Leaked source code of windows server 2003
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

780 lines
18 KiB

  1. /*++
  2. Copyright (c) 1997 Microsoft Corporation
  3. Module Name:
  4. timezone.c
  5. Abstract:
  6. This module is responsible for managing the mapping of timezones from windows 9x to
  7. windows Nt. Because of the fact that the timezone strings are different between the
  8. several different platforms (Win9x Win98 WinNt) and because it is important to end
  9. users that there timezone setting accurately reflect there geographic location, a
  10. somewhat complex method of mapping timezones is needed.
  11. Author:
  12. Marc R. Whitten (marcw) 09-Jul-1998
  13. Revision History:
  14. marcw 18-Aug-1998 Added timezone enum, support for retaining
  15. fixed matches.
  16. --*/
  17. #include "pch.h"
  18. #include "sysmigp.h"
  19. #define DBG_TIMEZONE "TimeZone"
  20. #define S_FIRSTBOOT TEXT("!!!First Boot!!!")
  21. TCHAR g_TimeZoneMap[20] = TEXT("");
  22. TCHAR g_CurrentTimeZone[MAX_TIMEZONE] = TEXT("");
  23. BOOL g_TimeZoneMapped = FALSE;
  24. //
  25. // Variable used by tztest tool.
  26. //
  27. HANDLE g_TzTestHiveSftInf = NULL;
  28. BOOL
  29. pBuildNtTimeZoneData (
  30. VOID
  31. )
  32. /*++
  33. Routine Description:
  34. pBuildNtTimeZone data reads the timezone information that is stored in
  35. hivesft.inf and organizes it into memdb. This data is used to look up
  36. display names of timezone indices.
  37. Arguments:
  38. None.
  39. Return Value:
  40. TRUE if the function completes successfully, FALSE
  41. otherwise.
  42. --*/
  43. {
  44. HINF inf = INVALID_HANDLE_VALUE;
  45. INFSTRUCT is = INITINFSTRUCT_POOLHANDLE;
  46. BOOL rSuccess = FALSE;
  47. PTSTR key = NULL;
  48. PTSTR value = NULL;
  49. PTSTR desc = NULL;
  50. PTSTR index = NULL;
  51. BOOL timeZonesFound = FALSE;
  52. TCHAR paddedIndex[20];
  53. PTSTR p = NULL;
  54. UINT i = 0;
  55. UINT count = 0;
  56. if (!g_TzTestHiveSftInf) {
  57. //
  58. // First, Read data from hivesft.inf.
  59. //
  60. inf = InfOpenInfInAllSources (S_HIVESFT_INF);
  61. }
  62. else {
  63. inf = g_TzTestHiveSftInf;
  64. }
  65. if (inf == INVALID_HANDLE_VALUE) {
  66. LOG ((LOG_ERROR, "Cannot load hivesoft.inf. Unable to build timezone information." ));
  67. return FALSE;
  68. }
  69. if (InfFindFirstLine (inf, S_ADDREG, NULL, &is)) {
  70. do {
  71. //
  72. // Cycle through all of the lines looking for timezone information.
  73. //
  74. key = InfGetStringField (&is, 2);
  75. if (key && IsPatternMatch (TEXT("*Time Zones*"), key)) {
  76. //
  77. // Remember that we have found the first timezone entry.
  78. //
  79. timeZonesFound = TRUE;
  80. //
  81. // Now, get value. We care about "display" and "index"
  82. //
  83. value = InfGetStringField (&is, 3);
  84. if (!value) {
  85. continue;
  86. }
  87. if (StringIMatch (value, S_DISPLAY)) {
  88. //
  89. // display string found.
  90. //
  91. desc = InfGetStringField (&is, 5);
  92. } else if (StringIMatch (value, S_INDEX)) {
  93. //
  94. // index value found.
  95. //
  96. index = InfGetStringField (&is, 5);
  97. }
  98. if (index && desc) {
  99. //
  100. // Make sure the index is 0 padded.
  101. //
  102. count = 3 - TcharCount (index);
  103. p = paddedIndex;
  104. for (i=0; i<count; i++) {
  105. *p = TEXT('0');
  106. p = _tcsinc (p);
  107. }
  108. StringCopy (p, index);
  109. //
  110. // we have all the information we need. Save this entry into memdb.
  111. //
  112. MemDbSetValueEx (MEMDB_CATEGORY_NT_TIMEZONES, paddedIndex, desc, NULL, 0, NULL);
  113. index = NULL;
  114. desc = NULL;
  115. }
  116. } else {
  117. //
  118. // Keep memory usage low.
  119. //
  120. InfResetInfStruct (&is);
  121. if (key) {
  122. if (timeZonesFound) {
  123. //
  124. // We have gathered all of the timezone information from hivesft.inf
  125. // we can abort our loop at this point.
  126. //
  127. break;
  128. }
  129. }
  130. }
  131. } while (InfFindNextLine(&is));
  132. } ELSE_DEBUGMSG ((DBG_ERROR, "[%s] not found in hivesft.inf!",S_ADDREG));
  133. //
  134. // Clean up resources
  135. //
  136. InfCleanUpInfStruct (&is);
  137. InfCloseInfFile (inf);
  138. return TRUE;
  139. }
  140. BOOL
  141. pBuild9xTimeZoneData (
  142. VOID
  143. )
  144. /*++
  145. Routine Description:
  146. pBuild9xTimeZone Data is responsible for reading the time zone information
  147. that is stored in win95upg.inf and organizing it into memdb. The timezone
  148. enumeration routines then use this data in order to find all Nt timezones
  149. that can map to a particular 9x timezone.
  150. Arguments:
  151. None.
  152. Return Value:
  153. TRUE if the data is successfully stored in memdb, FALSE
  154. otherwise.
  155. --*/
  156. {
  157. INFSTRUCT is = INITINFSTRUCT_POOLHANDLE;
  158. PTSTR desc = NULL;
  159. PTSTR index = NULL;
  160. UINT count = 0;
  161. PTSTR p = NULL;
  162. //
  163. // Now, read in information about win9x registry mappings.
  164. //
  165. if (InfFindFirstLine (g_Win95UpgInf, S_TIMEZONEMAPPINGS, NULL, &is)) {
  166. do {
  167. //
  168. // Get the display name and matching index(es) for this timezone.
  169. //
  170. desc = InfGetStringField (&is,0);
  171. index = InfGetStringField (&is,1);
  172. //
  173. // Enumerate the indices and save them into memdb.
  174. //
  175. count = 0;
  176. while (index) {
  177. p = _tcschr (index, TEXT(','));
  178. if (p) {
  179. *p = 0;
  180. }
  181. MemDbSetValueEx (
  182. MEMDB_CATEGORY_9X_TIMEZONES,
  183. desc,
  184. MEMDB_FIELD_INDEX,
  185. index,
  186. 0,
  187. NULL
  188. );
  189. count++;
  190. if (p) {
  191. index = _tcsinc(p);
  192. }
  193. else {
  194. //
  195. // Save away the count of possible nt timezones for this 9x timezone.
  196. //
  197. MemDbSetValueEx (
  198. MEMDB_CATEGORY_9X_TIMEZONES,
  199. desc,
  200. MEMDB_FIELD_COUNT,
  201. NULL,
  202. count,
  203. NULL
  204. );
  205. index = NULL;
  206. }
  207. }
  208. } while (InfFindNextLine (&is));
  209. }
  210. //
  211. // Clean up resources.
  212. //
  213. InfCleanUpInfStruct (&is);
  214. return TRUE;
  215. }
  216. BOOL
  217. pGetCurrentTimeZone (
  218. VOID
  219. )
  220. /*++
  221. Routine Description:
  222. pGetCurrentTimeZone retrieves the user's timezone from the windows 9x
  223. registry. The enumeration routines use this timezone in order to enumerate
  224. the possible matching timezones in the INF.
  225. Arguments:
  226. None.
  227. Return Value:
  228. TRUE if the function successfully retrieves the user's password, FALSE
  229. otherwise.
  230. --*/
  231. {
  232. BOOL rSuccess = TRUE;
  233. PCTSTR displayName = NULL;
  234. REGTREE_ENUM eTree;
  235. PCTSTR valueName = NULL;
  236. PCTSTR value = NULL;
  237. PCTSTR curTimeZone = NULL;
  238. HKEY hKey = NULL;
  239. //
  240. // Get the current timezone name, and set the valuename to the correct string.
  241. //
  242. hKey = OpenRegKeyStr (S_TIMEZONEINFORMATION);
  243. if (!hKey) {
  244. LOG ((LOG_ERROR, "Unable to open %s key.", S_TIMEZONEINFORMATION));
  245. return FALSE;
  246. }
  247. if ((curTimeZone = GetRegValueString (hKey, S_STANDARDNAME)) && !StringIMatch (curTimeZone, S_FIRSTBOOT)) {
  248. //
  249. // standard time. We need to look under the "STD" value to match this.
  250. //
  251. valueName = S_STD;
  252. } else if ((curTimeZone = GetRegValueString (hKey, S_DAYLIGHTNAME)) && !StringIMatch (curTimeZone, S_FIRSTBOOT)) {
  253. //
  254. // Daylight Savings Time. We need to look under the "DLT" value to match this.
  255. //
  256. valueName = S_DLT;
  257. } else {
  258. CloseRegKey (hKey);
  259. hKey = OpenRegKeyStr (TEXT("HKLM\\Software\\Microsoft\\Windows\\CurrentVersion\\Time Zones"));
  260. if (hKey) {
  261. if (curTimeZone = GetRegValueString (hKey, TEXT(""))) {
  262. valueName = S_STD;
  263. }
  264. }
  265. if (!valueName) {
  266. //
  267. // No timezone found!
  268. //
  269. DEBUGMSG((DBG_WHOOPS,"Unable to get Timezone name..User will have to enter timezone in GUI mode."));
  270. return FALSE;
  271. }
  272. }
  273. __try {
  274. //
  275. // Now we have to search through the timezones key and find the key that has a value equal to
  276. // the current timezone name. A big pain.
  277. //
  278. if (EnumFirstRegKeyInTree (&eTree, S_TIMEZONES)) {
  279. do {
  280. //
  281. // For each subkey, we must look for the string in valueName and
  282. // see if it matches.
  283. //
  284. value = GetRegValueString (eTree.CurrentKey->KeyHandle, valueName);
  285. if (value) {
  286. if (StringIMatch (value, curTimeZone)) {
  287. //
  288. // We found the key we were looking for and we can finally
  289. // gather the data we need.
  290. //
  291. displayName = GetRegValueString (eTree.CurrentKey->KeyHandle, S_DISPLAY);
  292. if (!displayName) {
  293. DEBUGMSG((DBG_WHOOPS,"Error! Timezone key found, but no Display value!"));
  294. AbortRegKeyTreeEnum (&eTree);
  295. rSuccess = FALSE;
  296. __leave;
  297. }
  298. //
  299. // Save away the current Timezone and leave the loop. We are done.
  300. //
  301. StringCopy (g_CurrentTimeZone, displayName);
  302. AbortRegKeyTreeEnum (&eTree);
  303. break;
  304. }
  305. MemFree (g_hHeap, 0, value);
  306. value = NULL;
  307. }
  308. } while (EnumNextRegKeyInTree (&eTree));
  309. }
  310. } __finally {
  311. if (curTimeZone) {
  312. MemFree (g_hHeap, 0, curTimeZone);
  313. }
  314. if (value) {
  315. MemFree (g_hHeap, 0, value);
  316. }
  317. if (displayName) {
  318. MemFree (g_hHeap, 0, displayName);
  319. }
  320. CloseRegKey (hKey);
  321. }
  322. return rSuccess;
  323. }
  324. BOOL
  325. pInitTimeZoneData (
  326. VOID
  327. )
  328. /*++
  329. Routine Description:
  330. pInitTimeZoneData is responsible for performing all of the initialization
  331. necessary to use the time zone enumeration routines.
  332. Arguments:
  333. None.
  334. Return Value:
  335. TRUE if initialization completes successfully, FALSE otherwise.
  336. --*/
  337. {
  338. BOOL rSuccess = TRUE;
  339. //
  340. // First, fill memdb with timezone
  341. // information regarding winnt and win9x
  342. // (from hivesft.inf and win95upg.inf)
  343. //
  344. if (!pBuildNtTimeZoneData ()) {
  345. LOG ((LOG_ERROR, "Unable to gather nt timezone information."));
  346. rSuccess = FALSE;
  347. }
  348. if (!pBuild9xTimeZoneData ()) {
  349. LOG ((LOG_ERROR, "Unable to gather 9x timezone information."));
  350. rSuccess = FALSE;
  351. }
  352. //
  353. // Next, get the user's timezone.
  354. //
  355. if (!pGetCurrentTimeZone ()) {
  356. LOG ((LOG_ERROR, "Failure trying to retrieve timezone information."));
  357. rSuccess = FALSE;
  358. }
  359. return rSuccess;
  360. }
  361. BOOL
  362. pEnumFirstNtTimeZone (
  363. OUT PTIMEZONE_ENUM EnumPtr
  364. )
  365. {
  366. BOOL rSuccess = FALSE;
  367. PTSTR p;
  368. EnumPtr -> MapCount = 0;
  369. if (MemDbEnumFirstValue (&(EnumPtr -> Enum), MEMDB_CATEGORY_NT_TIMEZONES"\\*", MEMDB_ALL_SUBLEVELS, MEMDB_ENDPOINTS_ONLY)) {
  370. do {
  371. EnumPtr -> MapCount++;
  372. } while (MemDbEnumNextValue (&(EnumPtr -> Enum)));
  373. }
  374. else {
  375. return FALSE;
  376. }
  377. MemDbEnumFirstValue (&(EnumPtr -> Enum), MEMDB_CATEGORY_NT_TIMEZONES"\\*", MEMDB_ALL_SUBLEVELS, MEMDB_ENDPOINTS_ONLY);
  378. p = _tcschr (EnumPtr->Enum.szName,TEXT('\\'));
  379. if (!p) {
  380. return FALSE;
  381. }
  382. *p = 0;
  383. EnumPtr -> MapIndex = EnumPtr -> Enum.szName;
  384. StringCopy (EnumPtr -> NtTimeZone, _tcsinc(p));
  385. return TRUE;
  386. }
  387. BOOL
  388. pEnumNextNtTimeZone (
  389. OUT PTIMEZONE_ENUM EnumPtr
  390. )
  391. {
  392. PTSTR p;
  393. if (!MemDbEnumNextValue(&EnumPtr -> Enum)) {
  394. return FALSE;
  395. }
  396. p = _tcschr (EnumPtr->Enum.szName,TEXT('\\'));
  397. if (!p) {
  398. return FALSE;
  399. }
  400. *p = 0;
  401. EnumPtr -> MapIndex = EnumPtr -> Enum.szName;
  402. StringCopy (EnumPtr -> NtTimeZone, _tcsinc(p));
  403. return TRUE;
  404. }
  405. BOOL
  406. EnumFirstTimeZone (
  407. OUT PTIMEZONE_ENUM EnumPtr,
  408. IN DWORD Flags
  409. )
  410. /*++
  411. Routine Description:
  412. EnumFirstTimeZone/EnumNextTimeZone enumerate the Nt timezones that can match the
  413. user's current Windows 9x time zone. In most cases, there will be only one,
  414. but, in some cases, there can be several.
  415. Arguments:
  416. EnumPtr - A pointer to a valid timezone enumeration structure. This
  417. variable holds the necessary state between timezone enumeration
  418. calls.
  419. Return Value:
  420. TRUE if there are any timezones to enumerate, FALSE otherwise.
  421. --*/
  422. {
  423. BOOL rSuccess = FALSE;
  424. TCHAR key[MEMDB_MAX];
  425. static BOOL firstTime = TRUE;
  426. if (firstTime) {
  427. if (!pInitTimeZoneData ()) {
  428. LOG ((LOG_ERROR, "Error initializing timezone data."));
  429. return FALSE;
  430. }
  431. firstTime = FALSE;
  432. }
  433. MYASSERT (EnumPtr);
  434. EnumPtr -> CurTimeZone = g_CurrentTimeZone;
  435. EnumPtr -> Flags = Flags;
  436. if (Flags & TZFLAG_ENUM_ALL) {
  437. return pEnumFirstNtTimeZone (EnumPtr);
  438. }
  439. if ((Flags & TZFLAG_USE_FORCED_MAPPINGS) && g_TimeZoneMapped) {
  440. //
  441. // We have a force mapping, so mapcount is 1.
  442. //
  443. EnumPtr -> MapCount = 1;
  444. }
  445. else {
  446. //
  447. // Get count of matches.
  448. //
  449. MemDbBuildKey (key, MEMDB_CATEGORY_9X_TIMEZONES, EnumPtr -> CurTimeZone, MEMDB_FIELD_COUNT, NULL);
  450. if (!MemDbGetValue (key, &(EnumPtr -> MapCount))) {
  451. DEBUGMSG ((
  452. DBG_WARNING,
  453. "EnumFirstTimeZone: Could not retrieve count of nt timezone matches from memdb for %s.",
  454. EnumPtr -> CurTimeZone
  455. ));
  456. return FALSE;
  457. }
  458. }
  459. DEBUGMSG ((DBG_TIMEZONE, "%d Nt time zones match the win9x timezone %s.", EnumPtr -> MapCount, EnumPtr -> CurTimeZone));
  460. if ((Flags & TZFLAG_USE_FORCED_MAPPINGS) && g_TimeZoneMapped) {
  461. //
  462. // Use the previously forced mapping.
  463. //
  464. EnumPtr -> MapIndex = g_TimeZoneMap;
  465. } else {
  466. //
  467. // Now, enumerate the matching map indexes in memdb.
  468. //
  469. rSuccess = MemDbGetValueEx (
  470. &(EnumPtr -> Enum),
  471. MEMDB_CATEGORY_9X_TIMEZONES,
  472. EnumPtr -> CurTimeZone,
  473. MEMDB_FIELD_INDEX
  474. );
  475. if (!rSuccess) {
  476. return FALSE;
  477. }
  478. EnumPtr -> MapIndex = EnumPtr -> Enum.szName;
  479. }
  480. //
  481. // Get the NT display string for this map index.
  482. //
  483. rSuccess = MemDbGetEndpointValueEx (MEMDB_CATEGORY_NT_TIMEZONES, EnumPtr->MapIndex, NULL, EnumPtr->NtTimeZone);
  484. return rSuccess;
  485. }
  486. BOOL
  487. EnumNextTimeZone (
  488. IN OUT PTIMEZONE_ENUM EnumPtr
  489. )
  490. {
  491. if (EnumPtr -> Flags & TZFLAG_ENUM_ALL) {
  492. return pEnumNextNtTimeZone (EnumPtr);
  493. }
  494. if ((EnumPtr -> Flags & TZFLAG_USE_FORCED_MAPPINGS) && g_TimeZoneMapped) {
  495. return FALSE;
  496. }
  497. if (!MemDbEnumNextValue (&(EnumPtr->Enum))) {
  498. return FALSE;
  499. }
  500. EnumPtr->MapIndex = EnumPtr->Enum.szName;
  501. return MemDbGetEndpointValueEx (MEMDB_CATEGORY_NT_TIMEZONES, EnumPtr->MapIndex, NULL, EnumPtr->NtTimeZone);
  502. }
  503. BOOL
  504. ForceTimeZoneMap (
  505. IN PCTSTR NtTimeZone
  506. )
  507. /*++
  508. Routine Description:
  509. ForceTimeZoneMap forces the mapping of a particular 9x timezone to a
  510. particular Nt timezone. This function is used in cases where there are
  511. multiple nt timezones that could map to a particular 9x timezone.
  512. Arguments:
  513. NtTimeZone - String containing the timezone to force mapping to.
  514. Return Value:
  515. TRUE if the function successfully updated the timezone mapping, FALSE
  516. otherwise.
  517. --*/
  518. {
  519. TIMEZONE_ENUM e;
  520. //
  521. // Find index that matches this timezone.
  522. //
  523. if (EnumFirstTimeZone (&e, TZFLAG_ENUM_ALL)) {
  524. do {
  525. if (StringIMatch (NtTimeZone, e.NtTimeZone)) {
  526. //
  527. // this is the index we need.
  528. //
  529. StringCopy (g_TimeZoneMap, e.MapIndex);
  530. g_TimeZoneMapped = TRUE;
  531. break;
  532. }
  533. } while (EnumNextTimeZone (&e));
  534. }
  535. return TRUE;
  536. }