12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070 |
- {
- This file is part of the Free Pascal run time library.
- Copyright (c) 1999-2000 by the Free Pascal development team.
- Dos unit for BP7 compatible RTL
- 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.
- **********************************************************************}
- {$inline on}
- {$IFNDEF FPC_DOTTEDUNITS}
- unit dos;
- {$ENDIF FPC_DOTTEDUNITS}
- interface
- Type
- searchrec = packed record
- fill : array[1..21] of byte;
- attr : byte;
- time : longint;
- { reserved : word; not in DJGPP V2 }
- size : longint;
- name : string[255]; { LFN Name, DJGPP uses only [12] but more can't hurt (PFV) }
- end;
- {$DEFINE HAS_REGISTERS}
- {$I registers.inc}
- {$i dosh.inc}
- {$IfDef SYSTEM_DEBUG_STARTUP}
- {$DEFINE FORCE_PROXY}
- {$endif SYSTEM_DEBUG_STARTUP}
- Const
- { This variable can be set to true
- to force use of !proxy command lines even for short
- strings, for debugging purposes mainly, as
- this might have negative impact if trying to
- call non-go32v2 programs }
- force_go32v2_proxy : boolean =
- {$ifdef FORCE_PROXY}
- true;
- {$DEFINE DEBUG_PROXY}
- {$else not FORCE_PROXY}
- false;
- {$endif not FORCE_PROXY}
- { This variable allows to use !proxy if command line is
- longer than 126 characters.
- This will only work if the called program knows how to handle
- those command lines.
- Luckily this is the case for Free Pascal compiled
- programs (even old versions)
- and go32v2 DJGPP programs.
- You can set this to false to get a warning to stderr
- if command line is too long. }
- Use_go32v2_proxy : boolean = true;
- { Added to interface so that there is no need to implement it
- both in dos and sysutils units }
- procedure exec_ansistring(path : string;comline : ansistring);
- procedure Intr(IntNo: Byte; var Regs: Registers);
- procedure MsDos(var Regs: Registers); external name 'FPC_MSDOS';
- implementation
- {$IFDEF FPC_DOTTEDUNITS}
- uses
- System.Strings, WinApi.WinProcs, WinApi.WinTypes;
- {$ELSE FPC_DOTTEDUNITS}
- uses
- strings, winprocs, wintypes;
- {$ENDIF FPC_DOTTEDUNITS}
- type
- PFarByte = ^Byte;far;
- PFarChar = ^AnsiChar;far;
- PFarWord = ^Word;far;
- {$DEFINE HAS_GETMSCOUNT}
- {$DEFINE HAS_INTR}
- {$DEFINE HAS_SETCBREAK}
- {$DEFINE HAS_GETCBREAK}
- {$DEFINE HAS_SETVERIFY}
- {$DEFINE HAS_GETVERIFY}
- {//$DEFINE HAS_SWAPVECTORS}
- {$DEFINE HAS_GETINTVEC}
- {$DEFINE HAS_SETINTVEC}
- {$DEFINE HAS_GETSHORTNAME}
- {$DEFINE HAS_GETLONGNAME}
- {$DEFINE FPC_FEXPAND_UNC} (* UNC paths are supported *)
- {$DEFINE FPC_FEXPAND_DRIVES} (* Full paths begin with drive specification *)
- {$I dos.inc}
- {******************************************************************************
- --- Dos Interrupt ---
- ******************************************************************************}
- var
- dosregs : registers;
- procedure Intr(IntNo: Byte; var Regs: Registers);assembler;
- asm
- // todo
- end;
- { in protected mode, loading invalid values into segment registers causes an
- exception, so we use this function to initialize our Registers structure }
- procedure ZeroSegRegs(var regs: Registers); inline;
- begin
- regs.DS:=0;
- regs.ES:=0;
- end;
- procedure LoadDosError;
- var
- r : registers;
- SimpleDosError : word;
- begin
- if (dosregs.flags and fcarry) <> 0 then
- begin
- { I got a extended error = 0
- while CarryFlag was set from Exec function }
- SimpleDosError:=dosregs.ax;
- ZeroSegRegs(r);
- r.ax:=$5900;
- r.bx:=$0;
- msdos(r);
- { conversion from word to integer !!
- gave a Bound check error if ax is $FFFF !! PM }
- doserror:=integer(r.ax);
- case doserror of
- 0 : DosError:=integer(SimpleDosError);
- 19 : DosError:=150;
- 21 : DosError:=152;
- end;
- end
- else
- doserror:=0;
- end;
- {******************************************************************************
- --- Info / Date / Time ---
- ******************************************************************************}
- function dosversion : word;
- begin
- ZeroSegRegs(dosregs);
- dosregs.ax:=$3000;
- msdos(dosregs);
- dosversion:=dosregs.ax;
- end;
- procedure getdate(var year,month,mday,wday : word);
- begin
- ZeroSegRegs(dosregs);
- dosregs.ax:=$2a00;
- msdos(dosregs);
- wday:=dosregs.al;
- year:=dosregs.cx;
- month:=dosregs.dh;
- mday:=dosregs.dl;
- end;
- procedure setdate(year,month,day : word);
- begin
- ZeroSegRegs(dosregs);
- dosregs.cx:=year;
- dosregs.dh:=month;
- dosregs.dl:=day;
- dosregs.ah:=$2b;
- msdos(dosregs);
- end;
- procedure gettime(var hour,minute,second,sec100 : word);
- begin
- ZeroSegRegs(dosregs);
- dosregs.ah:=$2c;
- msdos(dosregs);
- hour:=dosregs.ch;
- minute:=dosregs.cl;
- second:=dosregs.dh;
- sec100:=dosregs.dl;
- end;
- procedure settime(hour,minute,second,sec100 : word);
- begin
- ZeroSegRegs(dosregs);
- dosregs.ch:=hour;
- dosregs.cl:=minute;
- dosregs.dh:=second;
- dosregs.dl:=sec100;
- dosregs.ah:=$2d;
- msdos(dosregs);
- end;
- function GetMsCount: int64;
- begin
- GetMsCount:=GetTickCount;
- end;
- {******************************************************************************
- --- Exec ---
- ******************************************************************************}
- procedure exec_ansistring(path : string;comline : ansistring);
- var
- c: ansistring;
- pc: PAnsiChar;
- p: string;
- winexec_result: Word;
- m: MSG;
- begin
- { create command line }
- p:=path;
- { allow slash as backslash }
- DoDirSeparators(p);
- if Pos(' ',p)<>0 then
- c:='"'+p+'" '+comline
- else
- c:=p+' '+comline;
- pc:=PAnsiChar(c);
- winexec_result:=WinExec(FarAddr(pc^),SW_SHOW);
- if winexec_result<32 then
- begin
- doserror:=winexec_result;
- LastDosExitCode:=0;
- end
- else
- begin
- doserror:=0;
- { wait until the hinstance terminates }
- while GetModuleUsage(winexec_result)>0 do
- begin
- while PeekMessage(FarAddr(m),0,0,0,1) do
- begin
- TranslateMessage(FarAddr(m));
- DispatchMessage(FarAddr(m));
- end;
- end;
- { TODO: is there actually a way to receive the child exit code in win16??? }
- LastDosExitCode:=0;
- end;
- end;
- procedure exec(const path : pathstr;const comline : comstr);
- begin
- exec_ansistring(path, comline);
- end;
- procedure getcbreak(var breakvalue : boolean);
- begin
- ZeroSegRegs(dosregs);
- dosregs.ax:=$3300;
- msdos(dosregs);
- breakvalue:=dosregs.dl<>0;
- end;
- procedure setcbreak(breakvalue : boolean);
- begin
- ZeroSegRegs(dosregs);
- dosregs.ax:=$3301;
- dosregs.dl:=ord(breakvalue);
- msdos(dosregs);
- end;
- procedure getverify(var verify : boolean);
- begin
- ZeroSegRegs(dosregs);
- dosregs.ah:=$54;
- msdos(dosregs);
- verify:=dosregs.al<>0;
- end;
- procedure setverify(verify : boolean);
- begin
- ZeroSegRegs(dosregs);
- dosregs.ah:=$2e;
- dosregs.al:=ord(verify);
- msdos(dosregs);
- end;
- {******************************************************************************
- --- Disk ---
- ******************************************************************************}
- type
- ExtendedFat32FreeSpaceRec = packed record
- RetSize : word; { $00 }
- Strucversion : word; { $02 }
- SecPerClus, { $04 }
- BytePerSec, { $08 }
- AvailClusters, { $0C }
- TotalClusters, { $10 }
- AvailPhysSect, { $14 }
- TotalPhysSect, { $18 }
- AvailAllocUnits, { $1C }
- TotalAllocUnits : longword; { $20 }
- Dummy, { $24 }
- Dummy2 : longword; { $28 }
- end; { $2C }
- const
- IOCTL_INPUT = 3; //For request header command field
- CDFUNC_SECTSIZE = 7; //For cdrom control block func field
- CDFUNC_VOLSIZE = 8; //For cdrom control block func field
- type
- TRequestHeader = packed record
- length : byte; { $00 }
- subunit : byte; { $01 }
- command : byte; { $02 }
- status : word; { $03 }
- reserved1 : longword; { $05 }
- reserved2 : longword; { $09 }
- media_desc : byte; { $0D }
- transf_ofs : word; { $0E }
- transf_seg : word; { $10 }
- numbytes : word; { $12 }
- end; { $14 }
- TCDSectSizeReq = packed record
- func : byte; { $00 }
- mode : byte; { $01 }
- secsize : word; { $02 }
- end; { $04 }
- TCDVolSizeReq = packed record
- func : byte; { $00 }
- size : longword; { $01 }
- end; { $05 }
- function do_diskdata(drive : byte; Free : boolean) : Int64;
- var
- blocksize, freeblocks, totblocks : longword;
- { Get disk data via old int21/36 (GET FREE DISK SPACE). It's always supported
- even if it returns wrong values for volumes > 2GB and for cdrom drives when
- in pure DOS. Note that it's also the only way to get some data on WinNTs. }
- function DiskData_36 : boolean;
- begin
- DiskData_36:=false;
- ZeroSegRegs(dosregs);
- dosregs.dl:=drive;
- dosregs.ah:=$36;
- msdos(dosregs);
- if dosregs.ax=$FFFF then exit;
- blocksize:=dosregs.ax*dosregs.cx;
- freeblocks:=dosregs.bx;
- totblocks:=dosregs.dx;
- Diskdata_36:=true;
- end;
- { Get disk data via int21/7303 (FAT32 - GET EXTENDED FREE SPACE ON DRIVE).
- It is supported by win9x even in pure DOS }
- function DiskData_7303 : boolean;
- var
- s : shortstring;
- rec : ExtendedFat32FreeSpaceRec;
- begin
- DiskData_7303:=false;
- s:=chr(drive+$40)+':\'+#0;
- rec.Strucversion:=0;
- rec.RetSize := 0;
- { no need to call ZeroSegRegs(dosregs), because es and ds are initialized below }
- dosregs.dx:=Ofs(s[1]);
- dosregs.ds:=Seg(s[1]);
- dosregs.di:=Ofs(Rec);
- dosregs.es:=Seg(Rec);
- dosregs.cx:=Sizeof(ExtendedFat32FreeSpaceRec);
- dosregs.ax:=$7303;
- msdos(dosregs);
- if (dosregs.flags and fcarry) <> 0 then
- exit;
- if Rec.RetSize = 0 then
- exit;
- blocksize:=rec.SecPerClus*rec.BytePerSec;
- freeblocks:=rec.AvailAllocUnits;
- totblocks:=rec.TotalAllocUnits;
- DiskData_7303:=true;
- end;
- { Get disk data asking to MSCDEX. Pure DOS returns wrong values with
- int21/7303 or int21/36 if the drive is a CDROM drive }
- function DiskData_CDROM : boolean;
- var req : TRequestHeader;
- sectreq : TCDSectSizeReq;
- sizereq : TCDVolSizeReq;
- i : integer;
- drnum : byte;
- begin
- DiskData_CDROM:=false;
- exit;
- (* drnum:=drive-1; //for MSCDEX, 0 = a, 1 = b etc, unlike int21/36
- { Is this a CDROM drive? }
- dosregs.ax:=$150b;
- dosregs.cx:=drnum;
- intr($2f,dosregs);
- if (dosregs.bx<>$ADAD) or (dosregs.ax=0) then
- exit; // no, it isn't
- { Prepare the request header to send to the cdrom driver }
- FillByte(req,sizeof(req),0);
- req.length:=sizeof(req);
- req.command:=IOCTL_INPUT;
- req.transf_ofs:=Ofs(sectreq);
- req.transf_seg:=Seg(sectreq);
- req.numbytes:=sizeof(sectreq);
- { We're asking the sector size }
- sectreq.func:=CDFUNC_SECTSIZE;
- sectreq.mode:=0; //cooked
- sectreq.secsize:=0;
- for i:=1 to 2 do
- begin
- { Send the request to the cdrom driver }
- dosregs.ax:=$1510;
- dosregs.cx:=drnum;
- dosregs.es:=Seg(req);
- dosregs.bx:=Ofs(req);
- intr($2f,dosregs);
- { status = $800F means "disk changed". Try once more. }
- if (req.status and $800F) <> $800F then break;
- end;
- if (req.status<>$0100) or (req.numbytes<>sizeof(sectreq)) then
- exit; //An error occurred
- { Update the request header for the next request }
- FillByte(req,sizeof(req),0);
- req.length:=sizeof(req);
- req.command:=IOCTL_INPUT;
- req.transf_ofs:=Ofs(sizereq);
- req.transf_seg:=Seg(sizereq);
- req.numbytes:=sizeof(sizereq);
- { We're asking the volume size (in blocks) }
- sizereq.func:=CDFUNC_VOLSIZE;
- sizereq.size:=0;
- { Send the request to the cdrom driver }
- dosregs.ax:=$1510;
- dosregs.cx:=drnum;
- dosregs.es:=Seg(req);
- dosregs.bx:=Ofs(req);
- intr($2f,dosregs);
- if (req.status<>$0100) or (req.numbytes<>sizeof(sizereq)) then
- exit; //An error occurred
- blocksize:=sectreq.secsize;
- freeblocks:=0; //always 0 for a cdrom
- totblocks:=sizereq.size;
- DiskData_CDROM:=true;*)
- end;
- begin
- if drive=0 then
- begin
- ZeroSegRegs(dosregs);
- dosregs.ax:=$1900; //get current default drive
- msdos(dosregs);
- drive:=dosregs.al+1;
- end;
- if not DiskData_CDROM then
- if not DiskData_7303 then
- if not DiskData_36 then
- begin
- do_diskdata:=-1;
- exit;
- end;
- do_diskdata:=blocksize;
- if free then
- do_diskdata:=do_diskdata*freeblocks
- else
- do_diskdata:=do_diskdata*totblocks;
- end;
- function diskfree(drive : byte) : int64;
- begin
- diskfree:=Do_DiskData(drive,TRUE);
- end;
- function disksize(drive : byte) : int64;
- begin
- disksize:=Do_DiskData(drive,false);
- end;
- {******************************************************************************
- --- LFNFindfirst LFNFindNext ---
- ******************************************************************************}
- type
- LFNSearchRec=packed record
- attr,
- crtime,
- crtimehi,
- actime,
- actimehi,
- lmtime,
- lmtimehi,
- sizehi,
- size : longint;
- reserved : array[0..7] of byte;
- name : array[0..259] of byte;
- shortname : array[0..13] of byte;
- end;
- procedure LFNSearchRec2Dos(const w:LFNSearchRec;hdl:longint;var d:Searchrec;from_findfirst : boolean);
- var
- Len : integer;
- begin
- With w do
- begin
- FillChar(d,sizeof(SearchRec),0);
- if DosError=0 then
- len:=StrLen(@Name)
- else
- len:=0;
- d.Name[0]:=chr(len);
- Move(Name[0],d.Name[1],Len);
- d.Time:=lmTime;
- d.Size:=Size;
- d.Attr:=Attr and $FF;
- if (DosError<>0) and from_findfirst then
- hdl:=-1;
- Move(hdl,d.Fill,4);
- end;
- end;
- {$ifdef DEBUG_LFN}
- const
- LFNFileName : string = 'LFN.log';
- LFNOpenNb : longint = 0;
- LogLFN : boolean = false;
- var
- lfnfile : text;
- {$endif DEBUG_LFN}
- procedure LFNFindFirst(path:PAnsiChar;attr:longint;var s:searchrec);
- var
- w : LFNSearchRec;
- begin
- { allow slash as backslash }
- DoDirSeparators(path);
- { no need to call ZeroSegRegs(dosregs), because both ds and es are initialized below }
- dosregs.si:=1; { use ms-dos time }
- { don't include the label if not asked for it, needed for network drives }
- if attr=$8 then
- dosregs.cx:=8
- else
- dosregs.cx:=attr and (not 8);
- dosregs.dx:=Ofs(path^);
- dosregs.ds:=Seg(path^);
- dosregs.di:=Ofs(w);
- dosregs.es:=Seg(w);
- dosregs.ax:=$714e;
- msdos(dosregs);
- LoadDosError;
- if DosError=2 then
- DosError:=18;
- {$ifdef DEBUG_LFN}
- if (DosError=0) and LogLFN then
- begin
- Append(lfnfile);
- inc(LFNOpenNb);
- Writeln(lfnfile,LFNOpenNb,' LFNFindFirst called ',path);
- close(lfnfile);
- end;
- {$endif DEBUG_LFN}
- LFNSearchRec2Dos(w,dosregs.ax,s,true);
- end;
- procedure LFNFindNext(var s:searchrec);
- var
- hdl : longint;
- w : LFNSearchRec;
- begin
- Move(s.Fill,hdl,4);
- { no need to call ZeroSegRegs(dosregs), because both ds and es are initialized below }
- dosregs.si:=1; { use ms-dos time }
- dosregs.di:=Ofs(w);
- dosregs.es:=Seg(w);
- dosregs.ds:=0; { avoid invalid selector values in protected mode }
- dosregs.bx:=hdl;
- dosregs.ax:=$714f;
- msdos(dosregs);
- LoadDosError;
- LFNSearchRec2Dos(w,hdl,s,false);
- end;
- procedure LFNFindClose(var s:searchrec);
- var
- hdl : longint;
- begin
- Move(s.Fill,hdl,4);
- { Do not call MsDos if FindFirst returned with an error }
- if hdl=-1 then
- begin
- DosError:=0;
- exit;
- end;
- ZeroSegRegs(dosregs);
- dosregs.bx:=hdl;
- dosregs.ax:=$71a1;
- msdos(dosregs);
- LoadDosError;
- {$ifdef DEBUG_LFN}
- if (DosError=0) and LogLFN then
- begin
- Append(lfnfile);
- Writeln(lfnfile,LFNOpenNb,' LFNFindClose called ');
- close(lfnfile);
- if LFNOpenNb>0 then
- dec(LFNOpenNb);
- end;
- {$endif DEBUG_LFN}
- end;
- {******************************************************************************
- --- DosFindfirst DosFindNext ---
- ******************************************************************************}
- procedure dossearchrec2searchrec(var f : searchrec);
- var
- len : integer;
- begin
- { Check is necessary!! OS/2's VDM doesn't clear the name with #0 if the }
- { file doesn't exist! (JM) }
- if dosError = 0 then
- len:=StrLen(@f.Name)
- else len := 0;
- Move(f.Name[0],f.Name[1],Len);
- f.Name[0]:=chr(len);
- end;
- procedure DosFindfirst(path : PAnsiChar;attr : word;var f : searchrec);
- begin
- { allow slash as backslash }
- DoDirSeparators(path);
- dosregs.dx:=Ofs(f);
- dosregs.ds:=Seg(f);
- dosregs.es:=0; { avoid invalid selector values in protected mode }
- dosregs.ah:=$1a;
- msdos(dosregs);
- dosregs.cx:=attr;
- dosregs.dx:=Ofs(path^);
- dosregs.ds:=Seg(path^);
- dosregs.es:=0; { avoid invalid selector values in protected mode }
- dosregs.ah:=$4e;
- msdos(dosregs);
- LoadDosError;
- dossearchrec2searchrec(f);
- end;
- procedure Dosfindnext(var f : searchrec);
- begin
- dosregs.dx:=Ofs(f);
- dosregs.ds:=Seg(f);
- dosregs.es:=0; { avoid invalid selector values in protected mode }
- dosregs.ah:=$1a;
- msdos(dosregs);
- ZeroSegRegs(dosregs);
- dosregs.ah:=$4f;
- msdos(dosregs);
- LoadDosError;
- dossearchrec2searchrec(f);
- end;
- {******************************************************************************
- --- Findfirst FindNext ---
- ******************************************************************************}
- procedure findfirst(const path : pathstr;attr : word;var f : searchRec);
- var
- path0 : array[0..255] of AnsiChar;
- begin
- doserror:=0;
- strpcopy(path0,path);
- if LFNSupport then
- LFNFindFirst(path0,attr,f)
- else
- Dosfindfirst(path0,attr,f);
- end;
- procedure findnext(var f : searchRec);
- begin
- doserror:=0;
- if LFNSupport then
- LFNFindnext(f)
- else
- Dosfindnext(f);
- end;
- Procedure FindClose(Var f: SearchRec);
- begin
- DosError:=0;
- if LFNSupport then
- LFNFindClose(f);
- end;
- {procedure SwapIntVec(IntNo: Byte; var Vector: FarPointer);
- var
- tmpvec: FarPointer;
- begin
- GetIntVec(IntNo, tmpvec);
- SetIntVec(IntNo, Vector);
- Vector := tmpvec;
- end;
- procedure SwapVectors;
- begin
- SwapIntVec(0, SaveInt00);
- end;}
- {******************************************************************************
- --- File ---
- ******************************************************************************}
- Function FSearch(path: pathstr; dirlist: string): pathstr;
- var
- p1 : longint;
- s : searchrec;
- newdir : pathstr;
- begin
- { No wildcards allowed in these things }
- if (pos('?',path)<>0) or (pos('*',path)<>0) then
- begin
- fsearch:='';
- exit;
- end;
- { check if the file specified exists }
- findfirst(path,anyfile and not(directory),s);
- if doserror=0 then
- begin
- findclose(s);
- fsearch:=path;
- exit;
- end;
- findclose(s);
- { allow slash as backslash }
- DoDirSeparators(dirlist);
- repeat
- p1:=pos(';',dirlist);
- if p1<>0 then
- begin
- newdir:=copy(dirlist,1,p1-1);
- delete(dirlist,1,p1);
- end
- else
- begin
- newdir:=dirlist;
- dirlist:='';
- end;
- if (newdir<>'') and (not (newdir[length(newdir)] in [DirectorySeparator,DriveSeparator])) then
- newdir:=newdir+DirectorySeparator;
- findfirst(newdir+path,anyfile and not(directory),s);
- if doserror=0 then
- newdir:=newdir+path
- else
- newdir:='';
- findclose(s);
- until (dirlist='') or (newdir<>'');
- fsearch:=newdir;
- end;
- { change to short filename if successful DOS call PM }
- function GetShortName(var p : String) : boolean;
- var
- c : array[0..255] of AnsiChar;
- begin
- move(p[1],c[0],length(p));
- c[length(p)]:=#0;
- { no need to call ZeroSegRegs(dosregs), because es and ds are initialized below }
- dosregs.ax:=$7160;
- dosregs.cx:=1;
- dosregs.ds:=Seg(c);
- dosregs.si:=Ofs(c);
- dosregs.es:=Seg(c);
- dosregs.di:=Ofs(c);
- msdos(dosregs);
- LoadDosError;
- if DosError=0 then
- begin
- move(c[0],p[1],strlen(c));
- p[0]:=AnsiChar(strlen(c));
- GetShortName:=true;
- end
- else
- GetShortName:=false;
- end;
- { change to long filename if successful DOS call PM }
- function GetLongName(var p : String) : boolean;
- var
- c : array[0..260] of AnsiChar;
- begin
- move(p[1],c[0],length(p));
- c[length(p)]:=#0;
- { no need to call ZeroSegRegs(dosregs), because es and ds are initialized below }
- dosregs.ax:=$7160;
- dosregs.cx:=2;
- dosregs.ds:=Seg(c);
- dosregs.si:=Ofs(c);
- dosregs.es:=Seg(c);
- dosregs.di:=Ofs(c);
- msdos(dosregs);
- LoadDosError;
- if DosError=0 then
- begin
- c[255]:=#0;
- move(c[0],p[1],strlen(c));
- p[0]:=AnsiChar(strlen(c));
- GetLongName:=true;
- end
- else
- GetLongName:=false;
- end;
- {******************************************************************************
- --- Get/Set File Time,Attr ---
- ******************************************************************************}
- procedure getftime(var f;var time : longint);
- begin
- ZeroSegRegs(dosregs);
- dosregs.bx:=textrec(f).handle;
- dosregs.ax:=$5700;
- msdos(dosregs);
- loaddoserror;
- time:=(longint(dosregs.dx) shl 16)+dosregs.cx;
- end;
- procedure setftime(var f;time : longint);
- begin
- ZeroSegRegs(dosregs);
- dosregs.bx:=textrec(f).handle;
- dosregs.cx:=time and $ffff;
- dosregs.dx:=time shr 16;
- dosregs.ax:=$5701;
- msdos(dosregs);
- loaddoserror;
- end;
- procedure getfattr(var f;var attr : word);
- var
- path: PAnsiChar;
- {$ifndef FPC_ANSI_TEXTFILEREC}
- r: rawbytestring;
- {$endif not FPC_ANSI_TEXTFILEREC}
- begin
- {$ifdef FPC_ANSI_TEXTFILEREC}
- path:=@filerec(f).Name;
- {$else}
- r:=ToSingleByteFileSystemEncodedFileName(filerec(f).Name);
- path:=PAnsiChar(r);
- {$endif}
- dosregs.dx:=Ofs(path^);
- dosregs.ds:=Seg(path^);
- dosregs.es:=0; { avoid invalid selector values in protected mode }
- if LFNSupport then
- begin
- dosregs.ax:=$7143;
- dosregs.bx:=0;
- end
- else
- dosregs.ax:=$4300;
- msdos(dosregs);
- LoadDosError;
- Attr:=dosregs.cx;
- end;
- procedure setfattr(var f;attr : word);
- var
- path: PAnsiChar;
- {$ifndef FPC_ANSI_TEXTFILEREC}
- r: rawbytestring;
- {$endif not FPC_ANSI_TEXTFILEREC}
- begin
- { Fail for setting VolumeId. }
- if ((attr and VolumeID)<>0) then
- begin
- doserror:=5;
- exit;
- end;
- {$ifdef FPC_ANSI_TEXTFILEREC}
- path:=@filerec(f).Name;
- {$else}
- r:=ToSingleByteFileSystemEncodedFileName(filerec(f).Name);
- path:=PAnsiChar(r);
- {$endif}
- dosregs.dx:=Ofs(path);
- dosregs.ds:=Seg(path);
- dosregs.es:=0; { avoid invalid selector values in protected mode }
- if LFNSupport then
- begin
- dosregs.ax:=$7143;
- dosregs.bx:=1;
- end
- else
- dosregs.ax:=$4301;
- dosregs.cx:=attr;
- msdos(dosregs);
- LoadDosError;
- end;
- {******************************************************************************
- --- Environment ---
- ******************************************************************************}
- function GetEnvStr(EnvNo: Integer; var OutEnvStr: string): integer;
- var
- dos_env_ptr: LPSTR;
- Ch: AnsiChar;
- begin
- dos_env_ptr := GetDOSEnvironment;
- GetEnvStr := 1;
- OutEnvStr := '';
- repeat
- Ch := dos_env_ptr^;
- if (Ch = #0) and ((dos_env_ptr+1)^ = #0) then
- exit;
- if Ch = #0 then
- Inc(GetEnvStr);
- if (Ch <> #0) and (GetEnvStr = EnvNo) then
- OutEnvStr := OutEnvStr + Ch;
- Inc(dos_env_ptr);
- if Ofs(dos_env_ptr^) = 0 then
- exit;
- until false;
- end;
- function envcount : longint;
- var
- tmpstr: string;
- begin
- envcount := GetEnvStr(-1, tmpstr);
- end;
- function envstr (Index: longint): string;
- begin
- GetEnvStr(Index, envstr);
- end;
- Function GetEnv(envvar: string): string;
- var
- hs : string;
- eqpos : integer;
- I : integer;
- begin
- envvar:=upcase(envvar);
- getenv:='';
- for I := 1 to envcount do
- begin
- hs:=envstr(I);
- eqpos:=pos('=',hs);
- if upcase(copy(hs,1,eqpos-1))=envvar then
- begin
- getenv:=copy(hs,eqpos+1,length(hs)-eqpos);
- break;
- end;
- end;
- end;
- {******************************************************************************
- --- Get/SetIntVec ---
- ******************************************************************************}
- procedure GetIntVec(intno: Byte; var vector: farpointer); assembler;
- asm
- mov al, intno
- mov ah, 35h
- int 21h
- xchg ax, bx
- {$if defined(FPC_MM_TINY) or defined(FPC_MM_SMALL) or defined(FPC_MM_MEDIUM)}
- mov bx, vector
- mov [bx], ax
- mov ax, es
- mov [bx + 2], ax
- {$else}
- {$ifdef FPC_ENABLED_CLD}
- cld
- {$endif FPC_ENABLED_CLD}
- push es
- pop bx
- les di, vector
- stosw
- xchg ax, bx
- stosw
- {$endif}
- end;
- procedure SetIntVec(intno: Byte; vector: farpointer); assembler;
- asm
- push ds
- mov al, intno
- mov ah, 25h
- lds dx, word [vector]
- int 21h
- pop ds
- end;
- {$ifdef DEBUG_LFN}
- begin
- LogLFN:=(GetEnv('LOGLFN')<>'');
- assign(lfnfile,LFNFileName);
- {$I-}
- Reset(lfnfile);
- if IOResult<>0 then
- begin
- Rewrite(lfnfile);
- Writeln(lfnfile,'New lfn.log');
- end;
- close(lfnfile);
- {$endif DEBUG_LFN}
- end.
|