| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524 | {    This file is part of the Free Pascal run time library.    Copyright (c) 1999-2000 by Florian Klaempfl    This unit contains some routines to get informations about the    processor    See the file COPYING.FPC, included in this distribution,    for details about the copyright.    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. **********************************************************************}{$mode objfpc}{$goto on}{$IFNDEF FPC_DOTTEDUNITS}unit cpu;{$ENDIF FPC_DOTTEDUNITS}  interface{$IFDEF FPC_DOTTEDUNITS}    uses      System.SysUtils;{$ELSE FPC_DOTTEDUNITS}    uses      sysutils;{$ENDIF FPC_DOTTEDUNITS}    { returns true, if the processor supports the cpuid instruction }    function cpuid_support : boolean;    { returns true, if floating point is done by an emulator }    function floating_point_emulation : boolean;    { returns the contents of the cr0 register }    function cr0 : longint;    function CMOVSupport : boolean;inline;    function InterlockedCompareExchange128Support : boolean;    function AESSupport : boolean;inline;    function AVXSupport: boolean;inline;    function AVX2Support: boolean;inline;    function AVX512FSupport: boolean;inline;        function AVX512DQSupport: boolean;inline;        function AVX512IFMASupport: boolean;inline;        function AVX512PFSupport: boolean;inline;        function AVX512ERSupport: boolean;inline;        function AVX512CDSupport: boolean;inline;        function AVX512BWSupport: boolean;inline;        function AVX512VLSupport: boolean;inline;        function AVX512VBMISupport: boolean;inline;    function AVX512VBMI2Support: boolean;inline;    function AVX512VNNISupport: boolean;inline;    function VAESSupport: boolean;inline;    function VCLMULSupport: boolean;inline;    function AVX512BITALGSupport: boolean;inline;    function RDSEEDSupport: boolean;inline;    function ADXSupport: boolean;inline;    function SHASupport: boolean;inline;        function FMASupport: boolean;inline;    function POPCNTSupport: boolean;inline;    function LZCNTSupport: boolean;inline;    function SSE3Support: boolean;inline;    function SSSE3Support: boolean;inline;    function SSE41Support: boolean;inline;    function SSE42Support: boolean;inline;    function MOVBESupport: boolean;inline;    function F16CSupport: boolean;inline;    function RDRANDSupport: boolean;inline;    function RTMSupport: boolean;inline;    function BMI1Support: boolean;inline;    function BMI2Support: boolean;inline;    var      is_sse3_cpu : boolean = false;    function InterlockedCompareExchange128(var Target: Int128Rec; NewValue: Int128Rec; Comperand: Int128Rec): Int128Rec;  implementation{$ASMMODE INTEL}    var      _CMOVSupport,      _AESSupport,      _AVXSupport,      _AVX2Support,      _AVX512FSupport,      _AVX512DQSupport,      _AVX512IFMASupport,      _AVX512PFSupport,      _AVX512ERSupport,      _AVX512CDSupport,      _AVX512BWSupport,      _AVX512VLSupport,      _AVX512VBMISupport,      _AVX512VBMI2Support,      _VAESSupport,      _VCLMULSupport,      _AVX512VNNISupport,      _AVX512BITALGSupport,      _RDSEEDSupport,      _ADXSupport,      _SHASupport,      _FMASupport,      _POPCNTSupport,      _LZCNTSupport,      _SSE3Support,      _SSSE3Support,      _SSE41Support,      _SSE42Support,      _MOVBESupport,      _F16CSupport,      _RDRANDSupport,      _RTMSupport,      _BMI1Support,      _BMI2Support: boolean;{$ASMMODE ATT}    function InterlockedCompareExchange128(var Target: Int128Rec; NewValue: Int128Rec; Comperand: Int128Rec): Int128Rec;      begin{$if FPC_FULLVERSION >= 30101}{$ifndef FPC_PIC}        if _RTMSupport then          begin            asm{$ifdef USE_REAL_INSTRUCTIONS}         .Lretry:              xbegin .Lretry{$else}{   3d:	c7 f8 fa ff ff ff    	xbegin    }         .byte 0xc7,0xf8, 0xfa, 0xff, 0xff, 0xff{$endif}            end;            Result:=Target;            if (Result.Lo=Comperand.Lo) and (Result.Hi=Comperand.Hi) then              Target:=NewValue;            asm{$ifdef USE_REAL_INSTRUCTIONS}              xend{$else}  { 8a:	0f 01 d5             	xend    }         .byte 0x0f, 0x01, 0xd5{$endif}            end;          end        else{$endif FPC_PIC}{$endif FPC_FULLVERSION >= 30101}          RunError(217);      end;{$ASMMODE INTEL}    function cpuid_support : boolean;assembler;      {        Check if the ID-flag can be changed, if changed then CpuID is supported.        Tested under go32v1 and Linux on c6x86 with CpuID enabled and disabled (PFV)      }      asm         push    ebx         pushfd         pushfd         pop     eax         mov     ebx,eax         xor     eax,200000h         push    eax         popfd         pushfd         pop     eax         popfd         and     eax,200000h         and     ebx,200000h         cmp     eax,ebx         setnz   al         pop     ebx      end;    function cr0 : longint;assembler;      asm{$ifdef USE_REAL_INSTRUCTIONS}         mov eax,cr0{$else}         DB 0Fh,20h,0C0h{$endif}         { mov eax,cr0           special registers are not allowed in the assembler                parsers }      end;    function floating_point_emulation : boolean;      begin         {!!!! I don't know currently the position of the EM flag }         { $4 after Ralf Brown's list }         floating_point_emulation:=(cr0 and $4)<>0;      end;{$ASMMODE ATT}    function XGETBV(i : dword) : int64;assembler;      asm        movl %eax,%ecx{$ifdef USE_REAL_INSTRUCTIONS}        xgetbv{$else}        // older FPCs don't know the xgetbv opcode        .byte 0x0f,0x01,0xd0{$endif}      end;    procedure SetupSupport;      var        _edx,_ecx,_ebx,maxcpuidvalue : longint;      begin        is_sse3_cpu:=false;         if cpuid_support then           begin              asm                 pushl %ebx                 movl $0,%eax                 cpuid                 movl %eax,maxcpuidvalue                 popl %ebx              end;              asm                 pushl %ebx                 movl $1,%eax                 cpuid                 movl %edx,_edx                 movl %ecx,_ecx                 popl %ebx              end;              _CMOVSupport:=(_edx and $8000)<>0;              _AESSupport:=(_ecx and $2000000)<>0;              _POPCNTSupport:=(_ecx and $800000)<>0;              _SSE3Support:=(_ecx and $1)<>0;              _SSSE3Support:=(_ecx and $200)<>0;              _SSE41Support:=(_ecx and $80000)<>0;              _SSE42Support:=(_ecx and $100000)<>0;              _MOVBESupport:=(_ecx and $400000)<>0;              _F16CSupport:=(_ecx and $20000000)<>0;              _RDRANDSupport:=(_ecx and $40000000)<>0;              _AVXSupport:=                { XGETBV suspport? }                ((_ecx and $08000000)<>0) and                { xmm and ymm state enabled? }                ((XGETBV(0) and %110)=%110) and                { avx supported? }                ((_ecx and $10000000)<>0);              is_sse3_cpu:=(_ecx and $1)<>0;              _FMASupport:=_AVXSupport and ((_ecx and $1000)<>0);              asm                pushl %ebx                movl $0x80000001,%eax                cpuid                movl %ecx,_ecx                movl %edx,_edx                popl %ebx              end;              _LZCNTSupport:=(_ecx and $20)<>0;              if maxcpuidvalue>=7 then                begin                  asm                    pushl %ebx                    movl $7,%eax                    movl $0,%ecx                    cpuid                    movl %ebx,_ebx                    movl %ecx,_ecx                    movl %edx,_edx                    popl %ebx                  end;                  _AVX2Support:=_AVXSupport and ((_ebx and $20)<>0);                  _AVX512FSupport:=(_ebx and $10000)<>0;                  _AVX512DQSupport:=(_ebx and $20000)<>0;                  _RDSEEDSupport:=(_ebx and $40000)<>0;                  _ADXSupport:=(_ebx and $80000)<>0;                  _AVX512IFMASupport:=(_ebx and $200000)<>0;                  _AVX512PFSupport:=(_ebx and $4000000)<>0;                  _AVX512ERSupport:=(_ebx and $8000000)<>0;                  _AVX512CDSupport:=(_ebx and $10000000)<>0;                  _AVX512BWSupport:=(_ebx and $40000000)<>0;                  _AVX512VBMISupport:=(_ecx and $00000002)<>0;                  _AVX512VBMI2Support:=(_ecx and $00000040)<>0;                   _VAESSupport:=(_ecx and $00000200)<>0;                  _VCLMULSupport:=(_ecx and $00000400)<>0;                  _AVX512VNNISupport:=(_ecx and $00000800)<>0;                  _AVX512BITALGSupport:=(_ecx and $00001000)<>0;                  _SHASupport:=(_ebx and $20000000)<>0;                  _AVX512VLSupport:=(_ebx and $80000000)<>0;                  _BMI1Support:=(_ebx and $8)<>0;                  _BMI2Support:=(_ebx and $100)<>0;                  _RTMSupport:=((_ebx and $800)<>0);                end;           end;      end;    function InterlockedCompareExchange128Support : boolean;      begin        { 32 Bit CPUs have no 128 Bit interlocked exchange support,          but it can simulated using RTM }        result:=_RTMSupport;      end;    function CMOVSupport : boolean;      begin        result:=_CMOVSupport;      end;    function AESSupport : boolean;      begin        result:=_AESSupport;      end;    function AVXSupport: boolean;inline;      begin        result:=_AVXSupport;      end;    function AVX2Support: boolean;inline;      begin        result:=_AVX2Support;      end;    function AVX512FSupport: boolean;inline;      begin        result:=_AVX512FSupport;      end;    function AVX512DQSupport: boolean;inline;      begin        result:=_AVX512DQSupport;      end;    function AVX512IFMASupport: boolean;inline;          begin        result:=_AVX512IFMASupport;      end;    function AVX512PFSupport: boolean;inline;          begin        result:=_AVX512PFSupport;      end;    function AVX512ERSupport: boolean;inline;          begin        result:=_AVX512ERSupport;      end;    function AVX512CDSupport: boolean;inline;          begin        result:=_AVX512CDSupport;      end;    function AVX512BWSupport: boolean;inline;          begin        result:=_AVX512BWSupport;      end;    function AVX512VLSupport: boolean;inline;          begin        result:=_AVX512VLSupport;      end;    function AVX512VBMISupport: boolean;inline;      begin        result:=_AVX512VBMISupport;      end;    function AVX512VBMI2Support: boolean;inline;      begin        result:=_AVX512VBMI2Support;      end;    function VAESSupport: boolean;inline;      begin        result:=_VAESSupport;      end;    function VCLMULSupport: boolean;inline;      begin        result:=_VCLMULSupport;      end;    function AVX512VNNISupport: boolean;inline;          begin        result:=_AVX512VNNISupport;      end;    function AVX512BITALGSupport: boolean;inline;          begin        result:=_AVX512BITALGSupport;      end;    function RDSEEDSupport: boolean;inline;      begin        result:=_RDSEEDSupport;      end;    function ADXSupport: boolean;inline;      begin        result:=_ADXSupport;      end;     function SHASupport: boolean;inline;          begin        result:=_SHASupport;      end;   function FMASupport: boolean;inline;      begin        result:=_FMASupport;      end;    function POPCNTSupport: boolean;inline;      begin        result:=_POPCNTSupport;      end;    function LZCNTSupport: boolean;inline;      begin        result:=_LZCNTSupport;      end;    function SSE3Support: boolean;inline;      begin        result:=_SSE3Support;      end;    function SSSE3Support: boolean;inline;      begin        result:=_SSSE3Support;      end;    function SSE41Support: boolean;inline;      begin        result:=_SSE41Support;      end;    function SSE42Support: boolean;inline;      begin        result:=_SSE42Support;      end;    function MOVBESupport: boolean;inline;      begin        result:=_MOVBESupport;      end;    function F16CSupport: boolean;inline;      begin        result:=_F16CSupport;      end;    function RDRANDSupport: boolean;inline;      begin        result:=_RDRANDSupport;      end;    function RTMSupport: boolean;inline;      begin        result:=_RTMSupport;      end;    function BMI1Support: boolean;inline;      begin        result:=_BMI1Support;      end;    function BMI2Support: boolean;inline;      begin        result:=_BMI2Support;      end;begin  SetupSupport;end.
 |