| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820 | {    This file is part of the Free Pascal run time library.    Copyright (c) 2001 by members of the Free Pascal    development team    DOS unit template based on POSIX    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{$goto on}Const  FileNameLen = 255;Type  SearchRec = packed Record  {Fill : array[1..21] of byte;  Fill replaced with below}    DirPtr     : pointer;        {directory pointer for reading directory}    SearchAttr : Byte;        {attribute we are searching for}    Fill       : Array[1..16] of Byte; {future use}  {End of fill}    Attr       : Byte;        {attribute of found file}    Time       : LongInt;     {last modify date of found file}    Size       : LongInt;     {file size of found file}    Reserved   : Word;        {future use}    Name       : String[FileNameLen]; {name of found file}    SearchSpec : String[FileNameLen]; {search pattern}    SearchDir  : String[FileNameLen]; { path we are searching in }  End;{$DEFINE HAS_FILENAMELEN}{$I dosh.inc}Procedure AddDisk(const path:string);ImplementationUses  strings,posix;(* Potentially needed FPC_FEXPAND_* defines should be defined here. *){$I dos.inc}  { Used by AddDisk(), DiskFree() and DiskSize() }const  Drives   : byte = 4;  MAX_DRIVES = 26;var  DriveStr : array[4..MAX_DRIVES] of pchar;Function StringToPPChar(Var S:STring; var count : longint):ppchar;{  Create a PPChar to structure of pchars which are the arguments specified  in the string S. Especially usefull for creating an ArgV for Exec-calls}var  nr  : longint;  Buf : ^char;  p   : ppchar;begin  s:=s+#0;  buf:=@s[1];  nr:=0;  while(buf^<>#0) do   begin     while (buf^ in [' ',#8,#10]) do      inc(buf);     inc(nr);     while not (buf^ in [' ',#0,#8,#10]) do      inc(buf);   end;  getmem(p,nr*4);  StringToPPChar:=p;  if p=nil then   begin     Errno:=sys_enomem;     count := 0;     exit;   end;  buf:=@s[1];  while (buf^<>#0) do   begin     while (buf^ in [' ',#8,#10]) do      begin        buf^:=#0;        inc(buf);      end;     p^:=buf;     inc(p);     p^:=nil;     while not (buf^ in [' ',#0,#8,#10]) do      inc(buf);   end;   count := nr;end;{$i dos_beos.inc}    { include OS specific stuff }{******************************************************************************                        --- Info / Date / Time ---******************************************************************************}var  TZSeconds : longint;   { offset to add/ subtract from Epoch to get local time }  tzdaylight : boolean;  tzname     : array[boolean] of pchar;type  GTRec = packed Record    Year,    Month,    MDay,    WDay,    Hour,    Minute,    Second : Word;  End;Const{Date Calculation}  C1970 = 2440588;  D0    = 1461;  D1    = 146097;  D2    = 1721119;function WeekDay (y,m,d:longint):longint;{  Calculates th day of the week. returns -1 on error}var  u,v : longint;begin  if (m<1) or (m>12) or (y<1600) or (y>4000) or     (d<1) or (d>30+((m+ord(m>7)) and 1)-ord(m=2)) or     ((m*d=58) and (((y mod 4>0) or (y mod 100=0)) and (y mod 400>0))) then   WeekDay:=-1  else   begin     u:=m;     v:=y;     if m<3 then      begin        inc(u,12);        dec(v);      end;     WeekDay:=(d+2*u+((3*(u+1)) div 5)+v+(v div 4)-(v div 100)+(v div 400)+1) mod 7;   end;end;Procedure JulianToGregorian(JulianDN:LongInt;Var Year,Month,Day:Word);Var  YYear,XYear,Temp,TempMonth : LongInt;Begin  Temp:=((JulianDN-D2) shl 2)-1;  JulianDN:=Temp Div D1;  XYear:=(Temp Mod D1) or 3;  YYear:=(XYear Div D0);  Temp:=((((XYear mod D0)+4) shr 2)*5)-3;  Day:=((Temp Mod 153)+5) Div 5;  TempMonth:=Temp Div 153;  If TempMonth>=10 Then   Begin     inc(YYear);     dec(TempMonth,12);   End;  inc(TempMonth,3);  Month := TempMonth;  Year:=YYear+(JulianDN*100);end;Procedure EpochToLocal(epoch:time_t;var year,month,day,hour,minute,second:Word);{  Transforms Epoch time into local time (hour, minute,seconds)}Var  DateNum: time_t;Begin  Epoch:=Epoch+TZSeconds;  Datenum:=(Epoch Div 86400) + c1970;  JulianToGregorian(DateNum,Year,Month,day);  Epoch:=Abs(Epoch Mod 86400);  Hour:=Epoch Div 3600;  Epoch:=Epoch Mod 3600;  Minute:=Epoch Div 60;  Second:=Epoch Mod 60;End;Procedure GetDate(Var Year, Month, MDay, WDay: Word);var  hour,minute,second : word;  timeval : time_t;Begin  timeval := sys_time(timeval);  { convert the GMT time to local time }  EpochToLocal(timeval,year,month,mday,hour,minute,second);  Wday:=weekday(Year,Month,MDay);end;Procedure SetDate(Year, Month, Day: Word);Begin  {!!}End;Procedure GetTime(Var Hour, Minute, Second, Sec100: Word);var timeval : time_t; year,month,day: word;Begin  timeval := sys_time(timeval);  EpochToLocal(timeval,year,month,day,hour,minute,second);  Sec100 := 0;end;Procedure SetTime(Hour, Minute, Second, Sec100: Word);Begin  {!!}End;Procedure UnixDateToDt(SecsPast: LongInt; Var Dt: DateTime);Begin  EpochToLocal(SecsPast,dt.Year,dt.Month,dt.Day,dt.Hour,dt.Min,dt.Sec);End;{$ifndef DOS_HAS_EXEC}{******************************************************************************                               --- Exec ---******************************************************************************}Function  InternalWaitProcess(Pid:pid_t):Longint; { like WaitPid(PID,@result,0) Handling of Signal interrupts (errno=EINTR), returning the Exitcode of Process (>=0) or -Status if terminated}var     r,s     : cint;begin  repeat    s:=$7F00;    r:=sys_WaitPid(Pid,s,0);  until (r<>-1) or (Errno<>Sys_EINTR);  { When r = -1 or r = 0, no status is available, so there was an error. }  if (r=-1) or (r=0) then    InternalWaitProcess:=-1 { return -1 to indicate an error }  else   begin     { process terminated normally }     if wifexited(s)<>0 then       begin         { get status code }         InternalWaitProcess := wexitstatus(s);         exit;       end;     { process terminated due to a signal }     if wifsignaled(s)<>0 then       begin         { get signal number }         InternalWaitProcess := wstopsig(s);         exit;       end;     InternalWaitProcess:=-1;   end;end;Procedure Exec (Const Path: PathStr; Const ComLine: ComStr);var  pid    : pid_t;  tmp : string;  p : ppchar;  count: longint;  // The Error-Checking in the previous Version failed, since halt($7F) gives an WaitPid-status of $7F00  F: File;Begin{$IFOPT I+}{$DEFINE IOCHECK}{$ENDIF}{$I-}  { verify if the file to execute exists }  Assign(F,Path);  Reset(F,1);  if IOResult <> 0 then    { file not found }    begin      DosError := 2;      exit;    end  else    Close(F);{$IFDEF IOCHECK}{$I+}{$UNDEF IOCHECK}{$ENDIF}  LastDosExitCode:=0;  { Fork the process }  pid:=sys_Fork;  if pid=0 then   begin   {The child does the actual execution, and then exits}    tmp := Path+' '+ComLine;    p:=StringToPPChar(tmp,count);    if (p<>nil) and (p^<>nil) then    begin      sys_Execve(p^,p,Envp);    end;   {If the execve fails, we return an exitvalue of 127, to let it be known}     sys_exit(127);   end  else   if pid=-1 then         {Fork failed - parent only}    begin      DosError:=8;      exit    end;{We're in the parent, let's wait.}  LastDosExitCode:=InternalWaitProcess(pid); // WaitPid and result-convert  if (LastDosExitCode>=0) and (LastDosExitCode<>127) then DosError:=0 else     DosError:=8; // perhaps one time give an better errorEnd;{$ENDIF}{******************************************************************************                               --- Disk ---******************************************************************************}Procedure AddDisk(const path:string);begin  if not (DriveStr[Drives]=nil) then   FreeMem(DriveStr[Drives],StrLen(DriveStr[Drives])+1);  GetMem(DriveStr[Drives],length(Path)+1);  StrPCopy(DriveStr[Drives],path);  inc(Drives);  if Drives>26 then   Drives:=4;end;{******************************************************************************                       --- Findfirst FindNext ---******************************************************************************}Function FNMatch(const Pattern,Name:string):Boolean;Var  LenPat,LenName : longint;  Function DoFNMatch(i,j:longint):Boolean;  Var    Found : boolean;  Begin  Found:=true;  While Found and (i<=LenPat) Do   Begin     Case Pattern[i] of      '?' : Found:=(j<=LenName);      '*' : Begin            {find the next character in pattern, different of ? and *}              while Found and (i<LenPat) do                begin                inc(i);                case Pattern[i] of                  '*' : ;                  '?' : begin                          inc(j);                          Found:=(j<=LenName);                        end;                else                  Found:=false;                end;               end;            {Now, find in name the character which i points to, if the * or ?             wasn't the last character in the pattern, else, use up all the             chars in name}              Found:=true;              if (i<=LenPat) then                begin                repeat                {find a letter (not only first !) which maches pattern[i]}                while (j<=LenName) and (name[j]<>pattern[i]) do                  inc (j);                 if (j<LenName) then                  begin                    if DoFnMatch(i+1,j+1) then                     begin                       i:=LenPat;                       j:=LenName;{we can stop}                       Found:=true;                     end                    else                     inc(j);{We didn't find one, need to look further}                  end;               until (j>=LenName);                end              else                j:=LenName;{we can stop}            end;     else {not a wildcard character in pattern}       Found:=(j<=LenName) and (pattern[i]=name[j]);     end;     inc(i);     inc(j);   end;  DoFnMatch:=Found and (j>LenName);  end;Begin {start FNMatch}  LenPat:=Length(Pattern);  LenName:=Length(Name);  FNMatch:=DoFNMatch(1,1);End;Procedure FindClose(Var f: SearchRec);{  Closes dirptr if it is open}Begin  { could already have been closed }  if assigned(f.dirptr) then     sys_closedir(pdir(f.dirptr));  f.dirptr := nil;End;{ Returns a filled in searchRec structure }{ and TRUE if the specified file in s is  }{ found.                                  }Function FindGetFileInfo(s:string;var f:SearchRec):boolean;var  DT   : DateTime;  st   : stat;  Fmode : byte;  res: string;    { overlaid variable }  Dir : DirsTr;  Name : NameStr;  Ext: ExtStr;begin  FindGetFileInfo:=false;  res := s + #0;  if sys_stat(@res[1],st)<>0 then   exit;  if S_ISDIR(st.st_mode) then   fmode:=directory  else   fmode:=0;  if (st.st_mode and S_IWUSR)=0 then   fmode:=fmode or readonly;  FSplit(s,Dir,Name,Ext);  if Name[1]='.' then   fmode:=fmode or hidden;  If ((FMode and Not(f.searchattr))=0) Then   Begin     if Ext <> '' then       res := Name + Ext     else       res := Name;     f.Name:=res;     f.Attr:=FMode;     f.Size:=longint(st.st_size);     UnixDateToDT(st.st_mtime, DT);     PackTime(DT,f.Time);     FindGetFileInfo:=true;   End;end;Procedure FindNext(Var f: SearchRec);{  re-opens dir if not already in array and calls FindWorkProc}Var  FName,  SName    : string;  Found,  Finished : boolean;  p        : PDirEnt;Begin{Main loop}  SName:=f.SearchSpec;  Found:=False;  Finished:=(f.dirptr=nil);  While Not Finished Do   Begin     p:=sys_readdir(pdir(f.dirptr));     if p=nil then     begin      FName:=''     end     else      FName:=Strpas(@p^.d_name);     If FName='' Then      Finished:=True     Else      Begin        If FNMatch(SName,FName) Then         Begin           Found:=FindGetFileInfo(f.SearchDir+FName,f);           if Found then           begin            Finished:=true;           end;         End;      End;   End;{Shutdown}  If Found Then   Begin     DosError:=0;   End  Else   Begin     FindClose(f);     { FindClose() might be called thereafter also... }     f.dirptr := nil;     DosError:=18;   End;End;Procedure FindFirst(Const Path: PathStr; Attr: Word; Var f: SearchRec);{  opens dir}var res: string;  Dir : DirsTr;  Name : NameStr;  Ext: ExtStr;Begin  { initialize f.dirptr because it is used    }  { to see if we need to close the dir stream }  f.dirptr := nil;  if Path='' then   begin     DosError:=3;     exit;   end;  {We always also search for readonly and archive, regardless of Attr:}  f.SearchAttr := Attr or archive or readonly;{Wildcards?}  if (Pos('?',Path)=0)  and (Pos('*',Path)=0) then   begin     if FindGetFileInfo(Path,f) then      DosError:=0     else      begin        if ErrNo=Sys_ENOENT then         DosError:=3        else         DosError:=18;      end;     f.DirPtr:=nil;   end  else{Find Entry}   begin     FSplit(Path,Dir,Name,Ext);     if Ext <> '' then       res := Name + Ext     else       res := Name;     f.SearchSpec := res;     { if dir is an empty string }     { then this indicates that  }     { use the current working   }     { directory.                }     if dir = '' then        dir := './';     f.SearchDir := Dir;     { add terminating null character }     Dir := Dir + #0;     f.dirptr := sys_opendir(@Dir[1]);     if not assigned(f.dirptr) then     begin        DosError := 8;        exit;     end;     FindNext(f);   end;End;{******************************************************************************                               --- File ---******************************************************************************}Function FSearch(const path:pathstr;dirlist:string):pathstr;{  Searches for a file 'path' in the list of direcories in 'dirlist'.  returns an empty string if not found. Wildcards are NOT allowed.  If dirlist is empty, it is set to '.'}Var  NewDir : PathStr;  p1     : Longint;  Info   : Stat;  buffer : array[0..FileNameLen+1] of char;Begin  Move(path[1], Buffer, Length(path));  Buffer[Length(path)]:=#0;  if (length(Path)>0) and (path[1]='/') and (sys_stat(pchar(@Buffer),info)=0) then  begin    FSearch:=path;    exit;  end;{Replace ':' with ';'}  for p1:=1to length(dirlist) do   if dirlist[p1]=':' then    dirlist[p1]:=';';{Check for WildCards}  If (Pos('?',Path) <> 0) or (Pos('*',Path) <> 0) Then   FSearch:='' {No wildcards allowed in these things.}  Else   Begin     Dirlist:='.;'+dirlist;{Make sure current dir is first to be searched.}     Repeat       p1:=Pos(';',DirList);       If p1=0 Then        p1:=255;       NewDir:=Copy(DirList,1,P1 - 1);       if NewDir[Length(NewDir)]<>'/' then        NewDir:=NewDir+'/';       NewDir:=NewDir+Path;       Delete(DirList,1,p1);       Move(NewDir[1], Buffer, Length(NewDir));       Buffer[Length(NewDir)]:=#0;       if sys_stat(pchar(@Buffer),Info)=0 then        Begin          If Pos('./',NewDir)=1 Then           Delete(NewDir,1,2);        {DOS strips off an initial .\}        End       Else        NewDir:='';     Until (DirList='') or (Length(NewDir) > 0);     FSearch:=NewDir;   End;End;Procedure GetFAttr(var f; var attr : word);Var  info : stat;  LinAttr : mode_t;Begin  DosError:=0;  if sys_stat(@textrec(f).name,info)<>0 then   begin     Attr:=0;     DosError:=3;     exit;   end  else   LinAttr:=Info.st_Mode;  if S_ISDIR(LinAttr) then   Attr:=directory  else   Attr:=0;  if sys_Access(@textrec(f).name,W_OK)<>0 then   Attr:=Attr or readonly;  if (filerec(f).name[0]='.')  then   Attr:=Attr or hidden;end;Procedure getftime (var f; var time : longint);Var  Info: stat;  DT: DateTime;Begin  doserror:=0;  if sys_fstat(filerec(f).handle,info)<>0 then   begin     Time:=0;     doserror:=3;     exit   end  else   UnixDateToDT(Info.st_mtime,DT);  PackTime(DT,Time);End;{******************************************************************************                             --- Environment ---******************************************************************************}Function EnvCount: Longint;var  envcnt : longint;  p      : ppchar;Begin  envcnt:=0;  p:=envp;      {defined in syslinux}  while (p^<>nil) do   begin     inc(envcnt);     inc(p);   end;  EnvCount := envcntEnd;Function EnvStr (Index: longint): String;Var  i : longint;  p : ppchar;Begin  p:=envp;      {defined in syslinux}  i:=1;  envstr:='';  if (index < 1) or (index > EnvCount) then    exit;  while (i<Index) and (p^<>nil) do   begin     inc(i);     inc(p);   end;  if p<>nil then   envstr:=strpas(p^)End;Function GetEnv(EnvVar:string):string;{  Searches the environment for a string with name p and  returns a pchar to it's value.  A pchar is used to accomodate for strings of length > 255}var  ep    : ppchar;  found : boolean;  p1 : pchar;Begin  EnvVar:=EnvVar+'=';            {Else HOST will also find HOSTNAME, etc}  ep:=envp;  found:=false;  if ep<>nil then   begin     while (not found) and (ep^<>nil) do      begin        if strlcomp(@EnvVar[1],(ep^),length(EnvVar))=0 then         found:=true        else         inc(ep);      end;   end;  if found then   p1:=ep^+length(EnvVar)  else   p1:=nil;  if p1 = nil then    GetEnv := ''  else    GetEnv := StrPas(p1);end;Procedure setftime(var f; time : longint);Begin  {! No POSIX equivalent !}End;Procedure setfattr (var f;attr : word);Begin  {! No POSIX equivalent !}End;{ Include timezone routines }{$i timezone.inc}{******************************************************************************                            --- Initialization ---******************************************************************************}Initialization  InitLocalTime;finalization  DoneLocalTime;end.
 |