12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919192019211922192319241925192619271928192919301931193219331934193519361937193819391940194119421943194419451946194719481949195019511952195319541955195619571958195919601961196219631964196519661967196819691970197119721973197419751976197719781979198019811982198319841985198619871988198919901991199219931994199519961997199819992000200120022003200420052006200720082009201020112012201320142015201620172018201920202021202220232024202520262027202820292030203120322033203420352036203720382039204020412042204320442045204620472048204920502051205220532054205520562057205820592060206120622063206420652066206720682069207020712072207320742075207620772078207920802081208220832084208520862087208820892090209120922093209420952096209720982099210021012102210321042105210621072108210921102111211221132114211521162117211821192120212121222123212421252126212721282129213021312132213321342135213621372138 |
- {
- 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 = 64; { Stack size margin for stack checking }
- {$elseif defined(CPUZ80)}
- STACK_MARGIN = 64; { Stack size margin for stack checking }
- {$else}
- STACK_MARGIN = 16384; { Stack size margin for stack checking }
- {$endif}
- { Random / Randomize constants }
- OldRandSeed : Cardinal = 0;
- { 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}
- { 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}
- {$else not cpumipsel}
- {$ifdef cpumips}
- {$ifdef SYSPROCDEFINED}
- {$Error Can't determine processor type !}
- {$endif}
- {$i mips.inc} { Case dependent, don't change }
- {$define SYSPROCDEFINED}
- {$endif cpumips}
- {$endif not cpumipsel}
- {$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}
- {$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 : char);
- 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:char):SizeInt;
- begin
- IndexChar:=IndexByte(Buf,Len,byte(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
- FillByte(data,len,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) and not defined(VER3_0)}
- result := aligntoqword(p);
- {$else CPUM68K and not VER3_0}
- result := aligntoptr(p);
- {$endif CPUM68K and not VER3_0}
- end;
- {$if not defined(VER3_0) and not defined(VER3_2)}
- type
- TRttiDataCommon =
- {$ifdef FPC_REQUIRES_PROPER_ALIGNMENT}
- packed
- {$endif}
- record
- Attrs: Pointer;
- end;
- {$endif not VER3_0 and 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}
- {$i aliases.inc}
- {****************************************************************************
- Run-Time Type Information (RTTI) declarations
- ****************************************************************************}
- {$ifdef FPC_HAS_FEATURE_RTTI}
- {$i rttidecl.inc}
- {$endif FPC_HAS_FEATURE_RTTI}
- {*****************************************************************************
- 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}
- { Pascal translation of https://github.com/dajobe/libmtwist }
- {* -*- Mode: c; c-basic-offset: 2 -*-
- *
- * mt.c - Mersenne Twister functions
- *
- * This is free and unencumbered software released into the public domain.
- *
- * Anyone is free to copy, modify, publish, use, compile, sell, or
- * distribute this software, either in source code form or as a compiled
- * binary, for any purpose, commercial or non-commercial, and by any
- * means.
- *
- * In jurisdictions that recognize copyright laws, the author or authors
- * of this software dedicate any and all copyright interest in the
- * software to the public domain. We make this dedication for the benefit
- * of the public at large and to the detriment of our heirs and
- * successors. We intend this dedication to be an overt act of
- * relinquishment in perpetuity of all present and future rights to this
- * software under copyright law.
- *
- * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
- * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
- * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
- * IN NO EVENT SHALL THE AUTHORS BE LIABLE FOR ANY CLAIM, DAMAGES OR
- * OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
- * ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
- * OTHER DEALINGS IN THE SOFTWARE.
- *
- * For more information, please refer to <http://unlicense.org/>
- *
- *}
- {$R-} {range checking off}
- {$Q-} {overflow checking off}
- const
- MTWIST_N = 624;
- MTWIST_M = 397;
- MT_STATIC_SEED = 5489;
- MTWIST_UPPER_MASK = cardinal($80000000);
- MTWIST_LOWER_MASK = cardinal($7FFFFFFF);
- MTWIST_MATRIX_A = cardinal($9908B0DF);
- var
- mt_state: array[0..MTWIST_N-1] of cardinal;
- const
- mt_index: cardinal = MTWIST_N+1;
- function MTWIST_MIXBITS(u, v: cardinal): cardinal; inline;
- begin
- result:=(u and MTWIST_UPPER_MASK) or (v and MTWIST_LOWER_MASK);
- end;
- function MTWIST_TWIST(u, v: cardinal): cardinal; inline;
- begin
- { the construct at the end is equivalent to
- if odd(v) then
- MTWIST_MATRIX_A
- else
- 0
- }
- result:=(MTWIST_MIXBITS(u,v) shr 1) xor (cardinal(-(v and 1)) and MTWIST_MATRIX_A);
- end;
- procedure mtwist_init(seed: cardinal);
- var
- i: longint;
- begin
- mt_state[0]:=seed;
- for i:=1 to MTWIST_N-1 do
- mt_state[i]:=cardinal(1812433253) * (mt_state[i-1] xor (mt_state[i-1] shr 30)) + i;
- { still need to update the state }
- mt_index:=MTWIST_N;
- end;
- procedure mtwist_update_state;
- var
- count: longint;
- begin
- { The original C code uses pointers, doesn't play nice with JVM backend;
- it counts from N-M+1 downto 0 (0 not included) for the first loop, which
- should initialise the first N-M+1 elements -- doing so gives the wrong
- results though (different from the old generator, and it also doesn't
- match the algorithm description), so we use only N-M iterations. They don't
- seem to test this one element and its value does not impact subsequent
- numbers, so it's probably a bug in their implementation.
- }
- for count:=0 to MTWIST_N-MTWIST_M-1 do
- mt_state[count]:=mt_state[count+MTWIST_M] xor MTWIST_TWIST(mt_state[count],mt_state[count+1]);
- for count:=MTWIST_N-MTWIST_M to MTWIST_N-2 do
- mt_state[count]:=mt_state[count+(MTWIST_M-MTWIST_N)] xor MTWIST_TWIST(mt_state[count],mt_state[count+1]);
- mt_state[MTWIST_N-1]:=mt_state[MTWIST_M-1] xor MTWIST_TWIST(mt_state[MTWIST_N-1],mt_state[0]);
- mt_index:=0;
- end;
- function mtwist_u32rand: cardinal;
- var
- l_index :cardinal;
- begin
- l_index:=mt_index;
- inc(mt_index);
- if (RandSeed<>OldRandSeed) or
- (l_index>=MTWIST_N+1) then
- begin
- mtwist_init(RandSeed);
- { 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
- }
- RandSeed:=not(RandSeed);
- OldRandSeed:=RandSeed;
- l_index:=MTWIST_N;
- end;
- if l_index=MTWIST_N then
- begin
- mtwist_update_state;
- l_index:=0;
- mt_index:=1;
- end;
- result:=mt_state[l_index];
- result:=result xor (result shr 11);
- result:=result xor ((result shl 7) and cardinal($9D2C5680));
- result:=result xor ((result shl 15) and cardinal($EFC60000));
- result:=result xor (result shr 18);
- end;
- function random(l:longint): longint;
- begin
- { otherwise we can return values = l (JM) }
- if (l < 0) then
- inc(l);
- random := longint((int64(mtwist_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 < 0) then
- begin
- inc(l);
- q:=qword(-l)
- end
- else
- q:=qword(l);
- a:=mtwist_u32rand;
- b:=mtwist_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;
- begin
- random := mtwist_u32rand * (extended(1.0)/(int64(1) shl 32));
- end;
- {$endif}
- {$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'];
- 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 - STACK_MARGIN;
- 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 : {$if defined(VER2)}longint
- {$elseif defined(VER3_0)}sizeint
- {$else}ALUUInt{$endif}
- ;
- 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}
- {*****************************************************************************
- 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;
- function SysBackTraceStr (Addr: CodePointer): ShortString;
- begin
- SysBackTraceStr:=' $'+hexstr(addr);
- end;
- {$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):ppchar; // const ?
- {$ifdef EXCLUDE_COMPLEX_PROCS}
- begin
- runerror(217);
- end;
- {$else EXCLUDE_COMPLEX_PROCS}
- // Extra allocate reserveentries pchar'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 : ppchar;
- i : ObjpasInt;
- begin
- if High(s)<Low(s) Then Exit(NIL);
- Getmem(p,sizeof(pchar)*(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]:=pchar(s[i]);
- p[high(s)+1+Reserveentries]:=nil;
- ArrayStringToPPchar:=p;
- end;
- {$endif EXCLUDE_COMPLEX_PROCS}
- Function StringToPPChar(Var S:AnsiString;ReserveEntries:integer):ppchar;
- {
- Create a PPChar 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(PChar(S),ReserveEntries);
- end;
- {$endif FPC_HAS_FEATURE_ANSISTRINGS}
- Function StringToPPChar(S: PChar;ReserveEntries:integer):ppchar;
- var
- i,nr : ObjpasInt;
- Buf : ^char;
- p : ppchar;
- 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(pchar));
- 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 msg='' 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: pchar; inplace: boolean = true);
- var
- i : ObjpasInt;
- len : sizeint;
- newp : pchar;
- 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 : pchar;
- 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:=pchar(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 (s='') 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 (s='') 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 (s='') 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 (s='') or (InOutRes <> 0) then
- exit;
- Do_mkdir(GetDirStrFromShortstring(S));
- End;
- Procedure RmDir(Const s: shortstring);[IOCheck];
- Begin
- If (s='') or (InOutRes <> 0) then
- exit;
- Do_rmdir(GetDirStrFromShortstring(S));
- End;
- Procedure ChDir(Const s: shortstring);[IOCheck];
- Begin
- If (s='') 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 (s='') or (InOutRes <> 0) then
- exit;
- Do_mkdir(S);
- End;
- Procedure RmDir(Const s: UnicodeString);[IOCheck];
- Begin
- if (s='') or (InOutRes <> 0) then
- exit;
- Do_rmdir(S);
- End;
- Procedure ChDir(Const s: UnicodeString);[IOCheck];
- Begin
- if (s='') 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}
|