mirror of
				https://gitlab.com/freepascal.org/fpc/source.git
				synced 2025-11-04 09:19:39 +01:00 
			
		
		
		
	
		
			
				
	
	
		
			249 lines
		
	
	
		
			7.3 KiB
		
	
	
	
		
			ObjectPascal
		
	
	
	
	
	
			
		
		
	
	
			249 lines
		
	
	
		
			7.3 KiB
		
	
	
	
		
			ObjectPascal
		
	
	
	
	
	
{
 | 
						|
    Copyright (c) 1998-2004 by Florian Klaempfl
 | 
						|
 | 
						|
    Some basic types and constants for the code generation
 | 
						|
 | 
						|
    This program is free software; you can redistribute it and/or modify
 | 
						|
    it under the terms of the GNU General Public License as published by
 | 
						|
    the Free Software Foundation; either version 2 of the License, or
 | 
						|
    (at your option) any later version.
 | 
						|
 | 
						|
    This program is distributed in the hope that it will be useful,
 | 
						|
    but WITHOUT ANY WARRANTY; without even the implied warranty of
 | 
						|
    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 | 
						|
    GNU General Public License for more details.
 | 
						|
 | 
						|
    You should have received a copy of the GNU General Public License
 | 
						|
    along with this program; if not, write to the Free Software
 | 
						|
    Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
 | 
						|
 | 
						|
 ****************************************************************************
 | 
						|
}
 | 
						|
{ This unit exports some helper routines which are used across the code generator }
 | 
						|
unit cgutils;
 | 
						|
 | 
						|
{$i fpcdefs.inc}
 | 
						|
 | 
						|
  interface
 | 
						|
 | 
						|
    uses
 | 
						|
      globtype,
 | 
						|
      cclasses,
 | 
						|
      aasmbase,
 | 
						|
      cpubase,cgbase;
 | 
						|
 | 
						|
    type
 | 
						|
      { reference record, reordered for best alignment }
 | 
						|
      preference = ^treference;
 | 
						|
      treference = record
 | 
						|
         offset      : asizeint;
 | 
						|
         symbol,
 | 
						|
         relsymbol   : tasmsymbol;
 | 
						|
{$if defined(x86) or defined(m68k)}
 | 
						|
         segment,
 | 
						|
{$endif defined(x86) or defined(m68k)}
 | 
						|
         base,
 | 
						|
         index       : tregister;
 | 
						|
         refaddr     : trefaddr;
 | 
						|
         scalefactor : byte;
 | 
						|
{$ifdef arm}
 | 
						|
         symboldata  : tlinkedlistitem;
 | 
						|
         signindex   : shortint;
 | 
						|
         shiftimm    : byte;
 | 
						|
         addressmode : taddressmode;
 | 
						|
         shiftmode   : tshiftmode;
 | 
						|
{$endif arm}
 | 
						|
{$ifdef avr}
 | 
						|
         addressmode : taddressmode;
 | 
						|
{$endif avr}
 | 
						|
{$ifdef m68k}
 | 
						|
         { indexed increment and decrement mode }
 | 
						|
         { (An)+ and -(An)                      }
 | 
						|
         direction : tdirection;
 | 
						|
{$endif m68k}
 | 
						|
         alignment : byte;
 | 
						|
      end;
 | 
						|
 | 
						|
      tsubsetregister = record
 | 
						|
        subsetreg : tregister;
 | 
						|
        startbit, bitlen: byte;
 | 
						|
        subsetregsize: tcgsize;
 | 
						|
      end;
 | 
						|
 | 
						|
      tsubsetreference = record
 | 
						|
        ref: treference;
 | 
						|
        bitindexreg: tregister;
 | 
						|
        startbit, bitlen: byte;
 | 
						|
      end;
 | 
						|
 | 
						|
      tlocation = record
 | 
						|
         loc  : TCGLoc;
 | 
						|
         size : TCGSize;
 | 
						|
         case TCGLoc of
 | 
						|
{$ifdef cpuflags}
 | 
						|
            LOC_FLAGS : (resflags : tresflags);
 | 
						|
{$endif cpuflags}
 | 
						|
            LOC_CONSTANT : (
 | 
						|
              case longint of
 | 
						|
{$ifdef FPC_BIG_ENDIAN}
 | 
						|
                1 : (_valuedummy,value : aint);
 | 
						|
{$else FPC_BIG_ENDIAN}
 | 
						|
                1 : (value : aint);
 | 
						|
{$endif FPC_BIG_ENDIAN}
 | 
						|
                2 : (value64 : Int64);
 | 
						|
              );
 | 
						|
            LOC_CREFERENCE,
 | 
						|
            LOC_REFERENCE : (reference : treference);
 | 
						|
            { segment in reference at the same place as in loc_register }
 | 
						|
            LOC_REGISTER,
 | 
						|
            LOC_CREGISTER : (
 | 
						|
              case longint of
 | 
						|
                1 : (register : tregister;
 | 
						|
                     { some x86_64 targets require two function result registers }
 | 
						|
                     registerhi : tregister;
 | 
						|
{$ifdef m68k}
 | 
						|
                     { some m68k OSes require that the result is returned in d0 and a0
 | 
						|
                       the second location must be stored here }
 | 
						|
                     registeralias : tregister;
 | 
						|
{$endif m68k}
 | 
						|
                    );
 | 
						|
{$ifndef cpu64bitalu}
 | 
						|
                { overlay a 64 Bit register type }
 | 
						|
                2 : (register64 : tregister64);
 | 
						|
{$endif cpu64bitalu}
 | 
						|
{$ifdef avr}
 | 
						|
                3 : (registers : array[0..3] of tregister);
 | 
						|
{$endif avr}
 | 
						|
              );
 | 
						|
            LOC_SUBSETREG,
 | 
						|
            LOC_CSUBSETREG : (
 | 
						|
              sreg: tsubsetregister;
 | 
						|
            );
 | 
						|
            LOC_SUBSETREF : (
 | 
						|
              sref: tsubsetreference;
 | 
						|
            )
 | 
						|
      end;
 | 
						|
 | 
						|
 | 
						|
    { trerefence handling }
 | 
						|
 | 
						|
    {# Clear to zero a treference }
 | 
						|
    procedure reference_reset(var ref : treference; alignment: longint);
 | 
						|
    {# Clear to zero a treference, and set is base address
 | 
						|
       to base register.
 | 
						|
    }
 | 
						|
    procedure reference_reset_base(var ref : treference;base : tregister;offset, alignment : longint);
 | 
						|
    procedure reference_reset_symbol(var ref : treference;sym : tasmsymbol;offset, alignment : longint);
 | 
						|
    { This routine verifies if two references are the same, and
 | 
						|
       if so, returns TRUE, otherwise returns false.
 | 
						|
    }
 | 
						|
    function references_equal(const sref,dref : treference) : boolean;
 | 
						|
 | 
						|
    { tlocation handling }
 | 
						|
 | 
						|
    { cannot be used for loc_(c)reference, because that one requires an alignment }
 | 
						|
    procedure location_reset(var l : tlocation;lt:TCGNonRefLoc;lsize:TCGSize);
 | 
						|
    { for loc_(c)reference }
 | 
						|
    procedure location_reset_ref(var l : tlocation;lt:TCGRefLoc;lsize:TCGSize; alignment: longint);
 | 
						|
    procedure location_copy(var destloc:tlocation; const sourceloc : tlocation);
 | 
						|
    procedure location_swap(var destloc,sourceloc : tlocation);
 | 
						|
 | 
						|
    { returns r with the given alignment }
 | 
						|
    function setalignment(const r : treference;b : byte) : treference;
 | 
						|
 | 
						|
 | 
						|
implementation
 | 
						|
 | 
						|
uses
 | 
						|
  systems,
 | 
						|
  verbose;
 | 
						|
 | 
						|
{****************************************************************************
 | 
						|
                                  TReference
 | 
						|
****************************************************************************}
 | 
						|
 | 
						|
    procedure reference_reset(var ref : treference; alignment: longint);
 | 
						|
      begin
 | 
						|
        FillChar(ref,sizeof(treference),0);
 | 
						|
{$ifdef arm}
 | 
						|
        ref.signindex:=1;
 | 
						|
{$endif arm}
 | 
						|
        ref.alignment:=alignment;
 | 
						|
      end;
 | 
						|
 | 
						|
 | 
						|
    procedure reference_reset_base(var ref : treference;base : tregister;offset, alignment : longint);
 | 
						|
      begin
 | 
						|
        reference_reset(ref,alignment);
 | 
						|
        ref.base:=base;
 | 
						|
        ref.offset:=offset;
 | 
						|
      end;
 | 
						|
 | 
						|
 | 
						|
    procedure reference_reset_symbol(var ref : treference;sym : tasmsymbol;offset, alignment : longint);
 | 
						|
      begin
 | 
						|
        reference_reset(ref,alignment);
 | 
						|
        ref.symbol:=sym;
 | 
						|
        ref.offset:=offset;
 | 
						|
      end;
 | 
						|
 | 
						|
 | 
						|
    function references_equal(const sref,dref : treference):boolean;
 | 
						|
      begin
 | 
						|
        references_equal:=CompareByte(sref,dref,sizeof(treference))=0;
 | 
						|
      end;
 | 
						|
 | 
						|
 | 
						|
    { returns r with the given alignment }
 | 
						|
    function setalignment(const r : treference;b : byte) : treference;
 | 
						|
      begin
 | 
						|
        result:=r;
 | 
						|
        result.alignment:=b;
 | 
						|
      end;
 | 
						|
 | 
						|
{****************************************************************************
 | 
						|
                                  TLocation
 | 
						|
****************************************************************************}
 | 
						|
 | 
						|
    procedure location_reset(var l : tlocation;lt:TCGNonRefLoc;lsize:TCGSize);
 | 
						|
      begin
 | 
						|
        FillChar(l,sizeof(tlocation),0);
 | 
						|
        l.loc:=lt;
 | 
						|
        l.size:=lsize;
 | 
						|
        if l.loc in [LOC_REFERENCE,LOC_CREFERENCE] then
 | 
						|
          { call location_reset_ref instead }
 | 
						|
          internalerror(2009020705);
 | 
						|
      end;
 | 
						|
 | 
						|
    procedure location_reset_ref(var l: tlocation; lt: tcgrefloc; lsize: tcgsize;
 | 
						|
      alignment: longint);
 | 
						|
    begin
 | 
						|
      FillChar(l,sizeof(tlocation),0);
 | 
						|
      l.loc:=lt;
 | 
						|
      l.size:=lsize;
 | 
						|
{$ifdef arm}
 | 
						|
      l.reference.signindex:=1;
 | 
						|
{$endif arm}
 | 
						|
      l.reference.alignment:=alignment;
 | 
						|
    end;
 | 
						|
 | 
						|
 | 
						|
    procedure location_copy(var destloc:tlocation; const sourceloc : tlocation);
 | 
						|
      begin
 | 
						|
        destloc:=sourceloc;
 | 
						|
      end;
 | 
						|
 | 
						|
 | 
						|
    procedure location_swap(var destloc,sourceloc : tlocation);
 | 
						|
      var
 | 
						|
        swapl : tlocation;
 | 
						|
      begin
 | 
						|
        swapl := destloc;
 | 
						|
        destloc := sourceloc;
 | 
						|
        sourceloc := swapl;
 | 
						|
      end;
 | 
						|
 | 
						|
 | 
						|
end.
 | 
						|
 |