| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919192019211922192319241925192619271928192919301931193219331934193519361937193819391940194119421943194419451946194719481949195019511952195319541955195619571958195919601961196219631964 | {    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}{$MODESWITCH OUT}{ force ansistrings }{$H+}{$modeswitch typehelpers}{$modeswitch advancedrecords}uses  windows;{$DEFINE HAS_SLEEP}{$DEFINE HAS_OSERROR}{$DEFINE HAS_OSCONFIG}{$DEFINE HAS_OSUSERDIR}{$DEFINE HAS_CREATEGUID}{$DEFINE HAS_LOCALTIMEZONEOFFSET}{$DEFINE HAS_GETTICKCOUNT}{$DEFINE HAS_GETTICKCOUNT64}{$DEFINE HAS_FILEDATETIME}{$DEFINE OS_FILESETDATEBYNAME}{$DEFINE HAS_FILEGETDATETIMEINFO}// this target has an fileflush implementation, don't include dummy{$DEFINE SYSUTILS_HAS_FILEFLUSH_IMPL}{ used OS file system APIs use unicodestring }{$define SYSUTILS_HAS_UNICODESTR_FILEUTIL_IMPL}{ OS has an ansistring/single byte environment variable API }{$define SYSUTILS_HAS_ANSISTR_ENVVAR_IMPL}{ OS has a unicodestring/two byte environment variable API }{$define SYSUTILS_HAS_UNICODESTR_ENVVAR_IMPL}{ 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?const  MaxEraCount = 7;var  EraNames: array [1..MaxEraCount] of String;  EraYearOffsets: array [1..MaxEraCount] of Integer;{ 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;function GetFileVersion(const AFileName: UnicodeString): Cardinal;procedure GetFormatSettings;procedure GetLocaleFormatSettings(LCID: Integer; var FormatSettings: TFormatSettings); platform;implementation  uses    sysconst,    windirs;var   FindExInfoDefaults : TFINDEX_INFO_LEVELS = FindExInfoStandard;  FindFirstAdditionalFlags : DWord = 0;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    { useful 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:=GetFileVersionInfoSizeA(pchar(fn),@h);    if size>sizeof(buf) then      begin        getmem(bufp,size);        try          if GetFileVersionInfoA(pchar(fn),h,size,bufp) then            if VerQueryValue(bufp,'\',valrec,valsize) then              result:=valrec^.dwFileVersionMS;        finally          freemem(bufp);        end;      end    else      begin        if GetFileVersionInfoA(pchar(fn),h,size,@buf) then          if VerQueryValue(@buf,'\',valrec,valsize) then            result:=valrec^.dwFileVersionMS;      end;  end;function GetFileVersion(const AFileName:UnicodeString):Cardinal;  var    { useful only as long as we don't need to touch different stack pages }    buf : array[0..3071] of byte;    bufp : pointer;    fn : unicodestring;    valsize,    size : DWORD;    h : DWORD;    valrec : PVSFixedFileInfo;  begin    result:=$fffffff;    fn:=AFileName;    UniqueString(fn);    size:=GetFileVersionInfoSizeW(pwidechar(fn),@h);    if size>sizeof(buf) then      begin        getmem(bufp,size);        try          if GetFileVersionInfoW(pwidechar(fn),h,size,bufp) then            if VerQueryValue(bufp,'\',valrec,valsize) then              result:=valrec^.dwFileVersionMS;        finally          freemem(bufp);        end;      end    else      begin        if GetFileVersionInfoW(pwidechar(fn),h,size,@buf) then          if VerQueryValueW(@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 *)function ConvertEraYearString(Count ,Year,Month,Day : integer) : string; forward;function ConvertEraString(Count ,Year,Month,Day : integer) : string; forward;{ Include platform independent implementation part }{$i sysutils.inc}function GetTempFileName(Dir,Prefix: PChar; uUnique: DWORD; TempFileName: PChar):DWORD;begin  Result:= Windows.GetTempFileNameA(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:rawbytestring) : rawbytestring;{ returns empty string on errors }var  u: unicodestring;begin  { prevent data loss due to unsupported characters in ansi code page }  u:=ExpandUNCFileName(unicodestring(filename));  widestringmanager.Unicode2AnsiMoveProc(punicodechar(u),result,DefaultRTLFileSystemCodePage,length(u));end;function ExpandUNCFileName (const filename:unicodestring) : unicodestring;{ returns empty string on errors }var  s    : unicodestring;  size : dword;  rc   : dword;  buf : pwidechar;begin  s := ExpandFileName (filename);  s := s + #0;  size := max_path;  getmem(buf,size);  try    rc := WNetGetUniversalNameW (pwidechar(s), UNIVERSAL_NAME_INFO_LEVEL, buf, @size);    if rc=ERROR_MORE_DATA then      begin        buf:=reallocmem(buf,size);        rc := WNetGetUniversalNameW (pwidechar(s), UNIVERSAL_NAME_INFO_LEVEL, buf, @size);      end;    if rc = NO_ERROR then      Result := PRemoteNameInfoW(buf)^.lpUniversalName    else if rc = ERROR_NOT_CONNECTED then      Result := filename    else      Result := '';  finally    freemem(buf);  end;end;{****************************************************************************                              File Functions****************************************************************************}const  AccessMode: array[0..2] of Cardinal  = (    GENERIC_READ,    GENERIC_WRITE,    GENERIC_READ or GENERIC_WRITE or FILE_WRITE_ATTRIBUTES);  ShareModes: array[0..4] of Integer = (               0,               0,               FILE_SHARE_READ,               FILE_SHARE_WRITE,               FILE_SHARE_READ or FILE_SHARE_WRITE);function FileFlush(Handle: THandle): Boolean;begin  Result:= FlushFileBuffers(Handle);end;Function FileOpen (Const FileName : unicodestring; Mode : Integer) : THandle;begin  result := CreateFileW(PWideChar(FileName), dword(AccessMode[Mode and 3]),                       dword(ShareModes[(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 : UnicodeString) : THandle;begin  FileCreate:=FileCreate(FileName, fmShareExclusive, 0);end;Function FileCreate (Const FileName : UnicodeString; Rights:longint) : THandle;begin  FileCreate:=FileCreate(FileName, fmShareExclusive, Rights);end;Function FileCreate (Const FileName : UnicodeString; ShareMode : Integer; Rights : Integer) : THandle;begin  Result := CreateFileW(PwideChar(FileName), GENERIC_READ or GENERIC_WRITE,                       dword(ShareModes[(ShareMode and $F0) shr 4]), nil, CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, 0);end;Function FileRead (Handle : THandle; out 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;var  rslt: Int64Rec;begin  rslt := Int64Rec(FOffset);  rslt.lo := SetFilePointer(Handle, rslt.lo, @rslt.hi, Origin);  if (rslt.lo = $FFFFFFFF) and (GetLastError <> 0) then    rslt.hi := $FFFFFFFF;  Result := Int64(rslt);end;Procedure FileClose (Handle : THandle);begin  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 : UnicodeString): Int64;var  Handle: THandle;  FindData: TWin32FindDataW;  tmpdtime    : longint;begin  Handle := FindFirstFileW(Pwidechar(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,tmpdtime) then          begin            result:=tmpdtime;            exit;          end;    end;  Result := -1;end;type  TSymLinkResult = (    slrOk,    slrNoSymLink,    slrError  );function FileGetSymLinkTargetInt(const FileName: UnicodeString; out SymLinkRec: TUnicodeSymLinkRec; RaiseErrorOnMissing: Boolean): TSymLinkResult;{ reparse point specific declarations from Windows headers }const  IO_REPARSE_TAG_MOUNT_POINT = $A0000003;  IO_REPARSE_TAG_SYMLINK = $A000000C;  ERROR_REPARSE_TAG_INVALID = 4393;  FSCTL_GET_REPARSE_POINT = $900A8;  MAXIMUM_REPARSE_DATA_BUFFER_SIZE = 16 * 1024;  SYMLINK_FLAG_RELATIVE = 1;  FILE_FLAG_OPEN_REPARSE_POINT = $200000;  FILE_READ_EA = $8;type  TReparseDataBuffer = record    ReparseTag: ULONG;    ReparseDataLength: Word;    Reserved: Word;    SubstituteNameOffset: Word;    SubstituteNameLength: Word;    PrintNameOffset: Word;    PrintNameLength: Word;    case ULONG of      IO_REPARSE_TAG_MOUNT_POINT: (        PathBufferMount: array[0..4095] of WCHAR);      IO_REPARSE_TAG_SYMLINK: (        Flags: ULONG;        PathBufferSym: array[0..4095] of WCHAR);  end;const  CShareAny = FILE_SHARE_READ or FILE_SHARE_WRITE or FILE_SHARE_DELETE;  COpenReparse = FILE_FLAG_OPEN_REPARSE_POINT or FILE_FLAG_BACKUP_SEMANTICS;  CVolumePrefix = 'Volume';  CGlobalPrefix = '\\?\';var  HFile, Handle: THandle;  PBuffer: ^TReparseDataBuffer;  BytesReturned: DWORD;  guid: TGUID;begin  Result := slrError;  SymLinkRec := Default(TUnicodeSymLinkRec);  HFile := CreateFileW(PUnicodeChar(FileName), FILE_READ_EA, CShareAny, Nil, OPEN_EXISTING, COpenReparse, 0);  if HFile <> INVALID_HANDLE_VALUE then    try      GetMem(PBuffer, MAXIMUM_REPARSE_DATA_BUFFER_SIZE);      try        if DeviceIoControl(HFile, FSCTL_GET_REPARSE_POINT, Nil, 0,             PBuffer, MAXIMUM_REPARSE_DATA_BUFFER_SIZE, @BytesReturned, Nil) then begin          case PBuffer^.ReparseTag of            IO_REPARSE_TAG_MOUNT_POINT: begin              SymLinkRec.TargetName := WideCharLenToString(                @PBuffer^.PathBufferMount[4 { skip start '\??\' } +                  PBuffer^.SubstituteNameOffset div SizeOf(WCHAR)],                PBuffer^.SubstituteNameLength div SizeOf(WCHAR) - 4);              if (Length(SymLinkRec.TargetName) = Length(CVolumePrefix) + 2 { brackets } + 32 { guid } + 4 { - } + 1 { \ }) and                  (Copy(SymLinkRec.TargetName, 1, Length(CVolumePrefix)) = CVolumePrefix) and                  TryStringToGUID(String(Copy(SymLinkRec.TargetName, Length(CVolumePrefix) + 1, Length(SymLinkRec.TargetName) - Length(CVolumePrefix) - 1)), guid) then                SymLinkRec.TargetName := CGlobalPrefix + SymLinkRec.TargetName;            end;            IO_REPARSE_TAG_SYMLINK: begin              SymLinkRec.TargetName := WideCharLenToString(                @PBuffer^.PathBufferSym[PBuffer^.PrintNameOffset div SizeOf(WCHAR)],                PBuffer^.PrintNameLength div SizeOf(WCHAR));              if (PBuffer^.Flags and SYMLINK_FLAG_RELATIVE) <> 0 then                SymLinkRec.TargetName := ExpandFileName(ExtractFilePath(FileName) + SymLinkRec.TargetName);            end;          end;          if SymLinkRec.TargetName <> '' then begin            { the fields of WIN32_FILE_ATTRIBUTE_DATA match with the first fields of WIN32_FIND_DATA }            if GetFileAttributesExW(PUnicodeChar(SymLinkRec.TargetName), GetFileExInfoStandard, @SymLinkRec.FindData) then begin              SymLinkRec.Attr := SymLinkRec.FindData.dwFileAttributes;              SymLinkRec.Size := QWord(SymLinkRec.FindData.nFileSizeHigh) shl 32 + QWord(SymLinkRec.FindData.nFileSizeLow);            end else if RaiseErrorOnMissing then              raise EDirectoryNotFoundException.Create(SysErrorMessage(GetLastOSError))            else              SymLinkRec.TargetName := '';          end else begin            SetLastError(ERROR_REPARSE_TAG_INVALID);            Result := slrNoSymLink;          end;        end else          SetLastError(ERROR_REPARSE_TAG_INVALID);      finally        FreeMem(PBuffer);      end;    finally      CloseHandle(HFile);    end;  if SymLinkRec.TargetName <> '' then    Result := slrOkend;function FileGetSymLinkTarget(const FileName: UnicodeString; out SymLinkRec: TUnicodeSymLinkRec): Boolean;begin  Result := FileGetSymLinkTargetInt(FileName, SymLinkRec, True) = slrOk;end;function FileOrDirExists(const FileOrDirName: UnicodeString; CheckDir: Boolean; FollowLink: Boolean): Boolean;const  CDirAttributes: array[Boolean] of DWORD = (0, FILE_ATTRIBUTE_DIRECTORY);  function FoundByEnum: Boolean;  var    FindData: TWin32FindDataW;    Handle: THandle;  begin    { FindFirstFileEx is faster than FindFirstFile }    Handle := FindFirstFileExW(PUnicodeChar(FileOrDirName), FindExInfoDefaults , @FindData,                FindExSearchNameMatch, Nil, 0);    Result := Handle <> INVALID_HANDLE_VALUE;    if Result then begin      Windows.FindClose(Handle);      Result := (FindData.dwFileAttributes and FILE_ATTRIBUTE_DIRECTORY) = CDirAttributes[CheckDir];    end;  end;const  CNotExistsErrors = [    ERROR_FILE_NOT_FOUND,    ERROR_PATH_NOT_FOUND,    ERROR_INVALID_NAME, // protects from names in the form of masks like '*'    ERROR_INVALID_DRIVE,    ERROR_NOT_READY,    ERROR_INVALID_PARAMETER,    ERROR_BAD_PATHNAME,    ERROR_BAD_NETPATH,    ERROR_BAD_NET_NAME  ];var  Attr : DWord;  slr : TUnicodeSymLinkRec;  res : TSymLinkResult;begin  Attr := GetFileAttributesW(PUnicodeChar(FileOrDirName));  if Attr = INVALID_FILE_ATTRIBUTES then    Result := not (GetLastError in CNotExistsErrors) and FoundByEnum  else begin    Result := (Attr and FILE_ATTRIBUTE_DIRECTORY) = CDirAttributes[CheckDir];    if Result and FollowLink and ((Attr and FILE_ATTRIBUTE_REPARSE_POINT) <> 0) then begin      res := FileGetSymLinkTargetInt(FileOrDirName, slr, False);      case res of        slrOk:          Result := FileOrDirExists(slr.TargetName, CheckDir, False);        slrNoSymLink:          Result := True;        else          Result := False;      end;    end;  end;end;Function FileExists (Const FileName : UnicodeString; FollowLink : Boolean) : Boolean;begin  Result := FileOrDirExists(FileName, False, FollowLink);end;Function DirectoryExists (Const Directory : UnicodeString; FollowLink : Boolean) : Boolean;begin  Result := FileOrDirExists(Directory, True, FollowLink);end;Function FindMatch(var f: TAbstractSearchRec; var Name: UnicodeString) : Longint;var  tmpdtime : longint;begin  { Find file with correct attribute }  While (F.FindData.dwFileAttributes and cardinal(F.ExcludeAttr))<>0 do   begin     if not FindNextFileW (F.FindHandle,F.FindData) then      begin        Result:=GetLastError;        exit;      end;   end;  { Convert some attributes back }  WinToDosTime(F.FindData.ftLastWriteTime,tmpdtime);  F.Time:=tmpdtime;  f.size:=F.FindData.NFileSizeLow+(qword(maxdword)+1)*F.FindData.NFileSizeHigh;  f.attr:=F.FindData.dwFileAttributes;  Name:=F.FindData.cFileName;  Result:=0;end;Procedure InternalFindClose (var Handle: THandle; var FindData: TFindData);begin   if Handle <> INVALID_HANDLE_VALUE then    begin    Windows.FindClose(Handle);    Handle:=INVALID_HANDLE_VALUE;    end;end;function GetFinalPathNameByHandle(aHandle : THandle; Buf : LPSTR; BufSize : DWord; Flags : DWord) : DWORD; external 'kernel32' name 'GetFinalPathNameByHandleA';Const  VOLUME_NAME_NT = $2;Function FollowSymlink(const aLink: String): String;Var  Attrs: Cardinal;  aHandle: THandle;  oFlags: DWord;  Buf : Array[0..Max_Path] of AnsiChar;  Len : Integer;begin  Result:='';  FillChar(Buf,MAX_PATH+1,0);  if Not FileExists(aLink,False) then     exit;  if not CheckWin32Version(6, 0) then     exit;  Attrs:=GetFileAttributes(PChar(aLink));  if (Attrs=INVALID_FILE_ATTRIBUTES) or ((Attrs and faSymLink)=0) then    exit;  oFLags:=0;  // https://learn.microsoft.com/en-us/windows/win32/api/fileapi/nf-fileapi-createfilea  if (Attrs and faDirectory)=faDirectory then    oFlags:=FILE_FLAG_BACKUP_SEMANTICS;  aHandle:=CreateFile(PChar(aLink),GENERIC_READ,FILE_SHARE_READ,nil,OPEN_EXISTING,oFlags,0);  if aHandle=INVALID_HANDLE_VALUE then    exit;  try    Len:=GetFinalPathNameByHandle(aHandle,@Buf,MAX_PATH,VOLUME_NAME_NT);    If Len<=0 then       exit;     Result:=StrPas(PChar(@Buf));  finally    CloseHandle(aHandle);  end;end;Function InternalFindFirst (Const Path : UnicodeString; Attr : Longint; out Rslt : TAbstractSearchRec; var Name : UnicodeString) : Longint;begin  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:=FindFirstFileExW(PUnicodeChar(Path), FindExInfoDefaults , @Rslt.FindData,                      FindExSearchNameMatch, Nil, FindFirstAdditionalFlags);  If Rslt.FindHandle=Invalid_Handle_value then   begin     Result:=GetLastError;     exit;   end;  { Find file with correct attribute }  Result:=FindMatch(Rslt,Name);  if (Result<>0) then    InternalFindClose(Rslt.FindHandle,Rslt.FindData);end;Function InternalFindNext (Var Rslt : TAbstractSearchRec; var Name: UnicodeString) : Longint;begin  if FindNextFileW(Rslt.FindHandle, Rslt.FindData) then    Result := FindMatch(Rslt, Name)  else    Result := GetLastError;end;function FileGetDateTimeInfo(const FileName: string;  out DateTime: TDateTimeInfoRec; FollowLink: Boolean = True): Boolean;var  Data: TWin32FindDataW;  FN: unicodestring;begin  Result := False;  SetLastError(ERROR_SUCCESS);  FN:=FileName;  if Not GetFileAttributesExW(PWideChar(FileName), GetFileExInfoStandard, @Data) then    exit;  if ((Data.dwFileAttributes and faSymlink)=faSymlink) then    begin    if FollowLink then      begin      FN:=FollowSymlink(FileName);      if FN='' then         exit;       if not GetFileAttributesExW(PWideChar(FN), GetFileExInfoStandard, @Data) then        exit;      end;    end;       DateTime.Data:=Data;  Result:=True;end;Function FileGetDate (Handle : THandle) : Int64;Var  FT : TFileTime;  tmpdtime : longint;begin  If GetFileTime(Handle,nil,nil,@ft) and     WinToDosTime(FT,tmpdtime) then    begin      result:=tmpdtime;      exit;    end;  Result:=-1;end;Function FileGetDate (Handle : THandle; out FileDateTime: TDateTime) : Boolean;Var  FT : TFileTime;begin  Result :=     GetFileTime(Handle,nil,nil,@ft) and     FindDataTimeToDateTime(FT, FileDateTime);end;Function FileGetDateUTC (Handle : THandle; out FileDateTimeUTC: TDateTime) : Boolean;Var  FT : TFileTime;begin  Result :=     GetFileTime(Handle,nil,nil,@ft) and     FindDataTimeToUTC(FT, FileDateTimeUTC);end;Function FileSetDate (Handle : THandle;Age : Int64) : Longint;Var  FT: TFileTime;begin  Result := 0;  if DosToWinTime(Age,FT) and    SetFileTime(Handle, nil, nil, @FT) then    Exit;  Result := GetLastError;end;Function FileSetDate (Handle : THandle; const FileDateTime: TDateTime) : Longint;var  FT: TFiletime;  LT: TFiletime;  ST: TSystemTime;begin  DateTimeToSystemTime(FileDateTime,ST);  if SystemTimeToFileTime(ST,LT) and LocalFileTimeToFileTime(LT,FT)    and SetFileTime(Handle,nil,nil,@FT) then    Result:=0  else    Result:=GetLastError;end;Function FileSetDateUTC (Handle : THandle; const FileDateTimeUTC: TDateTime) : Longint;var  FT: TFiletime;  ST: TSystemTime;begin  DateTimeToSystemTime(FileDateTimeUTC,ST);  if SystemTimeToFileTime(ST,FT) and SetFileTime(Handle,nil,nil,@FT) then    Result:=0  else    Result:=GetLastError;end;{$IFDEF OS_FILESETDATEBYNAME}Function FileSetDate (Const FileName : UnicodeString;Age : Int64) : Longint;Var  fd : THandle;begin  FD := CreateFileW (PWideChar (FileName), GENERIC_READ or GENERIC_WRITE,                     FILE_SHARE_WRITE, nil, OPEN_EXISTING,                     FILE_FLAG_BACKUP_SEMANTICS, 0);  If (Fd<>feInvalidHandle) then    try      Result:=FileSetDate(fd,Age);    finally      FileClose(fd);    end  else    Result:=GetLastOSError;end;Function FileSetDate (Const FileName : UnicodeString;const FileDateTime : TDateTime) : Longint;Var  fd : THandle;begin  FD := CreateFileW (PWideChar (FileName), GENERIC_READ or GENERIC_WRITE,                     FILE_SHARE_WRITE, nil, OPEN_EXISTING,                     FILE_FLAG_BACKUP_SEMANTICS, 0);  If (Fd<>feInvalidHandle) then    try      Result:=FileSetDate(fd,FileDateTime);    finally      FileClose(fd);    end  else    Result:=GetLastOSError;end;Function FileSetDateUTC (Const FileName : UnicodeString;const FileDateTimeUTC : TDateTime) : Longint;Var  fd : THandle;begin  FD := CreateFileW (PWideChar (FileName), GENERIC_READ or GENERIC_WRITE,                     FILE_SHARE_WRITE, nil, OPEN_EXISTING,                     FILE_FLAG_BACKUP_SEMANTICS, 0);  If (Fd<>feInvalidHandle) then    try      Result:=FileSetDateUTC(fd,FileDateTimeUTC);    finally      FileClose(fd);    end  else    Result:=GetLastOSError;end;{$ENDIF}Function FileGetAttr (Const FileName : UnicodeString) : Longint;begin  Result:=Longint(GetFileAttributesW(PWideChar(FileName)));end;Function FileSetAttr (Const Filename : UnicodeString; Attr: longint) : Longint;begin  if SetFileAttributesW(PWideChar(FileName), Attr) then    Result:=0  else    Result := GetLastError;end;Function DeleteFile (Const FileName : UnicodeString) : Boolean;begin  Result:=Windows.DeleteFileW(PWidechar(FileName));end;Function RenameFile (Const OldName, NewName : UnicodeString) : Boolean;begin  Result := MoveFileW(PWideChar(OldName), PWideChar(NewName));end;{****************************************************************************                              Disk Functions****************************************************************************}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 : dword;  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 : dword;  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;{****************************************************************************                              Time Functions****************************************************************************}Procedure GetLocalTime(var SystemTime: TSystemTime);begin  windows.Getlocaltime(SystemTime);end;function GetUniversalTime(var SystemTime: TSystemTime): Boolean;begin  windows.GetSystemTime(SystemTime);  Result:=True;end;function GetLocalTimeOffset: Integer;var  TZInfo: TTimeZoneInformation;begin   case GetTimeZoneInformation(TZInfo) of     TIME_ZONE_ID_UNKNOWN:       Result := TZInfo.Bias;     TIME_ZONE_ID_STANDARD:       Result := TZInfo.Bias + TZInfo.StandardBias;     TIME_ZONE_ID_DAYLIGHT:       Result := TZInfo.Bias + TZInfo.DaylightBias;     else       Result := 0;   end;end;type  TGetTimeZoneInformationForYear = function(wYear: USHORT; lpDynamicTimeZoneInformation: PDynamicTimeZoneInformation;    var lpTimeZoneInformation: TTimeZoneInformation): BOOL;stdcall;var  GetTimeZoneInformationForYear:TGetTimeZoneInformationForYear=nil;function GetLocalTimeOffset(const DateTime: TDateTime; const InputIsUTC: Boolean; out Offset: Integer): Boolean;var  Year: Integer;const  DaysPerWeek = 7;  // MonthOf and YearOf are not available in SysUtils  function MonthOf(const AValue: TDateTime): Word;  var    Y,D : Word;  begin    DecodeDate(AValue,Y,Result,D);  end;  function YearOf(const AValue: TDateTime): Word;  var    D,M : Word;  begin    DecodeDate(AValue,Result,D,M);  end;  function RelWeekDayToDateTime(const SysTime: TSystemTime): TDateTime;  var    WeekDay, IncDays: Integer;  begin    // get first day in month    Result := EncodeDate(Year, SysTime.Month, 1);    WeekDay := DayOfWeek(Result)-1;    // get the correct first weekday in month    IncDays := SysTime.wDayOfWeek-WeekDay;    if IncDays<0 then      Inc(IncDays, DaysPerWeek);    // inc weeks    Result := Result+IncDays+DaysPerWeek*(SysTime.Day-1);    // SysTime.DayOfWeek=5 means the last one - check if we are not in the next month    while (MonthOf(Result)>SysTime.Month) do      Result := Result-DaysPerWeek;    Result := Result+EncodeTime(SysTime.Hour, SysTime.Minute, SysTime.Second, SysTime.Millisecond);  end;var  TZInfo: TTimeZoneInformation;  DSTStart, DSTEnd: TDateTime;begin  if not Assigned(GetTimeZoneInformationForYear) then    Exit(False);  Year := YearOf(DateTime);  TZInfo := Default(TTimeZoneInformation);  if not GetTimeZoneInformationForYear(Year, nil, TZInfo) then    Exit(False);  if (TZInfo.StandardDate.Month>0) and (TZInfo.DaylightDate.Month>0) then  begin // there is DST    // DaylightDate and StandardDate are local times    DSTStart := RelWeekDayToDateTime(TZInfo.DaylightDate);    DSTEnd := RelWeekDayToDateTime(TZInfo.StandardDate);    if InputIsUTC then    begin      DSTStart := DSTStart + (TZInfo.Bias+TZInfo.StandardBias)/MinsPerDay;      DSTEnd := DSTEnd + (TZInfo.Bias+TZInfo.DaylightBias)/MinsPerDay;    end;    if (DSTStart<=DateTime) and (DateTime<DSTEnd) then      Offset := TZInfo.Bias+TZInfo.DaylightBias    else      Offset := TZInfo.Bias+TZInfo.StandardBias;  end else // no DST    Offset := TZInfo.Bias;  Result := True;end;function GetTickCount: LongWord;begin  Result := Windows.GetTickCount;end;{$IFNDEF WINCE}type  TGetTickCount64 = function : QWord; stdcall;var  WinGetTickCount64: TGetTickCount64 = Nil;{$ENDIF}function GetTickCount64: QWord;begin{$IFNDEF WINCE}  if Assigned(WinGetTickCount64) then    Exit(WinGetTickCount64());  { on Vista and newer there is a GetTickCount64 implementation }  if Win32MajorVersion >= 6 then begin    WinGetTickCount64 := TGetTickCount64(GetProcAddress(GetModuleHandle('kernel32.dll'), 'GetTickCount64'));    Result := WinGetTickCount64();  end else{$ENDIF}    Result := Windows.GetTickCount;end;{****************************************************************************                              Misc Functions****************************************************************************}procedure sysbeep;begin  MessageBeep(0);end;{****************************************************************************                              Locale Functions****************************************************************************}function GetLocaleStr(LID, LT: Longint; const Def: string): AnsiString;var  L: Integer;  Buf: unicodestring;begin  L := GetLocaleInfoW(LID, LT, nil, 0);  if L > 0 then    begin      SetLength(Buf,L-1); // L includes terminating NULL      if l>1 Then        L := GetLocaleInfoW(LID, LT, @Buf[1], L);      result:=buf;    end  else    Result := Def;end;function GetLocaleChar(LID, LT: Longint; Def: Char): Char;var  Buf: array[0..3] of Char; // sdate allows 4 chars.begin  if GetLocaleInfoA(LID, LT, Buf, sizeof(buf)) > 0 then    Result := Buf[0]  else    Result := Def;end;function ConvertEraString(Count ,Year,Month,Day : integer) : string;  var    ASystemTime: TSystemTime;    wbuf: array[0..100] of WideChar;    ALCID : LCID;begin  Result := ''; if (Count<=0) then exit;  DateTimeToSystemTime(EncodeDate(Year,Month,Day),ASystemTime);  ALCID := GetThreadLocale;//  ALCID := SysLocale.DefaultLCID;  if GetDateFormatW(ALCID , DATE_USE_ALT_CALENDAR      , @ASystemTime, PWChar('gg')      , @wbuf, SizeOf(wbuf)) > 0 then  begin    if Count = 1 then      wbuf[1] := #0;    Result := string(WideString(wbuf));  end;end;function ConvertEraYearString(Count ,Year,Month,Day : integer) : string;  var    ALCID : LCID;    ASystemTime : TSystemTime;    AFormatText : string;    buf : array[0..100] of Char;begin  Result := '';  DateTimeToSystemTime(EncodeDate(Year,Month,Day),ASystemTime);  if Count <= 2 then    AFormatText := 'yy'  else    AFormatText := 'yyyy';  ALCID := GetThreadLocale;//  ALCID := SysLocale.DefaultLCID;  if GetDateFormatA(ALCID, DATE_USE_ALT_CALENDAR      , @ASystemTime, PChar(AFormatText)      , @buf, SizeOf(buf)) > 0 then  begin    Result := buf;    if (Count = 1) and (Result[1] = '0') then      Result := Copy(Result, 2, Length(Result)-1);  end;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;function EnumEraNames(Names: PChar): WINBOOL; stdcall;var  i : integer;begin  Result := False;  for i := Low(EraNames) to High(EraNames) do   if (EraNames[i] = '') then   begin     EraNames[i] := Names;     Result := True;     break;   end;end;function EnumEraYearOffsets(YearOffsets: PChar): WINBOOL; stdcall;var  i : integer;begin  Result := False;  for i := Low(EraYearOffsets) to High(EraYearOffsets) do   if (EraYearOffsets[i] = -1) then   begin     EraYearOffsets[i] := StrToIntDef(YearOffsets, 0);     Result := True;     break;   end;end;procedure GetEraNamesAndYearOffsets;  var    ACALID : CALID;    ALCID : LCID;    buf : array[0..10] of char;    i : integer;begin  for i:= 1 to MaxEraCount do   begin     EraNames[i] := '';  EraYearOffsets[i] := -1;   end;  ALCID := GetThreadLocale;  if GetLocaleInfoA(ALCID , LOCALE_IOPTIONALCALENDAR, buf, sizeof(buf)) <= 0 then exit;  ACALID := StrToIntDef(buf,1);  if ACALID in [3..5] then  begin    EnumCalendarInfoA(@EnumEraNames, ALCID, ACALID , CAL_SERASTRING);    EnumCalendarInfoA(@EnumEraYearOffsets, ALCID, ACALID, CAL_IYEAROFFSETRANGE);  end;(*1 CAL_GREGORIAN Gregorian (localized)2 CAL_GREGORIAN_US Gregorian (English strings always)3 CAL_JAPAN Japanese Emperor Era4 CAL_TAIWAN Taiwan Calendar5 CAL_KOREA Korean Tangun Era6 CAL_HIJRI Hijri (Arabic Lunar)7 CAL_THAI Thai8 CAL_HEBREW Hebrew (Lunar)9 CAL_GREGORIAN_ME_FRENCH Gregorian Middle East French10 CAL_GREGORIAN_ARABIC Gregorian Arabic11 CAL_GREGORIAN_XLIT_ENGLISH Gregorian transliterated English12 CAL_GREGORIAN_XLIT_FRENCH Gregorian transliterated French23 CAL_UMALQURA Windows Vista or later: Um Al Qura (Arabic lunar) calendar*)end;procedure GetLocaleFormatSettings(LCID: Integer; var FormatSettings: TFormatSettings);  function FixSeparator(const Format: string; const FromSeparator, ToSeparator: Char): string;  var    R: PChar;  begin    if (Format='') or (FromSeparator=ToSeparator) then      Exit(Format);    Result := Copy(Format, 1);    R := PChar(Result);    while R^<>#0 do      begin      if R^=FromSeparator then        R^:=ToSeparator;      Inc(R);      end;  end;var  HF  : Shortstring;  LID : Windows.LCID;  I,Day : longint;begin  LID := LCID;  with FormatSettings do    begin  { 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 := FixSeparator(GetLocaleStr(LID, LOCALE_SSHORTDATE, 'm/d/yy'), DateSeparator, '/');      LongDateFormat := FixSeparator(GetLocaleStr(LID, LOCALE_SLONGDATE, 'mmmm d, yyyy'), DateSeparator, '/');      { 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';      ShortTimeFormat := HF+':nn';      LongTimeFormat := HF + ':nn:ss';      { 12-hour system support }      if GetLocaleInt(LID, LOCALE_ITIME, 1) = 0 then      begin        LongTimeFormat := LongTimeFormat + ' AMPM';        ShortTimeFormat := ShortTimeFormat + ' AMPM';      end;      { 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);      ListSeparator := GetLocaleChar(LID, LOCALE_SLIST, ',');    end;end;procedure GetFormatSettings;begin  GetlocaleFormatSettings(GetThreadLocale, DefaultFormatSettings);end;Procedure InitLeadBytes;var  I,B,C,E: Byte;  Info: TCPInfo;begin  GetCPInfo(CP_ACP,Info);  I:=0;  With Info do    begin    B:=LeadByte[i];    E:=LeadByte[i+1];    while (I<MAX_LEADBYTES) and (B<>0) and (E<>0) do      begin      for C:=B to E do        Include(LeadBytes,AnsiChar(C));      Inc(I,2);      if (I<MAX_LEADBYTES) then        begin        B:=LeadByte[i];        E:=LeadByte[i+1];        end;      end;    end;   end;Procedure InitInternational;var{$if defined(CPU386) or defined(CPUX86_64)}  { A call to GetSystemMetrics changes the value of the 8087 Control Word on    Pentium4 with WinXP SP2 }  old8087CW: word;{$endif}  DefaultCustomLocaleID : LCID;   // typedef DWORD LCID;  DefaultCustomLanguageID : Word; // typedef WORD LANGID;begin  /// workaround for Windows 7 bug, see bug report #18574  SetThreadLocale(GetUserDefaultLCID);  InitInternationalGeneric;{$if defined(CPU386) or defined(CPUX86_64)}  old8087CW:=Get8087CW;{$endif}  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  DefaultCustomLocaleID := GetThreadLocale;  if DefaultCustomLocaleID <> 0 then    begin      { Locale Identifiers        +-------------+---------+-------------------------+        |   Reserved  | Sort ID |      Language ID        |        +-------------+---------+-------------------------+        31         20 19      16 15                       0   bit }      DefaultCustomLanguageID := DefaultCustomLocaleID and $FFFF; // 2^16      if DefaultCustomLanguageID <> 0 then        begin          SysLocale.DefaultLCID := DefaultCustomLocaleID;          { Language Identifiers            +-------------------------+-------------------------+            |     SubLanguage ID      |   Primary Language ID   |            +-------------------------+-------------------------+            15                      10  9                         0   bit  }          SysLocale.PriLangID := DefaultCustomLanguageID and $3ff; // 2^10          SysLocale.SubLangID := DefaultCustomLanguageID shr 10;        end;     end;{$if defined(CPU386) or defined(CPUX86_64)}  Set8087CW(old8087CW);{$endif}  GetFormatSettings;  if SysLocale.FarEast then GetEraNamesAndYearOffsets;end;{****************************************************************************                           Target Dependent****************************************************************************}function SysErrorMessage(ErrorCode: Integer): String;var  MsgBuffer: PWideChar;  Msg: UnicodeString;  len: longint;begin  len := FormatMessageW(FORMAT_MESSAGE_FROM_SYSTEM or                        FORMAT_MESSAGE_IGNORE_INSERTS or                        FORMAT_MESSAGE_ALLOCATE_BUFFER,                        nil,                        ErrorCode,                        MakeLangId(LANG_NEUTRAL, SUBLANG_DEFAULT),                        PWideChar(@MsgBuffer),                        0,                        nil);  // Remove trailing #13#10  if (len > 1) and (MsgBuffer[len - 2] = #13) and (MsgBuffer[len - 1] = #10) then    Dec(len, 2);  SetString(Msg, PUnicodeChar(MsgBuffer), len);  LocalFree(HLOCAL(MsgBuffer));  Result := Msg;end;{****************************************************************************                              Initialization code****************************************************************************}{$push}{ GetEnvironmentStrings cannot be checked by CheckPointer function }{$checkpointer off}Function GetEnvironmentVariable(Const EnvVar : String) : String;var   oemenvvar, oemstr : RawByteString;   i, hplen : longint;   hp,p : pchar;begin   oemenvvar:=uppercase(envvar);   SetCodePage(oemenvvar,CP_OEMCP);   Result:='';   p:=GetEnvironmentStringsA;   hp:=p;   while hp^<>#0 do     begin        oemstr:=hp;        { cache length, may change after uppercasing depending on code page }        hplen:=length(oemstr);        { all environment variables are encoded in the oem code page }        SetCodePage(oemstr,CP_OEMCP,false);        i:=pos('=',oemstr);        if uppercase(copy(oemstr,1,i-1))=oemenvvar then          begin             Result:=copy(oemstr,i+1,length(oemstr)-i);             break;          end;        { next string entry}        hp:=hp+hplen+1;     end;   FreeEnvironmentStringsA(p);end;Function GetEnvironmentVariable(Const EnvVar : UnicodeString) : UnicodeString;var   s, upperenv : Unicodestring;   i : longint;   hp,p : pwidechar;begin   Result:='';   p:=GetEnvironmentStringsW;   hp:=p;   upperenv:=uppercase(envvar);   while hp^<>#0 do     begin        s:=hp;        i:=pos('=',s);        if uppercase(copy(s,1,i-1))=upperenv then          begin             Result:=copy(s,i+1,length(s)-i);             break;          end;        { next string entry}        hp:=hp+strlen(hp)+1;     end;   FreeEnvironmentStringsW(p);end;Function GetEnvironmentVariableCount : Integer;var  hp,p : pchar;begin  Result:=0;  p:=GetEnvironmentStringsA;  hp:=p;  If (Hp<>Nil) then    while hp^<>#0 do      begin      Inc(Result);      hp:=hp+strlen(hp)+1;      end;  FreeEnvironmentStringsA(p);end;Function GetEnvironmentString(Index : Integer) : {$ifdef FPC_RTL_UNICODE}UnicodeString{$else}AnsiString{$endif};var  hp,p : pchar;{$ifdef FPC_RTL_UNICODE}  tmpstr : RawByteString;{$endif}begin  Result:='';  p:=GetEnvironmentStringsA;  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      begin{$ifdef FPC_RTL_UNICODE}        tmpstr:=hp;        SetCodePage(tmpstr,CP_OEMCP,false);        Result:=tmpstr;{$else}        Result:=hp;        SetCodePage(RawByteString(Result),CP_OEMCP,false);{$endif}      end;    end;  FreeEnvironmentStringsA(p);end;{$pop}function ExecuteProcess(Const Path: RawByteString; Const ComLine: RawByteString;Flags:TExecuteFlags=[]):integer;begin  result:=ExecuteProcess(Unicodestring(Path),UnicodeString(ComLine),Flags);end;function ExecuteProcess(Const Path: UnicodeString; Const ComLine: UnicodeString;Flags:TExecuteFlags=[]):integer;// win specific  functionvar  SI: TStartupInfoW;  PI: TProcessInformation;  Proc : THandle;  l    : DWord;  CommandLine : unicodestring;  e : EOSError;  ExecInherits : longbool;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;  ExecInherits:=ExecInheritsHandles in Flags;  if not CreateProcessW(nil, pwidechar(CommandLine),    Nil, Nil, ExecInherits,$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: RawByteString; Const ComLine: Array of RawByteString;Flags:TExecuteFlags=[]):integer;var  CommandLine: UnicodeString;  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 (UnicodeString(Path), CommandLine,Flags);end;function ExecuteProcess(Const Path: UnicodeString; Const ComLine: Array of UnicodeString;Flags:TExecuteFlags=[]):integer;var  CommandLine: UnicodeString;  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,Flags);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  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)));  kernel32dll:=GetModuleHandle('kernel32');  if kernel32dll<>0 then    GetDiskFreeSpaceEx:=TGetDiskFreeSpaceEx(GetProcAddress(kernel32dll,'GetDiskFreeSpaceExA'));  if Win32MajorVersion<6 then     FindExInfoDefaults := FindExInfoStandard; // also searches SFNs. XP only.  if (Win32MajorVersion>=6) and (Win32MinorVersion>=1) then     FindFirstAdditionalFlags := FIND_FIRST_EX_LARGE_FETCH; // win7 and 2008R2+  // GetTimeZoneInformationForYear is supported only on Vista and newer  if (kernel32dll<>0) and (Win32MajorVersion>=6) then    GetTimeZoneInformationForYear:=TGetTimeZoneInformationForYear(GetProcAddress(kernel32dll,'GetTimeZoneInformationForYear'));end;Function GetAppConfigDir(Global : Boolean) : String;begin  If Global then    Result:=GetWindowsSpecialDir(CSIDL_COMMON_APPDATA)  else    Result:=GetWindowsSpecialDir(CSIDL_LOCAL_APPDATA);  If (Result<>'') then    begin      if VendorName<>'' then        Result:=IncludeTrailingPathDelimiter(Result+VendorName);      Result:=IncludeTrailingPathDelimiter(Result+ApplicationName);    end  else    Result:=IncludeTrailingPathDelimiter(DGetAppConfigDir(Global));end;Function GetAppConfigFile(Global : Boolean; SubDir : Boolean) : String;begin  result:=DGetAppConfigFile(Global,SubDir);end;Function GetUserDir : String;begin  Result:=GetWindowsSpecialDir(CSIDL_PROFILE);end;Procedure InitSysConfigDir;begin  SetLength(SysConfigDir, MAX_PATH);  SetLength(SysConfigDir, GetWindowsDirectoryA(PChar(SysConfigDir), MAX_PATH));end;{****************************************************************************                    Target Dependent WideString stuff****************************************************************************}{ This is the case of Win9x. Limited to current locale of course, but it's better  than not working at all. }function DoCompareStringA(P1, P2: PWideChar; L1, L2: PtrUInt; Flags: DWORD): PtrInt;  var    a1, a2: AnsiString;  begin    if L1>0 then      widestringmanager.Wide2AnsiMoveProc(P1,a1,DefaultSystemCodePage,L1);    if L2>0 then      widestringmanager.Wide2AnsiMoveProc(P2,a2,DefaultSystemCodePage,L2);    SetLastError(0);    Result:=CompareStringA(LOCALE_USER_DEFAULT,Flags,pchar(a1),      length(a1),pchar(a2),length(a2))-2;  end;function DoCompareStringW(P1, P2: PWideChar; L1, L2: PtrUInt; Flags: DWORD): PtrInt;  begin    SetLastError(0);    Result:=CompareStringW(LOCALE_USER_DEFAULT,Flags,P1,L1,P2,L2)-2;    if GetLastError=0 then      Exit;    if GetLastError=ERROR_CALL_NOT_IMPLEMENTED then  // Win9x case      Result:=DoCompareStringA(P1, P2, L1, L2, Flags);    if GetLastError<>0 then      RaiseLastOSError;  end;const  WinAPICompareFlags : array [TCompareOption] of LongWord    = ({LINGUISTIC_IGNORECASE,  LINGUISTIC_IGNOREDIACRITIC, }NORM_IGNORECASE{,       NORM_IGNOREKANATYPE, NORM_IGNORENONSPACE, NORM_IGNORESYMBOLS, NORM_IGNOREWIDTH,       NORM_LINGUISTIC_CASING, SORT_DIGITSASNUMBERS, SORT_STRINGSORT});function Win32CompareWideString(const s1, s2 : WideString; Options : TCompareOptions) : PtrInt;Var  O : LongWord;  CO : TCompareOption;begin  O:=0;  for CO in TCompareOption do    if CO in Options then      O:=O or WinAPICompareFlags[CO];  Result:=DoCompareStringW(PWideChar(s1), PWideChar(s2), Length(s1), Length(s2), O);end;function Win32CompareTextWideString(const s1, s2 : WideString) : PtrInt;  begin    Result:=DoCompareStringW(PWideChar(s1), PWideChar(s2), Length(s1), Length(s2), NORM_IGNORECASE);  end;function Win32AnsiUpperCase(const s: string): string;  begin    if length(s)>0 then      begin        result:=s;        UniqueString(result);        CharUpperBuffA(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);        CharLowerBuffA(pchar(result),length(result));      end    else      result:='';  end;function Win32AnsiCompareStr(const S1, S2: string): PtrInt;  begin    result:=CompareStringA(LOCALE_USER_DEFAULT,0,pchar(s1),length(s1),      pchar(s2),length(s2))-2;  end;function Win32AnsiCompareText(const S1, S2: string): PtrInt;  begin    result:=CompareStringA(LOCALE_USER_DEFAULT,NORM_IGNORECASE,pchar(s1),length(s1),      pchar(s2),length(s2))-2;  end;function Win32AnsiStrComp(S1, S2: PChar): PtrInt;  begin    result:=CompareStringA(LOCALE_USER_DEFAULT,0,s1,-1,s2,-1)-2;  end;function Win32AnsiStrIComp(S1, S2: PChar): PtrInt;  begin    result:=CompareStringA(LOCALE_USER_DEFAULT,NORM_IGNORECASE,s1,-1,s2,-1)-2;  end;function Win32AnsiStrLComp(S1, S2: PChar; MaxLen: PtrUInt): PtrInt;  begin    result:=CompareStringA(LOCALE_USER_DEFAULT,0,s1,maxlen,s2,maxlen)-2;  end;function Win32AnsiStrLIComp(S1, S2: PChar; MaxLen: PtrUInt): PtrInt;  begin    result:=CompareStringA(LOCALE_USER_DEFAULT,NORM_IGNORECASE,s1,maxlen,s2,maxlen)-2;  end;function Win32AnsiStrLower(Str: PChar): PChar;  begin    CharLowerA(str);    result:=str;  end;function Win32AnsiStrUpper(Str: PChar): PChar;  begin    CharUpperA(str);    result:=str;  end;function Win32CompareUnicodeString(const s1, s2 : UnicodeString; Options : TCompareOptions) : PtrInt;Var  O : LongWord;  CO : TCompareOption;begin  O:=0;  for CO in TCompareOption do    if CO in Options then      O:=O or WinAPICompareFlags[CO];    Result:=DoCompareStringW(PWideChar(s1), PWideChar(s2), Length(s1), Length(s2), O);end;function Win32CompareTextUnicodeString(const s1, s2 : UnicodeString) : PtrInt;  begin    Result:=DoCompareStringW(PWideChar(s1), PWideChar(s2), Length(s1), Length(s2), NORM_IGNORECASE);  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    { return value: number of code points in the string. Whenever an invalid      code point is encountered, all characters part of this invalid code point      are considered to form one "character" and the next character is      considered to be the start of a new (possibly also invalid) code point }//!!!    CharLengthPCharProc : function(const Str: PChar): PtrInt;    { return value:      -1 if incomplete or invalid code point      0 if NULL character,      > 0 if that's the length in bytes of the code point }//!!!!    CodePointLengthProc : function(const Str: PChar; MaxLookAead: PtrInt): Ptrint;    widestringmanager.CompareWideStringProc:=@Win32CompareWideString;    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;    widestringmanager.CompareUnicodeStringProc:=@Win32CompareUnicodeString;  end;{ Platform-specific exception support }function WinExceptionObject(code: Longint; const rec: TExceptionRecord): Exception;var  entry: PExceptMapEntry;begin  entry := FindExceptMapEntry(code);  if assigned(entry) then    result:=entry^.cls.CreateRes(entry^.msg)  else    result:=EExternalException.CreateResFmt(@SExternalException,[rec.ExceptionCode]);  if result is EExternal then    EExternal(result).FExceptionRecord:=rec;end;function WinExceptionClass(code: longint): ExceptClass;var  entry: PExceptMapEntry;begin  entry := FindExceptMapEntry(code);  if assigned(entry) then    result:=entry^.cls  else    result:=EExternalException;end;Initialization  InitWin32Widestrings;  InitExceptions;       { Initialize exceptions. OS independent }{$ifdef mswindows}      { Keeps exe size down for systems that do not use SEH }  ExceptObjProc:=@WinExceptionObject;  ExceptClsProc:=@WinExceptionClass;{$endif mswindows}  InitLeadBytes;  InitInternational;    { Initialize internationalization settings }  LoadVersionInfo;  InitSysConfigDir;  OnBeep:=@SysBeep;Finalization  FreeTerminateProcs;  DoneExceptions;end.
 |