mirror of
				https://gitlab.com/freepascal.org/fpc/source.git
				synced 2025-10-31 14:51:35 +01:00 
			
		
		
		
	
		
			
				
	
	
		
			2249 lines
		
	
	
		
			84 KiB
		
	
	
	
		
			ObjectPascal
		
	
	
	
	
	
			
		
		
	
	
			2249 lines
		
	
	
		
			84 KiB
		
	
	
	
		
			ObjectPascal
		
	
	
	
	
	
| {
 | |
|     $Id$
 | |
|     Copyright (c) 1997-98 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;
 | |
| {**********************************************************************}
 | |
| { 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
 | |
|   m68k,tree;
 | |
| 
 | |
|    function assemble: ptree;
 | |
| 
 | |
| const
 | |
|  { this variable is TRUE if the lookup tables have already been setup  }
 | |
|  { for fast access. On the first call to assemble the tables are setup }
 | |
|  { and stay set up.                                                    }
 | |
|  _asmsorted: boolean = FALSE;
 | |
|  firstreg       = R_D0;
 | |
|  lastreg        = R_FPSR;
 | |
| 
 | |
| type
 | |
|  tiasmops = array[firstop..lastop] of string[7];
 | |
|  piasmops = ^tiasmops;
 | |
| 
 | |
|  tasmkeyword = string[6];
 | |
| 
 | |
| var
 | |
|  { sorted tables of opcodes }
 | |
|  iasmops: piasmops;
 | |
|  { uppercased tables of registers }
 | |
|  iasmregs: array[firstreg..lastreg] of string[6];
 | |
| 
 | |
| 
 | |
| Implementation
 | |
| 
 | |
| uses
 | |
|   files,globals,systems,AsmUtils,strings,hcodegen,scanner,aasm,
 | |
|   cobjects,verbose,symtable;
 | |
| 
 | |
| 
 | |
| type
 | |
|  tmotorolatoken = (
 | |
|    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);
 | |
| 
 | |
| 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');
 | |
| 
 | |
| 
 | |
| const
 | |
|   newline = #10;
 | |
|   firsttoken : boolean = TRUE;
 | |
|   operandnum : byte = 0;
 | |
| var
 | |
|  p : paasmoutput;
 | |
|  actasmtoken: tmotorolatoken;
 | |
|  actasmpattern: string;
 | |
|  c: char;
 | |
|  Instr: TInstruction;
 | |
|  labellist: TAsmLabelList;
 | |
|  old_exit : pointer;
 | |
| 
 | |
|    Procedure SetupTables;
 | |
|    { creates uppercased symbol tables for speed access }
 | |
|    var
 | |
|      i: tasmop;
 | |
|      j: tregister;
 | |
|    Begin
 | |
|      Message(assem_d_creating_lookup_tables);
 | |
|      { opcodes }
 | |
|      new(iasmops);
 | |
|      for i:=firstop to lastop do
 | |
|       iasmops^[i] := upper(mot_op2str[i]);
 | |
|      { opcodes }
 | |
|      for j:=firstreg to lastreg do
 | |
|       iasmregs[j] := upper(mot_reg2str[j]);
 | |
|    end;
 | |
| 
 | |
| 
 | |
|   {---------------------------------------------------------------------}
 | |
|   {                     Routines for the tokenizing                     }
 | |
|   {---------------------------------------------------------------------}
 | |
| 
 | |
| 
 | |
|    function is_asmopcode(s: string):Boolean;
 | |
|   {*********************************************************************}
 | |
|   { FUNCTION is_asmopcode(s: string):Boolean                            }
 | |
|   {  Description: Determines if the s string is a valid opcode          }
 | |
|   {  if so returns TRUE otherwise returns FALSE.                        }
 | |
|   {  Remark: Suffixes are also checked, as long as they are valid.      }
 | |
|   {*********************************************************************}
 | |
|    var
 | |
|     i: tasmop;
 | |
|     j: byte;
 | |
|    Begin
 | |
|      is_asmopcode := FALSE;
 | |
|      { first of all we remove the suffix }
 | |
|      j:=pos('.',s);
 | |
|      if j<>0 then
 | |
|       delete(s,j,2);
 | |
|      for i:=firstop to lastop do
 | |
|      begin
 | |
|        if  s = iasmops^[i] then
 | |
|        begin
 | |
|           is_asmopcode:=TRUE;
 | |
|           exit;
 | |
|        end;
 | |
|      end;
 | |
|    end;
 | |
| 
 | |
| 
 | |
| 
 | |
|    Procedure is_asmdirective(const s: string; var token: tmotorolatoken);
 | |
|   {*********************************************************************}
 | |
|   { FUNCTION is_asmdirective(s: string; var token: tinteltoken):Boolean }
 | |
|   {  Description: Determines if the s string is a valid directive       }
 | |
|   { (an operator can occur in operand fields, while a directive cannot) }
 | |
|   {  if so returns the directive token, otherwise does not change token.}
 | |
|   {*********************************************************************}
 | |
|    var
 | |
|     i:byte;
 | |
|    Begin
 | |
|      for i:=0 to _count_asmdirectives do
 | |
|      begin
 | |
|         if s=_asmdirectives[i] then
 | |
|         begin
 | |
|            token := tmotorolatoken(longint(firstdirective)+i);
 | |
|            exit;
 | |
|         end;
 | |
|      end;
 | |
|    end;
 | |
| 
 | |
| 
 | |
|    Procedure is_register(const s: string; var token: tmotorolatoken);
 | |
|   {*********************************************************************}
 | |
|   { PROCEDURE is_register(s: string; var token: tinteltoken);           }
 | |
|   {  Description: Determines if the s string is a valid register, if    }
 | |
|   {  so return token equal to A_REGISTER, otherwise does not change token}
 | |
|   {*********************************************************************}
 | |
|    Var
 | |
|     i: tregister;
 | |
|    Begin
 | |
|      for i:=firstreg to lastreg do
 | |
|      begin
 | |
|       if s=iasmregs[i] then
 | |
|       begin
 | |
|         token := AS_REGISTER;
 | |
|         exit;
 | |
|       end;
 | |
|      end;
 | |
|      { take care of other name for sp }
 | |
|      if s = 'A7' then
 | |
|      begin
 | |
|       token:=AS_REGISTER;
 | |
|       exit;
 | |
|      end;
 | |
|    end;
 | |
| 
 | |
| 
 | |
| 
 | |
|   Function GetToken: tmotorolatoken;
 | |
|   {*********************************************************************}
 | |
|   { FUNCTION GetToken: tinteltoken;                                     }
 | |
|   {  Description: This routine returns intel assembler tokens and       }
 | |
|   {  does some minor syntax error checking.                             }
 | |
|   {*********************************************************************}
 | |
|   var
 | |
|    j: integer;
 | |
|    token: tmotorolatoken;
 | |
|    forcelabel: boolean;
 | |
|    errorflag : boolean;
 | |
|   begin
 | |
|     errorflag := FALSE;
 | |
|     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;
 | |
|     current_scanner^.gettokenpos;
 | |
|     { Possiblities for first token in a statement:                }
 | |
|     {   Local Label, Label, Directive, Prefix or Opcode....       }
 | |
|     if firsttoken and not (c in [newline,#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;
 | |
|            gettoken := token;
 | |
|            exit;
 | |
|       end;
 | |
| 
 | |
|       { Are we trying to create an identifier with }
 | |
|       { an at-sign...?                             }
 | |
|       if forcelabel then
 | |
|        Message(assem_e_none_label_contain_at);
 | |
| 
 | |
|       If is_asmopcode(actasmpattern) then
 | |
|       Begin
 | |
|        gettoken := AS_OPCODE;
 | |
|        exit;
 | |
|       end;
 | |
|       is_asmdirective(actasmpattern, token);
 | |
|       if (token <> AS_NONE) then
 | |
|       Begin
 | |
|         gettoken := token;
 | |
|         exit
 | |
|       end
 | |
|       else
 | |
|       begin
 | |
|          gettoken := AS_NONE;
 | |
|          Message1(assem_e_invalid_operand,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);
 | |
|                              gettoken := 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
 | |
|                              Begin
 | |
|                                     gettoken := AS_OPCODE;
 | |
|                                     exit;
 | |
|                              end;
 | |
|                              is_register(actasmpattern, token);
 | |
|                              {is_asmoperator(actasmpattern,token);}
 | |
|                              is_asmdirective(actasmpattern,token);
 | |
|                              { if found }
 | |
|                              if (token <> AS_NONE) then
 | |
|                              begin
 | |
|                                gettoken := token;
 | |
|                                exit;
 | |
|                              end
 | |
|                              { this is surely an identifier }
 | |
|                              else
 | |
|                                token := AS_ID;
 | |
|                              gettoken := token;
 | |
|                              exit;
 | |
|                           end;
 | |
|            { override operator... not supported }
 | |
|            '&':       begin
 | |
|                          c:=current_scanner^.asmgetchar;
 | |
|                          gettoken := AS_AND;
 | |
|                       end;
 | |
|            { string or character }
 | |
|            '''' :
 | |
|                       begin
 | |
|                          actasmpattern:='';
 | |
|                          while true do
 | |
|                          begin
 | |
|                            if c = '''' then
 | |
|                            begin
 | |
|                               c:=current_scanner^.asmgetchar;
 | |
|                               if c=newline 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=newline 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=newline 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;
 | |
|                    gettoken := 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;
 | |
|                    gettoken := AS_HEXNUM;
 | |
|                    exit;
 | |
|                   end;
 | |
|            ',' : begin
 | |
|                    gettoken := AS_COMMA;
 | |
|                    c:=current_scanner^.asmgetchar;
 | |
|                    exit;
 | |
|                  end;
 | |
|            '(' : begin
 | |
|                    gettoken := AS_LPAREN;
 | |
|                    c:=current_scanner^.asmgetchar;
 | |
|                    exit;
 | |
|                  end;
 | |
|            ')' : begin
 | |
|                    gettoken := AS_RPAREN;
 | |
|                    c:=current_scanner^.asmgetchar;
 | |
|                    exit;
 | |
|                  end;
 | |
|            ':' : begin
 | |
|                    gettoken := AS_COLON;
 | |
|                    c:=current_scanner^.asmgetchar;
 | |
|                    exit;
 | |
|                  end;
 | |
| {           '.' : begin
 | |
|                    gettoken := AS_DOT;
 | |
|                    c:=current_scanner^.asmgetchar;
 | |
|                    exit;
 | |
|                  end; }
 | |
|            '+' : begin
 | |
|                    gettoken := AS_PLUS;
 | |
|                    c:=current_scanner^.asmgetchar;
 | |
|                    exit;
 | |
|                  end;
 | |
|            '-' : begin
 | |
|                    gettoken := AS_MINUS;
 | |
|                    c:=current_scanner^.asmgetchar;
 | |
|                    exit;
 | |
|                  end;
 | |
|            '*' : begin
 | |
|                    gettoken := AS_STAR;
 | |
|                    c:=current_scanner^.asmgetchar;
 | |
|                    exit;
 | |
|                  end;
 | |
|            '/' : begin
 | |
|                    gettoken := AS_SLASH;
 | |
|                    c:=current_scanner^.asmgetchar;
 | |
|                    exit;
 | |
|                  end;
 | |
|            '<' : begin
 | |
|                    c := current_scanner^.asmgetchar;
 | |
|                    { invalid characters }
 | |
|                    if c <> '<' then
 | |
|                     Message(assem_e_invalid_char_smaller);
 | |
|                    { still assume << }
 | |
|                    gettoken := AS_SHL;
 | |
|                    c := current_scanner^.asmgetchar;
 | |
|                    exit;
 | |
|                  end;
 | |
|            '>' : begin
 | |
|                    c := current_scanner^.asmgetchar;
 | |
|                    { invalid characters }
 | |
|                    if c <> '>' then
 | |
|                     Message(assem_e_invalid_char_greater);
 | |
|                    { still assume << }
 | |
|                    gettoken := AS_SHR;
 | |
|                    c := current_scanner^.asmgetchar;
 | |
|                    exit;
 | |
|                  end;
 | |
|            '|' : begin
 | |
|                    gettoken := AS_OR;
 | |
|                    c := current_scanner^.asmgetchar;
 | |
|                    exit;
 | |
|                  end;
 | |
|            '^' : begin
 | |
|                   gettoken := AS_XOR;
 | |
|                   c := current_scanner^.asmgetchar;
 | |
|                   exit;
 | |
|                  end;
 | |
|            '#' : begin
 | |
|                   gettoken:=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;
 | |
|                    gettoken := 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;
 | |
|                         gettoken := AS_INTNUM;
 | |
|                         exit;
 | |
|                      end;
 | |
|          ';' : begin
 | |
|                   repeat
 | |
|                      c:=current_scanner^.asmgetchar;
 | |
|                   until c=newline;
 | |
|                   firsttoken := TRUE;
 | |
|                   gettoken:=AS_SEPARATOR;
 | |
|                end;
 | |
| 
 | |
|          '{',#13,newline : begin
 | |
|                             c:=current_scanner^.asmgetchar;
 | |
|                             firsttoken := TRUE;
 | |
|                             gettoken:=AS_SEPARATOR;
 | |
|                            end;
 | |
|             else
 | |
|              Begin
 | |
|                Message(scan_f_illegal_char);
 | |
|              end;
 | |
| 
 | |
|       end; { end case }
 | |
|     end; { end else if }
 | |
|   end;
 | |
| 
 | |
| 
 | |
|   {---------------------------------------------------------------------}
 | |
|   {                     Routines for the parsing                        }
 | |
|   {---------------------------------------------------------------------}
 | |
| 
 | |
|      procedure consume(t : tmotorolatoken);
 | |
| 
 | |
|      begin
 | |
|        if t<>actasmtoken then
 | |
|         Message(assem_e_syntax_error);
 | |
|        actasmtoken:=gettoken;
 | |
|        { if the token must be ignored, then }
 | |
|        { get another token to parse.        }
 | |
|        if actasmtoken = AS_NONE then
 | |
|           actasmtoken := gettoken;
 | |
|       end;
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
|    function findregister(const s : string): tregister;
 | |
|   {*********************************************************************}
 | |
|   { FUNCTION findregister(s: string):tasmop;                            }
 | |
|   {  Description: Determines if the s string is a valid register,       }
 | |
|   {  if so returns correct tregister token, or R_NO if not found.       }
 | |
|   {*********************************************************************}
 | |
|    var
 | |
|     i: tregister;
 | |
|    begin
 | |
|      findregister := R_NO;
 | |
|      for i:=firstreg to lastreg do
 | |
|        if s = iasmregs[i] then
 | |
|        Begin
 | |
|          findregister := i;
 | |
|          exit;
 | |
|        end;
 | |
|     if s = 'A7' then
 | |
|     Begin
 | |
|       findregister := R_SP;
 | |
|       exit;
 | |
|     end;
 | |
|    end;
 | |
| 
 | |
| 
 | |
|    function findopcode(s: string): tasmop;
 | |
|   {*********************************************************************}
 | |
|   { FUNCTION findopcode(s: string): tasmop;                             }
 | |
|   {  Description: Determines if the s string is a valid opcode          }
 | |
|   {  if so returns correct tasmop token.                                }
 | |
|   {*********************************************************************}
 | |
|    var
 | |
|     i: tasmop;
 | |
|     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 stropsize size is used to }
 | |
|        { determine the size of the operands.             }
 | |
|        'B': instr.stropsize := S_B;
 | |
|        'W': instr.stropsize := S_W;
 | |
|        'L': instr.stropsize := S_L;
 | |
|        'S': instr.stropsize := S_FS;
 | |
|        'D': instr.stropsize := S_FL;
 | |
|        'X': instr.stropsize := S_FX;
 | |
|        else
 | |
|         Message1(assem_e_invalid_opcode,s);
 | |
|        end;
 | |
|        { delete everything starting from dot }
 | |
|        delete(s,j,length(s));
 | |
|      end;
 | |
|      for i:=firstop to lastop do
 | |
|        if  s = iasmops^[i] then
 | |
|        begin
 | |
|           findopcode:=i;
 | |
|           exit;
 | |
|        end;
 | |
|    end;
 | |
| 
 | |
|   Procedure InitAsmRef(var instr: TInstruction);
 | |
|   {*********************************************************************}
 | |
|   {  Description: This routine first check if the instruction is of     }
 | |
|   {  type OPR_NONE, or OPR_REFERENCE , if not it gives out an error.    }
 | |
|   {  If the operandtype = OPR_NONE or <> OPR_REFERENCE then it sets up  }
 | |
|   {  the operand type to OPR_REFERENCE, as well as setting up the ref   }
 | |
|   {  to point to the default segment.                                   }
 | |
|   {*********************************************************************}
 | |
|    Begin
 | |
|      With instr do
 | |
|      Begin
 | |
|         case operands[operandnum].operandtype of
 | |
|           OPR_REFERENCE: exit;
 | |
|           OPR_NONE: ;
 | |
|         else
 | |
|           Message(assem_e_invalid_operand_type);
 | |
|         end;
 | |
|         operands[operandnum].ref.direction := dir_none;
 | |
|         operands[operandnum].operandtype := OPR_REFERENCE;
 | |
|         operands[operandnum].ref.segment := R_DEFAULT_SEG;
 | |
|      end;
 | |
|    end;
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
|   Function CalculateExpression(expression: string): longint;
 | |
|   var
 | |
|     expr: TExprParse;
 | |
|   Begin
 | |
|    expr.Init;
 | |
|    CalculateExpression := expr.Evaluate(expression);
 | |
|    expr.Done;
 | |
|   end;
 | |
| 
 | |
| 
 | |
|   Procedure ConcatOpCode(var instr: TInstruction);
 | |
|   var
 | |
|     fits : boolean;
 | |
|     i: longint;
 | |
|     opsize: topsize;
 | |
|     optyp1, optyp2, optyp3: longint;
 | |
|     instruc: tasmop;
 | |
|     op: tasmop;
 | |
|   Begin
 | |
|      fits := FALSE;
 | |
|     { setup specific instructions for first pass }
 | |
|     instruc := instr.getinstruction;
 | |
| 
 | |
|     { Setup special operands }
 | |
|     { Convert to general form as to conform to the m68k opcode table }
 | |
|     if (instruc = A_ADDA) or (instruc = A_ADDI)
 | |
|        then instruc := A_ADD
 | |
|     else
 | |
|     { CMPM excluded because of GAS v1.34 BUG }
 | |
|     if (instruc = A_CMPA) or
 | |
|        (instruc = A_CMPI) then
 | |
|        instruc := A_CMP
 | |
|     else
 | |
|     if instruc = A_EORI then
 | |
|       instruc := A_EOR
 | |
|     else
 | |
|     if instruc = A_MOVEA then
 | |
|      instruc := A_MOVE
 | |
|     else
 | |
|     if instruc = A_ORI then
 | |
|       instruc := A_OR
 | |
|     else
 | |
|     if (instruc = A_SUBA) or (instruc = A_SUBI) then
 | |
|       instruc :=  A_SUB;
 | |
| 
 | |
|     { Setup operand types }
 | |
| 
 | |
| (*
 | |
|     in instruc <> A_MOVEM then
 | |
|     Begin
 | |
| 
 | |
|       while not(fits) do
 | |
|         begin
 | |
|          { set the instruction cache, if the instruction }
 | |
|          { occurs the first time                         }
 | |
|          if (it[i].i=instruc) and (ins_cache[instruc]=-1) then
 | |
|              ins_cache[instruc]:=i;
 | |
| 
 | |
|          if (it[i].i=instruc) and (instr.numops=it[i].ops) then
 | |
|          begin
 | |
|             { first fit }
 | |
|            case instr.numops of
 | |
|              0 : begin
 | |
|                    fits:=true;
 | |
|                    break;
 | |
|                 end;
 | |
|             1 :
 | |
|                 Begin
 | |
|                   if (optyp1 and it[i].o1)<>0 then
 | |
|                   Begin
 | |
|                     fits:=true;
 | |
|                      break;
 | |
|                   end;
 | |
|                 end;
 | |
|             2 : if ((optyp1 and it[i].o1)<>0) and
 | |
|                  ((optyp2 and it[i].o2)<>0) then
 | |
|                  Begin
 | |
|                        fits:=true;
 | |
|                        break;
 | |
|                  end
 | |
|             3 : if ((optyp1 and it[i].o1)<>0) and
 | |
|                  ((optyp2 and it[i].o2)<>0) and
 | |
|                  ((optyp3 and it[i].o3)<>0) then
 | |
|                  Begin
 | |
|                    fits:=true;
 | |
|                    break;
 | |
|                  end;
 | |
|            end; { end case }
 | |
|         end; { endif }
 | |
|         if it[i].i=A_NONE then
 | |
|         begin
 | |
|           { NO MATCH! }
 | |
|           Message(assem_e_invalid_combination_opcode_and_operand);
 | |
|           exit;
 | |
|         end;
 | |
|         inc(i);
 | |
|        end; { end while }
 | |
|              *)
 | |
|   fits:=TRUE;
 | |
| 
 | |
|   { We add the opcode to the opcode linked list }
 | |
|   if fits then
 | |
|   Begin
 | |
|     case instr.numops of
 | |
| 
 | |
|      0:
 | |
|         if instr.stropsize <> S_NO then
 | |
|           p^.concat(new(pai68k,op_none(instruc,instr.stropsize)))
 | |
|         else
 | |
|           p^.concat(new(pai68k,op_none(instruc,S_NO)));
 | |
|      1: Begin
 | |
|           case instr.operands[1].operandtype of
 | |
|            OPR_SYMBOL: Begin
 | |
|                              p^.concat(new(pai68k,op_ref(instruc,
 | |
|                                instr.stropsize, newreference(instr.operands[1].ref))));
 | |
|                          end;
 | |
|            OPR_CONSTANT: Begin
 | |
|                              p^.concat(new(pai68k,op_const(instruc,
 | |
|                                instr.stropsize, instr.operands[1].val)));
 | |
|                          end;
 | |
|            OPR_REGISTER:  p^.concat(new(pai68k,op_reg(instruc,
 | |
|                             instr.stropsize,instr.operands[1].reg)));
 | |
|            OPR_REFERENCE:
 | |
|                           if instr.stropsize <> S_NO then
 | |
|                           Begin
 | |
|                            p^.concat(new(pai68k,op_ref(instruc,
 | |
|                             instr.stropsize,newreference(instr.operands[1].ref))));
 | |
|                           end
 | |
|                           else
 | |
|                           Begin
 | |
|                               { special jmp and call case with }
 | |
|                               { symbolic references.           }
 | |
|                               if instruc in [A_BSR,A_JMP,A_JSR,A_BRA,A_PEA] then
 | |
|                               Begin
 | |
|                                 p^.concat(new(pai68k,op_ref(instruc,
 | |
|                                   S_NO,newreference(instr.operands[1].ref))));
 | |
|                               end
 | |
|                               else
 | |
|                                 Message(assem_e_invalid_opcode_and_operand);
 | |
|                           end;
 | |
|            OPR_NONE: Begin
 | |
|                        Message(assem_f_internal_error_in_concatopcode);
 | |
|                      end;
 | |
|           else
 | |
|            Begin
 | |
|              Message(assem_f_internal_error_in_concatopcode);
 | |
|            end;
 | |
|           end;
 | |
|         end;
 | |
|      2:
 | |
|         Begin
 | |
|                 With instr do
 | |
|                 Begin
 | |
|                 { source }
 | |
|                   case operands[1].operandtype of
 | |
|                   { reg,reg     }
 | |
|                   { reg,ref     }
 | |
|                    OPR_REGISTER:
 | |
|                      Begin
 | |
|                        case operands[2].operandtype of
 | |
|                          OPR_REGISTER:
 | |
|                             Begin
 | |
|                                p^.concat(new(pai68k,op_reg_reg(instruc,
 | |
|                                stropsize,operands[1].reg,operands[2].reg)));
 | |
|                             end;
 | |
|                          OPR_REFERENCE:
 | |
|                                   p^.concat(new(pai68k,op_reg_ref(instruc,
 | |
|                                   stropsize,operands[1].reg,newreference(operands[2].ref))));
 | |
|                        else { else case }
 | |
|                          Begin
 | |
|                            Message(assem_f_internal_error_in_concatopcode);
 | |
|                          end;
 | |
|                        end; { end inner case }
 | |
|                      end;
 | |
|                   { reglist, ref }
 | |
|                    OPR_REGLIST:
 | |
|                           Begin
 | |
|                             case operands[2].operandtype of
 | |
|                               OPR_REFERENCE :
 | |
|                                   p^.concat(new(pai68k,op_reglist_ref(instruc,
 | |
|                                   stropsize,operands[1].list,newreference(operands[2].ref))));
 | |
|                             else
 | |
|                              Begin
 | |
|                                Message(assem_f_internal_error_in_concatopcode);
 | |
|                              end;
 | |
|                             end; { end case }
 | |
|                           end;
 | |
| 
 | |
|                   { const,reg   }
 | |
|                   { const,const }
 | |
|                   { const,ref   }
 | |
|                    OPR_CONSTANT:
 | |
|                       case instr.operands[2].operandtype of
 | |
|                       { constant, constant does not have a specific size. }
 | |
|                         OPR_CONSTANT:
 | |
|                            p^.concat(new(pai68k,op_const_const(instruc,
 | |
|                            S_NO,operands[1].val,operands[2].val)));
 | |
|                         OPR_REFERENCE:
 | |
|                            Begin
 | |
|                                  p^.concat(new(pai68k,op_const_ref(instruc,
 | |
|                                  stropsize,operands[1].val,
 | |
|                                  newreference(operands[2].ref))))
 | |
|                            end;
 | |
|                         OPR_REGISTER:
 | |
|                            Begin
 | |
|                                  p^.concat(new(pai68k,op_const_reg(instruc,
 | |
|                                  stropsize,operands[1].val,
 | |
|                                  operands[2].reg)))
 | |
|                            end;
 | |
|                       else
 | |
|                          Begin
 | |
|                            Message(assem_f_internal_error_in_concatopcode);
 | |
|                          end;
 | |
|                       end; { end case }
 | |
|                    { ref,reg     }
 | |
|                    { ref,ref     }
 | |
|                    OPR_REFERENCE:
 | |
|                       case instr.operands[2].operandtype of
 | |
|                          OPR_REGISTER:
 | |
|                             Begin
 | |
|                               p^.concat(new(pai68k,op_ref_reg(instruc,
 | |
|                                stropsize,newreference(operands[1].ref),
 | |
|                                operands[2].reg)));
 | |
|                             end;
 | |
|                          OPR_REGLIST:
 | |
|                             Begin
 | |
|                               p^.concat(new(pai68k,op_ref_reglist(instruc,
 | |
|                                stropsize,newreference(operands[1].ref),
 | |
|                                operands[2].list)));
 | |
|                             end;
 | |
|                          OPR_REFERENCE: { special opcodes }
 | |
|                             p^.concat(new(pai68k,op_ref_ref(instruc,
 | |
|                             stropsize,newreference(operands[1].ref),
 | |
|                             newreference(operands[2].ref))));
 | |
|                       else
 | |
|                          Begin
 | |
|                            Message(assem_f_internal_error_in_concatopcode);
 | |
|                          end;
 | |
|                    end; { end inner case }
 | |
|                   end; { end case }
 | |
|                 end; { end with }
 | |
|         end;
 | |
|      3: Begin
 | |
|            if (instruc = A_DIVSL) or (instruc = A_DIVUL) or (instruc = A_MULU)
 | |
|            or (instruc = A_MULS) or (instruc = A_DIVS) or (instruc = A_DIVU) then
 | |
|            Begin
 | |
|              if (instr.operands[1].operandtype <> OPR_REGISTER)
 | |
|              or (instr.operands[2].operandtype <> OPR_REGISTER)
 | |
|              or (instr.operands[3].operandtype <> OPR_REGISTER) then
 | |
|              Begin
 | |
|                Message(assem_f_internal_error_in_concatopcode);
 | |
|              end
 | |
|              else
 | |
|              Begin
 | |
|                p^.concat(new(pai68k, op_reg_reg_reg(instruc,instr.stropsize,
 | |
|                  instr.operands[1].reg,instr.operands[2].reg,instr.operands[3].reg)));
 | |
|              end;
 | |
|            end
 | |
|            else
 | |
|             Message(assem_e_unsupported_opcode);
 | |
|         end;
 | |
|   end; { end case }
 | |
|  end;
 | |
|  end;
 | |
| 
 | |
| 
 | |
|     Procedure ConcatLabeledInstr(var instr: TInstruction);
 | |
|     Begin
 | |
|        if ((instr.getinstruction >= A_BCC) and (instr.getinstruction <= A_BVS))
 | |
|        or (instr.getinstruction = A_BRA) or (instr.getinstruction = A_BSR)
 | |
|        or (instr.getinstruction = A_JMP) or (instr.getinstruction = A_JSR)
 | |
|        or ((instr.getinstruction >= A_FBEQ) and (instr.getinstruction <= A_FBNGLE))
 | |
|        then
 | |
|        Begin
 | |
|         if instr.numops > 2 then
 | |
|           Message(assem_e_invalid_opcode)
 | |
|         else if instr.operands[1].operandtype <> OPR_LABINSTR then
 | |
|           Message(assem_e_invalid_opcode)
 | |
|         else if (instr.operands[1].operandtype = OPR_LABINSTR) and
 | |
|          (instr.numops = 1) then
 | |
|            if assigned(instr.operands[1].hl) then
 | |
|             ConcatLabel(p,instr.getinstruction, instr.operands[1].hl)
 | |
|            else
 | |
|             Message(assem_f_internal_error_in_findtype);
 | |
|        end
 | |
|        else
 | |
|        if ((instr.getinstruction >= A_DBCC) and (instr.getinstruction <= A_DBF))
 | |
|        or ((instr.getinstruction >= A_FDBEQ) and (instr.getinstruction <= A_FBDNGLE)) then
 | |
|        begin
 | |
|          p^.concat(new(pai_labeled,init_reg(instr.getinstruction,instr.operands[2].hl,
 | |
|            instr.operands[1].reg)));
 | |
|        end
 | |
|        else
 | |
|         Message(assem_e_invalid_operand);
 | |
|     end;
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
|     Function BuildExpression: 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;
 | |
|       tempstr: string;
 | |
|       l : longint;
 | |
|       errorflag: boolean;
 | |
|   Begin
 | |
|     errorflag := FALSE;
 | |
|     expr := '';
 | |
|     tempstr := '';
 | |
|     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 NOT SearchIConstant(actasmpattern,l) then
 | |
|                   Begin
 | |
|                     Message1(assem_e_invalid_const_symbol,actasmpattern);
 | |
|                     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 := BinaryToDec(actasmpattern);
 | |
|                       if tempstr = '' then
 | |
|                        Message(assem_f_error_converting_bin);
 | |
|                       expr:=expr+tempstr;
 | |
|                       Consume(AS_BINNUM);
 | |
|                  end;
 | |
| 
 | |
|       AS_HEXNUM: Begin
 | |
|                     tempstr := HexToDec(actasmpattern);
 | |
|                     if tempstr = '' then
 | |
|                      Message(assem_f_error_converting_hex);
 | |
|                     expr:=expr+tempstr;
 | |
|                     Consume(AS_HEXNUM);
 | |
|                 end;
 | |
|       AS_OCTALNUM: Begin
 | |
|                     tempstr := OctalToDec(actasmpattern);
 | |
|                     if tempstr = '' then
 | |
|                      Message(assem_f_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(assem_e_invalid_constant_expression);
 | |
|           { consume tokens until we find COMMA or SEPARATOR }
 | |
|           Consume(actasmtoken);
 | |
|           errorflag := TRUE;
 | |
|         End;
 | |
|       end;
 | |
|     Until false;
 | |
|   end;
 | |
| 
 | |
| 
 | |
|   Procedure 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;
 | |
|       tempstr: string;
 | |
|       r : extended;
 | |
|       code : word;
 | |
|       negativ : boolean;
 | |
|       errorflag: boolean;
 | |
|   Begin
 | |
|     errorflag := FALSE;
 | |
|     Repeat
 | |
|     negativ:=false;
 | |
|     expr := '';
 | |
|     tempstr := '';
 | |
|     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(assem_w_float_bin_ignored);
 | |
|                       Consume(AS_BINNUM);
 | |
|                  end;
 | |
| 
 | |
|       AS_HEXNUM: Begin
 | |
|                       { checking for real constants with this should use  }
 | |
|                       { real DECODING otherwise the compiler will crash!  }
 | |
|                     Message(assem_w_float_hex_ignored);
 | |
|                     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(assem_w_float_octal_ignored);
 | |
|                     Consume(AS_OCTALNUM);
 | |
|                   end;
 | |
|          else
 | |
|            Begin
 | |
|              { only write error once. }
 | |
|              if not errorflag then
 | |
|               Message(assem_e_invalid_real_const);
 | |
|              { 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(assem_e_invalid_real_const);
 | |
|                ConcatRealConstant(p,r,typ);
 | |
|             End
 | |
|           else
 | |
|             Begin
 | |
|               ConcatRealConstant(p,r,typ);
 | |
|             End;
 | |
|         end
 | |
|       else
 | |
|         Message(assem_e_invalid_real_const);
 | |
|     Until actasmtoken=AS_SEPARATOR;
 | |
|   end;
 | |
| 
 | |
| 
 | |
| 
 | |
|   Procedure BuildScaling(Var instr: TInstruction);
 | |
|   {*********************************************************************}
 | |
|   {  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 (instr.operands[operandnum].ref.scalefactor <> 0)
 | |
|      and (instr.operands[operandnum].ref.scalefactor <> 1) then
 | |
|       Message(assem_f_internal_error_in_buildscale);
 | |
|      case actasmtoken of
 | |
|         AS_INTNUM: str := actasmpattern;
 | |
|         AS_HEXNUM: str := HexToDec(actasmpattern);
 | |
|         AS_BINNUM: str := BinaryToDec(actasmpattern);
 | |
|         AS_OCTALNUM: str := OctalToDec(actasmpattern);
 | |
|      else
 | |
|         Message(assem_e_syntax_error);
 | |
|      end;
 | |
|      val(str, l, code);
 | |
|      if code <> 0 then
 | |
|       Message(assem_e_invalid_scaling_factor);
 | |
|      if ((l = 2) or (l = 4) or (l = 8) or (l = 1)) and (code = 0) then
 | |
|      begin
 | |
|         instr.operands[operandnum].ref.scalefactor := l;
 | |
|      end
 | |
|      else
 | |
|      Begin
 | |
|         Message(assem_e_invalid_scaling_value);
 | |
|         instr.operands[operandnum].ref.scalefactor := 0;
 | |
|      end;
 | |
|      if instr.operands[operandnum].ref.index = R_NO then
 | |
|      Begin
 | |
|         Message(assem_e_scaling_value_only_allowed_with_index);
 | |
|         instr.operands[operandnum].ref.scalefactor := 0;
 | |
|      end;
 | |
|     { Consume the scaling number }
 | |
|     Consume(actasmtoken);
 | |
|     if actasmtoken = AS_RPAREN then
 | |
|         Consume(AS_RPAREN)
 | |
|     else
 | |
|        Message(assem_e_invalid_scaling_value);
 | |
|     { // .Field.Field ... or separator/comma // }
 | |
|     if actasmtoken in [AS_COMMA,AS_SEPARATOR] then
 | |
|     Begin
 | |
|     end
 | |
|     else
 | |
|      Message(assem_e_syntax_error);
 | |
|   end;
 | |
| 
 | |
| 
 | |
|   Function BuildRefExpression: 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 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(assem_e_parenthesis_are_not_allowed);
 | |
|                   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
 | |
|                     Message1(assem_e_invalid_const_symbol,actasmpattern);
 | |
|                     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 := BinaryToDec(actasmpattern);
 | |
|                       if tempstr = '' then
 | |
|                        Message(assem_f_error_converting_bin);
 | |
|                       expr:=expr+tempstr;
 | |
|                       Consume(AS_BINNUM);
 | |
|                  end;
 | |
| 
 | |
|       AS_HEXNUM: Begin
 | |
|                     tempstr := HexToDec(actasmpattern);
 | |
|                     if tempstr = '' then
 | |
|                      Message(assem_f_error_converting_hex);
 | |
|                     expr:=expr+tempstr;
 | |
|                     Consume(AS_HEXNUM);
 | |
|                 end;
 | |
|       AS_OCTALNUM: Begin
 | |
|                     tempstr := OctalToDec(actasmpattern);
 | |
|                     if tempstr = '' then
 | |
|                      Message(assem_f_error_converting_octal);
 | |
|                     expr:=expr+tempstr;
 | |
|                     Consume(AS_OCTALNUM);
 | |
|                   end;
 | |
|       else
 | |
|         Begin
 | |
|           { write error only once. }
 | |
|           if not errorflag then
 | |
|            Message(assem_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 BuildReference(var Instr: TInstruction);
 | |
|   {*********************************************************************}
 | |
|   { 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.                               }
 | |
|   {*********************************************************************}
 | |
|   var
 | |
|     l:longint;
 | |
|     code: integer;
 | |
|     str: string;
 | |
|   Begin
 | |
|      Consume(AS_LPAREN);
 | |
|      Case actasmtoken of
 | |
|         { // (reg ... // }
 | |
|         AS_REGISTER: Begin
 | |
|                         instr.operands[operandnum].ref.base :=
 | |
|                            findregister(actasmpattern);
 | |
|                         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 (instr.operands[operandnum].ref.direction <> dir_none) then
 | |
|                               Message(assem_e_no_inc_and_dec_together)
 | |
|                              else
 | |
|                                instr.operands[operandnum].ref.direction := dir_inc;
 | |
|                              Consume(AS_PLUS);
 | |
|                            end;
 | |
|                            if not (actasmtoken in [AS_COMMA,AS_SEPARATOR]) then
 | |
|                              Begin
 | |
|                                Message(assem_e_invalid_reference);
 | |
|                                { error recovery ... }
 | |
|                                while actasmtoken <> AS_SEPARATOR do
 | |
|                                   Consume(actasmtoken);
 | |
|                              end;
 | |
|                              exit;
 | |
|                          end;
 | |
|                        { // (reg,reg .. // }
 | |
|                        Consume(AS_COMMA);
 | |
|                        if actasmtoken = AS_REGISTER then
 | |
|                        Begin
 | |
|                          instr.operands[operandnum].ref.index :=
 | |
|                            findregister(actasmpattern);
 | |
|                          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(assem_e_invalid_reference);
 | |
|                                   while actasmtoken <> AS_SEPARATOR do
 | |
|                                     Consume(actasmtoken);
 | |
|                                 end;
 | |
|                                 exit;
 | |
|                               end;
 | |
|                            AS_STAR:
 | |
|                               Begin
 | |
|                                 BuildScaling(instr);
 | |
|                               end;
 | |
|                          else
 | |
|                            Begin
 | |
|                              Message(assem_e_invalid_reference_syntax);
 | |
|                              while (actasmtoken <> AS_SEPARATOR) do
 | |
|                                Consume(actasmtoken);
 | |
|                            end;
 | |
|                          end; { end case }
 | |
|                        end
 | |
|                        else
 | |
|                           Begin
 | |
|                              Message(assem_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 := HexToDec(actasmpattern);
 | |
|                                         AS_BINNUM: str := BinaryToDec(actasmpattern);
 | |
|                                         AS_OCTALNUM: str := OctalToDec(actasmpattern);
 | |
|                                 else
 | |
|                                         Message(assem_e_syntax_error);
 | |
|                                 end;
 | |
|                                 Consume(actasmtoken);
 | |
|                                 val(str, l, code);
 | |
|                                 if code <> 0 then
 | |
|                                      Message(assem_e_invalid_reference_syntax)
 | |
|                                 else
 | |
|                                      instr.operands[operandnum].ref.offset := l;
 | |
|                                 Consume(AS_RPAREN);
 | |
|                                 if not (actasmtoken in [AS_COMMA,AS_SEPARATOR]) then
 | |
|                                 Begin
 | |
|                                       { error recovery ... }
 | |
|                                       Message(assem_e_invalid_reference);
 | |
|                                       while actasmtoken <> AS_SEPARATOR do
 | |
|                                         Consume(actasmtoken);
 | |
|                                 end;
 | |
|                                 exit;
 | |
|                             end;
 | |
|      else
 | |
|        Begin
 | |
| 
 | |
|          Message(assem_e_invalid_reference_syntax);
 | |
|          while (actasmtoken <> AS_SEPARATOR) do
 | |
|            Consume(actasmtoken);
 | |
|        end;
 | |
|      end; { end case }
 | |
|   end;
 | |
| 
 | |
| 
 | |
|   Procedure BuildOperand(var instr: TInstruction);
 | |
|   {*********************************************************************}
 | |
|   { EXIT CONDITION:  On exit the routine should point to either the     }
 | |
|   {       AS_COMMA or AS_SEPARATOR token.                               }
 | |
|   {*********************************************************************}
 | |
|   var
 | |
|     tempstr: string;
 | |
|     expr: string;
 | |
|     lab: Pasmlabel;
 | |
|     l : longint;
 | |
|     i: tregister;
 | |
|     hl: plabel;
 | |
|     reg_one, reg_two: tregister;
 | |
|     reglist: set of tregister;
 | |
|   Begin
 | |
|    reglist := [];
 | |
|    tempstr := '';
 | |
|    expr := '';
 | |
|    case actasmtoken of
 | |
|    { // Memory reference //  }
 | |
|      AS_LPAREN:
 | |
|                Begin
 | |
|                   initAsmRef(instr);
 | |
|                   BuildReference(instr);
 | |
|                end;
 | |
|    { // Constant expression //  }
 | |
|      AS_APPT:  Begin
 | |
|                       Consume(AS_APPT);
 | |
|                       if not (instr.operands[operandnum].operandtype in [OPR_NONE,OPR_CONSTANT]) then
 | |
|                          Message(assem_e_invalid_operand_type);
 | |
|                       { identifiers are handled by BuildExpression }
 | |
|                       instr.operands[operandnum].operandtype := OPR_CONSTANT;
 | |
|                       instr.operands[operandnum].val :=BuildExpression;
 | |
|                  end;
 | |
|    { // Constant memory offset .              // }
 | |
|    { // This must absolutely be followed by ( // }
 | |
|      AS_HEXNUM,AS_INTNUM,
 | |
|      AS_BINNUM,AS_OCTALNUM,AS_PLUS:
 | |
|                    Begin
 | |
|                       InitAsmRef(instr);
 | |
|                       instr.operands[operandnum].ref.offset:=BuildRefExpression;
 | |
|                       BuildReference(instr);
 | |
|                    end;
 | |
|    { // A constant expression, or a Variable ref. // }
 | |
|      AS_ID:  Begin
 | |
|               if actasmpattern[1] = '@' then
 | |
|               { // Label or Special symbol reference // }
 | |
|               Begin
 | |
|                  if actasmpattern = '@RESULT' then
 | |
|                    Begin
 | |
|                       InitAsmRef(instr);
 | |
|                       SetUpResult(instr,operandnum);
 | |
|                    end
 | |
|                  else
 | |
|                   if (actasmpattern = '@CODE') or (actasmpattern = '@DATA') then
 | |
|                     Message(assem_w_CODE_and_DATA_not_supported)
 | |
|                    else
 | |
|                   Begin
 | |
|                     delete(actasmpattern,1,1);
 | |
|                     if actasmpattern = '' then
 | |
|                      Message(assem_e_null_label_ref_not_allowed);
 | |
|                     lab := labellist.search(actasmpattern);
 | |
|                     { check if the label is already defined   }
 | |
|                     { if so, we then check if the plabel is   }
 | |
|                     { non-nil, if so we add it to instruction }
 | |
|                     if assigned(lab) then
 | |
|                      Begin
 | |
|                      if assigned(lab^.lab) then
 | |
|                        Begin
 | |
|                          instr.operands[operandnum].operandtype := OPR_LABINSTR;
 | |
|                          instr.operands[operandnum].hl := lab^.lab;
 | |
|                          instr.labeled := TRUE;
 | |
|                        end;
 | |
|                      end
 | |
|                     else
 | |
|                     { the label does not exist, create it }
 | |
|                     { emit the opcode, but set that the   }
 | |
|                     { label has not been emitted          }
 | |
|                      Begin
 | |
|                         getlabel(hl);
 | |
|                         labellist.insert(actasmpattern,hl,FALSE);
 | |
|                         instr.operands[operandnum].operandtype := OPR_LABINSTR;
 | |
|                         instr.operands[operandnum].hl := hl;
 | |
|                         instr.labeled := TRUE;
 | |
|                      end;
 | |
|                   end;
 | |
|                 Consume(AS_ID);
 | |
|                 if not (actasmtoken in [AS_SEPARATOR,AS_COMMA]) then
 | |
|                  Message(assem_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
 | |
|                       InitAsmRef(instr);
 | |
|                       instr.operands[operandnum].ref.offset:=BuildRefExpression;
 | |
|                       BuildReference(instr);
 | |
| 
 | |
| {                      if not (instr.operands[operandnum].operandtype in [OPR_NONE,OPR_CONSTANT]) then
 | |
|                         Message(assem_e_invalid_operand_type);
 | |
|                       instr.operands[operandnum].operandtype := OPR_CONSTANT;
 | |
|                       instr.operands[operandnum].val :=BuildExpression;}
 | |
|                     end
 | |
|                    else { is it a label variable ? }
 | |
|                     Begin
 | |
|                      { // 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) then
 | |
|                      Begin
 | |
|                         instr.operands[operandnum].operandtype := OPR_LABINSTR;
 | |
|                         instr.operands[operandnum].hl := hl;
 | |
|                         instr.labeled := TRUE;
 | |
|                         Consume(AS_ID);
 | |
|                         if not (actasmtoken in [AS_SEPARATOR,AS_COMMA]) then
 | |
|                          Message(assem_e_syntax_error);
 | |
|                      end
 | |
|                      else
 | |
|                      { is it a normal variable ? }
 | |
|                      Begin
 | |
|                       initAsmRef(instr);
 | |
|                       if not CreateVarInstr(instr,actasmpattern,operandnum) then
 | |
|                       Begin
 | |
|                          { not a variable.. }
 | |
|                          { check special variables.. }
 | |
|                          if actasmpattern = 'SELF' then
 | |
|                           { special self variable }
 | |
|                          Begin
 | |
|                            if assigned(procinfo._class) then
 | |
|                              Begin
 | |
|                                instr.operands[operandnum].ref.offset := procinfo.ESI_offset;
 | |
|                                instr.operands[operandnum].ref.base := procinfo.framepointer;
 | |
|                              end
 | |
|                            else
 | |
|                              Message(assem_e_cannot_use_SELF_outside_a_method);
 | |
|                          end
 | |
|                          else
 | |
|                          if (cs_compilesystem in aktswitches) then
 | |
|                          Begin
 | |
|                            if not assigned(instr.operands[operandnum].ref.symbol) then
 | |
|                            Begin
 | |
|                              instr.operands[operandnum].ref.symbol:=newpasstr(actasmpattern);
 | |
|                              Message1(assem_w_id_supposed_external,actasmpattern);
 | |
|                            end;
 | |
|                          end
 | |
|                          else
 | |
|                            Message1(assem_e_unknown_id,actasmpattern);
 | |
|                       end;
 | |
|                       expr := actasmpattern;
 | |
|                       Consume(AS_ID);
 | |
|                       case actasmtoken of
 | |
|                            AS_LPAREN: { indexing }
 | |
|                                         BuildReference(instr);
 | |
|                            AS_SEPARATOR,AS_COMMA: ;
 | |
|                       else
 | |
|                            Message(assem_e_syntax_error);
 | |
|                       end;
 | |
|                      end;
 | |
|                     end;
 | |
|                end;
 | |
|             end;
 | |
|    { // Pre-decrement mode reference or constant mem offset.   // }
 | |
|      AS_MINUS:    Begin
 | |
|                    Consume(AS_MINUS);
 | |
|                    if actasmtoken = AS_LPAREN then
 | |
|                    Begin
 | |
|                      InitAsmRef(instr);
 | |
|                      { indicate pre-decrement mode }
 | |
|                      instr.operands[operandnum].ref.direction := dir_dec;
 | |
|                      BuildReference(instr);
 | |
|                    end
 | |
|                    else
 | |
|                    if actasmtoken in [AS_OCTALNUM,AS_HEXNUM,AS_BINNUM,AS_INTNUM] then
 | |
|                    Begin
 | |
|                       InitAsmRef(instr);
 | |
|                       instr.operands[operandnum].ref.offset:=BuildRefExpression;
 | |
|                       { negate because was preceded by a negative sign! }
 | |
|                       instr.operands[operandnum].ref.offset:=-instr.operands[operandnum].ref.offset;
 | |
|                       BuildReference(instr);
 | |
|                    end
 | |
|                    else
 | |
|                    Begin
 | |
|                     Message(assem_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
 | |
|                    { save the type of register used. }
 | |
|                    tempstr := actasmpattern;
 | |
|                    Consume(AS_REGISTER);
 | |
|                    { // Simple register // }
 | |
|                    if (actasmtoken = AS_SEPARATOR) or (actasmtoken = AS_COMMA) then
 | |
|                    Begin
 | |
|                         if not (instr.operands[operandnum].operandtype in [OPR_NONE,OPR_REGISTER]) then
 | |
|                          Message(assem_e_invalid_operand_type);
 | |
|                         instr.operands[operandnum].operandtype := OPR_REGISTER;
 | |
|                         instr.operands[operandnum].reg := findregister(tempstr);
 | |
|                    end
 | |
|                    else
 | |
|                    { HERE WE MUST HANDLE THE SPECIAL CASE OF MOVEM AND FMOVEM }
 | |
|                    { // Individual register listing // }
 | |
|                    if (actasmtoken = AS_SLASH) then
 | |
|                    Begin
 | |
|                      reglist := [findregister(tempstr)];
 | |
|                      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
 | |
|                                         reglist := reglist + [findregister(actasmpattern)];
 | |
|                                         Consume(AS_REGISTER);
 | |
|                                        end;
 | |
|                           AS_SLASH: Consume(AS_SLASH);
 | |
|                           AS_SEPARATOR,AS_COMMA: break;
 | |
|                          else
 | |
|                           Begin
 | |
|                             Message(assem_e_invalid_reg_list_in_movem);
 | |
|                             Consume(actasmtoken);
 | |
|                           end;
 | |
|                          end; { end case }
 | |
|                        end; { end while }
 | |
|                        instr.operands[operandnum].operandtype:= OPR_REGLIST;
 | |
|                        instr.operands[operandnum].list := reglist;
 | |
|                      end
 | |
|                      else
 | |
|                       { error recovery ... }
 | |
|                       Begin
 | |
|                             Message(assem_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:=findregister(tempstr);
 | |
|                      if actasmtoken <> AS_REGISTER then
 | |
|                      Begin
 | |
|                        Message(assem_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:=findregister(actasmpattern);
 | |
|                       if reg_one > reg_two then
 | |
|                       begin
 | |
|                        for i:=reg_two to reg_one do
 | |
|                          reglist := reglist + [i];
 | |
|                       end
 | |
|                       else
 | |
|                       Begin
 | |
|                        for i:=reg_one to reg_two do
 | |
|                          reglist := reglist + [i];
 | |
|                       end;
 | |
|                       Consume(AS_REGISTER);
 | |
|                       if not (actasmtoken in [AS_SEPARATOR,AS_COMMA]) then
 | |
|                       Begin
 | |
|                        Message(assem_e_invalid_reg_list_in_movem);
 | |
|                        while not (actasmtoken in [AS_SEPARATOR,AS_COMMA]) do
 | |
|                          Consume(actasmtoken);
 | |
|                       end;
 | |
|                       { set up instruction }
 | |
|                       instr.operands[operandnum].operandtype:= OPR_REGLIST;
 | |
|                       instr.operands[operandnum].list := reglist;
 | |
|                      end;
 | |
|                    end
 | |
|                    else
 | |
|                    { DIVSL/DIVS/MULS/MULU with long for MC68020 only }
 | |
|                    if (actasmtoken = AS_COLON) then
 | |
|                    Begin
 | |
|                      if (aktoptprocessor = MC68020) or (cs_compilesystem in aktswitches) then
 | |
|                      Begin
 | |
|                        Consume(AS_COLON);
 | |
|                        if (actasmtoken = AS_REGISTER) then
 | |
|                        Begin
 | |
|                          { set up old field, since register is valid }
 | |
|                          instr.operands[operandnum].operandtype := OPR_REGISTER;
 | |
|                          instr.operands[operandnum].reg := findregister(tempstr);
 | |
|                          Inc(operandnum);
 | |
|                          instr.operands[operandnum].operandtype := OPR_REGISTER;
 | |
|                          instr.operands[operandnum].reg := findregister(actasmpattern);
 | |
|                          Consume(AS_REGISTER);
 | |
|                          if not (actasmtoken in [AS_SEPARATOR,AS_COMMA]) then
 | |
|                          Begin
 | |
|                           Message(assem_e_invalid_reg_list_for_opcode);
 | |
|                           while not (actasmtoken in [AS_SEPARATOR,AS_COMMA]) do
 | |
|                             Consume(actasmtoken);
 | |
|                          end;
 | |
|                        end;
 | |
|                      end
 | |
|                      else
 | |
|                      Begin
 | |
|                         Message(assem_e_68020_mode_required);
 | |
|                         if not (actasmtoken in [AS_SEPARATOR,AS_COMMA]) then
 | |
|                         Begin
 | |
|                           Message(assem_e_invalid_reg_list_for_opcode);
 | |
|                           while not (actasmtoken in [AS_SEPARATOR,AS_COMMA]) do
 | |
|                             Consume(actasmtoken);
 | |
|                         end;
 | |
|                      end;
 | |
|                    end
 | |
|                    else
 | |
|                     Message1(assem_e_syn_register,tempstr);
 | |
|                  end;
 | |
|      AS_SEPARATOR, AS_COMMA: ;
 | |
|     else
 | |
|      Begin
 | |
|       Message(assem_e_syn_opcode_operand);
 | |
|       Consume(actasmtoken);
 | |
|      end;
 | |
|   end; { end case }
 | |
|  end;
 | |
| 
 | |
| 
 | |
| 
 | |
|   Procedure 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
 | |
|    strlength: byte;
 | |
|    expr: string;
 | |
|    tempstr: string;
 | |
|    value : longint;
 | |
|   Begin
 | |
|       Repeat
 | |
|         Case actasmtoken of
 | |
|           AS_STRING: Begin
 | |
|                       if maxvalue = $ff then
 | |
|                          strlength := 1
 | |
|                       else
 | |
|                          Message(assem_e_string_not_allowed_as_const);
 | |
|                       expr := actasmpattern;
 | |
|                       if length(expr) > 1 then
 | |
|                        Message(assem_e_string_not_allowed_as_const);
 | |
|                       Consume(AS_STRING);
 | |
|                       Case actasmtoken of
 | |
|                        AS_COMMA: Consume(AS_COMMA);
 | |
|                        AS_SEPARATOR: ;
 | |
|                       else
 | |
|                        Message(assem_e_invalid_string_expression);
 | |
|                       end; { end case }
 | |
|                       ConcatString(p,expr);
 | |
|                     end;
 | |
|           AS_INTNUM,AS_BINNUM,
 | |
|           AS_OCTALNUM,AS_HEXNUM:
 | |
|                     Begin
 | |
|                       value:=BuildExpression;
 | |
|                       ConcatConstant(p,value,maxvalue);
 | |
|                     end;
 | |
|           AS_ID:
 | |
|                      Begin
 | |
|                       value:=BuildExpression;
 | |
|                       if value > maxvalue then
 | |
|                       Begin
 | |
|                          Message(assem_e_constant_out_of_bounds);
 | |
|                          { assuming a value of maxvalue }
 | |
|                          value := maxvalue;
 | |
|                       end;
 | |
|                       ConcatConstant(p,value,maxvalue);
 | |
|                   end;
 | |
|           { These terms can start an assembler expression }
 | |
|           AS_PLUS,AS_MINUS,AS_LPAREN,AS_NOT: Begin
 | |
|                                           value := BuildExpression;
 | |
|                                           ConcatConstant(p,value,maxvalue);
 | |
|                                          end;
 | |
|           AS_COMMA:  BEGIN
 | |
|                        Consume(AS_COMMA);
 | |
|                      END;
 | |
|           AS_SEPARATOR: ;
 | |
| 
 | |
|         else
 | |
|          Begin
 | |
|            Message(assem_f_internal_error_in_buildconstant);
 | |
|          end;
 | |
|     end; { end case }
 | |
|    Until actasmtoken = AS_SEPARATOR;
 | |
|   end;
 | |
| 
 | |
| 
 | |
|   Procedure 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(p,expr);
 | |
|                       Consume(AS_STRING);
 | |
|                     end;
 | |
|           AS_COMMA:  BEGIN
 | |
|                        Consume(AS_COMMA);
 | |
|                      END;
 | |
|           AS_SEPARATOR: ;
 | |
|         else
 | |
|          Begin
 | |
|           Consume(actasmtoken);
 | |
|           if not errorflag then
 | |
|            Message(assem_e_invalid_string_expression);
 | |
|           errorflag := TRUE;
 | |
|          end;
 | |
|     end; { end case }
 | |
|    Until actasmtoken = AS_SEPARATOR;
 | |
|   end;
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
|   Procedure BuildOpCode;
 | |
|   {*********************************************************************}
 | |
|   { 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 asmtok: tasmop;
 | |
|       op: tasmop;
 | |
|       expr: string;
 | |
|       segreg: tregister;
 | |
|   Begin
 | |
|     expr := '';
 | |
|     asmtok := A_NONE; { assmume no prefix          }
 | |
|     segreg := R_NO;   { assume no segment override }
 | |
| 
 | |
|     { //  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(assem_e_invalid_or_missing_opcode);
 | |
|       { error recovery }
 | |
|       While not (actasmtoken in [AS_SEPARATOR,AS_COMMA]) do
 | |
|          Consume(actasmtoken);
 | |
|       exit;
 | |
|     end
 | |
|     else
 | |
|     Begin
 | |
|       op := findopcode(actasmpattern);
 | |
|       instr.addinstr(op);
 | |
|       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 > MaxOperands then
 | |
|                     Message(assem_e_too_many_operands)
 | |
|                   else
 | |
|                     Inc(operandnum);
 | |
|                   Consume(AS_COMMA);
 | |
|                 end;
 | |
|          { // End of asm operands for this opcode // }
 | |
|          AS_SEPARATOR: ;
 | |
|        else
 | |
|          BuildOperand(instr);
 | |
|      end; { end case }
 | |
|     end; { end while }
 | |
|   end;
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
|   Function Assemble: Ptree;
 | |
|   {*********************************************************************}
 | |
|   { PROCEDURE Assemble;                                                 }
 | |
|   {  Description: Parses the att assembler syntax, parsing is done      }
 | |
|   {  according to GAs rules.                                            }
 | |
|   {*********************************************************************}
 | |
|   Var
 | |
|    hl: plabel;
 | |
|    labelptr,nextlabel : pasmlabel;
 | |
|    commname : string;
 | |
|    store_p : paasmoutput;
 | |
| 
 | |
|   Begin
 | |
|     Message(assem_d_start_motorola);
 | |
|     firsttoken := TRUE;
 | |
|     operandnum := 0;
 | |
|     { sets up all opcode and register tables in uppercase }
 | |
|     if not _asmsorted then
 | |
|     Begin
 | |
|       SetupTables;
 | |
|       _asmsorted := TRUE;
 | |
|     end;
 | |
|     p:=new(paasmoutput,init);
 | |
|     { save pointer code section }
 | |
|     store_p:=p;
 | |
|     { setup label linked list }
 | |
|     labellist.init;
 | |
|     c:=current_scanner^.asmgetchar;
 | |
|     actasmtoken:=gettoken;
 | |
|     while actasmtoken<>AS_END do
 | |
|     Begin
 | |
|       case actasmtoken of
 | |
|         AS_LLABEL: Begin
 | |
|                     labelptr := labellist.search(actasmpattern);
 | |
|                     if not assigned(labelptr) then
 | |
|                     Begin
 | |
|                         getlabel(hl);
 | |
|                         labellist.insert(actasmpattern,hl,TRUE);
 | |
|                         ConcatLabel(p,A_LABEL,hl);
 | |
|                     end
 | |
|                     else
 | |
|                     { the label has already been inserted into the  }
 | |
|                     { label list, either as an instruction label (in}
 | |
|                     { this case it has not been emitted), or as a   }
 | |
|                     { duplicate local symbol (in this case it has   }
 | |
|                     { already been emitted).                        }
 | |
|                     Begin
 | |
|                        if labelptr^.emitted then
 | |
|                         Message1(assem_e_dup_local_sym,'@'+labelptr^.name^)
 | |
|                        else
 | |
|                         Begin
 | |
|                           if assigned(labelptr^.lab) then
 | |
|                             ConcatLabel(p,A_LABEL,labelptr^.lab);
 | |
|                           labelptr^.emitted := TRUE;
 | |
|                         end;
 | |
|                     end;
 | |
|                     Consume(AS_LLABEL);
 | |
|                   end;
 | |
|         AS_LABEL: Begin
 | |
|                      { when looking for Pascal labels, these must }
 | |
|                      { be in uppercase.                           }
 | |
|                      if SearchLabel(upper(actasmpattern),hl) then
 | |
|                        ConcatLabel(p,A_LABEL, hl)
 | |
|                      else
 | |
|                      Begin
 | |
|                        Message1(assem_e_unknown_label_identifer,actasmpattern);
 | |
|                      end;
 | |
|                      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($ffffffff);
 | |
|                 end;
 | |
|         AS_XDEF:
 | |
|                   Begin
 | |
|                    { normal units should not be able to declare }
 | |
|                    { direct label names like this... anyhow     }
 | |
|                    { procedural calls in asm blocks are         }
 | |
|                    { supposedely replaced automatically         }
 | |
|                    if (cs_compilesystem in aktswitches) then
 | |
|                    begin
 | |
|                      Consume(AS_XDEF);
 | |
|                       if actasmtoken <> AS_ID then
 | |
|                        Message(assem_e_invalid_global_def)
 | |
|                       else
 | |
|                         ConcatPublic(p,actasmpattern);
 | |
|                       Consume(actasmtoken);
 | |
|                       if actasmtoken <> AS_SEPARATOR then
 | |
|                       Begin
 | |
|                         Message(assem_e_line_separator_expected);
 | |
|                         while actasmtoken <> AS_SEPARATOR do
 | |
|                          Consume(actasmtoken);
 | |
|                       end;
 | |
|                    end
 | |
|                    else
 | |
|                    begin
 | |
|                      Message(assem_w_xdef_not_supported);
 | |
|                      while actasmtoken <> AS_SEPARATOR do
 | |
|                        Consume(actasmtoken);
 | |
|                    end;
 | |
|                   end;
 | |
|         AS_ALIGN: Begin
 | |
|                     Message(assem_w_align_not_supported);
 | |
|                     while actasmtoken <> AS_SEPARATOR do
 | |
|                      Consume(actasmtoken);
 | |
|                   end;
 | |
|         AS_OPCODE: Begin
 | |
|                    instr.init;
 | |
|                    BuildOpcode;
 | |
|                    instr.numops := operandnum;
 | |
|                    if instr.labeled then
 | |
|                      ConcatLabeledInstr(instr)
 | |
|                    else
 | |
|                      ConcatOpCode(instr);
 | |
|                   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(assem_e_assemble_node_syntax_error);
 | |
|          { error recovery }
 | |
|          Consume(actasmtoken);
 | |
|       end;
 | |
|     end; { end case }
 | |
|   end; { end while }
 | |
|   { check if there were undefined symbols.   }
 | |
|   { if so, then list each of those undefined }
 | |
|   { labels.                                  }
 | |
|   if assigned(labellist.First) then
 | |
|   Begin
 | |
|     labelptr := labellist.First;
 | |
|     While labelptr <> nil do
 | |
|       Begin
 | |
|          nextlabel:=labelptr^.next;
 | |
|          if not labelptr^.emitted  then
 | |
|           Message1(assem_e_local_sym_not_found_in_asm_statement,'@'+labelptr^.name^);
 | |
|          labelptr:=nextlabel;
 | |
|       end;
 | |
|   end;
 | |
|   assemble := genasmnode(p);
 | |
|   labellist.done;
 | |
|   Message(assem_d_finish_motorola);
 | |
| end;
 | |
| 
 | |
| 
 | |
|     procedure ra68kmot_exit;{$ifndef FPC}far;{$endif}
 | |
|       begin
 | |
|          if assigned(iasmops) then
 | |
|            dispose(iasmops);
 | |
|          exitproc:=old_exit;
 | |
|       end;
 | |
| 
 | |
| 
 | |
| Begin
 | |
|    old_exit:=exitproc;
 | |
|    exitproc:=@ra68kmot_exit;
 | |
| end.
 | |
| {
 | |
|   $Log$
 | |
|   Revision 1.4  1998-07-14 14:47:02  peter
 | |
|     * released NEWINPUT
 | |
| 
 | |
|   Revision 1.3  1998/07/10 10:51:02  peter
 | |
|     * m68k updates
 | |
| 
 | |
|   Revision 1.2  1998/06/24 14:06:39  peter
 | |
|     * fixed the name changes
 | |
| 
 | |
|   Revision 1.1  1998/06/23 14:00:20  peter
 | |
|     * renamed RA* units
 | |
| 
 | |
|   Revision 1.5  1998/06/12 10:32:31  pierre
 | |
|     * column problem hopefully solved
 | |
|     + C vars declaration changed
 | |
| 
 | |
|   Revision 1.4  1998/06/04 23:51:56  peter
 | |
|     * m68k compiles
 | |
|     + .def file creation moved to gendef.pas so it could also be used
 | |
|       for win32
 | |
| 
 | |
|   Revision 1.3  1998/05/20 09:42:36  pierre
 | |
|     + UseTokenInfo now default
 | |
|     * unit in interface uses and implementation uses gives error now
 | |
|     * only one error for unknown symbol (uses lastsymknown boolean)
 | |
|       the problem came from the label code !
 | |
|     + first inlined procedures and function work
 | |
|       (warning there might be allowed cases were the result is still wrong !!)
 | |
|     * UseBrower updated gives a global list of all position of all used symbols
 | |
|       with switch -gb
 | |
| 
 | |
|   Revision 1.2  1998/04/29 10:34:01  pierre
 | |
|     + added some code for ansistring (not complete nor working yet)
 | |
|     * corrected operator overloading
 | |
|     * corrected nasm output
 | |
|     + started inline procedures
 | |
|     + added starstarn : use ** for exponentiation (^ gave problems)
 | |
|     + started UseTokenInfo cond to get accurate positions
 | |
| 
 | |
|   Revision 1.1.1.1  1998/03/25 11:18:15  root
 | |
|   * Restored version
 | |
| 
 | |
|   Revision 1.14  1998/03/22 12:45:38  florian
 | |
|     * changes of Carl-Eric to m68k target commit:
 | |
|       - wrong nodes because of the new string cg in intel, I had to create
 | |
|         this under m68k also ... had to work it out to fix potential alignment
 | |
|         problems --> this removes the crash of the m68k compiler.
 | |
|       - added absolute addressing in m68k assembler (required for Amiga startup)
 | |
|       - fixed alignment problems (because of byte return values, alignment
 | |
|         would not be always valid) -- is this ok if i change the offset if odd in
 | |
|         setfirsttemp ?? -- it seems ok...
 | |
| 
 | |
|   Revision 1.13  1998/03/10 16:27:43  pierre
 | |
|     * better line info in stabs debug
 | |
|     * symtabletype and lexlevel separated into two fields of tsymtable
 | |
|     + ifdef MAKELIB for direct library output, not complete
 | |
|     + ifdef CHAINPROCSYMS for overloaded seach across units, not fully
 | |
|       working
 | |
|     + ifdef TESTFUNCRET for setting func result in underfunction, not
 | |
|       working
 | |
| 
 | |
|   Revision 1.12  1998/03/10 01:17:25  peter
 | |
|     * all files have the same header
 | |
|     * messages are fully implemented, EXTDEBUG uses Comment()
 | |
|     + AG... files for the Assembler generation
 | |
| 
 | |
| }
 | |
| 
 | |
| 
 | 
