| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349 | {    This file is part of the Free Pascal run time library.    Copyright (c) 1999-2000 by Florian Klaempfl    member of the Free Pascal development team    Sysutils unit for EMX    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 sysutils;interface{$MODE objfpc}{$MODESWITCH OUT}{$IFDEF UNICODERTL}{$MODESWITCH UNICODESTRINGS}{$ELSE}{$H+}{$ENDIF}{$modeswitch typehelpers}{$modeswitch advancedrecords}uses Dos;{$DEFINE HAS_SLEEP}{ used OS file system APIs use ansistring }{$define SYSUTILS_HAS_ANSISTR_FILEUTIL_IMPL}{ OS has an ansistring/single byte environment variable API }{$define SYSUTILS_HAS_ANSISTR_ENVVAR_IMPL}{$DEFINE executeprocuni} (* Only 1 byte version of ExecuteProcess is provided by the OS *){ Include platform independent interface part }{$i sysutilh.inc}implementation  uses    sysconst;{$DEFINE FPC_FEXPAND_UNC} (* UNC paths are supported *){$DEFINE FPC_FEXPAND_DRIVES} (* Full paths begin with drive specification *){ Include platform independent implementation part }{$i sysutils.inc}{****************************************************************************                        System (imported) calls****************************************************************************}(* "uses DosCalls" could not be used here due to type    *)(* conflicts, so needed parts had to be redefined here). *)type        TFileStatus = object        end;        PFileStatus = ^TFileStatus;        TFileStatus3 = object (TFileStatus)            DateCreation,               {Date of file creation.}            TimeCreation,               {Time of file creation.}            DateLastAccess,             {Date of last access to file.}            TimeLastAccess,             {Time of last access to file.}            DateLastWrite,              {Date of last modification of file.}            TimeLastWrite:word;         {Time of last modification of file.}            FileSize,                   {Size of file.}            FileAlloc:cardinal;         {Amount of space the file really                                         occupies on disk.}            AttrFile:cardinal;          {Attributes of file.}        end;        PFileStatus3=^TFileStatus3;        TFileStatus4=object(TFileStatus3)            cbList:cardinal;            {Length of entire EA set.}        end;        PFileStatus4=^TFileStatus4;        TFileStatus3L = object (TFileStatus)            DateCreation,               {Date of file creation.}            TimeCreation,               {Time of file creation.}            DateLastAccess,             {Date of last access to file.}            TimeLastAccess,             {Time of last access to file.}            DateLastWrite,              {Date of last modification of file.}            TimeLastWrite:word;         {Time of last modification of file.}            FileSize,                   {Size of file.}            FileAlloc:int64;         {Amount of space the file really                                         occupies on disk.}            AttrFile:cardinal;          {Attributes of file.}        end;        PFileStatus3L=^TFileStatus3L;        TFileStatus4L=object(TFileStatus3L)            cbList:cardinal;            {Length of entire EA set.}        end;        PFileStatus4L=^TFileStatus4L;        TFileFindBuf3=object(TFileStatus)            NextEntryOffset: cardinal;  {Offset of next entry}            DateCreation,               {Date of file creation.}            TimeCreation,               {Time of file creation.}            DateLastAccess,             {Date of last access to file.}            TimeLastAccess,             {Time of last access to file.}            DateLastWrite,              {Date of last modification of file.}            TimeLastWrite:word;         {Time of last modification of file.}            FileSize,                   {Size of file.}            FileAlloc:cardinal;         {Amount of space the file really                                         occupies on disk.}            AttrFile:cardinal;          {Attributes of file.}            Name:shortstring;           {Also possible to use as ASCIIZ.                                         The byte following the last string                                         character is always zero.}        end;        PFileFindBuf3=^TFileFindBuf3;        TFileFindBuf4=object(TFileStatus)            NextEntryOffset: cardinal;  {Offset of next entry}            DateCreation,               {Date of file creation.}            TimeCreation,               {Time of file creation.}            DateLastAccess,             {Date of last access to file.}            TimeLastAccess,             {Time of last access to file.}            DateLastWrite,              {Date of last modification of file.}            TimeLastWrite:word;         {Time of last modification of file.}            FileSize,                   {Size of file.}            FileAlloc:cardinal;         {Amount of space the file really                                         occupies on disk.}            AttrFile:cardinal;          {Attributes of file.}            cbList:longint;             {Size of the file's extended attributes.}            Name:shortstring;           {Also possible to use as ASCIIZ.                                         The byte following the last string                                         character is always zero.}        end;        PFileFindBuf4=^TFileFindBuf4;        TFileFindBuf3L=object(TFileStatus)            NextEntryOffset: cardinal;  {Offset of next entry}            DateCreation,               {Date of file creation.}            TimeCreation,               {Time of file creation.}            DateLastAccess,             {Date of last access to file.}            TimeLastAccess,             {Time of last access to file.}            DateLastWrite,              {Date of last modification of file.}            TimeLastWrite:word;         {Time of last modification of file.}            FileSize,                   {Size of file.}            FileAlloc:int64;            {Amount of space the file really                                         occupies on disk.}            AttrFile:cardinal;          {Attributes of file.}            Name:shortstring;           {Also possible to use as ASCIIZ.                                         The byte following the last string                                         character is always zero.}        end;        PFileFindBuf3L=^TFileFindBuf3L;        TFileFindBuf4L=object(TFileStatus)            NextEntryOffset: cardinal;  {Offset of next entry}            DateCreation,               {Date of file creation.}            TimeCreation,               {Time of file creation.}            DateLastAccess,             {Date of last access to file.}            TimeLastAccess,             {Time of last access to file.}            DateLastWrite,              {Date of last modification of file.}            TimeLastWrite:word;         {Time of last modification of file.}            FileSize,                   {Size of file.}            FileAlloc:int64;            {Amount of space the file really                                         occupies on disk.}            AttrFile:cardinal;          {Attributes of file.}            cbList:cardinal;            {Size of the file's extended attributes.}            Name:shortstring;           {Also possible to use as ASCIIZ.                                         The byte following the last string                                         character is always zero.}        end;        PFileFindBuf4L=^TFileFindBuf4L; TFSInfo = record            case word of             1:              (File_Sys_ID,               Sectors_Per_Cluster,               Total_Clusters,               Free_Clusters: cardinal;               Bytes_Per_Sector: word);             2:                           {For date/time description,                                           see file searching realted                                           routines.}              (Label_Date,                {Date when volume label was created.}               Label_Time: word;          {Time when volume label was created.}               VolumeLabel: ShortString); {Volume label. Can also be used                                           as ASCIIZ, because the byte                                           following the last character of                                           the string is always zero.}           end; PFSInfo = ^TFSInfo; TCountryCode=record               Country,            {Country to query info about (0=current).}               CodePage: cardinal; {Code page to query info about (0=current).}              end; PCountryCode=^TCountryCode; TTimeFmt = (Clock12, Clock24); TCountryInfo=record               Country, CodePage: cardinal;  {Country and codepage requested.}               case byte of                0:                 (DateFormat: cardinal;     {1=ddmmyy 2=yymmdd 3=mmddyy}                  CurrencyUnit: array [0..4] of AnsiChar;                  ThousandSeparator: AnsiChar;  {Thousands separator.}                  Zero1: byte;              {Always zero.}                  DecimalSeparator: AnsiChar;   {Decimals separator,}                  Zero2: byte;                  DateSeparator: AnsiChar;      {Date separator.}                  Zero3: byte;                  TimeSeparator: AnsiChar;      {Time separator.}                  Zero4: byte;                  CurrencyFormat,           {Bit field:                                             Bit 0: 0=indicator before value                                                    1=indicator after value                                             Bit 1: 1=insert space after                                                      indicator.                                             Bit 2: 1=Ignore bit 0&1, replace                                                      decimal separator with                                                      indicator.}                  DecimalPlace: byte;       {Number of decimal places used in                                             currency indication.}                  TimeFormat: TTimeFmt;     {12/24 hour.}                  Reserve1: array [0..1] of word;                  DataSeparator: AnsiChar;      {Data list separator}                  Zero5: byte;                  Reserve2: array [0..4] of word);                1:                 (fsDateFmt: cardinal;      {1=ddmmyy 2=yymmdd 3=mmddyy}                  szCurrency: array [0..4] of AnsiChar;                                            {null terminated currency symbol}                  szThousandsSeparator: array [0..1] of AnsiChar;                                            {Thousands separator + #0}                  szDecimal: array [0..1] of AnsiChar;                                            {Decimals separator + #0}                  szDateSeparator: array [0..1] of AnsiChar;                                            {Date separator + #0}                  szTimeSeparator: array [0..1] of AnsiChar;                                            {Time separator + #0}                  fsCurrencyFmt,            {Bit field:                                             Bit 0: 0=indicator before value                                                    1=indicator after value                                             Bit 1: 1=insert space after                                                      indicator.                                             Bit 2: 1=Ignore bit 0&1, replace                                                      decimal separator with                                                      indicator}                  cDecimalPlace: byte;      {Number of decimal places used in                                             currency indication}                  fsTimeFmt: byte;          {0=12,1=24 hours}                  abReserved1: array [0..1] of word;                  szDataSeparator: array [0..1] of AnsiChar;                                            {Data list separator + #0}                  abReserved2: array [0..4] of word);              end; PCountryInfo=^TCountryInfo; TRequestData=record               PID,                {ID of process that wrote element.}               Data: cardinal;     {Information from process writing the data.}              end; PRequestData=^TRequestData;{Queue data structure for synchronously started sessions.} TChildInfo = record  case boolean of   false:    (SessionID,     Return: word);  {Return code from the child process.}   true:    (usSessionID,     usReturn: word);     {Return code from the child process.} end; PChildInfo = ^TChildInfo; TStartData=record  {Note: to omit some fields, use a length smaller than SizeOf(TStartData).}  Length:word;                {Length, in bytes, of datastructure                               (24/30/32/50/60).}  Related:word;               {Independent/child session (0/1).}  FgBg:word;                  {Foreground/background (0/1).}  TraceOpt:word;              {No trace/trace this/trace all (0/1/2).}  PgmTitle:PAnsiChar;             {Program title.}  PgmName:PAnsiChar;              {Filename to program.}  PgmInputs:PAnsiChar;            {Command parameters (nil allowed).}  TermQ:PAnsiChar;                {System queue. (nil allowed).}  Environment:PAnsiChar;          {Environment to pass (nil allowed).}  InheritOpt:word;            {Inherit environment from shell/                               inherit environment from parent (0/1).}  SessionType:word;           {Auto/full screen/window/presentation                               manager/full screen Dos/windowed Dos                               (0/1/2/3/4/5/6/7).}  Iconfile:PAnsiChar;             {Icon file to use (nil allowed).}  PgmHandle:cardinal;         {0 or the program handle.}  PgmControl:word;            {Bitfield describing initial state                               of windowed sessions.}  InitXPos,InitYPos:word;     {Initial top coordinates.}  InitXSize,InitYSize:word;   {Initial size.}  Reserved:word;  ObjectBuffer:PAnsiChar;         {If a module cannot be loaded, its                               name will be returned here.}  ObjectBuffLen:cardinal;     {Size of your buffer.} end; PStartData=^TStartData;const ilStandard      =  1; (* Use TFileStatus3/TFindFileBuf3 *) ilQueryEASize   =  2; (* Use TFileStatus4/TFindFileBuf4 *) ilQueryEAs      =  3; ilQueryFullName =  5; ilStandardL     = 11; (* Use TFileStatus3L/TFindFileBuf3L *) ilQueryEASizeL  = 12; (* Use TFileStatus4L/TFindFileBuf4L *) ilQueryEAsL     = 13; quFIFO     = 0; quLIFO     = 1; quPriority = 2; quNoConvert_Address = 0; quConvert_Address   = 4;{Start the new session independent or as a child.} ssf_Related_Independent = 0;    {Start new session independent                                  of the calling session.} ssf_Related_Child       = 1;    {Start new session as a child                                  session to the calling session.}{Start the new session in the foreground or in the background.} ssf_FgBg_Fore           = 0;    {Start new session in foreground.} ssf_FgBg_Back           = 1;    {Start new session in background.}{Should the program started in the new session be executed under conditions for tracing?} ssf_TraceOpt_None       = 0;    {No trace.} ssf_TraceOpt_Trace      = 1;    {Trace with no notification                                  of descendants.} ssf_TraceOpt_TraceAll   = 2;    {Trace all descendant sessions.                                  A termination queue must be                                  supplied and Related must be                                  ssf_Related_Child (=1).}{Will the new session inherit open file handles and environment from the calling process.} ssf_InhertOpt_Shell     = 0;    {Inherit from the shell.} ssf_InhertOpt_Parent    = 1;    {Inherit from the calling process.}{Specifies the type of session to start.} ssf_Type_Default        = 0;    {Use program's type.} ssf_Type_FullScreen     = 1;    {OS/2 full screen.} ssf_Type_WindowableVIO  = 2;    {OS/2 window.} ssf_Type_PM             = 3;    {Presentation Manager.} ssf_Type_VDM            = 4;    {DOS full screen.} ssf_Type_WindowedVDM    = 7;    {DOS window.}{Additional values for Windows programs} Prog_31_StdSeamlessVDM    = 15; {Windows 3.1 program in its                                  own windowed session.} Prog_31_StdSeamlessCommon = 16; {Windows 3.1 program in a                                  common windowed session.} Prog_31_EnhSeamlessVDM    = 17; {Windows 3.1 program in enhanced                                  compatibility mode in its own                                  windowed session.} Prog_31_EnhSeamlessCommon = 18; {Windows 3.1 program in enhanced                                  compatibility mode in a common                                  windowed session.} Prog_31_Enh               = 19; {Windows 3.1 program in enhanced                                  compatibility mode in a full                                  screen session.} Prog_31_Std               = 20; {Windows 3.1 program in a full                                  screen session.}{Specifies the initial attributes for a OS/2 window or DOS window session.} ssf_Control_Visible      = 0;   {Window is visible.} ssf_Control_Invisible    = 1;   {Window is invisible.} ssf_Control_Maximize     = 2;   {Window is maximized.} ssf_Control_Minimize     = 4;   {Window is minimized.} ssf_Control_NoAutoClose  = 8;   {Window will not close after                                  the program has ended.} ssf_Control_SetPos   = 32768;   {Use InitXPos, InitYPos,                                  InitXSize, and InitYSize for                                  the size and placement.}{This is the correct way to call external assembler procedures.}procedure syscall;external name '___SYSCALL';function DosSetFileInfo (Handle: THandle; InfoLevel: cardinal; AFileStatus: PFileStatus;        FileStatusLen: cardinal): cardinal; cdecl; external 'DOSCALLS' index 218;function DosQueryFSInfo (DiskNum, InfoLevel: cardinal; var Buffer: TFSInfo;               BufLen: cardinal): cardinal; cdecl; external 'DOSCALLS' index 278;function DosQueryFileInfo (Handle: THandle; InfoLevel: cardinal;           AFileStatus: PFileStatus; FileStatusLen: cardinal): cardinal; cdecl;                                                 external 'DOSCALLS' index 279;function DosScanEnv (Name: PAnsiChar; var Value: PAnsiChar): cardinal; cdecl;                                                 external 'DOSCALLS' index 227;function DosFindFirst (FileMask: PAnsiChar; var Handle: THandle; Attrib: cardinal;                       AFileStatus: PFileStatus; FileStatusLen: cardinal;                    var Count: cardinal; InfoLevel: cardinal): cardinal; cdecl;                                                 external 'DOSCALLS' index 264;function DosFindNext (Handle: THandle; AFileStatus: PFileStatus;                FileStatusLen: cardinal; var Count: cardinal): cardinal; cdecl;                                                 external 'DOSCALLS' index 265;function DosFindClose (Handle: THandle): cardinal; cdecl;                                                 external 'DOSCALLS' index 263;function DosQueryCtryInfo (Size: cardinal; var Country: TCountryCode;           var Res: TCountryInfo; var ActualSize: cardinal): cardinal; cdecl;                                                        external 'NLS' index 5;function DosMapCase (Size: cardinal; var Country: TCountryCode;                      AString: PAnsiChar): cardinal; cdecl; external 'NLS' index 7;procedure DosSleep (MSec: cardinal); cdecl; external 'DOSCALLS' index 229;function DosCreateQueue (var Handle: THandle; Priority:longint;                        Name: PAnsiChar): cardinal; cdecl;                                                  external 'QUECALLS' index 16;function DosReadQueue (Handle: THandle; var ReqBuffer: TRequestData;                      var DataLen: cardinal; var DataPtr: pointer;                      Element, Wait: cardinal; var Priority: byte;                      ASem: THandle): cardinal; cdecl;                                                   external 'QUECALLS' index 9;function DosCloseQueue (Handle: THandle): cardinal; cdecl;                                                  external 'QUECALLS' index 11;function DosStartSession (var AStartData: TStartData;                          var SesID, PID: cardinal): cardinal; cdecl;                                                    external 'SESMGR' index 37;function DosFreeMem(P:pointer):cardinal; cdecl; external 'DOSCALLS' index 304;{****************************************************************************                              File Functions****************************************************************************}const ofRead        = $0000;     {Open for reading} ofWrite       = $0001;     {Open for writing} ofReadWrite   = $0002;     {Open for reading/writing} doDenyRW      = $0010;     {DenyAll (no sharing)} faCreateNew   = $00010000; {Create if file does not exist} faOpenReplace = $00040000; {Truncate if file exists} faCreate      = $00050000; {Create if file does not exist, truncate otherwise} FindResvdMask = $00003737; {Allowed bits in attribute                             specification for DosFindFirst call.}{$ASMMODE INTEL}function FileOpen (const FileName: pointer; Mode: integer): longint; assembler;asm push ebx{$IFDEF REGCALL} mov ecx, edx mov edx, eax{$ELSE REGCALL} mov ecx, Mode mov edx, FileName{$ENDIF REGCALL}(* DenyNone if sharing not specified. *) mov eax, ecx xor eax, 112 jz @FOpenDefSharing cmp eax, 64 jbe @FOpen1@FOpenDefSharing: or ecx, 64@FOpen1: mov eax, 7F2Bh call syscall(* syscall __open() returns -1 in case of error, i.e. exactly what we need *) pop ebxend {['eax', 'ebx', 'ecx', 'edx']};function FileOpen (const FileName: rawbytestring; Mode: integer): longint;var  SystemFileName: RawByteString;begin  SystemFileName := ToSingleByteFileSystemEncodedFileName(FileName);  FileOpen := FileOpen(pointer(SystemFileName),Mode);end;function FileCreate (const FileName: RawByteString): longint;begin  FileCreate := FileCreate (FileName, ofReadWrite or faCreate or doDenyRW, 777);                                                       (* Sharing to DenyAll *)end;function FileCreate (const FileName: RawByteString; Rights: integer): longint;begin  FileCreate := FileCreate (FileName, ofReadWrite or faCreate or doDenyRW,                                              Rights); (* Sharing to DenyAll *)end;function FileCreate (const FileName: Pointer; ShareMode: integer; Rights: integer): longint; assembler;asm push ebx{$IFDEF REGCALL} mov ecx, edx mov edx, eax{$ELSE REGCALL} mov ecx, ShareMode mov edx, FileName{$ENDIF REGCALL} and ecx, 112 or ecx, ecx jz @FCDefSharing cmp ecx, 64 jbe @FCSharingOK@FCDefSharing: mov ecx, doDenyRW   (* Sharing to DenyAll *)@FCSharingOK: or ecx, ofReadWrite or faCreate mov eax, 7F2Bh call syscall pop ebxend {['eax', 'ebx', 'ecx', 'edx']};function FileCreate (const FileName: RawByteString; ShareMode: integer; Rights: integer): longint;var  SystemFileName: RawByteString;begin  SystemFileName := ToSingleByteFileSystemEncodedFileName(FileName);  FileCreate := FileCreate(pointer(SystemFileName),ShareMode,Rights);end;function FileRead (Handle: longint; Out Buffer; Count: longint): longint;                                                                     assembler;asm push ebx{$IFDEF REGCALL} mov ebx, eax{$ELSE REGCALL} mov ebx, Handle mov ecx, Count mov edx, Buffer{$ENDIF REGCALL} mov eax, 3F00h call syscall jnc @FReadEnd mov eax, -1@FReadEnd: pop ebxend {['eax', 'ebx', 'ecx', 'edx']};function FileWrite (Handle: longint; const Buffer; Count: longint): longint;                                                                     assembler;asm push ebx{$IFDEF REGCALL} mov ebx, eax{$ELSE REGCALL} mov ebx, Handle mov ecx, Count mov edx, Buffer{$ENDIF REGCALL} mov eax, 4000h call syscall jnc @FWriteEnd mov eax, -1@FWriteEnd: pop ebxend {['eax', 'ebx', 'ecx', 'edx']};function FileSeek (Handle, FOffset, Origin: longint): longint; assembler;asm push ebx{$IFDEF REGCALL} mov ebx, eax mov eax, ecx{$ELSE REGCALL} mov ebx, Handle mov eax, Origin mov edx, FOffset{$ENDIF REGCALL} mov ah, 42h call syscall jnc @FSeekEnd mov eax, -1@FSeekEnd: pop ebxend {['eax', 'ebx', 'edx']};function FileSeek (Handle: longint; FOffset: Int64; Origin: longint): Int64;begin  {$warning need to add 64bit call }  Result:=FileSeek(Handle,Longint(Foffset),Longint(Origin));end;procedure FileClose (Handle: longint);begin    if (Handle > 4) or ((os_mode = osOS2) and (Handle > 2)) then        asm            push ebx            mov eax, 3E00h            mov ebx, Handle            call syscall            pop ebx        end ['eax'];end;function FileTruncate (Handle: THandle; Size: Int64): boolean; assembler;asm push ebx{$IFDEF REGCALL} mov ebx, eax{$ELSE REGCALL} mov ebx, Handle{$ENDIF REGCALL} mov edx, dword ptr Size mov eax, dword ptr Size+4 or eax, eax mov eax, 0 jz @FTruncEnd  (* file sizes > 4 GB not supported with EMX *) mov eax, 7F25h push ebx call syscall pop ebx jc @FTruncEnd mov eax, 4202h mov edx, 0 call syscall mov eax, 0 jnc @FTruncEnd dec eax@FTruncEnd: pop ebxend {['eax', 'ebx', 'ecx', 'edx']};function FileAge (const FileName: RawByteString): Int64;var Handle: longint;begin    Handle := FileOpen (FileName, 0);    if Handle <> -1 then        begin            Result := FileGetDate (Handle);            FileClose (Handle);        end    else        Result := -1;end;function FileGetSymLinkTarget(const FileName: RawByteString; out SymLinkRec: TRawbyteSymLinkRec): Boolean;begin  Result := False;end;function FileExists (const FileName: RawByteString; FollowLink : Boolean): boolean;var  L: longint;begin  { no need to convert to DefaultFileSystemEncoding, FileGetAttr will do that }  if FileName = '' then   Result := false  else   begin    L := FileGetAttr (FileName);    Result := (L >= 0) and (L and (faDirectory or faVolumeID) = 0);(* Neither VolumeIDs nor directories are files. *)   end;end;type  PSearchRec = ^SearchRec;Function InternalFindFirst (Const Path : RawByteString; Attr : Longint; out Rslt : TAbstractSearchRec; var Name: RawByteString) : Longint;var  SystemEncodedPath: RawByteString;  SR: PSearchRec;  FStat: PFileFindBuf3L;  Count: cardinal;  Err: cardinal;begin  if os_mode = osOS2 then   begin    SystemEncodedPath:=ToSingleByteFileSystemEncodedFileName(Path);    New (FStat);    Rslt.FindHandle := THandle ($FFFFFFFF);    Count := 1;    if FSApi64 then     Err := DosFindFirst (PAnsiChar (SystemEncodedPath), Rslt.FindHandle,            Attr and FindResvdMask, FStat, SizeOf (FStat^), Count, ilStandardL)    else     Err := DosFindFirst (PAnsiChar (SystemEncodedPath), Rslt.FindHandle,            Attr and FindResvdMask, FStat, SizeOf (FStat^), Count, ilStandard);    if (Err = 0) and (Count = 0) then     Err := 18;    InternalFindFirst := -Err;    if Err = 0 then     begin      Rslt.ExcludeAttr := 0;      Rslt.Time := cardinal (FStat^.DateLastWrite) shl 16 +                                                          FStat^.TimeLastWrite;      if FSApi64 then       begin        Rslt.Size := FStat^.FileSize;        Name := FStat^.Name;        Rslt.Attr := FStat^.AttrFile;       end      else       begin        Rslt.Size := PFileFindBuf3 (FStat)^.FileSize;        Name := PFileFindBuf3 (FStat)^.Name;        Rslt.Attr := PFileFindBuf3 (FStat)^.AttrFile;       end;      SetCodePage(Name, DefaultFileSystemCodePage, false);     end    else     InternalFindClose (Rslt.FindHandle);    Dispose (FStat);   end  else   begin    Err := DOS.DosError;    GetMem (SR, SizeOf (SearchRec));    Rslt.FindHandle := longint(SR);    DOS.FindFirst (Path, Attr, SR^);    InternalFindFirst := -DOS.DosError;    if DosError = 0 then     begin      Rslt.Time := SR^.Time;(* Extend the supported file sizes from 2 GB to 4 GB at least. *)      Rslt.Size := cardinal (SR^.Size);      Rslt.Attr := SR^.Attr;      Rslt.ExcludeAttr := 0;      Name := SR^.Name;      SetCodePage(Name, DefaultFileSystemCodePage, false);     end;    DOS.DosError := Err;   end;end;Function InternalFindNext (var Rslt : TAbstractSearchRec; var Name : RawByteString) : Longint;var  SR: PSearchRec;  FStat: PFileFindBuf3L;  Count: cardinal;  Err: cardinal;begin  if os_mode = osOS2 then   begin    New (FStat);    Count := 1;    Err := DosFindNext (Rslt.FindHandle, FStat, SizeOf (FStat^), Count);    if (Err = 0) and (Count = 0) then     Err := 18;    InternalFindNext := -Err;    if Err = 0 then     begin      Rslt.ExcludeAttr := 0;      Rslt.Time := cardinal (FStat^.DateLastWrite) shl 16 +                                                          FStat^.TimeLastWrite;      if FSApi64 then       begin        Rslt.Size := FStat^.FileSize;        Name := FStat^.Name;        Rslt.Attr := FStat^.AttrFile;       end      else       begin        Rslt.Size := PFileFindBuf3 (FStat)^.FileSize;        Name := PFileFindBuf3 (FStat)^.Name;        Rslt.Attr := PFileFindBuf3 (FStat)^.AttrFile;       end;      SetCodePage(Name, DefaultFileSystemCodePage, false);     end;    Dispose (FStat);   end  else   begin    SR := PSearchRec (Rslt.FindHandle);    if SR <> nil then     begin      DOS.FindNext (SR^);      InternalFindNext := -DosError;      if DosError = 0 then       begin        Rslt.Time := SR^.Time;(* Extend the supported file sizes from 2 GB to 4 GB at least. *)        Rslt.Size := cardinal (SR^.Size);        Rslt.Attr := SR^.Attr;        Rslt.ExcludeAttr := 0;        Name := SR^.Name;        SetCodePage(Name, DefaultFileSystemCodePage, false);       end;     end;   end;end;Procedure InternalFindClose(var Handle: THandle);var SR: PSearchRec;begin    if os_mode = osOS2 then        begin            DosFindClose (Handle);        end    else        begin            SR := PSearchRec (Handle);            DOS.FindClose (SR^);            FreeMem (SR, SizeOf (SearchRec));        end;    Handle := 0;end;function FileGetDate (Handle: longint): Int64; assembler;asm push ebx{$IFDEF REGCALL} mov ebx, eax{$ELSE REGCALL} mov ebx, Handle{$ENDIF REGCALL} mov ax, 5700h call syscall mov eax, -1 jc @FGetDateEnd mov ax, dx shld eax, ecx, 16@FGetDateEnd: pop ebx xor edx,edxend {['eax', 'ebx', 'ecx', 'edx']};function FileSetDate (Handle: longint; Age: Int64): longint;var FStat: PFileStatus3;    RC: cardinal;begin    if os_mode = osOS2 thenbegin            New (FStat);            RC := DosQueryFileInfo (Handle, ilStandard, FStat,                                                              SizeOf (FStat^));            if RC <> 0 then                FileSetDate := -1            else                begin                    FStat^.DateLastAccess := Hi (dword (Age));                    FStat^.DateLastWrite := Hi (dword (Age));                    FStat^.TimeLastAccess := Lo (dword (Age));                    FStat^.TimeLastWrite := Lo (dword (Age));                    RC := DosSetFileInfo (Handle, ilStandard, FStat,                                                              SizeOf (FStat^));                    if RC <> 0 then                        FileSetDate := -1                    else                        FileSetDate := 0;                end;            Dispose (FStat);        end    else        asm            push ebx            mov ax, 5701h            mov ebx, Handle            mov cx, word ptr [Age]            mov dx, word ptr [Age + 2]            call syscall            jnc @FSetDateEnd            mov eax, -1@FSetDateEnd:            mov Result, eax            pop ebx        end ['eax', 'ecx', 'edx'];end;function FileGetAttr (const FileName: rawbytestring): longint; assembler;asm{$IFDEF REGCALL} mov edx, eax{$ELSE REGCALL} mov edx, FileName{$ENDIF REGCALL} mov ax, 4300h call syscall jnc @FGetAttrEnd mov eax, -1@FGetAttrEnd:end {['eax', 'edx']};function FileSetAttr (const Filename: RawByteString; Attr: longint): longint;var  SystemFileName: RawByteString;begin  SystemFileName:=ToSingleByteFileSystemEncodedFileName(Filename);  asm   mov ecx, Attr   mov edx, SystemFileName   mov ax, 4301h   call syscall   mov @result, 0   jnc @FSetAttrEnd   mov @result, -1  @FSetAttrEnd:  end ['eax', 'ecx', 'edx'];end;function DeleteFile (const FileName: rawbytestring): boolean;var  SystemFileName: RawByteString;begin  SystemFileName:=ToSingleByteFileSystemEncodedFileName(Filename);  asm   mov edx, SystemFileName   mov ax, 4100h   call syscall   mov @result, 0   jc @FDeleteEnd   mov @result, 1  @FDeleteEnd:  end ['eax', 'edx'];end;function RenameFile (const OldName, NewName: rawbytestring): boolean;var  OldSystemFileName, NewSystemFileName: RawByteString;Begin  OldSystemFileName:=ToSingleByteFileSystemEncodedFileName(OldName);  NewSystemFileName:=ToSingleByteFileSystemEncodedFileName(NewName);  asm   mov edx, OldSystemFileName   mov edi, NewSystemFileName   mov ax, 5600h   call syscall   mov @result, 0   jc @FRenameEnd   mov @result, 1  @FRenameEnd:  end ['eax', 'edx', 'edi'];end;{****************************************************************************                              Disk Functions****************************************************************************}{$ASMMODE ATT}function DiskFree (Drive: byte): int64;var FI: TFSinfo;    RC: cardinal;begin    if (os_mode = osDOS) or (os_mode = osDPMI) then    {Function 36 is not supported in OS/2.}        asm            pushl %ebx            movb Drive,%dl            movb $0x36,%ah            call syscall            cmpw $-1,%ax            je .LDISKFREE1            mulw %cx            mulw %bx            shll $16,%edx            movw %ax,%dx            movl $0,%eax            xchgl %edx,%eax            jmp .LDISKFREE2         .LDISKFREE1:            cltd         .LDISKFREE2:            popl %ebx            leave            ret        end    else        {In OS/2, we use the filesystem information.}        begin            RC := DosQueryFSInfo (Drive, 1, FI, SizeOf (FI));            if RC = 0 then                DiskFree := int64 (FI.Free_Clusters) *                   int64 (FI.Sectors_Per_Cluster) * int64 (FI.Bytes_Per_Sector)            else                DiskFree := -1;        end;end;function DiskSize (Drive: byte): int64;var FI: TFSinfo;    RC: cardinal;begin    if (os_mode = osDOS) or (os_mode = osDPMI) then        {Function 36 is not supported in OS/2.}        asm            pushl %ebx            movb Drive,%dl            movb $0x36,%ah            call syscall            movw %dx,%bx            cmpw $-1,%ax            je .LDISKSIZE1            mulw %cx            mulw %bx            shll $16,%edx            movw %ax,%dx            movl $0,%eax            xchgl %edx,%eax            jmp .LDISKSIZE2         .LDISKSIZE1:            cltd         .LDISKSIZE2:            popl %ebx            leave            ret        end    else        {In OS/2, we use the filesystem information.}        begin            RC := DosQueryFSinfo (Drive, 1, FI, SizeOf (FI));            if RC = 0 then                DiskSize := int64 (FI.Total_Clusters) *                   int64 (FI.Sectors_Per_Cluster) * int64 (FI.Bytes_Per_Sector)            else                DiskSize := -1;        end;end;function DirectoryExists (const Directory: RawByteString; FollowLink : Boolean): boolean;var  L: longint;begin  { no need to convert to DefaultFileSystemEncoding, FileGetAttr will do that }  if Directory = '' then   Result := false  else   begin    if ((Length (Directory) = 2) or        (Length (Directory) = 3) and        (Directory [3] in AllowDirectorySeparators)) and       (Directory [2] in AllowDriveSeparators) and       (UpCase (Directory [1]) in ['A'..'Z']) then(* Checking attributes for 'x:' is not possible but for 'x:.' it is. *)     L := FileGetAttr (Directory + '.')    else if (Directory [Length (Directory)] in AllowDirectorySeparators) and                                              (Length (Directory) > 1) and(* Do not remove '\' in '\\' (invalid path, possibly broken UNC path). *)      not (Directory [Length (Directory) - 1] in AllowDirectorySeparators) then     L := FileGetAttr (Copy (Directory, 1, Length (Directory) - 1))    else     L := FileGetAttr (Directory);    Result := (L > 0) and (L and faDirectory = faDirectory);   end;end;{****************************************************************************                              Time Functions****************************************************************************}{$ASMMODE INTEL}procedure GetLocalTime (var SystemTime: TSystemTime); assembler;asm(* Expects the default record alignment (word)!!! *) push edi{$IFDEF REGCALL} push eax{$ENDIF REGCALL} mov ah, 2Ah call syscall{$IFDEF REGCALL} pop edi{$ELSE REGCALL} mov edi, SystemTime{$ENDIF REGCALL} xchg ax, cx shl eax, 16 mov al, dh stosd mov al, dl shl eax, 16 mov al, cl stosd push edi mov ah, 2Ch call syscall pop edi xor eax, eax mov al, ch shl eax, 16 mov al, cl stosd xor eax, eax mov al, 10 mul dl shl eax, 16 mov al, dh rol eax, 16 stosd pop ediend {['eax', 'ecx', 'edx', 'edi']};{$asmmode default}{****************************************************************************                              Misc Functions****************************************************************************}{****************************************************************************                              Locale Functions****************************************************************************}procedure InitAnsi;var I: byte;    Country: TCountryCode;begin    for I := 0 to 255 do        UpperCaseTable [I] := Chr (I);    Move (UpperCaseTable, LowerCaseTable, SizeOf (UpperCaseTable));    if os_mode = osOS2 then        begin            FillChar (Country, SizeOf (Country), 0);            DosMapCase (SizeOf (UpperCaseTable), Country, @UpperCaseTable);        end    else        begin(* !!! TODO: DOS/DPMI mode support!!! *)        end;    for I := 0 to 255 do        if UpperCaseTable [I] <> Chr (I) then            LowerCaseTable [Ord (UpperCaseTable [I])] := Chr (I);end;procedure InitInternational;var Country: TCountryCode;    CtryInfo: TCountryInfo;    Size: cardinal;    RC: cardinal;begin  Size := 0;  FillChar (Country, SizeOf (Country), 0);  FillChar (CtryInfo, SizeOf (CtryInfo), 0);  RC := DosQueryCtryInfo (SizeOf (CtryInfo), Country, CtryInfo, Size);  if RC = 0 then      begin          DateSeparator := CtryInfo.DateSeparator;          case CtryInfo.DateFormat of           1: begin                  ShortDateFormat := 'd/m/y';                  LongDateFormat := 'dd" "mmmm" "yyyy';              end;           2: begin                  ShortDateFormat := 'y/m/d';                  LongDateFormat := 'yyyy" "mmmm" "dd';              end;           3: begin                  ShortDateFormat := 'm/d/y';                  LongDateFormat := 'mmmm" "dd" "yyyy';              end;          end;          TimeSeparator := CtryInfo.TimeSeparator;          DecimalSeparator := CtryInfo.DecimalSeparator;          ThousandSeparator := CtryInfo.ThousandSeparator;          CurrencyFormat := CtryInfo.CurrencyFormat;          CurrencyString := PAnsiChar (CtryInfo.CurrencyUnit);      end;  InitAnsi;  InitInternationalGeneric;end;function SysErrorMessage(ErrorCode: Integer): String;begin  Result:=Format(SUnknownErrorCode,[ErrorCode]);end;{****************************************************************************                              OS Utils****************************************************************************}Function GetEnvironmentVariable(Const EnvVar : String) : String;begin    GetEnvironmentVariable := GetEnvPChar (EnvVar);end;Function GetEnvironmentVariableCount : Integer;begin(*  Result:=FPCCountEnvVar(EnvP); - the amount is already known... *)  GetEnvironmentVariableCount := EnvC;end;Function GetEnvironmentString(Index : Integer) : {$ifdef FPC_RTL_UNICODE}UnicodeString{$else}AnsiString{$endif};begin  Result:=FPCGetEnvStrFromP (EnvP, Index);end;{$ASMMODE INTEL}procedure Sleep (Milliseconds: cardinal);begin if os_mode = osOS2 then DosSleep (Milliseconds) else  asm   mov edx, Milliseconds   mov eax, 7F30h   call syscall  end ['eax', 'edx'];end;{$ASMMODE DEFAULT}function ExecuteProcess (const Path: RawByteString; const ComLine: RawByteString;Flags:TExecuteFlags=[]):                                                                       integer;var HQ: THandle; SPID, STID, QName: shortstring; SD: TStartData; SID, PID: cardinal; RD: TRequestData; PCI: PChildInfo; CISize: cardinal; Prio: byte; E: EOSError; CommandLine: rawbytestring;begin if os_Mode = osOS2 then  begin   FillChar (SD, SizeOf (SD), 0);   SD.Length := 24;   SD.Related := ssf_Related_Child;   SD.PgmName := PAnsiChar (Path);   SD.PgmInputs := PAnsiChar (ComLine);   Str (GetProcessID, SPID);   Str (ThreadID, STID);   QName := '\QUEUES\FPC_ExecuteProcess_p' + SPID + 't' + STID + '.QUE'#0;   SD.TermQ := @QName [1];   Result := DosCreateQueue (HQ, quFIFO or quConvert_Address, @QName [1]);   if Result = 0 then    begin     Result := DosStartSession (SD, SID, PID);     if (Result = 0) or (Result = 457) then      begin       Result := DosReadQueue (HQ, RD, CISize, PCI, 0, 0, Prio, 0);       if Result = 0 then       begin        Result := PCI^.Return;        DosCloseQueue (HQ);        DosFreeMem (PCI);        Exit;       end;      end;     DosCloseQueue (HQ);    end;    if ComLine = '' then     CommandLine := Path    else     CommandLine := Path + ' ' + ComLine;    E := EOSError.CreateFmt (SExecuteProcessFailed, [CommandLine, Result]);    E.ErrorCode := Result;    raise E;  end else  begin   Dos.Exec (Path, ComLine);   if DosError <> 0 then    begin    if ComLine = '' then     CommandLine := Path    else     CommandLine := Path + ' ' + ComLine;      E := EOSError.CreateFmt (SExecuteProcessFailed, [CommandLine, DosError]);      E.ErrorCode := DosError;      raise E;    end;   ExecuteProcess := DosExitCode;  end;end;function ExecuteProcess (const Path: RawByteString;                                  const ComLine: array of RawByteString;Flags:TExecuteFlags=[]): integer;var  CommandLine: RawByteString;  I: integer;begin  Commandline := '';  for I := 0 to High (ComLine) do   if Pos (' ', ComLine [I]) <> 0 then    CommandLine := CommandLine + ' ' + '"' + ComLine [I] + '"'   else    CommandLine := CommandLine + ' ' + Comline [I];  ExecuteProcess := ExecuteProcess (Path, CommandLine);end;{****************************************************************************                              Initialization code****************************************************************************}Initialization  InitExceptions;       { Initialize exceptions. OS independent }  InitInternational;    { Initialize internationalization settings }Finalization  FreeTerminateProcs;  DoneExceptions;end.
 |