| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282 | {    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,cgutils,      aasmbase,aasmtai,assemble,aggas;    type      Tx86ATTAssembler=class(TGNUassembler)      private        procedure WriteReference(var ref : treference);        procedure WriteOper(const o:toper);        procedure WriteOper_jmp(const o:toper);      public        procedure WriteInstruction(hp: tai);override;      end;  implementation    uses      cutils,systems,      verbose,      itcpugas,      cgbase,      aasmcpu;{****************************************************************************                            TX86ATTASMOUTPUT ****************************************************************************}    procedure Tx86AttAssembler.WriteReference(var ref : treference);      begin        with ref do         begin           { have we 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             AsmWrite(gas_regname(segment)+':');           if assigned(symbol) then             AsmWrite(symbol.name);           if ref.refaddr=addr_pic then             AsmWrite('@GOTPCREL');           if offset<0 then             AsmWrite(tostr(offset))           else            if (offset>0) then             begin               if assigned(symbol) then                AsmWrite('+'+tostr(offset))               else                AsmWrite(tostr(offset));             end           else if (index=NR_NO) and (base=NR_NO) and (not assigned(symbol)) then             AsmWrite('0');           if (index<>NR_NO) and (base=NR_NO) then            begin              AsmWrite('(,'+gas_regname(index));              if scalefactor<>0 then               AsmWrite(','+tostr(scalefactor)+')')              else               AsmWrite(')');            end           else            if (index=NR_NO) and (base<>NR_NO) then              AsmWrite('('+gas_regname(base)+')')            else             if (index<>NR_NO) and (base<>NR_NO) then              begin                AsmWrite('('+gas_regname(base)+','+gas_regname(index));                if scalefactor<>0 then                 AsmWrite(','+tostr(scalefactor));                AsmWrite(')');              end;         end;      end;    procedure Tx86AttAssembler.WriteOper(const o:toper);      begin        case o.typ of          top_reg :            AsmWrite(gas_regname(o.reg));          top_ref :            if o.ref^.refaddr in [addr_no,addr_pic] then              WriteReference(o.ref^)            else              begin                AsmWrite('$');                if assigned(o.ref^.symbol) then                 AsmWrite(o.ref^.symbol.name);                if o.ref^.offset>0 then                 AsmWrite('+'+tostr(o.ref^.offset))                else                 if o.ref^.offset<0 then                  AsmWrite(tostr(o.ref^.offset))                else                 if not(assigned(o.ref^.symbol)) then                   AsmWrite('0');              end;          top_const :              AsmWrite('$'+tostr(o.val));          else            internalerror(10001);        end;      end;    procedure Tx86AttAssembler.WriteOper_jmp(const o:toper);      begin        case o.typ of          top_reg :            AsmWrite('*'+gas_regname(o.reg));          top_ref :            begin              if o.ref^.refaddr=addr_no then                begin                  AsmWrite('*');                  WriteReference(o.ref^);                end              else                begin                  AsmWrite(o.ref^.symbol.name);                  if o.ref^.refaddr=addr_pic then                    AsmWrite('@PLT');                  if o.ref^.offset>0 then                   AsmWrite('+'+tostr(o.ref^.offset))                  else                   if o.ref^.offset<0 then                    AsmWrite(tostr(o.ref^.offset));                end;            end;          top_const :            AsmWrite(tostr(o.val));          else            internalerror(10001);        end;      end;    procedure Tx86AttAssembler.WriteInstruction(hp: tai);      var       op       : tasmop;       calljmp  : boolean;       i        : integer;      begin        if hp.typ <> ait_instruction then          exit;        taicpu(hp).SetOperandOrder(op_att);        op:=taicpu(hp).opcode;        calljmp:=is_calljmp(op);        AsmWrite(#9);        { movsd should not be translated to movsl when there          are (xmm) arguments }        if (op=A_MOVSD) and (taicpu(hp).ops>0) then          AsmWrite('movsd')        else          AsmWrite(gas_op2str[op]);        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(               (taicpu(hp).ops<>0) and               (taicpu(hp).oper[0]^.typ=top_reg) and               (getregtype(taicpu(hp).oper[0]^.reg)=R_FPUREGISTER)              ) then          AsmWrite(gas_opsize2str[taicpu(hp).opsize]);        { process operands }        if taicpu(hp).ops<>0 then          begin            if calljmp then             begin               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                     AsmWrite(#9)                   else                     AsmWrite(',');                   WriteOper(taicpu(hp).oper[i]^);                 end;             end;          end;        AsmLn;      end;{*****************************************************************************                                  Initialize*****************************************************************************}    const{$ifdef x86_64}       as_x86_64_as_info : tasminfo =          (            id     : as_gas;            idtxt  : 'AS';            asmbin : 'as';            asmcmd : '--64 -o $OBJ $ASM';            supported_target : system_any;            flags : [af_allowdirect,af_needar,af_smartlink_sections,af_supports_dwarf];            labelprefix : '.L';            comment : '# ';          );{$else x86_64}       as_i386_as_info : tasminfo =          (            id     : as_gas;            idtxt  : 'AS';            asmbin : 'as';            asmcmd : '--32 -o $OBJ $ASM';            supported_target : system_any;            flags : [af_allowdirect,af_needar,af_smartlink_sections];            labelprefix : '.L';            comment : '# ';          );       as_i386_as_aout_info : tasminfo =          (            id           : as_i386_as_aout;            idtxt  : 'AS_AOUT';            asmbin : 'as';            asmcmd : '-o $OBJ $ASM';            supported_target : system_any;            flags : [af_allowdirect,af_needar];            labelprefix : 'L';            comment : '# ';          );{$endif x86_64}initialization{$ifdef x86_64}  RegisterAssembler(as_x86_64_as_info,Tx86ATTAssembler);{$else x86_64}  RegisterAssembler(as_i386_as_info,Tx86ATTAssembler);  RegisterAssembler(as_i386_as_aout_info,Tx86ATTAssembler);{$endif x86_64}end.
 |