| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220 | {    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    Sysutils unit for win32    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 sysutils;interface{$MODE objfpc}{ force ansistrings }{$H+}uses  windows;{$DEFINE HAS_SLEEP}{$DEFINE HAS_OSERROR}{$DEFINE HAS_OSCONFIG}{$DEFINE HAS_CREATEGUID}{ Include platform independent interface part }{$i sysutilh.inc}type  TSystemTime = Windows.TSystemTime;  EWin32Error = class(Exception)  public    ErrorCode : DWORD;  end;Var  Win32Platform : Longint;  Win32MajorVersion,  Win32MinorVersion,  Win32BuildNumber   : dword;  Win32CSDVersion    : ShortString;   // CSD record is 128 bytes only?{ Compatibility with Delphi }function Win32Check(res:boolean):boolean;inline;function WinCheck(res:boolean):boolean;function CheckWin32Version(Major,Minor : Integer ): Boolean;function CheckWin32Version(Major : Integer): Boolean;Procedure RaiseLastWin32Error;function GetFileVersion(const AFileName: string): Cardinal;procedure GetFormatSettings;implementation  uses    sysconst;function WinCheck(res:boolean):boolean;  begin    if not res then      RaiseLastOSError;    result:=res;  end;function Win32Check(res:boolean):boolean;inline;  begin    result:=WinCheck(res);  end;procedure RaiseLastWin32Error;  begin    RaiseLastOSError;  end;function CheckWin32Version(Major : Integer): Boolean;  begin    Result:=CheckWin32Version(Major,0)  end;function CheckWin32Version(Major,Minor: Integer): Boolean;  begin    Result:=(Win32MajorVersion>dword(Major)) or            ((Win32MajorVersion=dword(Major)) and (Win32MinorVersion>=dword(Minor)));  end;function GetFileVersion(const AFileName:string):Cardinal;  var    { usefull only as long as we don't need to touch different stack pages }    buf : array[0..3071] of byte;    bufp : pointer;    fn : string;    valsize,    size : DWORD;    h : DWORD;    valrec : PVSFixedFileInfo;  begin    result:=$fffffff;    fn:=AFileName;    UniqueString(fn);    size:=GetFileVersionInfoSize(pchar(fn),@h);    if size>sizeof(buf) then      begin        getmem(bufp,size);        try          if GetFileVersionInfo(pchar(fn),h,size,bufp) then            if VerQueryValue(bufp,'\',valrec,valsize) then              result:=valrec^.dwFileVersionMS;        finally          freemem(bufp);        end;      end    else      begin        if GetFileVersionInfo(pchar(fn),h,size,@buf) then          if VerQueryValue(@buf,'\',valrec,valsize) then            result:=valrec^.dwFileVersionMS;      end;  end;{$define HASCREATEGUID}{$define HASEXPANDUNCFILENAME}{$DEFINE FPC_NOGENERICANSIROUTINES}{$DEFINE FPC_FEXPAND_UNC} (* UNC paths are supported *){$DEFINE FPC_FEXPAND_DRIVES} (* Full paths begin with drive specification *){ Include platform independent implementation part }{$i sysutils.inc}function SysGetTempFileName(lpPathName:LPCSTR;                            lpPrefixString:LPCSTR;                            uUnique:UINT;                            lpTempFileName:LPSTR):UINT;stdcall;external 'kernel32' name 'GetTempFileNameA';function GetTempFileName(Dir,Prefix: PChar; uUnique: DWORD; TempFileName: PChar):DWORD;begin  Result:=SysGetTempFileName(Dir,Prefix,uUnique,TempFileName);end;{ UUID generation. }function CoCreateGuid(out guid: TGUID): HResult; stdcall; external 'ole32.dll' name 'CoCreateGuid';function SysCreateGUID(out Guid: TGUID): Integer;begin  Result := Integer(CoCreateGuid(Guid));end;function ExpandUNCFileName (const filename:string) : string;{ returns empty string on errors }var  s    : ansistring;  size : dword;  rc   : dword;  buf : pchar;begin  s := ExpandFileName (filename);  s := s + #0;  size := max_path;  getmem(buf,size);  try    rc := WNetGetUniversalName (pchar(s), UNIVERSAL_NAME_INFO_LEVEL, buf, @size);    if rc=ERROR_MORE_DATA then      begin        buf:=reallocmem(buf,size);        rc := WNetGetUniversalName (pchar(s), UNIVERSAL_NAME_INFO_LEVEL, buf, @size);      end;    if rc = NO_ERROR then      Result := PRemoteNameInfo(buf)^.lpUniversalName    else if rc = ERROR_NOT_CONNECTED then      Result := filename    else      Result := '';  finally    freemem(buf);  end;end;{****************************************************************************                              File Functions****************************************************************************}var  SetFilePointerEx : function(hFile : THandle;    liDistanceToMove : int64;lpNewFilePointer : pint64;    dwMoveMethod : DWord) : ByteBool;stdcall;Function FileOpen (Const FileName : string; Mode : Integer) : THandle;const  AccessMode: array[0..2] of Cardinal  = (    GENERIC_READ,    GENERIC_WRITE,    GENERIC_READ or GENERIC_WRITE);  ShareMode: array[0..4] of Integer = (               0,               0,               FILE_SHARE_READ,               FILE_SHARE_WRITE,               FILE_SHARE_READ or FILE_SHARE_WRITE);Var  FN : string;begin  FN:=FileName+#0;  result := CreateFile(@FN[1], dword(AccessMode[Mode and 3]),                       dword(ShareMode[(Mode and $F0) shr 4]), nil, OPEN_EXISTING,                       FILE_ATTRIBUTE_NORMAL, 0);  //if fail api return feInvalidHandle (INVALIDE_HANDLE=feInvalidHandle=-1)end;Function FileCreate (Const FileName : String) : THandle;Var  FN : string;begin  FN:=FileName+#0;  Result := CreateFile(@FN[1], GENERIC_READ or GENERIC_WRITE,                       0, nil, CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, 0);end;Function FileCreate (Const FileName : String; Mode:longint) : THandle;begin  FileCreate:=FileCreate(FileName);end;Function FileRead (Handle : THandle; Var Buffer; Count : longint) : Longint;Var  res : dword;begin  if ReadFile(Handle, Buffer, Count, res, nil) then   FileRead:=Res  else   FileRead:=-1;end;Function FileWrite (Handle : THandle; const Buffer; Count : Longint) : Longint;Var  Res : dword;begin  if WriteFile(Handle, Buffer, Count, Res, nil) then   FileWrite:=Res  else   FileWrite:=-1;end;Function FileSeek (Handle : THandle;FOffset,Origin : Longint) : Longint;begin  Result := longint(SetFilePointer(Handle, FOffset, nil, Origin));end;Function FileSeek (Handle : THandle; FOffset: Int64; Origin: Longint) : Int64;begin  if assigned(SetFilePointerEx) then    begin      if not(SetFilePointerEx(Handle, FOffset, @result, Origin)) then        Result:=-1;    end  else    Result:=longint(SetFilePointer(Handle, FOffset, nil, Origin));end;Procedure FileClose (Handle : THandle);begin  if Handle<=4 then   exit;  CloseHandle(Handle);end;Function FileTruncate (Handle : THandle;Size: Int64) : boolean;begin{  Result:=longint(SetFilePointer(handle,Size,nil,FILE_BEGIN))<>-1;}  if FileSeek (Handle, Size, FILE_BEGIN) = Size then   Result:=SetEndOfFile(handle)  else   Result := false;end;Function DosToWinTime (DTime:longint;Var Wtime : TFileTime):longbool;var  lft : TFileTime;begin  DosToWinTime:=DosDateTimeToFileTime(longrec(dtime).hi,longrec(dtime).lo,@lft) and                LocalFileTimeToFileTime(lft,Wtime);end;Function WinToDosTime (Var Wtime : TFileTime;var DTime:longint):longbool;var  lft : TFileTime;begin  WinToDosTime:=FileTimeToLocalFileTime(WTime,lft) and                FileTimeToDosDateTime(lft,Longrec(Dtime).Hi,LongRec(DTIME).lo);end;Function FileAge (Const FileName : String): Longint;var  Handle: THandle;  FindData: TWin32FindData;begin  Handle := FindFirstFile(Pchar(FileName), FindData);  if Handle <> INVALID_HANDLE_VALUE then    begin      Windows.FindClose(Handle);      if (FindData.dwFileAttributes and FILE_ATTRIBUTE_DIRECTORY) = 0 then        If WinToDosTime(FindData.ftLastWriteTime,Result) then          exit;    end;  Result := -1;end;Function FileExists (Const FileName : String) : Boolean;var  Attr:Dword;begin  Attr:=GetFileAttributes(PChar(FileName));  if Attr <> $ffffffff then    Result:= (Attr and FILE_ATTRIBUTE_DIRECTORY) = 0  else    Result:=False;end;Function DirectoryExists (Const Directory : String) : Boolean;var  Attr:Dword;begin  Attr:=GetFileAttributes(PChar(Directory));  if Attr <> $ffffffff then    Result:= (Attr and FILE_ATTRIBUTE_DIRECTORY) > 0  else    Result:=False;end;Function FindMatch(var f: TSearchRec) : Longint;begin  { Find file with correct attribute }  While (F.FindData.dwFileAttributes and cardinal(F.ExcludeAttr))<>0 do   begin     if not FindNextFile (F.FindHandle,F.FindData) then      begin        Result:=GetLastError;        exit;      end;   end;  { Convert some attributes back }  WinToDosTime(F.FindData.ftLastWriteTime,F.Time);  f.size:=F.FindData.NFileSizeLow;  f.attr:=F.FindData.dwFileAttributes;  f.Name:=StrPas(@F.FindData.cFileName[0]);  Result:=0;end;Function FindFirst (Const Path : String; Attr : Longint; out Rslt : TSearchRec) : Longint;begin  Rslt.Name:=Path;  Rslt.Attr:=attr;  Rslt.ExcludeAttr:=(not Attr) and ($1e);                 { $1e = faHidden or faSysFile or faVolumeID or faDirectory }  { FindFirstFile is a Win32 Call }  Rslt.FindHandle:=FindFirstFile (PChar(Path),Rslt.FindData);  If Rslt.FindHandle=Invalid_Handle_value then   begin     Result:=GetLastError;     exit;   end;  { Find file with correct attribute }  Result:=FindMatch(Rslt);end;Function FindNext (Var Rslt : TSearchRec) : Longint;begin  if FindNextFile(Rslt.FindHandle, Rslt.FindData) then    Result := FindMatch(Rslt)  else    Result := GetLastError;end;Procedure FindClose (Var F : TSearchrec);begin   if F.FindHandle <> INVALID_HANDLE_VALUE then    Windows.FindClose(F.FindHandle);end;Function FileGetDate (Handle : THandle) : Longint;Var  FT : TFileTime;begin  If GetFileTime(Handle,nil,nil,@ft) and     WinToDosTime(FT,Result) then    exit;  Result:=-1;end;Function FileSetDate (Handle : THandle;Age : Longint) : Longint;Var  FT: TFileTime;begin  Result := 0;  if DosToWinTime(Age,FT) and    SetFileTime(Handle, nil, nil, @FT) then    Exit;  Result := GetLastError;end;Function FileGetAttr (Const FileName : String) : Longint;begin  Result:=GetFileAttributes(PChar(FileName));end;Function FileSetAttr (Const Filename : String; Attr: longint) : Longint;begin  if SetFileAttributes(PChar(FileName), Attr) then    Result:=0  else    Result := GetLastError;end;Function DeleteFile (Const FileName : String) : Boolean;begin  Result:=Windows.DeleteFile(Pchar(FileName));end;Function RenameFile (Const OldName, NewName : String) : Boolean;begin  Result := MoveFile(PChar(OldName), PChar(NewName));end;{****************************************************************************                              Disk Functions****************************************************************************}function GetDiskFreeSpace(drive:pchar;var sector_cluster,bytes_sector,                          freeclusters,totalclusters:longint):longbool;         stdcall;external 'kernel32' name 'GetDiskFreeSpaceA';type   TGetDiskFreeSpaceEx = function(drive:pchar;var availableforcaller,total,free):longbool;stdcall;var GetDiskFreeSpaceEx : TGetDiskFreeSpaceEx;function diskfree(drive : byte) : int64;var  disk : array[1..4] of char;  secs,bytes,  free,total : longint;  qwtotal,qwfree,qwcaller : int64;begin  if drive=0 then   begin     disk[1]:='\';     disk[2]:=#0;   end  else   begin     disk[1]:=chr(drive+64);     disk[2]:=':';     disk[3]:='\';     disk[4]:=#0;   end;  if assigned(GetDiskFreeSpaceEx) then    begin       if GetDiskFreeSpaceEx(@disk[1],qwcaller,qwtotal,qwfree) then         diskfree:=qwfree       else         diskfree:=-1;    end  else    begin       if GetDiskFreeSpace(@disk[1],secs,bytes,free,total) then         diskfree:=int64(free)*secs*bytes       else         diskfree:=-1;    end;end;function disksize(drive : byte) : int64;var  disk : array[1..4] of char;  secs,bytes,  free,total : longint;  qwtotal,qwfree,qwcaller : int64;begin  if drive=0 then   begin     disk[1]:='\';     disk[2]:=#0;   end  else   begin     disk[1]:=chr(drive+64);     disk[2]:=':';     disk[3]:='\';     disk[4]:=#0;   end;  if assigned(GetDiskFreeSpaceEx) then    begin       if GetDiskFreeSpaceEx(@disk[1],qwcaller,qwtotal,qwfree) then         disksize:=qwtotal       else         disksize:=-1;    end  else    begin       if GetDiskFreeSpace(@disk[1],secs,bytes,free,total) then         disksize:=int64(total)*secs*bytes       else         disksize:=-1;    end;end;Function GetCurrentDir : String;begin  GetDir(0, result);end;Function SetCurrentDir (Const NewDir : String) : Boolean;begin  Result:=SetCurrentDirectory(PChar(NewDir));end;Function CreateDir (Const NewDir : String) : Boolean;begin  Result:=CreateDirectory(PChar(NewDir),nil);end;Function RemoveDir (Const Dir : String) : Boolean;begin  Result:=RemoveDirectory(PChar(Dir));end;{****************************************************************************                              Time Functions****************************************************************************}Procedure GetLocalTime(var SystemTime: TSystemTime);Var  Syst : Windows.TSystemtime;begin  windows.Getlocaltime(@syst);  SystemTime.year:=syst.wYear;  SystemTime.month:=syst.wMonth;  SystemTime.day:=syst.wDay;  SystemTime.hour:=syst.wHour;  SystemTime.minute:=syst.wMinute;  SystemTime.second:=syst.wSecond;  SystemTime.millisecond:=syst.wMilliSeconds;end;{****************************************************************************                              Misc Functions****************************************************************************}procedure Beep;begin  MessageBeep(0);end;{****************************************************************************                              Locale Functions****************************************************************************}function GetLocaleStr(LID, LT: Longint; const Def: string): ShortString;var  L: Integer;  Buf: array[0..255] of Char;begin  L := GetLocaleInfo(LID, LT, Buf, SizeOf(Buf));  if L > 0 then    SetString(Result, @Buf[0], L - 1)  else    Result := Def;end;function GetLocaleChar(LID, LT: Longint; Def: Char): Char;var  Buf: array[0..1] of Char;begin  if GetLocaleInfo(LID, LT, Buf, 2) > 0 then    Result := Buf[0]  else    Result := Def;end;Function GetLocaleInt(LID,TP,Def: LongInt): LongInt;Var  S: String;  C: Integer;Begin  S:=GetLocaleStr(LID,TP,'0');  Val(S,Result,C);  If C<>0 Then    Result:=Def;End;procedure GetFormatSettings;var  HF  : Shortstring;  LID : LCID;  I,Day : longint;begin  LID := GetThreadLocale;  { Date stuff }  for I := 1 to 12 do    begin    ShortMonthNames[I]:=GetLocaleStr(LID,LOCALE_SABBREVMONTHNAME1+I-1,ShortMonthNames[i]);    LongMonthNames[I]:=GetLocaleStr(LID,LOCALE_SMONTHNAME1+I-1,LongMonthNames[i]);    end;  for I := 1 to 7 do    begin    Day := (I + 5) mod 7;    ShortDayNames[I]:=GetLocaleStr(LID,LOCALE_SABBREVDAYNAME1+Day,ShortDayNames[i]);    LongDayNames[I]:=GetLocaleStr(LID,LOCALE_SDAYNAME1+Day,LongDayNames[i]);    end;  DateSeparator := GetLocaleChar(LID, LOCALE_SDATE, '/');  ShortDateFormat := GetLocaleStr(LID, LOCALE_SSHORTDATE, 'm/d/yy');  LongDateFormat := GetLocaleStr(LID, LOCALE_SLONGDATE, 'mmmm d, yyyy');  { Time stuff }  TimeSeparator := GetLocaleChar(LID, LOCALE_STIME, ':');  TimeAMString := GetLocaleStr(LID, LOCALE_S1159, 'AM');  TimePMString := GetLocaleStr(LID, LOCALE_S2359, 'PM');  if StrToIntDef(GetLocaleStr(LID, LOCALE_ITLZERO, '0'), 0) = 0 then    HF:='h'  else    HF:='hh';  // No support for 12 hour stuff at the moment...  ShortTimeFormat := HF+':nn';  LongTimeFormat := HF + ':nn:ss';  { Currency stuff }  CurrencyString:=GetLocaleStr(LID, LOCALE_SCURRENCY, '');  CurrencyFormat:=StrToIntDef(GetLocaleStr(LID, LOCALE_ICURRENCY, '0'), 0);  NegCurrFormat:=StrToIntDef(GetLocaleStr(LID, LOCALE_INEGCURR, '0'), 0);  { Number stuff }  ThousandSeparator:=GetLocaleChar(LID, LOCALE_STHOUSAND, ',');  DecimalSeparator:=GetLocaleChar(LID, LOCALE_SDECIMAL, '.');  CurrencyDecimals:=StrToIntDef(GetLocaleStr(LID, LOCALE_ICURRDIGITS, '0'), 0);end;Procedure InitInternational;var  { A call to GetSystemMetrics changes the value of the 8087 Control Word on    Pentium4 with WinXP SP2 }  old8087CW: word;begin  InitInternationalGeneric;  old8087CW:=Get8087CW;  SysLocale.MBCS:=GetSystemMetrics(SM_DBCSENABLED)<>0;  SysLocale.RightToLeft:=GetSystemMetrics(SM_MIDEASTENABLED)<>0;  SysLocale.DefaultLCID := $0409;  SysLocale.PriLangID := LANG_ENGLISH;  SysLocale.SubLangID := SUBLANG_ENGLISH_US;  // probably needs update with getthreadlocale. post 2.0.2  Set8087CW(old8087CW);  GetFormatSettings;end;{****************************************************************************                           Target Dependent****************************************************************************}function FormatMessageA(dwFlags     : DWORD;                        lpSource    : Pointer;                        dwMessageId : DWORD;                        dwLanguageId: DWORD;                        lpBuffer    : PCHAR;                        nSize       : DWORD;                        Arguments   : Pointer): DWORD; stdcall;external 'kernel32' name 'FormatMessageA';function SysErrorMessage(ErrorCode: Integer): String;const  MaxMsgSize = Format_Message_Max_Width_Mask;var  MsgBuffer: pChar;begin  GetMem(MsgBuffer, MaxMsgSize);  FillChar(MsgBuffer^, MaxMsgSize, #0);  FormatMessageA(FORMAT_MESSAGE_FROM_SYSTEM,                 nil,                 ErrorCode,                 MakeLangId(LANG_NEUTRAL, SUBLANG_DEFAULT),                 MsgBuffer,                 { This function allocs the memory }                 MaxMsgSize,                           { Maximum message size }                 nil);  SysErrorMessage := StrPas(MsgBuffer);  FreeMem(MsgBuffer, MaxMsgSize);end;{****************************************************************************                              Initialization code****************************************************************************}Function GetEnvironmentVariable(Const EnvVar : String) : String;var   s : string;   i : longint;   hp,p : pchar;begin   Result:='';   p:=GetEnvironmentStrings;   hp:=p;   while hp^<>#0 do     begin        s:=strpas(hp);        i:=pos('=',s);        if uppercase(copy(s,1,i-1))=upcase(envvar) then          begin             Result:=copy(s,i+1,length(s)-i);             break;          end;        { next string entry}        hp:=hp+strlen(hp)+1;     end;   FreeEnvironmentStrings(p);end;Function GetEnvironmentVariableCount : Integer;var  hp,p : pchar;begin  Result:=0;  p:=GetEnvironmentStrings;  hp:=p;  If (Hp<>Nil) then    while hp^<>#0 do      begin      Inc(Result);      hp:=hp+strlen(hp)+1;      end;  FreeEnvironmentStrings(p);end;Function GetEnvironmentString(Index : Integer) : String;var  hp,p : pchar;begin  Result:='';  p:=GetEnvironmentStrings;  hp:=p;  If (Hp<>Nil) then    begin    while (hp^<>#0) and (Index>1) do      begin      Dec(Index);      hp:=hp+strlen(hp)+1;      end;    If (hp^<>#0) then      Result:=StrPas(HP);    end;  FreeEnvironmentStrings(p);end;function ExecuteProcess(Const Path: AnsiString; Const ComLine: AnsiString):integer;var  SI: TStartupInfo;  PI: TProcessInformation;  Proc : THandle;  l    : DWord;  CommandLine : ansistring;  e : EOSError;begin  FillChar(SI, SizeOf(SI), 0);  SI.cb:=SizeOf(SI);  SI.wShowWindow:=1;  { always surround the name of the application by quotes    so that long filenames will always be accepted. But don't    do it if there are already double quotes, since Win32 does not    like double quotes which are duplicated!  }  if pos('"',path)=0 then    CommandLine:='"'+path+'"'  else    CommandLine:=path;  if ComLine <> '' then    CommandLine:=Commandline+' '+ComLine+#0  else    CommandLine := CommandLine + #0;  if not CreateProcess(nil, pchar(CommandLine),    Nil, Nil, False,$20, Nil, Nil, SI, PI) then    begin      e:=EOSError.CreateFmt(SExecuteProcessFailed,[CommandLine,GetLastError]);      e.ErrorCode:=GetLastError;      raise e;    end;  Proc:=PI.hProcess;  if WaitForSingleObject(Proc, dword($ffffffff)) <> $ffffffff then    begin      GetExitCodeProcess(Proc,l);      CloseHandle(Proc);      CloseHandle(PI.hThread);      result:=l;    end  else    begin      e:=EOSError.CreateFmt(SExecuteProcessFailed,[CommandLine,GetLastError]);      e.ErrorCode:=GetLastError;      CloseHandle(Proc);      CloseHandle(PI.hThread);      raise e;    end;end;function ExecuteProcess(Const Path: AnsiString; Const ComLine: Array of AnsiString):integer;var  CommandLine: AnsiString;  I: integer;begin  Commandline := '';  for I := 0 to High (ComLine) do   if Pos (' ', ComLine [I]) <> 0 then    CommandLine := CommandLine + ' ' + '"' + ComLine [I] + '"'   else    CommandLine := CommandLine + ' ' + Comline [I];  ExecuteProcess := ExecuteProcess (Path, CommandLine);end;Procedure Sleep(Milliseconds : Cardinal);begin  Windows.Sleep(MilliSeconds)end;Function GetLastOSError : Integer;begin  Result:=GetLastError;end;{****************************************************************************                              Initialization code****************************************************************************}var   kernel32dll : THandle;Procedure LoadVersionInfo;// and getfreespaceexVar   versioninfo : TOSVERSIONINFO;begin  kernel32dll:=0;  GetDiskFreeSpaceEx:=nil;  versioninfo.dwOSVersionInfoSize:=sizeof(versioninfo);  GetVersionEx(versioninfo);  Win32Platform:=versionInfo.dwPlatformId;  Win32MajorVersion:=versionInfo.dwMajorVersion;  Win32MinorVersion:=versionInfo.dwMinorVersion;  Win32BuildNumber:=versionInfo.dwBuildNumber;  Move (versioninfo.szCSDVersion ,Win32CSDVersion[1],128);  win32CSDVersion[0]:=chr(strlen(pchar(@versioninfo.szCSDVersion)));  if ((versioninfo.dwPlatformId=VER_PLATFORM_WIN32_WINDOWS) and    (versioninfo.dwBuildNUmber>=1000)) or    (versioninfo.dwPlatformId=VER_PLATFORM_WIN32_NT) then    begin       kernel32dll:=LoadLibrary('kernel32');       if kernel32dll<>0 then         GetDiskFreeSpaceEx:=TGetDiskFreeSpaceEx(GetProcAddress(kernel32dll,'GetDiskFreeSpaceExA'));    end;end;function FreeLibrary(hLibModule : THANDLE) : longbool;  stdcall;external 'kernel32' name 'FreeLibrary';function GetVersionEx(var VersionInformation:TOSVERSIONINFO) : longbool;  stdcall;external 'kernel32' name 'GetVersionExA';function LoadLibrary(lpLibFileName : pchar):THandle;  stdcall;external 'kernel32' name 'LoadLibraryA';function GetProcAddress(hModule : THandle;lpProcName : pchar) : pointer;  stdcall;external 'kernel32' name 'GetProcAddress';Const  CSIDL_PROGRAMS                = $0002; { %SYSTEMDRIVE%\Program Files                                      }  CSIDL_PERSONAL                = $0005; { %USERPROFILE%\My Documents                                       }  CSIDL_FAVORITES               = $0006; { %USERPROFILE%\Favorites                                          }  CSIDL_STARTUP                 = $0007; { %USERPROFILE%\Start menu\Programs\Startup                        }  CSIDL_RECENT                  = $0008; { %USERPROFILE%\Recent                                             }  CSIDL_SENDTO                  = $0009; { %USERPROFILE%\Sendto                                             }  CSIDL_STARTMENU               = $000B; { %USERPROFILE%\Start menu                                         }  CSIDL_MYMUSIC                 = $000D; { %USERPROFILE%\Documents\My Music                                 }  CSIDL_MYVIDEO                 = $000E; { %USERPROFILE%\Documents\My Videos                                }  CSIDL_DESKTOPDIRECTORY        = $0010; { %USERPROFILE%\Desktop                                            }  CSIDL_NETHOOD                 = $0013; { %USERPROFILE%\NetHood                                            }  CSIDL_TEMPLATES               = $0015; { %USERPROFILE%\Templates                                          }  CSIDL_COMMON_STARTMENU        = $0016; { %PROFILEPATH%\All users\Start menu                               }  CSIDL_COMMON_PROGRAMS         = $0017; { %PROFILEPATH%\All users\Start menu\Programs                      }  CSIDL_COMMON_STARTUP          = $0018; { %PROFILEPATH%\All users\Start menu\Programs\Startup              }  CSIDL_COMMON_DESKTOPDIRECTORY = $0019; { %PROFILEPATH%\All users\Desktop                                  }  CSIDL_APPDATA                 = $001A; { %USERPROFILE%\Application Data (roaming)                         }  CSIDL_PRINTHOOD               = $001B; { %USERPROFILE%\Printhood                                          }  CSIDL_LOCAL_APPDATA           = $001C; { %USERPROFILE%\Local Settings\Application Data (non roaming)      }  CSIDL_COMMON_FAVORITES        = $001F; { %PROFILEPATH%\All users\Favorites                                }  CSIDL_INTERNET_CACHE          = $0020; { %USERPROFILE%\Local Settings\Temporary Internet Files            }  CSIDL_COOKIES                 = $0021; { %USERPROFILE%\Cookies                                            }  CSIDL_HISTORY                 = $0022; { %USERPROFILE%\Local settings\History                             }  CSIDL_COMMON_APPDATA          = $0023; { %PROFILESPATH%\All Users\Application Data                        }  CSIDL_WINDOWS                 = $0024; { %SYSTEMROOT%                                                     }  CSIDL_SYSTEM                  = $0025; { %SYSTEMROOT%\SYSTEM32 (may be system on 95/98/ME)                }  CSIDL_PROGRAM_FILES           = $0026; { %SYSTEMDRIVE%\Program Files                                      }  CSIDL_MYPICTURES              = $0027; { %USERPROFILE%\My Documents\My Pictures                           }  CSIDL_PROFILE                 = $0028; { %USERPROFILE%                                                    }  CSIDL_PROGRAM_FILES_COMMON    = $002B; { %SYSTEMDRIVE%\Program Files\Common                               }  CSIDL_COMMON_TEMPLATES        = $002D; { %PROFILEPATH%\All Users\Templates                                }  CSIDL_COMMON_DOCUMENTS        = $002E; { %PROFILEPATH%\All Users\Documents                                }  CSIDL_COMMON_ADMINTOOLS       = $002F; { %PROFILEPATH%\All Users\Start Menu\Programs\Administrative Tools }  CSIDL_ADMINTOOLS              = $0030; { %USERPROFILE%\Start Menu\Programs\Administrative Tools           }  CSIDL_COMMON_MUSIC            = $0035; { %PROFILEPATH%\All Users\Documents\my music                       }  CSIDL_COMMON_PICTURES         = $0036; { %PROFILEPATH%\All Users\Documents\my pictures                    }  CSIDL_COMMON_VIDEO            = $0037; { %PROFILEPATH%\All Users\Documents\my videos                      }  CSIDL_CDBURN_AREA             = $003B; { %USERPROFILE%\Local Settings\Application Data\Microsoft\CD Burning }  CSIDL_PROFILES                = $003E; { %PROFILEPATH%                                                    }  CSIDL_FLAG_CREATE             = $8000; { (force creation of requested folder if it doesn't exist yet)     }Type  PFNSHGetFolderPath = Function(Ahwnd: HWND; Csidl: Integer; Token: THandle; Flags: DWord; Path: PChar): HRESULT; stdcall;var  SHGetFolderPath : PFNSHGetFolderPath = Nil;  CFGDLLHandle : THandle = 0;Procedure InitDLL;Var  P : Pointer;begin  CFGDLLHandle:=LoadLibrary('shell32.dll');  if (CFGDLLHandle<>0) then    begin    P:=GetProcAddress(CFGDLLHandle,'SHGetFolderPathA');    If (P=Nil) then      begin      FreeLibrary(CFGDLLHandle);      CFGDllHandle:=0;      end    else      SHGetFolderPath:=PFNSHGetFolderPath(P);    end;  If (P=Nil) then    begin    CFGDLLHandle:=LoadLibrary('shfolder.dll');    if (CFGDLLHandle<>0) then      begin      P:=GetProcAddress(CFGDLLHandle,'SHGetFolderPathA');      If (P=Nil) then        begin        FreeLibrary(CFGDLLHandle);        CFGDllHandle:=0;        end      else        ShGetFolderPath:=PFNSHGetFolderPath(P);      end;    end;  If (@ShGetFolderPath=Nil) then    Raise Exception.Create('Could not determine SHGetFolderPath Function');end;Function GetSpecialDir(ID :  Integer) : String;Var  APath : Array[0..MAX_PATH] of char;begin  Result:='';  if (CFGDLLHandle=0) then    InitDLL;  If (SHGetFolderPath<>Nil) then    begin    if SHGetFolderPath(0,ID or CSIDL_FLAG_CREATE,0,0,@APATH[0])=S_OK then      Result:=IncludeTrailingPathDelimiter(StrPas(@APath[0]));    end;end;Function GetAppConfigDir(Global : Boolean) : String;begin  If Global then    Result:=GetSpecialDir(CSIDL_COMMON_APPDATA)  else    Result:=GetSpecialDir(CSIDL_LOCAL_APPDATA);  If (Result<>'') then    begin      if VendorName<>'' then        Result:=IncludeTrailingPathDelimiter(Result+VendorName);      Result:=Result+ApplicationName;    end  else    Result:=DGetAppConfigDir(Global);end;Function GetAppConfigFile(Global : Boolean; SubDir : Boolean) : String;begin  result:=DGetAppConfigFile(Global,SubDir);end;Procedure InitSysConfigDir;begin  SetLength(SysConfigDir, MAX_PATH);  SetLength(SysConfigDir, GetWindowsDirectory(PChar(SysConfigDir), MAX_PATH));end;{****************************************************************************                    Target Dependent WideString stuff****************************************************************************}function Win32CompareWideString(const s1, s2 : WideString) : PtrInt;  begin    SetLastError(0);    Result:=CompareStringW(LOCALE_USER_DEFAULT,0,pwidechar(s1),      length(s1),pwidechar(s2),length(s2))-2;    if GetLastError<>0 then      RaiseLastOSError;  end;function Win32CompareTextWideString(const s1, s2 : WideString) : PtrInt;  begin    SetLastError(0);    Result:=CompareStringW(LOCALE_USER_DEFAULT,NORM_IGNORECASE,pwidechar(s1),      length(s1),pwidechar(s2),length(s2))-2;    if GetLastError<>0 then      RaiseLastOSError;  end;function Win32AnsiUpperCase(const s: string): string;  begin    if length(s)>0 then      begin        result:=s;        UniqueString(result);        CharUpperBuff(pchar(result),length(result));      end    else      result:='';  end;function Win32AnsiLowerCase(const s: string): string;  begin    if length(s)>0 then      begin        result:=s;        UniqueString(result);        CharLowerBuff(pchar(result),length(result));      end    else      result:='';  end;function Win32AnsiCompareStr(const S1, S2: string): PtrInt;  begin    result:=CompareString(LOCALE_USER_DEFAULT,0,pchar(s1),length(s1),      pchar(s2),length(s2))-2;  end;function Win32AnsiCompareText(const S1, S2: string): PtrInt;  begin    result:=CompareString(LOCALE_USER_DEFAULT,NORM_IGNORECASE,pchar(s1),length(s1),      pchar(s2),length(s2))-2;  end;function Win32AnsiStrComp(S1, S2: PChar): PtrInt;  begin    result:=CompareString(LOCALE_USER_DEFAULT,0,s1,-1,s2,-1)-2;  end;function Win32AnsiStrIComp(S1, S2: PChar): PtrInt;  begin    result:=CompareString(LOCALE_USER_DEFAULT,NORM_IGNORECASE,s1,-1,s2,-1)-2;  end;function Win32AnsiStrLComp(S1, S2: PChar; MaxLen: PtrUInt): PtrInt;  begin    result:=CompareString(LOCALE_USER_DEFAULT,0,s1,maxlen,s2,maxlen)-2;  end;function Win32AnsiStrLIComp(S1, S2: PChar; MaxLen: PtrUInt): PtrInt;  begin    result:=CompareString(LOCALE_USER_DEFAULT,NORM_IGNORECASE,s1,maxlen,s2,maxlen)-2;  end;function Win32AnsiStrLower(Str: PChar): PChar;  begin    CharLower(str);    result:=str;  end;function Win32AnsiStrUpper(Str: PChar): PChar;  begin    CharUpper(str);    result:=str;  end;{ there is a similiar procedure in the system unit which inits the fields which  are relevant already for the system unit }procedure InitWin32Widestrings;  begin//!!!    CharLengthPCharProc : function(const Str: PChar): PtrInt;    widestringmanager.CompareWideStringProc:=@Win32CompareWideString;    widestringmanager.CompareTextWideStringProc:=@Win32CompareTextWideString;    widestringmanager.UpperAnsiStringProc:=@Win32AnsiUpperCase;    widestringmanager.LowerAnsiStringProc:=@Win32AnsiLowerCase;    widestringmanager.CompareStrAnsiStringProc:=@Win32AnsiCompareStr;    widestringmanager.CompareTextAnsiStringProc:=@Win32AnsiCompareText;    widestringmanager.StrCompAnsiStringProc:=@Win32AnsiStrComp;    widestringmanager.StrICompAnsiStringProc:=@Win32AnsiStrIComp;    widestringmanager.StrLCompAnsiStringProc:=@Win32AnsiStrLComp;    widestringmanager.StrLICompAnsiStringProc:=@Win32AnsiStrLIComp;    widestringmanager.StrLowerAnsiStringProc:=@Win32AnsiStrLower;    widestringmanager.StrUpperAnsiStringProc:=@Win32AnsiStrUpper;  end;procedure SetupProcVars;  var    hinstLib : THandle;  begin    SetFilePointerEx:=nil;    hinstLib:=LoadLibrary(KernelDLL);    if hinstLib<>0 then      begin        pointer(SetFilePointerEx):=GetProcAddress(hinstLib,'SetFilePointerEx');        FreeLibrary(hinstLib);      end;  end;Initialization  InitWin32Widestrings;  InitExceptions;       { Initialize exceptions. OS independent }  InitInternational;    { Initialize internationalization settings }  LoadVersionInfo;  InitSysConfigDir;  SetupProcVars;Finalization  DoneExceptions;  if kernel32dll<>0 then   FreeLibrary(kernel32dll); if CFGDLLHandle<>0 then   FreeLibrary(CFGDllHandle);end.
 |