| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554 | {    Copyright (c) 1998-2006 by the Free Pascal team    This unit implements the generic part of the GNU assembler    (v2.8 or later) writer    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. ****************************************************************************}{ Base unit for writing GNU assembler output.}unit aggas;{$i fpcdefs.inc}interface    uses      cclasses,      globtype,globals,      aasmbase,aasmtai,aasmdata,aasmcpu,      assemble;    type      TCPUInstrWriter = class;      {# This is a derived class which is used to write         GAS styled assembler.      }      { TGNUAssembler }      TGNUAssembler=class(texternalassembler)      protected        function sectionname(atype:TAsmSectiontype;const aname:string;aorder:TAsmSectionOrder):string;virtual;        procedure WriteSection(atype:TAsmSectiontype;const aname:string;aorder:TAsmSectionOrder);        procedure WriteExtraHeader;virtual;        procedure WriteInstruction(hp: tai);        procedure WriteWeakSymbolDef(s: tasmsymbol); virtual;       public        function MakeCmdLine: TCmdStr; override;        procedure WriteTree(p:TAsmList);override;        procedure WriteAsmList;override;        destructor destroy; override;       private        setcount: longint;        procedure WriteDecodedSleb128(a: int64);        procedure WriteDecodedUleb128(a: qword);        function NextSetLabel: string;       protected        InstrWriter: TCPUInstrWriter;      end;      {# This is the base class for writing instructions.         The WriteInstruction() method must be overriden         to write a single instruction to the assembler         file.      }      TCPUInstrWriter = class        constructor create(_owner: TGNUAssembler);        procedure WriteInstruction(hp : tai); virtual; abstract;       protected        owner: TGNUAssembler;      end;      { TAppleGNUAssembler }      TAppleGNUAssembler=class(TGNUAssembler)       protected        function sectionname(atype:TAsmSectiontype;const aname:string;aorder:TAsmSectionOrder):string;override;        procedure WriteWeakSymbolDef(s: tasmsymbol); override;       end;      TAoutGNUAssembler=class(TGNUAssembler)        function sectionname(atype:TAsmSectiontype;const aname:string;aorder:TAsmSectionOrder):string;override;       end;implementation    uses      SysUtils,      cutils,cfileutl,systems,      fmodule,finput,verbose,      itcpugas,cpubase      ;    const      line_length = 70;    var      symendcount  : longint;    type{$ifdef cpuextended}      t80bitarray = array[0..9] of byte;{$endif cpuextended}      t64bitarray = array[0..7] of byte;      t32bitarray = array[0..3] of byte;{****************************************************************************}{                          Support routines                                  }{****************************************************************************}   function fixline(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);       for k:=j to i do        if s[k] in [#0..#31,#127..#255] then         s[k]:='.';       fixline:=Copy(s,j,i-j+1);     end;    function 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 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 extended2str(e : extended) : string;      var         hs : string;      begin         str(e,hs);      { replace space with + }         if hs[1]=' ' then          hs[1]:='+';         extended2str:='0d'+hs      end;  { convert floating point values }  { to correct endian             }  procedure swap64bitarray(var t: t64bitarray);    var     b: byte;    begin      b:= t[7];      t[7] := t[0];      t[0] := b;      b := t[6];      t[6] := t[1];      t[1] := b;      b:= t[5];      t[5] := t[2];      t[2] := b;      b:= t[4];      t[4] := t[3];      t[3] := b;   end;   procedure swap32bitarray(var t: t32bitarray);    var     b: byte;    begin      b:= t[1];      t[1]:= t[2];      t[2]:= b;      b:= t[0];      t[0]:= t[3];      t[3]:= b;    end;    const      ait_const2str : array[aitconst_128bit..aitconst_darwin_dwarf_delta32] of string[20]=(        #9'.fixme128'#9,#9'.quad'#9,#9'.long'#9,#9'.short'#9,#9'.byte'#9,        #9'.sleb128'#9,#9'.uleb128'#9,        #9'.rva'#9,#9'.secrel32'#9,#9'.quad'#9,#9'.long'#9      );{****************************************************************************}{                          GNU Assembler writer                              }{****************************************************************************}    destructor TGNUAssembler.Destroy;      begin        InstrWriter.free;        inherited destroy;      end;    function TGNUAssembler.MakeCmdLine: TCmdStr;      begin        result := inherited MakeCmdLine;        // MWE: disabled again. It generates dwarf info for the generated .s        //      files as well. This conflicts with the info we generate        // if target_dbg.id = dbg_dwarf then        //  result := result + ' --gdwarf-2';      end;    function TGNUAssembler.NextSetLabel: string;      begin        inc(setcount);        result := target_asm.labelprefix+'$set$'+tostr(setcount);      end;    function TGNUAssembler.sectionname(atype:TAsmSectiontype;const aname:string;aorder:TAsmSectionOrder):string;      const        secnames : array[TAsmSectiontype] of string[length('__DATA, __datacoal_nt,coalesced')] = ('',          '.text',          '.data',{ why doesn't .rodata work? (FK) }{ sometimes we have to create a data.rel.ro instead of .rodata, e.g. for  }{ vtables (and anything else containing relocations), otherwise those are }{ not relocated properly on e.g. linux/ppc64. g++ generates there for a   }{ vtable for a class called Window:                                       }{ .section .data.rel.ro._ZTV6Window,"awG",@progbits,_ZTV6Window,comdat    }{ TODO: .data.ro not yet working}{$if defined(arm) or defined(powerpc)}          '.rodata',{$else arm}          '.data',{$endif arm}{$if defined(m68k)} { Amiga/m68k GNU AS doesn't seem to like .rodata (KB) }          '.data',{$else}    '.rodata',{$endif}          '.bss',          '.threadvar',          '.pdata',          '', { stubs }          '__DATA,__nl_symbol_ptr',          '__DATA,__la_symbol_ptr',          '__DATA,__mod_init_func',          '__DATA,__mod_term_func',          '.stab',          '.stabstr',          '.idata$2','.idata$4','.idata$5','.idata$6','.idata$7','.edata',          '.eh_frame',          '.debug_frame','.debug_info','.debug_line','.debug_abbrev',          '.fpc',          '.toc',          '.init',          '.fini',          '.objc_class',          '.objc_meta_class',          '.objc_cat_cls_meth',          '.objc_cat_inst_meth',          '.objc_protocol',          '.objc_string_object',          '.objc_cls_meth',          '.objc_inst_meth',          '.objc_cls_refs',          '.objc_message_refs',          '.objc_symbols',          '.objc_category',          '.objc_class_vars',          '.objc_instance_vars',          '.objc_module_info',          '.objc_class_names',          '.objc_meth_var_types',          '.objc_meth_var_names',          '.objc_selector_strs',          '.objc_protocol_ext',          '.objc_class_ext',          '.objc_property',          '.objc_image_info',          '.objc_cstring_object',          '.objc_sel_fixup',          '__DATA,__objc_data',          '__DATA,__objc_const',          '.objc_superrefs',          '__DATA, __datacoal_nt,coalesced',          '.objc_classlist',          '.objc_nlclasslist',          '.objc_catlist',          '.obcj_nlcatlist',          '.objc_protolist'        );        secnames_pic : array[TAsmSectiontype] of string[length('__DATA, __datacoal_nt,coalesced')] = ('',          '.text',          '.data.rel',          '.data.rel',          '.data.rel',          '.bss',          '.threadvar',          '.pdata',          '', { stubs }          '__DATA,__nl_symbol_ptr',          '__DATA,__la_symbol_ptr',          '__DATA,__mod_init_func',          '__DATA,__mod_term_func',          '.stab',          '.stabstr',          '.idata$2','.idata$4','.idata$5','.idata$6','.idata$7','.edata',          '.eh_frame',          '.debug_frame','.debug_info','.debug_line','.debug_abbrev',          '.fpc',          '.toc',          '.init',          '.fini',          '.objc_class',          '.objc_meta_class',          '.objc_cat_cls_meth',          '.objc_cat_inst_meth',          '.objc_protocol',          '.objc_string_object',          '.objc_cls_meth',          '.objc_inst_meth',          '.objc_cls_refs',          '.objc_message_refs',          '.objc_symbols',          '.objc_category',          '.objc_class_vars',          '.objc_instance_vars',          '.objc_module_info',          '.objc_class_names',          '.objc_meth_var_types',          '.objc_meth_var_names',          '.objc_selector_strs',          '.objc_protocol_ext',          '.objc_class_ext',          '.objc_property',          '.objc_image_info',          '.objc_cstring_object',          '.objc_sel_fixup',          '__DATA, __objc_data',          '__DATA, __objc_const',          '.objc_superrefs',          '__DATA, __datacoal_nt,coalesced',          '.objc_classlist',          '.objc_nlclasslist',          '.objc_catlist',          '.obcj_nlcatlist',          '.objc_protolist'        );      var        sep     : string[3];        secname : string;      begin        if (cs_create_pic in current_settings.moduleswitches) and           not(target_info.system in systems_darwin) then          secname:=secnames_pic[atype]        else          secname:=secnames[atype];{$ifdef m68k}        { old Amiga GNU AS doesn't support .section .fpc }        if (atype=sec_fpc) and (target_info.system = system_m68k_amiga) then            secname:=secnames[sec_data];{$endif}        if (atype=sec_fpc) and (Copy(aname,1,3)='res') then          begin            result:=secname+'.'+aname;            exit;          end;        if (atype=sec_threadvar) and          (target_info.system=system_i386_win32) then          secname:='.tls';        { go32v2 stub only loads .text and .data sections, and allocates space for .bss.          Thus, data which normally goes into .rodata and .rodata_norel sections must          end up in .data section }        if (atype in [sec_rodata,sec_rodata_norel]) and          (target_info.system=system_i386_go32v2) then          secname:='.data';        { For bss we need to set some flags that are target dependent,          it is easier to disable it for smartlinking. It doesn't take up          filespace }        if not(target_info.system in systems_darwin) and           create_smartlink_sections and           (aname<>'') and           (atype<>sec_toc) and           { on embedded systems every byte counts, so smartlink bss too }           ((atype<>sec_bss) or (target_info.system in systems_embedded)) then          begin            case aorder of              secorder_begin :                sep:='.b_';              secorder_end :                sep:='.z_';              else                sep:='.n_';            end;            result:=secname+sep+aname          end        else          result:=secname;      end;    procedure TGNUAssembler.WriteSection(atype:TAsmSectiontype;const aname:string;aorder:TAsmSectionOrder);      var        s : string;      begin        AsmLn;        case target_info.system of         system_i386_OS2,         system_i386_EMX,         system_m68k_amiga,  { amiga has old GNU AS (2.14), which blews up from .section (KB) }         system_m68k_linux: ;         system_powerpc_darwin,         system_i386_darwin,         system_powerpc64_darwin,         system_x86_64_darwin,         system_arm_darwin:           begin             if (atype in [sec_stub,sec_objc_data,sec_objc_const,sec_data_coalesced]) then               AsmWrite('.section ');           end         else          AsmWrite('.section ');        end;        s:=sectionname(atype,aname,aorder);        AsmWrite(s);        case atype of          sec_fpc :            if aname = 'resptrs' then              AsmWrite(', "a", @progbits');          sec_stub :            begin              case target_info.system of                { there are processor-independent shortcuts available    }                { for this, namely .symbol_stub and .picsymbol_stub, but }                { they don't work and gcc doesn't use them either...     }                system_powerpc_darwin,                system_powerpc64_darwin:                  if (cs_create_pic in current_settings.moduleswitches) then                    AsmWriteln('__TEXT,__picsymbolstub1,symbol_stubs,pure_instructions,32')                  else                    AsmWriteln('__TEXT,__symbol_stub1,symbol_stubs,pure_instructions,16');                system_i386_darwin:                  AsmWriteln('__IMPORT,__jump_table,symbol_stubs,self_modifying_code+pure_instructions,5');                system_arm_darwin:                  if (cs_create_pic in current_settings.moduleswitches) then                    AsmWriteln('.section __TEXT,__picsymbolstub4,symbol_stubs,none,16')                  else                    AsmWriteln('.section __TEXT,__symbol_stub4,symbol_stubs,none,12')                { darwin/x86-64 uses RIP-based GOT addressing, no symbol stubs }                else                  internalerror(2006031101);              end;            end;        end;        AsmLn;        LastSecType:=atype;      end;    procedure TGNUAssembler.WriteDecodedUleb128(a: qword);      var        i,len : longint;        buf   : array[0..63] of byte;      begin        len:=EncodeUleb128(a,buf);        for i:=0 to len-1 do          begin            if (i > 0) then              AsmWrite(',');            AsmWrite(tostr(buf[i]));          end;      end;    procedure TGNUAssembler.WriteDecodedSleb128(a: int64);      var        i,len : longint;        buf   : array[0..255] of byte;      begin        len:=EncodeSleb128(a,buf);        for i:=0 to len-1 do          begin            if (i > 0) then              AsmWrite(',');            AsmWrite(tostr(buf[i]));          end;      end;    procedure TGNUAssembler.WriteTree(p:TAsmList);      function needsObject(hp : tai_symbol) : boolean;        begin          needsObject :=              (                assigned(hp.next) and                 (tai(hp.next).typ in [ait_const,ait_datablock,                  ait_real_32bit,ait_real_64bit,ait_real_80bit,ait_comp_64bit])              ) or              (hp.sym.typ=AT_DATA);          end;          procedure doalign(alignment: byte; use_op: boolean; fillop: byte; out last_align: longint);        var          i: longint;        begin          last_align:=alignment;          if alignment>1 then            begin              if not(target_info.system in systems_darwin) then                begin                  AsmWrite(#9'.balign '+tostr(alignment));                  if use_op then                    AsmWrite(','+tostr(fillop)){$ifdef x86}                  { force NOP as alignment op code }                  else if LastSecType=sec_code then                    AsmWrite(',0x90');{$endif x86}                end              else                begin                  { darwin as only supports .align }                  if not ispowerof2(alignment,i) then                    internalerror(2003010305);                  AsmWrite(#9'.align '+tostr(i));                  last_align:=i;                end;              AsmLn;            end;        end;    var      ch       : char;      hp       : tai;      hp1      : tailineinfo;      constdef : taiconst_type;      s,t      : string;      i,pos,l  : longint;      InlineLevel : longint;      last_align : longint;      co       : comp;      sin      : single;      d        : double;{$ifdef cpuextended}      e        : extended;{$endif cpuextended}      do_line  : boolean;      sepChar : char;    begin      if not assigned(p) then       exit;      last_align := 2;      InlineLevel:=0;      { lineinfo is only needed for al_procedures (PFV) }      do_line:=(cs_asm_source in current_settings.globalswitches) or               ((cs_lineinfo in current_settings.moduleswitches)                 and (p=current_asmdata.asmlists[al_procedures]));      hp:=tai(p.first);      while assigned(hp) do       begin         prefetch(pointer(hp.next)^);         if not(hp.typ in SkipLineInfo) then          begin            hp1 := hp as tailineinfo;            current_filepos:=hp1.fileinfo;             { no line info for inlined code }             if do_line and (inlinelevel=0) then              begin                { load infile }                if lastfileinfo.fileindex<>hp1.fileinfo.fileindex then                 begin                   infile:=current_module.sourcefiles.get_file(hp1.fileinfo.fileindex);                   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:=hp1.fileinfo.fileindex;                   { 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                       AsmWriteLn(target_asm.comment+'['+infile.name^+']');                       if assigned(lastinfile) then                         lastinfile.close;                     end;                   if (hp1.fileinfo.line<>lastfileinfo.line) and                      ((hp1.fileinfo.line<infile.maxlinebuf) or (InlineLevel>0)) then                     begin                       if (hp1.fileinfo.line<>0) and                          ((infile.linebuf^[hp1.fileinfo.line]>=0) or (InlineLevel>0)) then                         AsmWriteLn(target_asm.comment+'['+tostr(hp1.fileinfo.line)+'] '+                           fixline(infile.GetLineStr(hp1.fileinfo.line)));                       { set it to a negative value !                       to make that is has been read already !! PM }                       if (infile.linebuf^[hp1.fileinfo.line]>=0) then                         infile.linebuf^[hp1.fileinfo.line]:=-infile.linebuf^[hp1.fileinfo.line]-1;                     end;                 end;                lastfileinfo:=hp1.fileinfo;                lastinfile:=infile;              end;          end;         case hp.typ of           ait_comment :             Begin               AsmWrite(target_asm.comment);               AsmWritePChar(tai_comment(hp).str);               AsmLn;             End;           ait_regalloc :             begin               if (cs_asm_regalloc in current_settings.globalswitches) then                 begin                   AsmWrite(#9+target_asm.comment+'Register ');                   repeat                     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);                     AsmWrite(',');                   until false;                   AsmWrite(' ');                   AsmWriteLn(regallocstr[tai_regalloc(hp).ratype]);                 end;             end;           ait_tempalloc :             begin               if (cs_asm_tempalloc in current_settings.globalswitches) then                 begin{$ifdef EXTDEBUG}                   if assigned(tai_tempalloc(hp).problem) then                     AsmWriteLn(target_asm.comment+'Temp '+tostr(tai_tempalloc(hp).temppos)+','+                       tostr(tai_tempalloc(hp).tempsize)+' '+tai_tempalloc(hp).problem^)                   else{$endif EXTDEBUG}                     AsmWriteLn(target_asm.comment+'Temp '+tostr(tai_tempalloc(hp).temppos)+','+                       tostr(tai_tempalloc(hp).tempsize)+' '+tempallocstr[tai_tempalloc(hp).allocation]);                 end;             end;           ait_align :             begin               doalign(tai_align_abstract(hp).aligntype,tai_align_abstract(hp).use_op,tai_align_abstract(hp).fillop,last_align);             end;           ait_section :             begin               if tai_section(hp).sectype<>sec_none then                 WriteSection(tai_section(hp).sectype,tai_section(hp).name^,tai_section(hp).secorder)               else                 begin{$ifdef EXTDEBUG}                   AsmWrite(target_asm.comment);                   AsmWriteln(' sec_none');{$endif EXTDEBUG}                end;             end;           ait_datablock :             begin               if (target_info.system in systems_darwin) then                 begin                   { On Mac OS X you can't have common symbols in a shared library                     since those are in the TEXT section and the text section is                     read-only in shared libraries (so it can be shared among different                     processes). The alternate code creates some kind of common symbols                     in the data segment.                   }                   if tai_datablock(hp).is_global then                     begin                       asmwrite('.globl ');                       asmwriteln(tai_datablock(hp).sym.name);                       asmwriteln('.data');                       asmwrite('.zerofill __DATA, __common, ');                       asmwrite(tai_datablock(hp).sym.name);                       asmwriteln(', '+tostr(tai_datablock(hp).size)+','+tostr(last_align));                       if not(LastSecType in [sec_data,sec_none]) then                         writesection(LastSecType,'',secorder_default);                     end                   else                     begin                       asmwrite(#9'.lcomm'#9);                       asmwrite(tai_datablock(hp).sym.name);                       asmwrite(','+tostr(tai_datablock(hp).size));                       asmwrite(','+tostr(last_align));                       asmln;                     end;                 end               else                 begin{$ifdef USE_COMM_IN_BSS}                   if writingpackages then                     begin                       { The .comm is required for COMMON symbols. These are used                         in the shared library loading. All the symbols declared in                         the .so file need to resolve to the data allocated in the main                         program (PFV) }                       if tai_datablock(hp).is_global then                         begin                           asmwrite(#9'.comm'#9);                           asmwrite(tai_datablock(hp).sym.name);                           asmwrite(','+tostr(tai_datablock(hp).size));                           asmwrite(','+tostr(last_align));                           asmln;                         end                       else                         begin                           asmwrite(#9'.lcomm'#9);                           asmwrite(tai_datablock(hp).sym.name);                           asmwrite(','+tostr(tai_datablock(hp).size));                           asmwrite(','+tostr(last_align));                           asmln;                         end                     end                   else{$endif USE_COMM_IN_BSS}                     begin                       if Tai_datablock(hp).is_global then                         begin                           asmwrite(#9'.globl ');                           asmwriteln(Tai_datablock(hp).sym.name);                         end;                       if (target_info.system <> system_arm_linux) then                         sepChar := '@'                       else                         sepChar := '%';                       if (tf_needs_symbol_type in target_info.flags) then                         asmwriteln(#9'.type '+Tai_datablock(hp).sym.name+','+sepChar+'object');                       if (tf_needs_symbol_size in target_info.flags) and (tai_datablock(hp).size > 0) then                         asmwriteln(#9'.size '+Tai_datablock(hp).sym.name+','+tostr(Tai_datablock(hp).size));                       asmwrite(Tai_datablock(hp).sym.name);                       asmwriteln(':');                       asmwriteln(#9'.zero '+tostr(Tai_datablock(hp).size));                     end;                 end;             end;           ait_const:             begin               constdef:=tai_const(hp).consttype;               case constdef of{$ifndef cpu64bitaddr}                 aitconst_128bit :                    begin                      internalerror(200404291);                    end;                 aitconst_64bit :                    begin                      if assigned(tai_const(hp).sym) then                        internalerror(200404292);                      AsmWrite(ait_const2str[aitconst_32bit]);                      if target_info.endian = endian_little then                        begin                          AsmWrite(tostr(longint(lo(tai_const(hp).value))));                          AsmWrite(',');                          AsmWrite(tostr(longint(hi(tai_const(hp).value))));                        end                      else                        begin                          AsmWrite(tostr(longint(hi(tai_const(hp).value))));                          AsmWrite(',');                          AsmWrite(tostr(longint(lo(tai_const(hp).value))));                        end;                      AsmLn;                    end;{$endif cpu64bitaddr}                 aitconst_uleb128bit,                 aitconst_sleb128bit,{$ifdef cpu64bitaddr}                 aitconst_128bit,                 aitconst_64bit,{$endif cpu64bitaddr}                 aitconst_32bit,                 aitconst_16bit,                 aitconst_8bit,                 aitconst_rva_symbol,                 aitconst_secrel32_symbol,                 aitconst_darwin_dwarf_delta32,                 aitconst_darwin_dwarf_delta64:                   begin                     if (target_info.system in systems_darwin) and                        (constdef in [aitconst_uleb128bit,aitconst_sleb128bit]) then                       begin                         AsmWrite(ait_const2str[aitconst_8bit]);                         case tai_const(hp).consttype of                           aitconst_uleb128bit:                             WriteDecodedUleb128(qword(tai_const(hp).value));                           aitconst_sleb128bit:                             WriteDecodedSleb128(int64(tai_const(hp).value));                         end                       end                     else                       begin                         AsmWrite(ait_const2str[constdef]);                         l:=0;                         t := '';                         repeat                           if assigned(tai_const(hp).sym) then                             begin                               if assigned(tai_const(hp).endsym) then                                 begin                                   if (constdef in [aitconst_darwin_dwarf_delta32,aitconst_darwin_dwarf_delta64]) then                                     begin                                       s := NextSetLabel;                                       t := #9'.set '+s+','+tai_const(hp).endsym.name+'-'+tai_const(hp).sym.name;                                     end                                   else                                     s:=tai_const(hp).endsym.name+'-'+tai_const(hp).sym.name                                  end                               else                                 s:=tai_const(hp).sym.name;                               if tai_const(hp).value<>0 then                                 s:=s+tostr_with_plus(tai_const(hp).value);                             end                           else{$ifdef cpu64bitaddr}                             s:=tostr(tai_const(hp).value);{$else cpu64bitaddr}                             { 64 bit constants are already handled above in this case }                             s:=tostr(longint(tai_const(hp).value));{$endif cpu64bitaddr}                           AsmWrite(s);                           inc(l,length(s));                           { Values with symbols are written on a single line to improve                             reading of the .s file (PFV) }                           if assigned(tai_const(hp).sym) or                              not(LastSecType in [sec_data,sec_rodata,sec_rodata_norel]) or                              (l>line_length) or                              (hp.next=nil) or                              (tai(hp.next).typ<>ait_const) or                              (tai_const(hp.next).consttype<>constdef) or                              assigned(tai_const(hp.next).sym) then                             break;                           hp:=tai(hp.next);                           AsmWrite(',');                         until false;                         if (t <> '') then                           begin                             AsmLn;                             AsmWrite(t);                           end;                       end;                      AsmLn;                   end;                 else                   internalerror(200704251);               end;             end;           { the "and defined(FPC_HAS_TYPE_EXTENDED)" isn't optimal but currently the only solution             it prevents proper cross compilation to i386 though           }{$if defined(cpuextended) and defined(FPC_HAS_TYPE_EXTENDED)}           ait_real_80bit :             begin               if do_line then                AsmWriteLn(target_asm.comment+'value: '+extended2str(tai_real_80bit(hp).value));             { Make sure e is a extended type, bestreal could be               a different type (bestreal) !! (PFV) }               e:=tai_real_80bit(hp).value;               AsmWrite(#9'.byte'#9);               for i:=0 to 9 do                begin                  if i<>0 then                   AsmWrite(',');                  AsmWrite(tostr(t80bitarray(e)[i]));                end;               for i:=11 to tai_real_80bit(hp).savesize do                 AsmWrite(',0');               AsmLn;             end;{$endif cpuextended}           ait_real_64bit :             begin               if do_line then                AsmWriteLn(target_asm.comment+'value: '+double2str(tai_real_64bit(hp).value));               d:=tai_real_64bit(hp).value;               { swap the values to correct endian if required }               if source_info.endian <> target_info.endian then                 swap64bitarray(t64bitarray(d));               AsmWrite(#9'.byte'#9);{$ifdef arm}               if tai_real_64bit(hp).formatoptions=fo_hiloswapped then                 begin                   for i:=4 to 7 do                     begin                       if i<>4 then                         AsmWrite(',');                       AsmWrite(tostr(t64bitarray(d)[i]));                     end;                   for i:=0 to 3 do                     begin                       AsmWrite(',');                       AsmWrite(tostr(t64bitarray(d)[i]));                     end;                 end               else{$endif arm}                 begin                   for i:=0 to 7 do                     begin                       if i<>0 then                         AsmWrite(',');                       AsmWrite(tostr(t64bitarray(d)[i]));                     end;                 end;               AsmLn;             end;           ait_real_32bit :             begin               if do_line then                AsmWriteLn(target_asm.comment+'value: '+single2str(tai_real_32bit(hp).value));               sin:=tai_real_32bit(hp).value;               { swap the values to correct endian if required }               if source_info.endian <> target_info.endian then                 swap32bitarray(t32bitarray(sin));               AsmWrite(#9'.byte'#9);               for i:=0 to 3 do                begin                  if i<>0 then                   AsmWrite(',');                  AsmWrite(tostr(t32bitarray(sin)[i]));                end;               AsmLn;             end;           ait_comp_64bit :             begin               if do_line then                AsmWriteLn(target_asm.comment+'value: '+extended2str(tai_comp_64bit(hp).value));               AsmWrite(#9'.byte'#9);               co:=comp(tai_comp_64bit(hp).value);               { swap the values to correct endian if required }               if source_info.endian <> target_info.endian then                 swap64bitarray(t64bitarray(co));               for i:=0 to 7 do                begin                  if i<>0 then                   AsmWrite(',');                  AsmWrite(tostr(t64bitarray(co)[i]));                end;               AsmLn;             end;           ait_string :             begin               pos:=0;               for i:=1 to tai_string(hp).len do                begin                  if pos=0 then                   begin                     AsmWrite(#9'.ascii'#9'"');                     pos:=20;                   end;                  ch:=tai_string(hp).str[i-1];                  case ch of                     #0, {This can't be done by range, because a bug in FPC}                #1..#31,             #128..#255 : s:='\'+tostr(ord(ch) shr 6)+tostr((ord(ch) and 63) shr 3)+tostr(ord(ch) and 7);                    '"' : s:='\"';                    '\' : s:='\\';                  else                   s:=ch;                  end;                  AsmWrite(s);                  inc(pos,length(s));                  if (pos>line_length) or (i=tai_string(hp).len) then                   begin                     AsmWriteLn('"');                     pos:=0;                   end;                end;             end;           ait_label :             begin               if (tai_label(hp).labsym.is_used) then                begin                  if (tai_label(hp).labsym.bind=AB_PRIVATE_EXTERN) then                    begin                      AsmWrite(#9'.private_extern ');                      AsmWriteln(tai_label(hp).labsym.name);                    end;                  if tai_label(hp).labsym.bind in [AB_GLOBAL,AB_PRIVATE_EXTERN] then                   begin                     AsmWrite('.globl'#9);                     AsmWriteLn(tai_label(hp).labsym.name);                   end;                  AsmWrite(tai_label(hp).labsym.name);                  AsmWriteLn(':');                end;             end;           ait_symbol :             begin               if (tai_symbol(hp).sym.bind=AB_PRIVATE_EXTERN) then                 begin                   AsmWrite(#9'.private_extern ');                   AsmWriteln(tai_symbol(hp).sym.name);                 end;               if (target_info.system = system_powerpc64_linux) and                 (tai_symbol(hp).sym.typ = AT_FUNCTION) and (cs_profile in current_settings.moduleswitches) then                 begin                 AsmWriteLn('.globl _mcount');               end;               if tai_symbol(hp).is_global then                begin                  AsmWrite('.globl'#9);                  AsmWriteLn(tai_symbol(hp).sym.name);                end;               if (target_info.system = system_powerpc64_linux) and                 (tai_symbol(hp).sym.typ = AT_FUNCTION) then                 begin                   AsmWriteLn('.section ".opd", "aw"');                   AsmWriteLn('.align 3');                   AsmWriteLn(tai_symbol(hp).sym.name + ':');                   AsmWriteLn('.quad .' + tai_symbol(hp).sym.name + ', .TOC.@tocbase, 0');                   AsmWriteLn('.previous');                   AsmWriteLn('.size ' + tai_symbol(hp).sym.name + ', 24');                   if (tai_symbol(hp).is_global) then                     AsmWriteLn('.globl .' + tai_symbol(hp).sym.name);                   AsmWriteLn('.type .' + tai_symbol(hp).sym.name + ', @function');                   { the dotted name is the name of the actual function entry }                   AsmWrite('.');                 end               else                 begin                   if (target_info.system <> system_arm_linux) then                     sepChar := '@'                   else                     sepChar := '#';                   if (tf_needs_symbol_type in target_info.flags) then                     begin                       AsmWrite(#9'.type'#9 + tai_symbol(hp).sym.name);                       if (needsObject(tai_symbol(hp))) then                         AsmWriteLn(',' + sepChar + 'object')                       else                         AsmWriteLn(',' + sepChar + 'function');                     end;                 end;               if not(tai_symbol(hp).has_value) then                 AsmWriteLn(tai_symbol(hp).sym.name + ':')               else                 AsmWriteLn(tai_symbol(hp).sym.name + '=' + tostr(tai_symbol(hp).value));             end;{$ifdef arm}           ait_thumb_func:             begin               AsmWriteLn(#9'.thumb_func');             end;{$endif arm}           ait_symbol_end :             begin               if tf_needs_symbol_size in target_info.flags then                begin                  s:=target_asm.labelprefix+'e'+tostr(symendcount);                  inc(symendcount);                  AsmWriteLn(s+':');                  AsmWrite(#9'.size'#9);                  if (target_info.system = system_powerpc64_linux) and (tai_symbol_end(hp).sym.typ = AT_FUNCTION) then                    AsmWrite('.');                  AsmWrite(tai_symbol_end(hp).sym.name);                  AsmWrite(', '+s+' - ');                  if (target_info.system = system_powerpc64_linux) and (tai_symbol_end(hp).sym.typ = AT_FUNCTION) then                     AsmWrite('.');                  AsmWriteLn(tai_symbol_end(hp).sym.name);                end;             end;           ait_instruction :             begin               WriteInstruction(hp);             end;           ait_stab :             begin               if assigned(tai_stab(hp).str) then                 begin                   AsmWrite(#9'.'+stabtypestr[tai_stab(hp).stabtype]+' ');                   AsmWritePChar(tai_stab(hp).str);                   AsmLn;                 end;             end;           ait_force_line,           ait_function_name : ;           ait_cutobject :             begin               if SmartAsm then                begin                { only reset buffer if nothing has changed }                  if AsmSize=AsmStartSize then                   AsmClear                  else                   begin                     AsmClose;                     DoAssemble;                     AsmCreate(tai_cutobject(hp).place);                   end;                { avoid empty files }                  while assigned(hp.next) and (tai(hp.next).typ in [ait_cutobject,ait_section,ait_comment]) do                   begin                     if tai(hp.next).typ=ait_section then                       LastSecType:=tai_section(hp.next).sectype;                     hp:=tai(hp.next);                   end;                  if LastSecType<>sec_none then                    WriteSection(LastSecType,'',secorder_default);                  AsmStartSize:=AsmSize;                end;             end;           ait_marker :             if tai_marker(hp).kind=mark_NoLineInfoStart then               inc(InlineLevel)             else if tai_marker(hp).kind=mark_NoLineInfoEnd then               dec(InlineLevel);           ait_directive :             begin               AsmWrite('.'+directivestr[tai_directive(hp).directive]+' ');               if assigned(tai_directive(hp).name) then                 AsmWrite(tai_directive(hp).name^);               AsmLn;             end;           else             internalerror(2006012201);         end;         hp:=tai(hp.next);       end;    end;    procedure TGNUAssembler.WriteExtraHeader;      begin      end;    procedure TGNUAssembler.WriteInstruction(hp: tai);      begin        InstrWriter.WriteInstruction(hp);      end;    procedure TGNUAssembler.WriteWeakSymbolDef(s: tasmsymbol);      begin        AsmWriteLn(#9'.weak '+s.name);      end;    procedure TGNUAssembler.WriteAsmList;    var      n : string;      hal : tasmlisttype;      i: longint;    begin{$ifdef EXTDEBUG}      if assigned(current_module.mainsource) then       Comment(V_Debug,'Start writing gas-styled assembler output for '+current_module.mainsource^);{$endif}      if assigned(current_module.mainsource) then        n:=ExtractFileName(current_module.mainsource^)      else        n:=InputFileName;      { gcc does not add it either for Darwin (and AIX). Grep for        TARGET_ASM_FILE_START_FILE_DIRECTIVE in gcc/config/*.h      }      if not(target_info.system in systems_darwin) then        AsmWriteLn(#9'.file "'+FixFileName(n)+'"');      WriteExtraHeader;      AsmStartSize:=AsmSize;      symendcount:=0;      for hal:=low(TasmlistType) to high(TasmlistType) do        begin          AsmWriteLn(target_asm.comment+'Begin asmlist '+AsmlistTypeStr[hal]);          writetree(current_asmdata.asmlists[hal]);          AsmWriteLn(target_asm.comment+'End asmlist '+AsmlistTypeStr[hal]);        end;      { add weak symbol markers }      for i:=0 to current_asmdata.asmsymboldict.count-1 do        if (tasmsymbol(current_asmdata.asmsymboldict[i]).bind=AB_WEAK_EXTERNAL) then          writeweaksymboldef(tasmsymbol(current_asmdata.asmsymboldict[i]));      if create_smartlink_sections and         (target_info.system in systems_darwin) then        AsmWriteLn(#9'.subsections_via_symbols');      { "no executable stack" marker for Linux }      if (target_info.system in systems_linux) and         not(cs_executable_stack in current_settings.moduleswitches) then        begin          AsmWriteLn('.section .note.GNU-stack,"",%progbits');        end;      AsmLn;{$ifdef EXTDEBUG}      if assigned(current_module.mainsource) then       Comment(V_Debug,'Done writing gas-styled assembler output for '+current_module.mainsource^);{$endif EXTDEBUG}    end;{****************************************************************************}{                        Apple/GNU Assembler writer                          }{****************************************************************************}    function TAppleGNUAssembler.sectionname(atype:TAsmSectiontype;const aname:string;aorder:TAsmSectionOrder):string;      begin        if (target_info.system in systems_darwin) then          case atype of            sec_bss:              { all bss (lcomm) symbols are automatically put in the right }              { place by using the lcomm assembler directive               }              atype := sec_none;            sec_debug_frame,            sec_eh_frame:              begin                result := '.section __DWARF,__debug_info,regular,debug';                exit;              end;            sec_debug_line:              begin                result := '.section __DWARF,__debug_line,regular,debug';                exit;              end;            sec_debug_info:              begin                result := '.section __DWARF,__debug_info,regular,debug';                exit;              end;            sec_debug_abbrev:               begin                 result := '.section __DWARF,__debug_abbrev,regular,debug';                 exit;               end;            sec_rodata:              begin                result := '.const_data';                exit;              end;            sec_rodata_norel:              begin                result := '.const';                exit;              end;            sec_fpc:              begin                result := '.section __TEXT, .fpc, regular, no_dead_strip';                exit;              end;            sec_code:              begin                if (aname='fpc_geteipasebx') or                   (aname='fpc_geteipasecx') then                  begin                    result:='.section __TEXT,__textcoal_nt,coalesced,pure_instructions'#10'.weak_definition '+aname+                      #10'.private_extern '+aname;                    exit;                  end;              end;            sec_data_nonlazy:              begin                result:='.section __DATA, __nl_symbol_ptr,non_lazy_symbol_pointers';                exit;              end;            sec_data_lazy:              begin                result:='.section __DATA, __la_symbol_ptr,lazy_symbol_pointers';                exit;              end;            sec_init_func:              begin                result:='.section __DATA, __mod_init_func, mod_init_funcs';                exit;              end;            sec_term_func:              begin                result:='.section __DATA, __mod_term_func, mod_term_funcs';                exit;              end;            sec_objc_protocol_ext:              begin                result:='.section __OBJC, __protocol_ext, regular, no_dead_strip';                exit;              end;            sec_objc_class_ext:              begin                result:='.section __OBJC, __class_ext, regular, no_dead_strip';                exit;              end;            sec_objc_property:              begin                result:='.section __OBJC, __property, regular, no_dead_strip';                exit;              end;            sec_objc_image_info:              begin                result:='.section __OBJC, __image_info, regular, no_dead_strip';                exit;              end;            sec_objc_cstring_object:              begin                result:='.section __OBJC, __cstring_object, regular, no_dead_strip';                exit;              end;            sec_objc_sel_fixup:              begin                result:='.section __OBJC, __sel_fixup, regular, no_dead_strip';                exit;              end;            sec_objc_message_refs:              begin                if (target_info.system in systems_objc_nfabi) then                  begin                    result:='.section __DATA, __objc_selrefs, literal_pointers, no_dead_strip';                    exit;                  end;              end;            sec_objc_cls_refs:              begin                if (target_info.system in systems_objc_nfabi) then                  begin                    result:='.section __DATA, __objc_clsrefs, regular, no_dead_strip';                    exit;                  end;              end;            sec_objc_meth_var_names,            sec_objc_class_names:              begin                if (target_info.system in systems_objc_nfabi) then                  begin                    result:='.cstring';                    exit                  end;              end;            sec_objc_inst_meth,            sec_objc_cls_meth,            sec_objc_cat_inst_meth,            sec_objc_cat_cls_meth:              begin                if (target_info.system in systems_objc_nfabi) then                  begin                    result:='.section __DATA, __objc_const';                    exit;                  end;              end;            sec_objc_meta_class,            sec_objc_class:              begin                if (target_info.system in systems_objc_nfabi) then                  begin                    result:='.section __DATA, __objc_data';                    exit;                  end;              end;            sec_objc_sup_refs:              begin                result:='.section __DATA, __objc_superrefs, regular, no_dead_strip';                exit              end;            sec_objc_classlist:              begin                result:='.section __DATA, __objc_classlist, regular, no_dead_strip';                exit              end;            sec_objc_nlclasslist:              begin                result:='.section __DATA, __objc_nlclasslist, regular, no_dead_strip';                exit              end;            sec_objc_catlist:              begin                result:='.section __DATA, __objc_catlist, regular, no_dead_strip';                exit              end;            sec_objc_nlcatlist:              begin                result:='.section __DATA, __objc_nlcatlist, regular, no_dead_strip';                exit              end;            sec_objc_protolist:              begin                result:='.section __DATA, __objc_protolist, coalesced, no_dead_strip';                exit;              end;          end;        result := inherited sectionname(atype,aname,aorder);      end;    procedure TAppleGNUAssembler.WriteWeakSymbolDef(s: tasmsymbol);      begin        AsmWriteLn(#9'.weak_reference '+s.name);      end;{****************************************************************************}{                       a.out/GNU Assembler writer                           }{****************************************************************************}    function TAoutGNUAssembler.sectionname(atype:TAsmSectiontype;const aname:string;aorder:TAsmSectionOrder):string;    const(* Translation table - replace unsupported section types with basic ones. *)        SecXTable: array[TAsmSectionType] of TAsmSectionType = (         sec_none,         sec_code,         sec_data,         sec_data (* sec_rodata *),         sec_data (* sec_rodata_norel *),         sec_bss,         sec_data (* sec_threadvar *),         { used for wince exception handling }         sec_code (* sec_pdata *),         { used for darwin import stubs }         sec_code (* sec_stub *),         sec_data,(* sec_data_nonlazy *)         sec_data,(* sec_data_lazy *)         sec_data,(* sec_init_func *)         sec_data,(* sec_term_func *)         { stabs }         sec_stab,sec_stabstr,         { win32 }         sec_data (* sec_idata2 *),         sec_data (* sec_idata4 *),         sec_data (* sec_idata5 *),         sec_data (* sec_idata6 *),         sec_data (* sec_idata7 *),         sec_data (* sec_edata *),         { C++ exception handling unwinding (uses dwarf) }         sec_eh_frame,         { dwarf }         sec_debug_frame,         sec_debug_info,         sec_debug_line,         sec_debug_abbrev,         { ELF resources (+ references to stabs debug information sections) }         sec_code (* sec_fpc *),         { Table of contents section }         sec_code (* sec_toc *),         sec_code (* sec_init *),         sec_code (* sec_fini *),         sec_none (* sec_objc_class *),         sec_none (* sec_objc_meta_class *),         sec_none (* sec_objc_cat_cls_meth *),         sec_none (* sec_objc_cat_inst_meth *),         sec_none (* sec_objc_protocol *),         sec_none (* sec_objc_string_object *),         sec_none (* sec_objc_cls_meth *),         sec_none (* sec_objc_inst_meth *),         sec_none (* sec_objc_cls_refs *),         sec_none (* sec_objc_message_refs *),         sec_none (* sec_objc_symbols *),         sec_none (* sec_objc_category *),         sec_none (* sec_objc_class_vars *),         sec_none (* sec_objc_instance_vars *),         sec_none (* sec_objc_module_info *),         sec_none (* sec_objc_class_names *),         sec_none (* sec_objc_meth_var_types *),         sec_none (* sec_objc_meth_var_names *),         sec_none (* sec_objc_selector_strs *),         sec_none (* sec_objc_protocol_ext *),         sec_none (* sec_objc_class_ext *),         sec_none (* sec_objc_property *),         sec_none (* sec_objc_image_info *),         sec_none (* sec_objc_cstring_object *),         sec_none (* sec_objc_sel_fixup *),         sec_none (* sec_objc_data *),         sec_none (* sec_objc_const *),         sec_none (* sec_objc_sup_refs *),         sec_none (* sec_data_coalesced *),         sec_none (* sec_objc_classlist *),         sec_none (* sec_objc_nlclasslist *),         sec_none (* sec_objc_catlist *),         sec_none (* sec_objc_nlcatlist *),         sec_none (* sec_objc_protlist *)        );      begin        Result := inherited SectionName (SecXTable [AType], AName, AOrder);      end;{****************************************************************************}{                        Abstract Instruction Writer                         }{****************************************************************************}     constructor TCPUInstrWriter.create(_owner: TGNUAssembler);       begin         inherited create;         owner := _owner;       end;end.
 |