| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476 | {    Copyright (c) 1998-2002 by Carl Eric Codere and Peter Vreman    Copyright (c) 2014 by Jonas Maebe    Does the parsing for the AArch64 GNU AS styled inline assembler.    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. ****************************************************************************}Unit racpugas;{$i fpcdefs.inc}  Interface    uses      raatt,racpu,      aasmtai,      cgbase,cpubase;    type      { taarch64attreader }      taarch64attreader = class(tattreader)        actoppostfix : TOpPostfix;        actinsmmsubreg : TSubRegister;        actsehdirective : TAsmSehDirective;        function is_asmopcode(const s: string):boolean;override;        function is_register(const s:string):boolean;override;        function is_targetdirective(const s: string): boolean;override;        procedure handleopcode;override;        procedure handletargetdirective; override;       protected        procedure BuildReference(oper: taarch64operand; is64bit: boolean);        procedure BuildOperand(oper: taarch64operand; is64bit: boolean);        function TryBuildShifterOp(instr: taarch64instruction; opnr: longint) : boolean;        procedure BuildOpCode(instr: taarch64instruction);        procedure ReadSym(oper: taarch64operand; is64bit: boolean);        procedure ConvertCalljmp(instr: taarch64instruction);        function ToConditionCode(const hs: string; is_operand: boolean): tasmcond;        function ParseArrangementSpecifier(const hs: string): TSubRegister;        function ParseRegIndex(const hs: string): byte;      end;  Implementation    uses      { helpers }      cutils,      { global }      globtype,verbose,      systems,aasmbase,aasmdata,aasmcpu,      { symtable }      symconst,symsym,symdef,      procinfo,      rabase,rautils,      cgutils,paramgr;    function taarch64attreader.is_register(const s:string):boolean;      type        treg2str = record          name : string[3];          reg : tregister;        end;      const        extraregs : array[0..4] of treg2str = (          (name: 'FP' ; reg: NR_FP),          (name: 'LR' ; reg: NR_LR),          (name: 'XR' ; reg: NR_XR),          (name: 'IP0'; reg: NR_IP0),          (name: 'IP1'; reg: NR_IP1));      var        i : longint;      begin        result:=inherited is_register(s);        { reg found?          possible aliases are always 2 chars        }        if result or not(length(s) in [2]) then          exit;        for i:=low(extraregs) to high(extraregs) do          begin            if s=extraregs[i].name then              begin                actasmregister:=extraregs[i].reg;                result:=true;                actasmtoken:=AS_REGISTER;                exit;              end;          end;      end;    const      { Aarch64 subset of SEH directives. .seh_proc, .seh_endproc and .seh_endepilogue        excluded because they are generated automatically when needed. }      recognized_directives: set of TAsmSehDirective=[        ash_endprologue,ash_handler,ash_handlerdata,        ash_stackalloc,ash_nop,ash_savefplr,ash_savefplr_x,        ash_savereg,ash_savereg_x,ash_saveregp,ash_saveregp_x,        ash_savefreg,ash_savefreg_x,ash_savefregp,ash_savefregp_x,        ash_setfp,ash_addfp      ];    function taarch64attreader.is_targetdirective(const s: string): boolean;      var        i: TAsmSehDirective;      begin        result:=false;        if target_info.system<>system_aarch64_win64 then          exit;        for i:=low(TAsmSehDirective) to high(TAsmSehDirective) do          begin            if not (i in recognized_directives) then              continue;            if s=sehdirectivestr[i] then              begin                actsehdirective:=i;                result:=true;                break;              end;          end;        { allow SEH directives only in pure assember routines }        if result and not (po_assembler in current_procinfo.procdef.procoptions) then          begin            Message(asmr_e_seh_in_pure_asm_only);            result:=false;          end;      end;    procedure taarch64attreader.ReadSym(oper: taarch64operand; is64bit: boolean);      var         tempstr, mangledname : string;         typesize,l,k: aint;      begin        tempstr:=actasmpattern;        Consume(AS_ID);        { typecasting? }        if (actasmtoken=AS_LPAREN) and           SearchType(tempstr,typesize) then          begin            oper.hastype:=true;            Consume(AS_LPAREN);            BuildOperand(oper,is64bit);            Consume(AS_RPAREN);            if oper.opr.typ in [OPR_REFERENCE,OPR_LOCAL] then              oper.SetSize(typesize,true);          end        else          if not oper.SetupVar(tempstr,false) then            Message1(sym_e_unknown_id,tempstr);        { record.field ? }        if actasmtoken=AS_DOT then          begin            BuildRecordOffsetSize(tempstr,l,k,mangledname,false);            if (mangledname<>'') then              Message(asmr_e_invalid_reference_syntax);            inc(oper.opr.ref.offset,l);          end;      end;    Procedure taarch64attreader.BuildReference(oper: taarch64operand; is64bit: boolean);      procedure do_error;        begin          Message(asmr_e_invalid_reference_syntax);          RecoverConsume(false);        end;      procedure test_end(require_rbracket : boolean);        begin          if require_rbracket then begin            if not(actasmtoken=AS_RBRACKET) then              begin                do_error;                exit;              end            else              Consume(AS_RBRACKET);            if (actasmtoken=AS_NOT) then              begin                oper.opr.ref.addressmode:=AM_PREINDEXED;                Consume(AS_NOT);              end;          end;          if not(actasmtoken in [AS_SEPARATOR,AS_end]) then            do_error          else            begin{$IFDEF debugasmreader}              writeln('TEST_end_FINAL_OK. Created the following ref:');              writeln('oper.opr.ref.shiftimm=',oper.opr.ref.shiftimm);              writeln('oper.opr.ref.shiftmode=',ord(oper.opr.ref.shiftmode));              writeln('oper.opr.ref.index=',ord(oper.opr.ref.index));              writeln('oper.opr.ref.base=',ord(oper.opr.ref.base));              writeln('oper.opr.ref.signindex=',ord(oper.opr.ref.signindex));              writeln('oper.opr.ref.addressmode=',ord(oper.opr.ref.addressmode));              writeln;{$endIF debugasmreader}            end;        end;      function is_shifter_ref_operation(var a : tshiftmode) : boolean;        begin          a:=SM_NONE;          if (actasmpattern='LSL') then            a:=SM_LSL          else if (actasmpattern='UXTW') then            a:=SM_UXTW          else if (actasmpattern='SXTW') then            a:=SM_SXTW          else if (actasmpattern='SXTX') then            a:=SM_SXTX;          is_shifter_ref_operation:=not(a=SM_NONE);        end;      procedure read_index_shift(require_rbracket : boolean);        var          shift: aint;        begin          case actasmtoken of            AS_COMMA :              begin                Consume(AS_COMMA);                if not(actasmtoken=AS_ID) then                  do_error;                if is_shifter_ref_operation(oper.opr.ref.shiftmode) then                  begin                    Consume(actasmtoken);                    if actasmtoken=AS_HASH then                      begin                        Consume(AS_HASH);                        shift:=BuildConstExpression(false,true);                        if not(shift in [0,2+ord(is64bit)]) then                          do_error;                        oper.opr.ref.shiftimm:=shift;                        test_end(require_rbracket);                      end;                   end                 else                   begin                     do_error;                     exit;                   end;              end;            AS_RBRACKET :              if require_rbracket then                test_end(require_rbracket)              else                begin                  do_error;                  exit;                end;            AS_SEPARATOR,AS_END :              if not require_rbracket then                test_end(false)               else                 do_error;            else              begin                do_error;                exit;              end;          end;        end;      procedure read_index(require_rbracket : boolean);        var          recname : string;          o_int,s_int : aint;        begin          case actasmtoken of            AS_REGISTER :              begin                if getsupreg(actasmregister)=RS_XZR then                  Message1(asmr_e_invalid_ref_register,actasmpattern);                oper.opr.ref.index:=actasmregister;                Consume(AS_REGISTER);                read_index_shift(require_rbracket);                exit;              end;            AS_HASH : // constant              begin                Consume(AS_HASH);(*                if actasmtoken=AS_COLON then                  begin                    consume(AS_COLON);                    { GNU-style lower 12 bits of address of non-GOT-based                      access }                    if (actasmpattern='LO12') then                      begin                        consume(actasmtoken);                        consume(AS_COLON);                        if not oper.SetupVar(actasmpattern,false) then                          begin                            do_error;                            exit                          end;                        consume(AS_ID);                        oper.opr.ref.refaddr:=addr_??? (not gotpageoffset);                      end                    else                      begin                        do_error;                        exit                      end;                  end                else*)                  begin                    o_int:=BuildConstExpression(false,true);                    inc(oper.opr.ref.offset,o_int);                  end;                test_end(require_rbracket);                exit;              end;            AS_ID :              begin                recname:=actasmpattern;                Consume(AS_ID);                { Apple-style got page offset }                if actasmtoken=AS_AT then                  begin                    if not oper.SetupVar(recname,false) then                      begin                        do_error;                        exit                      end;                    consume(AS_AT);                    if actasmpattern='GOTPAGEOFF' then                      begin                        consume(actasmtoken);                        oper.opr.ref.refaddr:=addr_gotpageoffset;                      end                    else if actasmpattern='PAGEOFF' then                      begin                        consume(actasmtoken);                        oper.opr.ref.refaddr:=addr_pageoffset;                      end                    else                      begin                        do_error;                        exit                      end;                  end                else                  begin                    BuildRecordOffsetSize(recname,o_int,s_int,recname,false);                    inc(oper.opr.ref.offset,o_int);                  end;                test_end(require_rbracket);                exit;              end;            AS_AT:              begin                do_error;                exit;              end;            AS_RBRACKET :              begin                if require_rbracket then                  begin                    test_end(require_rbracket);                    exit;                  end                else                  begin                    do_error; // unexpected rbracket                    exit;                  end;              end;            AS_SEPARATOR,AS_end :              begin                if not require_rbracket then                  begin                    test_end(false);                    exit;                  end                else                  begin                    do_error;                    exit;                  end;              end;            else              begin                // unexpected token                do_error;                exit;              end;          end; // case        end;      procedure try_prepostindexed;        begin          Consume(AS_RBRACKET);          case actasmtoken of            AS_COMMA :              begin // post-indexed                Consume(AS_COMMA);                oper.opr.ref.addressmode:=AM_POSTINDEXED;                read_index(false);                exit;              end;            AS_NOT :              begin   // pre-indexed                Consume(AS_NOT);                oper.opr.ref.addressmode:=AM_PREINDEXED;                test_end(false);                exit;              end;            else              begin                test_end(false);                exit;              end;          end; // case        end;      begin        Consume(AS_LBRACKET);        oper.opr.ref.addressmode:=AM_OFFSET; // assume "neither PRE nor POST inc"        if actasmtoken=AS_REGISTER then          begin            if getsupreg(actasmregister)=RS_XZR then              Message1(asmr_e_invalid_ref_register,actasmpattern);            oper.opr.ref.base:=actasmregister;            Consume(AS_REGISTER);            case actasmtoken of              AS_RBRACKET :                begin                  try_prepostindexed;                  exit;                end;              AS_COMMA :                begin                  Consume(AS_COMMA);                  read_index(true);                  exit;                end;              else                begin                  Message(asmr_e_invalid_reference_syntax);                  RecoverConsume(false);                end;            end;          end        else          Begin            case actasmtoken of              AS_ID :                begin                  { TODO: local variables and parameters }                  Message(asmr_e_invalid_reference_syntax);                  RecoverConsume(false);                  exit;                end;              else                begin // elsecase                  Message(asmr_e_invalid_reference_syntax);                  RecoverConsume(false);                  exit;                end;            end;          end;      end;    function taarch64attreader.TryBuildShifterOp(instr: taarch64instruction; opnr: longint): boolean;      procedure handlepara(sm : tshiftmode);        begin          consume(AS_ID);          fillchar(instr.operands[opnr].opr,sizeof(instr.operands[opnr].opr),0);          instr.operands[opnr].opr.typ:=OPR_SHIFTEROP;          instr.operands[opnr].opr.shifterop.shiftmode:=sm;          if (sm=SM_LSL) or             (actasmtoken=AS_HASH) then            begin              consume(AS_HASH);              instr.operands[opnr].opr.shifterop.shiftimm:=BuildConstExpression(false,false);            end;        end;      const        shiftmode2str: array[SM_LSL..SM_SXTX] of string[4] =          ('LSL','LSR','ASR','ROR',           'UXTB','UXTH','UXTW','UXTX',           'SXTB','SXTH','SXTW','SXTX');      var        sm: tshiftmode;        i: longint;        usessp,        useszr: boolean;      begin        result:=false;        if (actasmtoken=AS_ID) then          begin            for sm:=low(shiftmode2str) to high(shiftmode2str) do              if actasmpattern=shiftmode2str[sm] then                begin                  handlepara(sm);                  if instr.operands[1].opr.typ=OPR_REGISTER then                    begin                      { the possible shifter ops depend on whether this                        instruction uses sp and/or zr }                      usessp:=false;                      useszr:=false;                      for i:=low(instr.operands) to pred(opnr) do                        begin                          if (instr.operands[i].opr.typ=OPR_REGISTER) then                            case getsupreg(instr.operands[i].opr.reg) of                              RS_XZR:                                useszr:=true;                              RS_SP:                                usessp:=true;                            end;                        end;                      result:=valid_shifter_operand(instr.opcode,useszr,usessp,instr.Is64bit,sm,instr.operands[opnr].opr.shifterop.shiftimm);                      if result then                        instr.Ops:=opnr;                    end;                  break;                end;          end;      end;    function taarch64attreader.ToConditionCode(const hs: string; is_operand: boolean): tasmcond;{$push}{$j-}      const        extracond2str: array[C_HS..C_LO] of string[2] = ('CS','CC');{$pop}      begin        case actopcode of          A_CSEL,A_CSINC,A_CSINV,A_CSNEG,A_CSET,A_CSETM,          A_CINC,A_CINV,A_CNEG,A_CCMN,A_CCMP,          A_B:            begin              { search for condition, conditions are always 2 chars }              if (is_operand<>(actopcode=A_B)) and                 (length(hs)>1) then                begin                  { workaround for DFA bug }                  result:=low(tasmcond);                  for result:=low(uppercond2str) to high(uppercond2str) do                    begin                      if hs=uppercond2str[result] then                        exit;                    end;                  for result:=low(extracond2str) to high(extracond2str) do                    begin                      if hs=extracond2str[result] then                        exit;                    end;                end;            end;          else            ;        end;        result:=C_None;      end;    function taarch64attreader.ParseArrangementSpecifier(const hs: string): TSubRegister;{$push}{$j-}      const        arrangements: array[R_SUBMM8B..R_SUBMM2D] of string[4] =          ('.8B','.16B','.4H','.8H','.2S','.4S','.1D','.2D');{$pop}      begin        if length(hs)>2 then          begin            for result:=low(arrangements) to high(arrangements) do              if hs=arrangements[result] then                exit;            result:=R_SUBNONE;          end        else          case hs of            '.B': result:=R_SUBMMB1;            '.H': result:=R_SUBMMH1;            '.S': result:=R_SUBMMS1;            '.D': result:=R_SUBMMD1;            else              result:=R_SUBNONE;          end      end;    function taarch64attreader.ParseRegIndex(const hs: string): byte;      var        b: cardinal;        error: longint;      begin        b:=0;        val(hs,b,error);        if (error<>0) then          Message(asmr_e_syn_constant)        else if b > 31 then          begin            Message(asmr_e_constant_out_of_bounds);            b:=0;          end;        result:=b;      end;    Procedure taarch64attreader.BuildOperand(oper: taarch64operand; is64bit: boolean);      var        expr: string;        typesize, l: aint;        procedure MaybeAddGotAddrMode;          begin            if actasmtoken=AS_AT then              begin                consume(AS_AT);                if actasmpattern='GOTPAGE' then                  oper.opr.ref.refaddr:=addr_gotpage                else if actasmpattern='GOTPAGEOFF' then                  oper.opr.ref.refaddr:=addr_gotpageoffset                else if actasmpattern='PAGE' then                  oper.opr.ref.refaddr:=addr_page                else if actasmpattern='PAGEOFF' then                  oper.opr.ref.refaddr:=addr_pageoffset                else                  Message(asmr_e_expr_illegal);                consume(actasmtoken);              end            else              oper.opr.ref.refaddr:=addr_pic;          end;        procedure AddLabelOperand(hl:tasmlabel);          begin            if not(actasmtoken in [AS_PLUS,AS_MINUS,AS_LPAREN]) and               is_calljmp(actopcode) then             begin               oper.opr.typ:=OPR_SYMBOL;               oper.opr.symbol:=hl;             end            else if (actopcode=A_ADR) or               (actopcode=A_ADRP) or               (actopcode=A_LDR) then              begin                oper.InitRef;                MaybeAddGotAddrMode;                oper.opr.ref.symbol:=hl;                if (actasmtoken in [AS_PLUS, AS_MINUS]) then                  begin                    l:=BuildConstExpression(true,false);                    oper.opr.ref.offset:=l;                  end;              end;          end;        procedure MaybeRecordOffset;          var            mangledname: string;            hasdot  : boolean;            l,            toffset,            tsize   : aint;          begin            if not(actasmtoken in [AS_DOT,AS_PLUS,AS_MINUS]) then              exit;            l:=0;            mangledname:='';            hasdot:=(actasmtoken=AS_DOT);            if hasdot then              begin                if expr<>'' then                  begin                    BuildRecordOffsetSize(expr,toffset,tsize,mangledname,false);                    if (oper.opr.typ<>OPR_CONSTANT) and                       (mangledname<>'') then                      Message(asmr_e_wrong_sym_type);                    inc(l,toffset);                    oper.SetSize(tsize,true);                  end;              end;            if actasmtoken in [AS_PLUS,AS_MINUS] then              inc(l,BuildConstExpression(true,false));            case oper.opr.typ of              OPR_LOCAL :                begin                  { don't allow direct access to fields of parameters, because that                    will generate buggy code. Allow it only for explicit typecasting }                  if hasdot and                     (not oper.hastype) then                    checklocalsubscript(oper.opr.localsym);                  inc(oper.opr.localsymofs,l)                end;              OPR_CONSTANT :                inc(oper.opr.val,l);              OPR_REFERENCE :                if (mangledname<>'') then                  begin                    if (oper.opr.val<>0) then                      Message(asmr_e_wrong_sym_type);                    oper.opr.typ:=OPR_SYMBOL;                    oper.opr.symbol:=current_asmdata.RefAsmSymbol(mangledname,AT_FUNCTION);                  end                else                  inc(oper.opr.val,l);              OPR_SYMBOL:                Message(asmr_e_invalid_symbol_ref);              else                internalerror(200309221);            end;          end;        function MaybeBuildReference(is64bit: boolean):boolean;          { Try to create a reference, if not a reference is found then false            is returned }          begin            MaybeBuildReference:=true;            case actasmtoken of              AS_INTNUM,              AS_MINUS,              AS_PLUS:                Begin                  oper.opr.ref.offset:=BuildConstExpression(True,False);                  if actasmtoken<>AS_LPAREN then                    Message(asmr_e_invalid_reference_syntax)                  else                    BuildReference(oper,is64bit);                end;              AS_LPAREN:                BuildReference(oper,is64bit);              AS_ID: { only a variable is allowed ... }                Begin                  ReadSym(oper,is64bit);                  case actasmtoken of                    AS_end,                    AS_SEPARATOR,                    AS_COMMA: ;                    AS_LPAREN:                      BuildReference(oper,is64bit);                  else                    Begin                      Message(asmr_e_invalid_reference_syntax);                      Consume(actasmtoken);                    end;                  end; {end case }                end;              else               MaybeBuildReference:=false;            end; { end case }          end;      function parsereg: tregister;         var           subreg: tsubregister;        begin          result:=actasmregister;          Consume(AS_REGISTER);          if (actasmtoken=AS_ID) and             (actasmpattern[1]='.') then            begin              subreg:=ParseArrangementSpecifier(upper(actasmpattern));              if (subreg<>R_SUBNONE) and                 (getregtype(result)=R_MMREGISTER) and                 ((actinsmmsubreg=R_SUBNONE) or                  (actinsmmsubreg=subreg)) then                begin                  setsubreg(result,subreg);                  { they all have to be the same }                  actinsmmsubreg:=subreg;                end              else                Message1(asmr_e_invalid_arrangement,actasmpattern);              Consume(AS_ID);            end          else if (getregtype(result)=R_MMREGISTER) then            begin              if actinsmmsubreg<>R_SUBNONE then                begin                  if (getsubreg(result)=R_SUBNONE) or                     (getsubreg(result)=actinsmmsubreg) then                    setsubreg(result,actinsmmsubreg)                  else                     Message1(asmr_e_invalid_arrangement,actasmpattern);                end              else if getsubreg(result)=R_SUBNONE then                { Vxx without an arrangement is invalid, use Qxx to specify the entire 128 bits}                Message1(asmr_e_invalid_arrangement,'');            end;        end;      var        tempreg: tregister;        hl: tasmlabel;        icond: tasmcond;        regindex: byte;      Begin        expr:='';        case actasmtoken of          AS_LBRACKET: { Memory reference or constant expression }            Begin              oper.InitRef;              BuildReference(oper,is64bit);            end;          AS_LSBRACKET: { register set }            begin              consume(AS_LSBRACKET);              oper.opr.typ:=OPR_REGSET;              oper.opr.basereg:=parsereg;              oper.opr.nregs:=1;              while (actasmtoken<>AS_RSBRACKET) and (oper.opr.nregs<4) do                begin                  if actasmtoken=AS_MINUS then                    begin                      consume(AS_MINUS);                      tempreg:=parsereg;                      if (getsupreg(tempreg)-getsupreg(oper.opr.basereg))>=4 then                        Message(asmr_e_a64_regset_too_large);                      oper.opr.nregs:=getsupreg(tempreg)-getsupreg(oper.opr.basereg)+1;                      if actasmtoken<>AS_COMMA then                        break;                    end;                  if actasmtoken=AS_COMMA then                    begin                      consume(AS_COMMA);                      tempreg:=parsereg;                      if (getsupreg(tempreg)-getsupreg(oper.opr.basereg))>=4 then                        Message(asmr_e_a64_regset_too_large);                      if getsupreg(tempreg)<>((getsupreg(oper.opr.basereg)+oper.opr.nregs) mod 32) then                        Message(asmr_e_a64_invalid_regset);                      inc(oper.opr.nregs);                    end                  else                    break;                end;              consume(AS_RSBRACKET);              if actasmtoken=AS_LBRACKET then                begin                  consume(AS_LBRACKET);                  oper.opr.regsetindex:=ParseRegIndex(actasmpattern);                  consume(AS_INTNUM);                  consume(AS_RBRACKET);                end              else                oper.opr.regsetindex:=255;              if not(actasmtoken in [AS_END,AS_SEPARATOR,AS_COMMA]) then                Message(asmr_e_syn_operand);            end;          AS_HASH: { Constant expression  }            Begin              Consume(AS_HASH);              BuildConstantOperand(oper);            end;          (*          AS_INTNUM,          AS_MINUS,          AS_PLUS:            Begin              { Constant memory offset }              { This must absolutely be followed by (  }              oper.InitRef;              oper.opr.ref.offset:=BuildConstExpression(True,False);              if actasmtoken<>AS_LPAREN then                begin                  ofs:=oper.opr.ref.offset;                  BuildConstantOperand(oper);                  inc(oper.opr.val,ofs);                end              else                BuildReference(oper,is64bit);            end;          *)          AS_ID: { A constant expression, or a Variable ref.  }            Begin              { Condition code? }              icond:=ToConditionCode(actasmpattern,true);              if icond<>C_None then                begin                  oper.opr.typ:=OPR_COND;                  oper.opr.cc:=icond;                  consume(AS_ID);                end              else              { Local Label ? }              if is_locallabel(actasmpattern) then               begin                 CreateLocalLabel(actasmpattern,hl,false);                 Consume(AS_ID);                 AddLabelOperand(hl);               end              else               { Check for label }               if SearchLabel(actasmpattern,hl,false) then                 begin                   Consume(AS_ID);                   AddLabelOperand(hl);                 end              else               { probably a variable or normal expression }               { or a procedure (such as in CALL ID)      }               begin                 { is it a constant ? }                 if SearchIConstant(actasmpattern,l) then                  begin                    if not (oper.opr.typ in [OPR_NONE,OPR_CONSTANT]) then                      Message(asmr_e_invalid_operand_type);                    BuildConstantOperand(oper);                  end                 else                  begin                    expr:=actasmpattern;                    Consume(AS_ID);                    { typecasting? }                    if (actasmtoken=AS_LPAREN) and                       SearchType(expr,typesize) then                     begin                       oper.hastype:=true;                       Consume(AS_LPAREN);                       BuildOperand(oper,is64bit);                       Consume(AS_RPAREN);                       if oper.opr.typ in [OPR_REFERENCE,OPR_LOCAL] then                         oper.SetSize(typesize,true);                     end                    else                     begin                       if not(oper.SetupVar(expr,false)) then                        Begin                          { look for special symbols ... }                          if expr= '__HIGH' then                            begin                              consume(AS_LPAREN);                              if not oper.setupvar('high'+actasmpattern,false) then                                Message1(sym_e_unknown_id,'high'+actasmpattern);                              consume(AS_ID);                              consume(AS_RPAREN);                            end                          else                           if expr = '__RESULT' then                            oper.SetUpResult                          else                           if expr = '__SELF' then                            oper.SetupSelf                          else                           if expr = '__OLDEBP' then                            oper.SetupOldEBP                          else                            Message1(sym_e_unknown_id,expr);                        end                       else if oper.opr.typ<>OPR_LOCAL then                         begin                           oper.InitRef;                           MaybeAddGotAddrMode;                         end;                     end;                  end;                  if actasmtoken=AS_DOT then                    MaybeRecordOffset;                  { add a constant expression? }                  if (actasmtoken=AS_PLUS) then                   begin                     l:=BuildConstExpression(true,false);                     case oper.opr.typ of                       OPR_CONSTANT :                         inc(oper.opr.val,l);                       OPR_LOCAL :                         inc(oper.opr.localsymofs,l);                       OPR_REFERENCE :                         inc(oper.opr.ref.offset,l);                       else                         internalerror(2003092005);                     end;                   end               end;              { Do we have a indexing reference, then parse it also }              if actasmtoken=AS_LPAREN then                BuildReference(oper,is64bit);            end;          { Register, a variable reference or a constant reference  }          AS_REGISTER:            Begin              { save the type of register used. }              tempreg:=parsereg;              regindex:=255;              if (getregtype(tempreg)=R_MMREGISTER) and                 (actasmtoken=AS_LBRACKET) then                begin                  consume(AS_LBRACKET);                  regindex:=ParseRegIndex(actasmpattern);                  consume(AS_INTNUM);                  consume(AS_RBRACKET);                end;              if actasmtoken in [AS_END,AS_SEPARATOR,AS_COMMA] then                begin                  if (oper.opr.typ<>OPR_NONE) then                    Message(asmr_e_invalid_operand_type);                  if regindex=255 then                    begin                      oper.opr.typ:=OPR_REGISTER;                      oper.opr.reg:=tempreg;                    end                  else                    begin                      oper.opr.typ:=OPR_INDEXEDREG;                      oper.opr.indexedreg:=tempreg;                      oper.opr.regindex:=regindex;                    end;                end              else                Message(asmr_e_syn_operand);            end;          AS_end,          AS_SEPARATOR,          AS_COMMA: ;        else          Begin            Message(asmr_e_syn_operand);            Consume(actasmtoken);          end;        end; { end case }      end;{*****************************************************************************                                taarch64attreader*****************************************************************************}    procedure taarch64attreader.BuildOpCode(instr: taarch64instruction);      var        operandnum : longint;      Begin        { opcode }        if (actasmtoken<>AS_OPCODE) then         Begin           Message(asmr_e_invalid_or_missing_opcode);           RecoverConsume(true);           exit;         end;        { Fill the instr object with the current state }        with instr do          begin            Opcode:=ActOpcode;            condition:=ActCondition;            oppostfix:=actoppostfix;          end;        Consume(AS_OPCODE);        { We are reading operands, so opcode will be an AS_ID }        operandnum:=1;        { Zero operand opcode ?  }        if actasmtoken in [AS_SEPARATOR,AS_end] then         begin           instr.Ops:=0;           exit;         end;        { Read the operands }        repeat          case actasmtoken of            AS_COMMA: { Operand delimiter }              Begin                { operandnum and not operandnum+1, because tinstruction is                  one-based and taicpu is zero-based)                }                if can_be_shifter_operand(instr.opcode,operandnum) then                  begin                    Consume(AS_COMMA);                    if not TryBuildShifterOp(instr,operandnum+1) then                      Message(asmr_e_illegal_shifterop_syntax);                    Inc(operandnum);                  end                else                  begin                    if operandnum>Max_Operands then                      Message(asmr_e_too_many_operands)                    else                      Inc(operandnum);                    Consume(AS_COMMA);                  end;              end;            AS_SEPARATOR,            AS_end : { End of asm operands for this opcode  }              begin                break;              end;          else            begin              BuildOperand(taarch64operand(instr.operands[operandnum]),instr.Is64bit);              instr.Ops:=operandnum;              if instr.operands[operandnum].opr.typ=OPR_REFERENCE then                if simple_ref_type(instr.opcode,instr.cgsize,instr.oppostfix,instr.operands[operandnum].opr.ref)<>sr_simple then                  Message(asmr_e_invalid_reference_syntax);                ;            end;          end; { end case }        until false;      end;    function taarch64attreader.is_asmopcode(const s: string):boolean;      const        { sorted by length so longer postfixes will match first }        postfix2strsorted : array[1..7] of string[3] = (          'SB','SH','SW',          'B','H','W',          'S');        postfixsorted : array[1..7] of TOpPostfix = (          PF_SB,PF_SH,PF_SW,          PF_B,PF_H,PF_W,          PF_S);                      { store replicate }        ldst14: array[boolean,boolean,'1'..'4'] of tasmop =          (((A_LD1,A_LD2,A_LD3,A_LD4),            (A_LD1R,A_LD2R,A_LD3R,A_LD4R)),           ((A_ST1,A_ST2,A_ST3,A_ST4),            (A_NONE,A_NONE,A_NONE,A_NONE)));      var        j  : longint;        hs : string;        maxlen : longint;      Begin        { making s a value parameter would break other assembler readers }        hs:=s;        is_asmopcode:=false;        { clear opcode }        actopcode:=A_None;        actcondition:=C_None;        { b.cond ? }        if (length(hs)=4) and           (hs[1]='B') and           (hs[2]='.') then          begin            actopcode:=A_B;            actasmtoken:=AS_OPCODE;            actcondition:=ToConditionCode(copy(hs,3,length(actasmpattern)-2),false);            if actcondition<>C_None then              is_asmopcode:=true;            exit;          end;        (* ldN(r)/stN.size ? (shorthand for "ldN(r)/stN { Vx.size, Vy.size } ..."          supported by clang and possibly gas *)        actinsmmsubreg:=R_SUBNONE;        if (length(s)>=5) and           (((hs[1]='L') and             (hs[2]='D')) or            ((hs[1]='S') and             (hs[2]='T'))) and           (hs[3] in ['1'..'4']) and           ((hs[4]='.') or            ((hs[4]='R') and             (hs[5]='.'))) then          begin            actinsmmsubreg:=ParseArrangementSpecifier(copy(hs,4+ord(hs[4]='R'),255));            if actinsmmsubreg=R_SUBNONE then              exit;            actopcode:=ldst14[hs[1]='S',hs[4]='R',hs[3]];            actasmtoken:=AS_OPCODE;            if actopcode<>A_NONE then              is_asmopcode:=true;            exit;          end;        maxlen:=max(length(hs),7);        actopcode:=A_NONE;        for j:=maxlen downto 1 do          begin            actopcode:=tasmop(PtrUInt(iasmops.Find(copy(hs,1,j))));            if actopcode<>A_NONE then              begin                actasmtoken:=AS_OPCODE;                { strip op code }                delete(hs,1,j);                break;              end;          end;        if actopcode=A_NONE then          exit;        { check for postfix }        if length(hs)>0 then          begin            for j:=low(postfixsorted) to high(postfixsorted) do              begin                if copy(hs,1,length(postfix2strsorted[j]))=postfix2strsorted[j] then                  begin                    actoppostfix:=postfixsorted[j];                    { strip postfix }                    delete(hs,1,length(postfix2strsorted[j]));                    break;                  end;              end;          end;        { if we stripped all postfixes, it's a valid opcode }        is_asmopcode:=length(hs)=0;      end;    procedure taarch64attreader.ConvertCalljmp(instr: taarch64instruction);      var        newopr : toprrec;      begin        if instr.Operands[1].opr.typ=OPR_REFERENCE then          begin            newopr.typ:=OPR_SYMBOL;            newopr.symbol:=instr.Operands[1].opr.ref.symbol;            newopr.symofs:=instr.Operands[1].opr.ref.offset;            if (instr.Operands[1].opr.ref.base<>NR_NO) or              (instr.Operands[1].opr.ref.index<>NR_NO) or              (instr.Operands[1].opr.ref.refaddr<>addr_pic) then              Message(asmr_e_syn_operand);            instr.Operands[1].opr:=newopr;          end;      end;    procedure taarch64attreader.handleopcode;      var        instr: taarch64instruction;      begin        instr:=taarch64instruction.Create(taarch64operand);        BuildOpcode(instr);        if is_calljmp(instr.opcode) then          ConvertCalljmp(instr);        {        instr.AddReferenceSizes;        instr.SetInstructionOpsize;        instr.CheckOperandSizes;        }        instr.ConcatInstruction(curlist);        instr.Free;        actoppostfix:=PF_None;      end;    procedure taarch64attreader.handletargetdirective;      function maxoffset(ash:TAsmSehDirective):aint;        begin          case ash of            ash_savefplr,            ash_saveregp,            ash_savereg,            ash_savefregp,            ash_savefreg:              result:=504;            ash_savefplr_x,            ash_saveregp_x,            ash_savefregp_x:              result:=-512;            ash_savereg_x,            ash_savefreg_x:              result:=-256;            ash_addfp:              result:=2040;            else              internalerror(2020041204);          end;        end;      procedure add_reg_with_offset(ash:TAsmSehDirective;hreg:tregister;hnum:aint;neg:boolean);        begin          if (neg and ((hnum>0) or (hnum<maxoffset(ash)) or (((-hnum) and $7)<>0))) or              (not neg and ((hnum<0) or (hnum>maxoffset(ash)) or ((hnum and $7)<>0))) then            Message1(asmr_e_bad_seh_directive_offset,sehdirectivestr[actsehdirective])          else            begin              if neg then                hnum:=-hnum;              if hreg=NR_NO then                curlist.concat(cai_seh_directive.create_offset(actsehdirective,hnum))              else                curlist.concat(cai_seh_directive.create_reg_offset(actsehdirective,hreg,hnum));            end;        end;      var        hreg,        hreg2 : TRegister;        hnum : aint;        flags : integer;        ai : tai_seh_directive;        hs : string;        err : boolean;      begin        if actasmtoken<>AS_TARGET_DIRECTIVE then          InternalError(2020033102);        Consume(AS_TARGET_DIRECTIVE);        Include(current_procinfo.flags,pi_has_unwind_info);        case actsehdirective of          ash_nop,          ash_setfp,          ash_endprologue,          ash_handlerdata:            curlist.concat(cai_seh_directive.create(actsehdirective));          ash_handler:            begin              hs:=actasmpattern;              Consume(AS_ID);              flags:=0;              err:=false;              while actasmtoken=AS_COMMA do                begin                  Consume(AS_COMMA);                  if actasmtoken=AS_AT then                    begin                      Consume(AS_AT);                      if actasmtoken=AS_ID then                        begin                          uppervar(actasmpattern);                          if actasmpattern='EXCEPT' then                            flags:=flags or 1                          else if actasmpattern='UNWIND' then                            flags:=flags or 2                          else                            err:=true;                          Consume(AS_ID);                        end                      else                        err:=true;                    end                  else                    err:=true;                  if err then                    begin                      Message(asmr_e_syntax_error);                      RecoverConsume(false);                      exit;                    end;                end;              ai:=cai_seh_directive.create_name(ash_handler,hs);              ai.data.flags:=flags;              curlist.concat(ai);            end;          ash_savefplr,          ash_savefplr_x:            begin              hnum:=BuildConstExpression(false,false);              add_reg_with_offset(actsehdirective,NR_NO,hnum,actsehdirective=ash_savefplr_x);            end;          ash_savereg,          ash_savereg_x:            begin              hreg:=actasmregister;              Consume(AS_REGISTER);              if (getregtype(hreg)<>R_INTREGISTER) or (getsubreg(hreg)<>R_SUBWHOLE) or (getsupreg(hreg)<19) then                Message1(asmr_e_bad_seh_directive_register,sehdirectivestr[actsehdirective]);              Consume(AS_COMMA);              hnum:=BuildConstExpression(false,false);              add_reg_with_offset(actsehdirective,hreg,hnum,actsehdirective=ash_savereg_x);            end;          ash_saveregp,          ash_saveregp_x:            begin              hreg:=actasmregister;              consume(AS_REGISTER);              if (getregtype(hreg)<>R_INTREGISTER) or (getsubreg(hreg)<>R_SUBWHOLE) or (getsupreg(hreg)<19) then                Message1(asmr_e_bad_seh_directive_register,sehdirectivestr[actsehdirective]);              consume(AS_COMMA);              hreg2:=actasmregister;              consume(AS_REGISTER);              if (getregtype(hreg2)<>R_INTREGISTER) or (getsubreg(hreg2)<>R_SUBWHOLE) or (getsupreg(hreg2)<>getsupreg(hreg)+1) then                Message1(asmr_e_bad_seh_directive_register,sehdirectivestr[actsehdirective]);              consume(AS_COMMA);              hnum:=BuildConstExpression(false,false);              add_reg_with_offset(actsehdirective,hreg,hnum,actsehdirective=ash_saveregp_x);            end;          ash_savefreg,          ash_savefreg_x:            begin              hreg:=actasmregister;              Consume(AS_REGISTER);              if (getregtype(hreg)<>R_MMREGISTER) or (getsubreg(hreg)<>R_SUBWHOLE) or (getsupreg(hreg)<8) then                Message1(asmr_e_bad_seh_directive_register,sehdirectivestr[actsehdirective]);              Consume(AS_COMMA);              hnum:=BuildConstExpression(false,false);              add_reg_with_offset(actsehdirective,hreg,hnum,actsehdirective=ash_savefreg_x);            end;          ash_savefregp,          ash_savefregp_x:            begin              hreg:=actasmregister;              consume(AS_REGISTER);              if (getregtype(hreg)<>R_MMREGISTER) or (getsubreg(hreg)<>R_SUBWHOLE) or (getsupreg(hreg)<8) then                Message1(asmr_e_bad_seh_directive_register,sehdirectivestr[actsehdirective]);              consume(AS_COMMA);              hreg2:=actasmregister;              consume(AS_REGISTER);              if (getregtype(hreg2)<>R_MMREGISTER) or (getsubreg(hreg2)<>R_SUBWHOLE) or (getsupreg(hreg2)<>getsupreg(hreg)+1) then                Message1(asmr_e_bad_seh_directive_register,sehdirectivestr[actsehdirective]);              consume(AS_COMMA);              hnum:=BuildConstExpression(false,false);              add_reg_with_offset(actsehdirective,hreg,hnum,actsehdirective=ash_savefregp_x);            end;          ash_stackalloc:            begin              hnum:=BuildConstExpression(false,false);              if (hnum<0) or (hnum>$FFFFFF) or ((hnum and 7)<>0) then                Message1(asmr_e_bad_seh_directive_offset,sehdirectivestr[ash_stackalloc])              else                curlist.concat(cai_seh_directive.create_offset(ash_stackalloc,hnum));            end;          else            InternalError(2020033103);        end;        if actasmtoken<>AS_SEPARATOR then          Consume(AS_SEPARATOR);      end;{*****************************************************************************                                     Initialize*****************************************************************************}const  asmmode_arm_att_info : tasmmodeinfo =          (            id    : asmmode_arm_gas;            idtxt : 'GAS';            casmreader : taarch64attreader;          );  asmmode_arm_standard_info : tasmmodeinfo =          (            id    : asmmode_standard;            idtxt : 'STANDARD';            casmreader : taarch64attreader;          );initialization  RegisterAsmMode(asmmode_arm_att_info);  RegisterAsmMode(asmmode_arm_standard_info);end.
 |