t_wasi.pas 6.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269
  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. ExeCmd[1] := 'wasm-ld $SONAME $GCSECTIONS $MAP -o $EXE';
  70. DllCmd[1] := 'wasm-ld $SONAME $GCSECTIONS $MAP -o $EXE';
  71. end;
  72. end;
  73. procedure tlinkerwasi.InitSysInitUnitName;
  74. begin
  75. if current_module.islibrary then
  76. sysinitunit:='si_dll'
  77. else
  78. sysinitunit:='si_prc';
  79. end;
  80. function tlinkerwasi.MakeExecutable:boolean;
  81. var
  82. GCSectionsStr : ansistring;
  83. binstr, cmdstr : Tcmdstr;
  84. InitStr,
  85. FiniStr,
  86. SoNameStr : string[80];
  87. mapstr,ltostr : TCmdStr;
  88. success : Boolean;
  89. tmp : TCmdStrListItem;
  90. tempFileName : ansistring;
  91. begin
  92. if not(cs_link_nolink in current_settings.globalswitches) then
  93. Message1(exec_i_linking,current_module.exefilename);
  94. { Create some replacements }
  95. mapstr:='';
  96. if (cs_link_map in current_settings.globalswitches) then
  97. mapstr:='-Map '+maybequoted(ChangeFileExt(current_module.exefilename,'.map'));
  98. if (cs_link_smart in current_settings.globalswitches) and
  99. create_smartlink_sections then
  100. GCSectionsStr:='--gc-sections'
  101. else
  102. GCSectionsStr:='';
  103. SoNameStr:='';
  104. SplitBinCmd(Info.ExeCmd[1],binstr,cmdstr);
  105. Replace(cmdstr,'$EXE',maybequoted(current_module.exefilename));
  106. tmp := TCmdStrListItem(ObjectFiles.First);
  107. while Assigned(tmp) do begin
  108. cmdstr := tmp.Str+ ' ' + cmdstr;
  109. tmp := TCmdStrListItem(tmp.Next);
  110. end;
  111. // if HasExports then
  112. // cmdstr := cmdstr + ' --export-dynamic'; //' --export-dynamic';
  113. cmdstr := cmdstr + ' --no-entry';
  114. if ts_wasm_threads in current_settings.targetswitches then
  115. begin
  116. cmdstr := cmdstr + ' --import-memory --shared-memory --initial-memory=16777216 --max-memory=16777216 -z stack-size=5242880 --global-base=1024';
  117. end;
  118. if (cs_link_strip in current_settings.globalswitches) then
  119. begin
  120. { only remove non global symbols and debugging info for a library }
  121. cmdstr := cmdstr + ' --strip-all';
  122. end;
  123. //Replace(cmdstr,'$OPT',Info.ExtraOptions);
  124. //Replace(cmdstr,'$RES',maybequoted(outputexedir+Info.ResName));
  125. //Replace(cmdstr,'$INIT',InitStr);
  126. //Replace(cmdstr,'$FINI',FiniStr);
  127. Replace(cmdstr,'$SONAME',SoNameStr);
  128. Replace(cmdstr,'$MAP',mapstr);
  129. //Replace(cmdstr,'$LTO',ltostr);
  130. Replace(cmdstr,'$GCSECTIONS',GCSectionsStr);
  131. success:=DoExec(FindUtil(utilsprefix+binstr),cmdstr,true,false);
  132. MakeExecutable:=success;
  133. end;
  134. function tlinkerwasi.MakeSharedLibrary: boolean;
  135. var
  136. GCSectionsStr : ansistring;
  137. binstr, cmdstr : Tcmdstr;
  138. InitStr,
  139. FiniStr,
  140. SoNameStr : string[80];
  141. mapstr,ltostr : TCmdStr;
  142. success : Boolean;
  143. tmp : TCmdStrListItem;
  144. tempFileName : ansistring;
  145. begin
  146. Result:=false;
  147. if not(cs_link_nolink in current_settings.globalswitches) then
  148. Message1(exec_i_linking,current_module.sharedlibfilename);
  149. { Create some replacements }
  150. mapstr:='';
  151. if (cs_link_map in current_settings.globalswitches) then
  152. mapstr:='-Map '+maybequoted(ChangeFileExt(current_module.sharedlibfilename,'.map'));
  153. if (cs_link_smart in current_settings.globalswitches) and
  154. create_smartlink_sections then
  155. GCSectionsStr:='--gc-sections'
  156. else
  157. GCSectionsStr:='';
  158. SoNameStr:='';
  159. SplitBinCmd(Info.DllCmd[1],binstr,cmdstr);
  160. Replace(cmdstr,'$EXE',maybequoted(current_module.sharedlibfilename));
  161. tmp := TCmdStrListItem(ObjectFiles.First);
  162. while Assigned(tmp) do begin
  163. cmdstr := tmp.Str+ ' ' + cmdstr;
  164. tmp := TCmdStrListItem(tmp.Next);
  165. end;
  166. // if HasExports then
  167. // cmdstr := cmdstr + ' --export-dynamic'; //' --export-dynamic';
  168. cmdstr := cmdstr + ' --no-entry';
  169. if (cs_link_strip in current_settings.globalswitches) then
  170. begin
  171. { only remove non global symbols and debugging info for a library }
  172. cmdstr := cmdstr + ' --strip-all';
  173. end;
  174. //Replace(cmdstr,'$OPT',Info.ExtraOptions);
  175. //Replace(cmdstr,'$RES',maybequoted(outputexedir+Info.ResName));
  176. //Replace(cmdstr,'$INIT',InitStr);
  177. //Replace(cmdstr,'$FINI',FiniStr);
  178. Replace(cmdstr,'$SONAME',SoNameStr);
  179. Replace(cmdstr,'$MAP',mapstr);
  180. //Replace(cmdstr,'$LTO',ltostr);
  181. Replace(cmdstr,'$GCSECTIONS',GCSectionsStr);
  182. success:=DoExec(FindUtil(utilsprefix+binstr),cmdstr,true,false);
  183. MakeSharedLibrary:=success;
  184. end;
  185. { texportlibwasi }
  186. procedure texportlibwasi.preparelib(const s: string);
  187. begin
  188. //nothing to happen. wasm files are modules
  189. end;
  190. procedure texportlibwasi.exportprocedure(hp: texported_item);
  191. var
  192. nm : TSymStr;
  193. begin
  194. nm := tprocdef(tprocsym(hp.sym).ProcdefList[0]).mangledname;
  195. current_asmdata.asmlists[al_exports].Concat(tai_export_name.create(hp.name^, nm, ie_Func));
  196. end;
  197. procedure texportlibwasi.exportvar(hp: texported_item);
  198. begin
  199. //inherited exportvar(hp);
  200. end;
  201. procedure texportlibwasi.generatelib;
  202. begin
  203. //inherited generatelib;
  204. end;
  205. initialization
  206. RegisterTarget(system_wasm32_wasi_info);
  207. RegisterImport(system_wasm32_wasi, timportlibwasi);
  208. RegisterExport(system_wasm32_wasi, texportlibwasi);
  209. RegisterLinker(ld_wasi, tlinkerwasi);
  210. end.