| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919192019211922192319241925192619271928192919301931193219331934193519361937193819391940194119421943194419451946194719481949195019511952195319541955195619571958195919601961196219631964196519661967196819691970197119721973197419751976197719781979198019811982198319841985198619871988198919901991199219931994199519961997199819992000200120022003200420052006200720082009201020112012201320142015201620172018201920202021202220232024202520262027202820292030203120322033203420352036203720382039204020412042204320442045204620472048204920502051205220532054205520562057205820592060206120622063206420652066206720682069207020712072207320742075207620772078207920802081208220832084208520862087208820892090209120922093209420952096209720982099210021012102210321042105210621072108210921102111211221132114211521162117211821192120212121222123212421252126212721282129213021312132213321342135213621372138213921402141214221432144214521462147214821492150215121522153215421552156215721582159216021612162216321642165216621672168216921702171217221732174217521762177217821792180218121822183218421852186218721882189219021912192219321942195219621972198219922002201220222032204220522062207220822092210221122122213221422152216221722182219222022212222222322242225222622272228222922302231223222332234223522362237223822392240224122422243224422452246224722482249225022512252225322542255225622572258225922602261226222632264226522662267226822692270227122722273227422752276227722782279228022812282228322842285228622872288228922902291229222932294229522962297229822992300230123022303230423052306230723082309231023112312231323142315231623172318231923202321232223232324232523262327232823292330233123322333233423352336233723382339234023412342234323442345234623472348234923502351235223532354235523562357235823592360236123622363236423652366236723682369237023712372237323742375237623772378237923802381238223832384238523862387238823892390239123922393239423952396239723982399240024012402240324042405240624072408240924102411241224132414241524162417241824192420242124222423242424252426242724282429243024312432243324342435243624372438243924402441244224432444244524462447244824492450245124522453245424552456245724582459246024612462246324642465246624672468246924702471247224732474247524762477247824792480248124822483248424852486248724882489249024912492249324942495249624972498249925002501250225032504250525062507250825092510251125122513251425152516251725182519252025212522252325242525252625272528252925302531253225332534253525362537253825392540254125422543254425452546254725482549255025512552255325542555255625572558255925602561256225632564256525662567256825692570257125722573257425752576257725782579258025812582258325842585258625872588258925902591259225932594259525962597259825992600260126022603260426052606260726082609261026112612261326142615261626172618261926202621262226232624262526262627262826292630263126322633263426352636263726382639264026412642264326442645264626472648264926502651265226532654265526562657265826592660266126622663266426652666266726682669267026712672267326742675267626772678267926802681268226832684268526862687268826892690269126922693269426952696269726982699270027012702270327042705270627072708270927102711271227132714271527162717271827192720272127222723272427252726272727282729273027312732273327342735273627372738273927402741274227432744274527462747274827492750275127522753275427552756275727582759276027612762276327642765276627672768276927702771277227732774277527762777277827792780278127822783278427852786278727882789279027912792279327942795279627972798279928002801280228032804280528062807280828092810281128122813281428152816281728182819282028212822282328242825282628272828282928302831283228332834283528362837283828392840284128422843284428452846284728482849285028512852285328542855285628572858285928602861286228632864286528662867286828692870287128722873287428752876287728782879288028812882288328842885288628872888288928902891289228932894289528962897289828992900290129022903290429052906290729082909291029112912291329142915291629172918291929202921292229232924292529262927292829292930293129322933293429352936293729382939294029412942294329442945294629472948294929502951295229532954295529562957295829592960296129622963 | {    Copyright (c) 1998-2004 by Peter Vreman    This unit handles the assemblerfile write and assembler calls of FPC    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. ****************************************************************************}{# @abstract(This unit handles the assembler file write and assembler calls of FPC)   Handles the calls to the actual external assemblers, as well as the generation   of object files for smart linking. Also contains the base class for writing   the assembler statements to file.}unit assemble;{$i fpcdefs.inc}interface    uses      SysUtils,      systems,globtype,globals,aasmbase,aasmtai,aasmdata,ogbase,owbase,finput;    const       { maximum of aasmoutput lists there will be }       maxoutputlists = ord(high(tasmlisttype))+1;       { buffer size for writing the .s file }       AsmOutSize=32768*4;    type      TAssembler=class(TObject)      public      {assembler info}        asminfo     : pasminfo;      {filenames}        path        : TPathStr;        name        : string;        AsmFileName,         { current .s and .o file }        ObjFileName,        ppufilename  : TPathStr;        asmprefix    : string;        SmartAsm     : boolean;        SmartFilesCount,        SmartHeaderCount : longint;        Constructor Create(info: pasminfo; smart:boolean);virtual;        Destructor Destroy;override;        procedure NextSmartName(place:tcutplace);        procedure MakeObject;virtual;abstract;      end;      TExternalAssembler = class;      IExternalAssemblerOutputFileDecorator=interface        function LinePrefix: AnsiString;        function LinePostfix: AnsiString;        function LineFilter(const s: AnsiString): AnsiString;        function LineEnding(const deflineending: ShortString): ShortString;      end;      TExternalAssemblerOutputFile=class      private        fdecorator: IExternalAssemblerOutputFileDecorator;      protected        owner: TExternalAssembler;      {outfile}        AsmSize,        AsmStartSize,        outcnt   : longint;        outbuf   : array[0..AsmOutSize-1] of char;        outfile  : file;        fioerror : boolean;        linestart: boolean;        Procedure AsmClear;        Procedure MaybeAddLinePrefix;        Procedure MaybeAddLinePostfix;        Procedure AsmWriteAnsiStringUnfiltered(const s: ansistring);      public        Constructor Create(_owner: TExternalAssembler);        Procedure RemoveAsm;virtual;        Procedure AsmFlush;        { mark the current output as the "empty" state (i.e., it only contains          headers/directives etc }        Procedure MarkEmpty;        { clears the assembler output if nothing was added since it was marked          as empty, and returns whether it was empty }        function ClearIfEmpty: boolean;        { these routines will write the filtered version of their argument          according to the current decorator }        procedure AsmWriteFiltered(const c:char);        procedure AsmWriteFiltered(const s:string);        procedure AsmWriteFiltered(const s:ansistring);        procedure AsmWriteFiltered(p:pchar; len: longint);        {# Write a string to the assembler file }        Procedure AsmWrite(const c:char);        Procedure AsmWrite(const s:string);        Procedure AsmWrite(const s:ansistring);        {# Write a string to the assembler file }        Procedure AsmWritePChar(p:pchar);        {# Write a string to the assembler file followed by a new line }        Procedure AsmWriteLn(const c:char);        Procedure AsmWriteLn(const s:string);        Procedure AsmWriteLn(const s:ansistring);        {# Write a new line to the assembler file }        Procedure AsmLn; virtual;        procedure AsmCreate(Aplace:tcutplace);        procedure AsmClose;        property ioerror: boolean read fioerror;        property decorator: IExternalAssemblerOutputFileDecorator read fdecorator write fdecorator;      end;      {# This is the base class which should be overridden for each each         assembler writer. It is used to actually assembler a file,         and write the output to the assembler file.      }      TExternalAssembler=class(TAssembler)      private       { output writer }        fwriter: TExternalAssemblerOutputFile;        ffreewriter: boolean;        procedure CreateSmartLinkPath(const s:TPathStr);      protected      {input source info}        lastfileinfo : tfileposinfo;        infile,        lastinfile   : tinputfile;      {last section type written}        lastsectype : TAsmSectionType;        procedure ResetSourceLines;        procedure WriteSourceLine(hp: tailineinfo);        procedure WriteTempalloc(hp: tai_tempalloc);        procedure WriteRealConstAsBytes(hp: tai_realconst; const dbdir: string; do_line: boolean);        function WriteComments(var hp: tai): boolean;        function single2str(d : single) : string; virtual;        function double2str(d : double) : string; virtual;        function extended2str(e : extended) : string; virtual;        function sleb128tostr(a : int64) : string;        function uleb128tostr(a : qword) : string;        Function DoPipe:boolean; virtual;        function CreateNewAsmWriter: TExternalAssemblerOutputFile; virtual;        {# Return true if the external assembler should run again }        function RerunAssembler: boolean; virtual;      public        {# Returns the complete path and executable name of the assembler           program.           It first tries looking in the UTIL directory if specified,           otherwise it searches in the free pascal binary directory, in           the current working directory and then in the  directories           in the $PATH environment.}        Function  FindAssembler:string;        {# Actually does the call to the assembler file. Returns false           if the assembling of the file failed.}        Function  CallAssembler(const command:string; const para:TCmdStr):Boolean;        Function  DoAssemble:boolean;virtual;        {# This routine should be overridden for each assembler, it is used           to actually write the abstract assembler stream to file.}        procedure WriteTree(p:TAsmList);virtual;        {# This routine should be overridden for each assembler, it is used           to actually write all the different abstract assembler streams           by calling for each stream type, the @var(WriteTree) method.}        procedure WriteAsmList;virtual;        {# Constructs the command line for calling the assembler }        function MakeCmdLine: TCmdStr; virtual;      public        Constructor Create(info: pasminfo; smart: boolean); override; final;        Constructor CreateWithWriter(info: pasminfo; wr: TExternalAssemblerOutputFile; freewriter, smart: boolean); virtual;        procedure MakeObject;override;        destructor Destroy; override;        property writer: TExternalAssemblerOutputFile read fwriter;      end;      TExternalAssemblerClass = class of TExternalAssembler;      { TInternalAssembler }      TInternalAssembler=class(TAssembler)      private{$ifdef ARM}        { true, if thumb instructions are generated }        Code16 : Boolean;{$endif ARM}        FCObjOutput : TObjOutputclass;        FCInternalAr : TObjectWriterClass;        { the aasmoutput lists that need to be processed }        lists        : byte;        list         : array[1..maxoutputlists] of TAsmList;        { current processing }        currlistidx  : byte;        currlist     : TAsmList;        procedure WriteStab(p:pchar);        function  MaybeNextList(var hp:Tai):boolean;        function  SetIndirectToSymbol(hp: Tai; const indirectname: string): Boolean;        function  TreePass0(hp:Tai):Tai;        function  TreePass1(hp:Tai):Tai;        function  TreePass2(hp:Tai):Tai;        procedure writetree;        procedure writetreesmart;      protected        ObjData   : TObjData;        ObjOutput : tObjOutput;        property CObjOutput:TObjOutputclass read FCObjOutput write FCObjOutput;        property CInternalAr : TObjectWriterClass read FCInternalAr write FCInternalAr;      public        constructor Create(info: pasminfo; smart: boolean);override;        destructor  destroy;override;        procedure MakeObject;override;      end;    TAssemblerClass = class of TAssembler;    Procedure GenerateAsm(smart:boolean);    { get an instance of an external GNU-style assembler that is compatible      with the current target, reusing an existing writer. Used by the LLVM      target to write inline assembler }    function GetExternalGnuAssemblerWithAsmInfoWriter(info: pasminfo; wr: TExternalAssemblerOutputFile): TExternalAssembler;    procedure RegisterAssembler(const r:tasminfo;c:TAssemblerClass);Implementation    uses{$ifdef hasunix}      unix,{$endif}      cutils,cfileutl,{$ifdef memdebug}      cclasses,{$endif memdebug}{$ifdef OMFOBJSUPPORT}      omfbase,      ogomf,{$endif OMFOBJSUPPORT}{$if defined(cpuextended) and defined(FPC_HAS_TYPE_EXTENDED)}{$else}{$ifdef FPC_SOFT_FPUX80}      sfpux80,{$endif FPC_SOFT_FPUX80}{$endif}{$ifdef WASM}      ogwasm,{$endif WASM}      cscript,fmodule,verbose,      cpubase,cpuinfo,triplet,      aasmcpu;    var      CAssembler : array[tasm] of TAssemblerClass;    function fixline(const s:string):string;     {       return s with all leading and ending spaces and tabs removed     }      var        i,j,k : integer;      begin        i:=length(s);        while (i>0) and (s[i] in [#9,' ']) do          dec(i);        j:=1;        while (j<i) and (s[j] in [#9,' ']) do          inc(j);        result := Copy(s, j, i - j + 1);        for k:=1 to length(result) do          if result[k] in [#0..#31,#127..#255] then            result[k]:='.';      end;{*****************************************************************************                                   TAssembler*****************************************************************************}    Constructor TAssembler.Create(info: pasminfo; smart: boolean);      begin        asminfo:=info;      { load start values }        AsmFileName:=current_module.AsmFilename;        ObjFileName:=current_module.ObjFileName;        name:=Lower(current_module.modulename^);        path:=current_module.outputpath;        asmprefix := current_module.asmprefix^;        if current_module.outputpath = '' then          ppufilename := ''        else          ppufilename := current_module.ppufilename;        SmartAsm:=smart;        SmartFilesCount:=0;        SmartHeaderCount:=0;        SmartLinkOFiles.Clear;      end;    Destructor TAssembler.Destroy;      begin      end;    procedure TAssembler.NextSmartName(place:tcutplace);      var        s : string;      begin        inc(SmartFilesCount);        if SmartFilesCount>999999 then         Message(asmw_f_too_many_asm_files);        case place of          cut_begin :            begin              inc(SmartHeaderCount);              s:=asmprefix+tostr(SmartHeaderCount)+'h';            end;          cut_normal :            s:=asmprefix+tostr(SmartHeaderCount)+'s';          cut_end :            s:=asmprefix+tostr(SmartHeaderCount)+'t';        end;        AsmFileName:=Path+FixFileName(s+tostr(SmartFilesCount)+target_info.asmext);        ObjFileName:=Path+FixFileName(s+tostr(SmartFilesCount)+target_info.objext);        { insert in container so it can be cleared after the linking }        SmartLinkOFiles.Insert(ObjFileName);      end;{*****************************************************************************                                 TAssemblerOutputFile*****************************************************************************}    procedure TExternalAssemblerOutputFile.RemoveAsm;      var        g : file;      begin        if cs_asm_leave in current_settings.globalswitches then         exit;        if cs_asm_extern in current_settings.globalswitches then         AsmRes.AddDeleteCommand(owner.AsmFileName)        else         begin           assign(g,owner.AsmFileName);           {$push} {$I-}            erase(g);           {$pop}           if ioresult<>0 then;         end;      end;    Procedure TExternalAssemblerOutputFile.AsmFlush;      begin        if outcnt>0 then         begin           { suppress i/o error }           {$push} {$I-}           BlockWrite(outfile,outbuf,outcnt);           {$pop}           fioerror:=fioerror or (ioresult<>0);           outcnt:=0;         end;      end;    procedure TExternalAssemblerOutputFile.MarkEmpty;      begin        AsmStartSize:=AsmSize      end;    function TExternalAssemblerOutputFile.ClearIfEmpty: boolean;      begin        result:=AsmSize=AsmStartSize;        if result then         AsmClear;      end;    procedure TExternalAssemblerOutputFile.AsmWriteFiltered(const c: char);      begin        MaybeAddLinePrefix;        AsmWriteAnsiStringUnfiltered(decorator.LineFilter(c));      end;    procedure TExternalAssemblerOutputFile.AsmWriteFiltered(const s: string);      begin        MaybeAddLinePrefix;        AsmWriteAnsiStringUnfiltered(decorator.LineFilter(s));      end;    procedure TExternalAssemblerOutputFile.AsmWriteFiltered(const s: ansistring);      begin        MaybeAddLinePrefix;        AsmWriteAnsiStringUnfiltered(decorator.LineFilter(s));      end;    procedure TExternalAssemblerOutputFile.AsmWriteFiltered(p: pchar; len: longint);      var        s: ansistring;      begin        MaybeAddLinePrefix;        s:='';        setlength(s,len);        move(p^,s[1],len);        AsmWriteAnsiStringUnfiltered(decorator.LineFilter(s));      end;    Procedure TExternalAssemblerOutputFile.AsmClear;      begin        outcnt:=0;      end;    procedure TExternalAssemblerOutputFile.MaybeAddLinePrefix;      begin        if assigned(decorator) and           linestart then          begin            AsmWriteAnsiStringUnfiltered(decorator.LinePrefix);            linestart:=false;          end;      end;    procedure TExternalAssemblerOutputFile.MaybeAddLinePostfix;      begin        if assigned(decorator) and           not linestart then          begin            AsmWriteAnsiStringUnfiltered(decorator.LinePostfix);            linestart:=true;          end;      end;    procedure TExternalAssemblerOutputFile.AsmWriteAnsiStringUnfiltered(const s: ansistring);      var        StartIndex, ToWrite: longint;      begin        if s='' then          exit;        if OutCnt+length(s)>=AsmOutSize then         AsmFlush;        StartIndex:=1;        ToWrite:=length(s);        while ToWrite>AsmOutSize do          begin            Move(s[StartIndex],OutBuf[OutCnt],AsmOutSize);            inc(OutCnt,AsmOutSize);            inc(AsmSize,AsmOutSize);            AsmFlush;            inc(StartIndex,AsmOutSize);            dec(ToWrite,AsmOutSize);          end;        Move(s[StartIndex],OutBuf[OutCnt],ToWrite);        inc(OutCnt,ToWrite);        inc(AsmSize,ToWrite);      end;    constructor TExternalAssemblerOutputFile.Create(_owner: TExternalAssembler);      begin        owner:=_owner;        linestart:=true;      end;    Procedure TExternalAssemblerOutputFile.AsmWrite(const c: char);      begin        if assigned(decorator) then          AsmWriteFiltered(c)        else          begin            if OutCnt+1>=AsmOutSize then             AsmFlush;            OutBuf[OutCnt]:=c;            inc(OutCnt);            inc(AsmSize);          end;      end;    Procedure TExternalAssemblerOutputFile.AsmWrite(const s:string);      begin        if s='' then          exit;        if assigned(decorator) then          AsmWriteFiltered(s)        else          begin            if OutCnt+length(s)>=AsmOutSize then             AsmFlush;            Move(s[1],OutBuf[OutCnt],length(s));            inc(OutCnt,length(s));            inc(AsmSize,length(s));          end;      end;    Procedure TExternalAssemblerOutputFile.AsmWrite(const s:ansistring);      begin        if s='' then          exit;        if assigned(decorator) then          AsmWriteFiltered(s)        else         AsmWriteAnsiStringUnfiltered(s);      end;    procedure TExternalAssemblerOutputFile.AsmWriteLn(const c: char);      begin        AsmWrite(c);        AsmLn;      end;    Procedure TExternalAssemblerOutputFile.AsmWriteLn(const s:string);      begin        AsmWrite(s);        AsmLn;      end;    Procedure TExternalAssemblerOutputFile.AsmWriteLn(const s: ansistring);      begin        AsmWrite(s);        AsmLn;      end;    Procedure TExternalAssemblerOutputFile.AsmWritePChar(p:pchar);      var        i,j : longint;      begin        i:=StrLen(p);        if i=0 then          exit;        if assigned(decorator) then          AsmWriteFiltered(p,i)        else          begin            j:=i;            while j>0 do             begin               i:=min(j,AsmOutSize);               if OutCnt+i>=AsmOutSize then                AsmFlush;               Move(p[0],OutBuf[OutCnt],i);               inc(OutCnt,i);               inc(AsmSize,i);               dec(j,i);               p:=pchar(@p[i]);             end;          end;      end;    Procedure TExternalAssemblerOutputFile.AsmLn;      var        newline: pshortstring;        newlineres: shortstring;        index: longint;      begin        MaybeAddLinePostfix;        if (cs_assemble_on_target in current_settings.globalswitches) then          newline:=@target_info.newline        else          newline:=@source_info.newline;        if assigned(decorator) then          begin            newlineres:=decorator.LineEnding(newline^);            newline:=@newlineres;          end;        if OutCnt>=AsmOutSize-length(newline^) then         AsmFlush;        index:=1;        repeat          OutBuf[OutCnt]:=newline^[index];          inc(OutCnt);          inc(AsmSize);          inc(index);        until index>length(newline^);      end;    procedure TExternalAssemblerOutputFile.AsmCreate(Aplace:tcutplace);{$ifdef hasamiga}      var        tempFileName: TPathStr;{$endif}      begin        if owner.SmartAsm then         owner.NextSmartName(Aplace);{$ifdef hasamiga}        { on Amiga/MorphOS try to redirect .s files to the T: assign, which is          for temp files, and usually (default setting) located in the RAM: drive.          This highly improves assembling speed for complex projects like the          compiler itself, especially on hardware with slow disk I/O.          Consider this as a poor man's pipe on Amiga, because real pipe handling          would be much more complex and error prone to implement. (KB) }        if (([cs_asm_extern,cs_asm_leave,cs_assemble_on_target] * current_settings.globalswitches) = []) then         begin          { try to have an unique name for the .s file }          tempFileName:=HexStr(GetProcessID shr 4,7)+ExtractFileName(owner.AsmFileName);{$ifndef morphos}          { old Amiga RAM: handler only allows filenames up to 30 char }          if Length(tempFileName) < 30 then{$endif}          owner.AsmFileName:='T:'+tempFileName;         end;{$endif}{$ifdef hasunix}        if owner.DoPipe then         begin           if owner.SmartAsm then            begin              if (owner.SmartFilesCount<=1) then               Message1(exec_i_assembling_smart,owner.name);            end           else             Message1(exec_i_assembling_pipe,owner.AsmFileName);           if checkverbosity(V_Executable) then             comment(V_Executable,'Executing "'+maybequoted(owner.FindAssembler)+'" with command line "'+               owner.MakeCmdLine+'"');           POpen(outfile,maybequoted(owner.FindAssembler)+' '+owner.MakeCmdLine,'W');         end        else{$endif}         begin           Assign(outfile,owner.AsmFileName);           {$push} {$I-}           Rewrite(outfile,1);           {$pop}           if ioresult<>0 then             begin               fioerror:=true;               Message1(exec_d_cant_create_asmfile,owner.AsmFileName);             end;         end;        outcnt:=0;        AsmSize:=0;        AsmStartSize:=0;      end;    procedure TExternalAssemblerOutputFile.AsmClose;      var        f : file;        FileAge : longint;      begin        AsmFlush;{$ifdef hasunix}        if owner.DoPipe then          begin            if PClose(outfile) <> 0 then              GenerateError;          end        else{$endif}         begin         {Touch Assembler time to ppu time is there is a ppufilename}           if owner.ppufilename<>'' then            begin              Assign(f,owner.ppufilename);              {$push} {$I-}              reset(f,1);              {$pop}              if ioresult=0 then               begin                 FileAge := FileGetDate(GetFileHandle(f));                 close(f);                 reset(outfile,1);                 FileSetDate(GetFileHandle(outFile),FileAge);               end;            end;           close(outfile);         end;      end;{*****************************************************************************                                 TExternalAssembler*****************************************************************************}    function TExternalAssembler.single2str(d : single) : string;      var         hs : string;      begin         str(d,hs);      { replace space with + }         if hs[1]=' ' then          hs[1]:='+';         single2str:='0d'+hs      end;    function TExternalAssembler.double2str(d : double) : string;      var         hs : string;      begin         str(d,hs);      { replace space with + }         if hs[1]=' ' then          hs[1]:='+';         double2str:='0d'+hs      end;    function TExternalAssembler.extended2str(e : extended) : string;      var         hs : string;      begin         str(e,hs);      { replace space with + }         if hs[1]=' ' then          hs[1]:='+';         extended2str:='0d'+hs      end;    function TExternalAssembler.sleb128tostr(a: int64): string;      var        i,len : longint;        buf   : array[0..31] of byte;      begin        result:='';        len:=EncodeSleb128(a,buf,0);        for i:=0 to len-1 do          begin            if (i > 0) then              result:=result+',';            result:=result+tostr(buf[i]);          end;      end;    function TExternalAssembler.uleb128tostr(a: qword): string;    var      i,len : longint;      buf   : array[0..31] of byte;    begin      result:='';      len:=EncodeUleb128(a,buf,0);      for i:=0 to len-1 do        begin          if (i > 0) then            result:=result+',';          result:=result+tostr(buf[i]);        end;    end;    Function TExternalAssembler.DoPipe:boolean;      begin{$ifdef hasunix}        DoPipe:=(cs_asm_pipe in current_settings.globalswitches) and                (([cs_asm_extern,cs_asm_leave,cs_assemble_on_target] * current_settings.globalswitches) = []) and                ((asminfo^.id in [as_gas,as_ggas,as_darwin,as_powerpc_xcoff,as_clang_gas,as_clang_llvm,as_clang_llvm_darwin,as_solaris_as,as_clang_asdarwin]));{$else hasunix}        DoPipe:=false;{$endif}      end;    function TExternalAssembler.CreateNewAsmWriter: TExternalAssemblerOutputFile;      begin        result:=TExternalAssemblerOutputFile.Create(self);      end;    Constructor TExternalAssembler.Create(info: pasminfo; smart: boolean);      begin        CreateWithWriter(info,CreateNewAsmWriter,true,smart);      end;    constructor TExternalAssembler.CreateWithWriter(info: pasminfo; wr: TExternalAssemblerOutputFile; freewriter,smart: boolean);      begin        inherited Create(info,smart);        fwriter:=wr;        ffreewriter:=freewriter;        if SmartAsm then          begin            path:=FixPath(ChangeFileExt(AsmFileName,target_info.smartext),false);            CreateSmartLinkPath(path);          end;      end;    procedure TExternalAssembler.CreateSmartLinkPath(const s:TPathStr);        procedure DeleteFilesWithExt(const AExt:string);        var          dir : TRawByteSearchRec;        begin          if findfirst(FixPath(s,false)+'*'+AExt,faAnyFile,dir) = 0 then            begin              repeat                DeleteFile(s+source_info.dirsep+dir.name);              until findnext(dir) <> 0;            end;          findclose(dir);        end;      var        hs  : TPathStr;      begin        if PathExists(s,false) then         begin           { the path exists, now we clean only all the .o and .s files }           DeleteFilesWithExt(target_info.objext);           DeleteFilesWithExt(target_info.asmext);         end        else         begin           hs:=s;           if hs[length(hs)] in ['/','\'] then            delete(hs,length(hs),1);           {$push} {$I-}            mkdir(hs);           {$pop}           if ioresult<>0 then;         end;      end;    const      lastas  : byte=255;    var      LastASBin : TCmdStr;    Function TExternalAssembler.FindAssembler:string;      var        asfound : boolean;        UtilExe  : string;        asmbin : TCmdStr;      begin        asfound:=false;        asmbin:=asminfo^.asmbin;        if (af_llvm in asminfo^.flags) then          asmbin:=asmbin+llvmutilssuffix;        if cs_assemble_on_target in current_settings.globalswitches then         begin           { If assembling on target, don't add any path PM }           FindAssembler:=utilsprefix+ChangeFileExt(asmbin,target_info.exeext);           exit;         end        else         UtilExe:=utilsprefix+ChangeFileExt(asmbin,source_info.exeext);        if lastas<>ord(asminfo^.id) then         begin           lastas:=ord(asminfo^.id);           { is an assembler passed ? }           if utilsdirectory<>'' then             asfound:=FindFile(UtilExe,utilsdirectory,false,LastASBin);           if not AsFound then             asfound:=FindExe(UtilExe,false,LastASBin);           if (not asfound) and not(cs_asm_extern in current_settings.globalswitches) then            begin              Message1(exec_e_assembler_not_found,LastASBin);              current_settings.globalswitches:=current_settings.globalswitches+[cs_asm_extern];            end;           if asfound then            Message1(exec_t_using_assembler,LastASBin);         end;        FindAssembler:=LastASBin;      end;    Function TExternalAssembler.CallAssembler(const command:string; const para:TCmdStr):Boolean;      var        DosExitCode : Integer;      begin        result:=true;        if (cs_asm_extern in current_settings.globalswitches) then          begin            if SmartAsm then              AsmRes.AddAsmCommand(command,para,Name+'('+TosTr(SmartFilesCount)+')')            else              AsmRes.AddAsmCommand(command,para,name);            exit;          end;        try          FlushOutput;          DosExitCode:=RequotedExecuteProcess(command,para);          if DosExitCode<>0          then begin            Message1(exec_e_error_while_assembling,tostr(dosexitcode));            result:=false;          end;        except on E:EOSError do          begin            Message1(exec_e_cant_call_assembler,tostr(E.ErrorCode));            current_settings.globalswitches:=current_settings.globalswitches+[cs_asm_extern];            result:=false;          end;        end;      end;    Function TExternalAssembler.DoAssemble:boolean;      begin        result:=true;        if DoPipe then         exit;        if not(cs_asm_extern in current_settings.globalswitches) then         begin           if SmartAsm then            begin              if (SmartFilesCount<=1) then               Message1(exec_i_assembling_smart,name);            end           else           Message1(exec_i_assembling,name);         end;        repeat          result:=CallAssembler(FindAssembler,MakeCmdLine)        until not(result) or not RerunAssembler;        if result then          writer.RemoveAsm        else          GenerateError;      end;    function TExternalAssembler.MakeCmdLine: TCmdStr;      function section_high_bound:longint;        var          alt : tasmlisttype;        begin          result:=0;          for alt:=low(tasmlisttype) to high(tasmlisttype) do            result:=result+current_asmdata.asmlists[alt].section_count;        end;      const        min_big_obj_section_count = $7fff;      begin        result:=asminfo^.asmcmd;        if af_llvm in target_asm.flags then          Replace(result,'$TRIPLET',targettriplet(triplet_llvm)){$ifdef arm}        else if (target_info.system=system_arm_ios) then          Replace(result,'$ARCH',lower(cputypestr[current_settings.cputype])){$endif arm}        ;        if (cs_assemble_on_target in current_settings.globalswitches) then         begin           Replace(result,'$ASM',maybequoted(ScriptFixFileName(AsmFileName)));           Replace(result,'$OBJ',maybequoted(ScriptFixFileName(ObjFileName)));         end        else         begin{$ifdef hasunix}          if DoPipe then            if not(asminfo^.id in [as_clang_gas,as_clang_asdarwin,as_clang_llvm,as_clang_llvm_darwin]) then              Replace(result,'$ASM','')            else              Replace(result,'$ASM','-')          else{$endif}             Replace(result,'$ASM',maybequoted(AsmFileName));           Replace(result,'$OBJ',maybequoted(ObjFileName));         end;         if (cs_create_pic in current_settings.moduleswitches) then           Replace(result,'$PIC','-KPIC')         else           Replace(result,'$PIC','');         if (cs_asm_source in current_settings.globalswitches) then           Replace(result,'$NOWARN','')         else           Replace(result,'$NOWARN','-W');         if target_info.endian=endian_little then           Replace(result,'$ENDIAN','-mlittle')         else           Replace(result,'$ENDIAN','-mbig');         { as we don't keep track of the amount of sections we created we simply           enable Big Obj COFF files always for targets that need them }         if (cs_asm_pre_binutils_2_25 in current_settings.globalswitches) or            not (target_info.system in systems_all_windows+systems_nativent-[system_i8086_win16]) or            (section_high_bound<min_big_obj_section_count) then           Replace(result,'$BIGOBJ','')         else           Replace(result,'$BIGOBJ','-mbig-obj');         Replace(result,'$EXTRAOPT',asmextraopt);      end;    function TExternalAssembler.RerunAssembler: boolean;      begin        result:=false;      end;    procedure TExternalAssembler.ResetSourceLines;      procedure DoReset(f:tinputfile);        var          i : longint;        begin          if not assigned(f) then            exit;          for i:=0 to f.maxlinebuf-1 do            if f.linebuf^[i]<0 then              f.linebuf^[i]:=-f.linebuf^[i]-1;        end;      begin        DoReset(infile);        DoReset(lastinfile);      end;    procedure TExternalAssembler.WriteSourceLine(hp: tailineinfo);      var        module : tmodule;      begin        { load infile }        if (lastfileinfo.moduleindex<>hp.fileinfo.moduleindex) or            (lastfileinfo.fileindex<>hp.fileinfo.fileindex) then          begin            { in case of a generic the module can be different }            if current_module.unit_index=hp.fileinfo.moduleindex then              module:=current_module            else              module:=get_module(hp.fileinfo.moduleindex);            { during the compilation of the system unit there are cases when              the fileinfo contains just zeros => invalid }            if assigned(module) then              infile:=module.sourcefiles.get_file(hp.fileinfo.fileindex)            else              infile:=nil;            if assigned(infile) then              begin                { open only if needed !! }                if (cs_asm_source in current_settings.globalswitches) then                  infile.open;              end;            { avoid unnecessary reopens of the same file !! }            lastfileinfo.fileindex:=hp.fileinfo.fileindex;            lastfileinfo.moduleindex:=hp.fileinfo.moduleindex;            { be sure to change line !! }            lastfileinfo.line:=-1;          end;        { write source }        if (cs_asm_source in current_settings.globalswitches) and          assigned(infile) then          begin            if (infile<>lastinfile) then              begin                writer.AsmWriteLn(asminfo^.comment+'['+infile.name+']');                if assigned(lastinfile) then                  lastinfile.close;              end;            if (hp.fileinfo.line<>lastfileinfo.line) and              (hp.fileinfo.line<infile.maxlinebuf) then              begin                if (hp.fileinfo.line<>0) and                  (infile.linebuf^[hp.fileinfo.line]>=0) then                  writer.AsmWriteLn(asminfo^.comment+'['+tostr(hp.fileinfo.line)+'] '+                  fixline(infile.GetLineStr(hp.fileinfo.line)));                { set it to a negative value !                  to make that is has been read already !! PM }                if (infile.linebuf^[hp.fileinfo.line]>=0) then                  infile.linebuf^[hp.fileinfo.line]:=-infile.linebuf^[hp.fileinfo.line]-1;              end;          end;        lastfileinfo:=hp.fileinfo;        lastinfile:=infile;      end;    procedure TExternalAssembler.WriteTempalloc(hp: tai_tempalloc);      begin{$ifdef EXTDEBUG}        if assigned(hp.problem) then          writer.AsmWriteLn(asminfo^.comment+'Temp '+tostr(hp.temppos)+','+          tostr(hp.tempsize)+' '+hp.problem^)        else{$endif EXTDEBUG}          writer.AsmWriteLn(asminfo^.comment+'Temp '+tostr(hp.temppos)+','+            tostr(hp.tempsize)+' '+tempallocstr[hp.allocation]);      end;    procedure TExternalAssembler.WriteRealConstAsBytes(hp: tai_realconst; const dbdir: string; do_line: boolean);      var        pdata: pbyte;        index, step, swapmask, real_byte_count: longint;        ssingle: single;        ddouble: double;{$ifdef FPC_COMP_IS_INT64}        ccomp: int64;{$else}        ccomp: comp;{$endif}        comp_data_size : byte;{$if defined(cpuextended) and defined(FPC_HAS_TYPE_EXTENDED)}        eextended: extended;{$else}{$ifdef FPC_SOFT_FPUX80}{$define USE_SOFT_FLOATX80}        f32 : float32;        f64 : float64;        eextended: floatx80;        gap_ofs_low,gap_ofs_high : byte;        gap_index, gap_size : byte;        has_gap : boolean;{$endif}{$endif cpuextended}      begin{$ifdef USE_SOFT_FLOATX80}        has_gap:=false;        gap_index:=0;        gap_size:=0;{$endif USE_SOFT_FLOATX80}        if do_line then          begin            case tai_realconst(hp).realtyp of              aitrealconst_s32bit:                writer.AsmWriteLn(asminfo^.comment+'s32bit real value: '+single2str(tai_realconst(hp).value.s32val));              aitrealconst_s64bit:                writer.AsmWriteLn(asminfo^.comment+'s64bit real value: '+double2str(tai_realconst(hp).value.s64val));{$if defined(cpuextended) and defined(FPC_HAS_TYPE_EXTENDED)}              { can't write full 80 bit floating point constants yet on non-x86 }              aitrealconst_s80bit:                writer.AsmWriteLn(asminfo^.comment+'s80bit real value: '+extended2str(tai_realconst(hp).value.s80val));{$else}{$ifdef USE_SOFT_FLOATX80}{$push}{$warn 6018 off} { Unreachable code due to compile time evaluation }             aitrealconst_s80bit:               begin                      if sizeof(tai_realconst(hp).value.s80val) = sizeof(double) then                   writer.AsmWriteLn(asminfo^.comment+'Emulated s80bit real value (on s64bit): '+double2str(tai_realconst(hp).value.s80val))                      else if sizeof(tai_realconst(hp).value.s80val) = sizeof(single) then                   writer.AsmWriteLn(asminfo^.comment+'Emulated s80bit real value (on s32bit): '+single2str(tai_realconst(hp).value.s80val))                else                      internalerror(2017091901);                     end;{$pop}{$endif}{$endif cpuextended}              aitrealconst_s64comp:                begin                  writer.AsmWriteLn(asminfo^.comment+'s64comp real value: '+extended2str(tai_realconst(hp).value.s64compval));                  comp_data_size:=sizeof(comp);                  if (comp_data_size<>tai_realconst(hp).datasize) then                    writer.AsmWriteLn(asminfo^.comment+'s64comp value type size is '+tostr(comp_data_size)+' but datasize is '+tostr(tai_realconst(hp).datasize));                end              else                internalerror(2014050604);            end;          end;        writer.AsmWrite(dbdir);        { generic float writing code: get start address of value, then write          byte by byte. Can't use fields directly, because e.g ts64comp is          defined as extended on x86 }        case tai_realconst(hp).realtyp of          aitrealconst_s32bit:            begin              ssingle:=single(tai_realconst(hp).value.s32val);              pdata:=@ssingle;            end;          aitrealconst_s64bit:            begin              ddouble:=double(tai_realconst(hp).value.s64val);              pdata:=@ddouble;            end;{$if defined(cpuextended) and defined(FPC_HAS_TYPE_EXTENDED)}          { can't write full 80 bit floating point constants yet on non-x86 }          aitrealconst_s80bit:            begin              eextended:=extended(tai_realconst(hp).value.s80val);              pdata:=@eextended;            end;{$else}{$ifdef USE_SOFT_FLOATX80}{$push}{$warn 6018 off} { Unreachable code due to compile time evaluation }          aitrealconst_s80bit:            begin              if sizeof(tai_realconst(hp).value.s80val) = sizeof(double) then                begin                  f64:=float64(double(tai_realconst(hp).value.s80val));                  if float64_is_signaling_nan(f64)<>0 then                    begin                      f64.low := 0;                      f64.high := longword($fff80000);                    end;                  eextended:=float64_to_floatx80(f64);                end              else if sizeof(tai_realconst(hp).value.s80val) = sizeof(single) then                begin                  f32:=float32(single(tai_realconst(hp).value.s80val));                  if float32_is_signaling_nan(f32)<>0 then                    begin                      f32 := longword($ffc00000);                    end;                  eextended:=float32_to_floatx80(f32);                end              else                internalerror(2017091902);              pdata:=@eextended;              if sizeof(eextended)>10 then                begin                  gap_ofs_high:=(pbyte(@eextended.high) - pbyte(@eextended));                  gap_ofs_low:=(pbyte(@eextended.low) - pbyte(@eextended));                  if (gap_ofs_low<gap_ofs_high) then                    begin                      gap_index:=gap_ofs_low+sizeof(eextended.low);                      gap_size:=gap_ofs_high-gap_index;                    end                  else                    begin                      gap_index:=gap_ofs_high+sizeof(eextended.high);                      gap_size:=gap_ofs_low-gap_index;                    end;                  if source_info.endian<>target_info.endian then                      gap_index:=gap_index+gap_size-1;                  has_gap:=gap_size <> 0;                end              else                has_gap:=false;            end;{$pop}{$endif}{$endif cpuextended}          aitrealconst_s64comp:            begin{$ifdef FPC_COMP_IS_INT64}              ccomp:=system.trunc(tai_realconst(hp).value.s64compval);{$else}              ccomp:=comp(tai_realconst(hp).value.s64compval);{$endif}              pdata:=@ccomp;            end;          else            internalerror(2014051001);        end;        real_byte_count:=tai_realconst(hp).datasize;        { write bytes in inverse order if source and target endianess don't          match }        if source_info.endian<>target_info.endian then          begin            { go from back to front }{$ifdef USE_SOFT_FLOATX80}            if has_gap then              index:=sizeof(eextended)-1            else{$endif USE_SOFT_FLOATX80}              index:=real_byte_count-1;            step:=-1;          end        else          begin            index:=0;            step:=1;          end;{$ifdef ARM}        { ARM-specific: low and high dwords of a double may be swapped }        if tai_realconst(hp).formatoptions=fo_hiloswapped then          begin            { only supported for double }            if tai_realconst(hp).datasize<>8 then              internalerror(2014050605);            { switch bit of the index so that the words are written in              the opposite order }            swapmask:=4;          end        else{$endif ARM}          swapmask:=0;        repeat{$ifdef USE_SOFT_FLOATX80}          if has_gap and (index=gap_index) then            index:=index+step*gap_size;{$endif USE_SOFT_FLOATX80}          writer.AsmWrite(tostr(pdata[index xor swapmask]));          inc(index,step);          dec(real_byte_count);          if real_byte_count<>0 then            writer.AsmWrite(',');        until real_byte_count=0;        { padding }        for real_byte_count:=tai_realconst(hp).datasize+1 to tai_realconst(hp).savesize do          writer.AsmWrite(',0');        writer.AsmLn;      end;    function TExternalAssembler.WriteComments(var hp: tai): boolean;      begin        result:=true;        case hp.typ of          ait_comment :            Begin              writer.AsmWrite(asminfo^.comment);              writer.AsmWritePChar(tai_comment(hp).str);              writer.AsmLn;            End;          ait_regalloc :            begin              if (cs_asm_regalloc in current_settings.globalswitches) then                begin                  writer.AsmWrite(#9+asminfo^.comment+'Register ');                  repeat                    writer.AsmWrite(std_regname(Tai_regalloc(hp).reg));                    if (hp.next=nil) or                       (tai(hp.next).typ<>ait_regalloc) or                       (tai_regalloc(hp.next).ratype<>tai_regalloc(hp).ratype) then                      break;                    hp:=tai(hp.next);                    writer.AsmWrite(',');                  until false;                  writer.AsmWrite(' ');                  writer.AsmWriteLn(regallocstr[tai_regalloc(hp).ratype]);                end;            end;          ait_tempalloc :            begin              if (cs_asm_tempalloc in current_settings.globalswitches) then                WriteTempalloc(tai_tempalloc(hp));            end;          ait_varloc:            begin              { ait_varloc is present here only when register allocation is not done ( -sr option ) }              if tai_varloc(hp).newlocationhi<>NR_NO then                writer.AsmWriteLn(asminfo^.comment+'Var '+tai_varloc(hp).varsym.realname+' located in register '+                  std_regname(tai_varloc(hp).newlocationhi)+':'+std_regname(tai_varloc(hp).newlocation))              else                writer.AsmWriteLn(asminfo^.comment+'Var '+tai_varloc(hp).varsym.realname+' located in register '+                  std_regname(tai_varloc(hp).newlocation));            end;          else            result:=false;        end;      end;    procedure TExternalAssembler.WriteTree(p:TAsmList);      begin      end;    procedure TExternalAssembler.WriteAsmList;      begin      end;    procedure TExternalAssembler.MakeObject;      begin        writer.AsmCreate(cut_normal);        FillChar(lastfileinfo, sizeof(lastfileinfo), 0);        lastfileinfo.line := -1;        lastinfile := nil;        lastsectype := sec_none;        WriteAsmList;        writer.AsmClose;        if not(writer.ioerror) then          DoAssemble;      end;    destructor TExternalAssembler.Destroy;      begin        if ffreewriter then          writer.Free;        inherited;      end;{*****************************************************************************                                  TInternalAssembler*****************************************************************************}    constructor TInternalAssembler.Create(info: pasminfo; smart: boolean);      begin        inherited;        ObjOutput:=nil;        ObjData:=nil;        SmartAsm:=smart;{$ifdef ARM}        Code16:=current_settings.instructionset=is_thumb;{$endif ARM}      end;   destructor TInternalAssembler.destroy;      begin        if assigned(ObjData) then          ObjData.free;        if assigned(ObjOutput) then          ObjOutput.free;      end;    procedure TInternalAssembler.WriteStab(p:pchar);        function consumecomma(var p:pchar):boolean;        begin          while (p^=' ') do            inc(p);          result:=(p^=',');          inc(p);        end;        function consumenumber(var p:pchar;out value:longint):boolean;        var          hs : string;          len,          code : integer;        begin          value:=0;          while (p^=' ') do            inc(p);          len:=0;          while (p^ in ['0'..'9']) do            begin              inc(len);              hs[len]:=p^;              inc(p);            end;          if len>0 then            begin              hs[0]:=chr(len);              val(hs,value,code);            end          else            code:=-1;          result:=(code=0);        end;        function consumeoffset(var p:pchar;out relocsym:tobjsymbol;out value:longint):boolean;        var          hs        : string;          len,          code      : integer;          pstart    : pchar;          sym       : tobjsymbol;          exprvalue : longint;          gotmin,          have_first_symbol,          have_second_symbol,          dosub     : boolean;        begin          result:=false;          value:=0;          relocsym:=nil;          gotmin:=false;          have_first_symbol:=false;          have_second_symbol:=false;          repeat            dosub:=false;            exprvalue:=0;            if gotmin then              begin                dosub:=true;                gotmin:=false;              end;            while (p^=' ') do              inc(p);            case p^ of              #0 :                break;              ' ' :                inc(p);              '0'..'9' :                begin                  len:=0;                  while (p^ in ['0'..'9']) do                    begin                      inc(len);                      hs[len]:=p^;                      inc(p);                    end;                  hs[0]:=chr(len);                  val(hs,exprvalue,code);                  if code<>0 then                    internalerror(200702251);                end;              '.','_',              'A'..'Z',              'a'..'z' :                begin                  pstart:=p;                  while not(p^ in [#0,' ','-','+']) do                    inc(p);                  len:=p-pstart;                  if len>255 then                    internalerror(200509187);                  hs[0]:=chr(len);                  move(pstart^,hs[1],len);                  sym:=objdata.symbolref(hs);                  { Second symbol? }                  if assigned(relocsym) then                    begin                      if have_second_symbol then                        internalerror(2007032201);                      have_second_symbol:=true;                      if not have_first_symbol then                        internalerror(2007032202);                      { second symbol should substracted to first }                      if not dosub then                        internalerror(2007032203);                      if (relocsym.objsection<>sym.objsection) then                        internalerror(2005091810);                      exprvalue:=relocsym.address-sym.address;                      relocsym:=nil;                      dosub:=false;                    end                  else                    begin                      relocsym:=sym;                      if assigned(sym.objsection) then                        begin                          { first symbol should be + }                          if not have_first_symbol and dosub then                            internalerror(2007032204);                          have_first_symbol:=true;                        end;                    end;                end;              '+' :                begin                  { nothing, by default addition is done }                  inc(p);                end;              '-' :                begin                  gotmin:=true;                  inc(p);                end;              else                internalerror(200509189);            end;            if dosub then              dec(value,exprvalue)            else              inc(value,exprvalue);          until false;          result:=true;        end;      var        stabstrlen,        ofs,        nline,        nidx,        nother,        i         : longint;        stab      : TObjStabEntry;        relocsym  : TObjSymbol;        pstr,        pcurr,        pendquote : pchar;        oldsec    : TObjSection;      begin        pcurr:=nil;        pstr:=nil;        pendquote:=nil;        relocsym:=nil;        ofs:=0;        { Parse string part }        if (p[0]='"') then          begin            pstr:=@p[1];            { Ignore \" inside the string }            i:=1;            while not((p[i]='"') and (p[i-1]<>'\')) and                  (p[i]<>#0) do              inc(i);            pendquote:=@p[i];            pendquote^:=#0;            pcurr:=@p[i+1];            if not consumecomma(pcurr) then              internalerror(200509181);          end        else          pcurr:=p;        { When in pass 1 then only alloc and leave }        if ObjData.currpass=1 then          begin            ObjData.StabsSec.Alloc(sizeof(TObjStabEntry));            if assigned(pstr) and (pstr[0]<>#0) then              ObjData.StabStrSec.Alloc(strlen(pstr)+1);          end        else          begin            { Stabs format: nidx,nother,nline[,offset] }            if not consumenumber(pcurr,nidx) then              internalerror(200509182);            if not consumecomma(pcurr) then              internalerror(200509183);            if not consumenumber(pcurr,nother) then              internalerror(200509184);            if not consumecomma(pcurr) then              internalerror(200509185);            if not consumenumber(pcurr,nline) then              internalerror(200509186);            if consumecomma(pcurr) then              consumeoffset(pcurr,relocsym,ofs);            { Generate stab entry }            if assigned(pstr) and (pstr[0]<>#0) then              begin                stabstrlen:=strlen(pstr);{$ifdef optimizestabs}                StabStrEntry:=nil;                if (nidx=N_SourceFile) or (nidx=N_IncludeFile) then                  begin                    hs:=strpas(pstr);                    StabstrEntry:=StabStrDict.Find(hs);                    if not assigned(StabstrEntry) then                      begin                        StabstrEntry:=TStabStrEntry.Create(hs);                        StabstrEntry:=StabStrSec.Size;                        StabStrDict.Insert(StabstrEntry);                        { generate new stab }                        StabstrEntry:=nil;                      end;                  end;                if assigned(StabstrEntry) then                  stab.strpos:=StabstrEntry.strpos                else{$endif optimizestabs}                  begin                    stab.strpos:=ObjData.StabStrSec.Size;                    ObjData.StabStrSec.write(pstr^,stabstrlen+1);                  end;              end            else              stab.strpos:=0;            stab.ntype:=byte(nidx);            stab.ndesc:=word(nline);            stab.nother:=byte(nother);            stab.nvalue:=ofs;            { Write the stab first without the value field. Then              write a the value field with relocation }            oldsec:=ObjData.CurrObjSec;            ObjData.SetSection(ObjData.StabsSec);            MaybeSwapStab(stab);            ObjData.Writebytes(stab,sizeof(TObjStabEntry)-4);            ObjData.Writereloc(stab.nvalue,4,relocsym,RELOC_ABSOLUTE32);            ObjData.setsection(oldsec);          end;        if assigned(pendquote) then          pendquote^:='"';      end;    function TInternalAssembler.MaybeNextList(var hp:Tai):boolean;      begin        { maybe end of list }        while not assigned(hp) do         begin           if currlistidx<lists then            begin              inc(currlistidx);              currlist:=list[currlistidx];              hp:=Tai(currList.first);            end           else            begin              MaybeNextList:=false;              exit;            end;         end;        MaybeNextList:=true;      end;    function TInternalAssembler.SetIndirectToSymbol(hp: Tai; const indirectname: string): Boolean;      var        objsym  : TObjSymbol;        indsym  : TObjSymbol;      begin        Result:=          Assigned(hp) and          (hp.typ=ait_symbol);        if not Result then          Exit;        objsym:=Objdata.SymbolRef(tai_symbol(hp).sym);        objsym.size:=0;        indsym := TObjSymbol(ObjData.ObjSymbolList.Find(indirectname));        if not Assigned(indsym) then          begin            { it's possible that indirect symbol is not present in the list,              so we must create it as undefined }            indsym:=ObjData.CObjSymbol.Create(ObjData.ObjSymbolList, indirectname);            indsym.typ:=AT_NONE;            indsym.bind:=AB_NONE;          end;        objsym.indsymbol:=indsym;        Result:=true;      end;    function TInternalAssembler.TreePass0(hp:Tai):Tai;      var        objsym,        objsymend : TObjSymbol;        cpu: tcputype;        eabi_section, TmpSection: TObjSection;      begin        while assigned(hp) do         begin{$ifdef DEBUG}           if not(hp.typ in SkipLineInfo) then             current_filepos:=tailineinfo(hp).fileinfo;{$endif DEBUG}            case hp.typ of             ait_align :               begin                 if tai_align_abstract(hp).aligntype>1 then                   begin                     { always use the maximum fillsize in this pass to avoid possible                       short jumps to become out of range }                     Tai_align_abstract(hp).fillsize:=Tai_align_abstract(hp).aligntype;                     ObjData.alloc(Tai_align_abstract(hp).fillsize);                     { may need to increase alignment of section }                     if tai_align_abstract(hp).aligntype>ObjData.CurrObjSec.secalign then                       ObjData.CurrObjSec.secalign:=tai_align_abstract(hp).aligntype;                   end                 else                   Tai_align_abstract(hp).fillsize:=0;               end;             ait_datablock :               begin{$ifdef USE_COMM_IN_BSS}                 if writingpackages and                    Tai_datablock(hp).is_global then                   ObjData.SymbolDefine(Tai_datablock(hp).sym)                 else{$endif USE_COMM_IN_BSS}                   begin                     ObjData.allocalign(used_align(size_2_align(Tai_datablock(hp).size),0,ObjData.CurrObjSec.secalign));                     ObjData.SymbolDefine(Tai_datablock(hp).sym);                     ObjData.alloc(Tai_datablock(hp).size);                   end;               end;             ait_realconst:               ObjData.alloc(tai_realconst(hp).savesize);             ait_const:               begin                 { if symbols are provided we can calculate the value for relative symbols.                   This is required for length calculation of leb128 constants }                 if assigned(tai_const(hp).sym) then                   begin                     objsym:=Objdata.SymbolRef(tai_const(hp).sym);                     { objsym already defined and there is endsym? }                     if assigned(objsym.objsection) and assigned(tai_const(hp).endsym) then                       begin                         objsymend:=Objdata.SymbolRef(tai_const(hp).endsym);                         { objsymend already defined? }                         if assigned(objsymend.objsection) then                           begin                             if objsymend.objsection<>objsym.objsection then                               begin                                 { leb128 relative constants are not relocatable, but other types are,                                   given that objsym belongs to the current section. }                                 if (Tai_const(hp).consttype in [aitconst_uleb128bit,aitconst_sleb128bit]) or                                    (objsym.objsection<>ObjData.CurrObjSec) then                                   InternalError(200404124);                               end{$push} {$R-}{$Q-}                             else                               Tai_const(hp).value:=objsymend.address-objsym.address+Tai_const(hp).symofs;                           end;{$pop}                       end;                   end;                 ObjData.alloc(tai_const(hp).size);               end;             ait_directive:               begin                 case tai_directive(hp).directive of                   asd_indirect_symbol:                     { handled in TreePass1 }                     ;                   asd_lazy_reference:                     begin                       if tai_directive(hp).name='' then                         Internalerror(2009112101);                       objsym:=ObjData.symbolref(tai_directive(hp).name);                       objsym.bind:=AB_LAZY;                     end;                   asd_reference:                     { ignore for now, but should be added}                     ;                   asd_cpu:                     begin                       ObjData.CPUType:=cpu_none;                       for cpu:=low(tcputype) to high(tcputype) do                         if cputypestr[cpu]=tai_directive(hp).name then                           begin                             ObjData.CPUType:=cpu;                             break;                           end;                     end;                   asd_weak_definition:                     begin                       if tai_directive(hp).name='' then                         Internalerror(2022040901);                       objsym:=ObjData.symbolref(tai_directive(hp).name);                       objsym.bind:=AB_WEAK;                     end;{$ifdef OMFOBJSUPPORT}                   asd_omf_linnum_line:                     { ignore for now, but should be added}                     ;{$endif OMFOBJSUPPORT}{$ifdef ARM}                   asd_thumb_func:                     ObjData.ThumbFunc:=true;                   asd_force_thumb:                     begin                       ObjData.ThumbFunc:=true;                       Code16:=true;                     end;                   asd_code:                     begin                       { ai_directive(hp).name can be only 16 or 32, this is checked by the reader }                       ObjData.ThumbFunc:=tai_directive(hp).name='16';                       Code16:=tai_directive(hp).name='16';                     end{$endif ARM}{$ifdef RISCV}                   asd_option:                     internalerror(2019031701);{$endif RISCV}                   else                     internalerror(2010011101);                 end;               end;             ait_section:               begin                 if Tai_section(hp).sectype=sec_user then                   ObjData.CreateSection(Tai_section(hp).sectype,Tai_section(hp).secflags,Tai_section(hp).secprogbits,Tai_section(hp).name^,Tai_section(hp).secorder)                 else                   ObjData.CreateSection(Tai_section(hp).sectype,Tai_section(hp).name^,Tai_section(hp).secorder);                 Tai_section(hp).sec:=ObjData.CurrObjSec;               end;             ait_symbol :               begin                 { needs extra support in the internal assembler }                 { the value is just ignored }                 {if tai_symbol(hp).has_value then                      internalerror(2009090804); ;}                 ObjData.SymbolDefine(Tai_symbol(hp).sym);               end;             ait_symbolpair :               with tai_symbolpair(hp) do                 ObjData.SymbolPairDefine(kind,sym^,value^);             ait_label :               ObjData.SymbolDefine(Tai_label(hp).labsym);             ait_string :               ObjData.alloc(Tai_string(hp).len);             ait_instruction :               begin{$ifdef arm}                 if code16 then                   include(taicpu(hp).flags,cf_thumb)                 else                   exclude(taicpu(hp).flags,cf_thumb);{$endif arm}                 { reset instructions which could change in pass 2 }                 Taicpu(hp).resetpass2;                 ObjData.alloc(Taicpu(hp).Pass1(ObjData));               end;             ait_cutobject :               if SmartAsm then                break;             ait_eabi_attribute :               begin                 eabi_section:=ObjData.findsection('.ARM.attributes');                 if not(assigned(eabi_section)) then                   begin                     TmpSection:=ObjData.CurrObjSec;                     ObjData.CreateSection(sec_arm_attribute,[],SPB_ARM_ATTRIBUTES,'',secorder_default);                     eabi_section:=ObjData.CurrObjSec;                     ObjData.setsection(TmpSection);                   end;                 if eabi_section.Size=0 then                   eabi_section.alloc(16);                 eabi_section.alloc(LengthUleb128(tai_eabi_attribute(hp).tag));                 case tai_eabi_attribute(hp).eattr_typ of                   eattrtype_dword:                     eabi_section.alloc(LengthUleb128(tai_eabi_attribute(hp).value));                   eattrtype_ntbs:                     if assigned(tai_eabi_attribute(hp).valuestr) then                       eabi_section.alloc(Length(tai_eabi_attribute(hp).valuestr^)+1)                     else                       eabi_section.alloc(1);                   else                     Internalerror(2019100701);                 end;               end;{$ifdef WASM}             ait_globaltype:               TWasmObjData(ObjData).DeclareGlobalType(tai_globaltype(hp));             ait_functype:               TWasmObjData(ObjData).DeclareFuncType_Pass0(tai_functype(hp));             ait_tagtype:               TWasmObjData(ObjData).DeclareTagType(tai_tagtype(hp));             ait_export_name:               TWasmObjData(ObjData).DeclareExportName(tai_export_name(hp));             ait_import_module:               TWasmObjData(ObjData).DeclareImportModule(tai_import_module(hp));             ait_import_name:               TWasmObjData(ObjData).DeclareImportName(tai_import_name(hp));             ait_local:               TWasmObjData(ObjData).DeclareLocals_Pass0(tai_local(hp));{$endif WASM}             else               ;           end;           hp:=Tai(hp.next);         end;        TreePass0:=hp;      end;    function TInternalAssembler.TreePass1(hp:Tai):Tai;      var        objsym,        objsymend : TObjSymbol;        cpu: tcputype;        eabi_section: TObjSection;      begin        while assigned(hp) do         begin{$ifdef DEBUG}           if not(hp.typ in SkipLineInfo) then             current_filepos:=tailineinfo(hp).fileinfo;{$endif DEBUG}           case hp.typ of             ait_align :               begin                 if tai_align_abstract(hp).aligntype>1 then                   begin                     { here we must determine the fillsize which is used in pass2 }                     Tai_align_abstract(hp).fillsize:=align(ObjData.CurrObjSec.Size,Tai_align_abstract(hp).aligntype)-                       ObjData.CurrObjSec.Size;                     { maximum number of bytes for alignment exeeded? }                     if (Tai_align_abstract(hp).aligntype<>Tai_align_abstract(hp).maxbytes) and                       (Tai_align_abstract(hp).fillsize>Tai_align_abstract(hp).maxbytes) then                       Tai_align_abstract(hp).fillsize:=align(ObjData.CurrObjSec.Size,Byte(Tai_align_abstract(hp).aligntype div 2))-                         ObjData.CurrObjSec.Size;                     ObjData.alloc(Tai_align_abstract(hp).fillsize);                   end;               end;             ait_datablock :               begin                 if (oso_data in ObjData.CurrObjSec.secoptions) and                    not (oso_sparse_data in ObjData.CurrObjSec.secoptions) then                   Message(asmw_e_alloc_data_only_in_bss);{$ifdef USE_COMM_IN_BSS}                 if writingpackages and                    Tai_datablock(hp).is_global then                   begin                     objsym:=ObjData.SymbolDefine(Tai_datablock(hp).sym);                     objsym.size:=Tai_datablock(hp).size;                     objsym.bind:=AB_COMMON;                     objsym.alignment:=needtowritealignmentalsoforELF;                   end                 else{$endif USE_COMM_IN_BSS}                   begin                     ObjData.allocalign(used_align(size_2_align(Tai_datablock(hp).size),0,ObjData.CurrObjSec.secalign));                     objsym:=ObjData.SymbolDefine(Tai_datablock(hp).sym);                     objsym.size:=Tai_datablock(hp).size;                     ObjData.alloc(Tai_datablock(hp).size);                   end;               end;             ait_realconst:               ObjData.alloc(tai_realconst(hp).savesize);             ait_const:               begin                 { Recalculate relative symbols }                 if assigned(tai_const(hp).sym) and                    assigned(tai_const(hp).endsym) then                   begin                     objsym:=Objdata.SymbolRef(tai_const(hp).sym);                     objsymend:=Objdata.SymbolRef(tai_const(hp).endsym);                     if Tai_const(hp).consttype in [aitconst_gottpoff,aitconst_tlsgd,aitconst_tlsdesc] then                       begin                         if objsymend.objsection<>ObjData.CurrObjSec then                           Internalerror(2019092801);                         Tai_const(hp).value:=objsymend.address-ObjData.CurrObjSec.Size+Tai_const(hp).symofs;                       end                     else if objsymend.objsection<>objsym.objsection then                       begin                         if (Tai_const(hp).consttype in [aitconst_uleb128bit,aitconst_sleb128bit]) or                            (objsym.objsection<>ObjData.CurrObjSec) then                           internalerror(200905042);                       end{$push} {$R-}{$Q-}                     else                       Tai_const(hp).value:=objsymend.address-objsym.address+Tai_const(hp).symofs;                   end;{$pop}                 if (Tai_const(hp).consttype in [aitconst_uleb128bit,aitconst_sleb128bit]) then                   Tai_const(hp).fixsize;                 ObjData.alloc(tai_const(hp).size);               end;             ait_section:               begin                 { use cached value }                 ObjData.setsection(Tai_section(hp).sec);               end;             ait_stab :               begin                 if assigned(Tai_stab(hp).str) then                   WriteStab(Tai_stab(hp).str);               end;             ait_symbol :               ObjData.SymbolDefine(Tai_symbol(hp).sym);             ait_symbol_end :               begin                 objsym:=ObjData.SymbolRef(Tai_symbol_end(hp).sym);                 objsym.size:=ObjData.CurrObjSec.Size-objsym.offset;               end;             ait_symbolpair:               with tai_symbolpair(hp) do                 ObjData.SymbolPairDefine(kind,sym^,value^);             ait_label :               ObjData.SymbolDefine(Tai_label(hp).labsym);             ait_string :               ObjData.alloc(Tai_string(hp).len);             ait_instruction :               ObjData.alloc(Taicpu(hp).Pass1(ObjData));             ait_cutobject :               if SmartAsm then                break;             ait_directive :               begin                 case tai_directive(hp).directive of                   asd_indirect_symbol:                     if tai_directive(hp).name='' then                       Internalerror(2009101103)                     else if not SetIndirectToSymbol(Tai(hp.Previous), tai_directive(hp).name) then                       Internalerror(2009101102);                   asd_lazy_reference:                     { handled in TreePass0 }                     ;                   asd_reference:                     { ignore for now, but should be added}                     ;                   asd_thumb_func:                     { ignore for now, but should be added}                     ;                   asd_force_thumb:                     { ignore for now, but should be added}                     ;                   asd_code:                     { ignore for now, but should be added}                     ;                   asd_option:                     { ignore for now, but should be added}                     ;                   asd_weak_definition:                     { ignore for now, but should be added}                     ;{$ifdef OMFOBJSUPPORT}                   asd_omf_linnum_line:                     { ignore for now, but should be added}                     ;{$endif OMFOBJSUPPORT}                   asd_cpu:                     begin                       ObjData.CPUType:=cpu_none;                       for cpu:=low(tcputype) to high(tcputype) do                         if cputypestr[cpu]=tai_directive(hp).name then                           begin                             ObjData.CPUType:=cpu;                             break;                           end;                     end;                   else                     internalerror(2010011102);                 end;               end;             ait_eabi_attribute :               begin                 eabi_section:=ObjData.findsection('.ARM.attributes');                 if not(assigned(eabi_section)) then                   Internalerror(2019100702);                 if eabi_section.Size=0 then                   eabi_section.alloc(16);                 eabi_section.alloc(LengthUleb128(tai_eabi_attribute(hp).tag));                 case tai_eabi_attribute(hp).eattr_typ of                   eattrtype_dword:                     eabi_section.alloc(LengthUleb128(tai_eabi_attribute(hp).value));                   eattrtype_ntbs:                     if assigned(tai_eabi_attribute(hp).valuestr) then                       eabi_section.alloc(Length(tai_eabi_attribute(hp).valuestr^)+1)                     else                       eabi_section.alloc(1);                   else                     Internalerror(2019100703);                 end;               end;{$ifdef WASM}             ait_functype:               TWasmObjData(ObjData).DeclareFuncType_Pass1(tai_functype(hp));             ait_local:               TWasmObjData(ObjData).DeclareLocals_Pass1(tai_local(hp));{$endif WASM}             else               ;           end;           hp:=Tai(hp.next);         end;        TreePass1:=hp;      end;    function TInternalAssembler.TreePass2(hp:Tai):Tai;      var        fillbuffer : tfillbuffer;        leblen : byte;        lebbuf : array[0..63] of byte;        objsym,        ref,        objsymend : TObjSymbol;        zerobuf : array[0..63] of byte;        relative_reloc: boolean;        pdata : pointer;	real_byte_count, index, step : longint;        ssingle : single;        ddouble : double;        {$if defined(cpuextended) and defined(FPC_HAS_TYPE_EXTENDED)}        eextended : extended;        {$else}        {$ifdef USE_SOFT_FLOATX80}        f32 : float32;        f64 : float64;        eextended : floatx80;        has_gap : boolean;        gap_ofs_low,gap_ofs_high : byte;        gap_index, gap_size : byte;        {$endif}        {$endif}{$ifdef FPC_COMP_IS_INT64}        ccomp: int64;{$else}        ccomp: comp;{$endif}        comp_data_size : byte;        tmp    : word;        cpu: tcputype;        ddword : dword;        b : byte;        w : word;        d : dword;        q : qword;        eabi_section: TObjSection;        s: String;        TmpDataPos: TObjSectionOfs;      begin        fillchar(zerobuf,sizeof(zerobuf),0);        fillchar(objsym,sizeof(objsym),0);        fillchar(objsymend,sizeof(objsymend),0);{$ifdef USE_SOFT_FLOATX80}        has_gap:=false;        gap_index:=0;        gap_size:=0;{$endif USE_SOFT_FLOATX80}        { main loop }        while assigned(hp) do         begin{$ifdef DEBUG}           if not(hp.typ in SkipLineInfo) then             current_filepos:=tailineinfo(hp).fileinfo;{$endif DEBUG}           case hp.typ of             ait_align :               begin                 if tai_align_abstract(hp).aligntype>ObjData.CurrObjSec.secalign then                   InternalError(2012072301);                 if oso_data in ObjData.CurrObjSec.secoptions then                   ObjData.writebytes(Tai_align_abstract(hp).calculatefillbuf(fillbuffer,oso_executable in ObjData.CurrObjSec.secoptions)^,                     Tai_align_abstract(hp).fillsize)                 else                   ObjData.alloc(Tai_align_abstract(hp).fillsize);               end;             ait_section :               begin                 { use cached value }                 ObjData.setsection(Tai_section(hp).sec);               end;             ait_symbol :               begin                 ObjOutput.exportsymbol(ObjData.SymbolRef(Tai_symbol(hp).sym));               end;            ait_symbol_end :               begin                 { recalculate size, as some preceding instructions                   could have been changed to smaller size }                 objsym:=ObjData.SymbolRef(Tai_symbol_end(hp).sym);                 objsym.size:=ObjData.CurrObjSec.Size-objsym.offset;               end;             ait_datablock :               begin                 ObjOutput.exportsymbol(ObjData.SymbolRef(Tai_datablock(hp).sym));{$ifdef USE_COMM_IN_BSS}                 if not(writingpackages and                        Tai_datablock(hp).is_global) then{$endif USE_COMM_IN_BSS}                   begin                     ObjData.allocalign(used_align(size_2_align(Tai_datablock(hp).size),0,ObjData.CurrObjSec.secalign));                     ObjData.alloc(Tai_datablock(hp).size);                   end;               end;             ait_realconst:               begin                 real_byte_count:=tai_realconst(hp).datasize;                 case tai_realconst(hp).realtyp of                   aitrealconst_s32bit:                     begin                       ssingle:=single(tai_realconst(hp).value.s32val);                       pdata:=@ssingle;                     end;                   aitrealconst_s64bit:                     begin                       ddouble:=double(tai_realconst(hp).value.s64val);                       pdata:=@ddouble;                     end;         {$if defined(cpuextended) and defined(FPC_HAS_TYPE_EXTENDED)}                   { can't write full 80 bit floating point constants yet on non-x86 }                   aitrealconst_s80bit:                     begin                       eextended:=extended(tai_realconst(hp).value.s80val);                       pdata:=@eextended;                     end;         {$else}         {$ifdef USE_SOFT_FLOATX80}           {$push}{$warn 6018 off} { Unreachable code due to compile time evaluation }                   aitrealconst_s80bit:                     begin                       if sizeof(tai_realconst(hp).value.s80val) = sizeof(double) then                         begin                           f64:=float64(double(tai_realconst(hp).value.s80val));                           if float64_is_signaling_nan(f64)<>0 then                             begin                               f64.low := 0;                               f64.high := longword($fff80000);                             end;                           eextended:=float64_to_floatx80(f64);                         end                       else if sizeof(tai_realconst(hp).value.s80val) = sizeof(single) then                         begin                           f32:=float32(single(tai_realconst(hp).value.s80val));                           if float32_is_signaling_nan(f32)<>0 then                             begin                               f32 := longword($ffc00000);                             end;                           eextended:=float32_to_floatx80(f32);                         end                       else                         internalerror(2017091903);                       pdata:=@eextended;                       if sizeof(eextended)>10 then                         begin                           gap_ofs_high:=(pbyte(@eextended.high) - pbyte(@eextended));                           gap_ofs_low:=(pbyte(@eextended.low) - pbyte(@eextended));                           if (gap_ofs_low<gap_ofs_high) then                             begin                               gap_index:=gap_ofs_low+sizeof(eextended.low);                               gap_size:=gap_ofs_high-gap_index;                             end                           else                             begin                               gap_index:=gap_ofs_high+sizeof(eextended.high);                               gap_size:=gap_ofs_low-gap_index;                             end;                           if source_info.endian<>target_info.endian then                             gap_index:=gap_index+gap_size-1;                           has_gap:=gap_size <> 0;                         end                       else                         has_gap:=false;                     end;           {$pop}         {$endif}         {$endif cpuextended}                   aitrealconst_s64comp:                     begin{$ifdef FPC_COMP_IS_INT64}                       ccomp:=system.trunc(tai_realconst(hp).value.s64compval);{$else}                       ccomp:=comp(tai_realconst(hp).value.s64compval);{$endif}                       pdata:=@ccomp;                     end;                   else                     internalerror(2015030501);                 end;                 if source_info.endian<>target_info.endian then                   { write bytes in inverse order if source and target endianess don't match }                   begin                     { go from back to front }{$ifdef USE_SOFT_FLOATX80}                     if has_gap then                       index:=sizeof(eextended)-1                     else{$endif USE_SOFT_FLOATX80}                       index:=real_byte_count-1;                     step:=-1;                   end                 else                   begin                     index:=0;                     step:=1;                   end;                 if (source_info.endian<>target_info.endian)                   {$ifdef USE_SOFT_FLOATX80} or has_gap{$endif} then                   begin                     d:=0;                     repeat{$ifdef USE_SOFT_FLOATX80}                       if has_gap and (index=gap_index) then                         index:=index+step*gap_size;{$endif USE_SOFT_FLOATX80}                       lebbuf[d]:=pbyte(pdata)[index];                       inc(index,step);                       dec(real_byte_count);                       inc(d);                     until real_byte_count=0;		     { d now bares the count value }                     pdata:=@lebbuf;                     ObjData.writebytes(pdata^,d);                     ObjData.writebytes(zerobuf,tai_realconst(hp).savesize-d);                   end		 else                   begin                     ObjData.writebytes(pdata^,tai_realconst(hp).datasize);                     ObjData.writebytes(zerobuf,tai_realconst(hp).savesize-tai_realconst(hp).datasize);		   end;               end;             ait_string :               ObjData.writebytes(Tai_string(hp).str^,Tai_string(hp).len);             ait_const :               begin                 { Recalculate relative symbols, addresses of forward references                   can be changed in treepass1 }                 relative_reloc:=false;                 if assigned(tai_const(hp).sym) and                    assigned(tai_const(hp).endsym) then                   begin                     objsym:=Objdata.SymbolRef(tai_const(hp).sym);                     objsymend:=Objdata.SymbolRef(tai_const(hp).endsym);                     relative_reloc:=(objsym.objsection<>objsymend.objsection);                     if Tai_const(hp).consttype in [aitconst_gottpoff] then                       begin                         if objsymend.objsection<>ObjData.CurrObjSec then                           Internalerror(2019092802);                         Tai_const(hp).value:=objsymend.address-ObjData.CurrObjSec.Size+Tai_const(hp).symofs;                       end                     else if Tai_const(hp).consttype in [aitconst_tlsgd,aitconst_tlsdesc] then                       begin                         if objsymend.objsection<>ObjData.CurrObjSec then                           Internalerror(2019092803);                         Tai_const(hp).value:=ObjData.CurrObjSec.Size-objsymend.address+Tai_const(hp).symofs;                       end                     else if objsymend.objsection<>objsym.objsection then                       begin                         if (Tai_const(hp).consttype in [aitconst_uleb128bit,aitconst_sleb128bit]) or                            (objsym.objsection<>ObjData.CurrObjSec) then                           internalerror(2019010301);                       end                     else{$push} {$R-}{$Q-}                       Tai_const(hp).value:=objsymend.address-objsym.address+Tai_const(hp).symofs;                   end;{$pop}                 case tai_const(hp).consttype of                   aitconst_64bit,                   aitconst_32bit,                   aitconst_16bit,                   aitconst_64bit_unaligned,                   aitconst_32bit_unaligned,                   aitconst_16bit_unaligned,                   aitconst_8bit :                     begin                       if assigned(tai_const(hp).sym) and                          not assigned(tai_const(hp).endsym) then                         ObjData.writereloc(Tai_const(hp).symofs,tai_const(hp).size,Objdata.SymbolRef(tai_const(hp).sym),RELOC_ABSOLUTE)                       else if relative_reloc then                         ObjData.writereloc(ObjData.CurrObjSec.size+tai_const(hp).size-objsym.address+tai_const(hp).symofs,tai_const(hp).size,objsymend,RELOC_RELATIVE)                       else                         if source_info.endian<>target_info.endian then                           begin                             case tai_const(hp).size of                                1 : begin                                      b:=byte(Tai_const(hp).value);                                      ObjData.writebytes(b,1);                                    end;                                2 : begin                                      w:=word(Tai_const(hp).value);                                      w:=swapendian(w);                                      ObjData.writebytes(w,2);                                    end;                                4 : begin                                      d:=dword(Tai_const(hp).value);                                      d:=swapendian(d);                                      ObjData.writebytes(d,4);                                    end;                                8 : begin                                      q:=qword(Tai_const(hp).value);                                      q:=swapendian(q);                                      ObjData.writebytes(q,8);                                    end;                             else                               internalerror(2024012502);                             end;                           end                         else                           ObjData.writebytes(Tai_const(hp).value,tai_const(hp).size);                     end;                   aitconst_rva_symbol :                     begin                       { PE32+? }                       if target_info.system in systems_peoptplus then                         ObjData.writereloc(Tai_const(hp).symofs,sizeof(longint),Objdata.SymbolRef(tai_const(hp).sym),RELOC_RVA)                       else                         ObjData.writereloc(Tai_const(hp).symofs,sizeof(pint),Objdata.SymbolRef(tai_const(hp).sym),RELOC_RVA);                     end;                   aitconst_secrel32_symbol :                     begin                       { Required for DWARF2 support under Windows }                       ObjData.writereloc(Tai_const(hp).symofs,sizeof(longint),Objdata.SymbolRef(tai_const(hp).sym),RELOC_SECREL32);                     end;{$ifdef i8086}                   aitconst_farptr :                     if assigned(tai_const(hp).sym) and                        not assigned(tai_const(hp).endsym) then                       ObjData.writereloc(Tai_const(hp).symofs,tai_const(hp).size,Objdata.SymbolRef(tai_const(hp).sym),RELOC_FARPTR)                     else if relative_reloc then                       internalerror(2015040601)                     else                       ObjData.writebytes(Tai_const(hp).value,tai_const(hp).size);                   aitconst_seg:                     if assigned(tai_const(hp).sym) and (tai_const(hp).size=2) then                       ObjData.writereloc(0,2,Objdata.SymbolRef(tai_const(hp).sym),RELOC_SEG)                     else                       internalerror(2015110502);                   aitconst_dgroup:                     ObjData.writereloc(0,2,nil,RELOC_DGROUP);                   aitconst_fardataseg:                     ObjData.writereloc(0,2,nil,RELOC_FARDATASEG);{$endif i8086}{$ifdef arm}                   aitconst_got:                     ObjData.writereloc(Tai_const(hp).symofs,sizeof(longint),Objdata.SymbolRef(tai_const(hp).sym),RELOC_GOT32);{                   aitconst_gottpoff:                     ObjData.writereloc(Tai_const(hp).symofs,sizeof(longint),Objdata.SymbolRef(tai_const(hp).sym),RELOC_TPOFF); }                   aitconst_tpoff:                     ObjData.writereloc(Tai_const(hp).symofs,sizeof(longint),Objdata.SymbolRef(tai_const(hp).sym),RELOC_TPOFF);                   aitconst_tlsgd:                     ObjData.writereloc(Tai_const(hp).symofs,sizeof(longint),Objdata.SymbolRef(tai_const(hp).sym),RELOC_TLSGD);                   aitconst_tlsdesc:                     begin                       { must be a relative symbol, thus value being valid }                       if not(assigned(tai_const(hp).sym)) or not(assigned(tai_const(hp).endsym)) then                         Internalerror(2019092904);                       ObjData.writereloc(Tai_const(hp).value,sizeof(longint),Objdata.SymbolRef(tai_const(hp).sym),RELOC_TLSDESC);                     end;{$endif arm}                   aitconst_dtpoff:                     { so far, the size of dtpoff is fixed to 4 bytes }                     ObjData.writereloc(Tai_const(hp).symofs,4,Objdata.SymbolRef(tai_const(hp).sym),RELOC_DTPOFF);                   aitconst_gotoff_symbol:                     ObjData.writereloc(Tai_const(hp).symofs,sizeof(longint),Objdata.SymbolRef(tai_const(hp).sym),RELOC_GOTOFF);                   aitconst_uleb128bit,                   aitconst_sleb128bit :                     begin                       if Tai_const(hp).fixed_size=0 then                         Internalerror(2019030302);                       if tai_const(hp).consttype=aitconst_uleb128bit then                         leblen:=EncodeUleb128(qword(Tai_const(hp).value),lebbuf,Tai_const(hp).fixed_size)                       else                         leblen:=EncodeSleb128(Tai_const(hp).value,lebbuf,Tai_const(hp).fixed_size);                       if leblen<>tai_const(hp).fixed_size then                         internalerror(200709271);                       ObjData.writebytes(lebbuf,leblen);                     end;                   aitconst_darwin_dwarf_delta32,                   aitconst_darwin_dwarf_delta64:                     ObjData.writebytes(Tai_const(hp).value,tai_const(hp).size);                   aitconst_half16bit,                   aitconst_gs:                     begin                       tmp:=Tai_const(hp).value div 2;                       ObjData.writebytes(tmp,2);                     end;                   else                     internalerror(200603254);                 end;               end;             ait_label :               begin                 { exporting shouldn't be necessary as labels are local,                   but it's better to be on the safe side (PFV) }                 ObjOutput.exportsymbol(ObjData.SymbolRef(Tai_label(hp).labsym));               end;             ait_instruction :               Taicpu(hp).Pass2(ObjData);             ait_stab :               WriteStab(Tai_stab(hp).str);             ait_function_name,             ait_force_line : ;             ait_cutobject :               if SmartAsm then                break;             ait_directive :               begin                 case tai_directive(hp).directive of                   asd_weak_definition,                   asd_weak_reference:                     begin                       objsym:=ObjData.symbolref(tai_directive(hp).name);                       if objsym.bind in [AB_EXTERNAL,AB_WEAK_EXTERNAL] then                         objsym.bind:=AB_WEAK_EXTERNAL                       else                         { TODO: should become a weak definition; for now, do                             the same as what was done for ait_weak }                         objsym.bind:=AB_WEAK_EXTERNAL;                     end;                   asd_cpu:                     begin                       ObjData.CPUType:=cpu_none;                       for cpu:=low(tcputype) to high(tcputype) do                         if cputypestr[cpu]=tai_directive(hp).name then                           begin                             ObjData.CPUType:=cpu;                             break;                           end;                     end;{$ifdef OMFOBJSUPPORT}                   asd_omf_linnum_line:                     begin                       TOmfObjSection(ObjData.CurrObjSec).LinNumEntries.Add(                         TOmfSubRecord_LINNUM_MsLink_Entry.Create(                           strtoint(tai_directive(hp).name),                           ObjData.CurrObjSec.Size                         ));                     end;{$endif OMFOBJSUPPORT}                   else                     ;                 end               end;             ait_symbolpair:               begin                 if tai_symbolpair(hp).kind=spk_set then                   begin                     objsym:=ObjData.symbolref(tai_symbolpair(hp).sym^);                     ref:=objdata.symbolref(tai_symbolpair(hp).value^);                     objsym.offset:=ref.offset;                     objsym.objsection:=ref.objsection;{$ifdef arm}                     objsym.ThumbFunc:=ref.ThumbFunc;{$endif arm}                   end;               end;{$ifndef DISABLE_WIN64_SEH}             ait_seh_directive :               tai_seh_directive(hp).generate_code(objdata);{$endif DISABLE_WIN64_SEH}             ait_eabi_attribute :               begin                 eabi_section:=ObjData.findsection('.ARM.attributes');                 if not(assigned(eabi_section)) then                   Internalerror(2019100704);                 if eabi_section.Size=0 then                   begin                     s:='A';                     eabi_section.write(s[1],1);                     ddword:=eabi_section.Size-1;                     if source_info.endian<>target_info.endian then                       ddword:=SwapEndian(ddword);                     eabi_section.write(ddword,4);                     s:='aeabi'#0;                     eabi_section.write(s[1],6);                     s:=#1;                     eabi_section.write(s[1],1);                     ddword:=eabi_section.Size-1-4-6-1;                     if source_info.endian<>target_info.endian then                       ddword:=SwapEndian(ddword);                     eabi_section.write(ddword,4);                   end;                 leblen:=EncodeUleb128(tai_eabi_attribute(hp).tag,lebbuf,0);                 eabi_section.write(lebbuf,leblen);                 case tai_eabi_attribute(hp).eattr_typ of                   eattrtype_dword:                     begin                       leblen:=EncodeUleb128(tai_eabi_attribute(hp).value,lebbuf,0);                       eabi_section.write(lebbuf,leblen);                     end;                   eattrtype_ntbs:                     begin                       if assigned(tai_eabi_attribute(hp).valuestr) then                         s:=tai_eabi_attribute(hp).valuestr^+#0                       else                         s:=#0;                       eabi_section.write(s[1],Length(s));                     end                   else                     Internalerror(2019100705);                 end;                 { update size of attributes section, write directly to the dyn. arrays as                   we do not increase the size of section }                 TmpDataPos:=eabi_section.Data.Pos;                 eabi_section.Data.seek(1);                 ddword:=eabi_section.Size-1;                 if source_info.endian<>target_info.endian then                   ddword:=SwapEndian(ddword);                 eabi_section.Data.write(ddword,4);                 eabi_section.Data.seek(12);                 ddword:=eabi_section.Size-1-4-6;                 if source_info.endian<>target_info.endian then                   ddword:=SwapEndian(ddword);                 eabi_section.Data.write(ddword,4);                 eabi_section.Data.Seek(TmpDataPos);               end;{$ifdef WASM}             ait_functype:               TWasmObjData(ObjData).DeclareFuncType_Pass2(tai_functype(hp));             ait_local:               TWasmObjData(ObjData).WriteLocals_Pass2(tai_local(hp));{$endif WASM}             else               ;           end;           hp:=Tai(hp.next);         end;        TreePass2:=hp;      end;    procedure TInternalAssembler.writetree;      label        doexit;      var        hp : Tai;        ObjWriter : TObjectWriter;      begin        ObjWriter:=TObjectwriter.create;        ObjOutput:=CObjOutput.Create(ObjWriter);        ObjData:=ObjOutput.newObjData(ObjFileName);        { Pass 0 }        ObjData.currpass:=0;        ObjData.createsection(sec_code);        ObjData.beforealloc;        { start with list 1 }        currlistidx:=1;        currlist:=list[currlistidx];        hp:=Tai(currList.first);        while assigned(hp) do         begin           hp:=TreePass0(hp);           MaybeNextList(hp);         end;        ObjData.afteralloc;        { leave if errors have occurred }        if errorcount>0 then         goto doexit;        { Pass 1 }        ObjData.currpass:=1;        ObjData.resetsections;        ObjData.beforealloc;        ObjData.createsection(sec_code);        { start with list 1 }        currlistidx:=1;        currlist:=list[currlistidx];        hp:=Tai(currList.first);        while assigned(hp) do         begin           hp:=TreePass1(hp);           MaybeNextList(hp);         end;        ObjData.createsection(sec_code);        ObjData.afteralloc;        { leave if errors have occurred }        if errorcount>0 then         goto doexit;        { Pass 2 }        ObjData.currpass:=2;        ObjData.resetsections;        ObjData.beforewrite;        ObjData.createsection(sec_code);        { start with list 1 }        currlistidx:=1;        currlist:=list[currlistidx];        hp:=Tai(currList.first);        while assigned(hp) do         begin           hp:=TreePass2(hp);           MaybeNextList(hp);         end;        ObjData.createsection(sec_code);        ObjData.afterwrite;        { don't write the .o file if errors have occurred }        if errorcount=0 then         begin           { write objectfile }           ObjOutput.startobjectfile(ObjFileName);           ObjOutput.writeobjectfile(ObjData);         end;      doexit:        { Cleanup }        ObjData.free;        ObjData:=nil;        ObjWriter.free;      end;    procedure TInternalAssembler.writetreesmart;      var        hp : Tai;        startsectype : TAsmSectiontype;        place: tcutplace;        ObjWriter : TObjectWriter;        startsecname: String;        startsecorder: TAsmSectionOrder;      begin        if not(cs_asm_leave in current_settings.globalswitches) and           not(af_needar in asminfo^.flags) then          ObjWriter:=CInternalAr.CreateAr(current_module.staticlibfilename)        else          ObjWriter:=TObjectwriter.create;        NextSmartName(cut_normal);        ObjOutput:=CObjOutput.Create(ObjWriter);        startsectype:=sec_none;        startsecname:='';        startsecorder:=secorder_default;        { start with list 1 }        currlistidx:=1;        currlist:=list[currlistidx];        hp:=Tai(currList.first);        while assigned(hp) do         begin           ObjData:=ObjOutput.newObjData(ObjFileName);           { Pass 0 }           ObjData.currpass:=0;           ObjData.resetsections;           ObjData.beforealloc;           if startsectype<>sec_none then             ObjData.CreateSection(startsectype,startsecname,startsecorder);           TreePass0(hp);           ObjData.afteralloc;           { leave if errors have occurred }           if errorcount>0 then             break;           { Pass 1 }           ObjData.currpass:=1;           ObjData.resetsections;           ObjData.beforealloc;           if startsectype<>sec_none then             ObjData.CreateSection(startsectype,startsecname,startsecorder);           TreePass1(hp);           ObjData.afteralloc;           { leave if errors have occurred }           if errorcount>0 then             break;           { Pass 2 }           ObjData.currpass:=2;           ObjOutput.startobjectfile(ObjFileName);           ObjData.resetsections;           ObjData.beforewrite;           if startsectype<>sec_none then             ObjData.CreateSection(startsectype,startsecname,startsecorder);           hp:=TreePass2(hp);           ObjData.afterwrite;           { leave if errors have occurred }           if errorcount>0 then             break;           { write the current objectfile }           ObjOutput.writeobjectfile(ObjData);           ObjData.free;           ObjData:=nil;           { end of lists? }           if not MaybeNextList(hp) then             break;           { we will start a new objectfile so reset everything }           { The place can still change in the next while loop, so don't init }           { the writer yet (JM)                                              }           if (hp.typ=ait_cutobject) then             place := Tai_cutobject(hp).place           else             place := cut_normal;           { avoid empty files }           startsectype:=sec_none;           startsecname:='';           startsecorder:=secorder_default;           while assigned(hp) and                 (Tai(hp).typ in [ait_marker,ait_comment,ait_section,ait_cutobject]) do            begin              if Tai(hp).typ=ait_section then                begin                  startsectype:=Tai_section(hp).sectype;                  startsecname:=Tai_section(hp).name^;                  startsecorder:=Tai_section(hp).secorder;                end;              if (Tai(hp).typ=ait_cutobject) then                place:=Tai_cutobject(hp).place;              hp:=Tai(hp.next);            end;           if not MaybeNextList(hp) then             break;           { start next objectfile }           NextSmartName(place);         end;        ObjData.free;        ObjData:=nil;        ObjWriter.free;      end;    procedure TInternalAssembler.MakeObject;    var to_do:set of TasmlistType;        i:TasmlistType;        procedure addlist(p:TAsmList);        begin          inc(lists);          list[lists]:=p;        end;      begin        to_do:=[low(Tasmlisttype)..high(Tasmlisttype)];        if usedeffileforexports then          exclude(to_do,al_exports);        if not(tf_section_threadvars in target_info.flags) then          exclude(to_do,al_threadvars);        for i:=low(TasmlistType) to high(TasmlistType) do          if (i in to_do) and (current_asmdata.asmlists[i]<>nil) and             (not current_asmdata.asmlists[i].empty) then            addlist(current_asmdata.asmlists[i]);        if SmartAsm then          writetreesmart        else          writetree;      end;{*****************************************************************************                     Generate Assembler Files Main Procedure*****************************************************************************}    Procedure GenerateAsm(smart:boolean);      var        a : TAssembler;      begin        if not assigned(CAssembler[target_asm.id]) then          Message(asmw_f_assembler_output_not_supported);        a:=CAssembler[target_asm.id].Create(@target_asm,smart);        a.MakeObject;        a.Free;      end;    function GetExternalGnuAssemblerWithAsmInfoWriter(info: pasminfo; wr: TExternalAssemblerOutputFile): TExternalAssembler;      var        asmkind: tasm;      begin        for asmkind in [as_gas,as_ggas,as_darwin,as_clang_gas,as_clang_asdarwin] do          if assigned(asminfos[asmkind]) and             (target_info.system in asminfos[asmkind]^.supported_targets) then            begin              result:=TExternalAssemblerClass(CAssembler[asmkind]).CreateWithWriter(asminfos[asmkind],wr,false,false);              exit;            end;        Internalerror(2015090604);      end;{*****************************************************************************                                 Init/Done*****************************************************************************}    procedure RegisterAssembler(const r:tasminfo;c:TAssemblerClass);      var        t : tasm;      begin        t:=r.id;        if assigned(asminfos[t]) then          writeln('Warning: Assembler is already registered!')        else          Getmem(asminfos[t],sizeof(tasminfo));        asminfos[t]^:=r;        CAssembler[t]:=c;      end;end.
 |