| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747 | {    This file is part of the Free Pascal run time library.    Copyright (c) 1999-2000 by Florian Klaempfl    member of the Free Pascal development team    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. **********************************************************************}  { Read filename handling functions implementation }  {$i fina.inc}  { variant error codes }  {$i varerror.inc}    Function FileSearch (Const Name, DirList : String; Options : TFileSearchoptions = [sfoImplicitCurrentDir]) : String;    Var      I : longint;      Temp : String;    begin      Result:=Name;      temp:=SetDirSeparators(DirList);      // Start with checking the file in the current directory      If (sfoImplicitCurrentDir in Options) and (Result <> '') and FileExists(Result) Then        exit;      while True do begin        If Temp = '' then          Break; // No more directories to search - fail        I:=pos(PathSeparator,Temp);        If I<>0 then          begin            Result:=Copy (Temp,1,i-1);            system.Delete(Temp,1,I);          end        else          begin            Result:=Temp;            Temp:='';          end;        If Result<>'' then          begin          If (sfoStripQuotes in Options) and (Result[1]='"') and (Result[Length(Result)]='"') then            Result:=Copy(Result,2,Length(Result)-2);          if (Result<>'') then            Result:=IncludeTrailingPathDelimiter(Result)+name;          end;        If (Result <> '') and FileExists(Result) Then          exit;      end;      result:='';    end;    Function FileSearch (Const Name, DirList : String; ImplicitCurrentDir : Boolean) : String;    begin      if ImplicitCurrentDir then        Result:=FileSearch(Name,DirList,[sfoImplicitCurrentDir])      else          Result:=FileSearch(Name,DirList,[]);    end;        Function ExeSearch (Const Name : String; Const DirList : String ='' ) : String;        Var      D : String;      O : TFileSearchOptions;    begin      D:=DirList;      if (D='') then        D:=GetEnvironmentVariable('PATH');    {$ifdef unix}      O:=[];    {$else unix}      O:=[sfoImplicitCurrentDir,sfoStripQuotes];    {$endif unix}      Result := FileSearch(Name, D, O);    end;  {$ifndef OS_FILEISREADONLY}  Function FileIsReadOnly(const FileName: String): Boolean;  begin    Result := (FileGetAttr(FileName) and faReadOnly) <> 0;  end;  {$endif OS_FILEISREADONLY}  {$ifndef OS_FILESETDATEBYNAME}  Function FileSetDate (Const FileName : String;Age : Longint) : Longint;  Var    fd : THandle;  begin    { at least windows requires fmOpenWrite here }    fd:=FileOpen(FileName,fmOpenWrite);    If (Fd<>feInvalidHandle) then      try        Result:=FileSetDate(fd,Age);      finally        FileClose(fd);      end    else  {$ifdef HAS_OSERROR}      Result:=GetLastOSError;  {$else}      Result:=-1;  {$endif}  end;  {$endif}  { Read String Handling functions implementation }  {$i sysstr.inc}  { Read date & Time function implementations }{$ifndef FPUNONE}  {$i dati.inc}{$endif}  { Read pchar handling functions implementation }  {$i syspch.inc}  { generic internationalisation code }  {$i sysint.inc}  { MCBS functions }  {$i sysansi.inc}  { wide string functions }  {$i syswide.inc}{$ifdef FPC_HAS_UNICODESTRING}  { unicode string functions }  {$i sysuni.inc}{$endif FPC_HAS_UNICODESTRING}    { threading stuff }  {$i sysuthrd.inc}  { OS utility code }  {$i osutil.inc}    procedure FreeAndNil(var obj);      var        temp: tobject;      begin        temp:=tobject(obj);        pointer(obj):=nil;        temp.free;      end;   function FileAge(const FileName: string; out FileDateTime: TDateTime; FollowLink: Boolean = True): Boolean;      Var     Info : TSearchRec;     A : Integer;         begin     for A:=1 to Length(FileName) do       If (FileName[A] in ['?','*']) then         Exit(False);     A:=0;     if Not FollowLink then       A:=A or faSymLink;     Result:=FindFirst(FileName,A,Info)=0;     If Result then        FileDateTime:=FileDatetoDateTime (Info.Time);     FindClose(Info);   end;  { Interfaces support }  {$i sysuintf.inc}    constructor Exception.Create(const msg : string);      begin         inherited create;         fmessage:=msg;      end;    constructor Exception.CreateFmt(const msg : string; const args : array of const);      begin         inherited create;         fmessage:=Format(msg,args);      end;    constructor Exception.CreateRes(ResString: PString);      begin         inherited create;         fmessage:=ResString^;      end;    constructor Exception.CreateResFmt(ResString: PString; const Args: array of const);      begin         inherited create;         fmessage:=Format(ResString^,args);      end;    constructor Exception.CreateHelp(const Msg: string; AHelpContext: Integer);      begin         inherited create;         fmessage:=Msg;         fhelpcontext:=AHelpContext;      end;    constructor Exception.CreateFmtHelp(const Msg: string; const Args: array of const;      AHelpContext: Integer);    begin       inherited create;       fmessage:=Format(Msg,args);       fhelpcontext:=AHelpContext;    end;    constructor Exception.CreateResHelp(ResString: PString; AHelpContext: Integer);    begin       inherited create;       fmessage:=ResString^;       fhelpcontext:=AHelpContext;    end;    constructor Exception.CreateResFmtHelp(ResString: PString; const Args: array of const;      AHelpContext: Integer);    begin       inherited create;       fmessage:=Format(ResString^,args);       fhelpcontext:=AHelpContext;    end;    procedure EHeapMemoryError.FreeInstance;    begin       if AllowFree then        inherited FreeInstance;    end;    Constructor EVariantError.CreateCode (Code : longint);    begin       case Code of         VAR_OK:           Create(SNoError);         VAR_PARAMNOTFOUND:           Create(SVarParamNotFound);         VAR_TYPEMISMATCH:           Create(SInvalidVarCast);         VAR_BADVARTYPE:           Create(SVarBadType);         VAR_OVERFLOW:           Create(SVarOverflow);         VAR_BADINDEX:           Create(SVarArrayBounds);         VAR_ARRAYISLOCKED:           Create(SVarArrayLocked);         VAR_NOTIMPL:           Create(SVarNotImplemented);         VAR_OUTOFMEMORY:           Create(SVarOutOfMemory);         VAR_INVALIDARG:           Create(SVarInvalid);         VAR_UNEXPECTED,         VAR_EXCEPTION:           Create(SVarUnexpected);         else           CreateFmt(SUnknownErrorCode,[Code]);       end;       ErrCode:=Code;    end;{$ifopt S+}{$define STACKCHECK_WAS_ON}{$S-}{$endif OPT S }Procedure CatchUnhandledException (Obj : TObject; Addr: Pointer; FrameCount: Longint; Frames: PPointer);[public,alias:'FPC_BREAK_UNHANDLED_EXCEPTION'];Var  Message : String;  i : longint;  hstdout : ^text;begin  hstdout:=@stdout;  Writeln(hstdout^,'An unhandled exception occurred at $',HexStr(PtrUInt(Addr),sizeof(PtrUInt)*2),' :');  if Obj is exception then   begin     Message:=Exception(Obj).ClassName+' : '+Exception(Obj).Message;     Writeln(hstdout^,Message);   end  else   Writeln(hstdout^,'Exception object ',Obj.ClassName,' is not of class Exception.');  Writeln(hstdout^,BackTraceStrFunc(Addr));  if (FrameCount>0) then    begin      for i:=0 to FrameCount-1 do        Writeln(hstdout^,BackTraceStrFunc(Frames[i]));    end;  Writeln(hstdout^,'');end;Var OutOfMemory : EOutOfMemory;    InValidPointer : EInvalidPointer;Procedure RunErrorToExcept (ErrNo : Longint; Address,Frame : Pointer);Var E : Exception;    HS : PString;begin  Case Errno of   1,203 : E:=OutOfMemory;   204 : E:=InvalidPointer;   2,3,4,5,6,100,101,102,103,105,106 : { I/O errors }     begin     Case Errno of       2 : HS:=@SFileNotFound;       3 : HS:=@SInvalidFileName;       4 : HS:=@STooManyOpenFiles;       5 : HS:=@SAccessDenied;       6 : HS:=@SInvalidFileHandle;       15 : HS:=@SInvalidDrive;       100 : HS:=@SEndOfFile;       101 : HS:=@SDiskFull;       102 : HS:=@SFileNotAssigned;       103 : HS:=@SFileNotOpen;       104 : HS:=@SFileNotOpenForInput;       105 : HS:=@SFileNotOpenForOutput;       106 : HS:=@SInvalidInput;     end;     E:=EinOutError.Create (HS^);     // this routine can be called from FPC_IOCHECK,     // which clears inoutres and then passes its     // original value to HandleErrorFrame() (which calls     // us). So use errno rather than IOResult, and clear     // InOutRes explicitly in case we can also be called     // from a place that does not clear InOutRes explicitly     EInoutError(E).ErrorCode:=errno;     inoutres:=0;     end;  // We don't set abstracterrorhandler, but we do it here.  // Unless the use sets another handler we'll get here anyway...  200 : E:=EDivByZero.Create(SDivByZero);  201 : E:=ERangeError.Create(SRangeError);  205 : E:=EOverflow.Create(SOverflow);  206 : E:=EOverflow.Create(SUnderflow);  207 : E:=EInvalidOp.Create(SInvalidOp);  211 : E:=EAbstractError.Create(SAbstractError);  212 : E:=EExternalException.Create(SExternalException);  214 : E:=EBusError.Create(SBusError);  215 : E:=EIntOverflow.Create(SIntOverflow);  216 : E:=EAccessViolation.Create(SAccessViolation);  217 : E:=EControlC.Create(SControlC);  218 : E:=EPrivilege.Create(SPrivilege);  219 : E:=EInvalidCast.Create(SInvalidCast);  220 : E:=EVariantError.Create(SInvalidVarCast);  221 : E:=EVariantError.Create(SInvalidVarOp);  222 : E:=EVariantError.Create(SDispatchError);  223 : E:=EVariantError.Create(SVarArrayCreate);  224 : E:=EVariantError.Create(SVarNotArray);  225 : E:=EVariantError.Create(SVarArrayBounds);  227 : E:=EAssertionFailed.Create(SAssertionFailed);  228 : E:=EIntfCastError.Create(SIntfCastError);  229 : E:=ESafecallException.Create(SSafecallException);  231 : E:=EConvertError.Create(SiconvError);  232 : E:=ENoThreadSupport.Create(SNoThreadSupport);  233 : E:=ENoWideStringSupport.Create(SSigQuit);  234 : E:=ENoWideStringSupport.Create(SMissingWStringManager);  else   E:=Exception.CreateFmt (SUnKnownRunTimeError,[Errno]);  end;  Raise E at Address,Frame;end;{$IFDEF HAS_OSERROR}Procedure RaiseLastOSError;overload;begin  RaiseLastOSError(GetLastOSError);end;Procedure RaiseLastOSError(LastError: Integer);overload;var  E : EOSError;begin  If (LastError<>0) then    E:=EOSError.CreateFmt(SOSError, [LastError, SysErrorMessage(LastError)])  else    E:=EOSError.Create(SUnkOSError);  E.ErrorCode:=LastError;  Raise E;end;{$else}Procedure RaiseLastOSError;overload;begin  Raise Exception.Create('RaiseLastOSError not implemented on this platform.');end;Procedure RaiseLastOSError(LastError: Integer);overload;begin  RaiseLastOSError;end;{$endif}Procedure AssertErrorHandler (Const Msg,FN : ShortString;LineNo:longint; TheAddr : pointer);Var  S : String;begin  If Msg='' then    S:=SAssertionFailed  else    S:=Msg;  Raise EAssertionFailed.Createfmt(SAssertError,[S,Fn,LineNo]) at get_caller_addr(theAddr), get_caller_frame(theAddr);end;{$ifdef STACKCHECK_WAS_ON}{$S+}{$endif}Procedure InitExceptions;{  Must install uncaught exception handler (ExceptProc)  and install exceptions for system exceptions or signals.  (e.g: SIGSEGV -> ESegFault or so.)}begin  ExceptionClass := Exception;  ExceptProc:=@CatchUnhandledException;  // Create objects that may have problems when there is no memory.  OutOfMemory:=EOutOfMemory.Create(SOutOfMemory);  OutOfMemory.AllowFree:=false;  InvalidPointer:=EInvalidPointer.Create(SInvalidPointer);  InvalidPointer.AllowFree:=false;  AssertErrorProc:=@AssertErrorHandler;  ErrorProc:=@RunErrorToExcept;  OnShowException:=Nil;end;Procedure DoneExceptions;begin  OutOfMemory.AllowFree:=true;  OutOfMemory.Free;  InValidPointer.AllowFree:=true;  InValidPointer.Free;end;{ Exception handling routines }function ExceptObject: TObject;begin  If RaiseList=Nil then    Result:=Nil  else    Result:=RaiseList^.FObject;end;function ExceptAddr: Pointer;begin  If RaiseList=Nil then    Result:=Nil  else    Result:=RaiseList^.Addr;end;function ExceptFrameCount: Longint;begin  If RaiseList=Nil then    Result:=0  else    Result:=RaiseList^.Framecount;end;function ExceptFrames: PPointer;begin  If RaiseList=Nil then    Result:=Nil  else    Result:=RaiseList^.Frames;end;function ExceptionErrorMessage(ExceptObject: TObject; ExceptAddr: Pointer;                               Buffer: PChar; Size: Integer): Integer;Var  S : AnsiString;  Len : Integer;begin  S:=Format(SExceptionErrorMessage,[ExceptAddr,ExceptObject.ClassName]);  If ExceptObject is Exception then    S:=Format('%s:'#10'%s',[S,Exception(ExceptObject).Message]);  Len:=Length(S);  If S[Len]<>'.' then    begin    S:=S+'.';    Inc(len);    end;  If Len>Size then    Len:=Size;  if Len > 0 then    Move(S[1],Buffer^,Len);  Result:=Len;end;procedure ShowException(ExceptObject: TObject; ExceptAddr: Pointer);// use shortstring. On exception, the heap may be corrupt.Var  Buf : ShortString;begin  SetLength(Buf,ExceptionErrorMessage(ExceptObject,ExceptAddr,@Buf[1],255));  If IsConsole Then    writeln(Buf)  else    If Assigned(OnShowException) Then      OnShowException (Buf);end;procedure Abort;begin  Raise EAbort.Create(SAbortError) at Pointer(Get_Caller_addr(Get_Frame));end;procedure OutOfMemoryError;begin  Raise OutOfMemory;end;{ ---------------------------------------------------------------------    Initialization/Finalization/exit code  ---------------------------------------------------------------------}Type  PPRecord = ^TPRecord;  TPRecord = Record    Func : TTerminateProc;    NextFunc : PPRecord;  end;Const  TPList : PPRecord = Nil;procedure AddTerminateProc(TermProc: TTerminateProc);Var  TPR : PPRecord;begin  New(TPR);  With TPR^ do    begin    NextFunc:=TPList;    Func:=TermProc;    end;  TPList:=TPR;end;function CallTerminateProcs: Boolean;Var  TPR : PPRecord;begin  Result:=True;  TPR:=TPList;  While Result and (TPR<>Nil) do    begin    Result:=TPR^.Func();    TPR:=TPR^.NextFunc;    end;end;{ ---------------------------------------------------------------------    Diskh functions, OS independent.  ---------------------------------------------------------------------}function ForceDirectories(Const Dir: string): Boolean;var  E: EInOutError;  ADrv : String;function DoForceDirectories(Const Dir: string): Boolean;var  ADir : String;  APath: String;begin  Result:=True;  ADir:=ExcludeTrailingPathDelimiter(Dir);  if (ADir='') then Exit;  if Not DirectoryExists(ADir) then    begin      APath := ExtractFilePath(ADir);      //this can happen on Windows if user specifies Dir like \user\name/test/      //and would, if not checked for, cause an infinite recusrsion and a stack overflow      if (APath = ADir) then Result := False        else Result:=DoForceDirectories(APath);    If Result then      Result := CreateDir(ADir);    end;end;function IsUncDrive(const Drv: String): Boolean;begin  Result := (Length(Drv) > 2) and (Drv[1] = PathDelim) and (Drv[2] = PathDelim);end;begin  Result := False;  ADrv := ExtractFileDrive(Dir);  if (ADrv<>'') and (not DirectoryExists(ADrv))  {$IFNDEF FORCEDIR_NO_UNC_SUPPORT} and (not IsUncDrive(ADrv)){$ENDIF} then Exit;  if Dir='' then    begin      E:=EInOutError.Create(SCannotCreateEmptyDir);      E.ErrorCode:=3;      Raise E;    end;  Result := DoForceDirectories(SetDirSeparators(Dir));end;Var  GUIDCalledRandomize : Boolean = False;Procedure GetRandomBytes(Var Buf; NBytes : Integer);Var  I : Integer;  P : PByte;begin  P:=@Buf;  If Not GUIDCalledRandomize then    begin    Randomize;    GUIDCalledRandomize:=True;    end;  For I:=0 to NBytes-1 do    P[i]:=Random(256);end;{$IFDEF HASCREATEGUID}Function SysCreateGUID(out GUID : TGUID) : Integer; forward;{$ENDIF}Function CreateGUID(out GUID : TGUID) : Integer;begin  If Assigned(OnCreateGUID) then    Result:=OnCreateGUID(GUID)  else    begin    {$IFDEF HASCREATEGUID}    Result:=SysCreateGUID(GUID);    {$ELSE}    GetRandomBytes(GUID,SizeOf(Guid));    guid.clock_seq_hi_and_reserved:=(guid.clock_seq_hi_and_reserved and $3F) + 64;    guid.time_hi_and_version      :=(guid.time_hi_and_version and $0FFF)+ $4000;    Result:=0;    {$ENDIF}    end;end;function SafeLoadLibrary(const FileName: AnsiString;  ErrorMode: DWord = {$ifdef windows}SEM_NOOPENFILEERRORBOX{$else windows}0{$endif windows}): HMODULE;{$if defined(cpui386) or defined(cpux86_64)}  var    mode : DWord;    fpucw : Word;    ssecw : DWord;{$endif}  begin{$if defined(win64) or defined(win32)}    mode:=SetErrorMode(ErrorMode);{$endif}    try{$if defined(cpui386) or defined(cpux86_64)}      fpucw:=Get8087CW;{$ifdef cpui386}      if has_sse_support then{$endif cpui386}        ssecw:=GetSSECSR;{$endif}{$if defined(windows) or defined(win32)}      Result:=LoadLibraryA(PChar(Filename));{$else}      Result:=0;{$endif}      finally{$if defined(cpui386) or defined(cpux86_64)}      Set8087CW(fpucw);{$ifdef cpui386}      if has_sse_support then{$endif cpui386}        SetSSECSR(ssecw);{$endif}{$if defined(win64) or defined(win32)}      SetErrorMode(mode);{$endif}    end;  end;function GetModuleName(Module: HMODULE): string;begin{$ifdef MSWINDOWS}  SetLength(Result,MAX_PATH);  SetLength(Result,GetModuleFileName(Module, Pchar(Result),Length(Result)));{$ELSE}  Result:='';{$ENDIF}end;{ Beep support }procedure Beep;begin  If Assigned(OnBeep) then    OnBeep;end;
 |