aasmdata.pas 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510
  1. {
  2. Copyright (c) 1998-2006 by Florian Klaempfl
  3. This unit implements an abstract asmoutput class for all processor types
  4. This program is free software; you can redistribute it and/or modify
  5. it under the terms of the GNU General Public License as published by
  6. the Free Software Foundation; either version 2 of the License, or
  7. (at your option) any later version.
  8. This program is distributed in the hope that it will be useful,
  9. but WITHOUT ANY WARRANTY; without even the implied warranty of
  10. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  11. GNU General Public License for more details.
  12. You should have received a copy of the GNU General Public License
  13. along with this program; if not, write to the Free Software
  14. Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
  15. ****************************************************************************
  16. }
  17. { @abstract(This unit implements an abstract asm output class for all processor types)
  18. This unit implements an abstract assembler output class for all processors, these
  19. are then overridden for each assembler writer to actually write the data in these
  20. classes to an assembler file.
  21. }
  22. unit aasmdata;
  23. {$i fpcdefs.inc}
  24. interface
  25. uses
  26. cutils,cclasses,
  27. globtype,globals,systems,
  28. cpuinfo,cpubase,
  29. cgbase,cgutils,
  30. symtype,
  31. aasmbase,ogbase;
  32. type
  33. { Type of AsmLists. The order is important for the layout of the
  34. information in the .o file. The stabs for the types must be defined
  35. before they can be referenced and therefor they need to be written
  36. first (PFV) }
  37. TAsmListType=(
  38. al_start,
  39. al_stabs,
  40. al_procedures,
  41. al_globals,
  42. al_const,
  43. al_typedconsts,
  44. al_rotypedconsts,
  45. al_threadvars,
  46. al_imports,
  47. al_exports,
  48. al_resources,
  49. al_rtti,
  50. al_dwarf_frame,
  51. al_dwarf_info,
  52. al_dwarf_abbrev,
  53. al_dwarf_line,
  54. al_picdata,
  55. al_resourcestrings,
  56. { Objective-C related sections }
  57. al_objc_data,
  58. { keep pool data separate, so we can generate new pool entries
  59. while emitting other data }
  60. al_objc_pools,
  61. al_end
  62. );
  63. { Type of constant 'pools'. Mostly for string types, but usable for
  64. floating point and large set constants, too. }
  65. TConstPoolType = (
  66. sp_invalid,
  67. sp_conststr,
  68. sp_shortstr,
  69. sp_longstr,
  70. sp_ansistr,
  71. sp_widestr,
  72. sp_unicodestr,
  73. sp_objcclassnamerefs,
  74. sp_varnamerefs,
  75. sp_objcclassnames,
  76. sp_objcvarnames,
  77. sp_objcvartypes,
  78. sp_objcprotocolrefs,
  79. sp_varsets,
  80. sp_floats
  81. );
  82. const
  83. AsmListTypeStr : array[TAsmListType] of string[24] =(
  84. 'al_begin',
  85. 'al_stabs',
  86. 'al_procedures',
  87. 'al_globals',
  88. 'al_const',
  89. 'al_typedconsts',
  90. 'al_rotypedconsts',
  91. 'al_threadvars',
  92. 'al_imports',
  93. 'al_exports',
  94. 'al_resources',
  95. 'al_rtti',
  96. 'al_dwarf_frame',
  97. 'al_dwarf_info',
  98. 'al_dwarf_abbrev',
  99. 'al_dwarf_line',
  100. 'al_picdata',
  101. 'al_resourcestrings',
  102. 'al_objc_data',
  103. 'al_objc_pools',
  104. 'al_end'
  105. );
  106. type
  107. TAsmList = class(tlinkedlist)
  108. constructor create;
  109. function empty : boolean;
  110. function getlasttaifilepos : pfileposinfo;
  111. end;
  112. TAsmCFI=class
  113. public
  114. constructor create;virtual;
  115. destructor destroy;override;
  116. procedure generate_code(list:TAsmList);virtual;
  117. procedure start_frame(list:TAsmList);virtual;
  118. procedure end_frame(list:TAsmList);virtual;
  119. procedure cfa_offset(list:TAsmList;reg:tregister;ofs:longint);virtual;
  120. procedure cfa_restore(list:TAsmList;reg:tregister);virtual;
  121. procedure cfa_def_cfa_register(list:TAsmList;reg:tregister);virtual;
  122. procedure cfa_def_cfa_offset(list:TAsmList;ofs:longint);virtual;
  123. end;
  124. TAsmCFIClass=class of TAsmCFI;
  125. TAsmData = class
  126. private
  127. { Symbols }
  128. FAsmSymbolDict : TFPHashObjectList;
  129. FAltSymbolList : TFPObjectList;
  130. FNextAltNr : longint;
  131. FNextLabelNr : array[TAsmLabeltype] of longint;
  132. { Call Frame Information for stack unwinding}
  133. FAsmCFI : TAsmCFI;
  134. public
  135. name,
  136. realname : string[80];
  137. NextVTEntryNr : longint;
  138. { Assembler lists }
  139. AsmLists : array[TAsmListType] of TAsmList;
  140. CurrAsmList : TAsmList;
  141. WideInits : TLinkedList;
  142. ResStrInits : TLinkedList;
  143. { hash tables for reusing constant storage }
  144. ConstPools : array[TConstPoolType] of THashSet;
  145. constructor create(const n:string);
  146. destructor destroy;override;
  147. { asmsymbol }
  148. function DefineAsmSymbol(const s : string;_bind:TAsmSymBind;_typ:Tasmsymtype) : TAsmSymbol;
  149. function WeakRefAsmSymbol(const s : string) : TAsmSymbol;
  150. function RefAsmSymbol(const s : string) : TAsmSymbol;
  151. function GetAsmSymbol(const s : string) : TAsmSymbol;
  152. { create new assembler label }
  153. procedure getlabel(out l : TAsmLabel;alt:TAsmLabeltype);
  154. procedure getjumplabel(out l : TAsmLabel);
  155. procedure getglobaljumplabel(out l : TAsmLabel);
  156. procedure getaddrlabel(out l : TAsmLabel);
  157. procedure getdatalabel(out l : TAsmLabel);
  158. { generate an alternative (duplicate) symbol }
  159. procedure GenerateAltSymbol(p:TAsmSymbol);
  160. procedure ResetAltSymbols;
  161. property AsmSymbolDict:TFPHashObjectList read FAsmSymbolDict;
  162. property AsmCFI:TAsmCFI read FAsmCFI;
  163. end;
  164. TTCInitItem = class(TLinkedListItem)
  165. sym: tsym;
  166. offset: aint;
  167. datalabel: TAsmSymbol;
  168. constructor Create(asym: tsym; aoffset: aint; alabel: TAsmSymbol);
  169. end;
  170. var
  171. CAsmCFI : TAsmCFIClass;
  172. current_asmdata : TAsmData;
  173. implementation
  174. uses
  175. verbose,
  176. aasmtai;
  177. {$ifdef MEMDEBUG}
  178. var
  179. memasmsymbols,
  180. memasmcfi,
  181. memasmlists : TMemDebug;
  182. {$endif MEMDEBUG}
  183. {*****************************************************************************
  184. TAsmCFI
  185. *****************************************************************************}
  186. constructor TAsmCFI.create;
  187. begin
  188. end;
  189. destructor TAsmCFI.destroy;
  190. begin
  191. end;
  192. procedure TAsmCFI.generate_code(list:TAsmList);
  193. begin
  194. end;
  195. procedure TAsmCFI.start_frame(list:TAsmList);
  196. begin
  197. end;
  198. procedure TAsmCFI.end_frame(list:TAsmList);
  199. begin
  200. end;
  201. procedure TAsmCFI.cfa_offset(list:TAsmList;reg:tregister;ofs:longint);
  202. begin
  203. end;
  204. procedure TAsmCFI.cfa_restore(list:TAsmList;reg:tregister);
  205. begin
  206. end;
  207. procedure TAsmCFI.cfa_def_cfa_register(list:TAsmList;reg:tregister);
  208. begin
  209. end;
  210. procedure TAsmCFI.cfa_def_cfa_offset(list:TAsmList;ofs:longint);
  211. begin
  212. end;
  213. {*****************************************************************************
  214. TTCInitItem
  215. *****************************************************************************}
  216. constructor TTCInitItem.Create(asym: tsym; aoffset: aint; alabel: TAsmSymbol);
  217. begin
  218. inherited Create;
  219. sym:=asym;
  220. offset:=aoffset;
  221. datalabel:=alabel;
  222. end;
  223. {*****************************************************************************
  224. TAsmList
  225. *****************************************************************************}
  226. constructor TAsmList.create;
  227. begin
  228. inherited create;
  229. { make sure the optimizer won't remove the first tai of this list}
  230. insert(tai_marker.create(mark_BlockStart));
  231. end;
  232. function TAsmList.empty : boolean;
  233. begin
  234. { there is always a mark_BlockStart available,
  235. see TAsmList.create }
  236. result:=(count<=1);
  237. end;
  238. function TAsmList.getlasttaifilepos : pfileposinfo;
  239. var
  240. hp : tlinkedlistitem;
  241. begin
  242. getlasttaifilepos := nil;
  243. if assigned(last) then
  244. begin
  245. { find the last file information record }
  246. if not (tai(last).typ in SkipLineInfo) then
  247. getlasttaifilepos:=@tailineinfo(last).fileinfo
  248. else
  249. { go through list backwards to find the first entry
  250. with line information
  251. }
  252. begin
  253. hp:=tai(last);
  254. while assigned(hp) and (tai(hp).typ in SkipLineInfo) do
  255. hp:=hp.Previous;
  256. { found entry }
  257. if assigned(hp) then
  258. getlasttaifilepos:=@tailineinfo(hp).fileinfo
  259. end;
  260. end;
  261. end;
  262. {****************************************************************************
  263. TAsmData
  264. ****************************************************************************}
  265. constructor TAsmData.create(const n:string);
  266. var
  267. alt : TAsmLabelType;
  268. hal : TAsmListType;
  269. begin
  270. inherited create;
  271. realname:=n;
  272. name:=upper(n);
  273. { symbols }
  274. FAsmSymbolDict:=TFPHashObjectList.create(true);
  275. FAltSymbolList:=TFPObjectList.Create(false);
  276. { labels }
  277. FNextAltNr:=1;
  278. for alt:=low(TAsmLabelType) to high(TAsmLabelType) do
  279. FNextLabelNr[alt]:=1;
  280. { AsmLists }
  281. CurrAsmList:=TAsmList.create;
  282. for hal:=low(TAsmListType) to high(TAsmListType) do
  283. AsmLists[hal]:=TAsmList.create;
  284. WideInits :=TLinkedList.create;
  285. ResStrInits:=TLinkedList.create;
  286. { CFI }
  287. FAsmCFI:=CAsmCFI.Create;
  288. end;
  289. destructor TAsmData.destroy;
  290. var
  291. hal : TAsmListType;
  292. hp : TConstPoolType;
  293. begin
  294. { Symbols }
  295. {$ifdef MEMDEBUG}
  296. memasmsymbols.start;
  297. {$endif}
  298. FAltSymbolList.free;
  299. FAsmSymbolDict.free;
  300. {$ifdef MEMDEBUG}
  301. memasmsymbols.stop;
  302. {$endif}
  303. { CFI }
  304. {$ifdef MEMDEBUG}
  305. memasmcfi.start;
  306. {$endif}
  307. FAsmCFI.free;
  308. {$ifdef MEMDEBUG}
  309. memasmcfi.stop;
  310. {$endif}
  311. { Lists }
  312. {$ifdef MEMDEBUG}
  313. memasmlists.start;
  314. {$endif}
  315. ResStrInits.free;
  316. WideInits.free;
  317. for hal:=low(TAsmListType) to high(TAsmListType) do
  318. AsmLists[hal].free;
  319. CurrAsmList.free;
  320. {$ifdef MEMDEBUG}
  321. memasmlists.stop;
  322. {$endif}
  323. for hp := low(TConstPoolType) to high(TConstPoolType) do
  324. ConstPools[hp].Free;
  325. end;
  326. function TAsmData.DefineAsmSymbol(const s : string;_bind:TAsmSymBind;_typ:Tasmsymtype) : TAsmSymbol;
  327. var
  328. hp : TAsmSymbol;
  329. begin
  330. hp:=TAsmSymbol(FAsmSymbolDict.Find(s));
  331. if assigned(hp) then
  332. begin
  333. { Redefine is allowed, but the types must be the same. The redefine
  334. is needed for Darwin where the labels are first allocated }
  335. if (hp.bind<>AB_EXTERNAL) then
  336. begin
  337. if (hp.bind<>_bind) and
  338. (hp.typ<>_typ) then
  339. internalerror(200603261);
  340. end;
  341. hp.typ:=_typ;
  342. hp.bind:=_bind;
  343. end
  344. else
  345. begin
  346. { Not found, insert it. }
  347. hp:=TAsmSymbol.create(AsmSymbolDict,s,_bind,_typ);
  348. end;
  349. result:=hp;
  350. end;
  351. function TAsmData.RefAsmSymbol(const s : string) : TAsmSymbol;
  352. begin
  353. result:=TAsmSymbol(FAsmSymbolDict.Find(s));
  354. if not assigned(result) then
  355. result:=TAsmSymbol.create(AsmSymbolDict,s,AB_EXTERNAL,AT_NONE)
  356. { one normal reference removes the "weak" character of a symbol }
  357. else if (result.bind=AB_WEAK_EXTERNAL) then
  358. result.bind:=AB_EXTERNAL;
  359. end;
  360. function TAsmData.WeakRefAsmSymbol(const s : string) : TAsmSymbol;
  361. begin
  362. result:=TAsmSymbol(FAsmSymbolDict.Find(s));
  363. if not assigned(result) then
  364. result:=TAsmSymbol.create(AsmSymbolDict,s,AB_WEAK_EXTERNAL,AT_NONE);
  365. end;
  366. function TAsmData.GetAsmSymbol(const s : string) : TAsmSymbol;
  367. begin
  368. result:=TAsmSymbol(FAsmSymbolDict.Find(s));
  369. end;
  370. procedure TAsmData.GenerateAltSymbol(p:TAsmSymbol);
  371. begin
  372. if not assigned(p.altsymbol) then
  373. begin
  374. p.altsymbol:=p.getaltcopy(AsmSymbolDict,FNextAltNr);
  375. FAltSymbolList.Add(p);
  376. end;
  377. end;
  378. procedure TAsmData.ResetAltSymbols;
  379. var
  380. i : longint;
  381. begin
  382. for i:=0 to FAltSymbolList.Count-1 do
  383. TAsmSymbol(FAltSymbolList[i]).altsymbol:=nil;
  384. FAltSymbolList.Clear;
  385. end;
  386. procedure TAsmData.getlabel(out l : TAsmLabel;alt:TAsmLabeltype);
  387. begin
  388. if (target_info.system in systems_linux) and
  389. (cs_create_smart in current_settings.moduleswitches) and
  390. (alt = alt_dbgline) then
  391. l:=TAsmLabel.createglobal(AsmSymbolDict,name,FNextLabelNr[alt],alt)
  392. else
  393. l:=TAsmLabel.createlocal(AsmSymbolDict,FNextLabelNr[alt],alt);
  394. inc(FNextLabelNr[alt]);
  395. end;
  396. procedure TAsmData.getjumplabel(out l : TAsmLabel);
  397. begin
  398. l:=TAsmLabel.createlocal(AsmSymbolDict,FNextLabelNr[alt_jump],alt_jump);
  399. inc(FNextLabelNr[alt_jump]);
  400. end;
  401. procedure TAsmData.getglobaljumplabel(out l : TAsmLabel);
  402. begin
  403. l:=TAsmLabel.createglobal(AsmSymbolDict,name,FNextLabelNr[alt_jump],alt_jump);
  404. inc(FNextLabelNr[alt_jump]);
  405. end;
  406. procedure TAsmData.getdatalabel(out l : TAsmLabel);
  407. begin
  408. l:=TAsmLabel.createglobal(AsmSymbolDict,name,FNextLabelNr[alt_data],alt_data);
  409. inc(FNextLabelNr[alt_data]);
  410. end;
  411. procedure TAsmData.getaddrlabel(out l : TAsmLabel);
  412. begin
  413. l:=TAsmLabel.createlocal(AsmSymbolDict,FNextLabelNr[alt_addr],alt_addr);
  414. inc(FNextLabelNr[alt_addr]);
  415. end;
  416. initialization
  417. {$ifdef MEMDEBUG}
  418. memasmsymbols:=TMemDebug.create('AsmSymbols');
  419. memasmsymbols.stop;
  420. memasmcfi:=TMemDebug.create('AsmCFI');
  421. memasmcfi.stop;
  422. memasmlists:=TMemDebug.create('AsmLists');
  423. memasmlists.stop;
  424. {$endif MEMDEBUG}
  425. CAsmCFI:=TAsmCFI;
  426. finalization
  427. {$ifdef MEMDEBUG}
  428. memasmsymbols.free;
  429. memasmcfi.free;
  430. memasmlists.free;
  431. {$endif MEMDEBUG}
  432. end.