| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248 | {    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{$IFDEF FPC_DOTTEDUNITS}Uses  DOSApi.GO32;{$ELSE FPC_DOTTEDUNITS}Uses  Go32;{$ENDIF FPC_DOTTEDUNITS}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}  Registers = Go32.Registers;{$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);implementation{$IFDEF FPC_DOTTEDUNITS}uses  System.Strings;{$ELSE FPC_DOTTEDUNITS}uses  strings;{$ENDIF FPC_DOTTEDUNITS}{$DEFINE HAS_GETMSCOUNT}{$DEFINE HAS_INTR}{$DEFINE HAS_SETCBREAK}{$DEFINE HAS_GETCBREAK}{$DEFINE HAS_SETVERIFY}{$DEFINE HAS_GETVERIFY}{$DEFINE HAS_SWAPVECTORS}{$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 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;     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;{******************************************************************************                        --- 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);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;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);end;function GetMsCount: int64;begin  GetMsCount := int64 (MemL [$40:$6c]) * 55;end;{******************************************************************************                               --- Exec ---******************************************************************************}const  DOS_MAX_COMMAND_LINE_LENGTH = 126;procedure exec_ansistring(path : string;comline : ansistring);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;  use_proxy       : boolean;  proxy_argc      : longint;  ExecBufSize, TB : longint;  ExecBufPtr      : PAnsiChar;  execblock       : texecblock;  c               : ansistring;  p               : string;  function paste_to_dos(src : string;add_cr_at_end, include_string_length : boolean) : boolean;  {Changed by Laaca - added parameter N}  var{    c : PAnsiChar;}    CLen : cardinal;    start_pos,ls : longint;  begin     paste_to_dos:=false;     if include_string_length then       start_pos:=0     else       start_pos:=1;     ls:=Length(src)-start_pos;{     if current_dos_buffer_pos+ls+3>transfer_buffer+tb_size then}     if Current_Dos_Buffer_Pos + LS + 3 > ExecBufSize then     begin      FreeMem (ExecBufPtr);      RunError(217);     end;{     getmem(c,ls+3);}     Move (Src [Start_Pos], ExecBufPtr [Current_Dos_Buffer_Pos], LS + 1);     Inc (Current_Dos_Buffer_Pos, LS + 1);     if add_cr_at_end then      begin        ExecBufPtr [Current_Dos_Buffer_Pos] := #13;        Inc (Current_Dos_Buffer_Pos);      end;      ExecBufPtr [Current_Dos_Buffer_Pos] := #0;      Inc (Current_Dos_Buffer_Pos);{      CLen := StrLen (C) + 1;     seg_move(get_ds,longint(c),dosmemselector,current_dos_buffer_pos,CLen);     current_dos_buffer_pos:=current_dos_buffer_pos+CLen;     freemem(c,ls+3);}     paste_to_dos:=true;  end;  procedure setup_proxy_cmdline;  const    MAX_ARGS = 128;  var    i : longint;    quote : AnsiChar;    end_of_arg, skip_char : boolean;    la_proxy_seg    : word;    la_proxy_ofs    : longint;    current_arg : string;    la_argv_ofs : array [0..MAX_ARGS] of word;  begin    quote:=#0;    current_arg:='';    proxy_argc:=0;    end_of_arg:=false;    while (TB + current_dos_buffer_pos) mod 16 <> 0 do      inc(current_dos_buffer_pos);    la_proxy_seg:=(TB + current_dos_buffer_pos) shr 4;    { Also copy parameter 0 }    la_argv_ofs[0]:=TB+current_dos_buffer_pos-la_proxy_seg*16;    { Note that this should be done before      alteriing p value }    paste_to_dos(p,false,false);    inc(proxy_argc);    for i:=1 to length(c) do      begin        skip_char:=false;        case c[i] of          #1..#32:            begin              if quote=#0 then                end_of_arg:=true;            end;          '"' :            begin              if quote=#0 then                begin                  quote:='"';                  skip_char:=true;                end              else if quote='"' then                end_of_arg:=true;            end;          '''' :            begin              if quote=#0 then                begin                  quote:='''';                  skip_char:=true;                end              else if quote='''' then                end_of_arg:=true;            end;        end;        if not end_of_arg and not skip_char then          current_arg:=current_arg+c[i];        if i=length(c) then          end_of_arg:=true;        if end_of_arg then          begin            { Allow empty args using "" or '' }            if (current_arg<>'') or (quote<>#0) then              begin                if proxy_argc>MAX_ARGS then                  begin                    writeln(stderr,'Too many arguments in Dos.exec');                    RunError(217);                  end;                la_argv_ofs[proxy_argc]:=TB + current_dos_buffer_pos - la_proxy_seg*16;    {$ifdef DEBUG_PROXY}                writeln(stderr,'arg ',proxy_argc,'="',current_arg,'"');    {$endif DEBUG_PROXY}                paste_to_dos(current_arg,false,false);                inc(proxy_argc);                quote:=#0;                current_arg:='';              end;            { Always reset end_of_arg boolean }            end_of_arg:=false;          end;      end;    la_proxy_ofs:=TB + current_dos_buffer_pos - la_proxy_seg*16;{    seg_move(get_ds,longint(@la_argv_ofs),dosmemselector,             current_dos_buffer_pos,proxy_argc*sizeof(word));}    Move (LA_ArgV_Ofs, ExecBufPtr [Current_Dos_Buffer_Pos],                                                   Proxy_ArgC * SizeOf (word));    current_dos_buffer_pos:=current_dos_buffer_pos + proxy_argc*sizeof(word);    c:='!proxy '+hexstr(proxy_argc,4)+' '+hexstr(la_proxy_seg,4)       +' '+hexstr(la_proxy_ofs,4);{$ifdef DEBUG_PROXY}    writeln(stderr,'Using comline "',c,'"');{$endif DEBUG_PROXY}  end;begin{ create command line }  c:=comline;   use_proxy:=false;  if force_go32v2_proxy then    Use_proxy:=true  else if length(c)>DOS_MAX_COMMAND_LINE_LENGTH then    begin      if Use_go32v2_proxy then        begin          Use_Proxy:=true;        end      else        begin           writeln(stderr,'Dos.exec command line truncated to ',                   DOS_MAX_COMMAND_LINE_LENGTH,' chars');           writeln(stderr,'Before: "',c,'"');           setlength(c, DOS_MAX_COMMAND_LINE_LENGTH);           writeln(stderr,'After: "',c,'"');         end;    end;{ create path }{$ifdef DEBUG_PROXY}  writeln(stderr,'Dos.exec path="',path,'"');{$endif DEBUG_PROXY}  p:=path;  if LFNSupport then    GetShortName(p);{ create buffer }  TB := Transfer_Buffer;  ExecBufSize := TB_Size;  GetMem (ExecBufPtr, ExecBufSize);  if ExecBufPtr = nil then   begin    DosError := 8;    Exit;   end;  la_env:=TB;  while (la_env and 15)<>0 do   inc(la_env);  current_dos_buffer_pos:=la_env - TB;{ copy environment }  for i:=1 to envcount do   paste_to_dos(envstr(i),false,false);  {the behaviour is still suboptimal because variable COMMAND is stripped out}  paste_to_dos(chr(0),false,false); { adds a double zero at the end }  if use_proxy then    setup_proxy_cmdline;{ allow slash as backslash }  DoDirSeparators(p);  { Add program to DosBuffer with    length at start }  la_p:=TB + current_dos_buffer_pos;  paste_to_dos(p,false,true);  { Add command line args to DosBuffer with    length at start and Carriage Return at end }  la_c:=TB + current_dos_buffer_pos;  paste_to_dos(c,true,true);  la_e:=TB + current_dos_buffer_pos;  fcb1_la:=la_e;  la_e:=la_e+16;  fcb2_la:=la_e;  la_e:=la_e+16;{$ifdef DEBUG_PROXY}  flush(stderr);{$endif DEBUG_PROXY}  seg_move (get_ds, PtrInt (ExecBufPtr), DosMemSelector, TB, Pred (Current_Dos_Buffer_Pos));{ allocate FCB see dosexec code }  arg_ofs:=1;  while (arg_ofs<length(c)) and (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 }  dosregs.ax:=$2901;  dosregs.ds:=(la_c+arg_ofs) shr 4;  dosregs.esi:=(la_c+arg_ofs) and 15;  dosregs.es:=fcb2_la shr 4;  dosregs.edi:=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+1;  dosregs.ds:=la_p shr 4;  dosregs.ebx:=la_p and 15+la_e-la_p;  dosregs.es:=la_p 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;procedure exec(const path : pathstr;const comline : comstr);begin  exec_ansistring(path, comline);end;procedure getcbreak(var breakvalue : boolean);begin  dosregs.ax:=$3300;  msdos(dosregs);  breakvalue:=dosregs.dl<>0;end;procedure setcbreak(breakvalue : boolean);begin  dosregs.ax:=$3301;  dosregs.dl:=ord(breakvalue);  msdos(dosregs);end;procedure getverify(var verify : boolean);begin  dosregs.ah:=$54;  msdos(dosregs);  verify:=dosregs.al<>0;end;procedure setverify(verify : boolean);begin  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 fieldtype  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;    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;    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);    if (dosregs.flags and fcarry) <> 0 then      exit;    copyfromdos(rec,Sizeof(ExtendedFat32FreeSpaceRec));    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;      status,byteswritten : word;      drnum : byte;  begin    DiskData_CDROM:=false;    drnum:=drive-1; //for MSCDEX, 0 = a, 1 = b etc, unlike int21/36    { Is this a CDROM drive? }    dosregs.ax:=$150b;    dosregs.cx:=drnum;    realintr($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:=tb_offset+sizeof(req); //CDROM control block will follow    req.transf_seg:=tb_segment;            //the request header    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 }      dosmemput(tb_segment,tb_offset,req,sizeof(req));      dosmemput(tb_segment,tb_offset+sizeof(req),sectreq,sizeof(sectreq));      dosregs.ax:=$1510;      dosregs.cx:=drnum;      dosregs.es:=tb_segment;      dosregs.bx:=tb_offset;      realintr($2f,dosregs);      dosmemget(tb_segment,tb_offset+3,status,2);      { status = $800F means "disk changed". Try once more. }      if (status and $800F) <> $800F then break;    end;    dosmemget(tb_segment,tb_offset+$12,byteswritten,2);    if (status<>$0100) or (byteswritten<>sizeof(sectreq)) then      exit; //An error occurred    dosmemget(tb_segment,tb_offset+sizeof(req),sectreq,sizeof(sectreq));  { Update the request header for the next request }    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 }    dosmemput(tb_segment,tb_offset,req,sizeof(req));    dosmemput(tb_segment,tb_offset+sizeof(req),sizereq,sizeof(sizereq));    dosregs.ax:=$1510;    dosregs.cx:=drnum;    dosregs.es:=tb_segment;    dosregs.bx:=tb_offset;    realintr($2f,dosregs);    dosmemget(tb_segment,tb_offset,req,sizeof(req));    if (req.status<>$0100) or (req.numbytes<>sizeof(sizereq)) then      exit; //An error occurred    dosmemget(tb_segment,tb_offset+sizeof(req)+1,sizereq.size,4);    blocksize:=sectreq.secsize;    freeblocks:=0; //always 0 for a cdrom    totblocks:=sizereq.size;    DiskData_CDROM:=true;  end;begin  if drive=0 then  begin    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 : 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;     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  i : longint;  w : LFNSearchRec;begin  { allow slash as backslash }  DoDirSeparators(path);  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) and $FF; { no required attributes }  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;  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}  copyfromdos(w,sizeof(LFNSearchRec));  LFNSearchRec2Dos(w,dosregs.ax,s,true);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,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;  dosregs.ebx:=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 : longint;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);var   i : longint;begin  { allow slash as backslash }  DoDirSeparators(path);  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..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;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  if _exception_exit<>nil then    if _v2prt0_exceptions_on then      _swap_out()    else      _swap_in();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;  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,256);     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..255] of AnsiChar;begin  move(p[1],c[0],length(p));  c[length(p)]:=#0;  copytodos(c,length(p)+1);  dosregs.ax:=$7160;  dosregs.cx:=2;  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,256);     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  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);{$ifndef FPC_ANSI_TEXTFILEREC}var  r: rawbytestring;{$endif not FPC_ANSI_TEXTFILEREC}begin{$ifdef FPC_ANSI_TEXTFILEREC}  copytodos(filerec(f).name,strlen(filerec(f).name)+1);{$else}  r:=ToSingleByteFileSystemEncodedFileName(filerec(f).name);  copytodos(PAnsiChar(r)^,length(r)+1);{$endif}  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);{$ifndef FPC_ANSI_TEXTFILEREC}var  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}  copytodos(filerec(f).name,strlen(filerec(f).name)+1);{$else}  r:=ToSingleByteFileSystemEncodedFileName(filerec(f).name);  copytodos(PAnsiChar(r)^,length(r)+1);{$endif}  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 : PPAnsiChar;begin  hp:=envp;  envcount:=0;  while assigned(hp^) do   begin     inc(envcount);     inc(hp);   end;end;function envstr (Index: longint): string;begin  if (index<=0) or (index>envcount) then    envstr:=''  else    envstr:=strpas(PPAnsiChar(pointer(envp)+SizeOf(PAnsiChar)*(index-1))^);end;Function  GetEnv(envvar: string): string;var  hp    : PPAnsiChar;  hs    : string;  eqpos : longint;begin  envvar:=upcase(envvar);  hp:=envp;  getenv:='';  while assigned(hp^) do   begin     hs:=strpas(hp^);     eqpos:=pos('=',hs);     if upcase(copy(hs,1,eqpos-1))=envvar then      begin        getenv:=copy(hs,eqpos+1,length(hs)-eqpos);        break;      end;     inc(hp);   end;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.
 |