mirror of
https://gitlab.com/freepascal.org/fpc/source.git
synced 2025-05-02 22:33:43 +02:00
429 lines
18 KiB
ObjectPascal
429 lines
18 KiB
ObjectPascal
{
|
|
Copyright (c) 20011 by Jonas Maebe
|
|
|
|
LLVM 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 nllvmutil;
|
|
|
|
{$i fpcdefs.inc}
|
|
|
|
interface
|
|
|
|
uses
|
|
globtype,cclasses,
|
|
aasmbase,aasmdata,aasmllvmmetadata, ngenutil,
|
|
symtype,symconst,symsym,symdef;
|
|
|
|
|
|
type
|
|
tllvmnodeutils = class(tnodeutils)
|
|
strict protected
|
|
class procedure insertbsssym(list: tasmlist; sym: tstaticvarsym; size: asizeint; varalign: shortint; _typ:Tasmsymtype); override;
|
|
class procedure InsertUsedList(var usedsyms: tfpobjectlist; const usedsymsname: TSymStr);
|
|
class procedure InsertInitFiniList(var procdefs: tfplist; const initfinisymsname: TSymStr);
|
|
class procedure InsertAsanGlobals;
|
|
public
|
|
class procedure InsertObjectInfo; override;
|
|
class procedure RegisterUsedAsmSym(sym: TAsmSymbol; def: tdef; compileronly: boolean); override;
|
|
class procedure RegisterModuleInitFunction(pd: tprocdef); override;
|
|
class procedure RegisterModuleFiniFunction(pd: tprocdef); override;
|
|
end;
|
|
|
|
|
|
implementation
|
|
|
|
uses
|
|
verbose,cutils,globals,fmodule,systems,finput,versioncmp,
|
|
aasmtai,cpubase,llvmbase,aasmllvm,
|
|
aasmcnst,nllvmtcon,
|
|
symbase,symtable,defutil,
|
|
llvminfo,llvmtype,llvmdef,
|
|
objcasm;
|
|
|
|
class procedure tllvmnodeutils.insertbsssym(list: tasmlist; sym: tstaticvarsym; size: asizeint; varalign: shortint; _typ:Tasmsymtype);
|
|
var
|
|
asmsym: tasmsymbol;
|
|
field1, field2: tsym;
|
|
begin
|
|
if sym.globalasmsym then
|
|
asmsym:=current_asmdata.DefineAsmSymbol(sym.mangledname,AB_GLOBAL,_typ,sym.vardef)
|
|
else if tf_supports_hidden_symbols in target_info.flags then
|
|
asmsym:=current_asmdata.DefineAsmSymbol(sym.mangledname,AB_PRIVATE_EXTERN,_typ,sym.vardef)
|
|
else
|
|
asmsym:=current_asmdata.DefineAsmSymbol(sym.mangledname,AB_LOCAL,_typ,sym.vardef);
|
|
if not(vo_is_thread_var in sym.varoptions) then
|
|
list.concat(taillvmdecl.createdef(asmsym,sym,sym.vardef,nil,sec_data,varalign))
|
|
else if tf_section_threadvars in target_info.flags then
|
|
list.concat(taillvmdecl.createtls(asmsym,sym,sym.vardef,varalign))
|
|
else
|
|
list.concat(taillvmdecl.createdef(asmsym,sym,
|
|
get_threadvar_record(sym.vardef,field1,field2),
|
|
nil,sec_data,varalign));
|
|
end;
|
|
|
|
|
|
type
|
|
TTypedAsmSym = class
|
|
sym: TAsmSymbol;
|
|
def: tdef;
|
|
constructor Create(s: TAsmSymbol; d: tdef);
|
|
end;
|
|
|
|
|
|
constructor TTypedAsmSym.Create(s: TAsmSymbol; d: tdef);
|
|
begin
|
|
sym:=s;
|
|
def:=d;
|
|
end;
|
|
|
|
|
|
function TypedAsmSymComparer(p1, p2: Pointer): Integer;
|
|
var
|
|
sym1: TTypedAsmSym absolute p1;
|
|
sym2: TTypedAsmSym absolute p2;
|
|
begin
|
|
result:=CompareStr(sym1.sym.Name,sym2.sym.Name);
|
|
end;
|
|
|
|
|
|
class procedure tllvmnodeutils.InsertUsedList(var usedsyms: tfpobjectlist;
|
|
const usedsymsname: TSymStr);
|
|
var
|
|
useddef: tdef;
|
|
tcb: ttai_typedconstbuilder;
|
|
prevasmsym: TAsmSymbol;
|
|
typedsym: TTypedAsmSym;
|
|
uniquesyms, i: longint;
|
|
begin
|
|
if usedsyms.count<>0 then
|
|
begin
|
|
{ a symbol can appear multiple times -> sort the list so we can filter out doubles }
|
|
usedsyms.Sort(@TypedAsmSymComparer);
|
|
{ count uniques }
|
|
prevasmsym:=nil;
|
|
uniquesyms:=0;
|
|
for i:=0 to usedsyms.count-1 do
|
|
begin
|
|
typedsym:=TTypedAsmSym(usedsyms[i]);
|
|
if (prevasmsym<>typedsym.sym) and
|
|
{ even though we already filter on pure assembler routines when adding the symbols,
|
|
some may slip through because of forward definitions that are not yet resolved }
|
|
not((typedsym.def.typ=procdef) and
|
|
(po_assembler in tprocdef(typedsym.def).procoptions)) then
|
|
inc(uniquesyms);
|
|
prevasmsym:=typedsym.sym;
|
|
end;
|
|
{ emit uniques }
|
|
prevasmsym:=nil;
|
|
tcb:=ctai_typedconstbuilder.create([tcalo_new_section]);
|
|
tllvmtai_typedconstbuilder(tcb).appendingdef:=true;
|
|
useddef:=carraydef.getreusable(voidpointertype,uniquesyms);
|
|
tcb.maybe_begin_aggregate(useddef);
|
|
for i:=0 to usedsyms.count-1 do
|
|
begin
|
|
typedsym:=TTypedAsmSym(usedsyms[i]);
|
|
if (prevasmsym<>typedsym.sym) and
|
|
not((typedsym.def.typ=procdef) and
|
|
(po_assembler in tprocdef(typedsym.def).procoptions)) then
|
|
begin
|
|
tcb.queue_init(voidpointertype);
|
|
tcb.queue_emit_asmsym(typedsym.sym,typedsym.def);
|
|
prevasmsym:=typedsym.sym;
|
|
end;
|
|
end;
|
|
tcb.maybe_end_aggregate(useddef);
|
|
current_asmdata.AsmLists[al_globals].concatlist(
|
|
tcb.get_final_asmlist(
|
|
current_asmdata.DefineAsmSymbol(
|
|
usedsymsname,AB_GLOBAL,AT_DATA,useddef),useddef,sec_user,
|
|
'llvm.metadata',0
|
|
)
|
|
);
|
|
tcb.free;
|
|
end;
|
|
usedsyms.free;
|
|
usedsyms:=nil;
|
|
end;
|
|
|
|
|
|
class procedure tllvmnodeutils.InsertInitFiniList(var procdefs: tfplist; const initfinisymsname: TSymStr);
|
|
var
|
|
itemdef: trecorddef;
|
|
arraydef: tarraydef;
|
|
pd: tprocdef;
|
|
fields: array[0..2] of tdef;
|
|
tcb: ttai_typedconstbuilder;
|
|
i: longint;
|
|
begin
|
|
if procdefs.count<>0 then
|
|
begin
|
|
pd:=tprocdef(procdefs[0]);
|
|
fields[0]:=s32inttype;
|
|
fields[1]:=cprocvardef.getreusableprocaddr(pd,pc_address_only);
|
|
fields[2]:=voidpointertype;
|
|
itemdef:=llvmgettemprecorddef(fields,C_alignment,
|
|
targetinfos[target_info.system]^.alignment.recordalignmin);
|
|
include(itemdef.defoptions,df_llvm_no_struct_packing);
|
|
include(itemdef.defoptions,df_llvm_no_typename);
|
|
tcb:=ctai_typedconstbuilder.create([tcalo_new_section]);
|
|
tllvmtai_typedconstbuilder(tcb).appendingdef:=true;
|
|
arraydef:=carraydef.getreusable(itemdef,procdefs.Count);
|
|
tcb.maybe_begin_aggregate(arraydef);
|
|
for i:=0 to procdefs.count-1 do
|
|
begin
|
|
tcb.maybe_begin_aggregate(itemdef);
|
|
tcb.emit_ord_const(65535,s32inttype);
|
|
tcb.emit_procdef_const(tprocdef(procdefs[i]));
|
|
tcb.emit_tai(Tai_const.Create_sym(nil),voidpointertype);
|
|
tcb.maybe_end_aggregate(itemdef);
|
|
end;
|
|
tcb.maybe_end_aggregate(arraydef);
|
|
current_asmdata.AsmLists[al_globals].concatlist(
|
|
tcb.get_final_asmlist(
|
|
current_asmdata.DefineAsmSymbol(
|
|
initfinisymsname,AB_GLOBAL,AT_DATA,arraydef),arraydef,sec_data,
|
|
initfinisymsname,voidpointertype.alignment
|
|
)
|
|
);
|
|
tcb.free;
|
|
end;
|
|
end;
|
|
|
|
|
|
class procedure tllvmnodeutils.InsertAsanGlobals;
|
|
var
|
|
asanglobal,
|
|
asanglobals,
|
|
globalfileloc: tai_llvmbasemetadatanode;
|
|
hp: tai;
|
|
hpdecl: taillvmdecl;
|
|
sourcefile: tinputfile;
|
|
module: tmodule;
|
|
list: TAsmList;
|
|
asmlisttype: TAsmListType;
|
|
begin
|
|
if not(cs_sanitize_address in current_settings.moduleswitches) or
|
|
(llvmflag_sanitizer_attributes in llvmversion_properties[current_settings.llvmversion]) then
|
|
exit;
|
|
asanglobals:=nil;
|
|
module:=get_module(current_filepos.moduleindex);
|
|
for asmlisttype:=low(asmlisttype) to high(asmlisttype) do
|
|
begin
|
|
list:=current_asmdata.AsmLists[asmlisttype];
|
|
if not assigned(list) then
|
|
continue;
|
|
hp:=tai(list.first);
|
|
while assigned(hp) do
|
|
begin
|
|
if (hp.typ=ait_llvmdecl) and
|
|
(ldf_definition in taillvmdecl(hp).flags) and
|
|
(taillvmdecl(hp).def.typ<>procdef) then
|
|
begin
|
|
if not assigned(asanglobals) then
|
|
begin
|
|
asanglobals:=tai_llvmnamedmetadatanode.create('llvm.asan.globals');
|
|
current_asmdata.AsmLists[al_rotypedconsts].concat(asanglobals);
|
|
end;
|
|
hpdecl:=taillvmdecl(hp);
|
|
|
|
globalfileloc:=tai_llvmunnamedmetadatanode.create;
|
|
current_asmdata.AsmLists[al_rotypedconsts].concat(globalfileloc);
|
|
if assigned(hpdecl.sym) then
|
|
begin
|
|
sourcefile:=get_source_file(hpdecl.sym.fileinfo.moduleindex,hpdecl.sym.fileinfo.fileindex);
|
|
globalfileloc.addvalue(tai_simpletypedconst.create(charpointertype,tai_string.Create(sourcefile.name)));
|
|
globalfileloc.addvalue(tai_simpletypedconst.create(s32inttype,tai_const.Create_32bit(hpdecl.sym.fileinfo.line)));
|
|
globalfileloc.addvalue(tai_simpletypedconst.create(s32inttype,tai_const.Create_32bit(hpdecl.sym.fileinfo.column)));
|
|
end
|
|
else
|
|
begin
|
|
sourcefile:=current_module.sourcefiles.get_file(1);
|
|
globalfileloc.addvalue(tai_simpletypedconst.create(charpointertype,tai_string.Create(sourcefile.name)));
|
|
globalfileloc.addvalue(tai_simpletypedconst.create(s32inttype,tai_const.Create_32bit(1)));
|
|
globalfileloc.addvalue(tai_simpletypedconst.create(s32inttype,tai_const.Create_32bit(1)));
|
|
end;
|
|
|
|
asanglobal:=tai_llvmunnamedmetadatanode.create;
|
|
current_asmdata.AsmLists[al_rotypedconsts].concat(asanglobal);
|
|
asanglobal.addvalue(tai_simpletypedconst.create(cpointerdef.getreusable(hpdecl.def),tai_const.Create_sym(hpdecl.namesym)));
|
|
asanglobal.addvalue(tai_simpletypedconst.create(llvm_metadatatype,llvm_getmetadatareftypedconst(globalfileloc)));
|
|
if assigned(hpdecl.sym) then
|
|
asanglobal.addvalue(tai_simpletypedconst.create(llvm_metadatatype,tai_string.Create(hpdecl.sym.RealName)))
|
|
else
|
|
asanglobal.addvalue(tai_simpletypedconst.create(llvm_metadatatype,tai_string.Create(hpdecl.namesym.Name)));
|
|
{ dynamic init }
|
|
asanglobal.addvalue(tai_simpletypedconst.create(llvmbool1type,tai_const.Create_8bit(ord(false))));
|
|
{ no asan }
|
|
asanglobal.addvalue(tai_simpletypedconst.create(llvmbool1type,tai_const.Create_8bit(ord((ldf_vectorized in taillvmdecl(hp).flags)))));
|
|
|
|
asanglobals.addvalue(tai_simpletypedconst.create(llvm_metadatatype,llvm_getmetadatareftypedconst(asanglobal)));
|
|
end;
|
|
hp:=tai(hp.next);
|
|
end;
|
|
end;
|
|
end;
|
|
|
|
|
|
class procedure tllvmnodeutils.InsertObjectInfo;
|
|
var
|
|
llvmmoduleflags,
|
|
objcmoduleflag,
|
|
dwarfversionflag: tai_llvmbasemetadatanode;
|
|
objcabiversion: longint;
|
|
begin
|
|
InsertAsanGlobals;
|
|
|
|
llvmmoduleflags:=tai_llvmnamedmetadatanode.create('llvm.module.flags');
|
|
current_asmdata.AsmLists[al_rotypedconsts].Concat(llvmmoduleflags);
|
|
|
|
if (m_objectivec1 in current_settings.modeswitches) then
|
|
begin
|
|
{ Objective-C ABI version }
|
|
if not(target_info.system in [system_powerpc_darwin,system_powerpc64_darwin,system_i386_darwin,system_x86_64_darwin]) or
|
|
(MacOSXVersionMin.relationto(10,5,0)>=0) then
|
|
objcabiversion:=2
|
|
else
|
|
objcabiversion:=1;
|
|
objcmoduleflag:=tai_llvmunnamedmetadatanode.create;
|
|
objcmoduleflag.addvalue(tai_simpletypedconst.create(s32inttype,tai_const.Create_32bit(1)));
|
|
objcmoduleflag.addvalue(tai_simpletypedconst.create(charpointertype,tai_string.Create('Objective-C Version')));
|
|
objcmoduleflag.addvalue(tai_simpletypedconst.create(s32inttype,tai_const.Create_32bit(objcabiversion)));
|
|
llvmmoduleflags.addvalue(llvm_getmetadatareftypedconst(objcmoduleflag));
|
|
current_asmdata.AsmLists[al_rotypedconsts].Concat(objcmoduleflag);
|
|
|
|
{ image info version }
|
|
objcmoduleflag:=tai_llvmunnamedmetadatanode.create;
|
|
objcmoduleflag.addvalue(tai_simpletypedconst.create(s32inttype,tai_const.Create_32bit(1)));
|
|
objcmoduleflag.addvalue(tai_simpletypedconst.create(charpointertype,tai_string.Create('Objective-C Image Info Version')));
|
|
objcmoduleflag.addvalue(tai_simpletypedconst.create(s32inttype,tai_const.Create_32bit(0)));
|
|
llvmmoduleflags.addvalue(llvm_getmetadatareftypedconst(objcmoduleflag));
|
|
current_asmdata.AsmLists[al_rotypedconsts].Concat(objcmoduleflag);
|
|
|
|
{ image info section }
|
|
objcmoduleflag:=tai_llvmunnamedmetadatanode.create;
|
|
objcmoduleflag.addvalue(tai_simpletypedconst.create(s32inttype,tai_const.Create_32bit(1)));
|
|
objcmoduleflag.addvalue(tai_simpletypedconst.create(charpointertype,tai_string.Create('Objective-C Image Info Section')));
|
|
objcmoduleflag.addvalue(tai_simpletypedconst.create(charpointertype,tai_string.Create(objc_section_name(sec_objc_image_info))));
|
|
llvmmoduleflags.addvalue(llvm_getmetadatareftypedconst(objcmoduleflag));
|
|
current_asmdata.AsmLists[al_rotypedconsts].Concat(objcmoduleflag);
|
|
|
|
{ garbage collection }
|
|
objcmoduleflag:=tai_llvmunnamedmetadatanode.create;
|
|
objcmoduleflag.addvalue(tai_simpletypedconst.create(s32inttype,tai_const.Create_32bit(1)));
|
|
objcmoduleflag.addvalue(tai_simpletypedconst.create(charpointertype,tai_string.Create('Objective-C Garbage Collection')));
|
|
objcmoduleflag.addvalue(tai_simpletypedconst.create(s32inttype,tai_const.Create_32bit(0)));
|
|
llvmmoduleflags.addvalue(llvm_getmetadatareftypedconst(objcmoduleflag));
|
|
current_asmdata.AsmLists[al_rotypedconsts].Concat(objcmoduleflag);
|
|
|
|
{ insert newly created defs in the implementation rather than interface symtable
|
|
(the interface symtable is sealed at this point) }
|
|
end;
|
|
|
|
{ debug information }
|
|
if (([cs_debuginfo,cs_lineinfo]*current_settings.moduleswitches)<>[]) and
|
|
(target_dbg.id in [dbg_dwarf2,dbg_dwarf3,dbg_dwarf4]) then
|
|
begin
|
|
{ the debug info version is the version of the debug info metadata
|
|
format }
|
|
dwarfversionflag:=tai_llvmunnamedmetadatanode.create;
|
|
dwarfversionflag.addvalue(tai_simpletypedconst.create(s32inttype,tai_const.Create_32bit(2)));
|
|
dwarfversionflag.addvalue(tai_simpletypedconst.create(charpointertype,tai_string.Create('Debug Info Version')));
|
|
dwarfversionflag.addvalue(tai_simpletypedconst.create(s32inttype,tai_const.Create_32bit(llvm_debuginfo_metadata_format[current_settings.llvmversion])));
|
|
llvmmoduleflags.addvalue(llvm_getmetadatareftypedconst(dwarfversionflag));
|
|
current_asmdata.AsmLists[al_rotypedconsts].Concat(dwarfversionflag);
|
|
|
|
{ dwarf version }
|
|
dwarfversionflag:=tai_llvmunnamedmetadatanode.create;
|
|
dwarfversionflag.addvalue(tai_simpletypedconst.create(s32inttype,tai_const.Create_32bit(2)));
|
|
dwarfversionflag.addvalue(tai_simpletypedconst.create(charpointertype,tai_string.Create('Dwarf Version')));
|
|
case target_dbg.id of
|
|
dbg_dwarf2:
|
|
dwarfversionflag.addvalue(tai_simpletypedconst.create(s32inttype,tai_const.Create_32bit(2)));
|
|
dbg_dwarf3:
|
|
dwarfversionflag.addvalue(tai_simpletypedconst.create(s32inttype,tai_const.Create_32bit(3)));
|
|
dbg_dwarf4:
|
|
dwarfversionflag.addvalue(tai_simpletypedconst.create(s32inttype,tai_const.Create_32bit(4)));
|
|
else
|
|
internalerror(2022022012);
|
|
end;
|
|
llvmmoduleflags.addvalue(llvm_getmetadatareftypedconst(dwarfversionflag));
|
|
current_asmdata.AsmLists[al_rotypedconsts].Concat(dwarfversionflag);
|
|
end;
|
|
|
|
symtablestack.push(current_module.localsymtable);
|
|
|
|
{ add the llvm.compiler.used array }
|
|
InsertUsedList(current_module.llvmcompilerusedsyms,'llvm.compiler.used');
|
|
{ add the llvm.used array }
|
|
InsertUsedList(current_module.llvmusedsyms,'llvm.used');
|
|
{ add the llvm.global_ctors array }
|
|
InsertInitFiniList(current_module.llvminitprocs,'llvm.global_ctors');
|
|
{ add the llvm.global_dtors array }
|
|
InsertInitFiniList(current_module.llvmfiniprocs,'llvm.global_dtors');
|
|
|
|
{ add "type xx = .." statements for all used recorddefs }
|
|
with TLLVMTypeInfo.Create do
|
|
begin
|
|
inserttypeinfo;
|
|
free;
|
|
end;
|
|
|
|
symtablestack.pop(current_module.localsymtable);
|
|
end;
|
|
|
|
|
|
class procedure tllvmnodeutils.RegisterUsedAsmSym(sym: TAsmSymbol; def: tdef; compileronly: boolean);
|
|
var
|
|
last: TTypedAsmSym;
|
|
begin
|
|
if compileronly then
|
|
begin
|
|
{ filter multiple adds in succession here already }
|
|
last:=TTypedAsmSym(current_module.llvmcompilerusedsyms.Last);
|
|
if not assigned(last) or
|
|
(last.sym<>sym) then
|
|
current_module.llvmcompilerusedsyms.Add(TTypedAsmSym.Create(sym,def))
|
|
end
|
|
else
|
|
begin
|
|
last:=TTypedAsmSym(current_module.llvmusedsyms.Last);
|
|
if not assigned(last) or
|
|
(last.sym<>sym) then
|
|
current_module.llvmusedsyms.Add(TTypedAsmSym.Create(sym,def))
|
|
end;
|
|
end;
|
|
|
|
|
|
class procedure tllvmnodeutils.RegisterModuleInitFunction(pd: tprocdef);
|
|
begin
|
|
current_module.llvminitprocs.add(pd);
|
|
end;
|
|
|
|
|
|
class procedure tllvmnodeutils.RegisterModuleFiniFunction(pd: tprocdef);
|
|
begin
|
|
current_module.llvmfiniprocs.add(pd);
|
|
end;
|
|
|
|
|
|
begin
|
|
cnodeutils:=tllvmnodeutils;
|
|
end.
|
|
|