mirror of
				https://gitlab.com/freepascal.org/fpc/source.git
				synced 2025-10-26 19:51:43 +01:00 
			
		
		
		
	
		
			
				
	
	
		
			1237 lines
		
	
	
		
			28 KiB
		
	
	
	
		
			ObjectPascal
		
	
	
	
	
	
			
		
		
	
	
			1237 lines
		
	
	
		
			28 KiB
		
	
	
	
		
			ObjectPascal
		
	
	
	
	
	
| {
 | |
|     $Id$
 | |
|     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.
 | |
| 
 | |
|  **********************************************************************}
 | |
| unit dos;
 | |
| interface
 | |
| 
 | |
| Uses
 | |
|   Go32;
 | |
| 
 | |
| 
 | |
| Const
 | |
|   {Bitmasks for CPU Flags}
 | |
|   fcarry     = $0001;
 | |
|   fparity    = $0004;
 | |
|   fauxiliary = $0010;
 | |
|   fzero      = $0040;
 | |
|   fsign      = $0080;
 | |
|   foverflow  = $0800;
 | |
| 
 | |
|   {Bitmasks for file attribute}
 | |
|   readonly  = $01;
 | |
|   hidden    = $02;
 | |
|   sysfile   = $04;
 | |
|   volumeid  = $08;
 | |
|   directory = $10;
 | |
|   archive   = $20;
 | |
|   anyfile   = $3F;
 | |
| 
 | |
|   {File Status}
 | |
|   fmclosed = $D7B0;
 | |
|   fminput  = $D7B1;
 | |
|   fmoutput = $D7B2;
 | |
|   fminout  = $D7B3;
 | |
| 
 | |
| 
 | |
| Type
 | |
| { Needed for LFN Support }
 | |
|   ComStr  = String[255];
 | |
|   PathStr = String[255];
 | |
|   DirStr  = String[255];
 | |
|   NameStr = String[255];
 | |
|   ExtStr  = String[255];
 | |
| 
 | |
| {
 | |
|   filerec.inc contains the definition of the filerec.
 | |
|   textrec.inc contains the definition of the textrec.
 | |
|   It is in a separate file to make it available in other units without
 | |
|   having to use the DOS unit for it.
 | |
| }
 | |
| {$i filerec.inc}
 | |
| {$i textrec.inc}
 | |
| 
 | |
|   DateTime = packed record
 | |
|     Year,
 | |
|     Month,
 | |
|     Day,
 | |
|     Hour,
 | |
|     Min,
 | |
|     Sec   : word;
 | |
|   End;
 | |
| 
 | |
|   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;
 | |
| 
 | |
|   Registers = Go32.Registers;
 | |
| 
 | |
| Var
 | |
|   DosError : integer;
 | |
| 
 | |
| {Interrupt}
 | |
| Procedure Intr(intno: byte; var regs: registers);
 | |
| Procedure MSDos(var regs: registers);
 | |
| 
 | |
| {Info/Date/Time}
 | |
| Function  DosVersion: Word;
 | |
| Procedure GetDate(var year, month, mday, wday: word);
 | |
| Procedure GetTime(var hour, minute, second, sec100: word);
 | |
| procedure SetDate(year,month,day: word);
 | |
| Procedure SetTime(hour,minute,second,sec100: word);
 | |
| Procedure UnpackTime(p: longint; var t: datetime);
 | |
| Procedure PackTime(var t: datetime; var p: longint);
 | |
| 
 | |
| {Exec}
 | |
| Procedure Exec(const path: pathstr; const comline: comstr);
 | |
| Function  DosExitCode: word;
 | |
| 
 | |
| {Disk}
 | |
| {$ifdef Int64}
 | |
|  Function  DiskFree(drive: byte) : int64;
 | |
|  Function  DiskSize(drive: byte) : int64;
 | |
| {$else}
 | |
|  Function  DiskFree(drive: byte) : longint;
 | |
|  Function  DiskSize(drive: byte) : longint;
 | |
| {$endif}
 | |
| Procedure FindFirst(const path: pathstr; attr: word; var f: searchRec);
 | |
| Procedure FindNext(var f: searchRec);
 | |
| Procedure FindClose(Var f: SearchRec);
 | |
| 
 | |
| {File}
 | |
| Procedure GetFAttr(var f; var attr: word);
 | |
| Procedure GetFTime(var f; var time: longint);
 | |
| Function  FSearch(path: pathstr; dirlist: string): pathstr;
 | |
| Function  FExpand(const path: pathstr): pathstr;
 | |
| Procedure FSplit(path: pathstr; var dir: dirstr; var name: namestr; var ext: extstr);
 | |
| function  GetShortName(var p : String) : boolean;
 | |
| 
 | |
| {Environment}
 | |
| Function  EnvCount: longint;
 | |
| Function  EnvStr(index: integer): string;
 | |
| Function  GetEnv(envvar: string): string;
 | |
| 
 | |
| {Misc}
 | |
| Procedure SetFAttr(var f; attr: word);
 | |
| Procedure SetFTime(var f; time: longint);
 | |
| Procedure GetCBreak(var breakvalue: boolean);
 | |
| Procedure SetCBreak(breakvalue: boolean);
 | |
| Procedure GetVerify(var verify: boolean);
 | |
| Procedure SetVerify(verify: boolean);
 | |
| 
 | |
| {Do Nothing Functions}
 | |
| Procedure SwapVectors;
 | |
| Procedure GetIntVec(intno: byte; var vector: pointer);
 | |
| Procedure SetIntVec(intno: byte; vector: pointer);
 | |
| Procedure Keep(exitcode: word);
 | |
| 
 | |
| 
 | |
| implementation
 | |
| 
 | |
| uses
 | |
|   strings;
 | |
| 
 | |
| {$ASMMODE ATT}
 | |
| 
 | |
| {******************************************************************************
 | |
|                            --- Dos Interrupt ---
 | |
| ******************************************************************************}
 | |
| 
 | |
| var
 | |
|   dosregs : registers;
 | |
| 
 | |
| procedure LoadDosError;
 | |
| var
 | |
|   r : registers;
 | |
|   SimpleDosError : word;
 | |
| begin
 | |
|   if (dosregs.flags and carryflag) <> 0 then
 | |
|    begin
 | |
|      { I got a extended error = 0
 | |
|        while CarryFlag was set from Exec function }
 | |
|      SimpleDosError:=dosregs.ax;
 | |
|      r.eax:=$5900;
 | |
|      r.ebx:=$0;
 | |
|      realintr($21,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;
 | |
| 
 | |
| 
 | |
| procedure intr(intno : byte;var regs : registers);
 | |
| begin
 | |
|   realintr(intno,regs);
 | |
| end;
 | |
| 
 | |
| 
 | |
| procedure msdos(var regs : registers);
 | |
| begin
 | |
|   intr($21,regs);
 | |
| end;
 | |
| 
 | |
| 
 | |
| {******************************************************************************
 | |
|                         --- Info / Date / Time ---
 | |
| ******************************************************************************}
 | |
| 
 | |
| function dosversion : word;
 | |
| begin
 | |
|   dosregs.ax:=$3000;
 | |
|   msdos(dosregs);
 | |
|   dosversion:=dosregs.ax;
 | |
| end;
 | |
| 
 | |
| 
 | |
| procedure getdate(var year,month,mday,wday : word);
 | |
| begin
 | |
|   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
 | |
|    dosregs.cx:=year;
 | |
|    dosregs.dh:=month;
 | |
|    dosregs.dl:=day;
 | |
|    dosregs.ah:=$2b;
 | |
|    msdos(dosregs);
 | |
|    DosError:=0;
 | |
| end;
 | |
| 
 | |
| 
 | |
| procedure gettime(var hour,minute,second,sec100 : word);
 | |
| begin
 | |
|   dosregs.ah:=$2c;
 | |
|   msdos(dosregs);
 | |
|   hour:=dosregs.ch;
 | |
|   minute:=dosregs.cl;
 | |
|   second:=dosregs.dh;
 | |
|   sec100:=dosregs.dl;
 | |
|   DosError:=0;
 | |
| end;
 | |
| 
 | |
| 
 | |
| procedure settime(hour,minute,second,sec100 : word);
 | |
| begin
 | |
|   dosregs.ch:=hour;
 | |
|   dosregs.cl:=minute;
 | |
|   dosregs.dh:=second;
 | |
|   dosregs.dl:=sec100;
 | |
|   dosregs.ah:=$2d;
 | |
|   msdos(dosregs);
 | |
|   DosError:=0;
 | |
| end;
 | |
| 
 | |
| 
 | |
| Procedure packtime(var t : datetime;var p : longint);
 | |
| Begin
 | |
|   p:=(t.sec shr 1)+(t.min shl 5)+(t.hour shl 11)+(t.day shl 16)+(t.month shl 21)+((t.year-1980) shl 25);
 | |
| End;
 | |
| 
 | |
| 
 | |
| Procedure unpacktime(p : longint;var t : datetime);
 | |
| Begin
 | |
|   with t do
 | |
|    begin
 | |
|      sec:=(p and 31) shl 1;
 | |
|      min:=(p shr 5) and 63;
 | |
|      hour:=(p shr 11) and 31;
 | |
|      day:=(p shr 16) and 31;
 | |
|      month:=(p shr 21) and 15;
 | |
|      year:=(p shr 25)+1980;
 | |
|    end;
 | |
| End;
 | |
| 
 | |
| 
 | |
| {******************************************************************************
 | |
|                                --- Exec ---
 | |
| ******************************************************************************}
 | |
| 
 | |
| var
 | |
|   lastdosexitcode : word;
 | |
| 
 | |
| procedure exec(const path : pathstr;const comline : comstr);
 | |
| type
 | |
|   realptr = packed record
 | |
|     ofs,seg : word;
 | |
|   end;
 | |
|   texecblock = packed record
 | |
|     envseg    : word;
 | |
|     comtail   : realptr;
 | |
|     firstFCB  : realptr;
 | |
|     secondFCB : realptr;
 | |
|     iniStack  : realptr;
 | |
|     iniCSIP   : realptr;
 | |
|   end;
 | |
| var
 | |
|   current_dos_buffer_pos,
 | |
|   arg_ofs,
 | |
|   i,la_env,
 | |
|   la_p,la_c,la_e,
 | |
|   fcb1_la,fcb2_la : longint;
 | |
|   execblock       : texecblock;
 | |
|   c,p             : string;
 | |
| 
 | |
|   function paste_to_dos(src : string) : boolean;
 | |
|   var
 | |
|     c : array[0..255] of char;
 | |
|   begin
 | |
|      paste_to_dos:=false;
 | |
|      if current_dos_buffer_pos+length(src)+1>transfer_buffer+tb_size then
 | |
|       RunError(217);
 | |
|      move(src[1],c[0],length(src));
 | |
|      c[length(src)]:=#0;
 | |
|      seg_move(get_ds,longint(@c),dosmemselector,current_dos_buffer_pos,length(src)+1);
 | |
|      current_dos_buffer_pos:=current_dos_buffer_pos+length(src)+1;
 | |
|      paste_to_dos:=true;
 | |
|   end;
 | |
| 
 | |
| begin
 | |
| { create command line }
 | |
|   move(comline[0],c[1],length(comline)+1);
 | |
|   c[length(comline)+2]:=#13;
 | |
|   c[0]:=char(length(comline)+2);
 | |
| { create path }
 | |
|   p:=path;
 | |
|   for i:=1 to length(p) do
 | |
|    if p[i]='/' then
 | |
|     p[i]:='\';
 | |
|   if LFNSupport then
 | |
|     GetShortName(p);
 | |
| { create buffer }
 | |
|   la_env:=transfer_buffer;
 | |
|   while (la_env and 15)<>0 do
 | |
|    inc(la_env);
 | |
|   current_dos_buffer_pos:=la_env;
 | |
| { copy environment }
 | |
|   for i:=1 to envcount do
 | |
|    paste_to_dos(envstr(i));
 | |
|   paste_to_dos(''); { adds a double zero at the end }
 | |
| { allow slash as backslash }
 | |
|   la_p:=current_dos_buffer_pos;
 | |
|   paste_to_dos(p);
 | |
|   la_c:=current_dos_buffer_pos;
 | |
|   paste_to_dos(c);
 | |
|   la_e:=current_dos_buffer_pos;
 | |
|   fcb1_la:=la_e;
 | |
|   la_e:=la_e+16;
 | |
|   fcb2_la:=la_e;
 | |
|   la_e:=la_e+16;
 | |
| { allocate FCB see dosexec code }
 | |
|   arg_ofs:=1;
 | |
|   while (c[arg_ofs] in [' ',#9]) do
 | |
|    inc(arg_ofs);
 | |
|   dosregs.ax:=$2901;
 | |
|   dosregs.ds:=(la_c+arg_ofs) shr 4;
 | |
|   dosregs.esi:=(la_c+arg_ofs) and 15;
 | |
|   dosregs.es:=fcb1_la shr 4;
 | |
|   dosregs.edi:=fcb1_la and 15;
 | |
|   msdos(dosregs);
 | |
| { allocate second FCB see dosexec code }
 | |
|   repeat
 | |
|     inc(arg_ofs);
 | |
|   until (c[arg_ofs] in [' ',#9,#13]);
 | |
|   if c[arg_ofs]<>#13 then
 | |
|    begin
 | |
|      repeat
 | |
|        inc(arg_ofs);
 | |
|      until not (c[arg_ofs] in [' ',#9]);
 | |
|    end;
 | |
|   dosregs.ax:=$2901;
 | |
|   dosregs.ds:=(la_c+arg_ofs) shr 4;
 | |
|   dosregs.si:=(la_c+arg_ofs) and 15;
 | |
|   dosregs.es:=fcb2_la shr 4;
 | |
|   dosregs.di:=fcb2_la and 15;
 | |
|   msdos(dosregs);
 | |
|   with execblock do
 | |
|    begin
 | |
|      envseg:=la_env shr 4;
 | |
|      comtail.seg:=la_c shr 4;
 | |
|      comtail.ofs:=la_c and 15;
 | |
|      firstFCB.seg:=fcb1_la shr 4;
 | |
|      firstFCB.ofs:=fcb1_la and 15;
 | |
|      secondFCB.seg:=fcb2_la shr 4;
 | |
|      secondFCB.ofs:=fcb2_la and 15;
 | |
|    end;
 | |
|   seg_move(get_ds,longint(@execblock),dosmemselector,la_e,sizeof(texecblock));
 | |
|   dosregs.edx:=la_p and 15;
 | |
|   dosregs.ds:=la_p shr 4;
 | |
|   dosregs.ebx:=la_e and 15;
 | |
|   dosregs.es:=la_e shr 4;
 | |
|   dosregs.ax:=$4b00;
 | |
|   msdos(dosregs);
 | |
|   LoadDosError;
 | |
|   if DosError=0 then
 | |
|    begin
 | |
|      dosregs.ax:=$4d00;
 | |
|      msdos(dosregs);
 | |
|      LastDosExitCode:=DosRegs.al
 | |
|    end
 | |
|   else
 | |
|    LastDosExitCode:=0;
 | |
| end;
 | |
| 
 | |
| 
 | |
| function dosexitcode : word;
 | |
| begin
 | |
|   dosexitcode:=lastdosexitcode;
 | |
| end;
 | |
| 
 | |
| 
 | |
| procedure getcbreak(var breakvalue : boolean);
 | |
| begin
 | |
|   DosError:=0;
 | |
|   dosregs.ax:=$3300;
 | |
|   msdos(dosregs);
 | |
|   breakvalue:=dosregs.dl<>0;
 | |
| end;
 | |
| 
 | |
| 
 | |
| procedure setcbreak(breakvalue : boolean);
 | |
| begin
 | |
|   DosError:=0;
 | |
|   dosregs.ax:=$3301;
 | |
|   dosregs.dl:=ord(breakvalue);
 | |
|   msdos(dosregs);
 | |
| end;
 | |
| 
 | |
| 
 | |
| procedure getverify(var verify : boolean);
 | |
| begin
 | |
|   DosError:=0;
 | |
|   dosregs.ah:=$54;
 | |
|   msdos(dosregs);
 | |
|   verify:=dosregs.al<>0;
 | |
| end;
 | |
| 
 | |
| 
 | |
| procedure setverify(verify : boolean);
 | |
| begin
 | |
|   DosError:=0;
 | |
|   dosregs.ah:=$2e;
 | |
|   dosregs.al:=ord(verify);
 | |
|   msdos(dosregs);
 | |
| end;
 | |
| 
 | |
| 
 | |
| {******************************************************************************
 | |
|                                --- Disk ---
 | |
| ******************************************************************************}
 | |
| 
 | |
| {$ifdef Int64}
 | |
| 
 | |
| TYPE  ExtendedFat32FreeSpaceRec=packed Record
 | |
|          RetSize           : WORD; { (ret) size of returned structure}
 | |
|          Strucversion      : WORD; {(call) structure version (0000h)
 | |
|                                     (ret) actual structure version (0000h)}
 | |
|          SecPerClus,               {number of sectors per cluster}
 | |
|          BytePerSec,               {number of bytes per sector}
 | |
|          AvailClusters,            {number of available clusters}
 | |
|          TotalClusters,            {total number of clusters on the drive}
 | |
|          AvailPhysSect,            {physical sectors available on the drive}
 | |
|          TotalPhysSect,            {total physical sectors on the drive}
 | |
|          AvailAllocUnits,          {Available allocation units}
 | |
|          TotalAllocUnits : DWORD;  {Total allocation units}
 | |
|          Dummy,Dummy2    : DWORD;  {8 bytes reserved}
 | |
|          END;
 | |
| 
 | |
| function do_diskdata(drive : byte; Free : BOOLEAN) : Int64;
 | |
| 
 | |
| VAR S    : String;
 | |
|     Rec  : ExtendedFat32FreeSpaceRec;
 | |
| 
 | |
| BEGIN
 | |
|  if (swap(dosversion)>=$070A) AND LFNSupport then
 | |
|   begin
 | |
|    DosError:=0;
 | |
|    S:='C:\'#0;
 | |
|    if Drive=0 then
 | |
|     begin
 | |
|      GetDir(Drive,S);
 | |
|      Setlength(S,4);
 | |
|      S[4]:=#0;
 | |
|     end
 | |
|    else
 | |
|     S[1]:=chr(Drive+64);
 | |
|    Rec.Strucversion:=0;
 | |
|    dosmemput(tb_segment,tb_offset,Rec,SIZEOF(ExtendedFat32FreeSpaceRec));
 | |
|    dosmemput(tb_segment,tb_offset+Sizeof(ExtendedFat32FreeSpaceRec)+1,S[1],4);
 | |
|    dosregs.dx:=tb_offset+Sizeof(ExtendedFat32FreeSpaceRec)+1;
 | |
|    dosregs.ds:=tb_segment;
 | |
|    dosregs.di:=tb_offset;
 | |
|    dosregs.es:=tb_segment;
 | |
|    dosregs.cx:=Sizeof(ExtendedFat32FreeSpaceRec);
 | |
|    dosregs.ax:=$7303;
 | |
|    msdos(dosregs);
 | |
|    LoadDosError;
 | |
|    copyfromdos(rec,Sizeof(ExtendedFat32FreeSpaceRec));
 | |
|    if Free then
 | |
|     Do_DiskData:=int64(rec.AvailAllocUnits)*rec.SecPerClus*rec.BytePerSec
 | |
|    else
 | |
|     Do_DiskData:=int64(rec.TotalAllocUnits)*rec.SecPerClus*rec.BytePerSec;
 | |
|    if doserror<>0 THEN {No error clausule in int except cf}
 | |
|     Do_DiskData:=-1;
 | |
|   end
 | |
|  else
 | |
|   begin
 | |
|    DosError:=0;
 | |
|    dosregs.dl:=drive;
 | |
|    dosregs.ah:=$36;
 | |
|    msdos(dosregs);
 | |
|    if dosregs.ax<>$FFFF then
 | |
|     begin
 | |
|      if Free then
 | |
|       Do_DiskData:=int64(dosregs.ax)*dosregs.bx*dosregs.cx
 | |
|      else
 | |
|       Do_DiskData:=int64(dosregs.ax)*dosregs.cx*dosregs.dx;
 | |
|     end
 | |
|    else
 | |
|     do_diskdata:=-1;
 | |
|   end;
 | |
| 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;
 | |
| {$else}
 | |
| 
 | |
| function diskfree(drive : byte) : longint;
 | |
| begin
 | |
|   DosError:=0;
 | |
|   dosregs.dl:=drive;
 | |
|   dosregs.ah:=$36;
 | |
|   msdos(dosregs);
 | |
|   if dosregs.ax<>$FFFF then
 | |
|    diskfree:=dosregs.ax*dosregs.bx*dosregs.cx
 | |
|   else
 | |
|    diskfree:=-1;
 | |
| end;
 | |
| 
 | |
| 
 | |
| function disksize(drive : byte) : longint;
 | |
| begin
 | |
|   DosError:=0;
 | |
|   dosregs.dl:=drive;
 | |
|   dosregs.ah:=$36;
 | |
|   msdos(dosregs);
 | |
|   if dosregs.ax<>$FFFF then
 | |
|    disksize:=dosregs.ax*dosregs.cx*dosregs.dx
 | |
|   else
 | |
|    disksize:=-1;
 | |
| end;
 | |
| 
 | |
| {$endif}
 | |
| 
 | |
| 
 | |
| {******************************************************************************
 | |
|                       --- 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);
 | |
| var
 | |
|   Len : longint;
 | |
| 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;
 | |
|      Move(hdl,d.Fill,4);
 | |
|    end;
 | |
| end;
 | |
| 
 | |
| 
 | |
| procedure LFNFindFirst(path:pchar;attr:longint;var s:searchrec);
 | |
| var
 | |
|   i : longint;
 | |
|   w : LFNSearchRec;
 | |
| begin
 | |
|   { allow slash as backslash }
 | |
|   for i:=0 to strlen(path) do
 | |
|     if path[i]='/' then path[i]:='\';
 | |
|   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.ecx:=8
 | |
|   else
 | |
|    dosregs.ecx:=attr and (not 8);
 | |
|   dosregs.edx:=tb_offset+Sizeof(LFNSearchrec)+1;
 | |
|   dosmemput(tb_segment,tb_offset+Sizeof(LFNSearchrec)+1,path^,strlen(path)+1);
 | |
|   dosregs.ds:=tb_segment;
 | |
|   dosregs.edi:=tb_offset;
 | |
|   dosregs.es:=tb_segment;
 | |
|   dosregs.ax:=$714e;
 | |
|   msdos(dosregs);
 | |
|   LoadDosError;
 | |
|   copyfromdos(w,sizeof(LFNSearchRec));
 | |
|   LFNSearchRec2Dos(w,dosregs.ax,s);
 | |
| end;
 | |
| 
 | |
| 
 | |
| procedure LFNFindNext(var s:searchrec);
 | |
| var
 | |
|   hdl : longint;
 | |
|   w   : LFNSearchRec;
 | |
| begin
 | |
|   Move(s.Fill,hdl,4);
 | |
|   dosregs.si:=1; { use ms-dos time }
 | |
|   dosregs.edi:=tb_offset;
 | |
|   dosregs.es:=tb_segment;
 | |
|   dosregs.ebx:=hdl;
 | |
|   dosregs.ax:=$714f;
 | |
|   msdos(dosregs);
 | |
|   LoadDosError;
 | |
|   copyfromdos(w,sizeof(LFNSearchRec));
 | |
|   LFNSearchRec2Dos(w,hdl,s);
 | |
| end;
 | |
| 
 | |
| 
 | |
| procedure LFNFindClose(var s:searchrec);
 | |
| var
 | |
|   hdl : longint;
 | |
| begin
 | |
|   Move(s.Fill,hdl,4);
 | |
|   dosregs.ebx:=hdl;
 | |
|   dosregs.ax:=$71a1;
 | |
|   msdos(dosregs);
 | |
|   LoadDosError;
 | |
| end;
 | |
| 
 | |
| 
 | |
| {******************************************************************************
 | |
|                      --- DosFindfirst DosFindNext ---
 | |
| ******************************************************************************}
 | |
| 
 | |
| procedure dossearchrec2searchrec(var f : searchrec);
 | |
| var
 | |
|   len : longint;
 | |
| begin
 | |
|   len:=StrLen(@f.Name);
 | |
|   Move(f.Name[0],f.Name[1],Len);
 | |
|   f.Name[0]:=chr(len);
 | |
| end;
 | |
| 
 | |
| 
 | |
| procedure DosFindfirst(path : pchar;attr : word;var f : searchrec);
 | |
| var
 | |
|    i : longint;
 | |
| begin
 | |
|   { allow slash as backslash }
 | |
|   for i:=0 to strlen(path) do
 | |
|     if path[i]='/' then path[i]:='\';
 | |
|   copytodos(f,sizeof(searchrec));
 | |
|   dosregs.edx:=tb_offset;
 | |
|   dosregs.ds:=tb_segment;
 | |
|   dosregs.ah:=$1a;
 | |
|   msdos(dosregs);
 | |
|   dosregs.ecx:=attr;
 | |
|   dosregs.edx:=tb_offset+Sizeof(searchrec)+1;
 | |
|   dosmemput(tb_segment,tb_offset+Sizeof(searchrec)+1,path^,strlen(path)+1);
 | |
|   dosregs.ds:=tb_segment;
 | |
|   dosregs.ah:=$4e;
 | |
|   msdos(dosregs);
 | |
|   copyfromdos(f,sizeof(searchrec));
 | |
|   LoadDosError;
 | |
|   dossearchrec2searchrec(f);
 | |
| end;
 | |
| 
 | |
| 
 | |
| procedure Dosfindnext(var f : searchrec);
 | |
| begin
 | |
|   copytodos(f,sizeof(searchrec));
 | |
|   dosregs.edx:=tb_offset;
 | |
|   dosregs.ds:=tb_segment;
 | |
|   dosregs.ah:=$1a;
 | |
|   msdos(dosregs);
 | |
|   dosregs.ah:=$4f;
 | |
|   msdos(dosregs);
 | |
|   copyfromdos(f,sizeof(searchrec));
 | |
|   LoadDosError;
 | |
|   dossearchrec2searchrec(f);
 | |
| end;
 | |
| 
 | |
| 
 | |
| {******************************************************************************
 | |
|                      --- Findfirst FindNext ---
 | |
| ******************************************************************************}
 | |
| 
 | |
| procedure findfirst(const path : pathstr;attr : word;var f : searchRec);
 | |
| var
 | |
|   path0 : array[0..256] of char;
 | |
| 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;
 | |
| 
 | |
| 
 | |
| type swap_proc = procedure;
 | |
| 
 | |
| var
 | |
|   _swap_in  : swap_proc;external name '_swap_in';
 | |
|   _swap_out : swap_proc;external name '_swap_out';
 | |
|   _exception_exit : pointer;external name '_exception_exit';
 | |
|   _v2prt0_exceptions_on : longbool;external name '_v2prt0_exceptions_on';
 | |
| 
 | |
| procedure swapvectors;
 | |
| begin
 | |
|   { DosError:=0; Who added this !!!!! }
 | |
|   if _exception_exit<>nil then
 | |
|     if _v2prt0_exceptions_on then
 | |
|       _swap_in()
 | |
|     else
 | |
|       _swap_out();
 | |
| end;
 | |
| 
 | |
| 
 | |
| {******************************************************************************
 | |
|                                --- File ---
 | |
| ******************************************************************************}
 | |
| 
 | |
| procedure fsplit(path : pathstr;var dir : dirstr;var name : namestr;var ext : extstr);
 | |
| var
 | |
|    dotpos,p1,i : longint;
 | |
| begin
 | |
|   { allow slash as backslash }
 | |
|   for i:=1 to length(path) do
 | |
|    if path[i]='/' then path[i]:='\';
 | |
|   { get drive name }
 | |
|   p1:=pos(':',path);
 | |
|   if p1>0 then
 | |
|     begin
 | |
|        dir:=path[1]+':';
 | |
|        delete(path,1,p1);
 | |
|     end
 | |
|   else
 | |
|     dir:='';
 | |
|   { split the path and the name, there are no more path informtions }
 | |
|   { if path contains no backslashes                                 }
 | |
|   while true do
 | |
|     begin
 | |
|        p1:=pos('\',path);
 | |
|        if p1=0 then
 | |
|          break;
 | |
|        dir:=dir+copy(path,1,p1);
 | |
|        delete(path,1,p1);
 | |
|     end;
 | |
|   { try to find out a extension }
 | |
|   if LFNSupport then
 | |
|     begin
 | |
|        Ext:='';
 | |
|        i:=Length(Path);
 | |
|        DotPos:=256;
 | |
|        While (i>0) Do
 | |
|          Begin
 | |
|             If (Path[i]='.') Then
 | |
|               begin
 | |
|                  DotPos:=i;
 | |
|                  break;
 | |
|               end;
 | |
|             Dec(i);
 | |
|          end;
 | |
|        Ext:=Copy(Path,DotPos,255);
 | |
|        Name:=Copy(Path,1,DotPos - 1);
 | |
|     end
 | |
|   else
 | |
|     begin
 | |
|        p1:=pos('.',path);
 | |
|        if p1>0 then
 | |
|          begin
 | |
|             ext:=copy(path,p1,4);
 | |
|             delete(path,p1,length(path)-p1+1);
 | |
|          end
 | |
|        else
 | |
|          ext:='';
 | |
|        name:=path;
 | |
|     end;
 | |
| end;
 | |
| 
 | |
| 
 | |
| function fexpand(const path : pathstr) : pathstr;
 | |
| var
 | |
|   s,pa : pathstr;
 | |
|   i,j  : longint;
 | |
| begin
 | |
|   getdir(0,s);
 | |
|   i:=ioresult;
 | |
|   if LFNSupport then
 | |
|    begin
 | |
|      pa:=path;
 | |
|    end
 | |
|   else
 | |
|    if FileNameCaseSensitive then
 | |
|     pa:=path
 | |
|    else
 | |
|     pa:=upcase(path);
 | |
| 
 | |
|   { allow slash as backslash }
 | |
|   for i:=1 to length(pa) do
 | |
|    if pa[i]='/' then
 | |
|     pa[i]:='\';
 | |
| 
 | |
|   if (length(pa)>1) and (pa[2]=':') and (pa[1] in ['A'..'Z','a'..'z']) then
 | |
|     begin
 | |
|        { Always uppercase driveletter }
 | |
|        if (pa[1] in ['a'..'z']) then
 | |
|         pa[1]:=Chr(Ord(Pa[1])-32);
 | |
|        { we must get the right directory }
 | |
|        getdir(ord(pa[1])-ord('A')+1,s);
 | |
|        i:=ioresult;
 | |
|        if (ord(pa[0])>2) and (pa[3]<>'\') then
 | |
|          if pa[1]=s[1] then
 | |
|            begin
 | |
|              { remove ending slash if it already exists }
 | |
|              if s[length(s)]='\' then
 | |
|               dec(s[0]);
 | |
|              pa:=s+'\'+copy (pa,3,length(pa));
 | |
|            end
 | |
|          else
 | |
|            pa:=pa[1]+':\'+copy (pa,3,length(pa))
 | |
|     end
 | |
|   else
 | |
|     if pa[1]='\' then
 | |
|       pa:=s[1]+':'+pa
 | |
|     else if s[0]=#3 then
 | |
|       pa:=s+pa
 | |
|     else
 | |
|       pa:=s+'\'+pa;
 | |
| 
 | |
| { Turbo Pascal gives current dir on drive if only drive given as parameter! }
 | |
| if length(pa) = 2 then
 | |
|  begin
 | |
|    getdir(byte(pa[1])-64,s);
 | |
|    pa := s;
 | |
|  end;
 | |
| 
 | |
| {First remove all references to '\.\'}
 | |
|   while pos ('\.\',pa)<>0 do
 | |
|    delete (pa,pos('\.\',pa),2);
 | |
| {Now remove also all references to '\..\' + of course previous dirs..}
 | |
|   repeat
 | |
|     i:=pos('\..\',pa);
 | |
|     if i<>0 then
 | |
|      begin
 | |
|        j:=i-1;
 | |
|        while (j>1) and (pa[j]<>'\') do
 | |
|         dec (j);
 | |
|        if pa[j+1] = ':' then j := 3;
 | |
|        delete (pa,j,i-j+3);
 | |
|      end;
 | |
|   until i=0;
 | |
| 
 | |
|   { Turbo Pascal gets rid of a \.. at the end of the path }
 | |
|   { Now remove also any reference to '\..'  at end of line
 | |
|     + of course previous dir.. }
 | |
|   i:=pos('\..',pa);
 | |
|   if i<>0 then
 | |
|    begin
 | |
|      if i = length(pa) - 2 then
 | |
|       begin
 | |
|         j:=i-1;
 | |
|         while (j>1) and (pa[j]<>'\') do
 | |
|          dec (j);
 | |
|         delete (pa,j,i-j+3);
 | |
|       end;
 | |
|       pa := pa + '\';
 | |
|     end;
 | |
|   { Remove End . and \}
 | |
|   if (length(pa)>0) and (pa[length(pa)]='.') then
 | |
|    dec(byte(pa[0]));
 | |
|   { if only the drive + a '\' is left then the '\' should be left to prevtn the program
 | |
|     accessing the current directory on the drive rather than the root!}
 | |
|   { if the last char of path = '\' then leave it in as this is what TP does! }
 | |
|   if ((length(pa)>3) and (pa[length(pa)]='\')) and (path[length(path)] <> '\') then
 | |
|    dec(byte(pa[0]));
 | |
|   { if only a drive is given in path then there should be a '\' at the
 | |
|     end of the string given back }
 | |
|   if length(pa) = 2 then pa := pa + '\';
 | |
|   fexpand:=pa;
 | |
| end;
 | |
| 
 | |
| 
 | |
| Function FSearch(path: pathstr; dirlist: string): pathstr;
 | |
| var
 | |
|   i,p1   : longint;
 | |
|   s      : searchrec;
 | |
|   newdir : pathstr;
 | |
| begin
 | |
| { check if the file specified exists }
 | |
|   findfirst(path,anyfile,s);
 | |
|   if doserror=0 then
 | |
|    begin
 | |
|      findclose(s);
 | |
|      fsearch:=path;
 | |
|      exit;
 | |
|    end;
 | |
| { No wildcards allowed in these things }
 | |
|   if (pos('?',path)<>0) or (pos('*',path)<>0) then
 | |
|     fsearch:=''
 | |
|   else
 | |
|     begin
 | |
|        { allow slash as backslash }
 | |
|        for i:=1 to length(dirlist) do
 | |
|          if dirlist[i]='/' then dirlist[i]:='\';
 | |
|        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 ['\',':'])) then
 | |
|           newdir:=newdir+'\';
 | |
|          findfirst(newdir+path,anyfile,s);
 | |
|          if doserror=0 then
 | |
|           newdir:=newdir+path
 | |
|          else
 | |
|           newdir:='';
 | |
|        until (dirlist='') or (newdir<>'');
 | |
|        fsearch:=newdir;
 | |
|     end;
 | |
|   findclose(s);
 | |
| end;
 | |
| 
 | |
| 
 | |
| { change to short filename if successful DOS call PM }
 | |
| function GetShortName(var p : String) : boolean;
 | |
| var
 | |
|   c : array[0..255] of char;
 | |
| begin
 | |
|   move(p[1],c[0],length(p));
 | |
|   c[length(p)]:=#0;
 | |
|   copytodos(@c,length(p)+1);
 | |
|   dosregs.ax:=$7160;
 | |
|   dosregs.cx:=1;
 | |
|   dosregs.ds:=tb_segment;
 | |
|   dosregs.si:=tb_offset;
 | |
|   dosregs.es:=tb_segment;
 | |
|   dosregs.di:=tb_offset;
 | |
|   msdos(dosregs);
 | |
|   LoadDosError;
 | |
|   if DosError=0 then
 | |
|    begin
 | |
|      copyfromdos(@c,255);
 | |
|      move(c[0],p[1],strlen(c));
 | |
|      p[0]:=char(strlen(c));
 | |
|      GetShortName:=true;
 | |
|    end
 | |
|   else
 | |
|    GetShortName:=false;
 | |
| end;
 | |
| 
 | |
| 
 | |
| {******************************************************************************
 | |
|                        --- Get/Set File Time,Attr ---
 | |
| ******************************************************************************}
 | |
| 
 | |
| procedure getftime(var f;var time : longint);
 | |
| begin
 | |
|   dosregs.bx:=textrec(f).handle;
 | |
|   dosregs.ax:=$5700;
 | |
|   msdos(dosregs);
 | |
|   loaddoserror;
 | |
|   time:=(dosregs.dx shl 16)+dosregs.cx;
 | |
| end;
 | |
| 
 | |
| 
 | |
| procedure setftime(var f;time : longint);
 | |
| begin
 | |
|   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);
 | |
| begin
 | |
|   copytodos(filerec(f).name,strlen(filerec(f).name)+1);
 | |
|   dosregs.edx:=tb_offset;
 | |
|   dosregs.ds:=tb_segment;
 | |
|   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);
 | |
| begin
 | |
|   copytodos(filerec(f).name,strlen(filerec(f).name)+1);
 | |
|   dosregs.edx:=tb_offset;
 | |
|   dosregs.ds:=tb_segment;
 | |
|   if LFNSupport then
 | |
|    begin
 | |
|      dosregs.ax:=$7143;
 | |
|      dosregs.bx:=1;
 | |
|    end
 | |
|   else
 | |
|    dosregs.ax:=$4301;
 | |
|   dosregs.cx:=attr;
 | |
|   msdos(dosregs);
 | |
|   LoadDosError;
 | |
| end;
 | |
| 
 | |
| 
 | |
| {******************************************************************************
 | |
|                              --- Environment ---
 | |
| ******************************************************************************}
 | |
| 
 | |
| function envcount : longint;
 | |
| var
 | |
|   hp : ppchar;
 | |
| begin
 | |
|   hp:=envp;
 | |
|   envcount:=0;
 | |
|   while assigned(hp^) do
 | |
|    begin
 | |
|      inc(envcount);
 | |
|      inc(hp);
 | |
|    end;
 | |
| end;
 | |
| 
 | |
| 
 | |
| function envstr(index : integer) : string;
 | |
| begin
 | |
|   if (index<=0) or (index>envcount) then
 | |
|    begin
 | |
|      envstr:='';
 | |
|      exit;
 | |
|    end;
 | |
|   envstr:=strpas(ppchar(pointer(envp)+4*(index-1))^);
 | |
| end;
 | |
| 
 | |
| 
 | |
| Function  GetEnv(envvar: string): string;
 | |
| var
 | |
|   hp      : ppchar;
 | |
|   hs    : string;
 | |
|   eqpos : longint;
 | |
| begin
 | |
|   envvar:=upcase(envvar);
 | |
|   hp:=envp;
 | |
|   getenv:='';
 | |
|   while assigned(hp^) do
 | |
|    begin
 | |
|      hs:=strpas(hp^);
 | |
|      eqpos:=pos('=',hs);
 | |
|      if copy(hs,1,eqpos-1)=envvar then
 | |
|       begin
 | |
|         getenv:=copy(hs,eqpos+1,255);
 | |
|         exit;
 | |
|       end;
 | |
|      inc(hp);
 | |
|    end;
 | |
| end;
 | |
| 
 | |
| 
 | |
| {******************************************************************************
 | |
|                              --- Not Supported ---
 | |
| ******************************************************************************}
 | |
| 
 | |
| Procedure keep(exitcode : word);
 | |
| Begin
 | |
| End;
 | |
| 
 | |
| Procedure getintvec(intno : byte;var vector : pointer);
 | |
| Begin
 | |
| End;
 | |
| 
 | |
| Procedure setintvec(intno : byte;vector : pointer);
 | |
| Begin
 | |
| End;
 | |
| 
 | |
| 
 | |
| end.
 | |
| {
 | |
|   $Log$
 | |
|   Revision 1.20  2000-02-02 17:34:49  pierre
 | |
|    * use int64 typecast to avoid overflows in diskfree and disksize
 | |
| 
 | |
|   Revision 1.19  2000/01/23 16:31:23  peter
 | |
|     * hasint64diskspace define changed to int64 so it's default now
 | |
| 
 | |
|   Revision 1.18  2000/01/23 12:28:38  marco
 | |
|    * Added diskfree and disksize with AH=71 dos functions (LFN/Fat32)
 | |
| 
 | |
|   Revision 1.17  2000/01/07 16:41:30  daniel
 | |
|     * copyright 2000
 | |
| 
 | |
|   Revision 1.16  2000/01/07 16:32:23  daniel
 | |
|     * copyright 2000 added
 | |
| 
 | |
|   Revision 1.15  1999/12/06 18:26:49  peter
 | |
|     * fpcmake updated for win32 commandline
 | |
| 
 | |
|   Revision 1.14  1999/11/09 11:07:50  pierre
 | |
|     * SwapVectors does not reset DosError anymore
 | |
|     + DosError is set to ax regsiter value if extended doserror function
 | |
|       retruns zero.
 | |
|     + Support for LFN in EXEC function using
 | |
|       function 7160 to get short filename counterpart
 | |
| 
 | |
|   Revision 1.13  1999/11/06 14:38:23  peter
 | |
|     * truncated log
 | |
| 
 | |
|   Revision 1.12  1999/09/10 17:14:09  peter
 | |
|     * better errorcode returning using int21h,5900
 | |
| 
 | |
|   Revision 1.11  1999/09/08 18:55:49  peter
 | |
|     * pointer fixes
 | |
| 
 | |
|   Revision 1.10  1999/08/13 21:23:15  peter
 | |
|     * fsearch checks first if the specified file exists and returns that
 | |
|       if it was found
 | |
| 
 | |
|   Revision 1.9  1999/05/16 17:08:58  peter
 | |
|     * fixed driveletter checking
 | |
| 
 | |
|   Revision 1.8  1999/05/08 19:47:22  peter
 | |
|     * check ioresult after getdir calls
 | |
| 
 | |
|   Revision 1.7  1999/05/04 23:55:50  pierre
 | |
|    * unneeded assembler code converted to pascal
 | |
| 
 | |
|   Revision 1.6  1999/04/28 11:42:44  peter
 | |
|     + FileNameCaseSensetive boolean
 | |
| 
 | |
|   Revision 1.5  1999/04/02 00:01:29  peter
 | |
|     * fixed LFNFindfirst on network drives
 | |
| 
 | |
|   Revision 1.4  1999/03/01 15:40:48  peter
 | |
|     * use external names
 | |
|     * removed all direct assembler modes
 | |
| 
 | |
|   Revision 1.3  1999/01/22 15:44:59  pierre
 | |
|    Daniel change removed : broke make cycle !!
 | |
| 
 | |
|   Revision 1.2  1999/01/22 10:07:03  daniel
 | |
|   - Findclose removed: This is TP incompatible!!
 | |
| 
 | |
|   Revision 1.1  1998/12/21 13:07:02  peter
 | |
|     * use -FE
 | |
| 
 | |
|   Revision 1.19  1998/11/23 13:53:59  peter
 | |
|     * more fexpand fixes from marco van de voort
 | |
| 
 | |
|   Revision 1.18  1998/11/23 12:48:02  peter
 | |
|     * fexpand('o:') fixed to return o:\ (from the mailinglist)
 | |
| 
 | |
|   Revision 1.17  1998/11/22 09:33:21  florian
 | |
|     * fexpand bug (temp. strings were too shoort) fixed, was reported
 | |
|       by Marco van de Voort
 | |
| 
 | |
|   Revision 1.16  1998/11/17 09:37:41  pierre
 | |
|    * explicit conversion from word dosreg.ax to integer doserror
 | |
| 
 | |
|   Revision 1.15  1998/11/01 20:27:18  peter
 | |
|     * fixed some doserror settings
 | |
| 
 | |
|   Revision 1.14  1998/10/22 15:05:28  pierre
 | |
|    * fsplit adapted to long filenames
 | |
| 
 | |
|   Revision 1.13  1998/09/16 16:47:24  peter
 | |
|     * merged fixes
 | |
| 
 | |
|   Revision 1.11.2.2  1998/09/16 16:16:04  peter
 | |
|     * go32v1 compiles again
 | |
| 
 | |
| }
 | |
| 
 | |
| 
 | 
