aasmdata.pas 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543
  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_init,
  51. al_ext_rtti,
  52. al_dwarf_frame,
  53. al_dwarf_info,
  54. al_dwarf_abbrev,
  55. al_dwarf_line,
  56. al_picdata,
  57. al_indirectpicdata,
  58. al_resourcestrings,
  59. { Objective-C related sections }
  60. al_objc_data,
  61. { keep pool data separate, so we can generate new pool entries
  62. while emitting other data }
  63. al_objc_pools,
  64. al_end
  65. );
  66. { Type of constant 'pools'. Mostly for string types, but usable for
  67. floating point and large set constants, too. }
  68. TConstPoolType = (
  69. sp_invalid,
  70. sp_conststr,
  71. sp_shortstr,
  72. sp_longstr,
  73. sp_ansistr,
  74. sp_widestr,
  75. sp_unicodestr,
  76. sp_objcclassnamerefs,
  77. sp_varnamerefs,
  78. sp_objcclassnames,
  79. sp_objcvarnames,
  80. sp_objcvartypes,
  81. sp_objcprotocolrefs,
  82. sp_varsets,
  83. sp_floats
  84. );
  85. const
  86. AsmListTypeStr : array[TAsmListType] of string[24] =(
  87. 'al_begin',
  88. 'al_stabs',
  89. 'al_procedures',
  90. 'al_globals',
  91. 'al_const',
  92. 'al_typedconsts',
  93. 'al_rotypedconsts',
  94. 'al_threadvars',
  95. 'al_imports',
  96. 'al_exports',
  97. 'al_resources',
  98. 'al_rtti',
  99. 'al_init',
  100. 'al_ext_rtti',
  101. 'al_dwarf_frame',
  102. 'al_dwarf_info',
  103. 'al_dwarf_abbrev',
  104. 'al_dwarf_line',
  105. 'al_picdata',
  106. 'al_indirectpicdata',
  107. 'al_resourcestrings',
  108. 'al_objc_data',
  109. 'al_objc_pools',
  110. 'al_end'
  111. );
  112. type
  113. TAsmList = class(tlinkedlist)
  114. constructor create;
  115. function empty : boolean;
  116. function getlasttaifilepos : pfileposinfo;
  117. end;
  118. TAsmCFI=class
  119. public
  120. constructor create;virtual;
  121. destructor destroy;override;
  122. procedure generate_code(list:TAsmList);virtual;
  123. procedure start_frame(list:TAsmList);virtual;
  124. procedure end_frame(list:TAsmList);virtual;
  125. procedure cfa_offset(list:TAsmList;reg:tregister;ofs:longint);virtual;
  126. procedure cfa_restore(list:TAsmList;reg:tregister);virtual;
  127. procedure cfa_def_cfa_register(list:TAsmList;reg:tregister);virtual;
  128. procedure cfa_def_cfa_offset(list:TAsmList;ofs:longint);virtual;
  129. end;
  130. TAsmCFIClass=class of TAsmCFI;
  131. { TAsmData }
  132. TAsmData = class
  133. private
  134. { Symbols }
  135. FAsmSymbolDict : TFPHashObjectList;
  136. FAltSymbolList : TFPObjectList;
  137. FNextAltNr : longint;
  138. FNextLabelNr : array[TAsmLabeltype] of longint;
  139. { Call Frame Information for stack unwinding}
  140. FAsmCFI : TAsmCFI;
  141. FConstPools : array[TConstPoolType] of THashSet;
  142. function GetConstPools(APoolType: TConstPoolType): THashSet;
  143. public
  144. name,
  145. realname : string[80];
  146. NextVTEntryNr : longint;
  147. { Assembler lists }
  148. AsmLists : array[TAsmListType] of TAsmList;
  149. CurrAsmList : TAsmList;
  150. WideInits : TLinkedList;
  151. ResStrInits : TLinkedList;
  152. constructor create(const n:string);
  153. destructor destroy;override;
  154. { asmsymbol }
  155. function DefineAsmSymbolByClass(symclass: TAsmSymbolClass; const s : TSymStr;_bind:TAsmSymBind;_typ:Tasmsymtype) : TAsmSymbol;
  156. function DefineAsmSymbol(const s : TSymStr;_bind:TAsmSymBind;_typ:Tasmsymtype) : TAsmSymbol;
  157. function WeakRefAsmSymbol(const s : TSymStr) : TAsmSymbol;
  158. function RefAsmSymbol(const s : TSymStr) : TAsmSymbol;
  159. function GetAsmSymbol(const s : TSymStr) : TAsmSymbol;
  160. { create new assembler label }
  161. procedure getlabel(out l : TAsmLabel;alt:TAsmLabeltype);
  162. procedure getjumplabel(out l : TAsmLabel);
  163. procedure getglobaljumplabel(out l : TAsmLabel);
  164. procedure getaddrlabel(out l : TAsmLabel);
  165. procedure getdatalabel(out l : TAsmLabel);
  166. { generate an alternative (duplicate) symbol }
  167. procedure GenerateAltSymbol(p:TAsmSymbol);
  168. procedure ResetAltSymbols;
  169. property AsmSymbolDict:TFPHashObjectList read FAsmSymbolDict;
  170. property AsmCFI:TAsmCFI read FAsmCFI;
  171. { hash tables for reusing constant storage }
  172. property ConstPools[APoolType:TConstPoolType]: THashSet read GetConstPools;
  173. end;
  174. TAsmDataClass = class of TAsmData;
  175. TTCInitItem = class(TLinkedListItem)
  176. sym: tsym;
  177. offset: aint;
  178. datalabel: TAsmSymbol;
  179. constructor Create(asym: tsym; aoffset: aint; alabel: TAsmSymbol);
  180. end;
  181. const
  182. casmdata: TAsmDataClass = TAsmData;
  183. var
  184. CAsmCFI : TAsmCFIClass;
  185. current_asmdata : TAsmData;
  186. implementation
  187. uses
  188. verbose,
  189. aasmtai;
  190. {$ifdef MEMDEBUG}
  191. var
  192. memasmsymbols,
  193. memasmcfi,
  194. memasmlists : TMemDebug;
  195. {$endif MEMDEBUG}
  196. {*****************************************************************************
  197. TAsmCFI
  198. *****************************************************************************}
  199. constructor TAsmCFI.create;
  200. begin
  201. end;
  202. destructor TAsmCFI.destroy;
  203. begin
  204. end;
  205. procedure TAsmCFI.generate_code(list:TAsmList);
  206. begin
  207. end;
  208. procedure TAsmCFI.start_frame(list:TAsmList);
  209. begin
  210. end;
  211. procedure TAsmCFI.end_frame(list:TAsmList);
  212. begin
  213. end;
  214. procedure TAsmCFI.cfa_offset(list:TAsmList;reg:tregister;ofs:longint);
  215. begin
  216. end;
  217. procedure TAsmCFI.cfa_restore(list:TAsmList;reg:tregister);
  218. begin
  219. end;
  220. procedure TAsmCFI.cfa_def_cfa_register(list:TAsmList;reg:tregister);
  221. begin
  222. end;
  223. procedure TAsmCFI.cfa_def_cfa_offset(list:TAsmList;ofs:longint);
  224. begin
  225. end;
  226. {*****************************************************************************
  227. TTCInitItem
  228. *****************************************************************************}
  229. constructor TTCInitItem.Create(asym: tsym; aoffset: aint; alabel: TAsmSymbol);
  230. begin
  231. inherited Create;
  232. sym:=asym;
  233. offset:=aoffset;
  234. datalabel:=alabel;
  235. end;
  236. {*****************************************************************************
  237. TAsmList
  238. *****************************************************************************}
  239. constructor TAsmList.create;
  240. begin
  241. inherited create;
  242. { make sure the optimizer won't remove the first tai of this list}
  243. insert(tai_marker.create(mark_BlockStart));
  244. end;
  245. function TAsmList.empty : boolean;
  246. begin
  247. { there is always a mark_BlockStart available,
  248. see TAsmList.create }
  249. result:=(count<=1);
  250. end;
  251. function TAsmList.getlasttaifilepos : pfileposinfo;
  252. var
  253. hp : tlinkedlistitem;
  254. begin
  255. getlasttaifilepos := nil;
  256. if assigned(last) then
  257. begin
  258. { find the last file information record }
  259. if not (tai(last).typ in SkipLineInfo) then
  260. getlasttaifilepos:=@tailineinfo(last).fileinfo
  261. else
  262. { go through list backwards to find the first entry
  263. with line information
  264. }
  265. begin
  266. hp:=tai(last);
  267. while assigned(hp) and (tai(hp).typ in SkipLineInfo) do
  268. hp:=hp.Previous;
  269. { found entry }
  270. if assigned(hp) then
  271. getlasttaifilepos:=@tailineinfo(hp).fileinfo
  272. end;
  273. end;
  274. end;
  275. {****************************************************************************
  276. TAsmData
  277. ****************************************************************************}
  278. function TAsmData.GetConstPools(APoolType: TConstPoolType): THashSet;
  279. begin
  280. if FConstPools[APoolType] = nil then
  281. case APoolType of
  282. sp_ansistr: FConstPools[APoolType] := TTagHashSet.Create(64, True, False);
  283. else
  284. FConstPools[APoolType] := THashSet.Create(64, True, False);
  285. end;
  286. Result := FConstPools[APoolType];
  287. end;
  288. constructor TAsmData.create(const n:string);
  289. var
  290. alt : TAsmLabelType;
  291. hal : TAsmListType;
  292. begin
  293. inherited create;
  294. realname:=n;
  295. name:=upper(n);
  296. { symbols }
  297. FAsmSymbolDict:=TFPHashObjectList.create(true);
  298. FAltSymbolList:=TFPObjectList.Create(false);
  299. { labels }
  300. FNextAltNr:=1;
  301. for alt:=low(TAsmLabelType) to high(TAsmLabelType) do
  302. FNextLabelNr[alt]:=1;
  303. { AsmLists }
  304. CurrAsmList:=TAsmList.create;
  305. for hal:=low(TAsmListType) to high(TAsmListType) do
  306. AsmLists[hal]:=TAsmList.create;
  307. WideInits :=TLinkedList.create;
  308. ResStrInits:=TLinkedList.create;
  309. { CFI }
  310. FAsmCFI:=CAsmCFI.Create;
  311. end;
  312. destructor TAsmData.destroy;
  313. var
  314. hal : TAsmListType;
  315. hp : TConstPoolType;
  316. begin
  317. { Symbols }
  318. {$ifdef MEMDEBUG}
  319. memasmsymbols.start;
  320. {$endif}
  321. FAltSymbolList.free;
  322. FAsmSymbolDict.free;
  323. {$ifdef MEMDEBUG}
  324. memasmsymbols.stop;
  325. {$endif}
  326. { CFI }
  327. {$ifdef MEMDEBUG}
  328. memasmcfi.start;
  329. {$endif}
  330. FAsmCFI.free;
  331. {$ifdef MEMDEBUG}
  332. memasmcfi.stop;
  333. {$endif}
  334. { Lists }
  335. {$ifdef MEMDEBUG}
  336. memasmlists.start;
  337. {$endif}
  338. ResStrInits.free;
  339. WideInits.free;
  340. for hal:=low(TAsmListType) to high(TAsmListType) do
  341. AsmLists[hal].free;
  342. CurrAsmList.free;
  343. {$ifdef MEMDEBUG}
  344. memasmlists.stop;
  345. {$endif}
  346. for hp := low(TConstPoolType) to high(TConstPoolType) do
  347. FConstPools[hp].Free;
  348. end;
  349. function TAsmData.DefineAsmSymbolByClass(symclass: TAsmSymbolClass; const s : TSymStr;_bind:TAsmSymBind;_typ:Tasmsymtype) : TAsmSymbol;
  350. var
  351. hp : TAsmSymbol;
  352. begin
  353. hp:=TAsmSymbol(FAsmSymbolDict.Find(s));
  354. if assigned(hp) then
  355. begin
  356. { Redefine is allowed, but the types must be the same. The redefine
  357. is needed for Darwin where the labels are first allocated }
  358. if not(hp.bind in [AB_EXTERNAL,AB_WEAK_EXTERNAL]) then
  359. begin
  360. if (hp.bind<>_bind) and
  361. (hp.typ<>_typ) then
  362. internalerror(200603261);
  363. end;
  364. hp.typ:=_typ;
  365. hp.bind:=_bind;
  366. end
  367. else
  368. begin
  369. { Not found, insert it. }
  370. hp:=symclass.create(AsmSymbolDict,s,_bind,_typ);
  371. end;
  372. result:=hp;
  373. end;
  374. function TAsmData.DefineAsmSymbol(const s : TSymStr;_bind:TAsmSymBind;_typ:Tasmsymtype) : TAsmSymbol;
  375. begin
  376. result:=DefineAsmSymbolByClass(TAsmSymbol,s,_bind,_typ);
  377. end;
  378. function TAsmData.RefAsmSymbol(const s : TSymStr) : TAsmSymbol;
  379. begin
  380. result:=TAsmSymbol(FAsmSymbolDict.Find(s));
  381. if not assigned(result) then
  382. result:=TAsmSymbol.create(AsmSymbolDict,s,AB_EXTERNAL,AT_NONE)
  383. { one normal reference removes the "weak" character of a symbol }
  384. else if (result.bind=AB_WEAK_EXTERNAL) then
  385. result.bind:=AB_EXTERNAL;
  386. end;
  387. function TAsmData.WeakRefAsmSymbol(const s : TSymStr) : TAsmSymbol;
  388. begin
  389. result:=TAsmSymbol(FAsmSymbolDict.Find(s));
  390. if not assigned(result) then
  391. result:=TAsmSymbol.create(AsmSymbolDict,s,AB_WEAK_EXTERNAL,AT_NONE);
  392. end;
  393. function TAsmData.GetAsmSymbol(const s : TSymStr) : TAsmSymbol;
  394. begin
  395. result:=TAsmSymbol(FAsmSymbolDict.Find(s));
  396. end;
  397. procedure TAsmData.GenerateAltSymbol(p:TAsmSymbol);
  398. begin
  399. if not assigned(p.altsymbol) then
  400. begin
  401. p.altsymbol:=p.getaltcopy(AsmSymbolDict,FNextAltNr);
  402. FAltSymbolList.Add(p);
  403. end;
  404. end;
  405. procedure TAsmData.ResetAltSymbols;
  406. var
  407. i : longint;
  408. begin
  409. for i:=0 to FAltSymbolList.Count-1 do
  410. TAsmSymbol(FAltSymbolList[i]).altsymbol:=nil;
  411. FAltSymbolList.Clear;
  412. end;
  413. procedure TAsmData.getlabel(out l : TAsmLabel;alt:TAsmLabeltype);
  414. begin
  415. if (target_info.system in (systems_linux + systems_bsd)) and
  416. (cs_create_smart in current_settings.moduleswitches) and
  417. (alt = alt_dbgline) then
  418. l:=TAsmLabel.createglobal(AsmSymbolDict,name,FNextLabelNr[alt],alt)
  419. else
  420. l:=TAsmLabel.createlocal(AsmSymbolDict,FNextLabelNr[alt],alt);
  421. inc(FNextLabelNr[alt]);
  422. end;
  423. procedure TAsmData.getjumplabel(out l : TAsmLabel);
  424. begin
  425. l:=TAsmLabel.createlocal(AsmSymbolDict,FNextLabelNr[alt_jump],alt_jump);
  426. inc(FNextLabelNr[alt_jump]);
  427. end;
  428. procedure TAsmData.getglobaljumplabel(out l : TAsmLabel);
  429. begin
  430. l:=TAsmLabel.createglobal(AsmSymbolDict,name,FNextLabelNr[alt_jump],alt_jump);
  431. inc(FNextLabelNr[alt_jump]);
  432. end;
  433. procedure TAsmData.getdatalabel(out l : TAsmLabel);
  434. begin
  435. l:=TAsmLabel.createglobal(AsmSymbolDict,name,FNextLabelNr[alt_data],alt_data);
  436. inc(FNextLabelNr[alt_data]);
  437. end;
  438. procedure TAsmData.getaddrlabel(out l : TAsmLabel);
  439. begin
  440. l:=TAsmLabel.createlocal(AsmSymbolDict,FNextLabelNr[alt_addr],alt_addr);
  441. inc(FNextLabelNr[alt_addr]);
  442. end;
  443. initialization
  444. {$ifdef MEMDEBUG}
  445. memasmsymbols:=TMemDebug.create('AsmSymbols');
  446. memasmsymbols.stop;
  447. memasmcfi:=TMemDebug.create('AsmCFI');
  448. memasmcfi.stop;
  449. memasmlists:=TMemDebug.create('AsmLists');
  450. memasmlists.stop;
  451. {$endif MEMDEBUG}
  452. CAsmCFI:=TAsmCFI;
  453. finalization
  454. {$ifdef MEMDEBUG}
  455. memasmsymbols.free;
  456. memasmcfi.free;
  457. memasmlists.free;
  458. {$endif MEMDEBUG}
  459. end.