| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229 | {    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}{ force ansistrings }{$H+}uses Dos;{$DEFINE HAS_SLEEP}{ 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;        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; 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 char;                  ThousandSeparator: char;  {Thousands separator.}                  Zero1: byte;              {Always zero.}                  DecimalSeparator: char;   {Decimals separator,}                  Zero2: byte;                  DateSeparator: char;      {Date separator.}                  Zero3: byte;                  TimeSeparator: char;      {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: char;      {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 char;                                            {null terminated currency symbol}                  szThousandsSeparator: array [0..1] of char;                                            {Thousands separator + #0}                  szDecimal: array [0..1] of char;                                            {Decimals separator + #0}                  szDateSeparator: array [0..1] of char;                                            {Date separator + #0}                  szTimeSeparator: array [0..1] of char;                                            {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 char;                                            {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:PChar;             {Program title.}  PgmName:PChar;              {Filename to program.}  PgmInputs:PChar;            {Command parameters (nil allowed).}  TermQ:PChar;                {System queue. (nil allowed).}  Environment:PChar;          {Environment to pass (nil allowed).}  InheritOpt:word;            {Inherit enviroment 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:PChar;             {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:PChar;         {If a module cannot be loaded, its                               name will be returned here.}  ObjectBuffLen:cardinal;     {Size of your buffer.} end; PStartData=^TStartData;const ilStandard      = 1; ilQueryEAsize   = 2; ilQueryEAs      = 3; ilQueryFullName = 5; 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: PChar; var Value: PChar): cardinal; cdecl;                                                 external 'DOSCALLS' index 227;function DosFindFirst (FileMask: PChar; 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: PChar): cardinal; cdecl; external 'NLS' index 7;procedure DosSleep (MSec: cardinal); cdecl; external 'DOSCALLS' index 229;function DosCreateQueue (var Handle: THandle; Priority:longint;                        Name: PChar): 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: string; 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}(* DenyAll if sharing not specified. *) test ecx, 112 jnz @FOpen1 or ecx, 16@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 FileCreate (const FileName: string): longint; assembler;asm push ebx{$IFDEF REGCALL} mov edx, eax{$ELSE REGCALL} mov edx, FileName{$ENDIF REGCALL} mov eax, 7F2Bh mov ecx, ofReadWrite or faCreate or doDenyRW   (* Sharing to DenyAll *) call syscall pop ebxend {['eax', 'ebx', 'ecx', 'edx']};function FileCreate (const FileName: string; Mode: integer): longint;begin  FileCreate:=FileCreate(FileName);end;function FileRead (Handle: longint; var 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: string): longint;var Handle: longint;begin    Handle := FileOpen (FileName, 0);    if Handle <> -1 then        begin            Result := FileGetDate (Handle);            FileClose (Handle);        end    else        Result := -1;end;function FileExists (const FileName: string): boolean; assembler;asm{$IFDEF REGCALL} mov edx, eax{$ELSE REGCALL} mov edx, FileName{$ENDIF REGCALL} mov ax, 4300h call syscall mov eax, 0 jc @FExistsEnd test cx, 18h jnz @FExistsEnd inc eax@FExistsEnd:end {['eax', 'ecx', 'edx']};type    TRec = record            T, D: word;        end;        PSearchRec = ^SearchRec;function FindFirst (const Path: string; Attr: longint; out Rslt: TSearchRec): longint;var SR: PSearchRec;    FStat: PFileFindBuf3;    Count: cardinal;    Err: cardinal;begin    if os_mode = osOS2 then        begin            New (FStat);            Rslt.FindHandle := THandle ($FFFFFFFF);            Count := 1;            Err := DosFindFirst (PChar (Path), Rslt.FindHandle,                   Attr and FindResvdMask, FStat, SizeOf (FStat^), Count,                                                                   ilStandard);            if (Err = 0) and (Count = 0) then Err := 18;            FindFirst := -Err;            if Err = 0 then                begin                    Rslt.Name := FStat^.Name;                    Rslt.Size := FStat^.FileSize;                    Rslt.Attr := FStat^.AttrFile;                    Rslt.ExcludeAttr := 0;                    TRec (Rslt.Time).T := FStat^.TimeLastWrite;                    TRec (Rslt.Time).D := FStat^.DateLastWrite;                end;            Dispose (FStat);        end    else        begin            Err := DOS.DosError;            GetMem (SR, SizeOf (SearchRec));            Rslt.FindHandle := longint(SR);            DOS.FindFirst (Path, Attr, SR^);            FindFirst := -DOS.DosError;            if DosError = 0 then                begin                    Rslt.Time := SR^.Time;                    Rslt.Size := SR^.Size;                    Rslt.Attr := SR^.Attr;                    Rslt.ExcludeAttr := 0;                    Rslt.Name := SR^.Name;                end;            DOS.DosError := Err;        end;end;function FindNext (var Rslt: TSearchRec): longint;var SR: PSearchRec;    FStat: PFileFindBuf3;    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;            FindNext := -Err;            if Err = 0 then                begin                    Rslt.Name := FStat^.Name;                    Rslt.Size := FStat^.FileSize;                    Rslt.Attr := FStat^.AttrFile;                    Rslt.ExcludeAttr := 0;                    TRec (Rslt.Time).T := FStat^.TimeLastWrite;                    TRec (Rslt.Time).D := FStat^.DateLastWrite;                end;            Dispose (FStat);        end    else        begin            SR := PSearchRec (Rslt.FindHandle);            if SR <> nil then                begin                    DOS.FindNext (SR^);                    FindNext := -DosError;                    if DosError = 0 then                        begin                            Rslt.Time := SR^.Time;                            Rslt.Size := SR^.Size;                            Rslt.Attr := SR^.Attr;                            Rslt.ExcludeAttr := 0;                            Rslt.Name := SR^.Name;                        end;                end;        end;end;procedure FindClose (var F: TSearchrec);var SR: PSearchRec;begin    if os_mode = osOS2 then        begin            DosFindClose (F.FindHandle);        end    else        begin            SR := PSearchRec (F.FindHandle);            DOS.FindClose (SR^);            FreeMem (SR, SizeOf (SearchRec));        end;    F.FindHandle := 0;end;function FileGetDate (Handle: longint): longint; 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 ebxend {['eax', 'ebx', 'ecx', 'edx']};function FileSetDate (Handle, Age: longint): 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 (Age);                    FStat^.DateLastWrite := Hi (Age);                    FStat^.TimeLastAccess := Lo (Age);                    FStat^.TimeLastWrite := Lo (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: string): 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: string; Attr: longint): longint; assembler;asm{$IFDEF REGCALL} mov ecx, edx mov edx, eax{$ELSE REGCALL} mov ecx, Attr mov edx, FileName{$ENDIF REGCALL} mov ax, 4301h call syscall mov eax, 0 jnc @FSetAttrEnd mov eax, -1@FSetAttrEnd:end {['eax', 'ecx', 'edx']};function DeleteFile (const FileName: string): boolean; assembler;asm{$IFDEF REGCALL} mov edx, eax{$ELSE REGCALL} mov edx, FileName{$ENDIF REGCALL} mov ax, 4100h call syscall mov eax, 0 jc @FDeleteEnd inc eax@FDeleteEnd:end {['eax', 'edx']};function RenameFile (const OldName, NewName: string): boolean; assembler;asm push edi{$IFDEF REGCALL} mov edx, eax mov edi, edx{$ELSE REGCALL} mov edx, OldName mov edi, NewName{$ENDIF REGCALL} mov ax, 5600h call syscall mov eax, 0 jc @FRenameEnd inc eax@FRenameEnd: pop ediend {['eax', 'edx', 'edi']};{****************************************************************************                              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 GetCurrentDir: string;begin GetDir (0, Result);end;function SetCurrentDir (const NewDir: string): boolean;begin{$I-} ChDir (NewDir); Result := (IOResult = 0);{$I+}end;function CreateDir (const NewDir: string): boolean;begin{$I-} MkDir (NewDir); Result := (IOResult = 0);{$I+}end;function RemoveDir (const Dir: string): boolean;begin{$I-} RmDir (Dir); Result := (IOResult = 0); {$I+}end;{$ASMMODE INTEL}function DirectoryExists (const Directory: string): boolean; assembler;asm{$IFDEF REGCALL} mov edx, eax{$ELSE REGCALL} mov edx, Directory{$ENDIF REGCALL} mov ax, 4300h call syscall mov eax, 0 jc @FExistsEnd test cx, 10h jz @FExistsEnd inc eax@FExistsEnd:end {['eax', 'ecx', 'edx']};{****************************************************************************                              Time Functions****************************************************************************}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 eax{$ELSE REGCALL} mov edi, SystemTime{$ENDIF REGCALL} mov ax, cx stosw xor eax, eax mov al, 10 mul dl shl eax, 16 mov al, dh stosd push edi mov ah, 2Ch call syscall pop edi xor eax, eax mov al, cl shl eax, 16 mov al, ch stosd mov al, dl shl eax, 16 mov al, dh stosd pop ediend {['eax', 'ecx', 'edx', 'edi']};{$asmmode default}{****************************************************************************                              Misc Functions****************************************************************************}procedure Beep;beginend;{****************************************************************************                              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 := PChar (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 := StrPas (GetEnvPChar (EnvVar));end;Function GetEnvironmentVariableCount : Integer;begin(*  Result:=FPCCountEnvVar(EnvP); - the amount is already known... *)  GetEnvironmentVariableCount := EnvC;end;Function GetEnvironmentString(Index : Integer) : String;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: AnsiString; const ComLine: AnsiString):                                                                       integer;var HQ: THandle; SPID, STID, QName: shortstring; SD: TStartData; SID, PID: cardinal; RD: TRequestData; PCI: PChildInfo; CISize: cardinal; Prio: byte; E: EOSError; CommandLine: ansistring;begin if os_Mode = osOS2 then  begin   FillChar (SD, SizeOf (SD), 0);   SD.Length := 24;   SD.Related := ssf_Related_Child;   SD.PgmName := PChar (Path);   SD.PgmInputs := PChar (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: AnsiString;                                  const ComLine: array of AnsiString): integer;var  CommandLine: AnsiString;  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  DoneExceptions;end.
 |