| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216 | {    Copyright (c) 2003 by Florian Klaempfl    This unit implements an asm for the ARM    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 the GNU Assembler writer for the ARM}unit agavrgas;{$i fpcdefs.inc}  interface    uses       globtype,       aasmtai,aasmdata,       aggas,       cpubase;    type      { TAVRGNUAssembler }      TAVRGNUAssembler=class(TGNUassembler)        constructor create(smart: boolean); override;       function MakeCmdLine: TCmdStr; override;      end;     TAVRInstrWriter=class(TCPUInstrWriter)        procedure WriteInstruction(hp : tai);override;     end;  implementation    uses       cutils,globals,verbose,       systems,       assemble,       aasmcpu,       itcpugas,       cpuinfo,       cgbase,cgutils;{****************************************************************************}{                         GNU Arm Assembler writer                           }{****************************************************************************}    constructor TAVRGNUAssembler.create(smart: boolean);      begin        inherited create(smart);        InstrWriter := TAVRInstrWriter.create(self);      end;{****************************************************************************}{                  Helper routines for Instruction Writer                    }{****************************************************************************}    Procedure TAVRInstrWriter.WriteInstruction(hp : tai);      function getreferencestring(var ref : treference) : string;        var          s : string;        begin           with ref do            begin  {$ifdef extdebug}              // if base=NR_NO then              //   internalerror(200308292);              // if ((index<>NR_NO) or (shiftmode<>SM_None)) and ((offset<>0) or (symbol<>nil)) then              //   internalerror(200308293);  {$endif extdebug}              if index<>NR_NO then                internalerror(2011021701)              else if base<>NR_NO then                begin                  if addressmode=AM_PREDRECEMENT then                    s:='-'                  else                    s:='';                  case base of                    NR_R26:                      s:=s+'X';                    NR_R28:                      s:=s+'Y';                    NR_R30:                      s:=s+'Z';                    else                      s:=gas_regname(base);                  end;                  if addressmode=AM_POSTINCREMENT then                    s:=s+'+';                  if offset>0 then                    s:=s+'+'+tostr(offset)                  else if offset<0 then                    s:=s+tostr(offset)                end              else if assigned(symbol) or (offset<>0) then                begin                  if assigned(symbol) then                    s:=ReplaceForbiddenChars(symbol.name)                  else                     s:='';                  if offset<0 then                    s:=s+tostr(offset)                  else if offset>0 then                    s:=s+'+'+tostr(offset);                  case refaddr of                    addr_hi8:                      s:='hi8('+s+')';                    addr_lo8:                      s:='lo8('+s+')';                    else                      s:='('+s+')';                  end;                end;            end;          getreferencestring:=s;        end;      function getopstr(const o:toper) : string;        var          hs : string;          first : boolean;          r : tsuperregister;        begin          case o.typ of            top_reg:              getopstr:=gas_regname(o.reg);            top_const:              getopstr:=tostr(longint(o.val));            top_ref:              if o.ref^.refaddr=addr_full then                begin                  hs:=ReplaceForbiddenChars(o.ref^.symbol.name);                  if o.ref^.offset>0 then                   hs:=hs+'+'+tostr(o.ref^.offset)                  else                   if o.ref^.offset<0 then                    hs:=hs+tostr(o.ref^.offset);                  getopstr:=hs;                end              else                getopstr:=getreferencestring(o.ref^);            else              internalerror(2002070604);          end;        end;    var op: TAsmOp;        s: string;        i: byte;        sep: string[3];    begin      op:=taicpu(hp).opcode;      s:=#9+gas_op2str[op]+cond2str[taicpu(hp).condition];      if taicpu(hp).ops<>0 then        begin          sep:=#9;          for i:=0 to taicpu(hp).ops-1 do            begin              s:=s+sep+getopstr(taicpu(hp).oper[i]^);              sep:=',';            end;        end;      owner.AsmWriteLn(s);    end;    function TAVRGNUAssembler.MakeCmdLine: TCmdStr;      begin        result := '-mmcu='+lower(cputypestr[current_settings.cputype])+' '+inherited MakeCmdLine;      end;    const       as_arm_gas_info : tasminfo =          (            id     : as_gas;            idtxt  : 'AS';            asmbin : 'as';            asmcmd : '-o $OBJ $ASM';            supported_targets : [system_avr_embedded];            flags : [af_allowdirect,af_needar,af_smartlink_sections];            labelprefix : '.L';            comment : '# ';          );begin  RegisterAssembler(as_arm_gas_info,TAVRGNUAssembler);end.
 |