mirror of
https://gitlab.com/freepascal.org/fpc/source.git
synced 2025-04-06 06:28:04 +02:00
2464 lines
64 KiB
PHP
2464 lines
64 KiB
PHP
{
|
||
|
||
This file is part of the Free Pascal Run time library.
|
||
Copyright (c) 1999-2008 by 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.
|
||
|
||
**********************************************************************}
|
||
|
||
{ ObjpasInt is the integer type, equivalent to Objpas.Integer (the Integer
|
||
type in ObjFpc and Delphi modes). It is defined here for use in the
|
||
implementation part of the System unit. }
|
||
{$ifdef CPU16}
|
||
type
|
||
ObjpasInt = SmallInt;
|
||
{$else CPU16}
|
||
type
|
||
ObjpasInt = LongInt;
|
||
{$endif CPU16}
|
||
|
||
{****************************************************************************
|
||
Local types
|
||
****************************************************************************}
|
||
|
||
{$ifdef FPC_HAS_FEATURE_EXITCODE}
|
||
{$if High(errorcode)<>maxExitCode}
|
||
{$define FPC_LIMITED_EXITCODE}
|
||
{$endif}
|
||
{$endif FPC_HAS_FEATURE_EXITCODE}
|
||
|
||
Procedure HandleError (Errno : TExitCode); external name 'FPC_HANDLEERROR';
|
||
Procedure HandleErrorFrame (Errno : TExitCode;frame : Pointer); forward;
|
||
Procedure HandleErrorAddrFrame (Errno : TExitCode;addr : CodePointer; frame : Pointer); forward;
|
||
Procedure HandleErrorAddrFrameInd (Errno : TExitCode;addr : CodePointer; frame : Pointer); forward;
|
||
|
||
{$ifdef FPC_HAS_FEATURE_TEXTIO}
|
||
type
|
||
FileFunc = Procedure(var t : TextRec);
|
||
{$endif FPC_HAS_FEATURE_TEXTIO}
|
||
|
||
const
|
||
{$if defined(CPUAVR)}
|
||
STACK_MARGIN_MAX = 64; { Stack size margin for stack checking }
|
||
{$elseif defined(CPUZ80)}
|
||
STACK_MARGIN_MAX = 64; { Stack size margin for stack checking }
|
||
{$elseif defined(CPULOONGARCH64)}
|
||
STACK_MARGIN_MAX = 65536; { Stack size margin for stack checking }
|
||
{$else}
|
||
STACK_MARGIN_MAX = 16384; { Stack size margin for stack checking }
|
||
{$endif}
|
||
StackMargin: ptruint = STACK_MARGIN_MAX;
|
||
{ Random / Randomize constants }
|
||
OldRandSeed : Cardinal = Cardinal(not Cardinal(0)); // see ReseedGlobalRNG and GlobalXsr128_32 initialization.
|
||
|
||
{ For Error Handling.}
|
||
ErrorBase : Pointer = nil;public name 'FPC_ERRORBASE';
|
||
|
||
{ Used by the ansi/widestrings and maybe also other things in the future }
|
||
var
|
||
{ widechar, because also used by widestring -> pwidechar conversions }
|
||
emptychar : widechar;public name 'FPC_EMPTYCHAR';
|
||
{$ifndef FPC_NO_GENERIC_STACK_CHECK}
|
||
{ if the OS does the stack checking, we don't need any stklen from the
|
||
main program }
|
||
initialstklen : SizeUint{$ifndef FPC_HAS_INDIRECT_ENTRY_INFORMATION}; external name '__stklen';{$else} = 0;{$endif}
|
||
{$endif FPC_NO_GENERIC_STACK_CHECK}
|
||
|
||
{$ifdef FPC_HAS_INDIRECT_ENTRY_INFORMATION}
|
||
EntryInformation: TEntryInformation;
|
||
{$endif FPC_HAS_INDIRECT_ENTRY_INFORMATION}
|
||
|
||
var
|
||
{$ifdef FPC_HAS_INDIRECT_ENTRY_INFORMATION}
|
||
{$ifdef FPC_HAS_RESSTRINITS}
|
||
FPCResStrInitTables : Pointer;public name '_FPC_ResStrInitTables';
|
||
{$endif FPC_HAS_RESSTRINITS}
|
||
FPCResourceStringTables : Pointer;public name '_FPC_ResourceStringTables';
|
||
FPCResLocation : Pointer;public name'_FPC_ResLocation';
|
||
{$else FPC_HAS_INDIRECT_ENTRY_INFORMATION}
|
||
{$ifdef FPC_HAS_RESSTRINITS}
|
||
FPCResStrInitTablesVar : record end; external name 'FPC_RESSTRINITTABLES';
|
||
FPCResStrInitTables : Pointer = @FPCResStrInitTablesVar;public name '_FPC_ResStrInitTables';
|
||
{$endif FPC_HAS_RESSTRINITS}
|
||
FPCResourceStringTablesVar : record end; External Name 'FPC_RESOURCESTRINGTABLES';
|
||
FPCResourceStringTables : Pointer = @FPCResourceStringTablesVar;public name '_FPC_ResourceStringTables';
|
||
{$endif FPC_HAS_INDIRECT_ENTRY_INFORMATION}
|
||
|
||
{$ifdef FPC_HAS_INDIRECT_ENTRY_INFORMATION}
|
||
procedure SetupEntryInformation(constref info: TEntryInformation);[public,alias:'_FPC_SetupEntryInformation'];
|
||
begin
|
||
EntryInformation := info;
|
||
FPCResStrInitTables := info.ResStrInitTables;
|
||
FPCResourceStringTables := info.ResourceStringTables;
|
||
FPCResLocation := info.ResLocation;
|
||
{$ifdef FPC_SYSTEM_HAS_OSSETUPENTRYINFORMATION}
|
||
OSSetupEntryInformation(info);
|
||
{$endif FPC_SYSTEM_HAS_OSSETUPENTRYINFORMATION}
|
||
end;
|
||
{$endif FPC_HAS_INDIRECT_ENTRY_INFORMATION}
|
||
|
||
{$if not defined(FPC_SYSTEM_FPU_CUSTOM_CONTROL_WORD) and not defined(FPUNONE) and not defined(FPC_SYSTEM_FPUCW_IMMUTABLE)}
|
||
var
|
||
{ Control word of the fpu that determes which exceptions will be thrown and
|
||
potentially other flags (precision, exception-happened flags, ... Used to
|
||
initialise the FPU when starting a new thread (in SysResetFPU).
|
||
|
||
Some platforms already have a legacy/existing global variable holding this
|
||
information (FPC_SYSTEM_FPU_CUSTOM_CONTROL_WORD). In case of FPU_NONE,
|
||
we don't support floating point at all. And in case of
|
||
FPC_SYSTEM_FPUCW_IMMUTABLE, either the control word cannot be changed
|
||
(e.g. wasm), or FPU operations are supported by only as softfloat }
|
||
DefaultFPUControlWord: TNativeFPUControlWord;
|
||
{$endif}
|
||
|
||
{ checks whether the given suggested size for the stack of the current
|
||
thread is acceptable. If this is the case, returns it unaltered.
|
||
Otherwise it should return an acceptable value.
|
||
|
||
Operating systems that automatically expand their stack on demand, should
|
||
simply return a very large value.
|
||
Operating systems which do not have a possibility to retrieve stack size
|
||
information, should simply return the given stklen value (This is the default
|
||
implementation).
|
||
}
|
||
{$ifdef FPC_HAS_FEATURE_STACKCHECK}
|
||
function CheckInitialStkLen(stklen : SizeUInt) : SizeUInt; forward;
|
||
{$endif FPC_HAS_FEATURE_STACKCHECK}
|
||
|
||
{*****************************************************************************
|
||
OS dependent Helpers/Syscalls
|
||
*****************************************************************************}
|
||
|
||
{ for some OSes do_isdevice is defined in sysos.inc, but for others (win32)
|
||
it isn't, and is used before the actual definition is encountered }
|
||
|
||
{$ifdef FPC_HAS_FEATURE_FILEIO}
|
||
function do_isdevice(handle:thandle):boolean;forward;
|
||
{$endif FPC_HAS_FEATURE_FILEIO}
|
||
|
||
|
||
{$i sysos.inc}
|
||
|
||
|
||
{****************************************************************************
|
||
Include processor specific routines
|
||
****************************************************************************}
|
||
|
||
{$ifdef FPC_USE_LIBC}
|
||
{$ifdef USE_CPU_MOVE}
|
||
{ Avoid use of generic C code for move procedure }
|
||
{$define FPC_SYSTEM_HAS_MOVE}
|
||
{$endif}
|
||
{ Under Haiku, bcopy cause a problem when searching for include file
|
||
in the compiler. So, we use the internal implementation for now
|
||
under BeOS and Haiku. }
|
||
{$ifndef BEOS}
|
||
{ prefer libc implementations over our own, as they're most likely faster }
|
||
{$i cgeneric.inc}
|
||
{ is now declared as external reference to another routine in the interface }
|
||
{$i cgenstr.inc}
|
||
{$endif}
|
||
{$ifdef USE_CPU_MOVE}
|
||
{ Avoid use of generic C code for move procedure }
|
||
{$undef FPC_SYSTEM_HAS_MOVE}
|
||
{$endif}
|
||
{$endif FPC_USE_LIBC}
|
||
|
||
{$ifdef cpui386}
|
||
{$ifdef SYSPROCDEFINED}
|
||
{$Error Can't determine processor type !}
|
||
{$endif}
|
||
{$i i386.inc} { Case dependent, don't change }
|
||
{$define SYSPROCDEFINED}
|
||
{$endif cpui386}
|
||
|
||
{$ifdef cpui8086}
|
||
{$ifdef SYSPROCDEFINED}
|
||
{$Error Can't determine processor type !}
|
||
{$endif}
|
||
{$i i8086.inc} { Case dependent, don't change }
|
||
{$define SYSPROCDEFINED}
|
||
{$endif cpui8086}
|
||
|
||
{$ifdef cpum68k}
|
||
{$ifdef SYSPROCDEFINED}
|
||
{$Error Can't determine processor type !}
|
||
{$endif}
|
||
{$i m68k.inc} { Case dependent, don't change }
|
||
{$define SYSPROCDEFINED}
|
||
{$endif cpum68k}
|
||
|
||
{$ifdef cpux86_64}
|
||
{$ifdef SYSPROCDEFINED}
|
||
{$Error Can't determine processor type !}
|
||
{$endif}
|
||
{$i x86_64.inc} { Case dependent, don't change }
|
||
{$define SYSPROCDEFINED}
|
||
{$endif cpux86_64}
|
||
|
||
{$ifdef cpupowerpc32}
|
||
{$ifdef SYSPROCDEFINED}
|
||
{$Error Can't determine processor type !}
|
||
{$endif}
|
||
{$i powerpc.inc} { Case dependent, don't change }
|
||
{$define SYSPROCDEFINED}
|
||
{$endif cpupowerpc32}
|
||
|
||
{$ifdef cpupowerpc64}
|
||
{$ifdef SYSPROCDEFINED}
|
||
{$Error Can't determine processor type !}
|
||
{$endif}
|
||
{$i powerpc64.inc} { Case dependent, don't change }
|
||
{$define SYSPROCDEFINED}
|
||
{$endif cpupowerpc64}
|
||
|
||
{$ifdef cpualpha}
|
||
{$ifdef SYSPROCDEFINED}
|
||
{$Error Can't determine processor type !}
|
||
{$endif}
|
||
{$i alpha.inc} { Case dependent, don't change }
|
||
{$define SYSPROCDEFINED}
|
||
{$endif cpualpha}
|
||
|
||
{$ifdef cpuiA64}
|
||
{$ifdef SYSPROCDEFINED}
|
||
{$Error Can't determine processor type !}
|
||
{$endif}
|
||
{$i ia64.inc} { Case dependent, don't change }
|
||
{$define SYSPROCDEFINED}
|
||
{$endif cpuiA64}
|
||
|
||
{$ifdef cpusparc}
|
||
{$ifdef SYSPROCDEFINED}
|
||
{$Error Can't determine processor type !}
|
||
{$endif}
|
||
{$i sparc.inc} { Case dependent, don't change }
|
||
{$define SYSPROCDEFINED}
|
||
{$endif cpusparc}
|
||
|
||
{$ifdef cpusparc64}
|
||
{$ifdef SYSPROCDEFINED}
|
||
{$Error Can't determine processor type !}
|
||
{$endif}
|
||
{$i sparc64.inc} { Case dependent, don't change }
|
||
{$define SYSPROCDEFINED}
|
||
{$endif cpusparc64}
|
||
|
||
{$ifdef cpuarm}
|
||
{$ifdef SYSPROCDEFINED}
|
||
{$Error Can't determine processor type !}
|
||
{$endif}
|
||
{$i armdefines.inc}
|
||
{$if defined(CPUTHUMB2)}
|
||
{$i thumb2.inc} { Case dependent, don't change }
|
||
{$else}
|
||
{$if defined(CPUTHUMB)}
|
||
{$i thumb.inc} { Case dependent, don't change }
|
||
{$else}
|
||
{$i arm.inc} { Case dependent, don't change }
|
||
{$endif}
|
||
{$endif}
|
||
{$define SYSPROCDEFINED}
|
||
{$endif cpuarm}
|
||
|
||
{$ifdef cpuavr}
|
||
{$ifdef SYSPROCDEFINED}
|
||
{$Error Can't determine processor type !}
|
||
{$endif}
|
||
{$i avr.inc} { Case dependent, don't change }
|
||
{$define SYSPROCDEFINED}
|
||
{$endif cpuavr}
|
||
|
||
{$ifdef cpumipsel}
|
||
{$ifdef SYSPROCDEFINED}
|
||
{$Error Can't determine processor type !}
|
||
{$endif}
|
||
{ there is no mipsel.inc, we use mips.inc instead }
|
||
{$i mips.inc} { Case dependent, don't change }
|
||
{$define SYSPROCDEFINED}
|
||
{$endif cpumipsel}
|
||
|
||
{$ifdef cpumipseb}
|
||
{$ifdef SYSPROCDEFINED}
|
||
{$Error Can't determine processor type !}
|
||
{$endif}
|
||
{$i mips.inc} { Case dependent, don't change }
|
||
{$define SYSPROCDEFINED}
|
||
{$endif cpumipseb}
|
||
|
||
{$ifdef cpumips64el}
|
||
{$ifdef SYSPROCDEFINED}
|
||
{$Error Can't determine processor type !}
|
||
{$endif}
|
||
{$i mips64el.inc} { Case dependent, don't change }
|
||
{$define SYSPROCDEFINED}
|
||
{$endif cpumips64el}
|
||
|
||
{$ifdef cpumips64eb}
|
||
{$ifdef SYSPROCDEFINED}
|
||
{$Error Can't determine processor type !}
|
||
{$endif}
|
||
{$i mips64.inc} { Case dependent, don't change }
|
||
{$define SYSPROCDEFINED}
|
||
{$endif cpumips64eb}
|
||
|
||
{$ifdef cpuaarch64}
|
||
{$ifdef SYSPROCDEFINED}
|
||
{$Error Can't determine processor type !}
|
||
{$endif}
|
||
{$i aarch64.inc} { Case dependent, don't change }
|
||
{$define SYSPROCDEFINED}
|
||
{$endif cpuaarch64}
|
||
|
||
{$ifdef cpuriscv32}
|
||
{$ifdef SYSPROCDEFINED}
|
||
{$Error Can't determine processor type !}
|
||
{$endif}
|
||
{$i riscv32.inc} { Case dependent, don't change }
|
||
{$define SYSPROCDEFINED}
|
||
{$endif cpuriscv32}
|
||
|
||
{$ifdef cpuriscv64}
|
||
{$ifdef SYSPROCDEFINED}
|
||
{$Error Can't determine processor type !}
|
||
{$endif}
|
||
{$i riscv64.inc} { Case dependent, don't change }
|
||
{$define SYSPROCDEFINED}
|
||
{$endif cpuriscv64}
|
||
|
||
{$ifdef cpuxtensa}
|
||
{$ifdef SYSPROCDEFINED}
|
||
{$Error Can't determine processor type !}
|
||
{$endif}
|
||
{$i xtensa.inc} { Case dependent, don't change }
|
||
{$define SYSPROCDEFINED}
|
||
{$endif cpuxtensa}
|
||
|
||
{$ifdef cpuz80}
|
||
{$ifdef SYSPROCDEFINED}
|
||
{$Error Can't determine processor type !}
|
||
{$endif}
|
||
{$i z80.inc} { Case dependent, don't change }
|
||
{$define SYSPROCDEFINED}
|
||
{$endif cpuz80}
|
||
|
||
{$ifdef cpuwasm32}
|
||
{$ifdef SYSPROCDEFINED}
|
||
{$Error Can't determine processor type !}
|
||
{$endif}
|
||
{$i wasm32.inc} { Case dependent, don't change }
|
||
{$define SYSPROCDEFINED}
|
||
{$endif cpuwasm32}
|
||
|
||
{$ifdef cpuloongarch64}
|
||
{$ifdef SYSPROCDEFINED}
|
||
{$Error Can't determine processor type !}
|
||
{$endif}
|
||
{$i loongarch64.inc} { Case dependent, don't change }
|
||
{$define SYSPROCDEFINED}
|
||
{$endif cpuloongarch64}
|
||
|
||
{$ifndef SYSPROCDEFINED}
|
||
{$Error Can't determine processor type !}
|
||
{$endif}
|
||
|
||
procedure fillchar(var x;count : {$ifdef FILLCHAR_HAS_SIZEUINT_COUNT}SizeUInt{$else}SizeInt{$endif};value : boolean);
|
||
begin
|
||
fillchar(x,count,byte(value));
|
||
end;
|
||
|
||
|
||
procedure fillchar(var x;count : {$ifdef FILLCHAR_HAS_SIZEUINT_COUNT}SizeUInt{$else}SizeInt{$endif};value : AnsiChar);
|
||
begin
|
||
fillchar(x,count,byte(value));
|
||
end;
|
||
|
||
|
||
procedure FillByte (var x;count : {$ifdef FILLCHAR_HAS_SIZEUINT_COUNT}SizeUInt{$else}SizeInt{$endif};value : byte );
|
||
begin
|
||
FillChar (X,Count,VALUE);
|
||
end;
|
||
|
||
|
||
function IndexChar(Const buf;len:SizeInt;b:AnsiChar):SizeInt;
|
||
begin
|
||
IndexChar:=IndexByte(Buf,Len,byte(B));
|
||
end;
|
||
|
||
|
||
function IndexChar(const buf; len: SizeInt; b: widechar): SizeInt;
|
||
|
||
begin
|
||
IndexChar:=IndexWord(buf,len,Word(b));
|
||
end;
|
||
|
||
|
||
function CompareChar(Const buf1,buf2;len:SizeInt):SizeInt;
|
||
begin
|
||
CompareChar:=CompareByte(buf1,buf2,len);
|
||
end;
|
||
|
||
|
||
procedure fpc_zeromem(p:pointer;len:sizeuint);
|
||
begin
|
||
FillChar(p^,len,0);
|
||
end;
|
||
|
||
|
||
procedure fpc_fillmem(out data;len:sizeuint;b : byte);
|
||
begin
|
||
FillChar(data,len,AnsiChar(b));
|
||
end;
|
||
|
||
{ Include generic pascal only routines which are not defined in the processor
|
||
specific include file }
|
||
{$I generic.inc}
|
||
|
||
|
||
{****************************************************************************
|
||
Set Handling
|
||
****************************************************************************}
|
||
|
||
{ Include set support which is processor specific}
|
||
{$i set.inc}
|
||
{ Include generic pascal routines for sets if the processor }
|
||
{ specific routines are not available. }
|
||
{$i genset.inc}
|
||
|
||
|
||
{****************************************************************************
|
||
Math Routines
|
||
****************************************************************************}
|
||
|
||
function Hi(b : byte): byte;{$ifdef SYSTEMINLINE}inline;{$endif}
|
||
begin
|
||
Hi := b shr 4
|
||
end;
|
||
|
||
function Lo(b : byte): byte;{$ifdef SYSTEMINLINE}inline;{$endif}
|
||
begin
|
||
Lo := b and $0f
|
||
end;
|
||
|
||
Function Swap (X : Word) : Word;{$ifdef SYSTEMINLINE}inline;{$endif}
|
||
Begin
|
||
Swap := SwapEndian(X);
|
||
End;
|
||
|
||
Function Swap (X : Integer) : Integer;{$ifdef SYSTEMINLINE}inline;{$endif}
|
||
Begin
|
||
Swap := SwapEndian(X);
|
||
End;
|
||
|
||
Function Swap (X : Longint) : Longint;{$ifdef SYSTEMINLINE}inline;{$endif}
|
||
Begin
|
||
Swap:=(X shl 16) + (X shr 16);
|
||
End;
|
||
|
||
Function Swap (X : Cardinal) : Cardinal;{$ifdef SYSTEMINLINE}inline;{$endif}
|
||
Begin
|
||
Swap:=(X shl 16) + (X shr 16);
|
||
End;
|
||
|
||
Function Swap (X : QWord) : QWord;{$ifdef SYSTEMINLINE}inline;{$endif}
|
||
Begin
|
||
Swap:=(X shl 32) + (X shr 32);
|
||
End;
|
||
|
||
Function Swap (X : Int64) : Int64;{$ifdef SYSTEMINLINE}inline;{$endif}
|
||
Begin
|
||
Swap:=(X shl 32) + (X shr 32);
|
||
End;
|
||
|
||
{$ifdef SUPPORT_DOUBLE}
|
||
operator := (b:real48) d:double;{$ifdef SYSTEMINLINE}inline;{$endif}
|
||
begin
|
||
D:=real2double(b);
|
||
end;
|
||
{$endif SUPPORT_DOUBLE}
|
||
|
||
{$ifdef SUPPORT_EXTENDED}
|
||
operator := (b:real48) e:extended;{$ifdef SYSTEMINLINE}inline;{$endif}
|
||
begin
|
||
e:=real2double(b);
|
||
end;
|
||
{$endif SUPPORT_EXTENDED}
|
||
|
||
{$ifndef FPUNONE}
|
||
{$ifdef FPC_USE_LIBC}
|
||
{ Include libc versions }
|
||
{$i cgenmath.inc}
|
||
{$endif FPC_USE_LIBC}
|
||
{ Include processor specific routines }
|
||
{$I math.inc}
|
||
{ Include generic version }
|
||
{$I genmath.inc}
|
||
{$endif}
|
||
|
||
{$i gencurr.inc}
|
||
|
||
|
||
function aligntoptr(p : pointer) : pointer;inline;
|
||
begin
|
||
{$ifdef FPC_REQUIRES_PROPER_ALIGNMENT}
|
||
result:=align(p,sizeof(p));
|
||
{$else FPC_REQUIRES_PROPER_ALIGNMENT}
|
||
result:=p;
|
||
{$endif FPC_REQUIRES_PROPER_ALIGNMENT}
|
||
end;
|
||
|
||
|
||
function aligntoqword(p : pointer) : pointer;inline;
|
||
{$push}
|
||
{$packrecords c}
|
||
type
|
||
TAlignCheck = record
|
||
b : byte;
|
||
q : qword;
|
||
end;
|
||
{$pop}
|
||
begin
|
||
{$ifdef FPC_REQUIRES_PROPER_ALIGNMENT}
|
||
result:=align(p,PtrInt(@TAlignCheck(nil^).q))
|
||
{$else FPC_REQUIRES_PROPER_ALIGNMENT}
|
||
result:=p;
|
||
{$endif FPC_REQUIRES_PROPER_ALIGNMENT}
|
||
end;
|
||
|
||
|
||
function AlignTypeData(p : pointer) : pointer; inline;
|
||
begin
|
||
{$if defined(CPUM68K)}
|
||
result := aligntoqword(p);
|
||
{$else CPUM68K}
|
||
result := aligntoptr(p);
|
||
{$endif CPUM68K}
|
||
end;
|
||
|
||
|
||
{$if not defined(VER3_2)}
|
||
type
|
||
TRttiDataCommon =
|
||
{$ifdef FPC_REQUIRES_PROPER_ALIGNMENT}
|
||
packed
|
||
{$endif}
|
||
record
|
||
Attrs: Pointer;
|
||
end;
|
||
{$endif not VER3_2}
|
||
|
||
|
||
{****************************************************************************
|
||
Subroutines for String handling
|
||
****************************************************************************}
|
||
|
||
{ Needs to be before RTTI handling }
|
||
|
||
{$i sstrings.inc}
|
||
|
||
{ requires sstrings.inc for initval }
|
||
{$I int64p.inc}
|
||
{$I int64.inc}
|
||
|
||
{Requires int64.inc, since that contains the VAL functions for int64 and qword}
|
||
{$ifdef FPC_HAS_FEATURE_ANSISTRINGS}
|
||
{$i astrings.inc}
|
||
{$endif FPC_HAS_FEATURE_ANSISTRINGS}
|
||
|
||
{$ifdef FPC_HAS_FEATURE_WIDESTRINGS}
|
||
{$ifndef FPC_WIDESTRING_EQUAL_UNICODESTRING}
|
||
{$i wstrings.inc}
|
||
{$endif FPC_WIDESTRING_EQUAL_UNICODESTRING}
|
||
{$i ustrings.inc}
|
||
{$endif FPC_HAS_FEATURE_WIDESTRINGS}
|
||
|
||
{****************************************************************************
|
||
Run-Time Type Information (RTTI) declarations
|
||
****************************************************************************}
|
||
|
||
{$ifdef FPC_HAS_FEATURE_RTTI}
|
||
{$i rttidecl.inc}
|
||
{$endif FPC_HAS_FEATURE_RTTI}
|
||
|
||
{$i aliases.inc}
|
||
|
||
{*****************************************************************************
|
||
Dynamic Array support
|
||
*****************************************************************************}
|
||
|
||
{$ifdef FPC_HAS_FEATURE_DYNARRAYS}
|
||
{$i dynarr.inc}
|
||
{$endif FPC_HAS_FEATURE_DYNARRAYS}
|
||
|
||
{*****************************************************************************
|
||
Object Pascal support
|
||
*****************************************************************************}
|
||
|
||
{$ifdef FPC_HAS_FEATURE_CLASSES}
|
||
{$i objpas.inc}
|
||
{$endif FPC_HAS_FEATURE_CLASSES}
|
||
|
||
{*****************************************************************************
|
||
Variant support
|
||
*****************************************************************************}
|
||
|
||
{$ifdef FPC_HAS_FEATURE_VARIANTS}
|
||
{$i variant.inc}
|
||
{$endif FPC_HAS_FEATURE_VARIANTS}
|
||
|
||
{****************************************************************************
|
||
Run-Time Type Information (RTTI)
|
||
****************************************************************************}
|
||
|
||
{$ifdef FPC_HAS_FEATURE_RTTI}
|
||
{$i rtti.inc}
|
||
{$endif FPC_HAS_FEATURE_RTTI}
|
||
|
||
{$if defined(FPC_HAS_FEATURE_RANDOM)}
|
||
{$ifndef FPC_USE_SIMPLE_RANDOM}
|
||
|
||
{$push} // random
|
||
{$r-,q-}
|
||
|
||
// SplitMix64, being a generator of fundamentally different nature,
|
||
// is a good (recommended by the author) way to seed Xoshiro.
|
||
//
|
||
// https://xoroshiro.di.unimi.it/splitmix64.c
|
||
//
|
||
// This is a generator with 64-bit state, 64-bit output, and 2^64 period.
|
||
|
||
type
|
||
SplitMix64 = object
|
||
procedure Setup(const seed: uint64); inline;
|
||
function Next: uint64;
|
||
private
|
||
state: uint64;
|
||
end;
|
||
|
||
procedure SplitMix64.Setup(const seed: uint64);
|
||
begin
|
||
state := seed;
|
||
end;
|
||
|
||
function SplitMix64.Next: uint64;
|
||
var
|
||
z: uint64;
|
||
begin
|
||
z := state + $9e3779b97f4a7c15;
|
||
state := z;
|
||
z := (z xor (z shr 30)) * $bf58476d1ce4e5b9;
|
||
z := (z xor (z shr 27)) * $94d049bb133111eb;
|
||
result := z xor (z shr 31);
|
||
end;
|
||
|
||
// Xoshiro128** is an all-purpose RNG.
|
||
// http://prng.di.unimi.it/xoshiro128starstar.c
|
||
//
|
||
// State must not be everywhere zero. With SplitMix64 initialization, it won't. :)
|
||
// 128-bit state, 32-bit output, 2^128-1 period.
|
||
|
||
type
|
||
Xoshiro128ss_32 = object
|
||
procedure Setup(const seed: uint64);
|
||
function Next: uint32; inline; // inlined as it is actually internal and called only through xsr128_32_u32rand
|
||
private
|
||
state: array[0 .. 3] of longword;
|
||
end;
|
||
|
||
procedure Xoshiro128ss_32.Setup(const seed: uint64);
|
||
var
|
||
sm: SplitMix64;
|
||
x: uint64;
|
||
begin
|
||
sm.Setup(seed);
|
||
x := sm.Next;
|
||
state[0] := Lo(x); state[1] := Hi(x);
|
||
x := sm.Next;
|
||
state[2] := Lo(x); state[3] := Hi(x);
|
||
end;
|
||
|
||
function Xoshiro128ss_32.Next: uint32;
|
||
var
|
||
s0, s1, s2, s3: uint32;
|
||
begin
|
||
s0 := state[0];
|
||
s1 := state[1];
|
||
s2 := state[2] xor s0;
|
||
s3 := state[3] xor s1;
|
||
result := RolDWord(s1 * 5, 7) * 9;
|
||
|
||
state[0] := s0 xor s3;
|
||
state[1] := s1 xor s2;
|
||
state[2] := s2 xor (s1 shl 9);
|
||
state[3] := RolDWord(s3, 11);
|
||
end;
|
||
|
||
var
|
||
// Just in case user sets RandSeed := not Cardinal(0) (RandSeed := $FFFFFFFF) from the start,
|
||
// thus bypassing first-time RandSeed <> OldRandSeed check,
|
||
// there will still be a precomputed initialization for RandSeed = $FFFFFFFF.
|
||
GlobalXsr128_32: Xoshiro128ss_32 = (state: ($AFF181C0, $73B13BA2, $1340D3B4, $61204305));
|
||
|
||
procedure ReseedGlobalRNG;
|
||
begin
|
||
{ Detect resets of randseed
|
||
|
||
This will break if someone coincidentally uses not(randseed) as the
|
||
next randseed, but it's much more common that you will reset randseed
|
||
to the same value as before to regenerate the same sequence of numbers
|
||
}
|
||
GlobalXsr128_32.Setup(RandSeed);
|
||
RandSeed := not RandSeed;
|
||
OldRandSeed := RandSeed;
|
||
end;
|
||
|
||
function xsr128_32_u32rand: uint32;
|
||
begin
|
||
if RandSeed <> OldRandSeed then ReseedGlobalRNG;
|
||
result := GlobalXsr128_32.Next;
|
||
end;
|
||
|
||
// There's still a flaw: repeated assignments of RandSeed := $FFFFFFFF from the start, i.e.
|
||
// RandSeed := $FFFFFFFF;
|
||
// writeln(random(10));
|
||
// RandSeed := $FFFFFFFF;
|
||
// writeln(random(10));
|
||
// won't reset RNG.
|
||
//
|
||
// But this is already the case with carefully crafted RandSeeds,
|
||
// so I doubt an additional check like "if (RandSeed <> OldRandSeed) or not RngInitialized" is justified.
|
||
|
||
function random(l:longint): longint;
|
||
begin
|
||
{ otherwise we can return values = l (JM) }
|
||
if (l < 0) then
|
||
inc(l);
|
||
random := longint((int64(xsr128_32_u32rand)*l) shr 32);
|
||
end;
|
||
|
||
function random(l:int64): int64;
|
||
var
|
||
a, b, c, d: cardinal;
|
||
q, bd, ad, bc, ac: qword;
|
||
carry: qword;
|
||
begin
|
||
if (l>=Low(int32)) and (l<=High(int32)) then
|
||
begin
|
||
{ random(longint(l)), inlined. This makes random(NativeType) on 64-bit platforms match 32-bit when possible. }
|
||
if (l < 0) then
|
||
inc(l);
|
||
exit(longint(int64(xsr128_32_u32rand)*l shr 32));
|
||
end;
|
||
|
||
if (l < 0) then
|
||
begin
|
||
inc(l);
|
||
q:=qword(-l)
|
||
end
|
||
else
|
||
q:=qword(l);
|
||
|
||
a:=xsr128_32_u32rand;
|
||
b:=xsr128_32_u32rand;
|
||
|
||
c:=q shr 32;
|
||
d:=cardinal(q);
|
||
|
||
bd:=qword(b)*d;
|
||
ad:=qword(a)*d;
|
||
bc:=qword(b)*c;
|
||
ac:=qword(a)*c;
|
||
|
||
// We only need the carry bit
|
||
carry:=((bd shr 32)+cardinal(ad)+cardinal(bc)) shr 32;
|
||
|
||
// Calculate the final result
|
||
result:=int64(carry+(ad shr 32)+(bc shr 32)+ac);
|
||
if l<0 then
|
||
result:=-result;
|
||
end;
|
||
|
||
{$ifndef FPUNONE}
|
||
function random: extended;
|
||
var
|
||
res: double;
|
||
r0, exponent: uint32;
|
||
begin
|
||
{ There are
|
||
|
||
- 2⁵² floats uniformly distributed over the range [0.5; 1): exponent = 2⁻¹, mantissa = (1.)all 52-bit values from 00...0 to 11...1,
|
||
- another 2⁵² in the range [0.25; 0.5): exponent = 2⁻²,
|
||
- another 2⁵² in the range [0.125; 0.25): exponent = 2⁻³,
|
||
|
||
and so on. Each next range is 0.5× the size of the previous one ⇒ 0.5× probability.
|
||
So we determine the range by flipping a coin until we get heads (exponent = 2^(−1 − BSF(infinite stream of random bits)),
|
||
and select one of 2⁵² values in that range.
|
||
|
||
Compared to this, random_52_bits / 2⁵² value loses N bits of precision when it falls in the Nth range; as a consequence,
|
||
minimum of 2^N random values has around N trailing zeros in its mantissa. }
|
||
|
||
{ 52 bits (double precision) go to mantissa: r0[0:19] + one_more_u32rand[0:31]. }
|
||
r0:=xsr128_32_u32rand;
|
||
PUint64(@res)^:=uint64(r0 and (1 shl 20-1)) shl 32 or xsr128_32_u32rand;
|
||
|
||
{ Exponent = −1 − (count of zeros in the stream of random bits). There are 12 bits left in r0. }
|
||
exponent:=1023-1-31; { Biased exponent, - 31 for Bsr(r0). }
|
||
if r0 shr 20=0 then
|
||
begin
|
||
inc(exponent,20); { Subtract 12 bits the first time and 32 on subsequent iterations. }
|
||
repeat
|
||
dec(exponent,32);
|
||
r0:=xsr128_32_u32rand;
|
||
until r0<>0;
|
||
end;
|
||
PUint64(@res)^:=PUint64(@res)^ or uint64(exponent+BsrDWord(r0)) shl 52;
|
||
result:=res;
|
||
end;
|
||
{$endif}
|
||
|
||
{$pop} // random
|
||
|
||
{$else FPC_USE_SIMPLE_RANDOM}
|
||
|
||
{ A simple implementation of random. TP/Delphi compatible. }
|
||
|
||
const
|
||
QRAN_A = 134775813;
|
||
QRAN_C = 1;
|
||
|
||
function rand_next: cardinal;
|
||
var
|
||
s: cardinal;
|
||
begin
|
||
s:=RandSeed*QRAN_A+QRAN_C;
|
||
RandSeed:=s;
|
||
rand_next:=s;
|
||
end;
|
||
|
||
function random(l: word): word;
|
||
var
|
||
s,ss: cardinal;
|
||
begin
|
||
s:=rand_next;
|
||
{ use 32-bit multiplications here }
|
||
ss:=(s shr 16)*l;
|
||
s:=(s and $FFFF)*l shr 16;
|
||
random:=(ss + s) shr 16;
|
||
end;
|
||
|
||
function random(l: longint): longint;
|
||
begin
|
||
random:=int64(rand_next)*l shr 32;
|
||
end;
|
||
|
||
function random(l:int64):int64;
|
||
begin
|
||
random:=random(longint(l));
|
||
end;
|
||
|
||
{$ifndef FPUNONE}
|
||
function random: extended;
|
||
const
|
||
c = 1.0/$10000/$10000;
|
||
begin
|
||
random:=rand_next*c;
|
||
end;
|
||
{$endif}
|
||
|
||
{$endif FPC_USE_SIMPLE_RANDOM}
|
||
{$endif FPC_HAS_FEATURE_RANDOM}
|
||
|
||
|
||
{****************************************************************************
|
||
Memory Management
|
||
****************************************************************************}
|
||
|
||
{$ifndef FPC_SYSTEM_HAS_PTR}
|
||
Function Ptr(sel,off : {$ifdef CPU16}Word{$else}Longint{$endif}) : farpointer;{$ifdef SYSTEMINLINE}inline;{$endif}
|
||
Begin
|
||
ptr:=farpointer((sel shl 4)+off);
|
||
End;
|
||
{$endif not FPC_SYSTEM_HAS_PTR}
|
||
|
||
{$ifndef FPC_SYSTEM_HAS_CSEG}
|
||
Function CSeg : Word;{$ifdef SYSTEMINLINE}inline;{$endif}
|
||
Begin
|
||
Cseg:=0;
|
||
End;
|
||
{$endif not FPC_SYSTEM_HAS_CSEG}
|
||
|
||
{$ifndef FPC_SYSTEM_HAS_DSEG}
|
||
Function DSeg : Word;{$ifdef SYSTEMINLINE}inline;{$endif}
|
||
Begin
|
||
Dseg:=0;
|
||
End;
|
||
{$endif not FPC_SYSTEM_HAS_DSEG}
|
||
|
||
{$ifndef FPC_SYSTEM_HAS_SSEG}
|
||
Function SSeg : Word;{$ifdef SYSTEMINLINE}inline;{$endif}
|
||
Begin
|
||
Sseg:=0;
|
||
End;
|
||
{$endif not FPC_SYSTEM_HAS_SSEG}
|
||
|
||
|
||
|
||
{$push}
|
||
{$R-}
|
||
{$I-}
|
||
{$Q-}
|
||
|
||
{*****************************************************************************
|
||
Miscellaneous
|
||
*****************************************************************************}
|
||
|
||
{$ifndef FPC_SYSTEM_HAS_STACKTOP}
|
||
function StackTop: pointer;
|
||
begin
|
||
{ Avoid wrap to zero on 32-bit }
|
||
if ptruint(StackBottom) > high(ptruint) - StackLength then
|
||
result:=pointer(ptruint(high(ptruint)))
|
||
else
|
||
result:=StackBottom + StackLength;
|
||
end;
|
||
{$endif FPC_SYSTEM_HAS_STACKTOP}
|
||
|
||
{$ifndef FPC_SYSTEM_HAS_GET_PC_ADDR}
|
||
{ This provides a dummy implementation
|
||
of get_pc_addr function, for CPU's that don't need
|
||
the instruction address to walk the stack. }
|
||
function get_pc_addr : codepointer;inline;
|
||
begin
|
||
get_pc_addr:=nil;
|
||
end;
|
||
{$endif ndef FPC_SYSTEM_HAS_GET_PC_ADDR}
|
||
|
||
{$ifndef FPC_SYSTEM_HAS_GET_CALLER_STACKINFO}
|
||
{ This provides a simple implementation
|
||
of get_caller_stackinfo procedure,
|
||
using get_caller_addr and get_caller_frame
|
||
functions. }
|
||
procedure get_caller_stackinfo(var framebp : pointer; var addr : codepointer);
|
||
var
|
||
nextbp : pointer;
|
||
nextaddr : codepointer;
|
||
begin
|
||
nextbp:=get_caller_frame(framebp,addr);
|
||
nextaddr:=get_caller_addr(framebp,addr);
|
||
framebp:=nextbp;
|
||
addr:=nextaddr;
|
||
end;
|
||
{$endif ndef FPC_SYSTEM_HAS_GET_CALLER_STACKINFO}
|
||
|
||
{$ifdef FPC_HAS_EXPLICIT_INTERLOCKED_POINTER}
|
||
{$ifndef FPC_SYSTEM_HAS_EXPLICIT_INTERLOCKED_POINTER}
|
||
|
||
function InterLockedDecrement (var Target: pointer) : pointer;
|
||
begin
|
||
Result := Pointer(InterLockedDecrement(PtrInt(Target)));
|
||
end;
|
||
|
||
function InterLockedIncrement (var Target: pointer) : pointer;
|
||
begin
|
||
Result := Pointer(InterLockedIncrement(PtrInt(Target)));
|
||
end;
|
||
|
||
function InterLockedExchange (var Target: pointer; Source : pointer) : pointer;
|
||
begin
|
||
Result := Pointer(InterLockedExchange(PtrInt(Target), PtrInt(Source)));
|
||
end;
|
||
|
||
function InterLockedExchangeAdd (var Target: pointer; Source : pointer) : pointer;
|
||
begin
|
||
Result := Pointer(InterLockedExchangeAdd(PtrInt(Target), PtrInt(Source)));
|
||
end;
|
||
|
||
function InterLockedCompareExchange (var Target: pointer; NewValue: pointer; Comperand: pointer): pointer;
|
||
begin
|
||
Result := Pointer(InterLockedCompareExchange(PtrInt(Target), PtrInt(NewValue), PtrInt(Comperand)));
|
||
end;
|
||
|
||
function InterLockedCompareExchangePointer (var Target: pointer; NewValue: pointer; Comperand: pointer): pointer;
|
||
begin
|
||
Result := Pointer(InterLockedCompareExchange(PtrInt(Target), PtrInt(NewValue), PtrInt(Comperand)));
|
||
end;
|
||
|
||
{$endif FPC_SYSTEM_HAS_EXPLICIT_INTERLOCKED_POINTER}
|
||
{$endif FPC_HAS_EXPLICIT_INTERLOCKED_POINTER}
|
||
|
||
procedure fpc_objecterror; compilerproc;
|
||
begin
|
||
HandleErrorAddrFrameInd(210,get_pc_addr,get_frame);
|
||
end;
|
||
|
||
procedure fpc_rangeerror;[public,alias:'FPC_RANGEERROR']; compilerproc;
|
||
begin
|
||
HandleErrorAddrFrameInd(201,get_pc_addr,get_frame);
|
||
end;
|
||
|
||
|
||
procedure fpc_divbyzero;[public,alias:'FPC_DIVBYZERO']; compilerproc;
|
||
begin
|
||
HandleErrorAddrFrameInd(200,get_pc_addr,get_frame);
|
||
end;
|
||
|
||
|
||
procedure fpc_overflow;[public,alias:'FPC_OVERFLOW']; compilerproc;
|
||
begin
|
||
HandleErrorAddrFrameInd(215,get_pc_addr,get_frame);
|
||
end;
|
||
|
||
|
||
procedure fpc_threaderror; [public,alias:'FPC_THREADERROR'];
|
||
begin
|
||
HandleErrorAddrFrameInd(236{RuntimeErrorExitCodes[reThreadError]},get_pc_addr,get_frame);
|
||
end;
|
||
|
||
|
||
procedure fpc_invalidpointer; [public,alias:'FPC_INVALIDPOINTER']; compilerproc;
|
||
begin
|
||
HandleErrorAddrFrameInd(216,get_pc_addr,get_frame);
|
||
end;
|
||
|
||
|
||
procedure fpc_iocheck;[public,alias:'FPC_IOCHECK']; compilerproc;
|
||
var
|
||
l : longint;
|
||
HInoutRes : PWord;
|
||
begin
|
||
HInOutRes:=@InoutRes;
|
||
if HInOutRes^<>0 then
|
||
begin
|
||
l:=HInOutRes^;
|
||
HInOutRes^:=0;
|
||
HandleErrorAddrFrameInd(l,get_pc_addr,get_frame);
|
||
end;
|
||
end;
|
||
|
||
|
||
Function IOResult:Word;
|
||
var
|
||
HInoutRes : PWord;
|
||
Begin
|
||
HInoutRes:=@InoutRes;
|
||
IOResult:=HInOutRes^;
|
||
HInOutRes^:=0;
|
||
End;
|
||
|
||
|
||
Function GetThreadID:TThreadID;{$ifdef SYSTEMINLINE}inline;{$endif}
|
||
begin
|
||
(* ThreadID is stored in a threadvar and made available in interface *)
|
||
(* to allow setup of this value during thread initialization. *)
|
||
GetThreadID := ThreadID;
|
||
end;
|
||
|
||
|
||
function fpc_safecallcheck(res : hresult) : hresult;[public,alias:'FPC_SAFECALLCHECK']; compilerproc; {$ifdef CPU86} register; {$endif}
|
||
begin
|
||
if res<0 then
|
||
begin
|
||
if assigned(SafeCallErrorProc) then
|
||
SafeCallErrorProc(res,get_frame);
|
||
HandleErrorAddrFrameInd(229,get_pc_addr,get_frame);
|
||
end;
|
||
result:=res;
|
||
end;
|
||
|
||
|
||
{*****************************************************************************
|
||
Stack check code
|
||
*****************************************************************************}
|
||
|
||
{ be compatible with old code }
|
||
{$ifdef FPC_NO_GENERIC_STACK_CHECK}
|
||
{$define NO_GENERIC_STACK_CHECK}
|
||
{$endif FPC_NO_GENERIC_STACK_CHECK}
|
||
|
||
{$IFNDEF NO_GENERIC_STACK_CHECK}
|
||
|
||
{$PUSH}
|
||
{$S-}
|
||
procedure fpc_stackcheck(stack_size:SizeUInt);[public,alias:'FPC_STACKCHECK']; compilerproc;
|
||
var
|
||
c : Pointer;
|
||
begin
|
||
{ Avoid recursive calls when called from the exit routines }
|
||
if StackError then
|
||
exit;
|
||
{ check stack alignment }
|
||
{$ifdef CPUI386}
|
||
{$ifdef FPC_STACKALIGNMENT}
|
||
{$if FPC_STACKALIGNMENT=16}
|
||
if ((PtrUInt(Sptr)+4) mod 16)<>0 then
|
||
begin
|
||
StackError:=true;
|
||
HandleError(202);
|
||
end;
|
||
{$endif FPC_STACKALIGNMENT=16}
|
||
{$endif FPC_STACKALIGNMENT}
|
||
{$endif CPUI386}
|
||
{ don't use stack_size, since the stack pointer has already been
|
||
decreased when this routine is called
|
||
}
|
||
c := Sptr - StackMargin;
|
||
if (c <= StackBottom) then
|
||
begin
|
||
StackError:=true;
|
||
HandleError(202);
|
||
end;
|
||
end;
|
||
{$POP}
|
||
|
||
{$ENDIF NO_GENERIC_STACK_CHECK}
|
||
|
||
{*****************************************************************************
|
||
Initialization / Finalization
|
||
*****************************************************************************}
|
||
|
||
const
|
||
maxunits=1024; { See also files.pas of the compiler source }
|
||
type
|
||
TInitFinalRec=record
|
||
InitProc,
|
||
FinalProc : TProcedure;
|
||
end;
|
||
TInitFinalTable = record
|
||
TableCount,
|
||
InitCount : ALUUInt;
|
||
Procs : array[1..maxunits] of TInitFinalRec;
|
||
end;
|
||
PInitFinalTable = ^TInitFinalTable;
|
||
|
||
|
||
{$ifndef FPC_HAS_INDIRECT_ENTRY_INFORMATION}
|
||
var
|
||
InitFinalTable : TInitFinalTable;external name 'INITFINAL';
|
||
{$endif FPC_HAS_INDIRECT_ENTRY_INFORMATION}
|
||
|
||
|
||
procedure fpc_InitializeUnits;[public,alias:'FPC_INITIALIZEUNITS']; compilerproc;
|
||
var
|
||
i : ALUUInt;
|
||
pt : PInitFinalTable;
|
||
begin
|
||
{ call cpu/fpu initialisation routine }
|
||
fpc_cpuinit;
|
||
{$ifdef FPC_HAS_INDIRECT_ENTRY_INFORMATION}
|
||
pt := PInitFinalTable(EntryInformation.InitFinalTable);
|
||
{$else FPC_HAS_INDIRECT_ENTRY_INFORMATION}
|
||
pt := @InitFinalTable;
|
||
{$endif FPC_HAS_INDIRECT_ENTRY_INFORMATION}
|
||
with pt^ do
|
||
begin
|
||
for i:=1 to ALUUInt(TableCount) do
|
||
begin
|
||
if assigned(Procs[i].InitProc) then
|
||
Procs[i].InitProc();
|
||
InitCount:=i;
|
||
end;
|
||
end;
|
||
if assigned(InitProc) then
|
||
TProcedure(InitProc)();
|
||
end;
|
||
|
||
|
||
procedure internal_initializeunits; external name 'FPC_INITIALIZEUNITS';
|
||
|
||
procedure fpc_LibInitializeUnits;[public,alias:'FPC_LIBINITIALIZEUNITS'];compilerproc;
|
||
begin
|
||
{$ifdef FPC_HAS_FEATURE_DYNLIBS}
|
||
IsLibrary:=true;
|
||
{ must also be set to true for packages when implemented }
|
||
ModuleIsLib:=true;
|
||
internal_initializeunits;
|
||
{$endif FPC_HAS_FEATURE_DYNLIBS}
|
||
end;
|
||
|
||
|
||
{$ifdef FPC_INIT_FINAL_UNITS_BY_CALLS}
|
||
procedure FinalizeUnits; external name 'FPC_FINALIZE_FUNC_TABLE';
|
||
|
||
{$else FPC_INIT_FINAL_UNITS_BY_CALLS}
|
||
procedure FinalizeUnits;[public,alias:'FPC_FINALIZEUNITS'];
|
||
begin
|
||
{$ifdef FPC_HAS_INDIRECT_ENTRY_INFORMATION}
|
||
with PInitFinalTable(EntryInformation.InitFinalTable)^ do
|
||
{$else FPC_HAS_INDIRECT_ENTRY_INFORMATION}
|
||
with InitFinalTable do
|
||
{$endif FPC_HAS_INDIRECT_ENTRY_INFORMATION}
|
||
begin
|
||
while (InitCount>0) do
|
||
begin
|
||
// we've to decrement the cound before calling the final. code
|
||
// else a halt in the final. code leads to a endless loop
|
||
dec(InitCount);
|
||
if assigned(Procs[InitCount+1].FinalProc) then
|
||
Procs[InitCount+1].FinalProc();
|
||
end;
|
||
end;
|
||
end;
|
||
{$endif FPC_INIT_FINAL_UNITS_BY_CALLS}
|
||
|
||
|
||
{*****************************************************************************
|
||
FPU Exceptions state
|
||
*****************************************************************************}
|
||
|
||
{$if defined(FPC_SYSTEM_FPUCW_IMMUTABLE) or defined(FPU_NONE)}
|
||
function GetNativeFPUControlWord: TNativeFPUControlWord; {$if defined(SYSTEMINLINE)}inline;{$endif}
|
||
begin
|
||
result:=default(TNativeFPUControlWord);
|
||
end;
|
||
|
||
procedure SetNativeFPUControlWord(const cw: TNativeFPUControlWord); {$if defined(SYSTEMINLINE)}inline;{$endif}
|
||
begin
|
||
end;
|
||
{$endif}
|
||
|
||
{*****************************************************************************
|
||
Error / Exit / ExitProc
|
||
*****************************************************************************}
|
||
|
||
Procedure system_exit;forward;{$ifdef FPC_SYSTEM_EXIT_NO_RETURN}noreturn;{$endif}
|
||
|
||
{$ifdef FPC_HAS_FEATURE_HEAP}
|
||
{$ifndef HAS_MEMORYMANAGER}
|
||
//not needed if independant memory manager
|
||
Procedure FinalizeHeap;forward;
|
||
{$endif HAS_MEMORYMANAGER}
|
||
{$endif FPC_HAS_FEATURE_HEAP}
|
||
|
||
{$ifdef FPC_HAS_FEATURE_CONSOLEIO}
|
||
procedure SysFlushStdIO;
|
||
begin
|
||
{ Make sure that all output is written to the redirected file }
|
||
if Textrec(Output).Mode=fmOutput then
|
||
Flush(Output);
|
||
if Textrec(ErrOutput).Mode=fmOutput then
|
||
Flush(ErrOutput);
|
||
{$ifndef FPC_STDOUT_TRUE_ALIAS}
|
||
if Textrec(stdout).Mode=fmOutput then
|
||
Flush(stdout);
|
||
if Textrec(StdErr).Mode=fmOutput then
|
||
Flush(StdErr);
|
||
{$endif FPC_STDOUT_TRUE_ALIAS}
|
||
end;
|
||
{$endif FPC_HAS_FEATURE_CONSOLEIO}
|
||
|
||
Procedure InternalExit;
|
||
var
|
||
current_exit : Procedure;
|
||
{$ifdef FPC_HAS_FEATURE_CONSOLEIO}
|
||
pstdout : ^Text;
|
||
{$endif}
|
||
{$if defined(MSWINDOWS) or defined(OS2)}
|
||
i : longint;
|
||
{$endif}
|
||
Begin
|
||
{$ifdef SYSTEMDEBUG}
|
||
writeln('InternalExit');
|
||
{$endif SYSTEMDEBUG}
|
||
{$ifndef CPUAVR}
|
||
while exitProc<>nil Do
|
||
Begin
|
||
InOutRes:=0;
|
||
current_exit:=tProcedure(exitProc);
|
||
exitProc:=nil;
|
||
current_exit();
|
||
End;
|
||
{$endif CPUAVR}
|
||
|
||
{$ifdef FPC_HAS_FEATURE_CONSOLEIO}
|
||
{ the embedded system unit itself contains no routines for console i/o
|
||
console i/o is done by the Consoleio unit which can do things like
|
||
redirection to seriell i/o }
|
||
{$ifndef EMBEDDED}
|
||
{ Show runtime error and exit }
|
||
if WriteErrorsToStdErr then
|
||
pstdout:=@stderr
|
||
else
|
||
pstdout:=@stdout;
|
||
If erroraddr<>nil Then
|
||
Begin
|
||
Writeln(pstdout^,'Runtime error ',Errorcode,' at $',hexstr(erroraddr));
|
||
{ to get a nice symify }
|
||
Writeln(pstdout^,BackTraceStrFunc(Erroraddr));
|
||
dump_stack(pstdout^,ErrorBase,ErrorAddr);
|
||
Writeln(pstdout^,'');
|
||
End;
|
||
SysFlushStdIO;
|
||
{$endif EMBEDDED}
|
||
{$endif FPC_HAS_FEATURE_CONSOLEIO}
|
||
|
||
{ Finalize units }
|
||
FinalizeUnits;
|
||
|
||
{$if defined(OS2)}
|
||
{ finally release the heap if possible, especially
|
||
important for DLLs.
|
||
Reset the array to nil, and finally also argv itself to
|
||
avoid double freeing problem in case this function gets called twice. }
|
||
if assigned(argv) then
|
||
begin
|
||
for i:=0 to argc-1 do
|
||
if assigned(argv[i]) then
|
||
begin
|
||
sysfreemem(argv[i]);
|
||
argv[i]:=nil;
|
||
end;
|
||
sysfreemem(argv);
|
||
argv:=nil;
|
||
end;
|
||
{$endif}
|
||
{$ifdef LINUX}
|
||
{sysfreemem already checks for nil}
|
||
{ Do not try to do anything if the heap manager already reported an error }
|
||
if (errorcode<>203) and (errorcode<>204) then
|
||
sysfreemem(calculated_cmdline);
|
||
{$endif}
|
||
{$ifdef BSD}
|
||
{ Do not try to do anything if the heap manager already reported an error }
|
||
if (errorcode<>203) and (errorcode<>204) then
|
||
sysfreemem(cmdline);
|
||
{$endif}
|
||
|
||
{$ifdef FPC_HAS_FEATURE_HEAP}
|
||
{$ifndef HAS_MEMORYMANAGER}
|
||
{$ifndef FPC_NO_DEFAULT_HEAP}
|
||
FinalizeHeap;
|
||
{$endif not FPC_NO_DEFAULT_HEAP}
|
||
{$endif not HAS_MEMORYMANAGER}
|
||
{$endif FPC_HAS_FEATURE_HEAP}
|
||
End;
|
||
|
||
|
||
Procedure fpc_do_exit;[Public,Alias:'FPC_DO_EXIT']; compilerproc;
|
||
begin
|
||
InternalExit;
|
||
System_exit;
|
||
end;
|
||
|
||
procedure internal_do_exit; external name 'FPC_DO_EXIT';
|
||
|
||
|
||
Procedure fpc_lib_exit;[Public,Alias:'FPC_LIB_EXIT'];
|
||
begin
|
||
InternalExit;
|
||
end;
|
||
|
||
|
||
Procedure Halt(ErrNum: TExitCode);noreturn;
|
||
Begin
|
||
{$ifdef FPC_HAS_FEATURE_EXITCODE}
|
||
{$ifdef FPC_LIMITED_EXITCODE}
|
||
if ErrNum > maxExitCode then
|
||
ExitCode:=255
|
||
else
|
||
{$endif FPC_LIMITED_EXITCODE}
|
||
ExitCode:=ErrNum;
|
||
{$endif FPC_HAS_FEATURE_EXITCODE}
|
||
internal_do_exit;
|
||
end;
|
||
|
||
|
||
{$ifndef FPC_SYSTEM_HAS_BACKTRACESTR}
|
||
function SysBackTraceStr (Addr: CodePointer): ShortString;
|
||
begin
|
||
SysBackTraceStr:=' $'+hexstr(addr);
|
||
end;
|
||
{$endif FPC_SYSTEM_HAS_BACKTRACESTR}
|
||
|
||
|
||
{$ifndef FPC_SYSTEM_HAS_CAPTUREBACKTRACE}
|
||
function CaptureBacktrace(skipframes,count:sizeint;frames:PCodePointer):sizeint;
|
||
var
|
||
curr_frame,prev_frame: pointer;
|
||
curr_addr: codepointer;
|
||
i: sizeint;
|
||
begin
|
||
curr_frame:=get_frame;
|
||
curr_addr:=get_pc_addr;
|
||
prev_frame:=curr_frame;
|
||
get_caller_stackinfo(curr_frame,curr_addr);
|
||
i:=-skipframes;
|
||
while (i<count) and (curr_frame>prev_frame) and
|
||
(curr_frame<StackTop) do
|
||
begin
|
||
if (curr_addr=nil) or
|
||
(curr_frame=nil) then
|
||
break;
|
||
if (i>=0) then
|
||
frames[i]:=curr_addr;
|
||
inc(i);
|
||
prev_frame:=curr_frame;
|
||
get_caller_stackinfo(curr_frame,curr_addr);
|
||
end;
|
||
if i<0 then
|
||
result:=0
|
||
else
|
||
result:=i;
|
||
end;
|
||
{$endif FPC_SYSTEM_HAS_CAPTUREBACKTRACE}
|
||
|
||
|
||
Procedure HandleErrorAddrFrame (Errno : TExitCode;addr : CodePointer; frame : Pointer);[public,alias:'FPC_BREAK_ERROR']; {$ifdef CPUI386} register; {$endif}
|
||
begin
|
||
If codepointer(ErrorProc)<>Nil then
|
||
ErrorProc(Errno,addr,frame);
|
||
errorcode:=word(Errno);
|
||
erroraddr:=addr;
|
||
errorbase:=frame;
|
||
Halt(errorcode);
|
||
end;
|
||
|
||
{ This is used internally by system skip first level,
|
||
and generated the same output as before, when
|
||
HandleErrorFrame function was used internally. }
|
||
Procedure HandleErrorAddrFrameInd (Errno : TExitCode;addr : CodePointer; frame : Pointer);
|
||
begin
|
||
get_caller_stackinfo (frame, addr);
|
||
HandleErrorAddrFrame (Errno,addr,frame);
|
||
end;
|
||
|
||
Procedure HandleErrorFrame (Errno : TExitCode;frame : Pointer);
|
||
{
|
||
Procedure to handle internal errors, i.e. not user-invoked errors
|
||
Internal function should ALWAYS call HandleError instead of RunError.
|
||
Can be used for exception handlers to specify the frame
|
||
}
|
||
begin
|
||
HandleErrorAddrFrame(Errno,get_caller_addr(frame),get_caller_frame(frame));
|
||
end;
|
||
|
||
|
||
procedure fpc_handleerror (Errno : TExitCode); compilerproc; [public,alias : 'FPC_HANDLEERROR'];
|
||
{
|
||
Procedure to handle internal errors, i.e. not user-invoked errors
|
||
Internal function should ALWAYS call HandleError instead of RunError.
|
||
}
|
||
begin
|
||
HandleErrorAddrFrameInd(Errno,get_pc_addr,get_frame);
|
||
end;
|
||
|
||
|
||
procedure RunError(w : word);[alias: 'FPC_RUNERROR'];noreturn;
|
||
var
|
||
bp : pointer;
|
||
pcaddr : codepointer;
|
||
begin
|
||
errorcode:=w;
|
||
pcaddr:=get_pc_addr;
|
||
bp:=get_frame;
|
||
get_caller_stackinfo(bp,pcaddr);
|
||
erroraddr:=pcaddr;
|
||
errorbase:=bp;
|
||
Halt(errorcode);
|
||
end;
|
||
|
||
|
||
Procedure RunError;{$ifdef SYSTEMINLINE}inline;{$endif}noreturn;
|
||
Begin
|
||
RunError (0);
|
||
End;
|
||
|
||
|
||
Procedure Halt;{$ifdef SYSTEMINLINE}inline;{$endif}noreturn;
|
||
Begin
|
||
Halt(0);
|
||
End;
|
||
|
||
|
||
Procedure Error(RunTimeError : TRunTimeError);
|
||
begin
|
||
RunError(RuntimeErrorExitCodes[RunTimeError]);
|
||
end;
|
||
|
||
|
||
Procedure dump_stack(var f : text;fp : Pointer; addr : CodePointer);
|
||
var
|
||
i : ObjpasInt;
|
||
prevfp : Pointer;
|
||
is_dev : boolean;
|
||
Begin
|
||
{$ifdef FPC_HAS_FEATURE_EXCEPTIONS}
|
||
try
|
||
{$endif FPC_HAS_FEATURE_EXCEPTIONS}
|
||
{ Frame of this procedure acts as StackBottom, fp values below that are invalid. }
|
||
prevfp:=get_frame;
|
||
i:=0;
|
||
is_dev:=do_isdevice(textrec(f).Handle);
|
||
{ sanity checks, new frame pointer must be always greater than the old one, further
|
||
it must point into the stack area, else something went wrong }
|
||
while (fp>prevfp) and (fp<StackTop) do
|
||
Begin
|
||
prevfp:=fp;
|
||
get_caller_stackinfo(fp,addr);
|
||
if (addr=nil) then
|
||
break;
|
||
Writeln(f,BackTraceStrFunc(addr));
|
||
if (fp=nil) then
|
||
break;
|
||
Inc(i);
|
||
If ((i>max_frame_dump) and is_dev) or (i>256) Then
|
||
break;
|
||
End;
|
||
{$ifdef FPC_HAS_FEATURE_EXCEPTIONS}
|
||
except
|
||
{ prevent endless dump if an exception occurred }
|
||
end;
|
||
{$endif FPC_HAS_FEATURE_EXCEPTIONS}
|
||
End;
|
||
|
||
|
||
procedure dump_stack(var f: text; skipframes: longint);
|
||
var
|
||
i,count: ObjpasInt;
|
||
frames: array [0..255] of codepointer;
|
||
begin
|
||
if do_isdevice(textrec(f).handle) then
|
||
count:=max_frame_dump
|
||
else
|
||
count:=255;
|
||
{$ifdef FPC_HAS_FEATURE_EXCEPTIONS}
|
||
try
|
||
{$endif FPC_HAS_FEATURE_EXCEPTIONS}
|
||
count:=CaptureBacktrace(skipframes+1,count,@frames[0]);
|
||
for i:=0 to count-1 do
|
||
writeln(f,BackTraceStrFunc(frames[i]));
|
||
{$ifdef FPC_HAS_FEATURE_EXCEPTIONS}
|
||
except
|
||
end;
|
||
{$endif FPC_HAS_FEATURE_EXCEPTIONS}
|
||
end;
|
||
|
||
|
||
{$ifdef FPC_HAS_FEATURE_EXCEPTIONS}
|
||
procedure DumpExceptionBackTrace(var f:text);
|
||
var
|
||
FrameNumber,
|
||
FrameCount : ObjpasInt;
|
||
Frames : PCodePointer;
|
||
begin
|
||
if RaiseList=nil then
|
||
exit;
|
||
WriteLn(f,BackTraceStrFunc(RaiseList^.Addr));
|
||
FrameCount:=RaiseList^.Framecount;
|
||
Frames:=RaiseList^.Frames;
|
||
for FrameNumber := 0 to FrameCount-1 do
|
||
WriteLn(f,BackTraceStrFunc(Frames[FrameNumber]));
|
||
end;
|
||
{$endif FPC_HAS_FEATURE_EXCEPTIONS}
|
||
|
||
|
||
{$ifdef FPC_HAS_FEATURE_HEAP}
|
||
Type
|
||
PExitProcInfo = ^TExitProcInfo;
|
||
TExitProcInfo = Record
|
||
Next : PExitProcInfo;
|
||
SaveExit : CodePointer;
|
||
Proc : TProcedure;
|
||
End;
|
||
const
|
||
ExitProcList: PExitProcInfo = nil;
|
||
|
||
Procedure DoExitProc;
|
||
var
|
||
P : PExitProcInfo;
|
||
Proc : TProcedure;
|
||
Begin
|
||
P:=ExitProcList;
|
||
ExitProcList:=P^.Next;
|
||
ExitProc:=P^.SaveExit;
|
||
Proc:=P^.Proc;
|
||
DisPose(P);
|
||
Proc();
|
||
End;
|
||
|
||
|
||
Procedure AddExitProc(Proc: TProcedure);
|
||
var
|
||
P : PExitProcInfo;
|
||
Begin
|
||
New(P);
|
||
P^.Next:=ExitProcList;
|
||
P^.SaveExit:=ExitProc;
|
||
P^.Proc:=Proc;
|
||
ExitProcList:=P;
|
||
ExitProc:=@DoExitProc;
|
||
End;
|
||
{$endif FPC_HAS_FEATURE_HEAP}
|
||
|
||
|
||
{$ifdef FPC_HAS_FEATURE_HEAP}
|
||
{$ifdef FPC_HAS_FEATURE_ANSISTRINGS}
|
||
function ArrayStringToPPchar(const S:Array of AnsiString;reserveentries:Longint):ppansichar; // const ?
|
||
{$ifdef EXCLUDE_COMPLEX_PROCS}
|
||
begin
|
||
runerror(217);
|
||
end;
|
||
{$else EXCLUDE_COMPLEX_PROCS}
|
||
// Extra allocate reserveentries pansichar's at the beginning (default param=0 after 1.0.x ?)
|
||
// Note: for internal use by skilled programmers only
|
||
// if "s" goes out of scope in the parent procedure, the pointer is dangling.
|
||
|
||
var p : ppansichar;
|
||
i : ObjpasInt;
|
||
begin
|
||
if High(s)<Low(s) Then Exit(NIL);
|
||
Getmem(p,sizeof(pansichar)*(high(s)-low(s)+ReserveEntries+2)); // one more for NIL, one more
|
||
// for cmd
|
||
if p=nil then
|
||
begin
|
||
{$ifdef xunix}
|
||
fpseterrno(ESysEnomem);
|
||
{$endif}
|
||
exit(NIL);
|
||
end;
|
||
for i:=low(s) to high(s) do
|
||
p[i+Reserveentries]:=pansichar(s[i]);
|
||
p[high(s)+1+Reserveentries]:=nil;
|
||
ArrayStringToPPchar:=p;
|
||
end;
|
||
{$endif EXCLUDE_COMPLEX_PROCS}
|
||
|
||
|
||
Function StringToPPChar(Var S:AnsiString;ReserveEntries:integer):ppansichar;
|
||
{
|
||
Create a PPAnsiChar to structure of pchars which are the arguments specified
|
||
in the string S. Especially useful for creating an ArgV for Exec-calls
|
||
}
|
||
|
||
begin
|
||
StringToPPChar:=StringToPPChar(PAnsiChar(S),ReserveEntries);
|
||
end;
|
||
{$endif FPC_HAS_FEATURE_ANSISTRINGS}
|
||
|
||
|
||
Function StringToPPChar(S: PAnsiChar;ReserveEntries:integer):ppansichar;
|
||
|
||
var
|
||
i,nr : ObjpasInt;
|
||
Buf : ^ansichar;
|
||
p : ppansichar;
|
||
|
||
begin
|
||
buf:=s;
|
||
nr:=1;
|
||
while (buf^<>#0) do // count nr of args
|
||
begin
|
||
while (buf^ in [' ',#9,#10]) do // Kill separators.
|
||
inc(buf);
|
||
inc(nr);
|
||
if buf^='"' Then // quotes argument?
|
||
begin
|
||
inc(buf);
|
||
while not (buf^ in [#0,'"']) do // then end of argument is end of string or next quote
|
||
inc(buf);
|
||
if buf^='"' then // skip closing quote.
|
||
inc(buf);
|
||
end
|
||
else
|
||
begin // else std
|
||
while not (buf^ in [' ',#0,#9,#10]) do
|
||
inc(buf);
|
||
end;
|
||
end;
|
||
getmem(p,(ReserveEntries+nr)*sizeof(pansichar));
|
||
StringToPPChar:=p;
|
||
if p=nil then
|
||
begin
|
||
{$ifdef xunix}
|
||
fpseterrno(ESysEnomem);
|
||
{$endif}
|
||
exit;
|
||
end;
|
||
for i:=1 to ReserveEntries do inc(p); // skip empty slots
|
||
buf:=s;
|
||
while (buf^<>#0) do
|
||
begin
|
||
while (buf^ in [' ',#9,#10]) do // Kill separators.
|
||
begin
|
||
buf^:=#0;
|
||
inc(buf);
|
||
end;
|
||
if buf^='"' Then // quotes argument?
|
||
begin
|
||
inc(buf);
|
||
p^:=buf;
|
||
inc(p);
|
||
p^:=nil;
|
||
while not (buf^ in [#0,'"']) do // then end of argument is end of string or next quote
|
||
inc(buf);
|
||
if buf^='"' then // skip closing quote.
|
||
begin
|
||
buf^:=#0;
|
||
inc(buf);
|
||
end;
|
||
end
|
||
else
|
||
begin
|
||
p^:=buf;
|
||
inc(p);
|
||
p^:=nil;
|
||
while not (buf^ in [' ',#0,#9,#10]) do
|
||
inc(buf);
|
||
end;
|
||
end;
|
||
end;
|
||
{$endif FPC_HAS_FEATURE_HEAP}
|
||
|
||
|
||
{*****************************************************************************
|
||
Abstract/Assert support.
|
||
*****************************************************************************}
|
||
|
||
procedure fpc_emptymethod;[public,alias : 'FPC_EMPTYMETHOD'];
|
||
begin
|
||
end;
|
||
|
||
|
||
procedure fpc_AbstractErrorIntern;compilerproc;[public,alias : 'FPC_ABSTRACTERROR'];
|
||
begin
|
||
If codepointer(AbstractErrorProc)<>nil then
|
||
AbstractErrorProc();
|
||
HandleErrorAddrFrameInd(211,get_pc_addr,get_frame);
|
||
end;
|
||
|
||
|
||
Procedure fpc_assert(Const Msg,FName:Shortstring;LineNo:Longint;
|
||
ErrorAddr:Pointer); [Public,Alias : 'FPC_ASSERT']; compilerproc;
|
||
begin
|
||
if codepointer(AssertErrorProc)<>nil then
|
||
AssertErrorProc(Msg,FName,LineNo,ErrorAddr)
|
||
else
|
||
HandleErrorAddrFrameInd(227,get_pc_addr,get_frame);
|
||
end;
|
||
|
||
|
||
Procedure SysAssert(Const Msg,FName:Shortstring;LineNo:Longint;ErrorAddr:Pointer);
|
||
begin
|
||
{$ifdef FPC_HAS_FEATURE_CONSOLEIO}
|
||
If Length(msg)=0 then
|
||
write(stderr,'Assertion failed')
|
||
else
|
||
write(stderr,msg);
|
||
Writeln(stderr,' (',FName,', line ',LineNo,').');
|
||
Writeln(stderr,'');
|
||
Halt(227);
|
||
{$endif FPC_HAS_FEATURE_CONSOLEIO}
|
||
end;
|
||
|
||
|
||
{*****************************************************************************
|
||
SetJmp/LongJmp support.
|
||
*****************************************************************************}
|
||
|
||
{$i setjump.inc}
|
||
|
||
|
||
{$pop} //{$I-,R-,Q-} before 'procedure fpc_rangeerror'
|
||
|
||
|
||
{*****************************************************************************
|
||
Heap
|
||
*****************************************************************************}
|
||
|
||
{$i sysheap.inc}
|
||
|
||
{$i heap.inc}
|
||
|
||
{*****************************************************************************
|
||
Thread support
|
||
*****************************************************************************}
|
||
|
||
{$ifdef FPC_HAS_FEATURE_THREADING}
|
||
{ Generic threadmanager }
|
||
{$i thread.inc}
|
||
|
||
{ Generic threadvar support }
|
||
{$i threadvr.inc}
|
||
|
||
{$ifdef DISABLE_NO_THREAD_MANAGER}
|
||
{ OS Dependent implementation }
|
||
{$i systhrd.inc}
|
||
{$endif DISABLE_NO_THREAD_MANAGER}
|
||
{$endif FPC_HAS_FEATURE_THREADING}
|
||
|
||
|
||
{*****************************************************************************
|
||
Dynamic library support
|
||
*****************************************************************************}
|
||
|
||
|
||
{$ifdef FPC_HAS_FEATURE_DYNLIBS}
|
||
{$i dynlib.inc}
|
||
|
||
{$ifdef DISABLE_NO_DYNLIBS_MANAGER}
|
||
{ OS Dependant implementation }
|
||
{$i sysdl.inc}
|
||
{$endif DISABLE_NO_DYNLIBS_MANAGER}
|
||
{$endif FPC_HAS_FEATURE_DYNLIBS}
|
||
|
||
|
||
{*****************************************************************************
|
||
File Handling
|
||
*****************************************************************************}
|
||
|
||
{$ifdef FPC_HAS_FEATURE_FILEIO}
|
||
{ Allow slash and backslash as separators }
|
||
procedure DoDirSeparators(var p: pansichar; inplace: boolean = true);
|
||
var
|
||
i : ObjpasInt;
|
||
len : sizeint;
|
||
newp : pansichar;
|
||
begin
|
||
len:=length(p);
|
||
newp:=nil;
|
||
for i:=0 to len do
|
||
if p[i] in AllowDirectorySeparators then
|
||
begin
|
||
if not inplace and
|
||
not assigned(newp) then
|
||
begin
|
||
getmem(newp,len+1);
|
||
move(p^,newp^,len+1);
|
||
p:=newp;
|
||
end;
|
||
p[i]:=DirectorySeparator;
|
||
end;
|
||
end;
|
||
|
||
procedure DoDirSeparators(var p: pwidechar; inplace: boolean = true);
|
||
var
|
||
i : ObjpasInt;
|
||
len : sizeint;
|
||
newp : pwidechar;
|
||
begin
|
||
len:=length(p);
|
||
newp:=nil;
|
||
for i:=0 to len do
|
||
if (ord(p[i])<255) and
|
||
(ansichar(ord(p[i])) in AllowDirectorySeparators) then
|
||
begin
|
||
if not inplace and
|
||
not assigned(newp) then
|
||
begin
|
||
getmem(newp,(len+1)*2);
|
||
move(p^,newp^,(len+1)*2);
|
||
p:=newp;
|
||
end;
|
||
p[i]:=DirectorySeparator;
|
||
end;
|
||
end;
|
||
|
||
procedure DoDirSeparators(var p:shortstring);
|
||
var
|
||
i : ObjpasInt;
|
||
begin
|
||
for i:=1 to length(p) do
|
||
if p[i] in AllowDirectorySeparators then
|
||
p[i]:=DirectorySeparator;
|
||
end;
|
||
|
||
|
||
{$ifdef FPC_HAS_FEATURE_ANSISTRINGS}
|
||
procedure DoDirSeparators(var ps:RawByteString);
|
||
var
|
||
i : ObjpasInt;
|
||
p : pansichar;
|
||
unique : boolean;
|
||
begin
|
||
unique:=false;
|
||
for i:=1 to length(ps) do
|
||
if ps[i] in AllowDirectorySeparators then
|
||
begin
|
||
if not unique then
|
||
begin
|
||
uniquestring(ps);
|
||
p:=pansichar(ps);
|
||
unique:=true;
|
||
end;
|
||
p[i-1]:=DirectorySeparator;
|
||
end;
|
||
end;
|
||
{$endif FPC_HAS_FEATURE_ANSISTRINGS}
|
||
|
||
|
||
{$ifdef FPC_HAS_FEATURE_UNICODESTRINGS}
|
||
procedure DoDirSeparators(var ps:UnicodeString);
|
||
var
|
||
i : ObjpasInt;
|
||
p : pwidechar;
|
||
unique : boolean;
|
||
begin
|
||
unique:=false;
|
||
for i:=1 to length(ps) do
|
||
if ps[i] in AllowDirectorySeparators then
|
||
begin
|
||
if not unique then
|
||
begin
|
||
uniquestring(ps);
|
||
p:=pwidechar(ps);
|
||
unique:=true;
|
||
end;
|
||
p[i-1]:=DirectorySeparator;
|
||
end;
|
||
end;
|
||
{$endif FPC_HAS_FEATURE_UNICODESTRINGS}
|
||
|
||
{$endif FPC_HAS_FEATURE_FILEIO}
|
||
|
||
{ OS dependent low level file functions }
|
||
{$ifdef FPC_HAS_FEATURE_FILEIO}
|
||
{$i sysfile.inc}
|
||
|
||
{$ifndef FPCRTL_FILESYSTEM_SINGLE_BYTE_API}
|
||
{$ifdef FPC_ANSI_TEXTFILEREC}
|
||
procedure do_open(var f; p: pansichar; flags: longint; pchangeable: boolean);
|
||
var
|
||
u: UnicodeString;
|
||
begin
|
||
widestringmanager.Ansi2UnicodeMoveProc(p,DefaultFileSystemCodePage,u,length(p));
|
||
do_open(f,pwidechar(u),flags,true);
|
||
end;
|
||
|
||
procedure do_erase(p: pansichar; pchangeable: boolean);
|
||
var
|
||
u: UnicodeString;
|
||
begin
|
||
widestringmanager.Ansi2UnicodeMoveProc(p,DefaultFileSystemCodePage,u,length(p));
|
||
do_erase(pwidechar(u),true);
|
||
end;
|
||
|
||
procedure do_rename(src, dst: pansichar; srcchangeable, dstchangeable: boolean);
|
||
var
|
||
usrc, udst: UnicodeString;
|
||
begin
|
||
widestringmanager.Ansi2UnicodeMoveProc(src,DefaultFileSystemCodePage,usrc,length(src));
|
||
widestringmanager.Ansi2UnicodeMoveProc(dst,DefaultFileSystemCodePage,udst,length(dst));
|
||
do_rename(pwidechar(usrc),pwidechar(udst),true,true);
|
||
end;
|
||
|
||
procedure do_rename(src: pansichar; dst: pwidechar; srcchangeable, dstchangeable: boolean);
|
||
var
|
||
usrc: UnicodeString;
|
||
begin
|
||
widestringmanager.Ansi2UnicodeMoveProc(src,DefaultFileSystemCodePage,usrc,length(src));
|
||
do_rename(pwidechar(usrc),dst,true,dstchangeable);
|
||
end;
|
||
{$endif FPC_ANSI_TEXTFILEREC}
|
||
{$endif not FPCRTL_FILESYSTEM_SINGLE_BYTE_API}
|
||
|
||
|
||
{$ifndef FPCRTL_FILESYSTEM_TWO_BYTE_API}
|
||
{$ifndef FPC_ANSI_TEXTFILEREC}
|
||
procedure do_open(var f; p: pwidechar; flags: longint; pchangeable: boolean);
|
||
var
|
||
s: RawByteString;
|
||
begin
|
||
widestringmanager.Unicode2AnsiMoveProc(p,s,DefaultFileSystemCodePage,length(p));
|
||
do_open(f,pansichar(s),flags,true);
|
||
end;
|
||
|
||
procedure do_erase(p: pwidechar; pchangeable: boolean);
|
||
var
|
||
s: RawByteString;
|
||
begin
|
||
widestringmanager.Unicode2AnsiMoveProc(p,s,DefaultFileSystemCodePage,length(p));
|
||
do_erase(pansichar(s),true);
|
||
end;
|
||
|
||
procedure do_rename(src, dst: pwidechar; srcchangeable, dstchangeable: boolean);
|
||
var
|
||
rsrc, rdst: RawByteString;
|
||
begin
|
||
widestringmanager.Unicode2AnsiMoveProc(src,rsrc,DefaultFileSystemCodePage,length(src));
|
||
widestringmanager.Unicode2AnsiMoveProc(dst,rdst,DefaultFileSystemCodePage,length(dst));
|
||
do_rename(pansichar(rsrc),pansichar(rdst),true,true);
|
||
end;
|
||
|
||
procedure do_rename(src: pwidechar; dst: pansichar; srcchangeable, dstchangeable: boolean);
|
||
var
|
||
rsrc: RawByteString;
|
||
begin
|
||
widestringmanager.Unicode2AnsiMoveProc(src,rsrc,DefaultFileSystemCodePage,length(src));
|
||
do_rename(pansichar(rsrc),dst,true,dstchangeable);
|
||
end;
|
||
{$endif not FPC_ANSI_TEXTFILEREC}
|
||
{$endif not FPCRTL_FILESYSTEM_TWO_BYTE_API}
|
||
|
||
{$endif FPC_HAS_FEATURE_FILEIO}
|
||
|
||
{ helper for targets supporting no ansistrings, it is used
|
||
by non-ansistring code }
|
||
function min(v1,v2 : SizeInt) : SizeInt;
|
||
begin
|
||
if v1<v2 then
|
||
result:=v1
|
||
else
|
||
result:=v2;
|
||
end;
|
||
|
||
{$ifdef FPC_HAS_FEATURE_TEXTIO}
|
||
{ Text file }
|
||
{$i text.inc}
|
||
{$endif FPC_HAS_FEATURE_TEXTIO}
|
||
|
||
{$ifdef FPC_HAS_FEATURE_FILEIO}
|
||
{ Untyped file }
|
||
{$i file.inc}
|
||
|
||
{ Typed file }
|
||
{$i typefile.inc}
|
||
{$endif FPC_HAS_FEATURE_FILEIO}
|
||
|
||
|
||
{*****************************************************************************
|
||
Directory Handling
|
||
*****************************************************************************}
|
||
|
||
{$ifdef FPC_HAS_FEATURE_FILEIO}
|
||
{ OS dependent dir functions }
|
||
{$i sysdir.inc}
|
||
|
||
|
||
{$ifdef FPC_HAS_FEATURE_ANSISTRINGS}
|
||
|
||
{$ifndef FPCRTL_FILESYSTEM_SINGLE_BYTE_API}
|
||
procedure do_getdir(drivenr : byte;var dir : rawbytestring);
|
||
var
|
||
u: unicodestring;
|
||
begin
|
||
Do_getdir(drivenr,u);
|
||
widestringmanager.Unicode2AnsiMoveProc(pwidechar(u),dir,DefaultRTLFileSystemCodePage,length(u));
|
||
end;
|
||
{$endif FPCRTL_FILESYSTEM_SINGLE_BYTE_API}
|
||
|
||
Procedure MkDir(Const s: RawByteString);[IOCheck];
|
||
Begin
|
||
If (Length(s)=0) or (InOutRes <> 0) then
|
||
exit;
|
||
{$ifdef FPCRTL_FILESYSTEM_SINGLE_BYTE_API}
|
||
Do_mkdir(ToSingleByteFileSystemEncodedFileName(S));
|
||
{$else FPCRTL_FILESYSTEM_SINGLE_BYTE_API}
|
||
Do_mkdir(S);
|
||
{$endif}
|
||
end;
|
||
|
||
|
||
Procedure RmDir(Const s: RawByteString);[IOCheck];
|
||
Begin
|
||
If (Length(s)=0) or (InOutRes <> 0) then
|
||
exit;
|
||
{$ifdef FPCRTL_FILESYSTEM_SINGLE_BYTE_API}
|
||
Do_rmdir(ToSingleByteFileSystemEncodedFileName(S));
|
||
{$else FPCRTL_FILESYSTEM_SINGLE_BYTE_API}
|
||
Do_rmdir(S);
|
||
{$endif}
|
||
End;
|
||
|
||
|
||
Procedure ChDir(Const s: RawByteString);[IOCheck];
|
||
Begin
|
||
If (Length(s)=0) or (InOutRes <> 0) then
|
||
exit;
|
||
{$ifdef FPCRTL_FILESYSTEM_SINGLE_BYTE_API}
|
||
Do_chdir(ToSingleByteFileSystemEncodedFileName(S));
|
||
{$else FPCRTL_FILESYSTEM_SINGLE_BYTE_API}
|
||
Do_chdir(S);
|
||
{$endif}
|
||
End;
|
||
|
||
|
||
Procedure getdir(drivenr:byte;Var dir:rawbytestring);
|
||
begin
|
||
Do_getdir(drivenr,dir);
|
||
{ we should return results in the DefaultRTLFileSystemCodePage -> convert if
|
||
necessary }
|
||
setcodepage(dir,DefaultRTLFileSystemCodePage,true);
|
||
end;
|
||
|
||
{ the generic shortstring ones are only implemented elsewhere for systems *not*
|
||
supporting ansi/unicodestrings; for now assume there are no systems that
|
||
support unicodestrings but not ansistrings }
|
||
|
||
{ avoid double string conversions }
|
||
{$ifdef FPCRTL_FILESYSTEM_SINGLE_BYTE_API}
|
||
function GetDirStrFromShortstring(const s: shortstring): RawByteString;
|
||
begin
|
||
GetDirStrFromShortstring:=ToSingleByteFileSystemEncodedFileName(ansistring(s));
|
||
end;
|
||
{$else FPCRTL_FILESYSTEM_SINGLE_BYTE_API}
|
||
function GetDirStrFromShortstring(const s: shortstring): UnicodeString;
|
||
begin
|
||
GetDirStrFromShortstring:=s;
|
||
end;
|
||
{$endif FPCRTL_FILESYSTEM_SINGLE_BYTE_API}
|
||
|
||
Procedure MkDir(Const s: shortstring);[IOCheck];
|
||
Begin
|
||
If (Length(s)=0) or (InOutRes <> 0) then
|
||
exit;
|
||
Do_mkdir(GetDirStrFromShortstring(S));
|
||
End;
|
||
|
||
|
||
Procedure RmDir(Const s: shortstring);[IOCheck];
|
||
Begin
|
||
If (Length(s)=0) or (InOutRes <> 0) then
|
||
exit;
|
||
Do_rmdir(GetDirStrFromShortstring(S));
|
||
End;
|
||
|
||
|
||
Procedure ChDir(Const s: shortstring);[IOCheck];
|
||
Begin
|
||
If (Length(S)=0) or (InOutRes <> 0) then
|
||
exit;
|
||
Do_chdir(GetDirStrFromShortstring(S));
|
||
End;
|
||
|
||
|
||
Procedure getdir(drivenr:byte;Var dir:shortstring);
|
||
var
|
||
s: rawbytestring;
|
||
begin
|
||
Do_getdir(drivenr,s);
|
||
if length(s)<=high(dir) then
|
||
dir:=s
|
||
else
|
||
inoutres:=3;
|
||
end;
|
||
{$endif FPC_HAS_FEATURE_ANSISTRINGS}
|
||
|
||
|
||
{$if defined(FPC_HAS_FEATURE_WIDESTRINGS)}
|
||
|
||
{$ifndef FPCRTL_FILESYSTEM_TWO_BYTE_API}
|
||
{ overloads required for mkdir/rmdir/chdir to ensure that the string is
|
||
converted to the right code page }
|
||
procedure do_mkdir(const s: unicodestring); {$ifdef SYSTEMINLINE}inline;{$endif}
|
||
begin
|
||
do_mkdir(ToSingleByteFileSystemEncodedFileName(s));
|
||
end;
|
||
|
||
|
||
procedure do_rmdir(const s: unicodestring); {$ifdef SYSTEMINLINE}inline;{$endif}
|
||
begin
|
||
do_rmdir(ToSingleByteFileSystemEncodedFileName(s));
|
||
end;
|
||
|
||
|
||
procedure do_chdir(const s: unicodestring); {$ifdef SYSTEMINLINE}inline;{$endif}
|
||
begin
|
||
do_chdir(ToSingleByteFileSystemEncodedFileName(s));
|
||
end;
|
||
|
||
|
||
procedure do_getdir(drivenr : byte;var dir : unicodestring);
|
||
var
|
||
s: rawbytestring;
|
||
begin
|
||
Do_getdir(drivenr,s);
|
||
dir:=unicodestring(s);
|
||
end;
|
||
{$endif FPCRTL_FILESYSTEM_TWO_BYTE_API}
|
||
|
||
Procedure MkDir(Const s: UnicodeString);[IOCheck];
|
||
Begin
|
||
if (Length(s)=0) or (InOutRes <> 0) then
|
||
exit;
|
||
Do_mkdir(S);
|
||
End;
|
||
|
||
|
||
Procedure RmDir(Const s: UnicodeString);[IOCheck];
|
||
Begin
|
||
if (Length(s)=0) or (InOutRes <> 0) then
|
||
exit;
|
||
Do_rmdir(S);
|
||
End;
|
||
|
||
|
||
Procedure ChDir(Const s: UnicodeString);[IOCheck];
|
||
Begin
|
||
if (Length(s)=0) or (InOutRes <> 0) then
|
||
exit;
|
||
Do_chdir(S);
|
||
End;
|
||
|
||
|
||
Procedure getdir(drivenr:byte;Var dir:unicodestring);
|
||
begin
|
||
Do_getdir(drivenr,dir);
|
||
end;
|
||
{$endif FPC_HAS_FEATURE_WIDESTRINGS}
|
||
|
||
{$endif FPC_HAS_FEATURE_FILEIO}
|
||
|
||
|
||
{*****************************************************************************
|
||
Resources support
|
||
*****************************************************************************}
|
||
|
||
{$i sysres.inc}
|
||
|
||
const
|
||
CtrlBreakHandler: TCtrlBreakHandler = nil;
|
||
|
||
{$IFNDEF FPC_HAS_SETCTRLBREAKHANDLER}
|
||
(* It is possible to provide platform specific implementation performing *)
|
||
(* special initialization; default implementation just sets the procedural *)
|
||
(* variable to make it available for use from the exception handler. *)
|
||
function SysSetCtrlBreakHandler (Handler: TCtrlBreakHandler): TCtrlBreakHandler;
|
||
begin
|
||
(* Return either nil or previous handler *)
|
||
SysSetCtrlBreakHandler := CtrlBreakHandler;
|
||
CtrlBreakHandler := Handler;
|
||
end;
|
||
{$ENDIF FPC_HAS_SETCTRLBREAKHANDLER}
|
||
|
||
{$ifdef cpu16}
|
||
function AtomicIncrement (var Target: smallint) : smallint; {$ifdef SYSTEMINLINE}inline;{$endif}
|
||
|
||
begin
|
||
Result:=InterlockedIncrement(Target);
|
||
end;
|
||
|
||
|
||
function AtomicDecrement (var Target: smallint) : smallint; {$ifdef SYSTEMINLINE}inline;{$endif}
|
||
|
||
begin
|
||
Result:=InterlockedDecrement(Target);
|
||
end;
|
||
|
||
|
||
function AtomicCmpExchange(var Target: smallint; NewValue, Comperand: smallint): smallint; {$ifdef SYSTEMINLINE}inline;{$endif}
|
||
|
||
begin
|
||
Result:=InterlockedCompareExchange(Target,NewValue,Comperand);
|
||
end;
|
||
|
||
|
||
function AtomicExchange (var Target: smallint;Source : smallint) : smallint; {$ifdef SYSTEMINLINE}inline;{$endif}
|
||
|
||
begin
|
||
Result:=InterlockedExchange(Target,Source);
|
||
end;
|
||
|
||
function AtomicIncrement (var Target: word) : word; {$ifdef SYSTEMINLINE}inline;{$endif}
|
||
|
||
begin
|
||
Result:=InterlockedIncrement(Target);
|
||
end;
|
||
|
||
|
||
function AtomicDecrement (var Target: word) : word; {$ifdef SYSTEMINLINE}inline;{$endif}
|
||
|
||
begin
|
||
Result:=InterlockedDecrement(Target);
|
||
end;
|
||
|
||
|
||
function AtomicCmpExchange(var Target: word; NewValue, Comperand: word): word; {$ifdef SYSTEMINLINE}inline;{$endif}
|
||
|
||
begin
|
||
Result:=InterlockedCompareExchange(Target, NewValue, Comperand);
|
||
end;
|
||
|
||
|
||
function AtomicExchange (var Target: word;Source : word) : word; {$ifdef SYSTEMINLINE}inline;{$endif}
|
||
|
||
begin
|
||
Result:=InterlockedExchange(Target,Source);
|
||
end;
|
||
{$endif cpu16}
|
||
|
||
function AtomicIncrement (var Target: longint) : longint; {$ifdef SYSTEMINLINE}inline;{$endif}
|
||
|
||
begin
|
||
Result:=InterlockedIncrement(Target);
|
||
end;
|
||
|
||
|
||
function AtomicDecrement (var Target: longint) : longint; {$ifdef SYSTEMINLINE}inline;{$endif}
|
||
|
||
begin
|
||
Result:=InterlockedDecrement(Target);
|
||
end;
|
||
|
||
|
||
function AtomicCmpExchange(var Target: longint; NewValue, Comperand: longint): longint; {$ifdef SYSTEMINLINE}inline;{$endif}
|
||
|
||
begin
|
||
Result:=InterlockedCompareExchange(Target,NewValue, Comperand);
|
||
end;
|
||
|
||
|
||
function AtomicExchange (var Target: longint;Source : longint) : longint; {$ifdef SYSTEMINLINE}inline;{$endif}
|
||
|
||
begin
|
||
Result:=InterlockedExchange(Target,Source);
|
||
end;
|
||
|
||
|
||
{$ifdef cpu64}
|
||
function AtomicIncrement (var Target: int64) : int64; {$ifdef SYSTEMINLINE}inline;{$endif}
|
||
|
||
begin
|
||
Result:=InterlockedIncrement64(Target);
|
||
end;
|
||
|
||
|
||
function AtomicDecrement (var Target: int64) : int64; {$ifdef SYSTEMINLINE}inline;{$endif}
|
||
|
||
begin
|
||
Result:=InterlockedDecrement64(Target);
|
||
end;
|
||
|
||
|
||
function AtomicCmpExchange(var Target: int64; NewValue, Comperand: int64): int64; {$ifdef SYSTEMINLINE}inline;{$endif}
|
||
|
||
begin
|
||
Result:=InterlockedCompareExchange64(Target,NewValue, Comperand);
|
||
end;
|
||
|
||
|
||
function AtomicExchange (var Target: int64;Source : int64) : int64; {$ifdef SYSTEMINLINE}inline;{$endif}
|
||
|
||
begin
|
||
|
||
Result:=InterlockedExchange64(Target,Source);
|
||
end;
|
||
{$endif cpu64}
|
||
|
||
|
||
{ Pointer overloads }
|
||
|
||
{$ifndef FPC_SYSTEM_DISABLE_INTERLOCK_POINTER_OVERLOAD}
|
||
|
||
function AtomicIncrement (var Target: pointer) : pointer; {$ifdef SYSTEMINLINE}inline;{$endif}
|
||
|
||
begin
|
||
{$IFDEF CPU64}
|
||
Result:=Pointer(InterlockedIncrement64(int64(Target)));
|
||
{$ELSE}
|
||
{$IFDEF CPU16}
|
||
Result:=Pointer(InterlockedIncrement(smallint(Target)));
|
||
{$ELSE}
|
||
Result:=Pointer(InterlockedIncrement(Longint(Target)));
|
||
{$ENDIF}
|
||
{$ENDIF}
|
||
end;
|
||
|
||
|
||
function AtomicDecrement (var Target: pointer) : pointer; {$ifdef SYSTEMINLINE}inline;{$endif}
|
||
|
||
begin
|
||
{$IFDEF CPU64}
|
||
Result:=Pointer(InterlockedDecrement64(Int64(Target)));
|
||
{$ELSE}
|
||
{$IFDEF CPU16}
|
||
Result:=Pointer(InterlockedDecrement(smallint(Target)));
|
||
{$ELSE}
|
||
Result:=Pointer(InterlockedDecrement(Longint(Target)));
|
||
{$ENDIF}
|
||
{$ENDIF}
|
||
end;
|
||
|
||
|
||
function AtomicCmpExchange(var Target: pointer; NewValue, Comperand: pointer): pointer; {$ifdef SYSTEMINLINE}inline;{$endif}
|
||
|
||
begin
|
||
{$IFDEF CPU64}
|
||
Result:=Pointer(InterlockedCompareExchange64(Int64(Target),Int64(NewValue), Int64(Comperand)));
|
||
{$ELSE}
|
||
{$IFDEF CPU16}
|
||
Result:=Pointer(InterlockedCompareExchange(smallint(Target),smallint(NewValue),smallint(Comperand)));
|
||
{$ELSE}
|
||
Result:=Pointer(InterlockedCompareExchange(LongInt(Target),LongInt(NewValue), LongInt(Comperand)));
|
||
{$ENDIF}
|
||
{$ENDIF}
|
||
end;
|
||
|
||
|
||
function AtomicExchange(var Target: pointer;Source : pointer) : pointer; {$ifdef SYSTEMINLINE}inline;{$endif}
|
||
|
||
begin
|
||
{$IFDEF CPU64}
|
||
Result:=Pointer(InterlockedExchange64(Int64(Target),Int64(Source)));
|
||
{$ELSE}
|
||
{$IFDEF CPU16}
|
||
Result:=Pointer(InterlockedExchange(smallint(Target),smallint(Source)));
|
||
{$ELSE}
|
||
Result:=Pointer(InterlockedExchange(LongInt(Target),LongInt(Source)));
|
||
{$ENDIF}
|
||
{$ENDIF}
|
||
end;
|
||
|
||
{$endif FPC_SYSTEM_DISABLE_INTERLOCK_POINTER_OVERLOAD}
|
||
|
||
function AtomicIncrement (var Target: Cardinal) : Cardinal; {$ifdef SYSTEMINLINE}inline;{$endif}
|
||
|
||
begin
|
||
Result:=Cardinal(InterlockedIncrement(Longint(Target)));
|
||
end;
|
||
|
||
|
||
function AtomicDecrement (var Target: Cardinal) : Cardinal; {$ifdef SYSTEMINLINE}inline;{$endif}
|
||
|
||
begin
|
||
Result:=Cardinal(InterlockedIncrement(Longint(Target)));
|
||
end;
|
||
|
||
|
||
function AtomicCmpExchange(var Target: Cardinal; NewValue, Comperand: Cardinal): Cardinal; {$ifdef SYSTEMINLINE}inline;{$endif}
|
||
|
||
begin
|
||
Result:=Cardinal(InterlockedCompareExchange(Longint(Target),Longint(NewValue), Longint(Comperand)));
|
||
end;
|
||
|
||
|
||
function AtomicExchange (var Target: Cardinal;Source : Cardinal) : Cardinal; {$ifdef SYSTEMINLINE}inline;{$endif}
|
||
|
||
begin
|
||
Result:=Cardinal(InterlockedExchange(Longint(Target),Longint(Source)));
|
||
end;
|
||
|
||
|
||
{$ifdef cpu64}
|
||
function AtomicIncrement (var Target: qword) : qword; {$ifdef SYSTEMINLINE}inline;{$endif}
|
||
|
||
begin
|
||
Result:=QWord(InterlockedIncrement64(Int64(Target)));
|
||
end;
|
||
|
||
|
||
function AtomicDecrement (var Target: qword) : qword; {$ifdef SYSTEMINLINE}inline;{$endif}
|
||
|
||
begin
|
||
Result:=QWord(InterlockedDecrement64(int64(Target)));
|
||
end;
|
||
|
||
|
||
function AtomicCmpExchange(var Target: qword; NewValue, Comperand: qword): qword; {$ifdef SYSTEMINLINE}inline;{$endif}
|
||
|
||
begin
|
||
Result:=QWord(InterlockedCompareExchange64(Int64(Target),Int64(NewValue), Int64(Comperand)));
|
||
end;
|
||
|
||
|
||
function AtomicExchange (var Target: qword;Source : qword) : qword; {$ifdef SYSTEMINLINE}inline;{$endif}
|
||
|
||
begin
|
||
Result:=QWord(InterlockedExchange64(Int64(Target),Int64(Source)));
|
||
end;
|
||
{$endif}
|