mirror of
				https://gitlab.com/freepascal.org/fpc/source.git
				synced 2025-10-25 07:21:27 +02:00 
			
		
		
		
	
		
			
				
	
	
		
			263 lines
		
	
	
		
			6.2 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			263 lines
		
	
	
		
			6.2 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
| {
 | |
|     This file is part of the Free Pascal run time library.
 | |
|     Copyright (c) 2001 by Free Pascal development team
 | |
| 
 | |
|     Low leve file functions
 | |
| 
 | |
|     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.
 | |
| 
 | |
|  **********************************************************************}
 | |
| 
 | |
| {****************************************************************************
 | |
| 
 | |
|                           Low Level File Routines
 | |
| 
 | |
| ****************************************************************************}
 | |
| 
 | |
| procedure allowslash(p:Pchar);
 | |
| {Allow slash as backslash.}
 | |
| var i:longint;
 | |
| begin
 | |
|     for i:=0 to strlen(p) do
 | |
|         if p[i]='/' then p[i]:='\';
 | |
| end;
 | |
| 
 | |
| procedure do_close(h:thandle);
 | |
| begin
 | |
| { Only three standard handles under real OS/2 }
 | |
|   if h>2 then
 | |
|   begin
 | |
|     InOutRes:=DosClose(h);
 | |
|   end;
 | |
| {$ifdef IODEBUG}
 | |
|   writeln('do_close: handle=', H, ', InOutRes=', InOutRes);
 | |
| {$endif}
 | |
| end;
 | |
| 
 | |
| procedure do_erase(p:Pchar);
 | |
| begin
 | |
|   allowslash(p);
 | |
|   inoutres:=DosDelete(p);
 | |
| end;
 | |
| 
 | |
| procedure do_rename(p1,p2:Pchar);
 | |
| begin
 | |
|   allowslash(p1);
 | |
|   allowslash(p2);
 | |
|   inoutres:=DosMove(p1, p2);
 | |
| end;
 | |
| 
 | |
| function do_read(h:thandle;addr:pointer;len:longint):longint;
 | |
| Var
 | |
|   T: cardinal;
 | |
| begin
 | |
| {$ifdef IODEBUG}
 | |
|   write('do_read: handle=', h, ', addr=', ptrint(addr), ', length=', len);
 | |
| {$endif}
 | |
|   InOutRes:=DosRead(H, Addr, Len, T);
 | |
|   do_read:= longint (T);
 | |
| {$ifdef IODEBUG}
 | |
|   writeln(', actual_len=', t, ', InOutRes=', InOutRes);
 | |
| {$endif}
 | |
| end;
 | |
| 
 | |
| function do_write(h:thandle;addr:pointer;len:longint) : longint;
 | |
| Var
 | |
|   T: cardinal;
 | |
| begin
 | |
| {$ifdef IODEBUG}
 | |
|   write('do_write: handle=', h, ', addr=', ptrint(addr), ', length=', len);
 | |
| {$endif}
 | |
|   InOutRes:=DosWrite(H, Addr, Len, T);
 | |
|   do_write:= longint (T);
 | |
| {$ifdef IODEBUG}
 | |
|   writeln(', actual_len=', t, ', InOutRes=', InOutRes);
 | |
| {$endif}
 | |
| end;
 | |
| 
 | |
| function do_filepos(handle:thandle): longint;
 | |
| var
 | |
|   PosActual: cardinal;
 | |
| begin
 | |
|   InOutRes:=DosSetFilePtr(Handle, 0, 1, PosActual);
 | |
|   do_filepos:=longint (PosActual);
 | |
| {$ifdef IODEBUG}
 | |
|   writeln('do_filepos: handle=', Handle, ', actual_pos=', PosActual, ', InOutRes=', InOutRes);
 | |
| {$endif}
 | |
| end;
 | |
| 
 | |
| procedure do_seek(handle:thandle;pos:longint);
 | |
| var
 | |
|   PosActual: cardinal;
 | |
| begin
 | |
|   InOutRes:=DosSetFilePtr(Handle, Pos, 0 {ZeroBased}, PosActual);
 | |
| {$ifdef IODEBUG}
 | |
|   writeln('do_seek: handle=', Handle, ', pos=', pos, ', actual_pos=', PosActual, ', InOutRes=', InOutRes);
 | |
| {$endif}
 | |
| end;
 | |
| 
 | |
| function do_seekend(handle:thandle):longint;
 | |
| var
 | |
|   PosActual: cardinal;
 | |
| begin
 | |
|   InOutRes:=DosSetFilePtr(Handle, 0, 2 {EndBased}, PosActual);
 | |
|   do_seekend:=longint (PosActual);
 | |
| {$ifdef IODEBUG}
 | |
|   writeln('do_seekend: handle=', Handle, ', actual_pos=', PosActual, ', InOutRes=', InOutRes);
 | |
| {$endif}
 | |
| end;
 | |
| 
 | |
| function do_filesize(handle:thandle):longint;
 | |
| var aktfilepos: cardinal;
 | |
| begin
 | |
|   aktfilepos:=do_filepos(handle);
 | |
|   do_filesize:=do_seekend(handle);
 | |
|   do_seek(handle,aktfilepos);
 | |
| end;
 | |
| 
 | |
| procedure do_truncate(handle:thandle;pos:longint);
 | |
| begin
 | |
|   InOutRes:=DosSetFileSize(Handle, Pos);
 | |
|   do_seekend(handle);
 | |
| end;
 | |
| 
 | |
| const
 | |
|     FileHandleCount: cardinal = 20;
 | |
| 
 | |
| function Increase_File_Handle_Count: boolean;
 | |
| var Err: word;
 | |
|     L1: longint;
 | |
|     L2: cardinal;
 | |
| begin
 | |
|   L1 := 10;
 | |
|   if DosSetRelMaxFH (L1, L2) <> 0 then
 | |
|     Increase_File_Handle_Count := false
 | |
|   else
 | |
|     if L2 > FileHandleCount then
 | |
|     begin
 | |
|       FileHandleCount := L2;
 | |
|       Increase_File_Handle_Count := true;
 | |
|     end
 | |
|     else
 | |
|       Increase_File_Handle_Count := false;
 | |
| end;
 | |
| 
 | |
| procedure do_open(var f;p:pchar;flags:longint);
 | |
| {
 | |
|   filerec and textrec have both handle and mode as the first items so
 | |
|   they could use the same routine for opening/creating.
 | |
| 
 | |
|   when (flags and $100)   the file will be append
 | |
|   when (flags and $1000)  the file will be truncate/rewritten
 | |
|   when (flags and $10000) there is no check for close (needed for textfiles)
 | |
| }
 | |
| var
 | |
|   Action, Attrib, OpenFlags, FM: Cardinal;
 | |
| begin
 | |
|   // convert unix slashes to normal slashes
 | |
|   allowslash(p);
 | |
| 
 | |
|   // close first if opened
 | |
|   if ((flags and $10000)=0) then
 | |
|   begin
 | |
|     case filerec(f).mode of
 | |
|       fminput,fmoutput,fminout : Do_Close(filerec(f).handle);
 | |
|       fmclosed:;
 | |
|     else
 | |
|       begin
 | |
|         inoutres:=102; {not assigned}
 | |
|         exit;
 | |
|       end;
 | |
|     end;
 | |
|   end;
 | |
| 
 | |
|   // reset file handle
 | |
|   filerec(f).handle := UnusedHandle;
 | |
| 
 | |
|   Attrib:=0;
 | |
|   OpenFlags:=0;
 | |
| 
 | |
|   // convert filesharing
 | |
|   FM := Flags and $FF and not (8);
 | |
| (* DenyNone if sharing not specified. *)
 | |
|   if FM and 112 = 0 then
 | |
|     FM := FM or 64;
 | |
|   // convert filemode to filerec modes and access mode
 | |
|   case (FM and 3) of
 | |
|     0: filerec(f).mode:=fminput;
 | |
|     1: filerec(f).mode:=fmoutput;
 | |
|     2: filerec(f).mode:=fminout;
 | |
|   end;
 | |
| 
 | |
|   if (flags and $1000)<>0 then
 | |
|     OpenFlags:=OpenFlags or 2 {doOverwrite} or 16 {doCreate} // Create/overwrite
 | |
|   else
 | |
|     OpenFlags:=OpenFlags or 1 {doOpen}; // Open existing
 | |
| 
 | |
|   // Handle Std I/O
 | |
|   if p[0]=#0 then
 | |
|   begin
 | |
|     case FileRec(f).mode of
 | |
|       fminput :
 | |
|         FileRec(f).Handle:=StdInputHandle;
 | |
|       fminout, // this is set by rewrite
 | |
|       fmoutput :
 | |
|         FileRec(f).Handle:=StdOutputHandle;
 | |
|       fmappend :
 | |
|         begin
 | |
|           FileRec(f).Handle:=StdOutputHandle;
 | |
|           FileRec(f).mode:=fmoutput; // fool fmappend
 | |
|         end;
 | |
|     end;
 | |
|     exit;
 | |
|   end;
 | |
| 
 | |
|   Attrib:=32 {faArchive};
 | |
| 
 | |
|   InOutRes:=DosOpen(p, FileRec(F).Handle, Action, 0, Attrib, OpenFlags, FM, nil);
 | |
| 
 | |
|   // If too many open files try to set more file handles and open again
 | |
|   if (InOutRes = 4) then
 | |
|     if Increase_File_Handle_Count then
 | |
|       InOutRes:=DosOpen(p, FileRec(F).Handle, Action, 0, Attrib, OpenFlags, FM, nil);
 | |
| 
 | |
|   If InOutRes<>0 then FileRec(F).Handle:=UnusedHandle;
 | |
| 
 | |
|   // If Handle created -> make some things
 | |
|   if (FileRec(F).Handle <> UnusedHandle) then
 | |
|   begin
 | |
| 
 | |
|     // Move to end of file for Append command
 | |
|     if ((Flags and $100) <> 0) then
 | |
|     begin
 | |
|       do_seekend(FileRec(F).Handle);
 | |
|       FileRec(F).Mode := fmOutput;
 | |
|     end;
 | |
| 
 | |
|   end;
 | |
| 
 | |
| {$ifdef IODEBUG}
 | |
|   writeln('do_open,', filerec(f).handle, ',', filerec(f).name, ',', filerec(f).mode, ', InOutRes=', InOutRes);
 | |
| {$endif}
 | |
| end;
 | |
| 
 | |
| function do_isdevice (Handle: THandle): boolean;
 | |
| var
 | |
|   HT, Attr: cardinal;
 | |
| begin
 | |
|   do_isdevice:=false;
 | |
|   If DosQueryHType(Handle, HT, Attr)<>0 then exit;
 | |
|   if ht=1 then do_isdevice:=true;
 | |
| end;
 | |
| {$ASMMODE ATT}
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | 
