| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693 | {    Copyright (c) 1998-2002 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 types which are used across the code generator }unit cgbase;{$i fpcdefs.inc}interface    uses      globtype,      symconst;    type       { Location types where value can be stored }       TCGLoc=(         LOC_INVALID,      { added for tracking problems}         LOC_VOID,         { no value is available }         LOC_CONSTANT,     { constant value }         LOC_JUMP,         { boolean results only, jump to false or true label }         LOC_FLAGS,        { boolean results only, flags are set }         LOC_REGISTER,     { in a processor register }         LOC_CREGISTER,    { Constant register which shouldn't be modified }         LOC_FPUREGISTER,  { FPU stack }         LOC_CFPUREGISTER, { if it is a FPU register variable on the fpu stack }         LOC_MMXREGISTER,  { MMX register }         { MMX register variable }         LOC_CMMXREGISTER,         { multimedia register }         LOC_MMREGISTER,         { Constant multimedia reg which shouldn't be modified }         LOC_CMMREGISTER,         { contiguous subset of bits of an integer register }         LOC_SUBSETREG,         LOC_CSUBSETREG,         { contiguous subset of bits in memory }         LOC_SUBSETREF,         LOC_CSUBSETREF,         { keep these last for range checking purposes }         LOC_CREFERENCE,   { in memory constant value reference (cannot change) }         LOC_REFERENCE     { in memory value }       );       TCGNonRefLoc=low(TCGLoc)..pred(LOC_CREFERENCE);       TCGRefLoc=LOC_CREFERENCE..LOC_REFERENCE;       { since we have only 16bit offsets, we need to be able to specify the high         and lower 16 bits of the address of a symbol of up to 64 bit }       trefaddr = (         addr_no,         addr_full,         addr_pic,         addr_pic_no_got         {$IF defined(POWERPC) or defined(POWERPC64) or defined(SPARC) or defined(MIPS)}         ,         addr_low,         // bits 48-63         addr_high,        // bits 32-47         {$IF defined(POWERPC64)}         addr_higher,      // bits 16-31         addr_highest,     // bits 00-15         {$ENDIF}         addr_higha        // bits 16-31, adjusted         {$IF defined(POWERPC64)}         ,         addr_highera,     // bits 32-47, adjusted         addr_highesta     // bits 48-63, adjusted         {$ENDIF}         {$ENDIF}         {$IFDEF AVR}         ,addr_lo8         ,addr_hi8         {$ENDIF}         );       {# Generic opcodes, which must be supported by all processors       }       topcg =       (          OP_NONE,          OP_MOVE,      { replaced operation with direct load }          OP_ADD,       { simple addition          }          OP_AND,       { simple logical and       }          OP_DIV,       { simple unsigned division }          OP_IDIV,      { simple signed division   }          OP_IMUL,      { simple signed multiply   }          OP_MUL,       { simple unsigned multiply }          OP_NEG,       { simple negate            }          OP_NOT,       { simple logical not       }          OP_OR,        { simple logical or        }          OP_SAR,       { arithmetic shift-right   }          OP_SHL,       { logical shift left       }          OP_SHR,       { logical shift right      }          OP_SUB,       { simple subtraction       }          OP_XOR,       { simple exclusive or      }          OP_ROL,       { rotate left              }          OP_ROR        { rotate right             }        );       {# Generic flag values - used for jump locations }       TOpCmp =       (          OC_NONE,          OC_EQ,           { equality comparison              }          OC_GT,           { greater than (signed)            }          OC_LT,           { less than (signed)               }          OC_GTE,          { greater or equal than (signed)   }          OC_LTE,          { less or equal than (signed)      }          OC_NE,           { not equal                        }          OC_BE,           { less or equal than (unsigned)    }          OC_B,            { less than (unsigned)             }          OC_AE,           { greater or equal than (unsigned) }          OC_A             { greater than (unsigned)          }        );       { OS_NO is also used memory references with large data that can         not be loaded in a register directly }       TCgSize = (OS_NO,                 { integer registers }                  OS_8,OS_16,OS_32,OS_64,OS_128,OS_S8,OS_S16,OS_S32,OS_S64,OS_S128,                 { single,double,extended,comp,float128 }                  OS_F32,OS_F64,OS_F80,OS_C64,OS_F128,                 { multi-media sizes: split in byte, word, dword, ... }                 { entities, then the signed counterparts             }                  OS_M8,OS_M16,OS_M32,OS_M64,OS_M128,                  OS_MS8,OS_MS16,OS_MS32,OS_MS64,OS_MS128);      { Register types }      TRegisterType = (        R_INVALIDREGISTER, { = 0 }        R_INTREGISTER,     { = 1 }        R_FPUREGISTER,     { = 2 }        { used by Intel only }        R_MMXREGISTER,     { = 3 }        R_MMREGISTER,      { = 4 }        R_SPECIALREGISTER, { = 5 }        R_ADDRESSREGISTER  { = 6 }      );      { Sub registers }      TSubRegister = (        R_SUBNONE, { = 0; no sub register possible }        R_SUBL,    { = 1; 8 bits, Like AL }        R_SUBH,    { = 2; 8 bits, Like AH }        R_SUBW,    { = 3; 16 bits, Like AX }        R_SUBD,    { = 4; 32 bits, Like EAX }        R_SUBQ,    { = 5; 64 bits, Like RAX }        { For Sparc floats that use F0:F1 to store doubles }        R_SUBFS,   { = 6; Float that allocates 1 FPU register }        R_SUBFD,   { = 7; Float that allocates 2 FPU registers }        R_SUBFQ,   { = 8; Float that allocates 4 FPU registers }        R_SUBMMS,  { = 9; single scalar in multi media register }        R_SUBMMD,  { = 10; double scalar in multi media register }        R_SUBMMWHOLE  { = 11; complete MM register, size depends on CPU }      );      TSubRegisterSet = set of TSubRegister;      TSuperRegister = type word;      {        The new register coding:        SuperRegister   (bits 0..15)        Subregister     (bits 16..23)        Register type   (bits 24..31)        TRegister is defined as an enum to make it incompatible        with TSuperRegister to avoid mixing them      }      TRegister = (        TRegisterLowEnum := Low(longint),        TRegisterHighEnum := High(longint)      );      TRegisterRec=packed record{$ifdef FPC_BIG_ENDIAN}         regtype : Tregistertype;         subreg  : Tsubregister;         supreg  : Tsuperregister;{$else FPC_BIG_ENDIAN}         supreg  : Tsuperregister;         subreg  : Tsubregister;         regtype : Tregistertype;{$endif FPC_BIG_ENDIAN}      end;      { A type to store register locations for 64 Bit values. }{$ifdef cpu64bitalu}      tregister64 = tregister;{$else cpu64bitalu}      tregister64 = record         reglo,reghi : tregister;      end;{$endif cpu64bitalu}      Tregistermmxset = record        reg0,reg1,reg2,reg3:Tregister      end;      { Set type definition for registers }      tcpuregisterset = set of byte;      tsuperregisterset = array[byte] of set of byte;      pmmshuffle = ^tmmshuffle;      { this record describes shuffle operations for mm operations; if a pointer a shuffle record        passed to an mm operation is nil, it means that the whole location is moved }      tmmshuffle = record        { describes how many shuffles are actually described, if len=0 then          moving the scalar with index 0 to the scalar with index 0 is meant }        len : byte;        { lower nibble of each entry of this array describes index of the source data index while          the upper nibble describes the destination index }        shuffles : array[1..1] of byte;      end;      Tsuperregisterarray=array[0..$ffff] of Tsuperregister;      Psuperregisterarray=^Tsuperregisterarray;      Tsuperregisterworklist=object        buflength,        buflengthinc,        length:word;        buf:Psuperregisterarray;        constructor init;        constructor copyfrom(const x:Tsuperregisterworklist);        destructor  done;        procedure clear;        procedure add(s:tsuperregister);        function addnodup(s:tsuperregister): boolean;        function get:tsuperregister;        function readidx(i:word):tsuperregister;        procedure deleteidx(i:word);        function delete(s:tsuperregister):boolean;      end;      psuperregisterworklist=^tsuperregisterworklist;    const       { alias for easier understanding }       R_SSEREGISTER = R_MMREGISTER;       { Invalid register number }       RS_INVALID    = high(tsuperregister);       { Maximum number of cpu registers per register type,         this must fit in tcpuregisterset }       maxcpuregister = 32;       tcgsize2size : Array[tcgsize] of integer =         { integer values }        (0,1,2,4,8,16,1,2,4,8,16,         { floating point values }         4,8,10,8,16,         { multimedia values }         1,2,4,8,16,1,2,4,8,16);       tfloat2tcgsize: array[tfloattype] of tcgsize =         (OS_F32,OS_F64,OS_F80,OS_F80,OS_C64,OS_C64,OS_F128);       tcgsize2tfloat: array[OS_F32..OS_C64] of tfloattype =         (s32real,s64real,s80real,s64comp);       tvarregable2tcgloc : array[tvarregable] of tcgloc = (LOC_VOID,          LOC_CREGISTER,LOC_CFPUREGISTER,LOC_CMMREGISTER,LOC_CREGISTER);       { Table to convert tcgsize variables to the correspondending         unsigned types }       tcgsize2unsigned : array[tcgsize] of tcgsize = (OS_NO,          OS_8,OS_16,OS_32,OS_64,OS_128,OS_8,OS_16,OS_32,OS_64,OS_128,          OS_F32,OS_F64,OS_F80,OS_C64,OS_F128,          OS_M8,OS_M16,OS_M32,OS_M64,OS_M128,OS_M8,OS_M16,OS_M32,          OS_M64,OS_M128);       tcgloc2str : array[TCGLoc] of string[12] = (            'LOC_INVALID',            'LOC_VOID',            'LOC_CONST',            'LOC_JUMP',            'LOC_FLAGS',            'LOC_REG',            'LOC_CREG',            'LOC_FPUREG',            'LOC_CFPUREG',            'LOC_MMXREG',            'LOC_CMMXREG',            'LOC_MMREG',            'LOC_CMMREG',            'LOC_SSETREG',            'LOC_CSSETREG',            'LOC_SSETREF',            'LOC_CSSETREF',            'LOC_CREF',            'LOC_REF'            );    var       mms_movescalar : pmmshuffle;    procedure supregset_reset(var regs:tsuperregisterset;setall:boolean;                              maxreg:Tsuperregister);{$ifdef USEINLINE}inline;{$endif}    procedure supregset_include(var regs:tsuperregisterset;s:tsuperregister);{$ifdef USEINLINE}inline;{$endif}    procedure supregset_exclude(var regs:tsuperregisterset;s:tsuperregister);{$ifdef USEINLINE}inline;{$endif}    function supregset_in(const regs:tsuperregisterset;s:tsuperregister):boolean;{$ifdef USEINLINE}inline;{$endif}    function newreg(rt:tregistertype;sr:tsuperregister;sb:tsubregister):tregister;{$ifdef USEINLINE}inline;{$endif}    function getsubreg(r:tregister):tsubregister;{$ifdef USEINLINE}inline;{$endif}    function getsupreg(r:tregister):tsuperregister;{$ifdef USEINLINE}inline;{$endif}    function getregtype(r:tregister):tregistertype;{$ifdef USEINLINE}inline;{$endif}    procedure setsubreg(var r:tregister;sr:tsubregister);{$ifdef USEINLINE}inline;{$endif}    procedure setsupreg(var r:tregister;sr:tsuperregister);{$ifdef USEINLINE}inline;{$endif}    function generic_regname(r:tregister):string;    {# From a constant numeric value, return the abstract code generator       size.    }    function int_cgsize(const a: tcgint): tcgsize;{$ifdef USEINLINE}inline;{$endif}    function int_float_cgsize(const a: tcgint): tcgsize;    { return the inverse condition of opcmp }    function inverse_opcmp(opcmp: topcmp): topcmp;{$ifdef USEINLINE}inline;{$endif}    { return the opcmp needed when swapping the operands }    function swap_opcmp(opcmp: topcmp): topcmp;{$ifdef USEINLINE}inline;{$endif}    { return whether op is commutative }    function commutativeop(op: topcg): boolean;{$ifdef USEINLINE}inline;{$endif}    { returns true, if shuffle describes a real shuffle operation and not only a move }    function realshuffle(shuffle : pmmshuffle) : boolean;    { returns true, if the shuffle describes only a move of the scalar at index 0 }    function shufflescalar(shuffle : pmmshuffle) : boolean;    { removes shuffling from shuffle, this means that the destenation index of each shuffle is copied to      the source }    procedure removeshuffles(var shuffle : tmmshuffle);implementation    uses      verbose;{******************************************************************************                             tsuperregisterworklist******************************************************************************}    constructor tsuperregisterworklist.init;    begin      length:=0;      buflength:=0;      buflengthinc:=16;      buf:=nil;    end;    constructor Tsuperregisterworklist.copyfrom(const x:Tsuperregisterworklist);    begin      self:=x;      if x.buf<>nil then        begin          getmem(buf,buflength*sizeof(Tsuperregister));          move(x.buf^,buf^,length*sizeof(Tsuperregister));        end;    end;    destructor tsuperregisterworklist.done;    begin      if assigned(buf) then        freemem(buf);    end;    procedure tsuperregisterworklist.add(s:tsuperregister);    begin      inc(length);      { Need to increase buffer length? }      if length>=buflength then        begin          inc(buflength,buflengthinc);          buflengthinc:=buflengthinc*2;          if buflengthinc>256 then             buflengthinc:=256;          reallocmem(buf,buflength*sizeof(Tsuperregister));        end;      buf^[length-1]:=s;    end;    function tsuperregisterworklist.addnodup(s:tsuperregister): boolean;    begin      addnodup := false;      if indexword(buf^,length,s) = -1 then        begin          add(s);          addnodup := true;        end;    end;    procedure tsuperregisterworklist.clear;    begin      length:=0;    end;    procedure tsuperregisterworklist.deleteidx(i:word);    begin      if i>=length then        internalerror(200310144);      buf^[i]:=buf^[length-1];      dec(length);    end;    function tsuperregisterworklist.readidx(i:word):tsuperregister;      begin        if (i >= length) then          internalerror(2005010601);        result := buf^[i];      end;    function tsuperregisterworklist.get:tsuperregister;    begin      if length=0 then        internalerror(200310142);      get:=buf^[0];      buf^[0]:=buf^[length-1];      dec(length);    end;    function tsuperregisterworklist.delete(s:tsuperregister):boolean;    var      i:longint;    begin      delete:=false;      { indexword in 1.0.x and 1.9.4 is broken }      i:=indexword(buf^,length,s);      if i<>-1 then        begin          deleteidx(i);          delete := true;        end;    end;    procedure supregset_reset(var regs:tsuperregisterset;setall:boolean;                              maxreg:Tsuperregister);{$ifdef USEINLINE}inline;{$endif}    begin      fillchar(regs,(maxreg+7) shr 3,-byte(setall));    end;    procedure supregset_include(var regs:tsuperregisterset;s:tsuperregister);{$ifdef USEINLINE}inline;{$endif}      begin        include(regs[s shr 8],(s and $ff));      end;    procedure supregset_exclude(var regs:tsuperregisterset;s:tsuperregister);{$ifdef USEINLINE}inline;{$endif}      begin        exclude(regs[s shr 8],(s and $ff));      end;    function supregset_in(const regs:tsuperregisterset;s:tsuperregister):boolean;{$ifdef USEINLINE}inline;{$endif}      begin        result:=(s and $ff) in regs[s shr 8];      end;    function newreg(rt:tregistertype;sr:tsuperregister;sb:tsubregister):tregister;{$ifdef USEINLINE}inline;{$endif}      begin        tregisterrec(result).regtype:=rt;        tregisterrec(result).supreg:=sr;        tregisterrec(result).subreg:=sb;      end;    function getsubreg(r:tregister):tsubregister;{$ifdef USEINLINE}inline;{$endif}      begin        result:=tregisterrec(r).subreg;      end;    function getsupreg(r:tregister):tsuperregister;{$ifdef USEINLINE}inline;{$endif}      begin        result:=tregisterrec(r).supreg;      end;    function getregtype(r:tregister):tregistertype;{$ifdef USEINLINE}inline;{$endif}      begin        result:=tregisterrec(r).regtype;      end;    procedure setsubreg(var r:tregister;sr:tsubregister);{$ifdef USEINLINE}inline;{$endif}      begin        tregisterrec(r).subreg:=sr;      end;    procedure setsupreg(var r:tregister;sr:tsuperregister);{$ifdef USEINLINE}inline;{$endif}      begin        tregisterrec(r).supreg:=sr;      end;    function generic_regname(r:tregister):string;      var        nr : string[12];      begin        str(getsupreg(r),nr);        case getregtype(r) of          R_INTREGISTER:            result:='ireg'+nr;          R_FPUREGISTER:            result:='freg'+nr;          R_MMREGISTER:            result:='mreg'+nr;          R_MMXREGISTER:            result:='xreg'+nr;          R_ADDRESSREGISTER:            result:='areg'+nr;          R_SPECIALREGISTER:            result:='sreg'+nr;          else            begin              result:='INVALID';              exit;            end;        end;        case getsubreg(r) of          R_SUBNONE:            ;          R_SUBL:            result:=result+'l';          R_SUBH:            result:=result+'h';          R_SUBW:            result:=result+'w';          R_SUBD:            result:=result+'d';          R_SUBQ:            result:=result+'q';          R_SUBFS:            result:=result+'fs';          R_SUBFD:            result:=result+'fd';          R_SUBMMD:            result:=result+'md';          R_SUBMMS:            result:=result+'ms';          R_SUBMMWHOLE:            result:=result+'ma';          else            internalerror(200308252);        end;      end;    function int_cgsize(const a: tcgint): tcgsize;{$ifdef USEINLINE}inline;{$endif}      const        size2cgsize : array[0..8] of tcgsize = (          OS_NO,OS_8,OS_16,OS_NO,OS_32,OS_NO,OS_NO,OS_NO,OS_64        );      begin        if a>8 then          result:=OS_NO        else          result:=size2cgsize[a];      end;    function int_float_cgsize(const a: tcgint): tcgsize;      begin        case a of          4 :            result:=OS_F32;          8 :            result:=OS_F64;          10 :            result:=OS_F80;          16 :            result:=OS_F128;          else            internalerror(200603211);        end;      end;    function inverse_opcmp(opcmp: topcmp): topcmp;{$ifdef USEINLINE}inline;{$endif}      const        list: array[TOpCmp] of TOpCmp =          (OC_NONE,OC_NE,OC_LTE,OC_GTE,OC_LT,OC_GT,OC_EQ,OC_A,OC_AE,           OC_B,OC_BE);      begin        inverse_opcmp := list[opcmp];      end;    function swap_opcmp(opcmp: topcmp): topcmp;{$ifdef USEINLINE}inline;{$endif}      const        list: array[TOpCmp] of TOpCmp =          (OC_NONE,OC_EQ,OC_LT,OC_GT,OC_LTE,OC_GTE,OC_NE,OC_AE,OC_A,           OC_BE,OC_B);      begin        swap_opcmp := list[opcmp];      end;    function commutativeop(op: topcg): boolean;{$ifdef USEINLINE}inline;{$endif}      const        list: array[topcg] of boolean =          (true,false,true,true,false,false,true,true,false,false,           true,false,false,false,false,true,false,false);      begin        commutativeop := list[op];      end;    function realshuffle(shuffle : pmmshuffle) : boolean;      var        i : longint;      begin        realshuffle:=true;        if (shuffle=nil) or (shuffle^.len=0) then          realshuffle:=false        else          begin            for i:=1 to shuffle^.len do              begin                if (shuffle^.shuffles[i] and $f)<>((shuffle^.shuffles[i] and $f0) shr 4) then                  exit;              end;            realshuffle:=false;          end;      end;    function shufflescalar(shuffle : pmmshuffle) : boolean;      begin        result:=shuffle^.len=0;      end;    procedure removeshuffles(var shuffle : tmmshuffle);      var        i : longint;      begin        if shuffle.len=0 then          exit;        for i:=1 to shuffle.len do          shuffle.shuffles[i]:=(shuffle.shuffles[i] and $f) or ((shuffle.shuffles[i] and $f0) shr 4);      end;initialization  new(mms_movescalar);  mms_movescalar^.len:=0;finalization  dispose(mms_movescalar);end.
 |