| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835 | {    Copyright (c) 1998-2000 by Carl Eric Codere    This unit does the parsing process for the motorola 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 ra68kmot;{$i fpcdefs.inc}{**********************************************************************}{ WARNING                                                              }{**********************************************************************}{  Any modification in the order or removal of terms in the tables     }{  in m68k.pas and asmo68k.pas  will BREAK the code in this unit,      }{  unless the appropriate changes are made to this unit. Addition      }{  of terms though, will not change the code herein.                   }{**********************************************************************}{---------------------------------------------------------------------------}{ LEFT TO DO                                                                }{---------------------------------------------------------------------------}{  o Add support for sized indexing such as in d0.l                         }{      presently only (an,dn) is supported for indexing --                  }{        size defaults to LONG.                                             }{  o Add support for MC68020 opcodes.                                       }{  o Add support for MC68020 adressing modes.                               }{  o Add operand checking with m68k opcode table in ConcatOpCode            }{  o Add Floating point support                                             }{---------------------------------------------------------------------------}  interface    uses      cutils,      globtype,cclasses,cpubase,      symconst,      aasmbase,      rabase,rasm,ra68k,rautils;    type      tasmtoken = (        AS_NONE,AS_LABEL,AS_LLABEL,AS_STRING,AS_HEXNUM,AS_OCTALNUM,        AS_BINNUM,AS_COMMA,AS_LBRACKET,AS_RBRACKET,AS_LPAREN,        AS_RPAREN,AS_COLON,AS_DOT,AS_PLUS,AS_MINUS,AS_STAR,AS_INTNUM,        AS_SEPARATOR,AS_ID,AS_REGISTER,AS_OPCODE,AS_SLASH,AS_APPT,AS_REALNUM,        AS_ALIGN,          {------------------ Assembler directives --------------------}        AS_DB,AS_DW,AS_DD,AS_XDEF,AS_END,          {------------------ Assembler Operators  --------------------}        AS_MOD,AS_SHL,AS_SHR,AS_NOT,AS_AND,AS_OR,AS_XOR);      tasmkeyword = string[10];      tm68kmotreader = class(tasmreader)         actasmtoken    : tasmtoken;         prevasmtoken   : tasmtoken;         procedure SetupTables;         function Assemble: tlinkedlist;override;         function is_asmopcode(const s: string) : boolean;         Function is_asmdirective(const s: string):boolean;         function is_register(const s:string):boolean;         procedure GetToken;         function consume(t : tasmtoken):boolean;         function findopcode(s: string; var opsize: topsize): tasmop;         Function BuildExpression(allow_symbol : boolean; asmsym : pshortstring) : longint;         Procedure BuildConstant(maxvalue: longint);         Procedure BuildRealConstant(typ : tfloattype);         Procedure BuildScaling(const oper:tm68koperand);         Function BuildRefExpression: longint;         procedure BuildReference(const oper:tm68koperand);         Procedure BuildOperand(const oper:tm68koperand);         Procedure BuildStringConstant(asciiz: boolean);         Procedure BuildOpCode(instr:Tm68kinstruction);      end;Implementation    uses       { global }       globals,verbose,       systems,       { aasm }       cpuinfo,aasmtai,aasmdata,aasmcpu,       cgbase,       { symtable }       symbase,symtype,symsym,symdef,symtable,       { pass 1 }       nbas,       { parser }       scanner,ag68kgas,       itcpugas       ;const   firstdirective = AS_DB;   lastdirective  = AS_END;   firstoperator  = AS_MOD;   lastoperator   = AS_XOR;   _count_asmdirectives = longint(lastdirective)-longint(firstdirective);   _count_asmoperators  = longint(lastoperator)-longint(firstoperator);   _asmdirectives : array[0.._count_asmdirectives] of tasmkeyword =    ('DC.B','DC.W','DC.L','XDEF','END');    { problems with shl,shr,not,and,or and xor, they are }    { context sensitive.                                 }    _asmoperators : array[0.._count_asmoperators] of tasmkeyword = (    'MOD','SHL','SHR','NOT','AND','OR','XOR');   token2str : array[tasmtoken] of tasmkeyword=(        'NONE','LABEL','LLABEL','STRING','HEXNUM','OCTALNUM',        'BINNUM',',','[',']','(',        ')',':','.','+','-','*','INTNUM',        'SEPARATOR','ID','REGISTER','OPCODE','/','APPT','REALNUM',        'ALIGN',          {------------------ Assembler directives --------------------}        'DB','DW','DD','XDEF','END',          {------------------ Assembler Operators  --------------------}        'MOD','SHL','SHR','NOT','AND','OR','XOR');const  firsttoken : boolean = TRUE;  operandnum : byte = 0;    procedure tm68kmotreader.SetupTables;      { creates uppercased symbol tables for speed access }      var        i : tasmop;      Begin        { opcodes }        iasmops:=TFPHashList.create;        for i:=firstop to lastop do          iasmops.Add(upper(gas_op2str[i]),Pointer(PtrInt(i)));      end;  {---------------------------------------------------------------------}  {                     Routines for the tokenizing                     }  {---------------------------------------------------------------------}    function tm68kmotreader.is_asmopcode(const s: string):boolean;      var        hs : string;        j : byte;      begin        is_asmopcode:=false;        { first of all we remove the suffix }        j:=pos('.',s);        if j>0 then          hs:=copy(s,1,j-1)        else          hs:=s;        { Search opcodes }        actopcode:=tasmop(PtrInt(iasmops.Find(hs)));        if actopcode<>A_NONE then          begin            actasmtoken:=AS_OPCODE;            result:=TRUE;            exit;          end;      end;   Function tm68kmotreader.is_asmdirective(const s: string):boolean;   var    i:byte;   begin     result:=false;     for i:=0 to _count_asmdirectives do     begin        if s=_asmdirectives[i] then        begin           actasmtoken := tasmtoken(longint(firstdirective)+i);           result:=true;           exit;        end;     end;   end;    function tm68kmotreader.is_register(const s:string):boolean;      begin        is_register:=false;        // FIX ME!!! Ugly, needs a proper fix (KB)        actasmregister:=gas_regnum_search('%'+lower(s));        if actasmregister<>NR_NO then          begin            is_register:=true;            actasmtoken:=AS_REGISTER;          end;      end;  Procedure tm68kmotreader.GetToken;  {*********************************************************************}  { FUNCTION GetToken: tinteltoken;                                     }  {  Description: This routine returns intel assembler tokens and       }  {  does some minor syntax error checking.                             }  {*********************************************************************}  var   token: tasmtoken;   forcelabel: boolean;  begin    forcelabel := FALSE;    actasmpattern :='';    {* INIT TOKEN TO NOTHING *}    token := AS_NONE;    { while space and tab , continue scan... }    while c in [' ',#9] do     c:=current_scanner.asmgetchar;    if not (c in [#10,#13,'{',';']) then     current_scanner.gettokenpos;    { Possiblities for first token in a statement:                }    {   Local Label, Label, Directive, Prefix or Opcode....       }    if firsttoken and not (c in [#10,#13,'{',';']) then    begin      firsttoken := FALSE;      if c = '@' then      begin        token := AS_LLABEL;   { this is a local label }        { Let us point to the next character }        c := current_scanner.asmgetchar;      end;      while c in ['A'..'Z','a'..'z','0'..'9','_','@','.'] do      begin         { if there is an at_sign, then this must absolutely be a label }         if c = '@' then forcelabel:=TRUE;         actasmpattern := actasmpattern + c;         c := current_scanner.asmgetchar;      end;      uppervar(actasmpattern);      if c = ':' then      begin           case token of             AS_NONE: token := AS_LABEL;             AS_LLABEL: ; { do nothing }           end; { end case }           { let us point to the next character }           c := current_scanner.asmgetchar;           actasmtoken := token;           exit;      end;      { Are we trying to create an identifier with }      { an at-sign...?                             }      if forcelabel then       Message(asmr_e_none_label_contain_at);      If is_asmopcode(actasmpattern) then       exit;      if is_asmdirective(actasmpattern) then        exit      else        begin          actasmtoken := AS_NONE;          Message1(asmr_e_invalid_or_missing_opcode,actasmpattern);        end;    end    else { else firsttoken }    { Here we must handle all possible cases                              }    begin      case c of         '@':   { possiblities : - local label reference , such as in jmp @local1 }                {                - @Result, @Code or @Data special variables.     }                            begin                             actasmpattern := c;                             c:= current_scanner.asmgetchar;                             while c in  ['A'..'Z','a'..'z','0'..'9','_','@','.'] do                             begin                               actasmpattern := actasmpattern + c;                               c := current_scanner.asmgetchar;                             end;                             uppervar(actasmpattern);                             actasmtoken := AS_ID;                             exit;                            end;      { identifier, register, opcode, prefix or directive }         'A'..'Z','a'..'z','_': begin                             actasmpattern := c;                             c:= current_scanner.asmgetchar;                             while c in  ['A'..'Z','a'..'z','0'..'9','_','.'] do                             begin                               actasmpattern := actasmpattern + c;                               c := current_scanner.asmgetchar;                             end;                             uppervar(actasmpattern);                             If is_asmopcode(actasmpattern) then                               exit;                             if is_register(actasmpattern) then                               exit;                             if is_asmdirective(actasmpattern) then                               exit;                             { this is surely an identifier }                             actasmtoken := AS_ID;                             exit;                          end;           { override operator... not supported }           '&':       begin                         c:=current_scanner.asmgetchar;                         actasmtoken := AS_AND;                      end;           { string or character }           '''' :                      begin                         actasmpattern:='';                         while true do                         begin                           if c = '''' then                           begin                              c:=current_scanner.asmgetchar;                              if c=#10 then                              begin                                 Message(scan_f_string_exceeds_line);                                 break;                              end;                              repeat                                  if c=''''then                                   begin                                       c:=current_scanner.asmgetchar;                                       if c='''' then                                        begin                                               actasmpattern:=actasmpattern+'''';                                               c:=current_scanner.asmgetchar;                                               if c=#10 then                                               begin                                                    Message(scan_f_string_exceeds_line);                                                    break;                                               end;                                        end                                        else break;                                   end                                   else                                   begin                                          actasmpattern:=actasmpattern+c;                                          c:=current_scanner.asmgetchar;                                          if c=#10 then                                            begin                                               Message(scan_f_string_exceeds_line);                                               break                                            end;                                   end;                              until false; { end repeat }                           end                           else break; { end if }                         end; { end while }                   token:=AS_STRING;                   actasmtoken := token;                   exit;                 end;           '$' :  begin                    c:=current_scanner.asmgetchar;                    while c in ['0'..'9','A'..'F','a'..'f'] do                    begin                      actasmpattern := actasmpattern + c;                      c := current_scanner.asmgetchar;                    end;                   actasmtoken := AS_HEXNUM;                   exit;                  end;           ',' : begin                   actasmtoken := AS_COMMA;                   c:=current_scanner.asmgetchar;                   exit;                 end;           '(' : begin                   actasmtoken := AS_LPAREN;                   c:=current_scanner.asmgetchar;                   exit;                 end;           ')' : begin                   actasmtoken := AS_RPAREN;                   c:=current_scanner.asmgetchar;                   exit;                 end;           ':' : begin                   actasmtoken := AS_COLON;                   c:=current_scanner.asmgetchar;                   exit;                 end;{           '.' : begin                   actasmtoken := AS_DOT;                   c:=current_scanner.asmgetchar;                   exit;                 end; }           '+' : begin                   actasmtoken := AS_PLUS;                   c:=current_scanner.asmgetchar;                   exit;                 end;           '-' : begin                   actasmtoken := AS_MINUS;                   c:=current_scanner.asmgetchar;                   exit;                 end;           '*' : begin                   actasmtoken := AS_STAR;                   c:=current_scanner.asmgetchar;                   exit;                 end;           '/' : begin                   actasmtoken := AS_SLASH;                   c:=current_scanner.asmgetchar;                   exit;                 end;           '<' : begin                   c := current_scanner.asmgetchar;                   { invalid characters }                   if c <> '<' then                    Message(asmr_e_invalid_char_smaller);                   { still assume << }                   actasmtoken := AS_SHL;                   c := current_scanner.asmgetchar;                   exit;                 end;           '>' : begin                   c := current_scanner.asmgetchar;                   { invalid characters }                   if c <> '>' then                    Message(asmr_e_invalid_char_greater);                   { still assume << }                   actasmtoken := AS_SHR;                   c := current_scanner.asmgetchar;                   exit;                 end;           '|' : begin                   actasmtoken := AS_OR;                   c := current_scanner.asmgetchar;                   exit;                 end;           '^' : begin                  actasmtoken := AS_XOR;                  c := current_scanner.asmgetchar;                  exit;                 end;           '#' : begin                  actasmtoken:=AS_APPT;                  c:=current_scanner.asmgetchar;                  exit;                 end;           '%' : begin                   c:=current_scanner.asmgetchar;                   while c in ['0','1'] do                   begin                     actasmpattern := actasmpattern + c;                     c := current_scanner.asmgetchar;                   end;                   actasmtoken := AS_BINNUM;                   exit;                 end;           { integer number }           '0'..'9': begin                        actasmpattern := c;                        c := current_scanner.asmgetchar;                        while c in ['0'..'9'] do                          begin                             actasmpattern := actasmpattern + c;                             c:= current_scanner.asmgetchar;                          end;                        actasmtoken := AS_INTNUM;                        exit;                     end;         ';' : begin                  repeat                     c:=current_scanner.asmgetchar;                  until c=#10;                  firsttoken := TRUE;                  actasmtoken:=AS_SEPARATOR;               end;         '{',#13,#10 : begin                            c:=current_scanner.asmgetchar;                            firsttoken := TRUE;                            actasmtoken:=AS_SEPARATOR;                           end;            else             begin               Message(scan_f_illegal_char);             end;      end; { end case }    end; { end else if }  end;  {---------------------------------------------------------------------}  {                     Routines for the parsing                        }  {---------------------------------------------------------------------}    function tm68kmotreader.consume(t : tasmtoken):boolean;      var        p: pointer;      begin        Consume:=true;        if t<>actasmtoken then         begin           p:=nil;           dword(p^):=0;           Message2(scan_f_syn_expected,token2str[t],token2str[actasmtoken]);           Consume:=false;         end;        repeat          gettoken;        until actasmtoken<>AS_NONE;      end;   function tm68kmotreader.findopcode(s: string; var opsize: topsize): tasmop;  {*********************************************************************}  { FUNCTION findopcode(s: string): tasmop;                             }  {  Description: Determines if the s string is a valid opcode          }  {  if so returns correct tasmop token.                                }  {*********************************************************************}   var    j: byte;    op_size: string;   begin     findopcode := A_NONE;     j:=pos('.',s);     if j<>0 then     begin       op_size:=copy(s,j+1,1);       case op_size[1] of       { For the motorola only opsize size is used to }       { determine the size of the operands.             }       'B': opsize := S_B;       'W': opsize := S_W;       'L': opsize := S_L;       'S': opsize := S_FS;       'D': opsize := S_FD;       'X': opsize := S_FX;       else        Message1(asmr_e_unknown_opcode,s);       end;       { delete everything starting from dot }       delete(s,j,length(s));     end;     result:=actopcode;   end;    Function tm68kmotreader.BuildExpression(allow_symbol : boolean; asmsym : pshortstring) : longint;  {*********************************************************************}  { FUNCTION BuildExpression: longint                                   }  {  Description: This routine calculates a constant expression to      }  {  a given value. The return value is the value calculated from       }  {  the expression.                                                    }  { The following tokens (not strings) are recognized:                  }  {    (,),SHL,SHR,/,*,NOT,OR,XOR,AND,MOD,+/-,numbers,ID to constants.  }  {*********************************************************************}  { ENTRY: On entry the token should be any valid expression token.     }  { EXIT:  On Exit the token points to either COMMA or SEPARATOR        }  { ERROR RECOVERY: Tries to find COMMA or SEPARATOR token by consuming }  {  invalid tokens.                                                    }  {*********************************************************************}  var expr: string;      hs, tempstr: string;      sym : tsym;      srsymtable : TSymtable;      hl : tasmlabel;      l : longint;      errorflag: boolean;  begin    errorflag := FALSE;    expr := '';    tempstr := '';    if allow_symbol then      asmsym^:='';    Repeat      Case actasmtoken of      AS_LPAREN: begin                  Consume(AS_LPAREN);                  expr := expr + '(';                end;      AS_RPAREN: begin                  Consume(AS_RPAREN);                  expr := expr + ')';                end;      AS_SHL:    begin                  Consume(AS_SHL);                  expr := expr + '<';                end;      AS_SHR:    begin                  Consume(AS_SHR);                  expr := expr + '>';                end;      AS_SLASH:  begin                  Consume(AS_SLASH);                  expr := expr + '/';                end;      AS_MOD:    begin                  Consume(AS_MOD);                  expr := expr + '%';                end;      AS_STAR:   begin                  Consume(AS_STAR);                  expr := expr + '*';                end;      AS_PLUS:   begin                  Consume(AS_PLUS);                  expr := expr + '+';                end;      AS_MINUS:  begin                  Consume(AS_MINUS);                  expr := expr + '-';                end;      AS_AND:    begin                  Consume(AS_AND);                  expr := expr + '&';                end;      AS_NOT:    begin                  Consume(AS_NOT);                  expr := expr + '~';                end;      AS_XOR:    begin                  Consume(AS_XOR);                  expr := expr + '^';                end;      AS_OR:     begin                  Consume(AS_OR);                  expr := expr + '|';                end;      AS_ID:    begin                  if SearchIConstant(actasmpattern,l) then                  begin                    str(l, tempstr);                    expr := expr + tempstr;                    Consume(AS_ID);                  End else                  if not allow_symbol then                  begin                    Message(asmr_e_syn_constant);                    l := 0;                  End else                  begin                    hs:='';                    if (expr[Length(expr)]='+') then                      Delete(expr,Length(expr),1)                    else if expr<>'' then                      begin                        Message(asmr_e_invalid_constant_expression);                        break;                      End;                    tempstr:=actasmpattern;                    consume(AS_ID);                    if (length(tempstr)>1) and (tempstr[1]='@') then                      begin                        CreateLocalLabel(tempstr,hl,false);                        hs:=hl.name                      end                    else if SearchLabel(tempstr,hl,false) then                      hs:=hl.name                    else                      begin                        searchsym(tempstr,sym,srsymtable);                        if assigned(sym) then                         begin                           case sym.typ of                             paravarsym,                             localvarsym :                               begin                                 Message(asmr_e_no_local_or_para_allowed);                                 hs:=tabstractvarsym(sym).mangledname;                               end;                             staticvarsym :                                   hs:=tstaticvarsym(sym).mangledname;                             procsym :                               begin                                 if tprocsym(sym).procdeflist.count>1 then                                      Message(asmr_w_calling_overload_func);                                 hs:=tprocdef(tprocsym(sym).procdeflist[0]).mangledname;                               end;                             typesym :                               begin                                 if not(ttypesym(sym).typedef.typ in [recorddef,objectdef]) then                                      Message(asmr_e_wrong_sym_type);                               end;                             else                               Message(asmr_e_wrong_sym_type);                           end;                        end                        else                           Message1(sym_e_unknown_id,tempstr);                      end;                     { symbol found? }                     if hs<>'' then                      begin                        if asmsym^='' then                         asmsym^:=hs                        else                         Message(asmr_e_cant_have_multiple_relocatable_symbols);                      end;                  end;                end;      AS_INTNUM:  begin                   expr := expr + actasmpattern;                   Consume(AS_INTNUM);                  end;      AS_BINNUM:  begin                      tempstr := tostr(ParseVal(actasmpattern,2));                      if tempstr = '' then                       Message(asmr_e_error_converting_binary);                      expr:=expr+tempstr;                      Consume(AS_BINNUM);                  end;      AS_HEXNUM: begin                    tempstr := tostr(ParseVal(actasmpattern,16));                    if tempstr = '' then                     Message(asmr_e_error_converting_hexadecimal);                    expr:=expr+tempstr;                    Consume(AS_HEXNUM);                end;      AS_OCTALNUM: begin                    tempstr := tostr(ParseVal(actasmpattern,8));                    if tempstr = '' then                     Message(asmr_e_error_converting_octal);                    expr:=expr+tempstr;                    Consume(AS_OCTALNUM);                  end;      { go to next term }      AS_COMMA: begin                  if not ErrorFlag then                    BuildExpression := CalculateExpression(expr)                  else                    BuildExpression := 0;                  Exit;               end;      { go to next symbol }      AS_SEPARATOR: begin                      if not ErrorFlag then                        BuildExpression := CalculateExpression(expr)                      else                        BuildExpression := 0;                      Exit;                   end;      else        begin          { only write error once. }          if not errorflag then           Message(asmr_e_invalid_constant_expression);          { consume tokens until we find COMMA or SEPARATOR }          Consume(actasmtoken);          errorflag := TRUE;        End;      end;    Until false;  end;  Procedure tm68kmotreader.BuildRealConstant(typ : tfloattype);  {*********************************************************************}  { PROCEDURE BuilRealConst                                             }  {  Description: This routine calculates a constant expression to      }  {  a given value. The return value is the value calculated from       }  {  the expression.                                                    }  { The following tokens (not strings) are recognized:                  }  {    +/-,numbers and real numbers                                     }  {*********************************************************************}  { ENTRY: On entry the token should be any valid expression token.     }  { EXIT:  On Exit the token points to either COMMA or SEPARATOR        }  { ERROR RECOVERY: Tries to find COMMA or SEPARATOR token by consuming }  {  invalid tokens.                                                    }  {*********************************************************************}  var expr: string;      r : extended;      code : word;      negativ : boolean;      errorflag: boolean;  begin    errorflag := FALSE;    Repeat    negativ:=false;    expr := '';    if actasmtoken=AS_PLUS then Consume(AS_PLUS)    else if actasmtoken=AS_MINUS then      begin         negativ:=true;         consume(AS_MINUS);      end;    Case actasmtoken of      AS_INTNUM:  begin                   expr := actasmpattern;                   Consume(AS_INTNUM);                 end;      AS_REALNUM:  begin                   expr := actasmpattern;                   { in ATT syntax you have 0d in front of the real }                   { should this be forced ?  yes i think so, as to }                   { conform to gas as much as possible.            }                   if (expr[1]='0') and (upper(expr[2])='D') then                     expr:=copy(expr,3,255);                   Consume(AS_REALNUM);                 end;      AS_BINNUM:  begin                      { checking for real constants with this should use  }                      { real DECODING otherwise the compiler will crash!  }                      Message(asmr_e_invalid_float_expr);                      expr:='0.0';                      Consume(AS_BINNUM);                 end;      AS_HEXNUM: begin                      { checking for real constants with this should use  }                      { real DECODING otherwise the compiler will crash!  }                    Message(asmr_e_invalid_float_expr);                    expr:='0.0';                    Consume(AS_HEXNUM);                end;      AS_OCTALNUM: begin                      { checking for real constants with this should use    }                      { real DECODING otherwise the compiler will crash!    }                      { xxxToDec using reals could be a solution, but the   }                      { problem is that these will crash the m68k compiler  }                      { when compiling -- because of lack of good fpu       }                      { support.                                           }                    Message(asmr_e_invalid_float_expr);                    expr:='0.0';                    Consume(AS_OCTALNUM);                  end;         else           begin             { only write error once. }             if not errorflag then              Message(asmr_e_invalid_float_expr);             { consume tokens until we find COMMA or SEPARATOR }             Consume(actasmtoken);             errorflag := TRUE;           End;         end;      { go to next term }      if (actasmtoken=AS_COMMA) or (actasmtoken=AS_SEPARATOR) then        begin          if negativ then expr:='-'+expr;          val(expr,r,code);          if code<>0 then            begin               r:=0;               Message(asmr_e_invalid_float_expr);               ConcatRealConstant(curlist,r,typ);            End          else            begin              ConcatRealConstant(curlist,r,typ);            End;        end      else        Message(asmr_e_invalid_float_expr);    Until actasmtoken=AS_SEPARATOR;  end;  Procedure tm68kmotreader.BuildConstant(maxvalue: longint);  {*********************************************************************}  { PROCEDURE BuildConstant                                             }  {  Description: This routine takes care of parsing a DB,DD,or DW      }  {  line and adding those to the assembler node. Expressions, range-   }  {  checking are fullly taken care of.                                 }  {   maxvalue: $ff -> indicates that this is a DB node.                }  {             $ffff -> indicates that this is a DW node.              }  {             $ffffffff -> indicates that this is a DD node.          }  {*********************************************************************}  { EXIT CONDITION:  On exit the routine should point to AS_SEPARATOR.  }  {*********************************************************************}  var   expr: string;   value : longint;  begin      Repeat        Case actasmtoken of          AS_STRING: begin                      if maxvalue <> $ff then                         Message(asmr_e_string_not_allowed_as_const);                      expr := actasmpattern;                      if length(expr) > 1 then                        Message(asmr_e_string_not_allowed_as_const);                      Consume(AS_STRING);                      Case actasmtoken of                       AS_COMMA: Consume(AS_COMMA);                       AS_SEPARATOR: ;                      else                       Message(asmr_e_invalid_string_expression);                      end; { end case }                      ConcatString(curlist,expr);                    end;          AS_INTNUM,AS_BINNUM,          AS_OCTALNUM,AS_HEXNUM:                    begin                      value:=BuildExpression(false,nil);                      ConcatConstant(curlist,value,maxvalue);                    end;          AS_ID:                     begin                      value:=BuildExpression(false,nil);                      if value > maxvalue then                      begin                         Message(asmr_e_constant_out_of_bounds);                         { assuming a value of maxvalue }                         value := maxvalue;                      end;                      ConcatConstant(curlist,value,maxvalue);                  end;          { These terms can start an assembler expression }          AS_PLUS,AS_MINUS,AS_LPAREN,AS_NOT: begin                                          value := BuildExpression(false,nil);                                          ConcatConstant(curlist,value,maxvalue);                                         end;          AS_COMMA:  begin                       Consume(AS_COMMA);                     END;          AS_SEPARATOR: ;        else         begin           Message(asmr_e_syntax_error);         end;    end; { end case }   Until actasmtoken = AS_SEPARATOR;  end;  Procedure TM68kMotreader.BuildScaling(const oper:tm68koperand);  {*********************************************************************}  {  Takes care of parsing expression starting from the scaling value   }  {  up to and including possible field specifiers.                     }  { EXIT CONDITION:  On exit the routine should point to  AS_SEPARATOR  }  { or AS_COMMA. On entry should point to the AS_STAR  token.           }  {*********************************************************************}  var str:string;      l: longint;      code: integer;  begin     Consume(AS_STAR);     if (oper.opr.ref.scalefactor <> 0)     and (oper.opr.ref.scalefactor <> 1) then      Message(asmr_e_wrong_base_index);     case actasmtoken of        AS_INTNUM: str := actasmpattern;        AS_HEXNUM: str := Tostr(ParseVal(actasmpattern,16));        AS_BINNUM: str := Tostr(ParseVal(actasmpattern,2));        AS_OCTALNUM: str := Tostr(ParseVal(actasmpattern,8));     else        Message(asmr_e_syntax_error);     end;     val(str, l, code);     if code <> 0 then      Message(asmr_e_wrong_scale_factor);     if ((l = 2) or (l = 4) or (l = 8) or (l = 1)) and (code = 0) then     begin        oper.opr.ref.scalefactor := l;     end     else     begin        Message(asmr_e_wrong_scale_factor);        oper.opr.ref.scalefactor := 0;     end;     if oper.opr.ref.index = NR_NO then     begin        Message(asmr_e_wrong_base_index);        oper.opr.ref.scalefactor := 0;     end;    { Consume the scaling number }    Consume(actasmtoken);    if actasmtoken = AS_RPAREN then        Consume(AS_RPAREN)    else       Message(asmr_e_wrong_scale_factor);    { // .Field.Field ... or separator/comma // }    if actasmtoken in [AS_COMMA,AS_SEPARATOR] then    begin    end    else     Message(asmr_e_syntax_error);  end;  Function TM68kMotreader.BuildRefExpression: longint;  {*********************************************************************}  { FUNCTION BuildRefExpression: longint                                   }  {  Description: This routine calculates a constant expression to      }  {  a given value. The return value is the value calculated from       }  {  the expression.                                                    }  { The following tokens (not strings) are recognized:                  }  {    SHL,SHR,/,*,NOT,OR,XOR,AND,MOD,+/-,numbers,ID to constants.      }  {*********************************************************************}  { ENTRY: On entry the token should be any valid expression token.     }  { EXIT:  On Exit the token points to the LPAREN token.                }  { ERROR RECOVERY: Tries to find COMMA or SEPARATOR token by consuming }  {  invalid tokens.                                                    }  {*********************************************************************}  var tempstr: string;      expr: string;    l : longint;    errorflag : boolean;  begin    errorflag := FALSE;    tempstr := '';    expr := '';    Repeat      Case actasmtoken of      AS_RPAREN: begin                   Message(asmr_e_syntax_error);                  Consume(AS_RPAREN);                end;      AS_SHL:    begin                  Consume(AS_SHL);                  expr := expr + '<';                end;      AS_SHR:    begin                  Consume(AS_SHR);                  expr := expr + '>';                end;      AS_SLASH:  begin                  Consume(AS_SLASH);                  expr := expr + '/';                end;      AS_MOD:    begin                  Consume(AS_MOD);                  expr := expr + '%';                end;      AS_STAR:   begin                  Consume(AS_STAR);                  expr := expr + '*';                end;      AS_PLUS:   begin                  Consume(AS_PLUS);                  expr := expr + '+';                end;      AS_MINUS:  begin                  Consume(AS_MINUS);                  expr := expr + '-';                end;      AS_AND:    begin                  Consume(AS_AND);                  expr := expr + '&';                end;      AS_NOT:    begin                  Consume(AS_NOT);                  expr := expr + '~';                end;      AS_XOR:    begin                  Consume(AS_XOR);                  expr := expr + '^';                end;      AS_OR:     begin                  Consume(AS_OR);                  expr := expr + '|';                end;      { End of reference }      AS_LPAREN: begin                     if not ErrorFlag then                        BuildRefExpression := CalculateExpression(expr)                     else                        BuildRefExpression := 0;                     { no longer in an expression }                     exit;                  end;      AS_ID:                begin                  if NOT SearchIConstant(actasmpattern,l) then                  begin                    Message(asmr_e_syn_constant);                    l := 0;                  end;                  str(l, tempstr);                  expr := expr + tempstr;                  Consume(AS_ID);                end;      AS_INTNUM:  begin                   expr := expr + actasmpattern;                   Consume(AS_INTNUM);                 end;      AS_BINNUM:  begin                      tempstr := Tostr(ParseVal(actasmpattern,2));                      if tempstr = '' then                       Message(asmr_e_error_converting_binary);                      expr:=expr+tempstr;                      Consume(AS_BINNUM);                 end;      AS_HEXNUM: begin                    tempstr := Tostr(ParseVal(actasmpattern,16));                    if tempstr = '' then                     Message(asmr_e_error_converting_hexadecimal);                    expr:=expr+tempstr;                    Consume(AS_HEXNUM);                end;      AS_OCTALNUM: begin                    tempstr := Tostr(ParseVal(actasmpattern,8));                    if tempstr = '' then                     Message(asmr_e_error_converting_octal);                    expr:=expr+tempstr;                    Consume(AS_OCTALNUM);                  end;      else        begin          { write error only once. }          if not errorflag then           Message(asmr_e_invalid_constant_expression);          BuildRefExpression := 0;          if actasmtoken in [AS_COMMA,AS_SEPARATOR] then exit;          { consume tokens until we find COMMA or SEPARATOR }          Consume(actasmtoken);          errorflag := TRUE;        end;      end;    Until false;  end;  {*********************************************************************}  { PROCEDURE BuildBracketExpression                                    }  {  Description: This routine builds up an expression after a LPAREN   }  {  token is encountered.                                              }  {   On entry actasmtoken should be equal to AS_LPAREN                 }  {*********************************************************************}  { EXIT CONDITION:  On exit the routine should point to either the     }  {       AS_COMMA or AS_SEPARATOR token.                               }  {*********************************************************************}  procedure TM68kMotreader.BuildReference(const oper:tm68koperand);    var      l:longint;      code: integer;      str: string;    begin       Consume(AS_LPAREN);       case actasmtoken of         { // (reg ... // }         AS_REGISTER:           begin             oper.opr.ref.base := actasmregister;             Consume(AS_REGISTER);             { can either be a register or a right parenthesis }             { // (reg)       // }             { // (reg)+      // }             if actasmtoken=AS_RPAREN then               begin                 Consume(AS_RPAREN);                 if actasmtoken = AS_PLUS then                 begin                   if (oper.opr.ref.direction <> dir_none) then                    Message(asmr_e_no_inc_and_dec_together)                   else                     oper.opr.ref.direction := dir_inc;                   Consume(AS_PLUS);                 end;                 if not (actasmtoken in [AS_COMMA,AS_SEPARATOR]) then                   begin                     Message(asmr_e_invalid_reference_syntax);                     { error recovery ... }                     while actasmtoken <> AS_SEPARATOR do                        Consume(actasmtoken);                   end;                   exit;               end;              { // (reg,reg .. // }              Consume(AS_COMMA);              if actasmtoken = AS_REGISTER then                begin                  oper.opr.ref.index :=                    actasmregister;                  Consume(AS_REGISTER);                  { check for scaling ... }                  case actasmtoken of                    AS_RPAREN:                       begin                         Consume(AS_RPAREN);                         if not (actasmtoken in [AS_COMMA,AS_SEPARATOR]) then                         begin                           { error recovery ... }                           Message(asmr_e_invalid_reference_syntax);                           while actasmtoken <> AS_SEPARATOR do                             Consume(actasmtoken);                         end;                         exit;                       end;                    AS_STAR:                       begin                         BuildScaling(oper);                       end;                    else                      begin                        Message(asmr_e_invalid_reference_syntax);                        while (actasmtoken <> AS_SEPARATOR) do                          Consume(actasmtoken);                      end;                  end; { end case }                end              else                begin                   Message(asmr_e_invalid_reference_syntax);                  while (actasmtoken <> AS_SEPARATOR) do                      Consume(actasmtoken);                end;           end;         AS_HEXNUM,AS_OCTALNUM,   { direct address }         AS_BINNUM,AS_INTNUM:           begin             case actasmtoken of               AS_INTNUM: str := actasmpattern;               AS_HEXNUM: str := Tostr(ParseVal(actasmpattern,16));               AS_BINNUM: str := Tostr(ParseVal(actasmpattern,2));               AS_OCTALNUM: str := Tostr(ParseVal(actasmpattern,8));              else                Message(asmr_e_syntax_error);             end;             Consume(actasmtoken);             val(str, l, code);             if code <> 0 then               Message(asmr_e_invalid_reference_syntax)             else               oper.opr.ref.offset := l;             Consume(AS_RPAREN);             if not (actasmtoken in [AS_COMMA,AS_SEPARATOR]) then             begin               { error recovery ... }               Message(asmr_e_invalid_reference_syntax);               while actasmtoken <> AS_SEPARATOR do                 Consume(actasmtoken);             end;             exit;           end;         else           begin             Message(asmr_e_invalid_reference_syntax);             while (actasmtoken <> AS_SEPARATOR) do               Consume(actasmtoken);           end;       end;    end;  Procedure TM68kMotreader.BuildOperand(const oper:tm68koperand);  {*********************************************************************}  { EXIT CONDITION:  On exit the routine should point to either the     }  {       AS_COMMA or AS_SEPARATOR token.                               }  {*********************************************************************}  var    expr: string;    tempstr: string;    lab: tasmlabel;    l : longint;    i: Tsuperregister;    r:Tregister;    hl: tasmlabel;    reg_one, reg_two: tregister;    regset: tcpuregisterset;    p: pointer;  begin   regset := [];   tempstr := '';   case actasmtoken of   { // Memory reference //  }     AS_LPAREN:               begin                  Oper.InitRef;                  BuildReference(oper);               end;   { // Constant expression //  }     AS_APPT:  begin                      Consume(AS_APPT);                      if not (oper.opr.typ in [OPR_NONE,OPR_CONSTANT]) then                         Message(asmr_e_invalid_operand_type);                      { identifiers are handled by BuildExpression }                      oper.opr.typ := OPR_CONSTANT;                      oper.opr.val :=BuildExpression(true,@tempstr);                      if tempstr<>'' then                        begin                          l:=oper.opr.val;                          oper.opr.typ := OPR_SYMBOL;                          oper.opr.symofs := l;                          oper.opr.symbol := current_asmdata.RefAsmSymbol(tempstr);                        end;                 end;   { // Constant memory offset .              // }   { // This must absolutely be followed by ( // }     AS_HEXNUM,AS_INTNUM,     AS_BINNUM,AS_OCTALNUM,AS_PLUS:                   begin                      Oper.InitRef;                      oper.opr.ref.offset:=BuildRefExpression;                      BuildReference(oper);                   end;   { // A constant expression, or a Variable ref. // }     AS_ID:  begin              Oper.InitRef;              if actasmpattern[1] = '@' then              { // Label or Special symbol reference // }              begin                 if actasmpattern = '@RESULT' then                    oper.SetUpResult                 else                 if actasmpattern = 'SELF' then                    oper.SetUpSelf                 else                 if (actasmpattern = '@CODE') or (actasmpattern = '@DATA') then                    Message(asmr_w_CODE_and_DATA_not_supported)                 else                  begin                    delete(actasmpattern,1,1);                    if actasmpattern = '' then                     Message(asmr_e_null_label_ref_not_allowed);                    CreateLocalLabel(actasmpattern,lab,false);                    oper.opr.typ := OPR_SYMBOL;                    oper.opr.symbol := lab;                    oper.opr.symofs := 0;//                    labeled := TRUE;                  end;                Consume(AS_ID);                if not (actasmtoken in [AS_SEPARATOR,AS_COMMA]) then                  Message(asmr_e_syntax_error);              end              { probably a variable or normal expression }              { or a procedure (such as in CALL ID)      }              else               begin                 { is it a constant ? }                 if SearchIConstant(actasmpattern,l) then                   begin                     Oper.InitRef;                     oper.opr.ref.offset:=BuildRefExpression;                     BuildReference(oper);                   end                 else { is it a label variable ? }                     { // ID[ , ID.Field.Field or simple ID // }                     { check if this is a label, if so then }                     { emit it as a label.                  }                     if SearchLabel(actasmpattern,hl,false) then                       begin                         oper.opr.typ := OPR_SYMBOL;                         oper.opr.symbol := hl;                         oper.opr.symofs := 0;//                         labeled := TRUE;                         Consume(AS_ID);                         if not (actasmtoken in [AS_SEPARATOR,AS_COMMA]) then                          Message(asmr_e_syntax_error);                       end                      else begin                       expr:=actasmpattern;                       Consume(AS_ID);                       { typecasting? }                       if SearchType(expr,l) then                        begin                          oper.hastype:=true;                          oper.typesize:=l;                          case actasmtoken of                            AS_LPAREN :                              begin                                { Support Type([Reference]) }                                Consume(AS_LPAREN);                                BuildOperand(oper{,true});                                Consume(AS_RPAREN);                              end;                            AS_LBRACKET :                              begin                                { Support Var.Type[Index] }                                { Convert @label.Byte[1] to reference }                                if oper.opr.typ=OPR_SYMBOL then                                  oper.initref;                              end;                          end;                        end                       else                        begin                          if not oper.SetupVar(expr,false) then                            begin                              { not a variable, check special variables.. }                              if expr = 'SELF' then                                oper.SetupSelf                              else begin                                writeln('unknown id: ',expr);                                Message1(sym_e_unknown_id,expr);                              end;                              expr:='';                            end;                         end;//                       Message1(sym_e_unknown_id,actasmpattern);                      end;                       case actasmtoken of                         AS_LPAREN: { indexing }                           BuildReference(oper);                         AS_SEPARATOR,AS_COMMA: begin                         end;                       else                         Message(asmr_e_syntax_error);                       end;                   end;               end;   { // Pre-decrement mode reference or constant mem offset.   // }     AS_MINUS:    begin                   Consume(AS_MINUS);                   if actasmtoken = AS_LPAREN then                   begin                     Oper.InitRef;                     { indicate pre-decrement mode }                     oper.opr.ref.direction := dir_dec;                     BuildReference(oper);                   end                   else                   if actasmtoken in [AS_OCTALNUM,AS_HEXNUM,AS_BINNUM,AS_INTNUM] then                   begin                      Oper.InitRef;                      oper.opr.ref.offset:=BuildRefExpression;                      { negate because was preceded by a negative sign! }                      oper.opr.ref.offset:=-oper.opr.ref.offset;                      BuildReference(oper);                   end                   else                   begin                    Message(asmr_e_syntax_error);                    while not (actasmtoken in [AS_SEPARATOR,AS_COMMA]) do                       Consume(actasmtoken);                   end;                  end;   { // Register, a variable reference or a constant reference // }     AS_REGISTER: begin//                   writeln('register! ',actasmpattern);                   { save the type of register used. }                   tempstr := actasmpattern;                   Consume(AS_REGISTER);                   { // Simple register // }                   if (actasmtoken = AS_SEPARATOR) or (actasmtoken = AS_COMMA) then                   begin//                        writeln('simple reg');                        if not (oper.opr.typ in [OPR_NONE,OPR_REGISTER]) then                         Message(asmr_e_invalid_operand_type);                        oper.opr.typ := OPR_REGISTER;                        oper.opr.reg := actasmregister;                   end                   else                   { HERE WE MUST HANDLE THE SPECIAL CASE OF MOVEM AND FMOVEM }                   { // Individual register listing // }                   if (actasmtoken = AS_SLASH) then                   begin                     r:=actasmregister;                     if getregtype(r)<>R_INTREGISTER then                       internalerror(200302191);                     include(regset,getsupreg(r));                     Consume(AS_SLASH);                     if actasmtoken = AS_REGISTER then                     begin                       While not (actasmtoken in [AS_SEPARATOR,AS_COMMA]) do                       begin                         case actasmtoken of                          AS_REGISTER: begin                                         if getregtype(r)<>R_INTREGISTER then                                           internalerror(200302191);                                         include(regset,getsupreg(r));                                         Consume(AS_REGISTER);                                       end;                          AS_SLASH: Consume(AS_SLASH);                          AS_SEPARATOR,AS_COMMA: break;                         else                          begin                            Message(asmr_e_invalid_reg_list_in_movem);                            Consume(actasmtoken);                          end;                         end; { end case }                       end; { end while }                       oper.opr.typ:= OPR_regset;                       oper.opr.regset := regset;                     end                     else                      { error recovery ... }                      begin                            Message(asmr_e_invalid_reg_list_in_movem);                            while not (actasmtoken in [AS_SEPARATOR,AS_COMMA]) do                               Consume(actasmtoken);                      end;                   end                   else                   { // Range register listing // }                   if (actasmtoken = AS_MINUS) then                   begin                     Consume(AS_MINUS);                     reg_one:=actasmregister;                     if actasmtoken <> AS_REGISTER then                     begin                       Message(asmr_e_invalid_reg_list_in_movem);                       while not (actasmtoken in [AS_SEPARATOR,AS_COMMA]) do                         Consume(actasmtoken);                     end                     else                     begin                      { determine the register range ... }                      reg_two:=actasmregister;                      if getregtype(reg_two)<>R_INTREGISTER then                        internalerror(200302191);                      if getsupreg(reg_one) > getsupreg(reg_two) then                       for i:=getsupreg(reg_two) to getsupreg(reg_one) do                         include(regset,i)                      else                       for i:=getsupreg(reg_one) to getsupreg(reg_two) do                         include(regset,i);                      Consume(AS_REGISTER);                      if not (actasmtoken in [AS_SEPARATOR,AS_COMMA]) then                      begin                       Message(asmr_e_invalid_reg_list_in_movem);                       while not (actasmtoken in [AS_SEPARATOR,AS_COMMA]) do                         Consume(actasmtoken);                      end;                      { set up instruction }                      oper.opr.typ:= OPR_regset;                      oper.opr.regset := regset;                     end;                   end                   else                   { DIVSL/DIVS/MULS/MULU with long for MC68020 only }                   if (actasmtoken = AS_COLON) then                   begin                     if (current_settings.cputype = cpu_MC68020) or (cs_compilesystem in current_settings.moduleswitches) then                     begin                       Consume(AS_COLON);                       if (actasmtoken = AS_REGISTER) then                       begin                         { set up old field, since register is valid }                         oper.opr.typ := OPR_REGISTER;                         oper.opr.reg := actasmregister;                         Inc(operandnum);                         oper.opr.typ := OPR_REGISTER;                         oper.opr.reg := actasmregister;                         Consume(AS_REGISTER);                         if not (actasmtoken in [AS_SEPARATOR,AS_COMMA]) then                         begin                          Message(asmr_e_invalid_reg_list_for_opcode);                          while not (actasmtoken in [AS_SEPARATOR,AS_COMMA]) do                            Consume(actasmtoken);                         end;                       end;                     end                     else                     begin                        Message1(asmr_e_higher_cpu_mode_required,'68020');                        if not (actasmtoken in [AS_SEPARATOR,AS_COMMA]) then                        begin                          Message(asmr_e_invalid_reg_list_for_opcode);                          while not (actasmtoken in [AS_SEPARATOR,AS_COMMA]) do                            Consume(actasmtoken);                        end;                     end;                   end                   else                    Message(asmr_e_invalid_register);                 end;     AS_SEPARATOR, AS_COMMA: ;    else     begin      writeln('looofasz');      Message(asmr_e_invalid_opcode_and_operand);      Consume(actasmtoken);     end;  end; { end case } end;  Procedure tm68kmotreader.BuildStringConstant(asciiz: boolean);  {*********************************************************************}  { PROCEDURE BuildStringConstant                                       }  {  Description: Takes care of a ASCII, or ASCIIZ directive.           }  {   asciiz: boolean -> if true then string will be null terminated.   }  {*********************************************************************}  { EXIT CONDITION:  On exit the routine should point to AS_SEPARATOR.  }  { On ENTRY: Token should point to AS_STRING                           }  {*********************************************************************}  var   expr: string;   errorflag : boolean;  begin      errorflag := FALSE;      Repeat        Case actasmtoken of          AS_STRING: begin                      expr:=actasmpattern;                      if asciiz then                       expr:=expr+#0;                      ConcatPasString(curlist,expr);                      Consume(AS_STRING);                    end;          AS_COMMA:  begin                       Consume(AS_COMMA);                     END;          AS_SEPARATOR: ;        else         begin          Consume(actasmtoken);          if not errorflag then           Message(asmr_e_invalid_string_expression);          errorflag := TRUE;         end;    end; { end case }   Until actasmtoken = AS_SEPARATOR;  end;  Procedure TM68kmotReader.BuildOpCode(instr:Tm68kinstruction);  {*********************************************************************}  { PROCEDURE BuildOpcode;                                              }  {  Description: Parses the intel opcode and operands, and writes it   }  {  in the TInstruction object.                                        }  {*********************************************************************}  { EXIT CONDITION:  On exit the routine should point to AS_SEPARATOR.  }  { On ENTRY: Token should point to AS_OPCODE                           }  {*********************************************************************}  var      operandnum : longint;  begin    { //  opcode                          // }    { allow for newline as in gas styled syntax }    { under DOS you get two AS_SEPARATOR !! }    while actasmtoken=AS_SEPARATOR do      Consume(AS_SEPARATOR);    if (actasmtoken <> AS_OPCODE) then    begin      Message(asmr_e_invalid_or_missing_opcode);      { error recovery }      While not (actasmtoken in [AS_SEPARATOR,AS_COMMA]) do         Consume(actasmtoken);      exit;    end    else    begin      Instr.opcode := findopcode(actasmpattern,instr.opsize);      Consume(AS_OPCODE);      { // Zero operand opcode ? // }      if actasmtoken = AS_SEPARATOR then        exit      else       operandnum := 1;    end;    While actasmtoken <> AS_SEPARATOR do    begin       case actasmtoken of         { //  Operand delimiter // }         AS_COMMA: begin                  if operandnum > Max_Operands then                    Message(asmr_e_too_many_operands)                  else                    Inc(operandnum);                  Consume(AS_COMMA);                end;         { // End of asm operands for this opcode // }         AS_SEPARATOR: ;       else         BuildOperand(Instr.Operands[operandnum] as tm68koperand);     end; { end case }    end; { end while }    instr.Ops:=operandnum;  end;    function tm68kmotreader.Assemble: tlinkedlist;      var        hl: tasmlabel;        instr : TM68kInstruction;      begin        Message(asmr_d_start_reading);        firsttoken := TRUE;        operandnum := 0;        { sets up all opcode and register tables in uppercase }        if not _asmsorted then          begin            SetupTables;            _asmsorted := TRUE;          end;        curlist:=TAsmList.Create;        { setup label linked list }        LocalLabelList:=TLocalLabelList.Create;        c:=current_scanner.asmgetchar;        gettoken;        while actasmtoken<>AS_END do          begin            case actasmtoken of              AS_LLABEL:                begin                  if CreateLocalLabel(actasmpattern,hl,true) then                    ConcatLabel(curlist,hl);                  Consume(AS_LLABEL);                end;              AS_LABEL:                begin                  { when looking for Pascal labels, these must }                  { be in uppercase.                           }                  if SearchLabel(upper(actasmpattern),hl,true) then                    ConcatLabel(curlist,hl)                  else                    Message1(asmr_e_unknown_label_identifier,actasmpattern);                  Consume(AS_LABEL);                end;              AS_DW:                begin                  Consume(AS_DW);                  BuildConstant($ffff);                end;              AS_DB:                begin                  Consume(AS_DB);                  BuildConstant($ff);                end;              AS_DD:                begin                  Consume(AS_DD);                  BuildConstant(longint($ffffffff));                end;              AS_XDEF:                begin                  Consume(AS_XDEF);                  if actasmtoken=AS_ID then                    ConcatPublic(curlist,actasmpattern);                  Consume(AS_ID);                  if actasmtoken<>AS_SEPARATOR then                   Consume(AS_SEPARATOR);                end;              AS_ALIGN:                begin                  Message(asmr_w_align_not_supported);                  while actasmtoken <> AS_SEPARATOR do                   Consume(actasmtoken);                end;              AS_OPCODE:                begin                  instr:=TM68kInstruction.Create(tm68koperand);                  BuildOpcode(instr);//                  instr.AddReferenceSizes;//                  instr.SetInstructionOpsize;//                  instr.CheckOperandSizes;                  if instr.labeled then                     instr.ConcatLabeledInstr(curlist)                  else begin                    instr.ConcatInstruction(curlist);                  end;                  instr.Free;{                  instr.init;                  BuildOpcode;                  instr.ops := operandnum;                  if instr.labeled then                    ConcatLabeledInstr(instr)                  else                    ConcatOpCode(instr);                  instr.done;}                end;              AS_SEPARATOR:                begin                  Consume(AS_SEPARATOR);                  { let us go back to the first operand }                  operandnum := 0;                end;              AS_END:                { end assembly block }                ;              else                begin                  Message(asmr_e_syntax_error);                  { error recovery }                  Consume(actasmtoken);                end;            end; { end case }          end; { end while }        { Check LocalLabelList }        LocalLabelList.CheckEmitted;        LocalLabelList.Free;        assemble:=curlist;        Message(asmr_d_finish_reading);      end;{*****************************************************************************                               Initialize*****************************************************************************}const  asmmode_m68k_mot_info : tasmmodeinfo =          (            id    : asmmode_m68k_mot;            idtxt : 'MOTOROLA';            casmreader : tm68kmotreader;          );  asmmode_m68k_standard_info : tasmmodeinfo =          (            id    : asmmode_standard;            idtxt : 'STANDARD';            casmreader : tm68kmotreader;          );begin  RegisterAsmMode(asmmode_m68k_mot_info);  RegisterAsmMode(asmmode_m68k_standard_info);end.
 |