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 ebx
- end {['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 ebx
- end {['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 ebx
- end {['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 ebx
- end {['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 ebx
- end {['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 ebx
- end {['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 ebx
- end {['eax', 'ebx', 'ecx', 'edx']};
- function FileSetDate (Handle, Age: longint): longint;
- var FStat: PFileStatus3;
- RC: cardinal;
- begin
- if os_mode = osOS2 then
- begin
- 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 edi
- end {['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 edi
- end {['eax', 'ecx', 'edx', 'edi']};
- {$asmmode default}
- {****************************************************************************
- Misc Functions
- ****************************************************************************}
- procedure Beep;
- begin
- end;
- {****************************************************************************
- 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.
|