t_wasi.pas 6.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265
  1. {
  2. Copyright (c) 2019 by Dmitry Boyarintsev
  3. This unit implements support import,export,link routines
  4. for the WASI target
  5. This program is free software; you can redistribute it and/or modify
  6. it under the terms of the GNU General Public License as published by
  7. the Free Software Foundation; either version 2 of the License, or
  8. (at your option) any later version.
  9. This program is distributed in the hope that it will be useful,
  10. but WITHOUT ANY WARRANTY; without even the implied warranty of
  11. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  12. GNU General Public License for more details.
  13. You should have received a copy of the GNU General Public License
  14. along with this program; if not, write to the Free Software
  15. Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
  16. ****************************************************************************
  17. }
  18. unit t_wasi;
  19. {$i fpcdefs.inc}
  20. interface
  21. uses
  22. systems,
  23. globtype, globals,
  24. aasmbase,
  25. cfileutl, cutils, cclasses,
  26. import, export, aasmdata, aasmcpu,
  27. fmodule, ogbase,
  28. symsym, symdef,
  29. link,
  30. i_wasi, tgcpu;
  31. type
  32. { texportlibwasi }
  33. texportlibwasi=class(texportlib)
  34. procedure preparelib(const s : string);override;
  35. procedure exportprocedure(hp : texported_item);override;
  36. procedure exportvar(hp : texported_item);override;
  37. procedure generatelib;override;
  38. end;
  39. { timportlibwasi }
  40. timportlibwasi = class(timportlib)
  41. procedure generatelib;override;
  42. end;
  43. { tlinkerwasi }
  44. tlinkerwasi=class(texternallinker)
  45. public
  46. constructor Create;override;
  47. procedure SetDefaultInfo;override;
  48. procedure InitSysInitUnitName;override;
  49. function MakeExecutable:boolean;override;
  50. function MakeSharedLibrary:boolean;override;
  51. end;
  52. implementation
  53. uses
  54. SysUtils,
  55. verbose;
  56. { timportlibwasi }
  57. procedure timportlibwasi.generatelib;
  58. begin
  59. end;
  60. { tlinkerwasi }
  61. constructor tlinkerwasi.Create;
  62. begin
  63. inherited Create;
  64. end;
  65. procedure tlinkerwasi.SetDefaultInfo;
  66. begin
  67. with Info do
  68. begin
  69. DllCmd[1] := 'wasm-ld $SONAME $GCSECTIONS $MAP -o $EXE';
  70. end;
  71. end;
  72. procedure tlinkerwasi.InitSysInitUnitName;
  73. begin
  74. sysinitunit:='si_prc';
  75. end;
  76. function tlinkerwasi.MakeExecutable:boolean;
  77. var
  78. GCSectionsStr : ansistring;
  79. binstr, cmdstr : Tcmdstr;
  80. InitStr,
  81. FiniStr,
  82. SoNameStr : string[80];
  83. mapstr,ltostr : TCmdStr;
  84. success : Boolean;
  85. tmp : TCmdStrListItem;
  86. tempFileName : ansistring;
  87. begin
  88. if not(cs_link_nolink in current_settings.globalswitches) then
  89. Message1(exec_i_linking,current_module.exefilename);
  90. { Create some replacements }
  91. mapstr:='';
  92. if (cs_link_map in current_settings.globalswitches) then
  93. mapstr:='-Map '+maybequoted(ChangeFileExt(current_module.exefilename,'.map'));
  94. if (cs_link_smart in current_settings.globalswitches) and
  95. create_smartlink_sections then
  96. GCSectionsStr:='--gc-sections'
  97. else
  98. GCSectionsStr:='';
  99. SoNameStr:='';
  100. SplitBinCmd(Info.DllCmd[1],binstr,cmdstr);
  101. Replace(cmdstr,'$EXE',maybequoted(current_module.exefilename));
  102. tmp := TCmdStrListItem(ObjectFiles.First);
  103. while Assigned(tmp) do begin
  104. cmdstr := tmp.Str+ ' ' + cmdstr;
  105. tmp := TCmdStrListItem(tmp.Next);
  106. end;
  107. // if HasExports then
  108. // cmdstr := cmdstr + ' --export-dynamic'; //' --export-dynamic';
  109. cmdstr := cmdstr + ' --no-entry';
  110. if (cs_link_strip in current_settings.globalswitches) then
  111. begin
  112. { only remove non global symbols and debugging info for a library }
  113. cmdstr := cmdstr + ' --strip-all';
  114. end;
  115. //Replace(cmdstr,'$OPT',Info.ExtraOptions);
  116. //Replace(cmdstr,'$RES',maybequoted(outputexedir+Info.ResName));
  117. //Replace(cmdstr,'$INIT',InitStr);
  118. //Replace(cmdstr,'$FINI',FiniStr);
  119. Replace(cmdstr,'$SONAME',SoNameStr);
  120. Replace(cmdstr,'$MAP',mapstr);
  121. //Replace(cmdstr,'$LTO',ltostr);
  122. Replace(cmdstr,'$GCSECTIONS',GCSectionsStr);
  123. success:=DoExec(FindUtil(utilsprefix+binstr),cmdstr,true,false);
  124. //SplitBinCmd(Info.DllCmd[2],binstr,cmdstr);
  125. //Replace(cmdstr,'$INPUT',current_module.objfilename );
  126. //Replace(cmdstr,'$EXE',maybequoted(current_module.exefilename));
  127. //DoExec(FindUtil(utilsprefix+binstr),cmdstr,false,false);
  128. MakeExecutable:=success;
  129. end;
  130. function tlinkerwasi.MakeSharedLibrary: boolean;
  131. var
  132. GCSectionsStr : ansistring;
  133. binstr, cmdstr : Tcmdstr;
  134. InitStr,
  135. FiniStr,
  136. SoNameStr : string[80];
  137. mapstr,ltostr : TCmdStr;
  138. success : Boolean;
  139. tmp : TCmdStrListItem;
  140. tempFileName : ansistring;
  141. begin
  142. //Result := true;
  143. //Result:=inherited MakeSharedLibrary;
  144. if (cs_link_smart in current_settings.globalswitches) and
  145. create_smartlink_sections then
  146. GCSectionsStr:='--gc-sections'
  147. else
  148. GCSectionsStr:='';
  149. SoNameStr:='';
  150. SplitBinCmd(Info.DllCmd[1],binstr,cmdstr);
  151. Replace(cmdstr,'$EXE',maybequoted(current_module.exefilename));
  152. tmp := TCmdStrListItem(ObjectFiles.First);
  153. while Assigned(tmp) do begin
  154. cmdstr := tmp.Str+ ' ' + cmdstr;
  155. tmp := TCmdStrListItem(tmp.Next);
  156. end;
  157. if HasExports then
  158. cmdstr := cmdstr + ' --export-dynamic'; //' --export-dynamic';
  159. cmdstr := cmdstr + ' --no-entry --allow-undefined';
  160. if (cs_link_strip in current_settings.globalswitches) then
  161. begin
  162. { only remove non global symbols and debugging info for a library }
  163. cmdstr := cmdstr + ' --strip-all';
  164. end;
  165. //Replace(cmdstr,'$OPT',Info.ExtraOptions);
  166. //Replace(cmdstr,'$RES',maybequoted(outputexedir+Info.ResName));
  167. //Replace(cmdstr,'$INIT',InitStr);
  168. //Replace(cmdstr,'$FINI',FiniStr);
  169. Replace(cmdstr,'$SONAME',SoNameStr);
  170. //Replace(cmdstr,'$MAP',mapstr);
  171. //Replace(cmdstr,'$LTO',ltostr);
  172. Replace(cmdstr,'$GCSECTIONS',GCSectionsStr);
  173. writeln(utilsprefix+binstr,' ',cmdstr);
  174. success:=DoExec(FindUtil(utilsprefix+binstr),cmdstr,true,false);
  175. //SplitBinCmd(Info.DllCmd[2],binstr,cmdstr);
  176. //Replace(cmdstr,'$INPUT',current_module.objfilename );
  177. //Replace(cmdstr,'$EXE',maybequoted(current_module.exefilename));
  178. //DoExec(FindUtil(utilsprefix+binstr),cmdstr,false,false);
  179. MakeSharedLibrary:=success;
  180. end;
  181. { texportlibwasi }
  182. procedure texportlibwasi.preparelib(const s: string);
  183. begin
  184. //nothing to happen. wasm files are modules
  185. end;
  186. procedure texportlibwasi.exportprocedure(hp: texported_item);
  187. var
  188. nm : TSymStr;
  189. begin
  190. nm := tprocdef(tprocsym(hp.sym).ProcdefList[0]).mangledname;
  191. current_asmdata.asmlists[al_exports].Concat(tai_export_name.create(hp.name^, nm, ie_Func));
  192. end;
  193. procedure texportlibwasi.exportvar(hp: texported_item);
  194. begin
  195. //inherited exportvar(hp);
  196. end;
  197. procedure texportlibwasi.generatelib;
  198. begin
  199. //inherited generatelib;
  200. end;
  201. initialization
  202. RegisterTarget(system_wasm32_wasi_info);
  203. RegisterImport(system_wasm32_wasi, timportlibwasi);
  204. RegisterExport(system_wasm32_wasi, texportlibwasi);
  205. RegisterLinker(ld_wasi, tlinkerwasi);
  206. end.