agllvm.pas 59 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744
  1. {
  2. Copyright (c) 1998-2013 by the Free Pascal team
  3. This unit implements the generic part of the LLVM IR writer
  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. unit agllvm;
  18. {$i fpcdefs.inc}
  19. interface
  20. uses
  21. cclasses,
  22. globtype,globals,systems,
  23. aasmbase,aasmtai,aasmdata,
  24. assemble,
  25. aasmllvm, aasmllvmmetadata;
  26. type
  27. TLLVMInstrWriter = class;
  28. TLLVMModuleInlineAssemblyDecorator = class(IExternalAssemblerOutputFileDecorator)
  29. function LineFilter(const s: AnsiString): AnsiString;
  30. function LinePrefix: AnsiString;
  31. function LinePostfix: AnsiString;
  32. function LineEnding(const deflineending: ShortString): ShortString;
  33. end;
  34. TLLVMFunctionInlineAssemblyDecorator = class(IExternalAssemblerOutputFileDecorator)
  35. function LineFilter(const s: AnsiString): AnsiString;
  36. function LinePrefix: AnsiString;
  37. function LinePostfix: AnsiString;
  38. function LineEnding(const deflineending: ShortString): ShortString;
  39. end;
  40. TLLVMAssember=class(texternalassembler)
  41. protected
  42. ffuncinlasmdecorator: TLLVMFunctionInlineAssemblyDecorator;
  43. fdecllevel: longint;
  44. procedure WriteExtraHeader;virtual;
  45. procedure WriteExtraFooter;virtual;
  46. procedure WriteInstruction(hp: tai);
  47. procedure WriteLlvmInstruction(hp: tai);
  48. procedure WriteDirectiveName(dir: TAsmDirective); virtual;
  49. procedure WriteRealConst(hp: tai_realconst; do_line: boolean);
  50. procedure WriteOrdConst(hp: tai_const);
  51. procedure WriteTai(const replaceforbidden: boolean; const do_line, inmetadata: boolean; var InlineLevel: cardinal; var asmblock: boolean; var hp: tai);
  52. public
  53. constructor CreateWithWriter(info: pasminfo; wr: TExternalAssemblerOutputFile; freewriter, smart: boolean); override;
  54. procedure WriteTree(p:TAsmList);override;
  55. procedure WriteAsmList;override;
  56. procedure WriteFunctionInlineAsmList(list: tasmlist);
  57. destructor destroy; override;
  58. protected
  59. InstrWriter: TLLVMInstrWriter;
  60. end;
  61. TLLVMLLCAssember=class(TLLVMAssember)
  62. public
  63. function MakeCmdLine: TCmdStr; override;
  64. end;
  65. TLLVMClangAssember=class(TLLVMAssember)
  66. public
  67. function MakeCmdLine: TCmdStr; override;
  68. function DoAssemble: boolean; override;
  69. function RerunAssembler: boolean; override;
  70. protected
  71. function DoPipe: boolean; override;
  72. private
  73. fnextpass: byte;
  74. end;
  75. {# This is the base class for writing instructions.
  76. The WriteInstruction() method must be overridden
  77. to write a single instruction to the assembler
  78. file.
  79. }
  80. TLLVMInstrWriter = class
  81. constructor create(_owner: TLLVMAssember);
  82. procedure WriteInstruction(hp : tai);
  83. protected
  84. owner: TLLVMAssember;
  85. fstr: TSymStr;
  86. function getopcodestr(hp: taillvm): TSymStr;
  87. function getopstr(const o:toper; refwithalign: boolean) : TSymStr;
  88. procedure writeparas(const paras: tfplist);
  89. procedure WriteAsmRegisterAllocationClobbers(list: tasmlist);
  90. end;
  91. implementation
  92. uses
  93. SysUtils,
  94. cutils,cfileutl,
  95. fmodule,verbose,
  96. objcasm,
  97. aasmcnst,symconst,symdef,symtable,
  98. llvmbase,itllvm,llvmdef,
  99. cgbase,cgutils,cpubase,cpuinfo,llvminfo;
  100. const
  101. line_length = 70;
  102. type
  103. {$ifdef cpuextended}
  104. t80bitarray = array[0..9] of byte;
  105. {$endif cpuextended}
  106. t64bitarray = array[0..7] of byte;
  107. t32bitarray = array[0..3] of byte;
  108. {****************************************************************************}
  109. { Support routines }
  110. {****************************************************************************}
  111. function single2str(d : single) : string;
  112. var
  113. hs : string;
  114. begin
  115. str(d,hs);
  116. { replace space with + }
  117. if hs[1]=' ' then
  118. hs[1]:='+';
  119. single2str:=hs
  120. end;
  121. function double2str(d : double) : string;
  122. var
  123. hs : string;
  124. begin
  125. str(d,hs);
  126. { replace space with + }
  127. if hs[1]=' ' then
  128. hs[1]:='+';
  129. double2str:=hs
  130. end;
  131. function extended2str(e : extended) : string;
  132. var
  133. hs : string;
  134. begin
  135. str(e,hs);
  136. { replace space with + }
  137. if hs[1]=' ' then
  138. hs[1]:='+';
  139. extended2str:=hs
  140. end;
  141. {****************************************************************************}
  142. { Decorator for module-level inline assembly }
  143. {****************************************************************************}
  144. function TLLVMModuleInlineAssemblyDecorator.LineFilter(const s: AnsiString): AnsiString;
  145. var
  146. i: longint;
  147. begin
  148. result:='';
  149. for i:=1 to length(s) do
  150. begin
  151. case s[i] of
  152. #0..#31,
  153. #127..#255,
  154. '"','\':
  155. result:=result+
  156. '\'+
  157. chr((ord(s[i]) shr 4)+ord('0'))+
  158. chr((ord(s[i]) and $f)+ord('0'));
  159. else
  160. result:=result+s[i];
  161. end;
  162. end;
  163. end;
  164. function TLLVMModuleInlineAssemblyDecorator.LinePrefix: AnsiString;
  165. begin
  166. result:='module asm "';
  167. end;
  168. function TLLVMModuleInlineAssemblyDecorator.LinePostfix: AnsiString;
  169. begin
  170. result:='"';
  171. end;
  172. function TLLVMModuleInlineAssemblyDecorator.LineEnding(const deflineending: ShortString): ShortString;
  173. begin
  174. result:=deflineending
  175. end;
  176. {****************************************************************************}
  177. { Decorator for function-level inline assembly }
  178. {****************************************************************************}
  179. function TLLVMFunctionInlineAssemblyDecorator.LineFilter(const s: AnsiString): AnsiString;
  180. var
  181. i: longint;
  182. begin
  183. result:='';
  184. for i:=1 to length(s) do
  185. begin
  186. case s[i] of
  187. { escape dollars }
  188. '$':
  189. result:=result+'$$';
  190. { ` is used as placeholder for a single dollar (reference to
  191. argument to the inline assembly) }
  192. '`':
  193. result:=result+'$';
  194. #0..#31,
  195. #127..#255,
  196. '"','\':
  197. result:=result+
  198. '\'+
  199. chr((ord(s[i]) shr 4)+ord('0'))+
  200. chr((ord(s[i]) and $f)+ord('0'));
  201. else
  202. result:=result+s[i];
  203. end;
  204. end;
  205. end;
  206. function TLLVMFunctionInlineAssemblyDecorator.LinePrefix: AnsiString;
  207. begin
  208. result:='';
  209. end;
  210. function TLLVMFunctionInlineAssemblyDecorator.LinePostfix: AnsiString;
  211. begin
  212. result:='';
  213. end;
  214. function TLLVMFunctionInlineAssemblyDecorator.LineEnding(const deflineending: ShortString): ShortString;
  215. begin
  216. result:='\0A';
  217. end;
  218. {****************************************************************************}
  219. { LLVM Instruction writer }
  220. {****************************************************************************}
  221. function getregisterstring(reg: tregister): ansistring;
  222. begin
  223. if getregtype(reg)=R_TEMPREGISTER then
  224. result:='%tmp.'
  225. else
  226. result:='%reg.'+tostr(byte(getregtype(reg)))+'_';
  227. result:=result+tostr(getsupreg(reg));
  228. end;
  229. function getreferencealignstring(var ref: treference) : ansistring;
  230. begin
  231. result:=', align '+tostr(ref.alignment);
  232. end;
  233. function getreferencestring(var ref : treference; withalign: boolean) : ansistring;
  234. begin
  235. result:='';
  236. if assigned(ref.relsymbol) or
  237. (assigned(ref.symbol) and
  238. (ref.base<>NR_NO)) or
  239. (ref.index<>NR_NO) or
  240. (ref.offset<>0) then
  241. begin
  242. result:=' **(error ref: ';
  243. if assigned(ref.symbol) then
  244. result:=result+'sym='+ref.symbol.name+', ';
  245. if assigned(ref.relsymbol) then
  246. result:=result+'sym='+ref.relsymbol.name+', ';
  247. if ref.base=NR_NO then
  248. result:=result+'base=NR_NO, ';
  249. if ref.index<>NR_NO then
  250. result:=result+'index<>NR_NO, ';
  251. if ref.offset<>0 then
  252. result:=result+'offset='+tostr(ref.offset);
  253. result:=result+')**';
  254. internalerror(2013060225);
  255. end;
  256. if ref.base<>NR_NO then
  257. result:=result+getregisterstring(ref.base)
  258. else if assigned(ref.symbol) then
  259. result:=result+LlvmAsmSymName(ref.symbol)
  260. else
  261. result:=result+'null';
  262. if withalign then
  263. result:=result+getreferencealignstring(ref);
  264. end;
  265. procedure TLLVMInstrWriter.writeparas(const paras: tfplist);
  266. var
  267. i: longint;
  268. tmpinline: cardinal;
  269. para: pllvmcallpara;
  270. tmpasmblock: boolean;
  271. hp: tai;
  272. begin
  273. tmpinline:=1;
  274. tmpasmblock:=false;
  275. owner.writer.AsmWrite(fstr);
  276. fstr:='';
  277. owner.writer.AsmWrite('(');
  278. for i:=0 to paras.count-1 do
  279. begin
  280. if i<>0 then
  281. owner.writer.AsmWrite(', ');
  282. para:=pllvmcallpara(paras[i]);
  283. owner.writer.AsmWrite(llvmencodetypename(para^.def));
  284. if para^.valueext<>lve_none then
  285. owner.writer.AsmWrite(llvmvalueextension2str[para^.valueext]);
  286. if para^.byval then
  287. owner.writer.AsmWrite(' byval');
  288. if para^.sret then
  289. owner.writer.AsmWrite(' sret');
  290. { For byval, this means "alignment on the stack" and of the passed source data.
  291. For other pointer parameters, this means "alignment of the passed source data" }
  292. if (para^.alignment<>std_param_align) or
  293. (para^.alignment<0) then
  294. begin
  295. owner.writer.AsmWrite(' align ');
  296. owner.writer.AsmWrite(tostr(abs(para^.alignment)));
  297. end;
  298. case para^.typ of
  299. top_reg:
  300. begin
  301. owner.writer.AsmWrite(' ');
  302. owner.writer.AsmWrite(getregisterstring(para^.register));
  303. end;
  304. top_ref:
  305. begin
  306. owner.writer.AsmWrite(' ');
  307. owner.writer.AsmWrite(llvmasmsymname(para^.sym));
  308. end;
  309. top_const:
  310. begin
  311. owner.writer.AsmWrite(' ');
  312. owner.writer.AsmWrite(tostr(para^.value));
  313. end;
  314. top_tai:
  315. begin
  316. tmpinline:=1;
  317. tmpasmblock:=false;
  318. hp:=para^.ai;
  319. owner.writer.AsmWrite(fstr);
  320. fstr:='';
  321. owner.WriteTai(false,false,para^.def=llvm_metadatatype,tmpinline,tmpasmblock,hp);
  322. end;
  323. { empty records }
  324. top_undef:
  325. owner.writer.AsmWrite(' undef');
  326. else
  327. internalerror(2014010801);
  328. end;
  329. end;
  330. owner.writer.AsmWrite(')');
  331. end;
  332. function llvmdoubletostr(const d: double): TSymStr;
  333. type
  334. tdoubleval = record
  335. case byte of
  336. 1: (d: double);
  337. 2: (i: int64);
  338. end;
  339. begin
  340. { "When using the hexadecimal form, constants of types half,
  341. float, and double are represented using the 16-digit form shown
  342. above (which matches the IEEE754 representation for double)"
  343. And always in big endian form (sign bit leftmost)
  344. }
  345. result:='0x'+hexstr(tdoubleval(d).i,16);
  346. end;
  347. {$if defined(cpuextended) and defined(FPC_HAS_TYPE_EXTENDED)}
  348. function llvmextendedtostr(const e: extended): TSymStr;
  349. var
  350. extendedval: record
  351. case byte of
  352. 1: (e: extended);
  353. 2: (r: packed record
  354. {$ifdef FPC_LITTLE_ENDIAN}
  355. l: int64;
  356. h: word;
  357. {$else FPC_LITTLE_ENDIAN}
  358. h: int64;
  359. l: word;
  360. {$endif FPC_LITTLE_ENDIAN}
  361. end;
  362. );
  363. end;
  364. begin
  365. extendedval.e:=e;
  366. { hex format is always big endian in llvm }
  367. result:='0xK'+hexstr(extendedval.r.h,sizeof(extendedval.r.h)*2)+
  368. hexstr(extendedval.r.l,sizeof(extendedval.r.l)*2);
  369. end;
  370. {$endif cpuextended}
  371. function TLLVMInstrWriter.getopstr(const o:toper; refwithalign: boolean) : TSymStr;
  372. var
  373. hp: tai;
  374. tmpinline: cardinal;
  375. tmpasmblock: boolean;
  376. begin
  377. case o.typ of
  378. top_reg:
  379. getopstr:=getregisterstring(o.reg);
  380. top_const:
  381. getopstr:=tostr(int64(o.val));
  382. top_ref:
  383. if o.ref^.refaddr=addr_full then
  384. begin
  385. getopstr:='';
  386. if assigned(o.ref^.symbol) then
  387. getopstr:=LlvmAsmSymName(o.ref^.symbol)
  388. else
  389. getopstr:='null';
  390. if o.ref^.offset<>0 then
  391. internalerror(2013060223);
  392. end
  393. else
  394. getopstr:=getreferencestring(o.ref^,refwithalign);
  395. top_def:
  396. begin
  397. getopstr:=llvmencodetypename(o.def);
  398. end;
  399. top_cond:
  400. begin
  401. getopstr:=llvm_cond2str[o.cond];
  402. end;
  403. top_fpcond:
  404. begin
  405. getopstr:=llvm_fpcond2str[o.fpcond];
  406. end;
  407. top_single,
  408. top_double:
  409. begin
  410. { "When using the hexadecimal form, constants of types half,
  411. float, and double are represented using the 16-digit form shown
  412. above (which matches the IEEE754 representation for double)"
  413. And always in big endian form (sign bit leftmost)
  414. }
  415. if o.typ=top_double then
  416. result:=llvmdoubletostr(o.dval)
  417. else
  418. result:=llvmdoubletostr(o.sval)
  419. end;
  420. top_para:
  421. begin
  422. writeparas(o.paras);
  423. result:='';
  424. end;
  425. top_tai:
  426. begin
  427. if assigned(o.ai) then
  428. begin
  429. tmpinline:=1;
  430. tmpasmblock:=false;
  431. hp:=o.ai;
  432. owner.writer.AsmWrite(fstr);
  433. fstr:='';
  434. owner.WriteTai(false,false,false,tmpinline,tmpasmblock,hp);
  435. end;
  436. result:='';
  437. end;
  438. {$if defined(cpuextended) and defined(FPC_HAS_TYPE_EXTENDED)}
  439. top_extended80:
  440. begin
  441. result:=llvmextendedtostr(o.eval);
  442. end;
  443. {$endif cpuextended}
  444. top_undef:
  445. result:='undef';
  446. top_callingconvention:
  447. result:=llvm_callingconvention_name(o.callingconvention);
  448. else
  449. internalerror(2013060227);
  450. end;
  451. end;
  452. procedure TLLVMInstrWriter.WriteAsmRegisterAllocationClobbers(list: tasmlist);
  453. var
  454. hp: tai;
  455. begin
  456. hp:=tai(list.first);
  457. while assigned(hp) do
  458. begin
  459. if (hp.typ=ait_regalloc) and
  460. (tai_regalloc(hp).ratype=ra_alloc) then
  461. begin
  462. owner.writer.AsmWrite(',~{');
  463. owner.writer.AsmWrite(std_regname(tai_regalloc(hp).reg));
  464. owner.writer.AsmWrite('}');
  465. end;
  466. hp:=tai(hp.next);
  467. end;
  468. end;
  469. procedure TLLVMInstrWriter.WriteInstruction(hp: tai);
  470. var
  471. op: tllvmop;
  472. tmpstr,
  473. sep: TSymStr;
  474. i, opstart: longint;
  475. nested: boolean;
  476. opdone,
  477. done: boolean;
  478. begin
  479. op:=taillvm(hp).llvmopcode;
  480. { we write everything immediately rather than adding it into a string,
  481. because operands may contain other tai that will also write things out
  482. (and their output must come after everything that was processed in this
  483. instruction, such as its opcode or previous operands) }
  484. if owner.fdecllevel=0 then
  485. owner.writer.AsmWrite(#9);
  486. sep:=' ';
  487. opdone:=false;
  488. done:=false;
  489. opstart:=0;
  490. nested:=false;
  491. case op of
  492. la_type:
  493. begin
  494. owner.writer.AsmWrite(llvmtypeidentifier(taillvm(hp).oper[0]^.def));
  495. owner.writer.AsmWrite(' = type ');
  496. owner.writer.AsmWrite(llvmencodetypedecl(taillvm(hp).oper[0]^.def));
  497. done:=true;
  498. end;
  499. la_asmblock:
  500. begin
  501. owner.writer.AsmWrite('call void asm sideeffect "');
  502. owner.WriteFunctionInlineAsmList(taillvm(hp).oper[0]^.asmlist);
  503. owner.writer.AsmWrite('","');
  504. { we pass all accessed local variables as in/out address parameters,
  505. since we don't analyze the assembly code to determine what exactly
  506. happens to them; this is also compatible with the regular code
  507. generators, which always place local place local variables
  508. accessed from assembly code in memory }
  509. for i:=0 to taillvm(hp).oper[1]^.paras.Count-1 do
  510. begin
  511. owner.writer.AsmWrite('=*m,');
  512. end;
  513. owner.writer.AsmWrite('~{memory},~{fpsr},~{flags}');
  514. WriteAsmRegisterAllocationClobbers(taillvm(hp).oper[0]^.asmlist);
  515. owner.writer.AsmWrite('"');
  516. writeparas(taillvm(hp).oper[1]^.paras);
  517. done:=true;
  518. end;
  519. la_load,
  520. la_getelementptr:
  521. begin
  522. if (taillvm(hp).oper[0]^.typ<>top_reg) or
  523. (taillvm(hp).oper[0]^.reg<>NR_NO) then
  524. owner.writer.AsmWrite(getopstr(taillvm(hp).oper[0]^,false)+' = ')
  525. else
  526. nested:=true;
  527. opstart:=1;
  528. if llvmflag_load_getelptr_type in llvmversion_properties[current_settings.llvmversion] then
  529. begin
  530. owner.writer.AsmWrite(getopcodestr(taillvm(hp)));
  531. opdone:=true;
  532. if nested then
  533. owner.writer.AsmWrite(' (')
  534. else
  535. owner.writer.AsmWrite(' ');
  536. { can't just dereference the type, because it may be an
  537. implicit pointer type such as a class -> resort to string
  538. manipulation... Not very clean :( }
  539. tmpstr:=llvmencodetypename(taillvm(hp).spilling_get_reg_type(0));
  540. if op=la_getelementptr then
  541. begin
  542. if tmpstr[length(tmpstr)]<>'*' then
  543. begin
  544. writeln(tmpstr);
  545. internalerror(2016071101);
  546. end
  547. else
  548. setlength(tmpstr,length(tmpstr)-1);
  549. end;
  550. owner.writer.AsmWrite(tmpstr);
  551. owner.writer.AsmWrite(',');
  552. end
  553. end;
  554. la_ret, la_br, la_switch, la_indirectbr,
  555. la_resume,
  556. la_unreachable,
  557. la_store,
  558. la_fence,
  559. la_cmpxchg,
  560. la_atomicrmw,
  561. la_catch,
  562. la_filter,
  563. la_cleanup:
  564. begin
  565. { instructions that never have a result }
  566. end;
  567. la_call,
  568. la_invoke:
  569. begin
  570. if taillvm(hp).oper[1]^.reg<>NR_NO then
  571. owner.writer.AsmWrite(getregisterstring(taillvm(hp).oper[1]^.reg)+' = ');
  572. opstart:=2;
  573. if llvmflag_call_no_ptr in llvmversion_properties[current_settings.llvmversion] then
  574. begin
  575. owner.writer.AsmWrite(getopcodestr(taillvm(hp)));
  576. tmpstr:=llvm_callingconvention_name(taillvm(hp).oper[2]^.callingconvention);
  577. if tmpstr<>'' then
  578. begin
  579. owner.writer.AsmWrite(' "');
  580. owner.writer.AsmWrite(tmpstr);
  581. owner.writer.AsmWrite('"');
  582. end;
  583. opdone:=true;
  584. tmpstr:=llvmencodetypename(taillvm(hp).oper[3]^.def);
  585. if tmpstr[length(tmpstr)]<>'*' then
  586. begin
  587. writeln(tmpstr);
  588. internalerror(2016071102);
  589. end
  590. else
  591. setlength(tmpstr,length(tmpstr)-1);
  592. owner.writer.AsmWrite(tmpstr);
  593. opstart:=4;
  594. end;
  595. end;
  596. la_blockaddress:
  597. begin
  598. { nested -> no type }
  599. if owner.fdecllevel = 0 then
  600. begin
  601. owner.writer.AsmWrite(getopstr(taillvm(hp).oper[0]^,false));
  602. owner.writer.AsmWrite(' ');
  603. end;
  604. owner.writer.AsmWrite('blockaddress(');
  605. owner.writer.AsmWrite(getopstr(taillvm(hp).oper[1]^,false));
  606. { getopstr would add a "label" qualifier, which blockaddress does
  607. not want }
  608. owner.writer.AsmWrite(',%');
  609. with taillvm(hp).oper[2]^ do
  610. begin
  611. if (typ<>top_ref) or
  612. (ref^.refaddr<>addr_full) then
  613. internalerror(2016112001);
  614. owner.writer.AsmWrite(ref^.symbol.name);
  615. end;
  616. nested:=true;
  617. done:=true;
  618. end;
  619. la_alloca:
  620. begin
  621. owner.writer.AsmWrite(getreferencestring(taillvm(hp).oper[0]^.ref^,false)+' = ');
  622. sep:=' ';
  623. opstart:=1;
  624. end;
  625. la_trunc, la_zext, la_sext, la_fptrunc, la_fpext,
  626. la_fptoui, la_fptosi, la_uitofp, la_sitofp,
  627. la_ptrtoint, la_inttoptr,
  628. la_bitcast:
  629. begin
  630. { destination can be empty in case of nested constructs, or
  631. data initialisers }
  632. if (taillvm(hp).oper[0]^.typ<>top_reg) or
  633. (taillvm(hp).oper[0]^.reg<>NR_NO) then
  634. owner.writer.AsmWrite(getopstr(taillvm(hp).oper[0]^,false)+' = ')
  635. else
  636. nested:=true;
  637. owner.writer.AsmWrite(getopcodestr(taillvm(hp)));
  638. if not nested then
  639. owner.writer.AsmWrite(' ')
  640. else
  641. owner.writer.AsmWrite(' (');
  642. owner.writer.AsmWrite(getopstr(taillvm(hp).oper[1]^,false));
  643. { if there's a tai operand, its def is used instead of an
  644. explicit def operand }
  645. if taillvm(hp).ops=4 then
  646. begin
  647. owner.writer.AsmWrite(' ');
  648. owner.writer.AsmWrite(getopstr(taillvm(hp).oper[2]^,false));
  649. opstart:=3;
  650. end
  651. else
  652. opstart:=2;
  653. owner.writer.AsmWrite(' to ');
  654. owner.writer.AsmWrite(getopstr(taillvm(hp).oper[opstart]^,false));
  655. done:=true;
  656. end
  657. else
  658. begin
  659. if (taillvm(hp).oper[0]^.typ<>top_reg) or
  660. (taillvm(hp).oper[0]^.reg<>NR_NO) then
  661. begin
  662. owner.writer.AsmWrite(getopstr(taillvm(hp).oper[0]^,true)+' = ');
  663. end
  664. else
  665. nested:=true;
  666. sep:=' ';
  667. opstart:=1
  668. end;
  669. end;
  670. { process operands }
  671. if not done then
  672. begin
  673. if not opdone then
  674. begin
  675. owner.writer.AsmWrite(getopcodestr(taillvm(hp)));
  676. if nested then
  677. owner.writer.AsmWrite(' (');
  678. end;
  679. if taillvm(hp).ops<>0 then
  680. begin
  681. for i:=opstart to taillvm(hp).ops-1 do
  682. begin
  683. owner.writer.AsmWrite(sep);
  684. { special invoke interjections: "to label X unwind label Y" }
  685. if (op=la_invoke) then
  686. case i of
  687. 6: owner.writer.AsmWrite('to ');
  688. 7: owner.writer.AsmWrite('unwind ');
  689. end;
  690. owner.writer.AsmWrite(getopstr(taillvm(hp).oper[i]^,op in [la_load,la_store]));
  691. if (taillvm(hp).oper[i]^.typ in [top_def,top_cond,top_fpcond]) or
  692. (op in [la_call,la_invoke,la_landingpad,la_catch,la_filter,la_cleanup]) then
  693. sep :=' '
  694. else
  695. sep:=', ';
  696. end;
  697. end;
  698. end;
  699. if op=la_alloca then
  700. owner.writer.AsmWrite(getreferencealignstring(taillvm(hp).oper[0]^.ref^));
  701. if nested then
  702. owner.writer.AsmWrite(')')
  703. else if owner.fdecllevel=0 then
  704. owner.writer.AsmLn;
  705. end;
  706. function TLLVMInstrWriter.getopcodestr(hp: taillvm): TSymStr;
  707. begin
  708. result:=llvm_op2str[hp.llvmopcode];
  709. case hp.llvmopcode of
  710. la_load:
  711. begin
  712. if vol_read in hp.oper[2]^.ref^.volatility then
  713. result:=result+' volatile';
  714. end;
  715. la_store:
  716. begin
  717. if vol_write in hp.oper[3]^.ref^.volatility then
  718. result:=result+' volatile';
  719. end;
  720. end;
  721. end;
  722. {****************************************************************************}
  723. { LLVM Assembler writer }
  724. {****************************************************************************}
  725. destructor TLLVMAssember.Destroy;
  726. begin
  727. InstrWriter.free;
  728. ffuncinlasmdecorator.free;
  729. inherited destroy;
  730. end;
  731. procedure TLLVMAssember.WriteTree(p:TAsmList);
  732. var
  733. hp : tai;
  734. InlineLevel : cardinal;
  735. asmblock: boolean;
  736. do_line : boolean;
  737. replaceforbidden: boolean;
  738. begin
  739. if not assigned(p) then
  740. exit;
  741. replaceforbidden:=asminfo^.dollarsign<>'$';
  742. InlineLevel:=0;
  743. asmblock:=false;
  744. { lineinfo is only needed for al_procedures (PFV) }
  745. do_line:=(cs_asm_source in current_settings.globalswitches) or
  746. ((cs_lineinfo in current_settings.moduleswitches)
  747. and (p=current_asmdata.asmlists[al_procedures]));
  748. hp:=tai(p.first);
  749. while assigned(hp) do
  750. begin
  751. prefetch(pointer(hp.next)^);
  752. if not(hp.typ in SkipLineInfo) then
  753. begin
  754. current_filepos:=tailineinfo(hp).fileinfo;
  755. { no line info for inlined code }
  756. if do_line and (inlinelevel=0) then
  757. WriteSourceLine(hp as tailineinfo);
  758. end;
  759. WriteTai(replaceforbidden, do_line, false, InlineLevel, asmblock, hp);
  760. hp:=tai(hp.next);
  761. end;
  762. end;
  763. procedure TLLVMAssember.WriteExtraHeader;
  764. begin
  765. writer.AsmWrite('target datalayout = "');
  766. writer.AsmWrite(target_info.llvmdatalayout);
  767. writer.AsmWriteln('"');
  768. writer.AsmWrite('target triple = "');
  769. writer.AsmWrite(llvm_target_name);
  770. writer.AsmWriteln('"');
  771. end;
  772. procedure TLLVMAssember.WriteExtraFooter;
  773. begin
  774. end;
  775. procedure TLLVMAssember.WriteInstruction(hp: tai);
  776. begin
  777. end;
  778. procedure TLLVMAssember.WriteLlvmInstruction(hp: tai);
  779. begin
  780. InstrWriter.WriteInstruction(hp);
  781. end;
  782. procedure TLLVMAssember.WriteRealConst(hp: tai_realconst; do_line: boolean);
  783. begin
  784. if fdecllevel=0 then
  785. begin
  786. case tai_realconst(hp).realtyp of
  787. aitrealconst_s32bit:
  788. writer.AsmWriteLn(asminfo^.comment+'value: '+single2str(tai_realconst(hp).value.s32val));
  789. aitrealconst_s64bit:
  790. writer.AsmWriteLn(asminfo^.comment+'value: '+double2str(tai_realconst(hp).value.s64val));
  791. {$if defined(cpuextended) and defined(FPC_HAS_TYPE_EXTENDED)}
  792. { can't write full 80 bit floating point constants yet on non-x86 }
  793. aitrealconst_s80bit:
  794. writer.AsmWriteLn(asminfo^.comment+'value: '+extended2str(tai_realconst(hp).value.s80val));
  795. {$endif cpuextended}
  796. aitrealconst_s64comp:
  797. writer.AsmWriteLn(asminfo^.comment+'value: '+extended2str(tai_realconst(hp).value.s64compval));
  798. else
  799. internalerror(2014050604);
  800. end;
  801. internalerror(2016120202);
  802. end;
  803. case hp.realtyp of
  804. aitrealconst_s32bit:
  805. writer.AsmWrite(llvmdoubletostr(hp.value.s32val));
  806. aitrealconst_s64bit:
  807. writer.AsmWriteln(llvmdoubletostr(hp.value.s64val));
  808. {$if defined(cpuextended) and defined(FPC_HAS_TYPE_EXTENDED)}
  809. aitrealconst_s80bit:
  810. writer.AsmWrite(llvmextendedtostr(hp.value.s80val));
  811. {$endif defined(cpuextended)}
  812. aitrealconst_s64comp:
  813. { handled as int64 most of the time in llvm }
  814. writer.AsmWrite(tostr(round(hp.value.s64compval)));
  815. else
  816. internalerror(2014062401);
  817. end;
  818. end;
  819. procedure TLLVMAssember.WriteOrdConst(hp: tai_const);
  820. var
  821. consttyp: taiconst_type;
  822. begin
  823. if fdecllevel=0 then
  824. internalerror(2016120203);
  825. consttyp:=hp.consttype;
  826. case consttyp of
  827. aitconst_got,
  828. aitconst_gotoff_symbol,
  829. aitconst_uleb128bit,
  830. aitconst_sleb128bit,
  831. aitconst_rva_symbol,
  832. aitconst_secrel32_symbol,
  833. aitconst_darwin_dwarf_delta32,
  834. aitconst_darwin_dwarf_delta64,
  835. aitconst_half16bit,
  836. aitconst_gs:
  837. internalerror(2014052901);
  838. aitconst_128bit,
  839. aitconst_64bit,
  840. aitconst_32bit,
  841. aitconst_16bit,
  842. aitconst_8bit,
  843. aitconst_16bit_unaligned,
  844. aitconst_32bit_unaligned,
  845. aitconst_64bit_unaligned:
  846. begin
  847. if fdecllevel=0 then
  848. writer.AsmWrite(asminfo^.comment);
  849. { can't have compile-time differences between symbols; these are
  850. normally for PIC, but llvm takes care of that for us }
  851. if assigned(hp.endsym) then
  852. internalerror(2014052902);
  853. if assigned(hp.sym) then
  854. begin
  855. writer.AsmWrite(LlvmAsmSymName(hp.sym));
  856. { can't have offsets }
  857. if hp.value<>0 then
  858. if fdecllevel<>0 then
  859. internalerror(2014052903)
  860. else
  861. writer.AsmWrite(' -- symbol offset: ' + tostr(hp.value));
  862. end
  863. else if hp.value=0 then
  864. writer.AsmWrite('zeroinitializer')
  865. else
  866. writer.AsmWrite(tostr(hp.value));
  867. {
  868. // activate in case of debugging IE 2016120203
  869. if fdecllevel=0 then
  870. writer.AsmLn;
  871. }
  872. end;
  873. else
  874. internalerror(200704251);
  875. end;
  876. end;
  877. procedure TLLVMAssember.WriteTai(const replaceforbidden: boolean; const do_line, inmetadata: boolean; var InlineLevel: cardinal; var asmblock: boolean; var hp: tai);
  878. procedure WriteLinkageVibilityFlags(bind: TAsmSymBind);
  879. begin
  880. case bind of
  881. AB_EXTERNAL,
  882. AB_EXTERNAL_INDIRECT:
  883. writer.AsmWrite(' external');
  884. AB_COMMON:
  885. writer.AsmWrite(' common');
  886. AB_LOCAL:
  887. writer.AsmWrite(' internal');
  888. AB_GLOBAL,
  889. AB_INDIRECT:
  890. ;
  891. AB_WEAK_EXTERNAL:
  892. writer.AsmWrite(' extern_weak');
  893. AB_PRIVATE_EXTERN:
  894. begin
  895. if not(llvmflag_linker_private in llvmversion_properties[current_settings.llvmversion]) then
  896. writer.AsmWrite(' hidden')
  897. else
  898. writer.AsmWrite(' linker_private');
  899. end
  900. else
  901. internalerror(2014020104);
  902. end;
  903. end;
  904. procedure WriteFunctionFlags(pd: tprocdef);
  905. begin
  906. if (pos('FPC_SETJMP',upper(pd.mangledname))<>0) or
  907. (pd.mangledname=(target_info.cprefix+'setjmp')) then
  908. writer.AsmWrite(' returns_twice');
  909. if po_inline in pd.procoptions then
  910. writer.AsmWrite(' inlinehint');
  911. if po_noinline in pd.procoptions then
  912. writer.AsmWrite(' noinline');
  913. { ensure that functions that happen to have the same name as a
  914. standard C library function, but which are implemented in Pascal,
  915. are not considered to have the same semantics as the C function with
  916. the same name }
  917. if not(po_external in pd.procoptions) then
  918. writer.AsmWrite(' nobuiltin');
  919. if po_noreturn in pd.procoptions then
  920. writer.AsmWrite(' noreturn');
  921. end;
  922. procedure WriteTypedConstData(hp: tai_abstracttypedconst; metadata: boolean);
  923. var
  924. p: tai_abstracttypedconst;
  925. pval: tai;
  926. defstr: TSymStr;
  927. first, gotstring: boolean;
  928. begin
  929. if hp.def<>llvm_metadatatype then
  930. begin
  931. defstr:=llvmencodetypename(hp.def)
  932. end
  933. else
  934. begin
  935. defstr:=''
  936. end;
  937. { write the struct, array or simple type }
  938. case hp.adetyp of
  939. tck_record:
  940. begin
  941. if not(metadata) then
  942. begin
  943. writer.AsmWrite(defstr);
  944. writer.AsmWrite(' <{');
  945. end
  946. else
  947. begin
  948. writer.AsmWrite(' !{');
  949. end;
  950. first:=true;
  951. for p in tai_aggregatetypedconst(hp) do
  952. begin
  953. if not first then
  954. writer.AsmWrite(', ')
  955. else
  956. first:=false;
  957. WriteTypedConstData(p,metadata);
  958. end;
  959. if not(metadata) then
  960. begin
  961. writer.AsmWrite('}>');
  962. end
  963. else
  964. begin
  965. writer.AsmWrite('}');
  966. end;
  967. end;
  968. tck_array:
  969. begin
  970. if not(metadata) then
  971. begin
  972. writer.AsmWrite(defstr);
  973. end;
  974. first:=true;
  975. gotstring:=false;
  976. for p in tai_aggregatetypedconst(hp) do
  977. begin
  978. if not first then
  979. writer.AsmWrite(', ')
  980. else
  981. begin
  982. writer.AsmWrite(' ');
  983. if (tai_abstracttypedconst(p).adetyp=tck_simple) and
  984. (tai_simpletypedconst(p).val.typ=ait_string) then
  985. begin
  986. gotstring:=true;
  987. end
  988. else
  989. begin
  990. if not metadata then
  991. begin
  992. writer.AsmWrite('[');
  993. end
  994. else
  995. begin
  996. writer.AsmWrite('!{');
  997. end;
  998. end;
  999. first:=false;
  1000. end;
  1001. { cannot concat strings and other things }
  1002. if gotstring and
  1003. not metadata and
  1004. ((tai_abstracttypedconst(p).adetyp<>tck_simple) or
  1005. (tai_simpletypedconst(p).val.typ<>ait_string)) then
  1006. internalerror(2014062701);
  1007. WriteTypedConstData(p,metadata);
  1008. end;
  1009. if not gotstring then
  1010. begin
  1011. if not metadata then
  1012. begin
  1013. writer.AsmWrite(']');
  1014. end
  1015. else
  1016. begin
  1017. writer.AsmWrite('}');
  1018. end;
  1019. end;
  1020. end;
  1021. tck_simple:
  1022. begin
  1023. pval:=tai_simpletypedconst(hp).val;
  1024. if (pval.typ<>ait_string) and
  1025. (defstr<>'') then
  1026. begin
  1027. writer.AsmWrite(defstr);
  1028. writer.AsmWrite(' ');
  1029. end;
  1030. WriteTai(replaceforbidden,do_line,metadata,InlineLevel,asmblock,pval);
  1031. end;
  1032. end;
  1033. end;
  1034. procedure WriteLlvmMetadataNode(hp: tai_llvmbasemetadatanode);
  1035. begin
  1036. { must only appear at the top level }
  1037. if fdecllevel<>0 then
  1038. internalerror(2019050111);
  1039. writer.AsmWrite('!');
  1040. writer.AsmWrite(tai_llvmbasemetadatanode(hp).name);
  1041. writer.AsmWrite(' =');
  1042. inc(fdecllevel);
  1043. WriteTypedConstData(hp,true);
  1044. writer.AsmLn;
  1045. dec(fdecllevel);
  1046. end;
  1047. var
  1048. hp2: tai;
  1049. s: string;
  1050. sstr: TSymStr;
  1051. i: longint;
  1052. ch: ansichar;
  1053. begin
  1054. case hp.typ of
  1055. ait_comment :
  1056. begin
  1057. writer.AsmWrite(asminfo^.comment);
  1058. writer.AsmWritePChar(tai_comment(hp).str);
  1059. if fdecllevel<>0 then
  1060. internalerror(2015090601);
  1061. writer.AsmLn;
  1062. end;
  1063. ait_regalloc :
  1064. begin
  1065. if (cs_asm_regalloc in current_settings.globalswitches) then
  1066. begin
  1067. writer.AsmWrite(#9+asminfo^.comment+'Register ');
  1068. repeat
  1069. writer.AsmWrite(std_regname(Tai_regalloc(hp).reg));
  1070. if (hp.next=nil) or
  1071. (tai(hp.next).typ<>ait_regalloc) or
  1072. (tai_regalloc(hp.next).ratype<>tai_regalloc(hp).ratype) then
  1073. break;
  1074. hp:=tai(hp.next);
  1075. writer.AsmWrite(',');
  1076. until false;
  1077. writer.AsmWrite(' ');
  1078. writer.AsmWriteLn(regallocstr[tai_regalloc(hp).ratype]);
  1079. end;
  1080. end;
  1081. ait_tempalloc :
  1082. begin
  1083. if (cs_asm_tempalloc in current_settings.globalswitches) then
  1084. WriteTempalloc(tai_tempalloc(hp));
  1085. end;
  1086. ait_align,
  1087. ait_section :
  1088. begin
  1089. { ignore, specified as part of declarations -- don't write
  1090. comment, because could appear in the middle of an aggregate
  1091. constant definition }
  1092. end;
  1093. ait_datablock :
  1094. begin
  1095. writer.AsmWrite(asminfo^.comment);
  1096. writer.AsmWriteln('datablock');
  1097. end;
  1098. ait_const:
  1099. begin
  1100. WriteOrdConst(tai_const(hp));
  1101. end;
  1102. ait_realconst :
  1103. begin
  1104. WriteRealConst(tai_realconst(hp), do_line);
  1105. end;
  1106. ait_string :
  1107. begin
  1108. if fdecllevel=0 then
  1109. internalerror(2016120201);
  1110. if not inmetadata then
  1111. writer.AsmWrite('c"')
  1112. else
  1113. writer.AsmWrite('!"');
  1114. for i:=1 to tai_string(hp).len do
  1115. begin
  1116. ch:=tai_string(hp).str[i-1];
  1117. case ch of
  1118. #0, {This can't be done by range, because a bug in FPC}
  1119. #1..#31,
  1120. #128..#255,
  1121. '"',
  1122. '\' : s:='\'+hexStr(ord(ch),2);
  1123. else
  1124. s:=ch;
  1125. end;
  1126. writer.AsmWrite(s);
  1127. end;
  1128. writer.AsmWrite('"');
  1129. end;
  1130. ait_label :
  1131. begin
  1132. if not asmblock and
  1133. (tai_label(hp).labsym.is_used) then
  1134. begin
  1135. if (tai_label(hp).labsym.bind=AB_PRIVATE_EXTERN) then
  1136. begin
  1137. { should be emitted as part of the variable/function def }
  1138. internalerror(2013010703);
  1139. end;
  1140. if tai_label(hp).labsym.bind in [AB_GLOBAL, AB_PRIVATE_EXTERN] then
  1141. begin
  1142. { should be emitted as part of the variable/function def }
  1143. //internalerror(2013010704);
  1144. writer.AsmWriteln(asminfo^.comment+'global/privateextern label: '+tai_label(hp).labsym.name);
  1145. end;
  1146. if replaceforbidden then
  1147. writer.AsmWrite(ReplaceForbiddenAsmSymbolChars(tai_label(hp).labsym.name))
  1148. else
  1149. writer.AsmWrite(tai_label(hp).labsym.name);
  1150. writer.AsmWriteLn(':');
  1151. end;
  1152. end;
  1153. ait_symbol :
  1154. begin
  1155. if fdecllevel=0 then
  1156. writer.AsmWrite(asminfo^.comment);
  1157. writer.AsmWriteln(LlvmAsmSymName(tai_symbol(hp).sym));
  1158. { todo }
  1159. if tai_symbol(hp).has_value then
  1160. internalerror(2014062402);
  1161. end;
  1162. ait_llvmdecl:
  1163. begin
  1164. if taillvmdecl(hp).def.typ=procdef then
  1165. begin
  1166. if not(ldf_definition in taillvmdecl(hp).flags) then
  1167. begin
  1168. writer.AsmWrite('declare');
  1169. writer.AsmWrite(llvmencodeproctype(tprocdef(taillvmdecl(hp).def), taillvmdecl(hp).namesym.name, lpd_decl));
  1170. WriteFunctionFlags(tprocdef(taillvmdecl(hp).def));
  1171. writer.AsmLn;
  1172. end
  1173. else
  1174. begin
  1175. writer.AsmWrite('define');
  1176. if ldf_weak in taillvmdecl(hp).flags then
  1177. writer.AsmWrite(' weak');
  1178. WriteLinkageVibilityFlags(taillvmdecl(hp).namesym.bind);
  1179. writer.AsmWrite(llvmencodeproctype(tprocdef(taillvmdecl(hp).def), '', lpd_def));
  1180. WriteFunctionFlags(tprocdef(taillvmdecl(hp).def));
  1181. if assigned(tprocdef(taillvmdecl(hp).def).personality) then
  1182. begin
  1183. writer.AsmWrite(' personality i8* bitcast (');
  1184. writer.AsmWrite(llvmencodeproctype(tprocdef(taillvmdecl(hp).def).personality, '', lpd_procvar));
  1185. writer.AsmWrite('* ');
  1186. writer.AsmWrite(llvmmangledname(tprocdef(taillvmdecl(hp).def).personality.mangledname));
  1187. writer.AsmWrite(' to i8*)');
  1188. end;
  1189. writer.AsmWriteln(' {');
  1190. end;
  1191. end
  1192. else
  1193. begin
  1194. writer.AsmWrite(LlvmAsmSymName(taillvmdecl(hp).namesym));
  1195. writer.AsmWrite(' =');
  1196. if ldf_weak in taillvmdecl(hp).flags then
  1197. writer.AsmWrite(' weak');
  1198. if ldf_appending in taillvmdecl(hp).flags then
  1199. writer.AsmWrite(' appending');
  1200. WriteLinkageVibilityFlags(taillvmdecl(hp).namesym.bind);
  1201. writer.AsmWrite(' ');
  1202. if (ldf_tls in taillvmdecl(hp).flags) then
  1203. writer.AsmWrite('thread_local ');
  1204. if ldf_unnamed_addr in taillvmdecl(hp).flags then
  1205. writer.AsmWrite('unnamed_addr ');
  1206. if taillvmdecl(hp).sec in [sec_rodata,sec_rodata_norel] then
  1207. writer.AsmWrite('constant ')
  1208. else
  1209. writer.AsmWrite('global ');
  1210. if not assigned(taillvmdecl(hp).initdata) then
  1211. begin
  1212. writer.AsmWrite(llvmencodetypename(taillvmdecl(hp).def));
  1213. if not(taillvmdecl(hp).namesym.bind in [AB_EXTERNAL, AB_WEAK_EXTERNAL,AB_EXTERNAL_INDIRECT]) then
  1214. writer.AsmWrite(' zeroinitializer');
  1215. end
  1216. else
  1217. begin
  1218. inc(fdecllevel);
  1219. { can't have an external symbol with initialisation data }
  1220. if taillvmdecl(hp).namesym.bind in [AB_EXTERNAL, AB_WEAK_EXTERNAL] then
  1221. internalerror(2014052905);
  1222. { bitcast initialisation data to the type of the constant }
  1223. { write initialisation data }
  1224. hp2:=tai(taillvmdecl(hp).initdata.first);
  1225. while assigned(hp2) do
  1226. begin
  1227. WriteTai(replaceforbidden,do_line,inmetadata,InlineLevel,asmblock,hp2);
  1228. hp2:=tai(hp2.next);
  1229. end;
  1230. dec(fdecllevel);
  1231. end;
  1232. { custom section name? }
  1233. case taillvmdecl(hp).sec of
  1234. sec_user:
  1235. begin
  1236. writer.AsmWrite(', section "');
  1237. writer.AsmWrite(taillvmdecl(hp).secname);
  1238. writer.AsmWrite('"');
  1239. end;
  1240. low(TObjCAsmSectionType)..high(TObjCAsmSectionType):
  1241. begin
  1242. writer.AsmWrite(', section "');
  1243. writer.AsmWrite(objc_section_name(taillvmdecl(hp).sec));
  1244. writer.AsmWrite('"');
  1245. end;
  1246. end;
  1247. { sections whose name starts with 'llvm.' are for LLVM
  1248. internal use and don't have an alignment }
  1249. if pos('llvm.',taillvmdecl(hp).secname)<>1 then
  1250. begin
  1251. { alignment }
  1252. writer.AsmWrite(', align ');
  1253. writer.AsmWriteln(tostr(taillvmdecl(hp).alignment));
  1254. end
  1255. else
  1256. writer.AsmLn;
  1257. end;
  1258. end;
  1259. ait_llvmalias:
  1260. begin
  1261. writer.AsmWrite(LlvmAsmSymName(taillvmalias(hp).newsym));
  1262. writer.AsmWrite(' = alias ');
  1263. WriteLinkageVibilityFlags(taillvmalias(hp).bind);
  1264. if taillvmalias(hp).def.typ=procdef then
  1265. sstr:=llvmencodeproctype(tabstractprocdef(taillvmalias(hp).def), '', lpd_alias)
  1266. else
  1267. sstr:=llvmencodetypename(taillvmalias(hp).def);
  1268. writer.AsmWrite(sstr);
  1269. if llvmflag_alias_double_type in llvmversion_properties[current_settings.llvmversion] then
  1270. begin
  1271. writer.AsmWrite(', ');
  1272. writer.AsmWrite(sstr);
  1273. end;
  1274. writer.AsmWrite('* ');
  1275. writer.AsmWriteln(LlvmAsmSymName(taillvmalias(hp).oldsym));
  1276. end;
  1277. ait_llvmmetadatanode:
  1278. begin
  1279. WriteLlvmMetadataNode(tai_llvmbasemetadatanode(hp));
  1280. end;
  1281. ait_llvmmetadatareftypedconst:
  1282. begin
  1283. { must only appear as an element in a typed const }
  1284. if fdecllevel=0 then
  1285. internalerror(2019050110);
  1286. writer.AsmWrite('!');
  1287. writer.AsmWrite(tai_llvmbasemetadatanode(tai_llvmmetadatareftypedconst(hp).val).name);
  1288. end;
  1289. ait_llvmmetadatarefoperand:
  1290. begin
  1291. { must only appear as an operand }
  1292. if fdecllevel=0 then
  1293. internalerror(2019050110);
  1294. writer.AsmWrite('!');
  1295. writer.AsmWrite(tai_llvmmetadatareferenceoperand(hp).id);
  1296. writer.AsmWrite(' !');
  1297. writer.AsmWrite(tai_llvmmetadatareferenceoperand(hp).value.name);
  1298. end;
  1299. ait_symbolpair:
  1300. begin
  1301. { should be emitted as part of the symbol def }
  1302. internalerror(2013010708);
  1303. end;
  1304. ait_symbol_end :
  1305. begin
  1306. if tai_symbol_end(hp).sym.typ=AT_FUNCTION then
  1307. writer.AsmWriteln('}')
  1308. else
  1309. writer.AsmWriteln('; ait_symbol_end error, should not be generated');
  1310. // internalerror(2013010711);
  1311. end;
  1312. ait_instruction :
  1313. begin
  1314. WriteInstruction(hp);
  1315. end;
  1316. ait_llvmins:
  1317. begin
  1318. WriteLlvmInstruction(hp);
  1319. end;
  1320. ait_stab :
  1321. begin
  1322. internalerror(2013010712);
  1323. end;
  1324. ait_force_line,
  1325. ait_function_name :
  1326. ;
  1327. ait_cutobject :
  1328. begin
  1329. end;
  1330. ait_marker :
  1331. case
  1332. tai_marker(hp).kind of
  1333. mark_NoLineInfoStart:
  1334. inc(InlineLevel);
  1335. mark_NoLineInfoEnd:
  1336. dec(InlineLevel);
  1337. { these cannot be nested }
  1338. mark_AsmBlockStart:
  1339. asmblock:=true;
  1340. mark_AsmBlockEnd:
  1341. asmblock:=false;
  1342. end;
  1343. ait_directive :
  1344. begin
  1345. { CPU directive is commented out for the LLVM }
  1346. if tai_directive(hp).directive=asd_cpu then
  1347. writer.AsmWrite(asminfo^.comment);
  1348. WriteDirectiveName(tai_directive(hp).directive);
  1349. if tai_directive(hp).name <>'' then
  1350. writer.AsmWrite(tai_directive(hp).name);
  1351. if fdecllevel<>0 then
  1352. internalerror(2015090602);
  1353. writer.AsmLn;
  1354. end;
  1355. ait_seh_directive :
  1356. begin
  1357. internalerror(2013010713);
  1358. end;
  1359. ait_varloc:
  1360. begin
  1361. if tai_varloc(hp).newlocationhi<>NR_NO then
  1362. writer.AsmWrite(strpnew('Var '+tai_varloc(hp).varsym.realname+' located in register '+
  1363. std_regname(tai_varloc(hp).newlocationhi)+':'+std_regname(tai_varloc(hp).newlocation)))
  1364. else
  1365. writer.AsmWrite(strpnew('Var '+tai_varloc(hp).varsym.realname+' located in register '+
  1366. std_regname(tai_varloc(hp).newlocation)));
  1367. if fdecllevel<>0 then
  1368. internalerror(2015090603);
  1369. writer.AsmLn;
  1370. end;
  1371. ait_typedconst:
  1372. begin
  1373. WriteTypedConstData(tai_abstracttypedconst(hp),false);
  1374. end
  1375. else
  1376. internalerror(2019012010);
  1377. end;
  1378. end;
  1379. constructor TLLVMAssember.CreateWithWriter(info: pasminfo; wr: TExternalAssemblerOutputFile; freewriter, smart: boolean);
  1380. begin
  1381. inherited;
  1382. InstrWriter:=TLLVMInstrWriter.create(self);
  1383. end;
  1384. procedure TLLVMAssember.WriteDirectiveName(dir: TAsmDirective);
  1385. begin
  1386. writer.AsmWrite('.'+directivestr[dir]+' ');
  1387. end;
  1388. procedure TLLVMAssember.WriteAsmList;
  1389. var
  1390. hal : tasmlisttype;
  1391. a: TExternalAssembler;
  1392. decorator: TLLVMModuleInlineAssemblyDecorator;
  1393. begin
  1394. WriteExtraHeader;
  1395. for hal:=low(TasmlistType) to high(TasmlistType) do
  1396. begin
  1397. if not assigned(current_asmdata.asmlists[hal]) or
  1398. current_asmdata.asmlists[hal].Empty then
  1399. continue;
  1400. writer.AsmWriteLn(asminfo^.comment+'Begin asmlist '+AsmlistTypeStr[hal]);
  1401. if not(hal in [al_pure_assembler,al_dwarf_frame]) then
  1402. writetree(current_asmdata.asmlists[hal])
  1403. else
  1404. begin
  1405. { write routines using the target-specific external assembler
  1406. writer, filtered using the LLVM module-level assembly
  1407. decorator }
  1408. decorator:=TLLVMModuleInlineAssemblyDecorator.Create;
  1409. writer.decorator:=decorator;
  1410. a:=GetExternalGnuAssemblerWithAsmInfoWriter(asminfo,writer);
  1411. a.WriteTree(current_asmdata.asmlists[hal]);
  1412. writer.decorator:=nil;
  1413. decorator.free;
  1414. a.free;
  1415. end;
  1416. writer.AsmWriteLn(asminfo^.comment+'End asmlist '+AsmlistTypeStr[hal]);
  1417. end;
  1418. writer.AsmLn;
  1419. end;
  1420. procedure TLLVMAssember.WriteFunctionInlineAsmList(list: tasmlist);
  1421. var
  1422. a: TExternalAssembler;
  1423. begin
  1424. if not assigned(ffuncinlasmdecorator) then
  1425. ffuncinlasmdecorator:=TLLVMFunctionInlineAssemblyDecorator.create;
  1426. if assigned(writer.decorator) then
  1427. internalerror(2016110201);
  1428. writer.decorator:=ffuncinlasmdecorator;
  1429. a:=GetExternalGnuAssemblerWithAsmInfoWriter(asminfo,writer);
  1430. a.WriteTree(list);
  1431. a.free;
  1432. writer.decorator:=nil;
  1433. end;
  1434. {****************************************************************************}
  1435. { LLVM Instruction Writer }
  1436. {****************************************************************************}
  1437. constructor TLLVMInstrWriter.create(_owner: TLLVMAssember);
  1438. begin
  1439. inherited create;
  1440. owner := _owner;
  1441. end;
  1442. {****************************************************************************}
  1443. { llc Assember }
  1444. {****************************************************************************}
  1445. function TLLVMLLCAssember.MakeCmdLine: TCmdStr;
  1446. var
  1447. optstr: TCmdStr;
  1448. begin
  1449. result:=inherited;
  1450. { standard optimization flags for llc -- todo: this needs to be split
  1451. into a call to opt and one to llc }
  1452. if cs_opt_level3 in current_settings.optimizerswitches then
  1453. optstr:='-O3'
  1454. else if cs_opt_level2 in current_settings.optimizerswitches then
  1455. optstr:='-O2'
  1456. else if cs_opt_level1 in current_settings.optimizerswitches then
  1457. optstr:='-O1'
  1458. else
  1459. optstr:='-O0';
  1460. { stack frame elimination }
  1461. if not(cs_opt_stackframe in current_settings.optimizerswitches) then
  1462. optstr:=optstr+' -disable-fp-elim';
  1463. { fast math }
  1464. if cs_opt_fastmath in current_settings.optimizerswitches then
  1465. optstr:=optstr+' -enable-unsafe-fp-math -fp-contract=fast'; { -enable-fp-mad support depends on version }
  1466. { smart linking }
  1467. if cs_create_smart in current_settings.moduleswitches then
  1468. optstr:=optstr+' -data-sections -function-sections';
  1469. { pic }
  1470. if cs_create_pic in current_settings.moduleswitches then
  1471. optstr:=optstr+' -relocation-model=pic'
  1472. else if not(target_info.system in systems_darwin) then
  1473. optstr:=optstr+' -relocation-model=static'
  1474. else
  1475. optstr:=optstr+' -relocation-model=dynamic-no-pic';
  1476. { force object output instead of textual assembler code }
  1477. optstr:=optstr+' -filetype=obj';
  1478. if fputypestrllvm[current_settings.fputype]<>'' then
  1479. optstr:=optstr+' -mattr=+'+fputypestrllvm[current_settings.fputype];
  1480. replace(result,'$OPT',optstr);
  1481. end;
  1482. {****************************************************************************}
  1483. { clang Assember }
  1484. {****************************************************************************}
  1485. function TLLVMClangAssember.MakeCmdLine: TCmdStr;
  1486. var
  1487. wpostr,
  1488. optstr: TCmdStr;
  1489. begin
  1490. wpostr:='';
  1491. if cs_lto in current_settings.moduleswitches then
  1492. begin
  1493. case fnextpass of
  1494. 0:
  1495. begin
  1496. ObjFileName:=ChangeFileExt(ObjFileName,'.bc');
  1497. wpostr:=' -flto';
  1498. end;
  1499. 1:
  1500. begin
  1501. ObjFileName:=ChangeFileExt(ObjFileName,'.o');
  1502. end;
  1503. end;
  1504. end;
  1505. result:=inherited;
  1506. { standard optimization flags for llc -- todo: this needs to be split
  1507. into a call to opt and one to llc }
  1508. if cs_opt_level3 in current_settings.optimizerswitches then
  1509. optstr:='-O3'
  1510. else if cs_opt_level2 in current_settings.optimizerswitches then
  1511. optstr:='-O2'
  1512. else if cs_opt_level1 in current_settings.optimizerswitches then
  1513. optstr:='-O1'
  1514. else
  1515. optstr:='-O0';
  1516. optstr:=optstr+wpostr;
  1517. { stack frame elimination }
  1518. if not(cs_opt_stackframe in current_settings.optimizerswitches) then
  1519. optstr:=optstr+' -fno-omit-frame-pointer'
  1520. else
  1521. optstr:=optstr+' -fomit-frame-pointer';
  1522. { fast math }
  1523. if cs_opt_fastmath in current_settings.optimizerswitches then
  1524. optstr:=optstr+' -ffast-math';
  1525. { smart linking }
  1526. if cs_create_smart in current_settings.moduleswitches then
  1527. optstr:=optstr+' -fdata-sections -ffunction-sections';
  1528. { pic }
  1529. if cs_create_pic in current_settings.moduleswitches then
  1530. optstr:=optstr+' -fpic'
  1531. else if not(target_info.system in systems_darwin) then
  1532. optstr:=optstr+' -static'
  1533. else
  1534. optstr:=optstr+' -mdynamic-no-pic';
  1535. if not(target_info.system in systems_darwin) then
  1536. begin
  1537. optstr:=optstr+' --target='+llvm_target_name;
  1538. end;
  1539. if fputypestrllvm[current_settings.fputype]<>'' then
  1540. optstr:=optstr+' -m'+fputypestrllvm[current_settings.fputype];
  1541. replace(result,'$OPT',optstr);
  1542. inc(fnextpass);
  1543. end;
  1544. function TLLVMClangAssember.DoAssemble: boolean;
  1545. begin
  1546. fnextpass:=0;
  1547. result:=inherited;
  1548. end;
  1549. function TLLVMClangAssember.RerunAssembler: boolean;
  1550. begin
  1551. result:=
  1552. (cs_lto in current_settings.moduleswitches) and
  1553. (fnextpass<=1);
  1554. end;
  1555. function TLLVMClangAssember.DoPipe: boolean;
  1556. begin
  1557. result:=
  1558. not(cs_lto in current_settings.moduleswitches) and
  1559. inherited;
  1560. end;
  1561. const
  1562. as_llvm_llc_info : tasminfo =
  1563. (
  1564. id : as_llvm_llc;
  1565. idtxt : 'LLVM-LLC';
  1566. asmbin : 'llc';
  1567. asmcmd: '$OPT -o $OBJ $ASM';
  1568. supported_targets : [system_x86_64_linux,system_x86_64_darwin,system_aarch64_linux,system_arm_linux];
  1569. flags : [af_smartlink_sections];
  1570. labelprefix : 'L';
  1571. comment : '; ';
  1572. dollarsign: '$';
  1573. );
  1574. as_llvm_clang_info : tasminfo =
  1575. (
  1576. id : as_llvm_clang;
  1577. idtxt : 'LLVM-CLANG';
  1578. asmbin : 'clang';
  1579. asmcmd: '$OPT $DARWINVERSION -c -o $OBJ $ASM';
  1580. supported_targets : [system_x86_64_linux,system_x86_64_darwin,system_aarch64_linux,system_arm_linux];
  1581. flags : [af_smartlink_sections];
  1582. labelprefix : 'L';
  1583. comment : '; ';
  1584. dollarsign: '$';
  1585. );
  1586. begin
  1587. RegisterAssembler(as_llvm_llc_info,TLLVMLLCAssember);
  1588. RegisterAssembler(as_llvm_clang_info,TLLVMClangAssember);
  1589. end.