| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296 | {    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+}uses  windows;{$DEFINE HAS_SLEEP}{$DEFINE HAS_OSERROR}{$DEFINE HAS_OSCONFIG}{$DEFINE HAS_OSUSERDIR}{$DEFINE HAS_CREATEGUID}{$DEFINE HAS_LOCALTIMEZONEOFFSET}{ 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;procedure GetFormatSettings;procedure GetLocaleFormatSettings(LCID: Integer; var FormatSettings: TFormatSettings); platform;implementation  uses    sysconst,    windirs;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:=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 *)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: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****************************************************************************}const  AccessMode: array[0..2] of Cardinal  = (    GENERIC_READ,    GENERIC_WRITE,    GENERIC_READ or GENERIC_WRITE);  ShareModes: array[0..4] of Integer = (               0,               0,               FILE_SHARE_READ,               FILE_SHARE_WRITE,               FILE_SHARE_READ or FILE_SHARE_WRITE);Function FileOpen (Const FileName : string; Mode : Integer) : THandle;begin  result := CreateFile(PChar(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 : String) : THandle;begin  FileCreate:=FileCreate(FileName, fmShareExclusive, 0);end;Function FileCreate (Const FileName : String; Rights:longint) : THandle;begin  FileCreate:=FileCreate(FileName, fmShareExclusive, Rights);end;Function FileCreate (Const FileName : String; ShareMode : Integer; Rights : Integer) : THandle;begin  Result := CreateFile(PChar(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  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+(qword(maxdword)+1)*F.FindData.NFileSizeHigh;  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:=Longint(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****************************************************************************}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;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);begin  windows.Getlocaltime(SystemTime);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;                                                                      {****************************************************************************                              Misc Functions****************************************************************************}procedure sysbeep;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..3] of Char; // sdate allows 4 chars.begin  if GetLocaleInfo(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;    buf: array[0..100] of char;    ALCID : LCID;    PriLangID : Word;    SubLangID : Word;begin  Result := ''; if (Count<=0) then exit;  DateTimeToSystemTime(EncodeDate(Year,Month,Day),ASystemTime);  ALCID := GetThreadLocale;//  ALCID := SysLocale.DefaultLCID;  if GetDateFormat(ALCID , DATE_USE_ALT_CALENDAR      , @ASystemTime, PChar('gg')      , @buf, SizeOf(buf)) > 0 then  begin    Result := buf;    if Count = 1 then    begin      PriLangID := ALCID and $3FF;      SubLangID := (ALCID and $FFFF) shr 10;      case PriLangID of        LANG_JAPANESE:          begin            Result := Copy(WideString(Result),1,1);          end;        LANG_CHINESE:          if (SubLangID = SUBLANG_CHINESE_TRADITIONAL) then          begin            Result := Copy(WideString(Result),1,1);          end;      end;    end;  end;// if Result = '' then Result := StringOfChar('G',Count);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 GetDateFormat(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 GetLocaleInfo(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); 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 := 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);      ListSeparator := GetLocaleChar(LID, LOCALE_SLIST, ',');    end;end;procedure GetFormatSettings;begin  GetlocaleFormatSettings(GetThreadLocale, DefaultFormatSettings);end;Procedure InitInternational;var  { A call to GetSystemMetrics changes the value of the 8087 Control Word on    Pentium4 with WinXP SP2 }  old8087CW: word;  DefaultCustomLocaleID : LCID;   // typedef DWORD LCID;  DefaultCustomLanguageID : Word; // typedef WORD LANGID;begin  /// workaround for Windows 7 bug, see bug report #18574  SetThreadLocale(GetUserDefaultLCID);  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  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;  Set8087CW(old8087CW);  GetFormatSettings;  if SysLocale.FarEast then GetEraNamesAndYearOffsets;end;{****************************************************************************                           Target Dependent****************************************************************************}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;Flags:TExecuteFlags=[]):integer;// win specific  functionvar  SI: TStartupInfo;  PI: TProcessInformation;  Proc : THandle;  l    : DWord;  CommandLine : ansistring;  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 CreateProcess(nil, pchar(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: AnsiString; Const ComLine: Array of AnsiString;Flags:TExecuteFlags=[]):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,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'));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, GetWindowsDirectory(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,L1);    if L2>0 then      widestringmanager.Wide2AnsiMoveProc(P2,a2,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;function Win32CompareWideString(const s1, s2 : WideString) : PtrInt;  begin    Result:=DoCompareStringW(PWideChar(s1), PWideChar(s2), Length(s1), Length(s2), 0);  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);        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;function Win32CompareUnicodeString(const s1, s2 : UnicodeString) : PtrInt;  begin    Result:=DoCompareStringW(PWideChar(s1), PWideChar(s2), Length(s1), Length(s2), 0);  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.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;    widestringmanager.CompareUnicodeStringProc:=@Win32CompareUnicodeString;    widestringmanager.CompareTextUnicodeStringProc:=@Win32CompareTextUnicodeString;  end;Initialization  InitWin32Widestrings;  InitExceptions;       { Initialize exceptions. OS independent }  InitInternational;    { Initialize internationalization settings }  LoadVersionInfo;  InitSysConfigDir;  OnBeep:=@SysBeep;Finalization  DoneExceptions;end.
 |