mirror of
https://gitlab.com/freepascal.org/fpc/source.git
synced 2025-04-05 05:58:02 +02:00
194 lines
6.0 KiB
ObjectPascal
194 lines
6.0 KiB
ObjectPascal
{
|
|
Copyright (c) 2021 by Nikolay Nikolov
|
|
|
|
WebAssembly version of some node tree helper routines
|
|
|
|
This program is free software; you can redistribute it and/or modify
|
|
it under the terms of the GNU General Public License as published by
|
|
the Free Software Foundation; either version 2 of the License, or
|
|
(at your option) any later version.
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with this program; if not, write to the Free Software
|
|
Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
|
|
|
****************************************************************************
|
|
}
|
|
unit nwasmutil;
|
|
|
|
{$i fpcdefs.inc}
|
|
|
|
interface
|
|
|
|
uses
|
|
ngenutil,
|
|
symsym;
|
|
|
|
type
|
|
|
|
{ twasmnodeutils }
|
|
|
|
twasmnodeutils = class(tnodeutils)
|
|
public
|
|
class procedure insertbssdata(sym : tstaticvarsym); override;
|
|
class procedure InsertObjectInfo; override;
|
|
end;
|
|
|
|
implementation
|
|
|
|
uses
|
|
cclasses,
|
|
globtype,globals,
|
|
cpubase,
|
|
aasmbase,aasmdata,aasmtai,aasmcpu,
|
|
hlcgobj,hlcgcpu,
|
|
symdef,symtype,symconst,symcpu,
|
|
fmodule;
|
|
|
|
{ twasmnodeutils }
|
|
|
|
class procedure twasmnodeutils.insertbssdata(sym: tstaticvarsym);
|
|
var
|
|
symcpu: tcpustaticvarsym;
|
|
begin
|
|
symcpu:=tcpustaticvarsym(sym);
|
|
if symcpu.is_wasm_global then
|
|
begin
|
|
if sym.globalasmsym then
|
|
current_asmdata.asmlists[al_start].Concat(tai_globaltype.create_global(symcpu.mangledname,symcpu.get_wasm_global_vardef_type,false,symcpu.vardef))
|
|
else
|
|
current_asmdata.asmlists[al_start].Concat(tai_globaltype.create_local(symcpu.mangledname,symcpu.get_wasm_global_vardef_type,false,symcpu.vardef));
|
|
end
|
|
else
|
|
inherited;
|
|
end;
|
|
|
|
class procedure twasmnodeutils.InsertObjectInfo;
|
|
|
|
var
|
|
modules: TFPList;
|
|
|
|
function ModuleExists(m: tmodule): Boolean;
|
|
var
|
|
q: tmodule;
|
|
begin
|
|
result:=modules.IndexOf(Pointer(m))>=0;
|
|
end;
|
|
|
|
procedure AddModule(m: tmodule);
|
|
begin
|
|
modules.Add(Pointer(m));
|
|
end;
|
|
|
|
procedure WriteImportDll(list: TAsmList; proc: tprocdef);
|
|
begin
|
|
thlcgwasm(hlcg).g_procdef(list,proc,false);
|
|
list.Concat(tai_import_module.create(proc.mangledname,proc.import_dll^));
|
|
list.Concat(tai_import_name.create(proc.mangledname,proc.import_name^));
|
|
end;
|
|
|
|
procedure InsertModuleInfo(list : TAsmList;u: tmodule);
|
|
var
|
|
i: Integer;
|
|
def : tdef;
|
|
proc : tprocdef;
|
|
cur_unit : tused_unit;
|
|
begin
|
|
if ModuleExists(u) then
|
|
exit;
|
|
AddModule(u);
|
|
|
|
cur_unit:=tused_unit(u.used_units.First);
|
|
while assigned(cur_unit) do
|
|
begin
|
|
InsertModuleInfo(list,cur_unit.u);
|
|
cur_unit:=tused_unit(cur_unit.Next);
|
|
end;
|
|
|
|
if ((u.moduleflags * [mf_init,mf_finalize])<>[]) and assigned(u.globalsymtable) then
|
|
begin
|
|
if mf_init in u.moduleflags then
|
|
list.Concat(tai_functype.create(make_mangledname('INIT$',u.globalsymtable,''),TWasmFuncType.Create([],[]),false));
|
|
if mf_finalize in u.moduleflags then
|
|
list.Concat(tai_functype.create(make_mangledname('FINALIZE$',u.globalsymtable,''),TWasmFuncType.Create([],[]),false));
|
|
end;
|
|
for i:=0 to u.deflist.Count-1 do
|
|
begin
|
|
def:=tdef(u.deflist[i]);
|
|
if assigned(def) and (tdef(def).typ = procdef) then
|
|
begin
|
|
proc := tprocdef(def);
|
|
if (po_external in proc.procoptions) and (po_has_importdll in proc.procoptions) then
|
|
WriteImportDll(list,proc)
|
|
else if not proc.owner.iscurrentunit or (po_external in proc.procoptions) then
|
|
thlcgwasm(hlcg).g_procdef(list,proc,false)
|
|
else
|
|
thlcgwasm(hlcg).g_procdef(list,proc,true);
|
|
end;
|
|
end;
|
|
end;
|
|
|
|
var
|
|
i : integer;
|
|
def : tdef;
|
|
proc : tprocdef;
|
|
list : TAsmList;
|
|
cur_unit: tused_unit;
|
|
begin
|
|
inherited;
|
|
|
|
modules:=TFPList.Create;
|
|
|
|
list:=current_asmdata.asmlists[al_start];
|
|
|
|
list.Concat(tai_globaltype.create(STACK_POINTER_SYM,wbt_i32,false));
|
|
|
|
if ts_wasm_threads in current_settings.targetswitches then
|
|
begin
|
|
list.Concat(tai_globaltype.create(TLS_SIZE_SYM,wbt_i32,true));
|
|
list.Concat(tai_globaltype.create(TLS_ALIGN_SYM,wbt_i32,true));
|
|
list.Concat(tai_globaltype.create(TLS_BASE_SYM,wbt_i32,false));
|
|
end;
|
|
|
|
if ts_wasm_native_exceptions in current_settings.targetswitches then
|
|
begin
|
|
list.Concat(tai_tagtype.create(FPC_EXCEPTION_TAG_SYM, []));
|
|
list.Concat(tai_symbol.Create_Weak(current_asmdata.WeakRefAsmSymbol(FPC_EXCEPTION_TAG_SYM,AT_WASM_EXCEPTION_TAG),0));
|
|
end;
|
|
|
|
for i:=0 to current_module.deflist.Count-1 do
|
|
begin
|
|
def:=tdef(current_module.deflist[i]);
|
|
{ since commit 48986 deflist might have NIL entries }
|
|
if assigned(def) and (def.typ=procdef) then
|
|
begin
|
|
proc := tprocdef(def);
|
|
if po_external in proc.procoptions then
|
|
if po_has_importdll in proc.procoptions then
|
|
WriteImportDll(list,proc)
|
|
else
|
|
thlcgwasm(hlcg).g_procdef(list,proc,false);
|
|
end;
|
|
end;
|
|
create_hlcodegen;
|
|
InsertModuleInfo(list,current_module);
|
|
cur_unit:=tused_unit(usedunits.First);
|
|
while assigned(cur_unit) do
|
|
begin
|
|
InsertModuleInfo(list,cur_unit.u);
|
|
cur_unit:=tused_unit(cur_unit.Next);
|
|
end;
|
|
destroy_hlcodegen;
|
|
|
|
modules.Free;
|
|
end;
|
|
|
|
begin
|
|
cnodeutils:=twasmnodeutils;
|
|
end.
|