123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264 |
- {
- This file is part of the Free Pascal run time library.
- Copyright (c) 1999-2000 by Florian Klaempfl
- member of the Free Pascal development team
- 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.
- **********************************************************************}
- { This unit contains some helpful stuff to deal with the mmx extensions }
- unit mmx;
- interface
- type
- tmmxshortint = array[0..7] of shortint;
- tmmxbyte = array[0..7] of byte;
- tmmxword = array[0..3] of word;
- tmmxinteger = array[0..3] of integer;
- tmmxlongint = array[0..1] of longint;
- tmmxcardinal = array[0..1] of cardinal;
- { for the AMD 3D }
- tmmxsingle = array[0..1] of single;
- pmmxshortint = ^tmmxshortint;
- pmmxbyte = ^tmmxbyte;
- pmmxword = ^tmmxword;
- pmmxinteger = ^tmmxinteger;
- {$ifdef HASFIXED}
- pmmxfixed = ^tmmxfixed;
- {$endif HASFIXED}
- pmmxlongint = ^tmmxlongint;
- pmmxcardinal = ^tmmxcardinal;
- { for the AMD 3D }
- pmmxsingle = ^tmmxsingle;
- const
- is_mmx_cpu : boolean = false;
- is_sse_cpu : boolean = false;
- is_sse2_cpu : boolean = false;
- is_amd_3d_cpu : boolean = false;
- is_amd_3d_dsp_cpu : boolean = false;
- is_amd_3d_mmx_cpu : boolean = false;
- { sets all floating point registers to empty
- (use this after mmx usage)
- }
- procedure emms;
- procedure femms;
- implementation
- uses
- cpu;
- {$ASMMODE ATT}
- { return base type of processor: 0 - is Unknown, 10 - is AMD (AuthenticAMD), }
- { 20 - is Intel (GenuineIntel) }
- function getdevel:byte;
- var
- _ebx,_ecx,_edx : longint;
- begin
- getdevel:=0;
- if cpuid_support then
- begin
- asm
- pushl %ebx
- movl $0,%eax
- cpuid
- movl %ebx,_ebx
- movl %ecx,_ecx
- movl %edx,_edx
- popl %ebx
- end;
- if ((_ebx=$68747541) and (_ecx=$444D4163) and (_edx=$69746E65)) then getdevel:=10;
- if ((_ebx=$756E6547) and (_ecx=$6C65746E) and (_edx=$49656E69)) then getdevel:=20;
- end
- end;
- { returns true, if the processor supports the mmx instructions }
- function mmx_support : boolean;
- var
- _edx : longint;
- begin
- if cpuid_support then
- begin
- asm
- pushl %ebx
- movl $1,%eax
- cpuid
- movl %edx,_edx
- popl %ebx
- end;
- mmx_support:=(_edx and $800000)<>0;
- end
- else
- { a cpu with without cpuid instruction supports never mmx }
- mmx_support:=false;
- end;
- function amd_3d_support : boolean;
- var
- _edx : longint;
- begin
- { are there third party cpus supporting amd 3d instructions? }
- if cpuid_support and (getdevel=10) then
- begin
- asm
- pushl %ebx
- movl $0x80000001,%eax
- cpuid
- movl %edx,_edx
- popl %ebx
- end;
- amd_3d_support:=(_edx and $80000000)<>0;
- end
- else
- { a cpu with without cpuid instruction supports never mmx }
- amd_3d_support:=false;
- end;
- function amd_3d_dsp_support : boolean;
- var
- _edx : longint;
- begin
- { are there third party cpus supporting amd dsp instructions? }
- if cpuid_support and (getdevel=10) then
- begin
- asm
- pushl %ebx
- movl $0x80000001,%eax
- cpuid
- movl %edx,_edx
- popl %ebx
- end;
- amd_3d_dsp_support:=(_edx and $40000000)<>0;
- end
- else
- { a cpu with without cpuid instruction supports never mmx }
- amd_3d_dsp_support:=false;
- end;
- function amd_3d_mmx_support : boolean;
- var
- _edx : longint;
- begin
- { are there third party cpus supporting amd mmx instructions? }
- if cpuid_support and (getdevel=10) then
- begin
- asm
- pushl %ebx
- movl $0x80000001,%eax
- cpuid
- movl %edx,_edx
- popl %ebx
- end;
- amd_3d_mmx_support:=(_edx and $400000)<>0;
- end
- else
- { a cpu with without cpuid instruction supports never mmx }
- amd_3d_mmx_support:=false;
- end;
- function sse_support : boolean;
- var
- _edx : longint;
- begin
- if cpuid_support then
- begin
- asm
- pushl %ebx
- movl $1,%eax
- cpuid
- movl %edx,_edx
- popl %ebx
- end;
- sse_support:=(_edx and $2000000)<>0;
- end
- else
- { a cpu with without cpuid instruction supports never sse }
- sse_support:=false;
- end;
- function sse2_support : boolean;
- var
- _edx : longint;
- begin
- if cpuid_support then
- begin
- asm
- pushl %ebx
- movl $1,%eax
- cpuid
- movl %edx,_edx
- popl %ebx
- end;
- sse2_support:=(_edx and $4000000)<>0;
- end
- else
- { a cpu with without cpuid instruction supports never sse2 }
- sse2_support:=false;
- end;
- procedure emms;assembler;
- asm
- { emms instruction not supported by older GNU as version,
- like 2.6 used by EMX target }
- // emms
- .byte 0x0f, 0x77
- end;
- procedure femms;assembler;
- asm
- { femms instruction not supported with older as versions }
- .byte 0x0f, 0x0e
- end;
- var
- oldexitproc : pointer;
- procedure mmxexitproc;
- begin
- exitproc:=oldexitproc;
- if is_amd_3d_cpu then femms else emms;
- end;
- begin
- if mmx_support then
- begin
- is_mmx_cpu:=true;
- if amd_3d_support then
- begin
- is_amd_3d_cpu:=true;
- is_amd_3d_dsp_cpu:=amd_3d_dsp_support;
- is_amd_3d_mmx_cpu:=amd_3d_mmx_support;
- end;
- is_sse_cpu:=sse_support;
- is_sse2_cpu:=sse2_support;
- { the exit code sets the fpu stack to empty }
- oldexitproc:=exitproc;
- exitproc:=@mmxexitproc;
- end;
- end.
|