| 1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528 | { ****************************************************************************    This file is part of the Free Pascal run time library.    Copyright (c) 1999-2015 by Free Pascal development team    Free Pascal - OS/2 runtime library    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.****************************************************************************}unit System;interface{$ifdef SYSTEMDEBUG}  {$define SYSTEMEXCEPTIONDEBUG}  {.$define IODEBUG}  {.$define DEBUGENVIRONMENT}  {.$define DEBUGARGUMENTS}  {.$define DEBUGOSERRORS}{$endif SYSTEMDEBUG}{$DEFINE OS2EXCEPTIONS}{$DEFINE OS2UNICODE}{$define DISABLE_NO_THREAD_MANAGER}{$define DISABLE_NO_DYNLIBS_MANAGER}{$DEFINE HAS_GETCPUCOUNT}{$define FPC_SYSTEM_HAS_SYSDLH}{$I systemh.inc}const  LineEnding = #13#10;{ LFNSupport is defined separately below!!! }  DirectorySeparator = '\';  DriveSeparator = ':';  ExtensionSeparator = '.';  PathSeparator = ';';  AllowDirectorySeparators : set of char = ['\','/'];  AllowDriveSeparators : set of char = [':'];{ FileNameCaseSensitive and FileNameCasePreserving are defined separately below!!! }  MaxExitCode = 65535;  MaxPathLen = 260;(* MaxPathLen is referenced as constant from unit SysUtils   *)(* - changing to variable or typed constant is not possible. *)  AllFilesMask = '*';  RealMaxPathLen: word = MaxPathLen;(* Default value only - real value queried from the system on startup. *)type  TOS = (osDOS, osOS2, osDPMI); (* For compatibility with target EMX *)  TUConvObject = pointer;  TLocaleObject = pointer;const  OS_Mode: TOS = osOS2; (* For compatibility with target EMX *)  First_Meg: pointer = nil; (* For compatibility with target EMX *)  UnusedHandle=-1;  StdInputHandle=0;  StdOutputHandle=1;  StdErrorHandle=2;  LFNSupport: boolean = true;  FileNameCaseSensitive: boolean = false;  FileNameCasePreserving: boolean = true;  CtrlZMarksEOF: boolean = true; (* #26 is considered as end of file *)  RTLUsesWinCP: boolean = true; (* UnicodeString manager shall treat *)(* codepage numbers passed to RTL functions as those used under MS Windows *)(* and translates them to their OS/2 equivalents if necessary.             *)  sLineBreak = LineEnding;  DefaultTextLineBreakStyle : TTextLineBreakStyle = tlbsCRLF;var{ C-compatible arguments and environment }  argc  : longint;  argv  : ppchar;  envp  : ppchar;  EnvC: cardinal;(* Pointer to the block of environment variables - used e.g. in unit Dos. *)  Environment: PChar;var(* Type / run mode of the current process: *)(* 0 .. full screen OS/2 session           *)(* 1 .. DOS session                        *)(* 2 .. VIO windowable OS/2 session        *)(* 3 .. Presentation Manager OS/2 session  *)(* 4 .. detached (background) OS/2 process *)  ApplicationType: cardinal;const  HeapAllocFlags: cardinal = $53; (* Compatible to VP/2 *)  (* mfPag_Commit or mfObj_Tile or mfPag_Write or mfPag_Read *)function ReadUseHighMem: boolean;procedure WriteUseHighMem (B: boolean);(* Is allocation of memory above 512 MB address limit allowed? Even if use   *)(* of high memory is supported by the underlying OS/2 version, just a subset *)(* of OS/2 API functions can work with memory buffers located in high        *)(* memory. Since FPC RTL allocates heap using memory pools received from     *)(* the operating system and thus memory allocation from the operating system *)(* may happen at a different time than allocation of memory from FPC heap,   *)(* use of high memory shall be enabled only if the given program is ensured  *)(* not to use any OS/2 API function beyond the limited set supporting it any *)(* time between enabling this feature and program termination.               *)property  UseHighMem: boolean read ReadUseHighMem write WriteUseHighMem;(* UseHighMem is provided for compatibility with 2.0.x. *){$IFDEF OS2UNICODE}function OS2CPtoRtlCP (CP: cardinal; ReqFlags: byte;                                  var UConvObj: TUConvObject): TSystemCodepage;function RtlCPtoOS2CP (RtlCP: TSystemCodepage; ReqFlags: byte;                                         var UConvObj: TUConvObject): cardinal;function OS2CPtoRtlCP (CP: cardinal; ReqFlags: byte): TSystemCodepage;function RtlCPtoOS2CP (RtlCP: TSystemCodepage; ReqFlags: byte): cardinal;(* function RtlChangeCP (CP: TSystemCodePage; const stdcp: TStandardCodePageEnum): longint; *){$ENDIF OS2UNICODE}const(* Are file sizes > 2 GB (64-bit) supported on the current system? *)  FSApi64: boolean = false;(* Is full Unicode support provided by the underlying OS/2 version available *)(* and successfully initialized (otherwise dummy routines need to be used).  *)  UniAPI: boolean = false;(* Support for tracking I/O errors returned by OS/2 API calls - emulation *)(* of GetLastError / fpGetError functionality used e.g. in Sysutils.      *)type  TOSErrorWatch = procedure (Error: cardinal);procedure NoErrorTracking (Error: cardinal);(* This shall be invoked whenever a non-zero error is returned by OS/2 APIs *)(* used in the RTL. Direct OS/2 API calls in user programs are not covered! *)const  OSErrorWatch: TOSErrorWatch = @NoErrorTracking;function SetOSErrorTracking (P: pointer): pointer;procedure SetDefaultOS2FileType (FType: ShortString);procedure SetDefaultOS2Creator (Creator: ShortString);type  TDosOpenL = function (FileName: PChar; var Handle: THandle;                        var Action: cardinal; InitSize: int64;                        Attrib, OpenFlags, FileMode: cardinal;                                                 EA: pointer): cardinal; cdecl;  TDosSetFilePtrL = function (Handle: THandle; Pos: int64; Method: cardinal;                                        var PosActual: int64): cardinal; cdecl;  TDosSetFileSizeL = function (Handle: THandle; Size: int64): cardinal; cdecl;  TUniCreateUConvObject = function (const CpName: PWideChar;                               var UConv_Object: TUConvObject): longint; cdecl;  TUniFreeUConvObject = function (UConv_Object: TUConvObject): longint; cdecl;  TUniMapCpToUcsCp = function (const Codepage: cardinal;                   CodepageName: PWideChar; const N: cardinal): longint; cdecl;  TUniUConvFromUcs = function (UConv_Object: TUConvObject;       var UcsBuf: PWideChar; var UniCharsLeft: longint; var OutBuf: PChar;         var OutBytesLeft: longint; var NonIdentical: longint): longint; cdecl;  TUniUConvToUcs = function (UConv_Object: TUConvObject; var InBuf: PChar;   var InBytesLeft: longint; var UcsBuf: PWideChar; var UniCharsLeft: longint;                                    var NonIdentical: longint): longint; cdecl;  TUniToLower = function (UniCharIn: WideChar): WideChar; cdecl;  TUniToUpper = function (UniCharIn: WideChar): WideChar; cdecl;  TUniStrColl = function (Locale_Object: TLocaleObject;                                  const UCS1, UCS2: PWideChar): longint; cdecl;  TUniCreateLocaleObject = function (LocaleSpecType: longint;                             const LocaleSpec: pointer;                             var Locale_Object: TLocaleObject): longint; cdecl;  TUniFreeLocaleObject = function (Locale_Object: TLocaleObject): longint;                                                                         cdecl;  TUniMapCtryToLocale = function (CountryCode: cardinal; LocaleName: PWideChar;                                             BufSize: longint): longint; cdecl;const  DosCallsHandle: THandle = THandle (-1);{$IFDEF OS2UNICODE}  UConvHandle: THandle = THandle (-1);  LibUniHandle: THandle = THandle (-1);{$ENDIF OS2UNICODE}var  Sys_DosOpenL: TDosOpenL;  Sys_DosSetFilePtrL: TDosSetFilePtrL;  Sys_DosSetFileSizeL: TDosSetFileSizeL;{$IFDEF OS2UNICODE}  Sys_UniCreateUConvObject: TUniCreateUConvObject;  Sys_UniFreeUConvObject: TUniFreeUConvObject;  Sys_UniMapCpToUcsCp: TUniMapCpToUcsCp;  Sys_UniUConvFromUcs: TUniUConvFromUcs;  Sys_UniUConvToUcs: TUniUConvToUcs;  Sys_UniToLower: TUniToLower;  Sys_UniToUpper: TUniToUpper;  Sys_UniStrColl: TUniStrColl;  Sys_UniCreateLocaleObject: TUniCreateLocaleObject;  Sys_UniFreeLocaleObject: TUniFreeLocaleObject;  Sys_UniMapCtryToLocale: TUniMapCtryToLocale;{$ENDIF OS2UNICODE}{$IFDEF SYSTEMDEBUG}var  SysLastOSError: cardinal;{$ENDIF SYSTEMDEBUG}function GetDynLibsError: longint;function GetDynLibsErrPath: PChar;implementation{*****************************************************************************                        System unit initialization.****************************************************************************}{$I system.inc}{*****************************************************************************                           Exception handling.****************************************************************************}{$IFDEF OS2EXCEPTIONS}var  { value of the stack segment    to check if the call stack can be written on exceptions }  _SS : Cardinal;function Is_Prefetch (P: pointer): boolean;  var    A: array [0..15] of byte;    DoAgain: boolean;    InstrLo, InstrHi, OpCode: byte;    I: longint;    MemSize, MemAttrs: cardinal;    RC: cardinal;  begin    Is_Prefetch := false;    MemSize := SizeOf (A);    RC := DosQueryMem (P, MemSize, MemAttrs);    if RC <> 0 then     OSErrorWatch (RC)    else if (MemAttrs and (mfPag_Free or mfPag_Commit) <> 0)                                               and (MemSize >= SizeOf (A)) then     Move (P^, A [0], SizeOf (A))    else     Exit;    I := 0;    DoAgain := true;    while DoAgain and (I < 15) do      begin        OpCode := A [I];        InstrLo := OpCode and $f;        InstrHi := OpCode and $f0;        case InstrHi of          { prefix? }          $20, $30:            DoAgain := (InstrLo and 7) = 6;          $60:            DoAgain := (InstrLo and $c) = 4;          $f0:            DoAgain := InstrLo in [0, 2, 3];          $0:            begin              Is_Prefetch := (InstrLo = $f) and (A [I + 1] in [$D, $18]);              Exit;            end;          else            DoAgain := false;        end;        Inc (I);      end;  end;const  MaxExceptionLevel = 16;  ExceptLevel: byte = 0;var  ExceptEIP: array [0..MaxExceptionLevel - 1] of longint;  ExceptError: array [0..MaxExceptionLevel - 1] of byte;  ResetFPU: array [0..MaxExceptionLevel - 1] of boolean;{$ifdef SYSTEMEXCEPTIONDEBUG}procedure DebugHandleErrorAddrFrame (Error: longint; Addr, Frame: pointer);begin if IsConsole then  begin   Write (StdErr, ' HandleErrorAddrFrame (error = ', Error);   Write (StdErr, ', addr = ', hexstr (PtrUInt (Addr), 8));   WriteLn (StdErr, ', frame = ', hexstr (PtrUInt (Frame), 8), ')');  end; HandleErrorAddrFrame (Error, Addr, Frame);end;{$endif SYSTEMEXCEPTIONDEBUG}procedure JumpToHandleErrorFrame;var EIP, EBP, Error: longint;{$IFDEF SYSTEMEXCEPTIONDEBUG} ESP, EBP1: longint;{$ENDIF SYSTEMEXCEPTIONDEBUG}begin (* save ebp *) asm  movl (%ebp),%eax  movl %eax,ebp{$IFDEF SYSTEMEXCEPTIONDEBUG}  movl %ebp,%eax  movl %eax,EBP1  movl %esp,%eax  movl %eax,ESP{$ENDIF SYSTEMEXCEPTIONDEBUG} end;{$ifdef SYSTEMEXCEPTIONDEBUG} if IsConsole then  WriteLn (StdErr, 'Exception level at start of JumpToHandleErrorFrame = ', ExceptLevel);{$endif SYSTEMEXCEPTIONDEBUG} if (ExceptLevel > 0) then  Dec (ExceptLevel); EIP := ExceptEIP [ExceptLevel]; Error := ExceptError [ExceptLevel];{$ifdef SYSTEMEXCEPTIONDEBUG} if IsConsole then  begin   WriteLn (StdErr, 'In JumpToHandleErrorFrame error = ', Error);   WriteLn (StdErr, 'EBP on entry: ', HexStr (EBP1, 8));   WriteLn (StdErr, 'Previous EBP: ', HexStr (EBP, 8));   WriteLn (StdErr, 'ESP on entry: ', HexStr (ESP, 8));  end;{$endif SYSTEMEXCEPTIONDEBUG} if ResetFPU [ExceptLevel] then  SysResetFPU; { build a fake stack } asm{$ifdef REGCALL}  movl   ebp,%ecx  movl   eip,%edx  movl   error,%eax  pushl  eip  movl   ebp,%ebp // Change frame pointer{$else}  movl   ebp,%eax  pushl  %eax  movl   eip,%eax  pushl  %eax  movl   error,%eax  pushl  %eax  movl   eip,%eax  pushl  %eax  movl   ebp,%ebp // Change frame pointer{$endif}{$ifdef SYSTEMEXCEPTIONDEBUG}  jmpl   DebugHandleErrorAddrFrame{$else not SYSTEMEXCEPTIONDEBUG}  jmpl   HandleErrorAddrFrame{$endif SYSTEMEXCEPTIONDEBUG} end;end;function System_Exception_Handler (Report: PExceptionReportRecord;                                   RegRec: PExceptionRegistrationRecord;                                   Context: PContextRecord;                                   DispContext: pointer): cardinal; cdecl;var Res: cardinal; Err: byte; Must_Reset_FPU: boolean; RC: cardinal;{$IFDEF SYSTEMEXCEPTIONDEBUG} CurSS, CurESP, CurEBP: cardinal; B: byte;{$ENDIF SYSTEMEXCEPTIONDEBUG}begin{$ifdef SYSTEMEXCEPTIONDEBUG} if IsConsole then  begin    asm      pushl %eax      xorl %eax,%eax      movw %ss,%ax      movl %eax,CurSS      movl %esp,%eax      movl %eax,CurESP      movl %ebp,%eax      movl %eax,CurEBP      popl %eax    end;    WriteLn (StdErr, '------------------------------------------------------');    WriteLn (StdErr, 'In System_Exception_Handler, error = ',                                            HexStr (Report^.Exception_Num, 8));    WriteLn (StdErr, 'Handler flags = ', HexStr (Report^.HandlerFlags, 8));    WriteLn (StdErr, 'Nested_RepRec = ', HexStr (PtrUInt (Report^.Nested_RepRec), 8));    WriteLn (StdErr, 'Amount of passed parameters = ', Report^.ParamCount);    WriteLn (StdErr, 'Context SS = ', HexStr (Context^.Reg_SS, 8),                                         ', current SS = ', HexStr (CurSS, 8));    WriteLn (StdErr, 'Current ESP = ', HexStr (CurESP, 8),                                       ', current EBP = ', HexStr (CurEBP, 8));    WriteLn (StdErr, 'Context flags = ', HexStr (Context^.ContextFlags, 8));    WriteLn (StdErr, 'Thread ID = ', ThreadID);    if Context^.ContextFlags and Context_Control <> 0 then     begin      WriteLn (StdErr, 'EBP = ', HexStr (Context^.Reg_EBP, 8),                     ', SS = ', HexStr (Context^.Reg_SS, 8),                     ', ESP = ', HexStr (Context^.Reg_ESP, 8));      WriteLn (StdErr, 'CS = ', HexStr (Context^.Reg_CS, 8),                     ', EIP = ', HexStr (Context^.Reg_EIP, 8),                     ', EFlags = ', HexStr (Context^.Flags, 8));     end;    if Context^.ContextFlags and Context_Floating_Point <> 0 then     begin      for B := 1 to 6 do       Write (StdErr, 'Ctx Env [', B, '] = ', HexStr (Context^.Env [B], 8),                                                                         ', ');      WriteLn (StdErr, 'Ctx Env [7] = ', HexStr (Context^.Env [7], 8));      for B := 0 to 6 do       Write (StdErr, 'FPU stack [', B, '] = ', Context^.FPUStack [B], ', ');      WriteLn (StdErr, 'FPU stack [7] = ', Context^.FPUStack [7]);     end;    if Context^.ContextFlags and Context_Segments <> 0 then     WriteLn (StdErr, 'GS = ', HexStr (Context^.Reg_GS, 8),                    ', FS = ', HexStr (Context^.Reg_FS, 8),                    ', ES = ', HexStr (Context^.Reg_ES, 8),                    ', DS = ', HexStr (Context^.Reg_DS, 8));    if Context^.ContextFlags and Context_Integer <> 0 then     begin      WriteLn (StdErr, 'EDI = ', HexStr (Context^.Reg_EDI, 8),                     ', ESI = ', HexStr (Context^.Reg_ESI, 8));      WriteLn (StdErr, 'EAX = ', HexStr (Context^.Reg_EAX, 8),                     ', EBX = ', HexStr (Context^.Reg_EBX, 8),                     ', ECX = ', HexStr (Context^.Reg_ECX, 8),                     ', EDX = ', HexStr (Context^.Reg_EDX, 8));     end;  end;{$endif SYSTEMEXCEPTIONDEBUG} Res := Xcpt_Continue_Search; if Context^.Reg_SS = _SS then  begin   Err := 0;   Must_Reset_FPU := true;{$ifdef SYSTEMEXCEPTIONDEBUG}   if IsConsole then    Writeln (StdErr, 'Exception  ', HexStr (Report^.Exception_Num, 8));{$endif SYSTEMEXCEPTIONDEBUG}   case Report^.Exception_Num of    Xcpt_Integer_Divide_By_Zero,    Xcpt_Float_Divide_By_Zero:      Err := 200;    Xcpt_Array_Bounds_Exceeded:     begin      Err := 201;      Must_Reset_FPU := false;     end;    Xcpt_Unable_To_Grow_Stack:     begin      Err := 202;      Must_Reset_FPU := false;     end;    Xcpt_Float_Overflow:     Err := 205;    Xcpt_Float_Denormal_Operand,    Xcpt_Float_Underflow:     Err := 206;    {Context^.FloatSave.StatusWord := Context^.FloatSave.StatusWord and $ffffff00;}    Xcpt_Float_Inexact_Result,    Xcpt_Float_Invalid_Operation,    Xcpt_Float_Stack_Check:     Err := 207;    Xcpt_Integer_Overflow:     begin      Err := 215;      Must_Reset_FPU := false;     end;    Xcpt_Illegal_Instruction:          { if we're testing sse support, simply set the flag and continue }     if SSE_Check then      begin       OS_Supports_SSE := false;          { skip the offending movaps %xmm7, %xmm6 instruction }       Inc (Context^.Reg_EIP, 3);       Report^.Exception_Num := 0;       Res := Xcpt_Continue_Execution;      end     else      Err := 216;    Xcpt_Access_Violation:     { Athlon prefetch bug? }     if Is_Prefetch (pointer (Context^.Reg_EIP)) then      begin       { if yes, then retry }       Report^.Exception_Num := 0;       Res := Xcpt_Continue_Execution;      end     else      begin       Err := 216;{$ifdef SYSTEMEXCEPTIONDEBUG}       if IsConsole and (Report^.ParamCount >= 2) then        begin         Writeln (StdErr, 'Access violation flags: ', Report^.Parameters [0]);         WriteLn (StdErr, 'Fault address: ', HexStr (Report^.Parameters [1], 8));        end;{$endif SYSTEMEXCEPTIONDEBUG}      end;    Xcpt_Signal:     case Report^.Parameters [0] of      Xcpt_Signal_KillProc:       Err := 217;      Xcpt_Signal_Break,      Xcpt_Signal_Intr:       if Assigned (CtrlBreakHandler) then        if CtrlBreakHandler (Report^.Parameters [0] = Xcpt_Signal_Break) then         begin{$IFDEF SYSTEMEXCEPTIONDEBUG}          WriteLn (StdErr, 'CtrlBreakHandler returned true');{$ENDIF SYSTEMEXCEPTIONDEBUG}          Report^.Exception_Num := 0;          Res := Xcpt_Continue_Execution;          RC := DosAcknowledgeSignalException (Report^.Parameters [0]);          if RC <> 0 then           OSErrorWatch (RC);         end        else         Err := 217;     end;    Xcpt_Privileged_Instruction:     begin      Err := 218;      Must_Reset_FPU := false;     end;    else     begin      if ((Report^.Exception_Num and Xcpt_Severity_Code)                                                   = Xcpt_Fatal_Exception) then       Err := 217      else       Err := 255;     end;   end;   if (Err <> 0) and (ExceptLevel < MaxExceptionLevel) (* TH: The following line is necessary to avoid an endless loop *)                 and (Report^.Exception_Num < Xcpt_Process_Terminate)                                                                    then    begin     ExceptEIP [ExceptLevel] := Context^.Reg_EIP;     ExceptError [ExceptLevel] := Err;     ResetFPU [ExceptLevel] := Must_Reset_FPU;     Inc (ExceptLevel);     Context^.Reg_EIP := cardinal (@JumpToHandleErrorFrame);     Report^.Exception_Num := 0;     if Must_Reset_FPU and                   (Context^.ContextFlags and Context_Floating_Point <> 0) then      begin       { Control word is index 1 }       Context^.Env [1] := Default8087CW;       { Status word is index 2 }       Context^.Env [2] := Context^.Env [2] and not FPU_ExceptionMask;       { Tag word is index 3 }       Context^.Env [3] := $FFFF;{$ifdef SYSTEMEXCEPTIONDEBUG}       WriteLn (StdErr, 'After FPU status reset in context record:');       for B := 1 to 2 do        Write (StdErr, 'Ctx Env [', B, '] = ', HexStr (Context^.Env [B], 8),                                                                         ', ');       WriteLn (StdErr, 'Ctx Env [3] = ', HexStr (Context^.Env [3], 8));{$endif SYSTEMEXCEPTIONDEBUG}      end;     Res := Xcpt_Continue_Execution;{$ifdef SYSTEMEXCEPTIONDEBUG}     if IsConsole then      begin       WriteLn (StdErr, 'Exception Continue Exception set at ',                                   HexStr (ExceptEIP [Pred (ExceptLevel)], 8));       WriteLn (StdErr, 'EIP changed to ',                              HexStr (Context^.Reg_EIP, 8), ', error = ', Err);       WriteLn (StdErr, 'Exception level = ', ExceptLevel);       WriteLn (StdErr, 'ResetFPU = ', ResetFPU [Pred (ExceptLevel)]);      end;{$endif SYSTEMEXCEPTIONDEBUG}    end;  end else  if (Report^.Exception_Num = Xcpt_Signal) and    (Report^.Parameters [0] and (Xcpt_Signal_Intr or Xcpt_Signal_Break) <> 0)                                           and Assigned (CtrlBreakHandler) then{$IFDEF SYSTEMEXCEPTIONDEBUG}   begin    WriteLn (StdErr, 'XCPT_SIGNAL caught, CtrlBreakHandler assigned, Param = ',                                                       Report^.Parameters [0]);{$ENDIF SYSTEMEXCEPTIONDEBUG}   if CtrlBreakHandler (Report^.Parameters [0] = Xcpt_Signal_Break) then    begin{$IFDEF SYSTEMEXCEPTIONDEBUG}     WriteLn (StdErr, 'CtrlBreakHandler returned true');{$ENDIF SYSTEMEXCEPTIONDEBUG}     Report^.Exception_Num := 0;     Res := Xcpt_Continue_Execution;     RC := DosAcknowledgeSignalException (Report^.Parameters [0]);     if RC <> 0 then      OSErrorWatch (RC);    end   else    Err := 217;{$IFDEF SYSTEMEXCEPTIONDEBUG}   end  else   if IsConsole then    begin     WriteLn (StdErr, 'Ctx flags = ', HexStr (Context^.ContextFlags, 8));     if Context^.ContextFlags and Context_Floating_Point <> 0 then      begin       for B := 1 to 6 do        Write (StdErr, 'Ctx Env [', B, '] = ', HexStr (Context^.Env [B], 8),                                                                         ', ');        WriteLn (StdErr, 'Ctx Env [7] = ', HexStr (Context^.Env [7], 8));       for B := 0 to 6 do        Write (StdErr, 'FPU stack [', B, '] = ', Context^.FPUStack [B], ', ');       WriteLn (StdErr, 'FPU stack [7] = ', Context^.FPUStack [7]);      end;     if Context^.ContextFlags and Context_Segments <> 0 then      WriteLn (StdErr, 'GS = ', HexStr (Context^.Reg_GS, 8),                     ', FS = ', HexStr (Context^.Reg_FS, 8),                     ', ES = ', HexStr (Context^.Reg_ES, 8),                     ', DS = ', HexStr (Context^.Reg_DS, 8));     if Context^.ContextFlags and Context_Integer <> 0 then      begin       WriteLn (StdErr, 'EDI = ', HexStr (Context^.Reg_EDI, 8),                      ', ESI = ', HexStr (Context^.Reg_ESI, 8));       WriteLn (StdErr, 'EAX = ', HexStr (Context^.Reg_EAX, 8),                      ', EBX = ', HexStr (Context^.Reg_EBX, 8),                      ', ECX = ', HexStr (Context^.Reg_ECX, 8),                      ', EDX = ', HexStr (Context^.Reg_EDX, 8));      end;     if Context^.ContextFlags and Context_Control <> 0 then      begin       WriteLn (StdErr, 'EBP = ', HexStr (Context^.Reg_EBP, 8),                      ', SS = ', HexStr (Context^.Reg_SS, 8),                      ', ESP = ', HexStr (Context^.Reg_ESP, 8));       WriteLn (StdErr, 'CS = ', HexStr (Context^.Reg_CS, 8),                      ', EIP = ', HexStr (Context^.Reg_EIP, 8),                      ', EFlags = ', HexStr (Context^.Flags, 8));      end;    end;{$endif SYSTEMEXCEPTIONDEBUG} System_Exception_Handler := Res;end;var  ExcptReg: PExceptionRegistrationRecord; public name '_excptregptr';{$ifdef SYSTEMEXCEPTIONDEBUG}var OldExceptAddr, NewExceptAddr: PtrUInt;{$endif SYSTEMEXCEPTIONDEBUG}procedure Install_Exception_Handler;var T: cardinal; RC: cardinal;begin{$ifdef SYSTEMEXCEPTIONDEBUG}(* ThreadInfoBlock is located at FS:[0], the first      *)(* entry is pointer to head of exception handler chain. *) asm  movl $0,%eax  movl %fs:(%eax),%eax  movl %eax, OldExceptAddr end;{$endif SYSTEMEXCEPTIONDEBUG} with ExcptReg^ do  begin   Prev_Structure := nil;   ExceptionHandler := TExceptionHandler (@System_Exception_Handler);  end; (* Disable pop-up windows for errors and exceptions *) DosError (deDisableExceptions); DosSetExceptionHandler (ExcptReg^); if IsConsole then  begin   RC := DosSetSignalExceptionFocus (1, T);   if RC <> 0 then    OSErrorWatch (RC);   RC := DosAcknowledgeSignalException (Xcpt_Signal_Intr);   if RC <> 0 then    OSErrorWatch (RC);   RC := DosAcknowledgeSignalException (Xcpt_Signal_Break);   if RC <> 0 then    OSErrorWatch (RC);  end;{$ifdef SYSTEMEXCEPTIONDEBUG} asm  movl $0,%eax  movl %fs:(%eax),%eax  movl %eax, NewExceptAddr end;{$endif SYSTEMEXCEPTIONDEBUG}end;{$IFDEF SYSTEMDEBUG}const  OrigOSErrorWatch: TOSErrorWatch = nil;procedure TrackLastOSError (Error: cardinal);begin  SysLastOSError := Error;{$IFDEF DEBUGOSERRORS}  if IsConsole then   WriteLn (StdErr, 'Some OS/2 API returned error ', Error);{$ENDIF DEBUGOSERRORS}  OrigOSErrorWatch (Error);end;{$ENDIF SYSTEMDEBUG}procedure Remove_Exception_Handlers;var  RC: cardinal;begin  RC := DosUnsetExceptionHandler (ExcptReg^);  if RC <> 0 then   OSErrorWatch (RC);end;{$ENDIF OS2EXCEPTIONS}procedure system_exit;begin(*  if IsLibrary then    ExitDLL(ExitCode);*)(*  if not IsConsole then   begin     Close(stderr);     Close(stdout);     Close(erroutput);     Close(Input);     Close(Output);   end;*){$IFDEF OS2EXCEPTIONS}  Remove_Exception_Handlers;{$ENDIF OS2EXCEPTIONS}  DosExit (1{process}, exitcode);end;{$ASMMODE ATT}{****************************************************************************                    Miscellaneous related routines.****************************************************************************}function paramcount:longint;assembler;asm    movl argc,%eax    decl %eaxend {['EAX']};function paramstr(l:longint):string;var p:^Pchar;begin  if (l>=0) and (l<=paramcount) then  begin    p:=argv;    paramstr:=strpas(p[l]);  end    else paramstr:='';end;procedure randomize;var  dt: TSysDateTime;begin  // Hmm... Lets use timer  DosGetDateTime(dt);  randseed:=dt.hour+(dt.minute shl 8)+(dt.second shl 16)+(dt.sec100 shl 32);end;{****************************************************************************                    Error Message writing using messageboxes****************************************************************************}const  WinInitialize: TWinInitialize = nil;  WinCreateMsgQueue: TWinCreateMsgQueue = nil;  WinMessageBox: TWinMessageBox = nil;  EnvSize: cardinal = 0;var  ErrorBuf: array [0..ErrorBufferLength] of char;  ErrorLen: longint;  PMWinHandle: cardinal;function ErrorWrite (var F: TextRec): integer;{  An error message should always end with #13#10#13#10}var  P: PChar;  I: longint;begin  if F.BufPos > 0 then   begin     if F.BufPos + ErrorLen > ErrorBufferLength then       I := ErrorBufferLength - ErrorLen     else       I := F.BufPos;     Move (F.BufPtr^, ErrorBuf [ErrorLen], I);     Inc (ErrorLen, I);     ErrorBuf [ErrorLen] := #0;   end;  if ErrorLen > 3 then   begin     P := @ErrorBuf [ErrorLen];     for I := 1 to 4 do      begin        Dec (P);        if not (P^ in [#10, #13]) then          break;      end;   end;   if ErrorLen = ErrorBufferLength then     I := 4;   if (I = 4) then    begin      WinMessageBox (0, 0, @ErrorBuf, PChar ('Error'), 0, MBStyle);      ErrorLen := 0;    end;  F.BufPos := 0;  ErrorWrite := 0;end;function ErrorClose (var F: TextRec): integer;begin  if ErrorLen > 0 then   begin     WinMessageBox (0, 0, @ErrorBuf, PChar ('Error'), 0, MBStyle);     ErrorLen := 0;   end;  ErrorLen := 0;  ErrorClose := 0;end;function ErrorOpen (var F: TextRec): integer;begin  TextRec(F).InOutFunc := @ErrorWrite;  TextRec(F).FlushFunc := @ErrorWrite;  TextRec(F).CloseFunc := @ErrorClose;  ErrorOpen := 0;end;procedure AssignError (var T: Text);begin  Assign (T, '');  TextRec (T).OpenFunc := @ErrorOpen;  Rewrite (T);end;procedure SysInitStdIO;(*var  RC: cardinal;*)begin  { Setup stdin, stdout and stderr, for GUI apps redirect stderr,stdout to be    displayed in a messagebox }(*  StdInputHandle := longint(GetStdHandle(cardinal(STD_INPUT_HANDLE)));  StdOutputHandle := longint(GetStdHandle(cardinal(STD_OUTPUT_HANDLE)));  StdErrorHandle := longint(GetStdHandle(cardinal(STD_ERROR_HANDLE)));  if not IsConsole then   begin    RC := DosLoadModule (nil, 0, 'PMWIN', PMWinHandle);    if RC <> 0 then     OSErrorWatch (RC)    else     begin      RC := DosQueryProcAddr (PMWinHandle, 789, nil, pointer (WinMessageBox));      if RC <> 0 then       OSErrorWatch (RC)      else       begin        RC := DosQueryProcAddr (PMWinHandle, 763, nil, pointer (WinInitialize));        if RC <> 0 then         OSErrorWatch (RC)        else         begin          RC := DosQueryProcAddr (PMWinHandle, 716, nil, pointer (WinCreateMsgQueue));          if RC <> 0 then           OSErrorWatch (RC)          else           begin            WinInitialize (0);            WinCreateMsgQueue (0, 0);           end         end       end     end;    if RC <> 0 then     HandleError (2);     AssignError (StdErr);     AssignError (StdOut);     Assign (Output, '');     Assign (Input, '');   end  else   begin*)     OpenStdIO (Input, fmInput, StdInputHandle);     OpenStdIO (Output, fmOutput, StdOutputHandle);     OpenStdIO (ErrOutput, fmOutput, StdErrorHandle);     OpenStdIO (StdOut, fmOutput, StdOutputHandle);     OpenStdIO (StdErr, fmOutput, StdErrorHandle);(*   end;*)end;function strcopy(dest,source : pchar) : pchar;assembler;var  saveeax,saveesi,saveedi : longint;asm        movl    %edi,saveedi        movl    %esi,saveesi{$ifdef REGCALL}        movl    %eax,saveeax        movl    %edx,%edi{$else}        movl    source,%edi{$endif}        testl   %edi,%edi        jz      .LStrCopyDone        leal    3(%edi),%ecx        andl    $-4,%ecx        movl    %edi,%esi        subl    %edi,%ecx{$ifdef REGCALL}        movl    %eax,%edi{$else}        movl    dest,%edi{$endif}        jz      .LStrCopyAligned.LStrCopyAlignLoop:        movb    (%esi),%al        incl    %edi        incl    %esi        testb   %al,%al        movb    %al,-1(%edi)        jz      .LStrCopyDone        decl    %ecx        jnz     .LStrCopyAlignLoop        .balign  16.LStrCopyAligned:        movl    (%esi),%eax        movl    %eax,%edx        leal    0x0fefefeff(%eax),%ecx        notl    %edx        addl    $4,%esi        andl    %edx,%ecx        andl    $0x080808080,%ecx        jnz     .LStrCopyEndFound        movl    %eax,(%edi)        addl    $4,%edi        jmp     .LStrCopyAligned.LStrCopyEndFound:        testl   $0x0ff,%eax        jz      .LStrCopyByte        testl   $0x0ff00,%eax        jz      .LStrCopyWord        testl   $0x0ff0000,%eax        jz      .LStrCopy3Bytes        movl    %eax,(%edi)        jmp     .LStrCopyDone.LStrCopy3Bytes:        xorb     %dl,%dl        movw     %ax,(%edi)        movb     %dl,2(%edi)        jmp     .LStrCopyDone.LStrCopyWord:        movw    %ax,(%edi)        jmp     .LStrCopyDone.LStrCopyByte:        movb    %al,(%edi).LStrCopyDone:{$ifdef REGCALL}        movl    saveeax,%eax{$else}        movl    dest,%eax{$endif}        movl    saveedi,%edi        movl    saveesi,%esiend;threadvar  DefaultCreator: ShortString;  DefaultFileType: ShortString;procedure SetDefaultOS2FileType (FType: ShortString);begin{$WARNING Not implemented yet!}  DefaultFileType := FType;end;procedure SetDefaultOS2Creator (Creator: ShortString);begin{$WARNING Not implemented yet!}  DefaultCreator := Creator;end;(* The default handler does not store the OS/2 API error codes. *)procedure NoErrorTracking (Error: cardinal);beginend;function SetOSErrorTracking (P: pointer): pointer;begin SetOSErrorTracking := OSErrorWatch; if P = nil then  OSErrorWatch := @NoErrorTracking else  OSErrorWatch := TOSErrorWatch (P);end;procedure InitEnvironment;var env_count : longint;    cp : pchar;begin  env_count:=0;  cp:=environment;  while cp ^ <> #0 do    begin    inc(env_count);    while (cp^ <> #0) do inc(longint(cp)); { skip to NUL }    inc(longint(cp)); { skip to next character }    end;  envp := sysgetmem((env_count+1) * sizeof(pchar));  envc := env_count;  if (envp = nil) then exit;  cp:=environment;  env_count:=0;  while cp^ <> #0 do  begin    envp[env_count] := sysgetmem(strlen(cp)+1);    strcopy(envp[env_count], cp);{$IfDef DEBUGENVIRONMENT}    Writeln(stderr,'env ',env_count,' = "',envp[env_count],'"');{$EndIf}    inc(env_count);    while (cp^ <> #0) do      inc(longint(cp)); { skip to NUL }    inc(longint(cp)); { skip to next character }  end;  envp[env_count]:=nil;end;var(* Initialized by system unit initialization *)  PIB: PProcessInfoBlock;procedure InitArguments;var  arglen,  count   : PtrInt;  argstart,  pc,arg  : pchar;  quote   : char;  argvlen : PtrInt;  RC: cardinal;  procedure allocarg(idx,len: PtrInt);{    var      oldargvlen : PtrInt;}    begin      if idx>=argvlen then       begin{         oldargvlen:=argvlen;}         argvlen:=(idx+8) and (not 7);         sysreallocmem(argv,argvlen*sizeof(pointer));{         fillchar(argv[oldargvlen],(argvlen-oldargvlen)*sizeof(pointer),0);}       end;      { use realloc to reuse already existing memory }      { always allocate, even if length is zero, since }      { the arg. is still present!                     }      ArgV [Idx] := SysAllocMem (Succ (Len));    end;begin  CmdLine := SysAllocMem (MaxPathLen);  ArgV := SysAllocMem (8 * SizeOf (pointer));  ArgLen := StrLen (PChar (PIB^.Cmd));  Inc (ArgLen);  RC := DosQueryModuleName (PIB^.Handle, MaxPathLen, CmdLine);  if RC = 0 then   ArgVLen := Succ (StrLen (CmdLine))  else(* Error occurred - use program name from command line as fallback. *)   begin    Move (PIB^.Cmd^, CmdLine, ArgLen);    ArgVLen := ArgLen;   end;{ Get ArgV [0] }  ArgV [0] := SysAllocMem (ArgVLen);  Move (CmdLine^, ArgV [0]^, ArgVLen);  Count := 1;(* PC points to leading space after program name on command line *)  PC := PChar (PIB^.Cmd) + ArgLen;(* ArgLen contains size of command line arguments including leading space. *)  ArgLen := Succ (StrLen (PC));  SysReallocMem (CmdLine, ArgVLen + Succ (ArgLen));  Move (PC^, CmdLine [ArgVLen], Succ (ArgLen));(* ArgV has space for 8 parameters from the first allocation. *)  ArgVLen := 8;  { process arguments }  while pc^<>#0 do   begin     { skip leading spaces }     while pc^ in [#1..#32] do      inc(pc);     if pc^=#0 then      break;     { calc argument length }     quote:=' ';     argstart:=pc;     arglen:=0;     while (pc^<>#0) do      begin        case pc^ of          #1..#32 :            begin              if quote<>' ' then               inc(arglen)              else               break;            end;          '"' :            begin              if quote<>'''' then               begin                 if pchar(pc+1)^<>'"' then                  begin                    if quote='"' then                     quote:=' '                    else                     quote:='"';                  end                 else                  inc(pc);               end              else               inc(arglen);            end;          '''' :            begin              if quote<>'"' then               begin                 if pchar(pc+1)^<>'''' then                  begin                    if quote=''''  then                     quote:=' '                    else                     quote:='''';                  end                 else                  inc(pc);               end              else               inc(arglen);            end;          else            inc(arglen);        end;        inc(pc);      end;     { copy argument }     { Don't copy the first one, it is already there.}     If Count<>0 then      begin        allocarg(count,arglen);        quote:=' ';        pc:=argstart;        arg:=argv[count];        while (pc^<>#0) do         begin           case pc^ of             #1..#32 :               begin                 if quote<>' ' then                  begin                    arg^:=pc^;                    inc(arg);                  end                 else                  break;               end;             '"' :               begin                 if quote<>'''' then                  begin                    if pchar(pc+1)^<>'"' then                     begin                       if quote='"' then                        quote:=' '                       else                        quote:='"';                     end                    else                     inc(pc);                  end                 else                  begin                    arg^:=pc^;                    inc(arg);                  end;               end;             '''' :               begin                 if quote<>'"' then                  begin                    if pchar(pc+1)^<>'''' then                     begin                       if quote=''''  then                        quote:=' '                       else                        quote:='''';                     end                    else                     inc(pc);                  end                 else                  begin                    arg^:=pc^;                    inc(arg);                  end;               end;             else               begin                 arg^:=pc^;                 inc(arg);               end;           end;           inc(pc);         end;        arg^:=#0;      end; {$IfDef DEBUGARGUMENTS}     Writeln(stderr,'dos arg ',count,' #',arglen,'#',argv[count],'#'); {$EndIf}     inc(count);   end;  { get argc and create an nil entry }  argc:=count;  allocarg(argc,0);  { free unused memory }  sysreallocmem(argv,(argc+1)*sizeof(pointer));end;function GetFileHandleCount: longint;var L1: longint;    L2: cardinal;    RC: cardinal;begin    L1 := 0; (* Don't change the amount, just check. *)    RC := DosSetRelMaxFH (L1, L2);    if RC <> 0 then     begin      GetFileHandleCount := 50;      OSErrorWatch (RC);     end    else     GetFileHandleCount := L2;end;function CheckInitialStkLen (StkLen: SizeUInt): SizeUInt;begin  CheckInitialStkLen := StkLen;end;var  TIB: PThreadInfoBlock;  RC: cardinal;  P: pointer;  DW: cardinal;const  DosCallsName: array [0..8] of char = 'DOSCALLS'#0;{$IFDEF OS2UNICODE}  {$I sysucode.inc}{$ENDIF OS2UNICODE}begin{$IFDEF OS2EXCEPTIONS}  asm   xorl %eax,%eax   movw %ss,%ax   movl %eax,_SS  end;{$ENDIF OS2EXCEPTIONS}  DosGetInfoBlocks (@TIB, @PIB);  StackLength := CheckInitialStkLen (InitialStkLen);  { OS/2 has top of stack in TIB^.StackLimit - unlike Windows where it is in TIB^.Stack }  StackBottom := TIB^.StackLimit - StackLength;  {Set type of application}  ApplicationType := PIB^.ProcType;  ProcessID := PIB^.PID;  ThreadID := TIB^.TIB2^.TID;  IsConsole := ApplicationType <> 3;{$IFDEF SYSTEMDEBUG}  SysLastOSError := 0;  OrigOSErrorWatch := TOSErrorWatch (SetOSErrorTracking (@TrackLastOSError));{$ENDIF SYSTEMDEBUG}  {Query maximum path length (QSV_MAX_PATH_LEN = 1)}  RC := DosQuerySysInfo (1, 1, DW, SizeOf (DW));  if RC = 0 then   RealMaxPathLen := DW  else   OSErrorWatch (RC);  ExitProc := nil;{$IFDEF OS2EXCEPTIONS}  Install_Exception_Handler;{$ENDIF OS2EXCEPTIONS}  (* Initialize the amount of file handles *)  FileHandleCount := GetFileHandleCount;  {Initialize the heap.}  (* Logic is following:     The heap is initially restricted to low address space (< 512 MB).     If underlying OS/2 version allows using more than 512 MB per process     (OS/2 WarpServer for e-Business, eComStation, possibly OS/2 Warp 4.0     with FP13 and above as well), use of this high memory is allowed for     future memory allocations at the end of System unit initialization.     The consequences are that the compiled application can allocate more     memory, but it must make sure to use direct DosAllocMem calls if it     needs a memory block for some system API not supporting high memory.     This is probably no problem for direct calls to these APIs, but     there might be situations when a memory block needs to be passed     to a 3rd party DLL which in turn calls such an API call. In case     of problems usage of high memory can be turned off by setting     UseHighMem to false - the program should change the setting at its     very beginning (e.g. in initialization section of the first unit     listed in the "uses" section) to avoid having preallocated memory     from the high memory region before changing value of this variable. *)  InitHeap;  Sys_DosOpenL := @DummyDosOpenL;  Sys_DosSetFilePtrL := @DummyDosSetFilePtrL;  Sys_DosSetFileSizeL := @DummyDosSetFileSizeL;  RC := DosQueryModuleHandle (@DosCallsName [0], DosCallsHandle);  if RC = 0 then   begin    RC := DosQueryProcAddr (DosCallsHandle, OrdDosOpenL, nil, P);    if RC = 0 then     begin      Sys_DosOpenL := TDosOpenL (P);      RC := DosQueryProcAddr (DosCallsHandle, OrdDosSetFilePtrL, nil, P);      if RC = 0 then       begin        Sys_DosSetFilePtrL := TDosSetFilePtrL (P);        RC := DosQueryProcAddr (DosCallsHandle, OrdDosSetFileSizeL, nil, P);        if RC = 0 then         begin          Sys_DosSetFileSizeL := TDosSetFileSizeL (P);          FSApi64 := true;         end;       end;     end;    if RC <> 0 then     OSErrorWatch (RC);    RC := DosQueryProcAddr (DosCallsHandle, OrdDosAllocThreadLocalMemory,                                                                       nil, P);    if RC = 0 then     begin      DosAllocThreadLocalMemory := TDosAllocThreadLocalMemory (P);      RC := DosQueryProcAddr (DosCallsHandle, OrdDosAllocThreadLocalMemory,                                                                       nil, P);      if RC = 0 then       begin        DosFreeThreadLocalMemory := TDosFreeThreadLocalMemory (P);        TLSAPISupported := true;       end      else       OSErrorWatch (RC);     end    else     OSErrorWatch (RC);   end  else   OSErrorWatch (RC);  { ... and exceptions }  SysInitExceptions;  fpc_cpucodeinit;  InitUnicodeStringManager;{$IFDEF OS2UNICODE}  InitOS2WideStringManager;  InitDefaultCP;{$ELSE OS2UNICODE}(* Otherwise called within InitDefaultCP... *)  RC := DosQueryCP (SizeOf (CPArr), @CPArr, ReturnedSize);  if (RC <> 0) and (RC <> 473) then   begin    OSErrorWatch (RC);    CPArr [0] := 850;   end  else if (ReturnedSize < 4) then   CPArr [0] := 850;  DefaultFileSystemCodePage := CPArr [0];{$ENDIF OS2UNICODE}  DefaultSystemCodePage := DefaultFileSystemCodePage;  DefaultRTLFileSystemCodePage := DefaultFileSystemCodePage;  DefaultUnicodeCodePage := CP_UTF16;  { ... and I/O }  SysInitStdIO;  { no I/O-Error }  InOutRes:=0;  {Initialize environment (must be after InitHeap because allocates memory)}  Environment := pointer (PIB^.Env);  InitEnvironment;  InitArguments;  DefaultCreator := '';  DefaultFileType := '';  InitSystemThreads;  InitSystemDynLibs;{$IFDEF EXTDUMPGROW}{    Int_HeapSize := high (cardinal);}{$ENDIF EXTDUMPGROW}{$ifdef SYSTEMEXCEPTIONDEBUG}  if IsConsole then   WriteLn (StdErr, 'Old exception ', HexStr (OldExceptAddr, 8),   ', new exception ', HexStr (NewExceptAddr, 8), ', _SS = ', HexStr (_SS, 8));{$endif SYSTEMEXCEPTIONDEBUG}end.
 |