| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612 | {    Copyright (c) 1998-2002 by Florian Klaempfl    This unit implements an asmoutput class for i386 AT&T syntax    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. ****************************************************************************}{ This unit implements an asmoutput class for i386 AT&T syntax}unit agx86att;{$i fpcdefs.inc}interface    uses      cclasses,cpubase,      globals,globtype,cgutils,      aasmbase,aasmtai,aasmdata,assemble,aggas;    type      Tx86ATTAssembler=class(TGNUassembler)        constructor create(smart: boolean); override;        function MakeCmdLine: TCmdStr; override;      end;      Tx86AppleGNUAssembler=class(TAppleGNUassembler)        constructor create(smart: boolean); override;      end;      Tx86AoutGNUAssembler=class(TAoutGNUassembler)        constructor create(smart: boolean); override;      end;     Tx86InstrWriter=class(TCPUInstrWriter)       private        procedure WriteReference(var ref : treference);        procedure WriteOper(const o:toper);        procedure WriteOper_jmp(const o:toper);       protected        fskipPopcountSuffix: boolean;        { http://gcc.gnu.org/bugzilla/show_bug.cgi?id=56656 }        fNoInterUnitMovQ: boolean;       public        procedure WriteInstruction(hp: tai);override;     end;  implementation    uses      cutils,systems,      verbose,      itcpugas,      cgbase,      aasmcpu;{****************************************************************************                            Tx86ATTAssembler ****************************************************************************}    constructor Tx86ATTAssembler.create(smart: boolean);      begin        inherited create(smart);        InstrWriter := Tx86InstrWriter.create(self);      end;    function TX86ATTAssembler.MakeCmdLine: TCmdStr;      var        FormatName : string;      begin        result:=Inherited MakeCmdLine;{$ifdef i386}        case target_info.system of          system_i386_go32v2:            FormatName:='coff';          system_i386_wdosx,          system_i386_win32:            FormatName:='win32';          system_i386_embedded:            FormatName:='obj';          system_i386_linux,          system_i386_beos:            FormatName:='elf';          system_i386_darwin:            FormatName:='macho32';        else          FormatName:='elf';        end;{$endif i386}{$ifdef x86_64}        case target_info.system of          system_x86_64_win64:            FormatName:='win64';          system_x86_64_darwin:            FormatName:='macho64';          system_x86_64_linux:            FormatName:='elf64';        else          FormatName:='elf64';        end;{$endif x86_64}        Replace(result,'$FORMAT',FormatName);      end;{****************************************************************************                          Tx86AppleGNUAssembler ****************************************************************************}    constructor Tx86AppleGNUAssembler.create(smart: boolean);      begin        inherited create(smart);        InstrWriter := Tx86InstrWriter.create(self);        { Apple's assembler does not support a size suffix for popcount }        Tx86InstrWriter(InstrWriter).fskipPopcountSuffix := true;        { Apple's assembler is broken regarding some movq suffix handling }        Tx86InstrWriter(InstrWriter).fNoInterUnitMovQ := true;      end;{****************************************************************************                          Tx86AoutGNUAssembler ****************************************************************************}    constructor Tx86AoutGNUAssembler.create(smart: boolean);      begin        inherited create(smart);        InstrWriter := Tx86InstrWriter.create(self);      end;{****************************************************************************                            Tx86InstrWriter ****************************************************************************}    procedure Tx86InstrWriter.WriteReference(var ref : treference);      begin        with ref do         begin           { do we have a segment prefix ? }           { These are probably not correctly handled under GAS }           { should be replaced by coding the segment override  }           { directly! - DJGPP FAQ                              }           if segment<>NR_NO then             owner.AsmWrite(gas_regname(segment)+':');           if assigned(symbol) then             owner.AsmWrite(symbol.name);           if assigned(relsymbol) then             owner.AsmWrite('-'+relsymbol.name);           if ref.refaddr=addr_pic then{$ifdef x86_64}             begin               { local symbols don't have to (and in case of Mac OS X: cannot)                 be accessed via the GOT               }               if not assigned(ref.symbol) or                  (ref.symbol.bind<>AB_LOCAL) then                 owner.AsmWrite('@GOTPCREL');             end;{$else x86_64}             owner.AsmWrite('@GOT');{$endif x86_64}           if offset<0 then             owner.AsmWrite(tostr(offset))           else            if (offset>0) then             begin               if assigned(symbol) then                owner.AsmWrite('+'+tostr(offset))               else                owner.AsmWrite(tostr(offset));             end           else if (index=NR_NO) and (base=NR_NO) and (not assigned(symbol)) then             owner.AsmWrite('0');           if (index<>NR_NO) and (base=NR_NO) then            begin              if scalefactor in [0,1] then                { Switching index to base position gives shorter                  assembler instructions }                begin                  owner.AsmWrite('('+gas_regname(index)+')');                end              else                begin                  owner.AsmWrite('(,'+gas_regname(index));                  if scalefactor<>0 then                   owner.AsmWrite(','+tostr(scalefactor)+')')                  else                   owner.AsmWrite(')');                end;            end           else            if (index=NR_NO) and (base<>NR_NO) then              owner.AsmWrite('('+gas_regname(base)+')')            else             if (index<>NR_NO) and (base<>NR_NO) then              begin                owner.AsmWrite('('+gas_regname(base)+','+gas_regname(index));                if scalefactor<>0 then                 owner.AsmWrite(','+tostr(scalefactor));                owner.AsmWrite(')');              end;         end;      end;    procedure Tx86InstrWriter.WriteOper(const o:toper);      begin        case o.typ of          top_reg :            owner.AsmWrite(gas_regname(o.reg));          top_ref :            if o.ref^.refaddr in [addr_no,addr_pic,addr_pic_no_got] then              WriteReference(o.ref^)            else              begin                owner.AsmWrite('$');                if assigned(o.ref^.symbol) then                 owner.AsmWrite(o.ref^.symbol.name);                if o.ref^.offset>0 then                 owner.AsmWrite('+'+tostr(o.ref^.offset))                else                 if o.ref^.offset<0 then                  owner.AsmWrite(tostr(o.ref^.offset))                else                 if not(assigned(o.ref^.symbol)) then                   owner.AsmWrite('0');              end;          top_const :              owner.AsmWrite('$'+tostr(o.val));          else            internalerror(10001);        end;      end;    procedure Tx86InstrWriter.WriteOper_jmp(const o:toper);      begin        case o.typ of          top_reg :            owner.AsmWrite('*'+gas_regname(o.reg));          top_ref :            begin              if o.ref^.refaddr in [addr_no,addr_pic_no_got] then                begin                  owner.AsmWrite('*');                  WriteReference(o.ref^);                end              else                begin                  owner.AsmWrite(o.ref^.symbol.name);                  if o.ref^.refaddr=addr_pic then                    owner.AsmWrite('@PLT');                  if o.ref^.offset>0 then                   owner.AsmWrite('+'+tostr(o.ref^.offset))                  else                   if o.ref^.offset<0 then                    owner.AsmWrite(tostr(o.ref^.offset));                end;            end;          top_const :            owner.AsmWrite(tostr(o.val));          else            internalerror(10001);        end;      end;    procedure Tx86InstrWriter.WriteInstruction(hp: tai);      var       op       : tasmop;{$ifdef x86_64}       val      : aint;{$endif}       calljmp  : boolean;       need_second_mov : boolean;       i        : integer;       sreg     : string;      begin        if hp.typ <> ait_instruction then          exit;        taicpu(hp).SetOperandOrder(op_att);        op:=taicpu(hp).opcode;        calljmp:=is_calljmp(op);        { constant values in the 32 bit range are sign-extended to          64 bits, but this is not what we want.  PM 2010-09-02          the fix consists of simply setting only the 4-byte register          as the upper 4-bytes will be zeroed at the same time. }        need_second_mov:=false;        // BUGFIX GAS-assembler        // Intel "Intel 64 and IA-32 Architectures Software Developers manual 12/2011        // Intel:       VCVTDQ2PD  YMMREG, YMMREG/mem128 ((intel syntax))        // GAS:         VCVTDQ2PD  YMMREG, XMMREG/mem128 ((intel syntax))        if (op = A_VCVTDQ2PD) and           (taicpu(hp).ops = 2) and           (taicpu(hp).oper[0]^.typ = top_reg) and           (taicpu(hp).oper[1]^.typ = top_reg) then        begin          if ((taicpu(hp).oper[0]^.ot and OT_YMMREG) = OT_YMMREG) and             ((taicpu(hp).oper[1]^.ot and OT_YMMREG) = OT_YMMREG) then          begin            // change registertype in oper[0] from OT_YMMREG to OT_XMMREG            taicpu(hp).oper[0]^.ot := taicpu(hp).oper[0]^.ot and not(OT_YMMREG) or OT_XMMREG;            sreg := gas_regname(taicpu(hp).oper[0]^.reg);            if (copy(sreg, 1, 2) = '%y') or               (copy(sreg, 1, 2) = '%Y') then              taicpu(hp).oper[0]^.reg := gas_regnum_search('%x' + copy(sreg, 3, length(sreg) - 2));          end;        end;{$ifdef x86_64}        if (op=A_MOV) and (taicpu(hp).opsize=S_Q) and           (taicpu(hp).oper[0]^.typ = top_const) then           begin             val := taicpu(hp).oper[0]^.val;             if (val > int64($7fffffff)) and (val < int64($100000000)) then               begin                 owner.AsmWrite(target_asm.comment);                 owner.AsmWritePChar('Fix for Win64-GAS bug');                 owner.AsmLn;                 taicpu(hp).opsize:=S_L;                 if taicpu(hp).oper[1]^.typ = top_reg then                   setsubreg(taicpu(hp).oper[1]^.reg,R_SUBD)                 else if taicpu(hp).oper[1]^.typ = top_ref then                   need_second_mov:=true                 else                   internalerror(20100902);               end;           end;{$endif x86_64}        { see fNoInterUnitMovQ declaration comment }        if fNoInterUnitMovQ then          begin            if ((op=A_MOVQ) or                (op=A_VMOVQ)) and               (((taicpu(hp).oper[0]^.typ=top_reg) and                 (getregtype(taicpu(hp).oper[0]^.reg)=R_INTREGISTER)) or                ((taicpu(hp).oper[1]^.typ=top_reg) and                 (getregtype(taicpu(hp).oper[1]^.reg)=R_INTREGISTER))) then              begin                if op=A_MOVQ then                  op:=A_MOVD                else                  op:=A_VMOVD;                taicpu(hp).opcode:=op;              end;          end;        owner.AsmWrite(#9);        { movsd should not be translated to movsl when there          are (xmm) arguments }        if (op=A_MOVSD) and (taicpu(hp).ops>0) then          owner.AsmWrite('movsd')        else          owner.AsmWrite(gas_op2str[op]);        owner.AsmWrite(cond2str[taicpu(hp).condition]);        { suffix needed ?  fnstsw,fldcw don't support suffixes          with binutils 2.9.5 under linux }{        if (Taicpu(hp).oper[0]^.typ=top_reg) and            (Taicpu(hp).oper[0]^.reg.enum>lastreg) then          internalerror(200301081);}        if (not calljmp) and           (gas_needsuffix[op]<>AttSufNONE) and           (op<>A_FNSTSW) and           (op<>A_FSTSW) and           (op<>A_FNSTCW) and           (op<>A_FSTCW) and           (op<>A_FLDCW) and           (not fskipPopcountSuffix or            (op<>A_POPCNT)) and           not(               (taicpu(hp).ops<>0) and               (taicpu(hp).oper[0]^.typ=top_reg) and               (getregtype(taicpu(hp).oper[0]^.reg)=R_FPUREGISTER)              ) then        begin          if gas_needsuffix[op] = AttSufMM then          begin            for i:=0 to taicpu(hp).ops-1 do            begin              if (taicpu(hp).oper[i]^.typ = top_ref) then              begin                case taicpu(hp).oper[i]^.ot and OT_SIZE_MASK of                   OT_BITS32: begin                                owner.AsmWrite(gas_opsize2str[S_L]);                                break;                              end;                   OT_BITS64: begin                                owner.AsmWrite(gas_opsize2str[S_Q]);                                break;                              end;                  OT_BITS128: begin                                owner.AsmWrite(gas_opsize2str[S_XMM]);                                break;                              end;                  OT_BITS256: begin                                owner.AsmWrite(gas_opsize2str[S_YMM]);                                break;                              end;                           0: begin                                owner.AsmWrite(gas_opsize2str[taicpu(hp).opsize]);                                break;                              end;                end;              end;            end;          end          else owner.AsmWrite(gas_opsize2str[taicpu(hp).opsize]);        end;        { process operands }        if taicpu(hp).ops<>0 then          begin            if calljmp then             begin               owner.AsmWrite(#9);               WriteOper_jmp(taicpu(hp).oper[0]^);             end            else             begin               for i:=0 to taicpu(hp).ops-1 do                 begin                   if i=0 then                     owner.AsmWrite(#9)                   else                     owner.AsmWrite(',');                   WriteOper(taicpu(hp).oper[i]^);                 end;             end;          end;        owner.AsmLn;        if need_second_mov then          begin            taicpu(hp).oper[0]^.val:=0;            inc(taicpu(hp).oper[1]^.ref^.offset,4);            WriteInstruction(hp);          end;      end;{*****************************************************************************                                  Initialize*****************************************************************************}    const{$ifdef x86_64}       as_x86_64_as_info : tasminfo =          (            id     : as_gas;            idtxt  : 'AS';            asmbin : 'as';            asmcmd : '--64 -o $OBJ $EXTRAOPT $ASM';            supported_targets : [system_x86_64_linux,system_x86_64_freebsd,                                 system_x86_64_win64,system_x86_64_embedded,                                 system_x86_64_openbsd,system_x86_64_netbsd,                                 system_x86_64_dragonfly];            flags : [af_needar,af_smartlink_sections,af_supports_dwarf];            labelprefix : '.L';            comment : '# ';            dollarsign: '$';          );       as_x86_64_yasm_info : tasminfo =          (            id     : as_yasm;            idtxt  : 'YASM';            asmbin : 'yasm';            asmcmd : '-a x86 -p gas -f $FORMAT -o $OBJ $EXTRAOPT $ASM';            supported_targets : [system_x86_64_linux,system_x86_64_freebsd,system_x86_64_win64,system_x86_64_embedded];            flags : [af_needar,af_smartlink_sections,af_supports_dwarf];            labelprefix : '.L';            comment : '# ';            dollarsign: '$';          );       as_x86_64_gas_info : tasminfo =          (            id     : as_ggas;            idtxt  : 'GAS';            asmbin : 'gas';            asmcmd : '--64 -o $OBJ $EXTRAOPT $ASM';            supported_targets : [system_x86_64_solaris];            flags : [af_needar,af_smartlink_sections,af_supports_dwarf];            labelprefix : '.L';            comment : '# ';            dollarsign: '$';          );       as_x86_64_gas_darwin_info : tasminfo =          (            id     : as_darwin;            idtxt  : 'AS-Darwin';            asmbin : 'as';            asmcmd : '-o $OBJ $EXTRAOPT $ASM -arch x86_64';            supported_targets : [system_x86_64_darwin];            flags : [af_needar,af_smartlink_sections,af_supports_dwarf];            labelprefix : 'L';            comment : '# ';            dollarsign: '$';          );{$else x86_64}       as_i386_as_info : tasminfo =          (            id     : as_gas;            idtxt  : 'AS';            asmbin : 'as';            asmcmd : '--32 -o $OBJ $EXTRAOPT $ASM';            supported_targets : [system_i386_GO32V2,system_i386_linux,system_i386_Win32,system_i386_freebsd,system_i386_solaris,system_i386_beos,                                system_i386_netbsd,system_i386_Netware,system_i386_qnx,system_i386_wdosx,system_i386_openbsd,                                system_i386_netwlibc,system_i386_wince,system_i386_embedded,system_i386_symbian,system_i386_haiku,system_x86_6432_linux,                                system_i386_nativent,system_i386_android,system_i386_aros];            flags : [af_needar,af_smartlink_sections,af_supports_dwarf];            labelprefix : '.L';            comment : '# ';            dollarsign: '$';          );       as_i386_yasm_info : tasminfo =          (            id     : as_yasm;            idtxt  : 'YASM';            asmbin : 'yasm';            asmcmd : '-a x86 -p gas -f $FORMAT -o $OBJ $EXTRAOPT $ASM';            supported_targets : [system_i386_GO32V2,system_i386_linux,system_i386_Win32,system_i386_freebsd,system_i386_solaris,system_i386_beos,                                system_i386_netbsd,system_i386_Netware,system_i386_qnx,system_i386_wdosx,system_i386_openbsd,                                system_i386_netwlibc,system_i386_wince,system_i386_embedded,system_i386_symbian,system_i386_haiku,system_x86_6432_linux,                                system_i386_nativent];            flags : [af_needar,af_smartlink_sections,af_supports_dwarf];            labelprefix : '.L';            comment : '# ';            dollarsign: '$';          );       as_i386_as_aout_info : tasminfo =          (            id           : as_i386_as_aout;            idtxt  : 'AS_AOUT';            asmbin : 'as';            asmcmd : '-o $OBJ $EXTRAOPT $ASM';            supported_targets : [system_i386_linux,system_i386_OS2,system_i386_freebsd,system_i386_netbsd,system_i386_openbsd,system_i386_EMX,system_i386_embedded];            flags : [af_needar,af_stabs_use_function_absolute_addresses];            labelprefix : 'L';            comment : '# ';            dollarsign: '$';          );       as_i386_gas_darwin_info : tasminfo =          (            id     : as_darwin;            idtxt  : 'AS-Darwin';            asmbin : 'as';            asmcmd : '-o $OBJ $EXTRAOPT $ASM -arch i386';            supported_targets : [system_i386_darwin,system_i386_iphonesim];            flags : [af_needar,af_smartlink_sections,af_supports_dwarf,af_stabs_use_function_absolute_addresses];            labelprefix : 'L';            comment : '# ';            dollarsign: '$';          );       as_i386_gas_info : tasminfo =          (            id     : as_ggas;            idtxt  : 'GAS';            asmbin : 'gas';            asmcmd : '--32 -o $OBJ $EXTRAOPT $ASM';            supported_targets : [system_i386_GO32V2,system_i386_linux,system_i386_Win32,system_i386_freebsd,system_i386_solaris,system_i386_beos,                                system_i386_netbsd,system_i386_Netware,system_i386_qnx,system_i386_wdosx,system_i386_openbsd,                                system_i386_netwlibc,system_i386_wince,system_i386_embedded,system_i386_symbian,system_i386_haiku,                                system_x86_6432_linux,system_i386_android];            flags : [af_needar,af_smartlink_sections,af_supports_dwarf];            labelprefix : '.L';            comment : '# ';            dollarsign: '$';          );{$endif x86_64}initialization{$ifdef x86_64}  RegisterAssembler(as_x86_64_as_info,Tx86ATTAssembler);  RegisterAssembler(as_x86_64_yasm_info,Tx86ATTAssembler);  RegisterAssembler(as_x86_64_gas_info,Tx86ATTAssembler);  RegisterAssembler(as_x86_64_gas_darwin_info,Tx86AppleGNUAssembler);{$else x86_64}  RegisterAssembler(as_i386_as_info,Tx86ATTAssembler);  RegisterAssembler(as_i386_gas_info,Tx86ATTAssembler);  RegisterAssembler(as_i386_yasm_info,Tx86ATTAssembler);  RegisterAssembler(as_i386_gas_darwin_info,Tx86AppleGNUAssembler);  RegisterAssembler(as_i386_as_aout_info,Tx86AoutGNUAssembler);{$endif x86_64}end.
 |