Source code of Windows XP (NT5)
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.

3330 lines
103 KiB

  1. /*++
  2. Copyright (c) 1991-2000, Microsoft Corporation All rights reserved.
  3. Module Name:
  4. tables.c
  5. Abstract:
  6. This file contains functions that manipulate or return information
  7. about the different tables used by the NLS API.
  8. External Routines found in this file:
  9. AllocTables
  10. GetUnicodeFileInfo
  11. GetGeoFileInfo
  12. GetCTypeFileInfo
  13. GetDefaultSortkeyFileInfo
  14. GetDefaultSortTablesFileInfo
  15. GetSortkeyFileInfo
  16. GetSortTablesFileInfo
  17. GetCodePageFileInfo
  18. GetLanguageFileInfo
  19. GetLocaleFileInfo
  20. MakeCPHashNode
  21. MakeLangHashNode
  22. MakeLocHashNode
  23. GetCPHashNode
  24. GetLangHashNode
  25. GetLocHashNode
  26. GetCalendar
  27. Revision History:
  28. 05-31-91 JulieB Created.
  29. --*/
  30. //
  31. // Include Files.
  32. //
  33. #include "nls.h"
  34. //
  35. // Constant Declarations.
  36. //
  37. #define SEM_NOERROR (SEM_FAILCRITICALERRORS | \
  38. SEM_NOGPFAULTERRORBOX | \
  39. SEM_NOOPENFILEERRORBOX)
  40. //
  41. // Global Variables.
  42. //
  43. PTBL_PTRS pTblPtrs; // ptr to structure of table ptrs
  44. //
  45. // Forward Declarations.
  46. //
  47. BOOL
  48. IsValidSortId(
  49. LCID Locale);
  50. ULONG
  51. GetLanguageExceptionInfo(void);
  52. LPWORD
  53. GetLinguisticLanguageInfo(
  54. LCID Locale);
  55. ULONG
  56. CreateAndCopyLanguageExceptions(
  57. LCID Locale,
  58. LPWORD *ppBaseAddr);
  59. BOOL FASTCALL
  60. FindLanguageExceptionPointers(
  61. LCID Locale,
  62. PL_EXCEPT_HDR *ppExceptHdr,
  63. PL_EXCEPT *ppExceptTbl);
  64. void FASTCALL
  65. CopyLanguageExceptionInfo(
  66. LPWORD pBaseAddr,
  67. PL_EXCEPT_HDR pExceptHdr,
  68. PL_EXCEPT pExceptTbl);
  69. BOOL FASTCALL
  70. FindExceptionPointers(
  71. LCID Locale,
  72. PEXCEPT_HDR *ppExceptHdr,
  73. PEXCEPT *ppExceptTbl,
  74. PVOID *ppIdeograph,
  75. PULONG pReturn);
  76. void FASTCALL
  77. CopyExceptionInfo(
  78. PSORTKEY pSortkey,
  79. PEXCEPT_HDR pExceptHdr,
  80. PEXCEPT pExceptTbl,
  81. PVOID pIdeograph);
  82. ULONG
  83. WaitOnEvent(
  84. LPWORD pSem);
  85. //-------------------------------------------------------------------------//
  86. // INTERNAL MACROS //
  87. //-------------------------------------------------------------------------//
  88. ////////////////////////////////////////////////////////////////////////////
  89. //
  90. // GET_HASH_VALUE
  91. //
  92. // Returns the hash value for given value and the given table size.
  93. //
  94. // DEFINED AS A MACRO.
  95. //
  96. // 05-31-91 JulieB Created.
  97. ////////////////////////////////////////////////////////////////////////////
  98. #define GET_HASH_VALUE(Value, TblSize) (Value % TblSize)
  99. ////////////////////////////////////////////////////////////////////////////
  100. //
  101. // CREATE_CODEPAGE_HASH_NODE
  102. //
  103. // Creates a code page hash node and stores the pointer to it in pHashN.
  104. //
  105. // NOTE: This macro may return if an error is encountered.
  106. //
  107. // DEFINED AS A MACRO.
  108. //
  109. // 05-31-91 JulieB Created.
  110. ////////////////////////////////////////////////////////////////////////////
  111. #define CREATE_CODEPAGE_HASH_NODE( CodePage, \
  112. pHashN ) \
  113. { \
  114. /* \
  115. * Allocate CP_HASH structure. \
  116. */ \
  117. if ((pHashN = (PCP_HASH)NLS_ALLOC_MEM(sizeof(CP_HASH))) == NULL) \
  118. { \
  119. return (ERROR_OUTOFMEMORY); \
  120. } \
  121. \
  122. /* \
  123. * Fill in the CodePage value. \
  124. */ \
  125. pHashN->CodePage = CodePage; \
  126. \
  127. /* \
  128. * Make sure the pfnCPProc value is NULL for now. \
  129. */ \
  130. pHashN->pfnCPProc = NULL; \
  131. }
  132. ////////////////////////////////////////////////////////////////////////////
  133. //
  134. // CREATE_LOCALE_HASH_NODE
  135. //
  136. // Creates a locale hash node and stores the pointer to it in pHashN.
  137. //
  138. // NOTE: This macro may return if an error is encountered.
  139. //
  140. // DEFINED AS A MACRO.
  141. //
  142. // 05-31-91 JulieB Created.
  143. ////////////////////////////////////////////////////////////////////////////
  144. #define CREATE_LOCALE_HASH_NODE( Locale, \
  145. pHashN ) \
  146. { \
  147. /* \
  148. * Allocate LOC_HASH structure. \
  149. */ \
  150. if ((pHashN = (PLOC_HASH)NLS_ALLOC_MEM(sizeof(LOC_HASH))) == NULL) \
  151. { \
  152. return (ERROR_OUTOFMEMORY); \
  153. } \
  154. \
  155. /* \
  156. * Fill in the Locale value. \
  157. */ \
  158. pHashN->Locale = Locale; \
  159. }
  160. ////////////////////////////////////////////////////////////////////////////
  161. //
  162. // FIND_CP_HASH_NODE
  163. //
  164. // Searches for the cp hash node for the given locale. The result is
  165. // put in pHashN. If no node exists, pHashN will be NULL.
  166. //
  167. // DEFINED AS A MACRO.
  168. //
  169. // 05-31-91 JulieB Created.
  170. ////////////////////////////////////////////////////////////////////////////
  171. #define FIND_CP_HASH_NODE( CodePage, \
  172. pHashN ) \
  173. { \
  174. UINT Index; /* hash value */ \
  175. \
  176. /* \
  177. * Get hash value. \
  178. */ \
  179. Index = GET_HASH_VALUE(CodePage, CP_TBL_SIZE); \
  180. \
  181. /* \
  182. * Make sure the hash node still doesn't exist in the table. \
  183. */ \
  184. pHashN = (pTblPtrs->pCPHashTbl)[Index]; \
  185. while ((pHashN != NULL) && (pHashN->CodePage != CodePage)) \
  186. { \
  187. pHashN = pHashN->pNext; \
  188. } \
  189. }
  190. ////////////////////////////////////////////////////////////////////////////
  191. //
  192. // IsCPHashNodeLoaded
  193. //
  194. // Wrapper for the FIND_CP_HASH_NODE macro so that we can call this from
  195. // mbcs.c. Return TRUE if the node already exists, otherwise false. False
  196. // could still indicate a valid code page, just not one already loaded.
  197. //
  198. // 05-31-02 ShawnSte Created.
  199. ////////////////////////////////////////////////////////////////////////////
  200. BOOL IsCPHashNodeLoaded( UINT CodePage )
  201. {
  202. PCP_HASH pHashN; // ptr to CP hash node
  203. //
  204. // Get hash node.
  205. //
  206. FIND_CP_HASH_NODE(CodePage, pHashN);
  207. return (pHashN != NULL);
  208. }
  209. ////////////////////////////////////////////////////////////////////////////
  210. //
  211. // FIND_LOCALE_HASH_NODE
  212. //
  213. // Searches for the locale hash node for the given locale. The result is
  214. // put in pHashN. If no node exists, pHashN will be NULL.
  215. //
  216. // DEFINED AS A MACRO.
  217. //
  218. // 05-31-91 JulieB Created.
  219. ////////////////////////////////////////////////////////////////////////////
  220. #define FIND_LOCALE_HASH_NODE( Locale, \
  221. pHashN ) \
  222. { \
  223. UINT Index; /* hash value */ \
  224. \
  225. \
  226. /* \
  227. * Get hash value. \
  228. */ \
  229. Index = GET_HASH_VALUE(Locale, LOC_TBL_SIZE); \
  230. \
  231. /* \
  232. * Get hash node. \
  233. */ \
  234. pHashN = (pTblPtrs->pLocHashTbl)[Index]; \
  235. while ((pHashN != NULL) && (pHashN->Locale != Locale)) \
  236. { \
  237. pHashN = pHashN->pNext; \
  238. } \
  239. }
  240. ////////////////////////////////////////////////////////////////////////////
  241. //
  242. // EXIST_LANGUAGE_INFO
  243. //
  244. // Checks to see if the casing tables have been added to the locale
  245. // hash node.
  246. //
  247. // Must check the LOWER CASE pointer, since that value is set last in
  248. // the hash node.
  249. //
  250. // DEFINED AS A MACRO.
  251. //
  252. // 05-31-91 JulieB Created.
  253. ////////////////////////////////////////////////////////////////////////////
  254. #define EXIST_LANGUAGE_INFO(pHashN) (pHashN->pLowerCase)
  255. ////////////////////////////////////////////////////////////////////////////
  256. //
  257. // EXIST_LINGUIST_LANGUAGE_INFO
  258. //
  259. // Checks to see if the linguistic casing tables have been added to the locale
  260. // hash node.
  261. //
  262. // Must check the LOWER CASE pointer, since that value is set last in
  263. // the hash node.
  264. //
  265. // DEFINED AS A MACRO.
  266. //
  267. // 08-30-95 JulieB Created.
  268. ////////////////////////////////////////////////////////////////////////////
  269. #define EXIST_LINGUIST_LANGUAGE_INFO(pHashN) (pHashN->pLowerLinguist)
  270. ////////////////////////////////////////////////////////////////////////////
  271. //
  272. // EXIST_LOCALE_INFO
  273. //
  274. // Checks to see if the locale tables have been added to the locale
  275. // hash node.
  276. //
  277. // Must check the FIXED locale pointer, since that value is set last in
  278. // the hash node.
  279. //
  280. // DEFINED AS A MACRO.
  281. //
  282. // 05-31-91 JulieB Created.
  283. ////////////////////////////////////////////////////////////////////////////
  284. #define EXIST_LOCALE_INFO(pHashN) (pHashN->pLocaleFixed)
  285. ////////////////////////////////////////////////////////////////////////////
  286. //
  287. // INSERT_CP_HASH_NODE
  288. //
  289. // Inserts a CP hash node into the global CP hash table. It assumes that
  290. // all unused hash values in the table are pointing to NULL. If there is
  291. // a collision, the new node will be added FIRST in the list.
  292. //
  293. // DEFINED AS A MACRO.
  294. //
  295. // 05-31-91 JulieB Created.
  296. ////////////////////////////////////////////////////////////////////////////
  297. #define INSERT_CP_HASH_NODE( pHashN, \
  298. pBaseAddr ) \
  299. { \
  300. UINT Index; /* hash value */ \
  301. PCP_HASH pSearch; /* ptr to CP hash node for search */ \
  302. \
  303. \
  304. /* \
  305. * Get hash value. \
  306. */ \
  307. Index = GET_HASH_VALUE(pHashN->CodePage, CP_TBL_SIZE); \
  308. \
  309. /* \
  310. * Enter table pointers critical section. \
  311. */ \
  312. RtlEnterCriticalSection(&gcsTblPtrs); \
  313. \
  314. /* \
  315. * Make sure the hash node still doesn't exist in the table. \
  316. */ \
  317. pSearch = (pTblPtrs->pCPHashTbl)[Index]; \
  318. while ((pSearch != NULL) && (pSearch->CodePage != pHashN->CodePage)) \
  319. { \
  320. pSearch = pSearch->pNext; \
  321. } \
  322. \
  323. /* \
  324. * If the hash node does not exist, insert the new one. \
  325. * Otherwise, free it. \
  326. */ \
  327. if (pSearch == NULL) \
  328. { \
  329. /* \
  330. * Insert hash node into hash table. \
  331. */ \
  332. pHashN->pNext = (pTblPtrs->pCPHashTbl)[Index]; \
  333. (pTblPtrs->pCPHashTbl)[Index] = pHashN; \
  334. } \
  335. else \
  336. { \
  337. /* \
  338. * Free the resources allocated. \
  339. */ \
  340. if (pBaseAddr) \
  341. { \
  342. UnMapSection(pBaseAddr); \
  343. } \
  344. NLS_FREE_MEM(pHashN); \
  345. } \
  346. \
  347. /* \
  348. * Leave table pointers critical section. \
  349. */ \
  350. RtlLeaveCriticalSection(&gcsTblPtrs); \
  351. }
  352. ////////////////////////////////////////////////////////////////////////////
  353. //
  354. // INSERT_LOC_HASH_NODE
  355. //
  356. // Inserts a LOC hash node into the global LOC hash table. It assumes
  357. // that all unused hash values in the table are pointing to NULL. If
  358. // there is a collision, the new node will be added FIRST in the list.
  359. //
  360. // DEFINED AS A MACRO.
  361. //
  362. // 05-31-91 JulieB Created.
  363. ////////////////////////////////////////////////////////////////////////////
  364. #define INSERT_LOC_HASH_NODE( pHashN, \
  365. pBaseAddr ) \
  366. { \
  367. UINT Index; /* hash value */ \
  368. PLOC_HASH pSearch; /* ptr to LOC hash node for search */ \
  369. \
  370. \
  371. /* \
  372. * Get hash value. \
  373. */ \
  374. Index = GET_HASH_VALUE(pHashN->Locale, LOC_TBL_SIZE); \
  375. \
  376. /* \
  377. * Enter table pointers critical section. \
  378. */ \
  379. RtlEnterCriticalSection(&gcsTblPtrs); \
  380. \
  381. /* \
  382. * Make sure the hash node still doesn't exist in the table. \
  383. */ \
  384. pSearch = (pTblPtrs->pLocHashTbl)[Index]; \
  385. while ((pSearch != NULL) && (pSearch->Locale != pHashN->Locale)) \
  386. { \
  387. pSearch = pSearch->pNext; \
  388. } \
  389. \
  390. /* \
  391. * If the hash node does not exist, insert the new one. \
  392. * Otherwise, free it. \
  393. */ \
  394. if (pSearch == NULL) \
  395. { \
  396. /* \
  397. * Insert hash node into hash table. \
  398. */ \
  399. pHashN->pNext = (pTblPtrs->pLocHashTbl)[Index]; \
  400. (pTblPtrs->pLocHashTbl)[Index] = pHashN; \
  401. } \
  402. else \
  403. { \
  404. /* \
  405. * Free the resources allocated. \
  406. */ \
  407. if (pBaseAddr) \
  408. { \
  409. UnMapSection(pBaseAddr); \
  410. } \
  411. if ((pHashN->pSortkey != pTblPtrs->pDefaultSortkey) && \
  412. (pHashN->pSortkey != NULL)) \
  413. { \
  414. UnMapSection(((LPWORD)(pHashN->pSortkey)) - SORTKEY_HEADER); \
  415. } \
  416. NLS_FREE_MEM(pHashN); \
  417. } \
  418. \
  419. /* \
  420. * Leave table pointers critical section. \
  421. */ \
  422. RtlLeaveCriticalSection(&gcsTblPtrs); \
  423. }
  424. ////////////////////////////////////////////////////////////////////////////
  425. //
  426. // GET_CP_SECTION_NAME
  427. //
  428. // Gets the section name for a given code page.
  429. //
  430. // NOTE: This macro may return if an error is encountered.
  431. //
  432. // DEFINED AS A MACRO.
  433. //
  434. // 05-31-91 JulieB Created.
  435. ////////////////////////////////////////////////////////////////////////////
  436. #define GET_CP_SECTION_NAME( CodePage, \
  437. pwszSecName, \
  438. cchSecName, \
  439. pObSecName ) \
  440. { \
  441. if (rc = GetNlsSectionName( CodePage, \
  442. 10, \
  443. 0, \
  444. NLS_SECTION_CPPREFIX, \
  445. pwszSecName, \
  446. cchSecName)) \
  447. { \
  448. return (rc); \
  449. } \
  450. RtlInitUnicodeString(pObSecName, pwszSecName); \
  451. }
  452. ////////////////////////////////////////////////////////////////////////////
  453. //
  454. // GET_SORTKEY_SECTION_NAME
  455. //
  456. // Gets the sortkey section name for a given locale.
  457. //
  458. // NOTE: This macro may return if an error is encountered.
  459. //
  460. // DEFINED AS A MACRO.
  461. //
  462. // 05-31-91 JulieB Created.
  463. ////////////////////////////////////////////////////////////////////////////
  464. #define GET_SORTKEY_SECTION_NAME( Locale, \
  465. pwszSecName, \
  466. cchSecName, \
  467. pObSecName ) \
  468. { \
  469. if (rc = GetNlsSectionName( Locale, \
  470. 16, \
  471. 8, \
  472. NLS_SECTION_SORTKEY, \
  473. pwszSecName, \
  474. cchSecName)) \
  475. { \
  476. return (rc); \
  477. } \
  478. RtlInitUnicodeString(pObSecName, pwszSecName); \
  479. }
  480. ////////////////////////////////////////////////////////////////////////////
  481. //
  482. // GET_LANG_SECTION_NAME
  483. //
  484. // Gets the section name for a given language.
  485. //
  486. // NOTE: This macro may return if an error is encountered.
  487. //
  488. // DEFINED AS A MACRO.
  489. //
  490. // 05-31-91 JulieB Created.
  491. ////////////////////////////////////////////////////////////////////////////
  492. #define GET_LANG_SECTION_NAME( Locale, \
  493. pwszSecName, \
  494. cchSecName, \
  495. pObSecName ) \
  496. { \
  497. if (rc = GetNlsSectionName( Locale, \
  498. 16, \
  499. 8, \
  500. NLS_SECTION_LANGPREFIX, \
  501. pwszSecName, \
  502. cchSecName)) \
  503. { \
  504. return (rc); \
  505. } \
  506. RtlInitUnicodeString(pObSecName, pwszSecName); \
  507. }
  508. //-------------------------------------------------------------------------//
  509. // EXTERNAL ROUTINES //
  510. //-------------------------------------------------------------------------//
  511. ////////////////////////////////////////////////////////////////////////////
  512. //
  513. // AllocTables
  514. //
  515. // Allocates the global table pointers structure. It then allocates the
  516. // code page and locale hash tables and saves the pointers to the tables
  517. // in the global table pointers structure.
  518. //
  519. // 05-31-91 JulieB Created.
  520. ////////////////////////////////////////////////////////////////////////////
  521. ULONG AllocTables()
  522. {
  523. //
  524. // Allocate global table pointers structure.
  525. //
  526. if ((pTblPtrs = (PTBL_PTRS)NLS_ALLOC_MEM(sizeof(TBL_PTRS))) == NULL)
  527. {
  528. KdPrint(("NLSAPI: Allocation for TABLE PTRS structure FAILED.\n"));
  529. return (ERROR_OUTOFMEMORY);
  530. }
  531. //
  532. // Allocate code page hash table.
  533. //
  534. if ((pTblPtrs->pCPHashTbl =
  535. (PCP_HASH_TBL)NLS_ALLOC_MEM(sizeof(PCP_HASH) * CP_TBL_SIZE)) == NULL)
  536. {
  537. KdPrint(("NLSAPI: Allocation for CODE PAGE hash table FAILED.\n"));
  538. return (ERROR_OUTOFMEMORY);
  539. }
  540. //
  541. // Allocate locale hash table.
  542. //
  543. if ((pTblPtrs->pLocHashTbl =
  544. (PLOC_HASH_TBL)NLS_ALLOC_MEM(sizeof(PLOC_HASH) * LOC_TBL_SIZE)) == NULL)
  545. {
  546. KdPrint(("NLSAPI: Allocation for LOCALE hash table FAILED.\n"));
  547. return (ERROR_OUTOFMEMORY);
  548. }
  549. //
  550. // Return success.
  551. //
  552. return (NO_ERROR);
  553. }
  554. ////////////////////////////////////////////////////////////////////////////
  555. //
  556. // GetUnicodeFileInfo
  557. //
  558. // Opens and Maps a view of the section for the unicode file. It then
  559. // fills in the appropriate fields of the global table pointers structure.
  560. //
  561. // 05-31-91 JulieB Created.
  562. ////////////////////////////////////////////////////////////////////////////
  563. ULONG GetUnicodeFileInfo()
  564. {
  565. HANDLE hSec = (HANDLE)0; // section handle
  566. UNICODE_STRING ObSecName; // section name
  567. LPWORD pBaseAddr; // ptr to base address of section
  568. ULONG rc = 0L; // return code
  569. WORD offCZ; // offset to FOLDCZONE table
  570. WORD offHG; // offset to HIRAGANA table
  571. WORD offKK; // offset to KATAKANA table
  572. WORD offHW; // offset to HALFWIDTH table
  573. WORD offFW; // offset to FULLWIDTH table
  574. WORD offTR; // offset to TRADITIONAL table
  575. WORD offSP; // offset to SIMPLIFIED table
  576. WORD offPre; // offset to PRECOMPOSED table
  577. WORD offComp; // offset to COMPOSITE table
  578. PCOMP_INFO pComp; // ptr to COMP_INFO structure
  579. //
  580. // Make sure we're in the critical section when entering this call.
  581. //
  582. ASSERT(NtCurrentTeb()->ClientId.UniqueThread == gcsTblPtrs.OwningThread);
  583. //
  584. // Make sure the unicode information is not already there.
  585. // If it is, return success.
  586. //
  587. // Since we're already in the critical section here, there is no
  588. // need to check ALL of the pointers set in this routine. Just
  589. // check one of them.
  590. //
  591. if (pTblPtrs->pADigit != NULL)
  592. {
  593. return (NO_ERROR);
  594. }
  595. //
  596. // Open and Map a view of the section.
  597. //
  598. RtlInitUnicodeString(&ObSecName, NLS_SECTION_UNICODE);
  599. if (rc = OpenSection( &hSec,
  600. &ObSecName,
  601. (PVOID *)&pBaseAddr,
  602. SECTION_MAP_READ,
  603. TRUE ))
  604. {
  605. return (rc);
  606. }
  607. //
  608. // Get the offsets.
  609. //
  610. offCZ = pBaseAddr[0];
  611. offHG = offCZ + pBaseAddr[offCZ];
  612. offKK = offHG + pBaseAddr[offHG];
  613. offHW = offKK + pBaseAddr[offKK];
  614. offFW = offHW + pBaseAddr[offHW];
  615. offTR = offFW + pBaseAddr[offFW];
  616. offSP = offTR + pBaseAddr[offTR];
  617. offPre = offSP + pBaseAddr[offSP];
  618. offComp = offPre + pBaseAddr[offPre];
  619. //
  620. // Allocate COMP_INFO structure.
  621. //
  622. if ((pComp = (PCOMP_INFO)NLS_ALLOC_MEM(sizeof(COMP_INFO))) == NULL)
  623. {
  624. return (ERROR_OUTOFMEMORY);
  625. }
  626. //
  627. // Fill in the COMPOSITE information.
  628. //
  629. pComp->NumBase = LOBYTE((pBaseAddr + offComp)[2]);
  630. pComp->NumNonSp = HIBYTE((pBaseAddr + offComp)[2]);
  631. pComp->pBase = pBaseAddr + offComp + CO_HEADER;
  632. pComp->pNonSp = pComp->pBase + ((pBaseAddr + offComp)[0]);
  633. pComp->pGrid = pComp->pNonSp + ((pBaseAddr + offComp)[1]);
  634. //
  635. // Attach ASCIIDIGITS table to tbl ptrs structure.
  636. //
  637. pTblPtrs->pADigit = pBaseAddr + AD_HEADER;
  638. //
  639. // Attach FOLDCZONE table to tbl ptrs structure.
  640. //
  641. pTblPtrs->pCZone = pBaseAddr + offCZ + CZ_HEADER;
  642. //
  643. // Attach HIRAGANA table to tbl ptrs structure.
  644. //
  645. pTblPtrs->pHiragana = pBaseAddr + offHG + HG_HEADER;
  646. //
  647. // Attach KATAKANA table to tbl ptrs structure.
  648. //
  649. pTblPtrs->pKatakana = pBaseAddr + offKK + KK_HEADER;
  650. //
  651. // Attach HALFWIDTH table to tbl ptrs structure.
  652. //
  653. pTblPtrs->pHalfWidth = pBaseAddr + offHW + HW_HEADER;
  654. //
  655. // Attach FULLWIDTH table to tbl ptrs structure.
  656. //
  657. pTblPtrs->pFullWidth = pBaseAddr + offFW + FW_HEADER;
  658. //
  659. // Attach TRADITIONAL table to tbl ptrs structure.
  660. //
  661. pTblPtrs->pTraditional = pBaseAddr + offTR + TR_HEADER;
  662. //
  663. // Attach SIMPLIFIED table to tbl ptrs structure.
  664. //
  665. pTblPtrs->pSimplified = pBaseAddr + offSP + SP_HEADER;
  666. //
  667. // Attach PRECOMPOSED table to tbl ptrs structure.
  668. //
  669. pTblPtrs->pPreComposed = pBaseAddr + offPre + PC_HEADER;
  670. //
  671. // Attach COMP_INFO to tbl ptrs structure.
  672. //
  673. pTblPtrs->pComposite = pComp;
  674. //
  675. // Return success.
  676. //
  677. return (NO_ERROR);
  678. }
  679. ////////////////////////////////////////////////////////////////////////////
  680. //
  681. // GetGeoFileInfo
  682. //
  683. // Opens and Maps a view of the section for the geo file. It then
  684. // fills in the appropriate field of the global table pointers structure.
  685. // Before calling this function you should check pGeoInfo member.
  686. //
  687. ////////////////////////////////////////////////////////////////////////////
  688. ULONG GetGeoFileInfo()
  689. {
  690. HANDLE hSec = (HANDLE)0; // section handle
  691. UNICODE_STRING ObSecName; // section name
  692. LPWORD pBaseAddr; // ptr to base address of section
  693. ULONG rc = NO_ERROR; // return code
  694. //
  695. // Enter the critical section to set up the GEO tables.
  696. //
  697. RtlEnterCriticalSection(&gcsTblPtrs);
  698. //
  699. // Make sure the Geographical Information table is
  700. // not already there. If it is, return TRUE.
  701. //
  702. // Since we're already in the critical section here, there is no
  703. // need to check ALL of the pointers set in this routine. Just
  704. // check one of them.
  705. //
  706. if (pTblPtrs->pGeoInfo != NULL)
  707. {
  708. RtlLeaveCriticalSection(&gcsTblPtrs);
  709. return (NO_ERROR);
  710. }
  711. //
  712. // Create and map the section, and then save the pointer.
  713. //
  714. if ((rc = CsrBasepNlsCreateSection(NLS_CREATE_SECTION_GEO, 0, &hSec)) == NO_ERROR)
  715. {
  716. //
  717. // Map a View of the Section.
  718. //
  719. if ((rc = MapSection( hSec,
  720. &pBaseAddr,
  721. PAGE_READONLY,
  722. TRUE )) != NO_ERROR)
  723. {
  724. RtlLeaveCriticalSection(&gcsTblPtrs);
  725. return (rc);
  726. }
  727. }
  728. else
  729. {
  730. RtlLeaveCriticalSection(&gcsTblPtrs);
  731. return (rc);
  732. }
  733. //
  734. // Attach GeoInfo mapping table, GEO/LCID mapping table and
  735. // GEO/ISO639 name mapping table to the tbl ptrs structure. We initialize
  736. // the pGeoinfo member at the end so we don't get a race condition.
  737. //
  738. pTblPtrs->nGeoLCID = ((PGEOTABLEHDR)pBaseAddr)->nGeoLCID;
  739. pTblPtrs->pGeoLCID = (PGEOLCID)((PBYTE)pBaseAddr + ((PGEOTABLEHDR)pBaseAddr)->dwOffsetGeoLCID);
  740. pTblPtrs->nGeoInfo = ((PGEOTABLEHDR)pBaseAddr)->nGeoInfo;
  741. pTblPtrs->pGeoInfo = (PGEOINFO)((PBYTE)pBaseAddr + ((PGEOTABLEHDR)pBaseAddr)->dwOffsetGeoInfo);
  742. //
  743. // Leave table pointers critical section.
  744. //
  745. RtlLeaveCriticalSection(&gcsTblPtrs);
  746. //
  747. // Return success.
  748. //
  749. return (NO_ERROR);
  750. }
  751. ////////////////////////////////////////////////////////////////////////////
  752. //
  753. // GetCTypeFileInfo
  754. //
  755. // Opens and Maps a view of the section for the given ctype. It then
  756. // fills in the appropriate field of the global table pointers structure.
  757. //
  758. // 05-31-91 JulieB Created.
  759. ////////////////////////////////////////////////////////////////////////////
  760. ULONG GetCTypeFileInfo()
  761. {
  762. HANDLE hSec = (HANDLE)0; // section handle
  763. UNICODE_STRING ObSecName; // section name
  764. LPWORD pBaseAddr; // ptr to base address of section
  765. ULONG rc = 0L; // return code
  766. //
  767. // Make sure the ctype information is not already there.
  768. // If it is, return success.
  769. //
  770. // Must check the 844 table rather than the mapping table, since
  771. // the 844 table is set AFTER the mapping table below. Otherwise,
  772. // there is a race condition, since we're not in a critical section.
  773. //
  774. if (pTblPtrs->pCType844 != NULL)
  775. {
  776. return (NO_ERROR);
  777. }
  778. //
  779. // Enter table pointers critical section.
  780. //
  781. RtlEnterCriticalSection(&gcsTblPtrs);
  782. if (pTblPtrs->pCType844 != NULL)
  783. {
  784. RtlLeaveCriticalSection(&gcsTblPtrs);
  785. return (NO_ERROR);
  786. }
  787. //
  788. // Open and Map a view of the section.
  789. //
  790. RtlInitUnicodeString(&ObSecName, NLS_SECTION_CTYPE);
  791. if (rc = OpenSection( &hSec,
  792. &ObSecName,
  793. (PVOID *)&pBaseAddr,
  794. SECTION_MAP_READ,
  795. TRUE ))
  796. {
  797. RtlLeaveCriticalSection(&gcsTblPtrs);
  798. return (rc);
  799. }
  800. //
  801. // Attach CTYPE mapping table and 8:4:4 table to tbl ptrs structure.
  802. //
  803. // The pCType844 value must be set LAST, since this is the pointer
  804. // that is checked to see that the ctype information has been
  805. // initialized.
  806. //
  807. pTblPtrs->pCTypeMap = (PCT_VALUES)(pBaseAddr + CT_HEADER);
  808. pTblPtrs->pCType844 = (PCTYPE)((LPBYTE)(pBaseAddr + 1) +
  809. ((PCTYPE_HDR)pBaseAddr)->MapSize);
  810. //
  811. // Leave table pointers critical section.
  812. //
  813. RtlLeaveCriticalSection(&gcsTblPtrs);
  814. //
  815. // Return success.
  816. //
  817. return (NO_ERROR);
  818. }
  819. ////////////////////////////////////////////////////////////////////////////
  820. //
  821. // GetDefaultSortkeyFileInfo
  822. //
  823. // Opens and Maps a view of the section for the default sortkey table. It
  824. // then stores the pointer to the table in the global pointer table.
  825. //
  826. // NOTE: THIS ROUTINE SHOULD ONLY BE CALLED AT PROCESS STARTUP. If it is
  827. // called from other than process startup, a critical section must
  828. // be placed around the assigning of the pointers to pTblPtrs.
  829. //
  830. // 05-31-91 JulieB Created.
  831. ////////////////////////////////////////////////////////////////////////////
  832. ULONG GetDefaultSortkeyFileInfo()
  833. {
  834. HANDLE hSec = (HANDLE)0; // section handle
  835. UNICODE_STRING ObSecName; // section name
  836. LPWORD pBaseAddr; // ptr to base address of section
  837. ULONG rc = 0L; // return code
  838. SECTION_BASIC_INFORMATION SecInfo; // section information - query
  839. //
  840. // Make sure we're in the critical section when entering this call.
  841. //
  842. ASSERT(NtCurrentTeb()->ClientId.UniqueThread == gcsTblPtrs.OwningThread);
  843. //
  844. // Open and Map a view of the section if it hasn't been done yet.
  845. //
  846. if (pTblPtrs->pDefaultSortkey != NULL)
  847. {
  848. return (NO_ERROR);
  849. }
  850. RtlInitUnicodeString(&ObSecName, NLS_SECTION_SORTKEY);
  851. if (rc = OpenSection( &hSec,
  852. &ObSecName,
  853. (PVOID *)&pBaseAddr,
  854. SECTION_MAP_READ | SECTION_QUERY,
  855. FALSE ))
  856. {
  857. KdPrint(("NLSAPI: Could NOT Open Section %wZ - %lx.\n", &ObSecName, rc));
  858. return (rc);
  859. }
  860. //
  861. // Query size of default section.
  862. //
  863. rc = NtQuerySection( hSec,
  864. SectionBasicInformation,
  865. &SecInfo,
  866. sizeof(SecInfo),
  867. NULL );
  868. //
  869. // Close the section handle.
  870. //
  871. NtClose(hSec);
  872. //
  873. // Check for error from NtQuerySection.
  874. //
  875. if (!NT_SUCCESS(rc))
  876. {
  877. KdPrint(("NLSAPI: Could NOT Query Section %wZ - %lx.\n", &ObSecName, rc));
  878. return (rc);
  879. }
  880. //
  881. // Get Default Sortkey Information.
  882. //
  883. pTblPtrs->pDefaultSortkey = (PSORTKEY)(pBaseAddr + SORTKEY_HEADER);
  884. pTblPtrs->DefaultSortkeySize = SecInfo.MaximumSize;
  885. //
  886. // Return success.
  887. //
  888. return (NO_ERROR);
  889. }
  890. ////////////////////////////////////////////////////////////////////////////
  891. //
  892. // GetDefaultSortTablesFileInfo
  893. //
  894. // Opens and Maps a view of the section for the sort tables. It then
  895. // stores the pointers to the various tables in the global pointer table.
  896. //
  897. // NOTE: THIS ROUTINE SHOULD ONLY BE CALLED AT PROCESS STARTUP. If it is
  898. // called from other than process startup, a critical section must
  899. // be placed around the assigning of the pointers to pTblPtrs.
  900. //
  901. // 05-31-91 JulieB Created.
  902. ////////////////////////////////////////////////////////////////////////////
  903. ULONG GetDefaultSortTablesFileInfo()
  904. {
  905. HANDLE hSec = (HANDLE)0; // section handle
  906. UNICODE_STRING ObSecName; // section name
  907. LPWORD pBaseAddr; // word ptr to base address of section
  908. DWORD Num; // number of entries in table
  909. PCOMPRESS_HDR pCompressHdr; // ptr to compression header
  910. PEXCEPT_HDR pExceptHdr; // ptr to exception header
  911. ULONG rc = 0L; // return code
  912. //
  913. // Make sure we're in the critical section when entering this call.
  914. //
  915. ASSERT(NtCurrentTeb()->ClientId.UniqueThread == gcsTblPtrs.OwningThread);
  916. //
  917. // Open and Map a view of the section if it hasn't been done yet.
  918. //
  919. // Since we're already in the critical section here, there is no
  920. // need to check ALL of the pointers set in this routine. Just
  921. // check one of them.
  922. //
  923. if (pTblPtrs->pReverseDW != NULL)
  924. {
  925. return (NO_ERROR);
  926. }
  927. RtlInitUnicodeString(&ObSecName, NLS_SECTION_SORTTBLS);
  928. if (rc = OpenSection( &hSec,
  929. &ObSecName,
  930. (PVOID *)&pBaseAddr,
  931. SECTION_MAP_READ,
  932. TRUE ))
  933. {
  934. return (rc);
  935. }
  936. //
  937. // Get Reverse Diacritic Information.
  938. //
  939. Num = *((LPDWORD)pBaseAddr);
  940. if (Num > 0)
  941. {
  942. pTblPtrs->NumReverseDW = Num;
  943. pTblPtrs->pReverseDW = (PREVERSE_DW)(pBaseAddr + REV_DW_HEADER);
  944. }
  945. pBaseAddr += REV_DW_HEADER + (Num * (sizeof(REVERSE_DW) / sizeof(WORD)));
  946. //
  947. // Get Double Compression Information.
  948. //
  949. Num = *((LPDWORD)pBaseAddr);
  950. if (Num > 0)
  951. {
  952. pTblPtrs->NumDblCompression = Num;
  953. pTblPtrs->pDblCompression = (PDBL_COMPRESS)(pBaseAddr + DBL_COMP_HEADER);
  954. }
  955. pBaseAddr += DBL_COMP_HEADER + (Num * (sizeof(DBL_COMPRESS) / sizeof(WORD)));
  956. //
  957. // Get Ideograph Lcid Exception Information.
  958. //
  959. Num = *((LPDWORD)pBaseAddr);
  960. if (Num > 0)
  961. {
  962. pTblPtrs->NumIdeographLcid = Num;
  963. pTblPtrs->pIdeographLcid = (PIDEOGRAPH_LCID)(pBaseAddr + IDEO_LCID_HEADER);
  964. }
  965. pBaseAddr += IDEO_LCID_HEADER + (Num * (sizeof(IDEOGRAPH_LCID) / sizeof(WORD)));
  966. //
  967. // Get Expansion Information.
  968. //
  969. Num = *((LPDWORD)pBaseAddr);
  970. if (Num > 0)
  971. {
  972. pTblPtrs->NumExpansion = Num;
  973. pTblPtrs->pExpansion = (PEXPAND)(pBaseAddr + EXPAND_HEADER);
  974. }
  975. pBaseAddr += EXPAND_HEADER + (Num * (sizeof(EXPAND) / sizeof(WORD)));
  976. //
  977. // Get Compression Information.
  978. //
  979. Num = *((LPDWORD)pBaseAddr);
  980. if (Num > 0)
  981. {
  982. pTblPtrs->NumCompression = Num;
  983. pTblPtrs->pCompressHdr = (PCOMPRESS_HDR)(pBaseAddr + COMPRESS_HDR_OFFSET);
  984. pTblPtrs->pCompression = (PCOMPRESS)(pBaseAddr + COMPRESS_HDR_OFFSET +
  985. (Num * (sizeof(COMPRESS_HDR) /
  986. sizeof(WORD))));
  987. }
  988. pCompressHdr = pTblPtrs->pCompressHdr;
  989. pBaseAddr = (LPWORD)(pTblPtrs->pCompression) +
  990. (pCompressHdr[Num - 1]).Offset;
  991. pBaseAddr += (((pCompressHdr[Num - 1]).Num2) *
  992. (sizeof(COMPRESS_2) / sizeof(WORD)));
  993. pBaseAddr += (((pCompressHdr[Num - 1]).Num3) *
  994. (sizeof(COMPRESS_3) / sizeof(WORD)));
  995. //
  996. // Get Exception Information.
  997. //
  998. Num = *((LPDWORD)pBaseAddr);
  999. if (Num > 0)
  1000. {
  1001. pTblPtrs->NumException = Num;
  1002. pTblPtrs->pExceptHdr = (PEXCEPT_HDR)(pBaseAddr + EXCEPT_HDR_OFFSET);
  1003. pTblPtrs->pException = (PEXCEPT)(pBaseAddr + EXCEPT_HDR_OFFSET +
  1004. (Num * (sizeof(EXCEPT_HDR) /
  1005. sizeof(WORD))));
  1006. }
  1007. pExceptHdr = pTblPtrs->pExceptHdr;
  1008. pBaseAddr = (LPWORD)(pTblPtrs->pException) +
  1009. (pExceptHdr[Num - 1]).Offset;
  1010. pBaseAddr += (((pExceptHdr[Num - 1]).NumEntries) *
  1011. (sizeof(EXCEPT) / sizeof(WORD)));
  1012. //
  1013. // Get Multiple Weights Information.
  1014. //
  1015. Num = (DWORD)(*pBaseAddr);
  1016. if (Num > 0)
  1017. {
  1018. pTblPtrs->NumMultiWeight = Num;
  1019. pTblPtrs->pMultiWeight = (PMULTI_WT)(pBaseAddr + MULTI_WT_HEADER);
  1020. }
  1021. pBaseAddr += MULTI_WT_HEADER + (Num * (sizeof(MULTI_WT) / sizeof(WORD)));
  1022. //
  1023. // Get Jamo Index Table.
  1024. //
  1025. Num = (DWORD)(*pBaseAddr);
  1026. if (Num > 0)
  1027. {
  1028. //
  1029. // The Jamo Index table size is (Num) bytes.
  1030. //
  1031. pTblPtrs->NumJamoIndex = Num;
  1032. pTblPtrs->pJamoIndex = (PJAMO_TABLE)(pBaseAddr + JAMO_INDEX_HEADER);
  1033. }
  1034. pBaseAddr += JAMO_INDEX_HEADER + (Num * (sizeof(JAMO_TABLE) / sizeof(WORD)));
  1035. //
  1036. // Get Jamo Composition State Machine Table.
  1037. //
  1038. Num = (DWORD)(*pBaseAddr);
  1039. if (Num > 0)
  1040. {
  1041. pTblPtrs->NumJamoComposition = Num;
  1042. pTblPtrs->pJamoComposition = (PJAMO_COMPOSE_STATE)(pBaseAddr + JAMO_COMPOSITION_HEADER);
  1043. }
  1044. //
  1045. // The following line is used to move pBaseAddr to the next field.
  1046. // Uncomment it if you are adding more fields.
  1047. //
  1048. // pBaseAddr += JAMO_COMPOSITION_HEADER + (Num * (sizeof(JAMO_COMPOSE_STATE) / sizeof(WORD)));
  1049. //
  1050. // Return success.
  1051. //
  1052. return (NO_ERROR);
  1053. }
  1054. ////////////////////////////////////////////////////////////////////////////
  1055. //
  1056. // GetSortkeyFileInfo
  1057. //
  1058. // Opens and Maps a view of the section for the sortkey file. It then
  1059. // fills in the appropriate field of the global table pointers structure.
  1060. //
  1061. // 05-31-91 JulieB Created.
  1062. ////////////////////////////////////////////////////////////////////////////
  1063. ULONG GetSortkeyFileInfo(
  1064. LCID Locale,
  1065. PLOC_HASH pHashN)
  1066. {
  1067. HANDLE hSec = (HANDLE)0; // section handle
  1068. UNICODE_STRING ObSecName; // section name
  1069. LPWORD pBaseAddr; // ptr to base address of section
  1070. ULONG rc = 0L; // return code
  1071. PEXCEPT_HDR pExceptHdr; // ptr to exception header
  1072. PEXCEPT pExceptTbl; // ptr to exception table
  1073. PVOID pIdeograph; // ptr to ideograph exception table
  1074. WCHAR wszSecName[MAX_SMALL_BUF_LEN]; // Place for the section name string \
  1075. NTSTATUS Status;
  1076. //
  1077. // Make sure we're in the critical section when entering this call.
  1078. //
  1079. ASSERT(NtCurrentTeb()->ClientId.UniqueThread == gcsTblPtrs.OwningThread);
  1080. //
  1081. // Make sure the default sortkey table is loaded. If it's not, then
  1082. // we shouldn't bother to continue here since the sorting table won't
  1083. // be created properly. pHashN->pSortkey will already be NULL, so
  1084. // we don't need to set it. Return NO_ERROR here to allow kernel32
  1085. // to initialize in case this is winlogon.
  1086. //
  1087. if (pTblPtrs->pDefaultSortkey == NULL)
  1088. {
  1089. KdPrint(("NLSAPI: No Default Sorting Table Loaded.\n"));
  1090. return (NO_ERROR);
  1091. }
  1092. //
  1093. // Try to Open and Map a view of the section (read only).
  1094. //
  1095. GET_SORTKEY_SECTION_NAME(Locale, wszSecName, MAX_SMALL_BUF_LEN, &ObSecName);
  1096. if (rc = OpenSection( &hSec,
  1097. &ObSecName,
  1098. (PVOID *)&pBaseAddr,
  1099. SECTION_MAP_READ,
  1100. FALSE ))
  1101. {
  1102. //
  1103. // Open failed.
  1104. // See if any exceptions exist for given Locale ID.
  1105. //
  1106. rc = NO_ERROR;
  1107. if (!FindExceptionPointers( Locale,
  1108. &pExceptHdr,
  1109. &pExceptTbl,
  1110. &pIdeograph,
  1111. &rc ))
  1112. {
  1113. //
  1114. // No exceptions for locale, so attach the default sortkey
  1115. // table pointer to the hash node and return success.
  1116. //
  1117. pHashN->pSortkey = pTblPtrs->pDefaultSortkey;
  1118. return (NO_ERROR);
  1119. }
  1120. else
  1121. {
  1122. //
  1123. // See if an error occurred.
  1124. //
  1125. if (rc != NO_ERROR)
  1126. {
  1127. //
  1128. // This occurs if the ideograph exception file could not be
  1129. // created or mapped. Return an error in this case.
  1130. //
  1131. // return (rc);
  1132. //
  1133. // On second thought, don't return an error. Returning an
  1134. // error can cause kernel32 to not initialize (which, if this
  1135. // is winlogon, leads to an unbootable system). Let's just
  1136. // patch things up and move on.
  1137. //
  1138. // LATER -- log an error in the logfile.
  1139. //
  1140. pHashN->IfIdeographFailure = TRUE;
  1141. pHashN->pSortkey = pTblPtrs->pDefaultSortkey;
  1142. return (NO_ERROR);
  1143. }
  1144. //
  1145. // Exceptions from default sortkey table exist for the given
  1146. // locale. Need to get the correct sortkey table.
  1147. // Create a section and call the server to lock it in.
  1148. //
  1149. Status = CsrBasepNlsCreateSection( NLS_CREATE_SORT_SECTION,
  1150. Locale,
  1151. &hSec );
  1152. //
  1153. // Check return from server call.
  1154. //
  1155. rc = (ULONG)Status;
  1156. if (!NT_SUCCESS(rc))
  1157. {
  1158. if (hSec != NULL)
  1159. {
  1160. NtClose(hSec);
  1161. }
  1162. return (rc);
  1163. }
  1164. //
  1165. // Map the section for ReadWrite.
  1166. //
  1167. if (rc = MapSection( hSec,
  1168. (PVOID *)&pBaseAddr,
  1169. PAGE_READWRITE,
  1170. FALSE ))
  1171. {
  1172. NtClose(hSec);
  1173. return (rc);
  1174. }
  1175. //
  1176. // Copy the Default Sortkey Table to the New Section.
  1177. //
  1178. RtlMoveMemory( (PVOID)pBaseAddr,
  1179. (PVOID)((LPWORD)(pTblPtrs->pDefaultSortkey) -
  1180. SORTKEY_HEADER),
  1181. (ULONG)(pTblPtrs->DefaultSortkeySize.LowPart) );
  1182. //
  1183. // Copy exception information to the table.
  1184. //
  1185. CopyExceptionInfo( (PSORTKEY)(pBaseAddr + SORTKEY_HEADER),
  1186. pExceptHdr,
  1187. pExceptTbl,
  1188. pIdeograph);
  1189. //
  1190. // Write a 1 to the WORD semaphore (table may now be read).
  1191. //
  1192. *pBaseAddr = 1;
  1193. //
  1194. // Unmap the section for Write and remap it for Read.
  1195. //
  1196. if ((rc = UnMapSection(pBaseAddr)) ||
  1197. (rc = MapSection( hSec,
  1198. (PVOID *)&pBaseAddr,
  1199. PAGE_READONLY,
  1200. FALSE )))
  1201. {
  1202. NtClose(hSec);
  1203. return (rc);
  1204. }
  1205. }
  1206. }
  1207. //
  1208. // Close the section handle.
  1209. //
  1210. NtClose(hSec);
  1211. //
  1212. // Check semaphore bit in file. Make sure that the open
  1213. // succeeded AFTER all exceptions were added to the memory
  1214. // mapped section.
  1215. //
  1216. if (*pBaseAddr == 0)
  1217. {
  1218. //
  1219. // Another process is still adding the appropriate exception
  1220. // information. Must wait for its completion.
  1221. //
  1222. if (rc = WaitOnEvent(pBaseAddr))
  1223. {
  1224. UnMapSection(pBaseAddr);
  1225. return (rc);
  1226. }
  1227. }
  1228. //
  1229. // Save pointer in hash node.
  1230. //
  1231. pHashN->pSortkey = (PSORTKEY)(pBaseAddr + SORTKEY_HEADER);
  1232. //
  1233. // Return success.
  1234. //
  1235. return (NO_ERROR);
  1236. }
  1237. ////////////////////////////////////////////////////////////////////////////
  1238. //
  1239. // GetSortTablesFileInfo
  1240. //
  1241. // Stores the appropriate sort table pointers for the given locale in
  1242. // the given locale hash node.
  1243. //
  1244. // 05-31-91 JulieB Created.
  1245. ////////////////////////////////////////////////////////////////////////////
  1246. void GetSortTablesFileInfo(
  1247. LCID Locale,
  1248. PLOC_HASH pHashN)
  1249. {
  1250. DWORD ctr; // loop counter
  1251. PREVERSE_DW pRevDW; // ptr to reverse diacritic table
  1252. PDBL_COMPRESS pDblComp; // ptr to double compression table
  1253. PCOMPRESS_HDR pCompHdr; // ptr to compression header
  1254. //
  1255. // Make sure we're in the critical section when entering this call.
  1256. //
  1257. ASSERT(NtCurrentTeb()->ClientId.UniqueThread == gcsTblPtrs.OwningThread);
  1258. //
  1259. // Check for Reverse Diacritic Locale.
  1260. //
  1261. pRevDW = pTblPtrs->pReverseDW;
  1262. for (ctr = pTblPtrs->NumReverseDW; ctr > 0; ctr--, pRevDW++)
  1263. {
  1264. if (*pRevDW == (DWORD)Locale)
  1265. {
  1266. pHashN->IfReverseDW = TRUE;
  1267. break;
  1268. }
  1269. }
  1270. //
  1271. // Check for Compression.
  1272. //
  1273. pCompHdr = pTblPtrs->pCompressHdr;
  1274. for (ctr = pTblPtrs->NumCompression; ctr > 0; ctr--, pCompHdr++)
  1275. {
  1276. if (pCompHdr->Locale == (DWORD)Locale)
  1277. {
  1278. pHashN->IfCompression = TRUE;
  1279. pHashN->pCompHdr = pCompHdr;
  1280. if (pCompHdr->Num2 > 0)
  1281. {
  1282. pHashN->pCompress2 = (PCOMPRESS_2)
  1283. (((LPWORD)(pTblPtrs->pCompression)) +
  1284. (pCompHdr->Offset));
  1285. }
  1286. if (pCompHdr->Num3 > 0)
  1287. {
  1288. pHashN->pCompress3 = (PCOMPRESS_3)
  1289. (((LPWORD)(pTblPtrs->pCompression)) +
  1290. (pCompHdr->Offset) +
  1291. (pCompHdr->Num2 *
  1292. (sizeof(COMPRESS_2) / sizeof(WORD))));
  1293. }
  1294. break;
  1295. }
  1296. }
  1297. //
  1298. // Check for Double Compression.
  1299. //
  1300. if (pHashN->IfCompression)
  1301. {
  1302. pDblComp = pTblPtrs->pDblCompression;
  1303. for (ctr = pTblPtrs->NumDblCompression; ctr > 0; ctr--, pDblComp++)
  1304. {
  1305. if (*pDblComp == (DWORD)Locale)
  1306. {
  1307. pHashN->IfDblCompression = TRUE;
  1308. break;
  1309. }
  1310. }
  1311. }
  1312. }
  1313. ////////////////////////////////////////////////////////////////////////////
  1314. //
  1315. // LoadCodePageAsDLL
  1316. //
  1317. // Try to load a code page as a DLL. If succeeded, the CodePage procedure
  1318. // is set.
  1319. //
  1320. // 05-27-99 SamerA Created.
  1321. ////////////////////////////////////////////////////////////////////////////
  1322. ULONG LoadCodePageAsDLL(
  1323. UINT CodePage,
  1324. LPFN_CP_PROC *ppfnCPProc)
  1325. {
  1326. WCHAR pDllName[MAX_PATH_LEN]; // ptr to DLL name
  1327. HANDLE hModCPDll; // module handle of code page DLL
  1328. ULONG rc = ERROR_INVALID_PARAMETER; // return code
  1329. UINT ErrorMode; // error mode
  1330. //
  1331. // Get the DLL name to load.
  1332. //
  1333. pDllName[0] = 0;
  1334. *ppfnCPProc = NULL;
  1335. if (NO_ERROR == GetCodePageDLLPathName(CodePage, pDllName, MAX_PATH_LEN) &&
  1336. NlsIsDll(pDllName))
  1337. {
  1338. //
  1339. // Load the DLL and get the procedure address.
  1340. // Turn off hard error popups.
  1341. //
  1342. ErrorMode = SetErrorMode(SEM_NOERROR);
  1343. SetErrorMode(SEM_NOERROR | ErrorMode);
  1344. hModCPDll = LoadLibrary(pDllName);
  1345. SetErrorMode(ErrorMode);
  1346. if (hModCPDll)
  1347. {
  1348. *ppfnCPProc =
  1349. (LPFN_CP_PROC)GetProcAddress( hModCPDll,
  1350. NLS_CP_DLL_PROC_NAME );
  1351. }
  1352. if (*ppfnCPProc == NULL)
  1353. {
  1354. if (hModCPDll)
  1355. {
  1356. rc = TYPE_E_DLLFUNCTIONNOTFOUND;
  1357. FreeLibrary(hModCPDll);
  1358. }
  1359. else
  1360. {
  1361. rc = TYPE_E_CANTLOADLIBRARY;
  1362. }
  1363. }
  1364. else
  1365. {
  1366. rc = NO_ERROR;
  1367. }
  1368. }
  1369. return (rc);
  1370. }
  1371. ////////////////////////////////////////////////////////////////////////////
  1372. //
  1373. // GetCodePageFileInfo
  1374. //
  1375. // Opens and Maps a view of the section for the given code page. It then
  1376. // creates and inserts a hash node into the global CP hash table.
  1377. //
  1378. // If the section cannot be opened, it then queries the registry to see if
  1379. // the information has been added since the initialization of the DLL. If
  1380. // so, then it creates the section and then opens and maps a view of it.
  1381. //
  1382. // 05-31-91 JulieB Created.
  1383. ////////////////////////////////////////////////////////////////////////////
  1384. ULONG GetCodePageFileInfo(
  1385. UINT CodePage,
  1386. PCP_HASH *ppNode)
  1387. {
  1388. HANDLE hSec = (HANDLE)0; // section handle
  1389. UNICODE_STRING ObSecName; // section name
  1390. LPWORD pBaseAddr = NULL; // ptr to base address of section
  1391. ULONG rc = 0L; // return code
  1392. BOOL IsDLL; // true if dll instead of data file
  1393. LPFN_CP_PROC pfnCPProc; // Code Page DLL Procedure
  1394. WCHAR wszSecName[MAX_SMALL_BUF_LEN]; // Place for the section name string \
  1395. //
  1396. // Make sure we're in the critical section when entering this call.
  1397. //
  1398. ASSERT(NtCurrentTeb()->ClientId.UniqueThread == gcsTblPtrs.OwningThread);
  1399. //
  1400. // See if we're dealing with a DLL or an NLS data file.
  1401. //
  1402. IsDLL = ((CodePage >= NLS_CP_DLL_RANGE) &&
  1403. (CodePage < NLS_CP_ALGORITHM_RANGE));
  1404. if (IsDLL)
  1405. {
  1406. //
  1407. // Try loading the codepage DLL
  1408. //
  1409. ULONG _rc = LoadCodePageAsDLL(CodePage, &pfnCPProc);
  1410. if (_rc)
  1411. {
  1412. if (ERROR_INVALID_PARAMETER == _rc)
  1413. {
  1414. //
  1415. // Not a valid DLL, try loading it as a normal data file
  1416. //
  1417. IsDLL = FALSE;
  1418. }
  1419. else
  1420. {
  1421. //
  1422. // Failed to load the DLL or can't find the function entry
  1423. // Return the error code
  1424. //
  1425. return (rc);
  1426. }
  1427. }
  1428. }
  1429. if (!IsDLL)
  1430. {
  1431. //
  1432. // Open and Map a view of the section.
  1433. //
  1434. GET_CP_SECTION_NAME(CodePage, wszSecName, MAX_SMALL_BUF_LEN, &ObSecName);
  1435. rc = OpenSection( &hSec,
  1436. &ObSecName,
  1437. (PVOID *)&pBaseAddr,
  1438. SECTION_MAP_READ,
  1439. TRUE );
  1440. if (!NT_SUCCESS(rc))
  1441. {
  1442. //
  1443. // Open failed, so try to create the section.
  1444. // If the creation is successful, the section will be mapped
  1445. // to the current process.
  1446. //
  1447. rc = CsrBasepNlsCreateSection(NLS_CREATE_CODEPAGE_SECTION, CodePage,
  1448. &hSec );
  1449. if (NT_SUCCESS(rc))
  1450. {
  1451. rc = MapSection( hSec,
  1452. &pBaseAddr,
  1453. PAGE_READONLY,
  1454. TRUE ); // Close handle, even if fail
  1455. }
  1456. if (!NT_SUCCESS(rc))
  1457. {
  1458. //
  1459. // Allow the default ACP and default OEMCP to work if
  1460. // it's only the registry that is corrupt. If there is
  1461. // still an error, return the error code that was returned
  1462. // from the OpenSection call.
  1463. //
  1464. if (CodePage == NLS_DEFAULT_ACP)
  1465. {
  1466. //
  1467. // Create the default ACP section.
  1468. //
  1469. if (!NT_SUCCESS(CsrBasepNlsCreateSection(NLS_CREATE_SECTION_DEFAULT_ACP, 0,
  1470. &hSec )))
  1471. {
  1472. return (rc);
  1473. }
  1474. else
  1475. {
  1476. //
  1477. // Map the section.
  1478. //
  1479. if (!NT_SUCCESS(MapSection( hSec,
  1480. (PVOID *)&pBaseAddr,
  1481. PAGE_READONLY,
  1482. TRUE )))
  1483. {
  1484. return (rc);
  1485. }
  1486. KdPrint(("NLSAPI: Registry is corrupt - Default ACP.\n"));
  1487. }
  1488. }
  1489. else if (CodePage == NLS_DEFAULT_OEMCP)
  1490. {
  1491. //
  1492. // Create the default OEMCP section.
  1493. //
  1494. if (!NT_SUCCESS(CsrBasepNlsCreateSection( NLS_CREATE_SECTION_DEFAULT_OEMCP, 0,
  1495. &hSec )))
  1496. {
  1497. return (rc);
  1498. }
  1499. else
  1500. {
  1501. //
  1502. // Map the section.
  1503. //
  1504. if (!NT_SUCCESS(MapSection( hSec,
  1505. (PVOID *)&pBaseAddr,
  1506. PAGE_READONLY,
  1507. TRUE )))
  1508. {
  1509. return (rc);
  1510. }
  1511. KdPrint(("NLSAPI: Registry is corrupt - Default OEMCP.\n"));
  1512. }
  1513. }
  1514. else
  1515. {
  1516. //
  1517. // Return the error code that was returned from the
  1518. // OpenSection call.
  1519. //
  1520. return (rc);
  1521. }
  1522. }
  1523. }
  1524. }
  1525. //
  1526. // Make the hash node and return the result.
  1527. //
  1528. return (MakeCPHashNode( CodePage,
  1529. pBaseAddr,
  1530. ppNode,
  1531. IsDLL,
  1532. pfnCPProc ));
  1533. }
  1534. ////////////////////////////////////////////////////////////////////////////
  1535. //
  1536. // GetLanguageFileInfo
  1537. //
  1538. // Opens and Maps a view of the section for the casing tables and sorting
  1539. // tables for the given locale.
  1540. //
  1541. // If the section cannot be opened, it then queries the registry to see if
  1542. // the information has been added since the initialization of the DLL. If
  1543. // so, then it creates the section and then opens and maps a view of it.
  1544. //
  1545. // 05-31-91 JulieB Created.
  1546. ////////////////////////////////////////////////////////////////////////////
  1547. ULONG GetLanguageFileInfo(
  1548. LCID Locale,
  1549. PLOC_HASH *ppNode,
  1550. BOOLEAN fCreateNode,
  1551. DWORD dwFlags)
  1552. {
  1553. LPWORD pBaseAddr = NULL; // ptr to base address of section
  1554. MEMORY_BASIC_INFORMATION MemoryBasicInfo;
  1555. //
  1556. // Make sure we're in the critical section when entering this call.
  1557. //
  1558. ASSERT(NtCurrentTeb()->ClientId.UniqueThread == gcsTblPtrs.OwningThread);
  1559. //
  1560. // See if the default language table has been stored yet.
  1561. //
  1562. if (pTblPtrs->pDefaultLanguage == NULL)
  1563. {
  1564. //
  1565. // Save the default language table and its size in the
  1566. // table pointers structure.
  1567. //
  1568. pTblPtrs->pDefaultLanguage = NtCurrentPeb()->UnicodeCaseTableData;
  1569. NtQueryVirtualMemory( NtCurrentProcess(),
  1570. pTblPtrs->pDefaultLanguage,
  1571. MemoryBasicInformation,
  1572. &MemoryBasicInfo,
  1573. sizeof(MEMORY_BASIC_INFORMATION),
  1574. NULL );
  1575. pTblPtrs->LinguistLangSize.QuadPart = MemoryBasicInfo.RegionSize;
  1576. ASSERT(MemoryBasicInfo.RegionSize > 0);
  1577. }
  1578. //
  1579. // See if we should load the culturally correct language table.
  1580. //
  1581. if (dwFlags)
  1582. {
  1583. if (pTblPtrs->pLangException == NULL)
  1584. {
  1585. GetLanguageExceptionInfo();
  1586. }
  1587. //
  1588. // Get the default linguistic language table for the given locale.
  1589. //
  1590. pBaseAddr = GetLinguisticLanguageInfo(Locale);
  1591. }
  1592. //
  1593. // Get the casing table and sorting table pointers.
  1594. //
  1595. return (MakeLangHashNode( Locale,
  1596. pBaseAddr,
  1597. ppNode,
  1598. fCreateNode ));
  1599. }
  1600. ////////////////////////////////////////////////////////////////////////////
  1601. //
  1602. // GetLocaleFileInfo
  1603. //
  1604. // Opens and Maps a view of the section for the given locale. It then
  1605. // creates and inserts a hash node into the global LOCALE hash table.
  1606. //
  1607. // If the section cannot be opened, it then queries the registry to see if
  1608. // the information has been added since the initialization of the DLL. If
  1609. // so, then it creates the section and then opens and maps a view of it.
  1610. //
  1611. // 05-31-91 JulieB Created.
  1612. ////////////////////////////////////////////////////////////////////////////
  1613. ULONG GetLocaleFileInfo(
  1614. LCID Locale,
  1615. PLOC_HASH *ppNode,
  1616. BOOLEAN fCreateNode)
  1617. {
  1618. HANDLE hSec = (HANDLE)0; // section handle
  1619. UNICODE_STRING ObSecName; // section name
  1620. LPWORD pBaseAddr; // ptr to base address of section
  1621. ULONG rc = 0L; // return code
  1622. //
  1623. // Make sure we're in the critical section when entering this call.
  1624. //
  1625. ASSERT(NtCurrentTeb()->ClientId.UniqueThread == gcsTblPtrs.OwningThread);
  1626. //
  1627. // Open and Map a view of the section if it hasn't been done yet.
  1628. //
  1629. if ((pBaseAddr = pTblPtrs->pLocaleInfo) == NULL)
  1630. {
  1631. //
  1632. // Get the locale file section pointer.
  1633. //
  1634. RtlInitUnicodeString(&ObSecName, NLS_SECTION_LOCALE);
  1635. if (rc = OpenSection( &hSec,
  1636. &ObSecName,
  1637. (PVOID *)&pBaseAddr,
  1638. SECTION_MAP_READ,
  1639. TRUE ))
  1640. {
  1641. return (rc);
  1642. }
  1643. //
  1644. // Store pointer to locale file and calendar info in table
  1645. // structure.
  1646. //
  1647. pTblPtrs->pLocaleInfo = pBaseAddr;
  1648. pTblPtrs->NumCalendars = ((PLOC_CAL_HDR)pBaseAddr)->NumCalendars;
  1649. pTblPtrs->pCalendarInfo = pBaseAddr +
  1650. ((PLOC_CAL_HDR)pBaseAddr)->CalOffset;
  1651. }
  1652. //
  1653. // Make the hash node and return the result.
  1654. //
  1655. return (MakeLocHashNode( Locale,
  1656. pBaseAddr,
  1657. ppNode,
  1658. fCreateNode ));
  1659. }
  1660. ////////////////////////////////////////////////////////////////////////////
  1661. //
  1662. // MakeCPHashNode
  1663. //
  1664. // Creates the hash node for the code page and assigns the fields of the
  1665. // hash node to point at the appropriate places in the file.
  1666. //
  1667. // 05-31-91 JulieB Created.
  1668. ////////////////////////////////////////////////////////////////////////////
  1669. ULONG MakeCPHashNode(
  1670. UINT CodePage,
  1671. LPWORD pBaseAddr,
  1672. PCP_HASH *ppNode,
  1673. BOOL IsDLL,
  1674. LPFN_CP_PROC pfnCPProc)
  1675. {
  1676. PCP_HASH pHashN; // ptr to CP hash node
  1677. WORD offMB; // offset to MB table
  1678. WORD offWC; // offset to WC table
  1679. PGLYPH_TABLE pGlyph; // ptr to glyph table info
  1680. PDBCS_RANGE pRange; // ptr to DBCS range
  1681. //
  1682. // Allocate CP_HASH structure and fill in the CodePage value.
  1683. //
  1684. CREATE_CODEPAGE_HASH_NODE(CodePage, pHashN);
  1685. //
  1686. // See if we're dealing with a DLL or an NLS data file.
  1687. //
  1688. if (IsDLL)
  1689. {
  1690. if (pfnCPProc == NULL)
  1691. {
  1692. NLS_FREE_MEM(pHashN);
  1693. return (ERROR_INVALID_PARAMETER);
  1694. }
  1695. pHashN->pfnCPProc = pfnCPProc;
  1696. }
  1697. else
  1698. {
  1699. //
  1700. // Get the offsets.
  1701. //
  1702. offMB = pBaseAddr[0];
  1703. offWC = offMB + pBaseAddr[offMB];
  1704. //
  1705. // Attach CP Info to CP hash node.
  1706. //
  1707. pHashN->pCPInfo = (PCP_TABLE)(pBaseAddr + CP_HEADER);
  1708. //
  1709. // Attach MB table to CP hash node.
  1710. //
  1711. pHashN->pMBTbl = pBaseAddr + offMB + MB_HEADER;
  1712. //
  1713. // Attach Glyph table to CP hash node (if it exists).
  1714. // Also, set the pointer to the DBCS ranges based on whether or
  1715. // not the GLYPH table is present.
  1716. //
  1717. pGlyph = pHashN->pMBTbl + MB_TBL_SIZE;
  1718. if (pGlyph[0] != 0)
  1719. {
  1720. pHashN->pGlyphTbl = pGlyph + GLYPH_HEADER;
  1721. pRange = pHashN->pDBCSRanges = pHashN->pGlyphTbl + GLYPH_TBL_SIZE;
  1722. }
  1723. else
  1724. {
  1725. pRange = pHashN->pDBCSRanges = pGlyph + GLYPH_HEADER;
  1726. }
  1727. //
  1728. // Attach DBCS information to CP hash node.
  1729. //
  1730. if (pRange[0] > 0)
  1731. {
  1732. //
  1733. // Set the pointer to the offsets section.
  1734. //
  1735. pHashN->pDBCSOffsets = pRange + DBCS_HEADER;
  1736. }
  1737. //
  1738. // Attach WC table to CP hash node.
  1739. //
  1740. pHashN->pWC = pBaseAddr + offWC + WC_HEADER;
  1741. }
  1742. //
  1743. // Insert hash node into hash table.
  1744. //
  1745. INSERT_CP_HASH_NODE(pHashN, pBaseAddr);
  1746. //
  1747. // Save the pointer to the hash node.
  1748. //
  1749. if (ppNode != NULL)
  1750. {
  1751. *ppNode = pHashN;
  1752. }
  1753. //
  1754. // Return success.
  1755. //
  1756. return (NO_ERROR);
  1757. }
  1758. ////////////////////////////////////////////////////////////////////////////
  1759. //
  1760. // MakeLangHashNode
  1761. //
  1762. // Gets the pointers to the casing tables and the sorting tables and
  1763. // stores them in the locale hash node given.
  1764. //
  1765. // If fCreateNode is FALSE, then *ppNode should contain a valid pointer
  1766. // to a LOC hash node. Also, the table critical section must be entered
  1767. // before calling this routine.
  1768. //
  1769. // 05-31-91 JulieB Created.
  1770. ////////////////////////////////////////////////////////////////////////////
  1771. ULONG MakeLangHashNode(
  1772. LCID Locale,
  1773. LPWORD pBaseAddr,
  1774. PLOC_HASH *ppNode,
  1775. BOOLEAN fCreateNode)
  1776. {
  1777. LPWORD pBaseDefault; // ptr to default language section
  1778. PLOC_HASH pHashN; // ptr to LOC hash node
  1779. ULONG rc = 0L; // return code
  1780. //
  1781. // If fCreateNode is TRUE, then allocate LOC_HASH structure.
  1782. //
  1783. if (fCreateNode)
  1784. {
  1785. CREATE_LOCALE_HASH_NODE(Locale, pHashN);
  1786. }
  1787. else
  1788. {
  1789. pHashN = *ppNode;
  1790. }
  1791. //
  1792. // See if the sorting tables still need to be attached.
  1793. //
  1794. if (pHashN->pSortkey == NULL)
  1795. {
  1796. //
  1797. // Get the sortkey table and attach it to the hash node.
  1798. //
  1799. if (rc = GetSortkeyFileInfo(Locale, pHashN))
  1800. {
  1801. if (fCreateNode)
  1802. {
  1803. NLS_FREE_MEM(pHashN);
  1804. }
  1805. return (rc);
  1806. }
  1807. //
  1808. // Get the appropriate sorting tables for the locale.
  1809. //
  1810. GetSortTablesFileInfo(Locale, pHashN);
  1811. }
  1812. //
  1813. // See if the default casing tables still need to be attached.
  1814. //
  1815. if (!EXIST_LANGUAGE_INFO(pHashN))
  1816. {
  1817. //
  1818. // Get the pointer to the base of the default table.
  1819. //
  1820. pBaseDefault = pTblPtrs->pDefaultLanguage;
  1821. //
  1822. // Attach the UPPERCASE table to the hash node.
  1823. //
  1824. pHashN->pUpperCase = pBaseDefault + LANG_HEADER + UP_HEADER;
  1825. //
  1826. // Attach the LOWERCASE table to the hash node.
  1827. //
  1828. // This value must be set LAST, since this is the pointer that
  1829. // is checked to see that the language information has been
  1830. // initialized.
  1831. //
  1832. pHashN->pLowerCase = pBaseDefault + LANG_HEADER +
  1833. pBaseDefault[LANG_HEADER] + LO_HEADER;
  1834. }
  1835. //
  1836. // See if there is a linguistic table to attach.
  1837. //
  1838. if (pBaseAddr)
  1839. {
  1840. //
  1841. // Attach the UPPERCASE Linguistic table to the hash node.
  1842. //
  1843. pHashN->pUpperLinguist = pBaseAddr + LANG_HEADER + UP_HEADER;
  1844. //
  1845. // Attach the LOWERCASE Linguistic table to the hash node.
  1846. //
  1847. // This value must be set LAST, since this is the pointer that
  1848. // is checked to see that the language information has been
  1849. // initialized.
  1850. //
  1851. pHashN->pLowerLinguist = pBaseAddr + LANG_HEADER +
  1852. pBaseAddr[LANG_HEADER] + LO_HEADER;
  1853. }
  1854. //
  1855. // If fCreateNode is TRUE, then insert hash node and save pointer.
  1856. //
  1857. if (fCreateNode)
  1858. {
  1859. //
  1860. // Insert LOC hash node into hash table.
  1861. //
  1862. INSERT_LOC_HASH_NODE(pHashN, pBaseAddr);
  1863. //
  1864. // Save the pointer to the hash node.
  1865. //
  1866. if (ppNode != NULL)
  1867. {
  1868. *ppNode = pHashN;
  1869. }
  1870. }
  1871. //
  1872. // Return success.
  1873. //
  1874. return (NO_ERROR);
  1875. }
  1876. ////////////////////////////////////////////////////////////////////////////
  1877. //
  1878. // MakeLocHashNode
  1879. //
  1880. // Gets the pointers to the locale tables and stores them in the locale
  1881. // hash node given.
  1882. //
  1883. // NOTE: If a critical section is needed to touch pHashN, then the
  1884. // critical section must be entered before calling this routine.
  1885. //
  1886. // 05-31-91 JulieB Created.
  1887. ////////////////////////////////////////////////////////////////////////////
  1888. ULONG MakeLocHashNode(
  1889. LCID Locale,
  1890. LPWORD pBaseAddr,
  1891. PLOC_HASH *ppNode,
  1892. BOOLEAN fCreateNode)
  1893. {
  1894. LANGID Language; // language id
  1895. PLOC_HASH pHashN; // ptr to LOC hash node
  1896. DWORD Num; // total number of locales
  1897. PLOCALE_HDR pFileHdr; // ptr to locale header entry
  1898. ULONG rc = 0L; // return code
  1899. //
  1900. // Save the language id.
  1901. //
  1902. Language = LANGIDFROMLCID(Locale);
  1903. //
  1904. // Search for the right locale id information.
  1905. //
  1906. Num = ((PLOC_CAL_HDR)pBaseAddr)->NumLocales;
  1907. pFileHdr = (PLOCALE_HDR)(pBaseAddr + LOCALE_HDR_OFFSET);
  1908. for (; (Num != 0) && (pFileHdr->Locale != Language); Num--, pFileHdr++)
  1909. ;
  1910. //
  1911. // See if the locale was found in the file.
  1912. //
  1913. if (Num != 0)
  1914. {
  1915. //
  1916. // Locale id was found, so increment the pointer to point at
  1917. // the beginning of the locale information.
  1918. //
  1919. pBaseAddr += pFileHdr->Offset;
  1920. }
  1921. else
  1922. {
  1923. //
  1924. // Return an error. The given locale is not supported.
  1925. //
  1926. return (ERROR_INVALID_PARAMETER);
  1927. }
  1928. //
  1929. // If fCreateNode is TRUE, then allocate LOC_HASH structure.
  1930. //
  1931. if (fCreateNode)
  1932. {
  1933. CREATE_LOCALE_HASH_NODE(Locale, pHashN);
  1934. }
  1935. else
  1936. {
  1937. pHashN = *ppNode;
  1938. }
  1939. //
  1940. // Attach Information to structure.
  1941. //
  1942. // The pLocaleFixed value must be set LAST, since this is the pointer
  1943. // that is checked to see that the locale information has been
  1944. // initialized.
  1945. //
  1946. pHashN->pLocaleHdr = (PLOCALE_VAR)pBaseAddr;
  1947. pHashN->pLocaleFixed = (PLOCALE_FIXED)(pBaseAddr +
  1948. (sizeof(LOCALE_VAR) / sizeof(WORD)));
  1949. //
  1950. // If fCreateNode is TRUE, then insert hash node and save pointer.
  1951. //
  1952. if (fCreateNode)
  1953. {
  1954. //
  1955. // Insert LOC hash node into hash table.
  1956. //
  1957. INSERT_LOC_HASH_NODE(pHashN, pBaseAddr);
  1958. //
  1959. // Save the pointer to the hash node.
  1960. //
  1961. if (ppNode != NULL)
  1962. {
  1963. *ppNode = pHashN;
  1964. }
  1965. }
  1966. //
  1967. // Return success.
  1968. //
  1969. return (NO_ERROR);
  1970. }
  1971. ////////////////////////////////////////////////////////////////////////////
  1972. //
  1973. // GetCPHashNode
  1974. //
  1975. // Returns a pointer to the appropriate CP hash node given the codepage.
  1976. // If no table could be found for the given codepage, NULL is returned.
  1977. //
  1978. // 05-31-91 JulieB Created.
  1979. ////////////////////////////////////////////////////////////////////////////
  1980. PCP_HASH FASTCALL GetCPHashNode(
  1981. UINT CodePage)
  1982. {
  1983. PCP_HASH pHashN; // ptr to CP hash node
  1984. //
  1985. // Get hash node.
  1986. //
  1987. FIND_CP_HASH_NODE(CodePage, pHashN);
  1988. //
  1989. // If the hash node does not exist, try to get the tables
  1990. // from the appropriate data file.
  1991. //
  1992. // NOTE: No need to check error code from GetCodePageFileInfo,
  1993. // because pHashN is not touched if there was an
  1994. // error. Thus, pHashN will still be NULL, and an
  1995. // "error" will be returned from this routine.
  1996. //
  1997. if (pHashN == NULL)
  1998. {
  1999. //
  2000. // Hash node does NOT exist.
  2001. //
  2002. RtlEnterCriticalSection(&gcsTblPtrs);
  2003. FIND_CP_HASH_NODE(CodePage, pHashN);
  2004. if (pHashN == NULL)
  2005. {
  2006. GetCodePageFileInfo(CodePage, &pHashN);
  2007. }
  2008. RtlLeaveCriticalSection(&gcsTblPtrs);
  2009. }
  2010. //
  2011. // Return pointer to hash node.
  2012. //
  2013. return (pHashN);
  2014. }
  2015. ////////////////////////////////////////////////////////////////////////////
  2016. //
  2017. // GetLangHashNode
  2018. //
  2019. // Returns a pointer to the appropriate LOC hash node given the locale.
  2020. // If no table could be found for the given locale, NULL is returned.
  2021. //
  2022. // 05-31-91 JulieB Created.
  2023. ////////////////////////////////////////////////////////////////////////////
  2024. PLOC_HASH FASTCALL GetLangHashNode(
  2025. LCID Locale,
  2026. DWORD dwFlags)
  2027. {
  2028. PLOC_HASH pHashN; // ptr to LOC hash node
  2029. //
  2030. // Get hash node.
  2031. //
  2032. FIND_LOCALE_HASH_NODE(Locale, pHashN);
  2033. //
  2034. // If the hash node does not exist, try to get the tables
  2035. // from the appropriate data file.
  2036. //
  2037. // NOTE: No need to check error code from GetLanguageFileInfo,
  2038. // because pHashN is not touched if there was an
  2039. // error. Thus, pHashN will still be NULL, and an
  2040. // "error" will be returned from this routine.
  2041. //
  2042. if (pHashN == NULL)
  2043. {
  2044. //
  2045. // If a sort id exists, make sure it's valid.
  2046. //
  2047. if (SORTIDFROMLCID(Locale))
  2048. {
  2049. if (!IsValidSortId(Locale))
  2050. {
  2051. return (NULL);
  2052. }
  2053. }
  2054. //
  2055. // Hash node does NOT exist.
  2056. //
  2057. RtlEnterCriticalSection(&gcsTblPtrs);
  2058. FIND_LOCALE_HASH_NODE(Locale, pHashN);
  2059. if (pHashN == NULL)
  2060. {
  2061. //
  2062. // Hash node still does NOT exist.
  2063. //
  2064. GetLanguageFileInfo(Locale, &pHashN, TRUE, dwFlags);
  2065. RtlLeaveCriticalSection(&gcsTblPtrs);
  2066. return (pHashN);
  2067. }
  2068. RtlLeaveCriticalSection(&gcsTblPtrs);
  2069. }
  2070. //
  2071. // Hash node DOES exist.
  2072. //
  2073. if (!EXIST_LANGUAGE_INFO(pHashN) ||
  2074. ((dwFlags != 0) && !EXIST_LINGUIST_LANGUAGE_INFO(pHashN)))
  2075. {
  2076. //
  2077. // Casing tables and sorting tables not yet stored in
  2078. // hash node.
  2079. //
  2080. RtlEnterCriticalSection(&gcsTblPtrs);
  2081. if (!EXIST_LANGUAGE_INFO(pHashN) ||
  2082. ((dwFlags != 0) && !EXIST_LINGUIST_LANGUAGE_INFO(pHashN)))
  2083. {
  2084. if (GetLanguageFileInfo(Locale, &pHashN, FALSE, dwFlags))
  2085. {
  2086. RtlLeaveCriticalSection(&gcsTblPtrs);
  2087. return (NULL);
  2088. }
  2089. }
  2090. RtlLeaveCriticalSection(&gcsTblPtrs);
  2091. }
  2092. //
  2093. // Return pointer to hash node.
  2094. //
  2095. return (pHashN);
  2096. }
  2097. ////////////////////////////////////////////////////////////////////////////
  2098. //
  2099. // GetLocHashNode
  2100. //
  2101. // Returns a pointer to the appropriate LOC hash node given the locale.
  2102. // If no table could be found for the given locale, NULL is returned.
  2103. //
  2104. // 05-31-91 JulieB Created.
  2105. ////////////////////////////////////////////////////////////////////////////
  2106. PLOC_HASH FASTCALL GetLocHashNode(
  2107. LCID Locale)
  2108. {
  2109. PLOC_HASH pHashN; // ptr to LOC hash node
  2110. //
  2111. // Get hash node.
  2112. //
  2113. FIND_LOCALE_HASH_NODE(Locale, pHashN);
  2114. //
  2115. // If the hash node does not exist, try to get the table
  2116. // from locale.nls.
  2117. //
  2118. // NOTE: No need to check error code from GetLocaleFileInfo,
  2119. // because pHashN is not touched if there was an
  2120. // error. Thus, pHashN will still be NULL, and an
  2121. // "error" will be returned from this routine.
  2122. //
  2123. if (pHashN == NULL)
  2124. {
  2125. //
  2126. // If a sort id exists, make sure it's valid.
  2127. //
  2128. if (SORTIDFROMLCID(Locale))
  2129. {
  2130. if (!IsValidSortId(Locale))
  2131. {
  2132. return (NULL);
  2133. }
  2134. }
  2135. //
  2136. // Hash node does NOT exist.
  2137. //
  2138. RtlEnterCriticalSection(&gcsTblPtrs);
  2139. FIND_LOCALE_HASH_NODE(Locale, pHashN);
  2140. if (pHashN == NULL)
  2141. {
  2142. //
  2143. // Hash node still does NOT exist.
  2144. //
  2145. GetLocaleFileInfo(Locale, &pHashN, TRUE);
  2146. RtlLeaveCriticalSection(&gcsTblPtrs);
  2147. return (pHashN);
  2148. }
  2149. RtlLeaveCriticalSection(&gcsTblPtrs);
  2150. }
  2151. //
  2152. // Hash node DOES exist.
  2153. //
  2154. if (!EXIST_LOCALE_INFO(pHashN))
  2155. {
  2156. //
  2157. // Locale tables not yet stored in hash node.
  2158. //
  2159. RtlEnterCriticalSection(&gcsTblPtrs);
  2160. if (!EXIST_LOCALE_INFO(pHashN))
  2161. {
  2162. if (GetLocaleFileInfo(Locale, &pHashN, FALSE))
  2163. {
  2164. RtlLeaveCriticalSection(&gcsTblPtrs);
  2165. return (NULL);
  2166. }
  2167. }
  2168. RtlLeaveCriticalSection(&gcsTblPtrs);
  2169. }
  2170. //
  2171. // Return pointer to hash node.
  2172. //
  2173. return (pHashN);
  2174. }
  2175. ////////////////////////////////////////////////////////////////////////////
  2176. //
  2177. // GetCalendar
  2178. //
  2179. // Gets the pointer to the specific calendar table. It stores it in the
  2180. // calendar information array in the global table pointers structure if it
  2181. // was not done yet.
  2182. //
  2183. // 05-31-91 JulieB Created.
  2184. ////////////////////////////////////////////////////////////////////////////
  2185. ULONG GetCalendar(
  2186. CALID Calendar,
  2187. PCAL_INFO *ppCalInfo)
  2188. {
  2189. PCALENDAR_HDR pCalHdr; // ptr to beginning of calendar header
  2190. DWORD Num; // total number of calendars
  2191. //
  2192. // Get number of calendars.
  2193. //
  2194. Num = pTblPtrs->NumCalendars;
  2195. //
  2196. // Make sure calendar id is within the appropriate range.
  2197. //
  2198. if (Calendar > Num)
  2199. {
  2200. return (ERROR_INVALID_PARAMETER);
  2201. }
  2202. //
  2203. // Check to see if calendar info has already been found.
  2204. //
  2205. if ((*ppCalInfo = (pTblPtrs->pCalTbl)[Calendar]) != NULL)
  2206. {
  2207. //
  2208. // Return success. Calendar info was found.
  2209. //
  2210. return (NO_ERROR);
  2211. }
  2212. RtlEnterCriticalSection(&gcsTblPtrs);
  2213. if ((*ppCalInfo = (pTblPtrs->pCalTbl)[Calendar]) != NULL)
  2214. {
  2215. //
  2216. // Return success. Calendar info was found.
  2217. //
  2218. RtlLeaveCriticalSection(&gcsTblPtrs);
  2219. return (NO_ERROR);
  2220. }
  2221. //
  2222. // Search for the appropriate calendar id information.
  2223. //
  2224. pCalHdr = (PCALENDAR_HDR)(pTblPtrs->pCalendarInfo);
  2225. while ((Num != 0) && (pCalHdr->Calendar != Calendar))
  2226. {
  2227. Num--;
  2228. pCalHdr++;
  2229. }
  2230. //
  2231. // See if the calendar was found in the file.
  2232. //
  2233. if (Num != 0)
  2234. {
  2235. //
  2236. // Calendar id was found.
  2237. //
  2238. // Store the pointer to the beginning of the calendar info
  2239. // in the calendar table array.
  2240. //
  2241. *ppCalInfo = (PCAL_INFO)((LPWORD)(pTblPtrs->pCalendarInfo) +
  2242. pCalHdr->Offset);
  2243. (pTblPtrs->pCalTbl)[Calendar] = *ppCalInfo;
  2244. //
  2245. // Return success. Calendar info was found.
  2246. //
  2247. RtlLeaveCriticalSection(&gcsTblPtrs);
  2248. return (NO_ERROR);
  2249. }
  2250. RtlLeaveCriticalSection(&gcsTblPtrs);
  2251. //
  2252. // Calendar id was not found in the locale.nls file.
  2253. // Return an error. The given calendar is not supported.
  2254. //
  2255. return (ERROR_INVALID_PARAMETER);
  2256. }
  2257. //-------------------------------------------------------------------------//
  2258. // INTERNAL ROUTINES //
  2259. //-------------------------------------------------------------------------//
  2260. ////////////////////////////////////////////////////////////////////////////
  2261. //
  2262. // IsValidSortId
  2263. //
  2264. // Checks to see if the given locale has a valid sort id.
  2265. //
  2266. // 11-15-96 JulieB Created.
  2267. ////////////////////////////////////////////////////////////////////////////
  2268. BOOL IsValidSortId(
  2269. LCID Locale)
  2270. {
  2271. WCHAR pTmpBuf[MAX_PATH]; // temp buffer
  2272. PKEY_VALUE_FULL_INFORMATION pKeyValueFull; // ptr to query info
  2273. BYTE pStatic[MAX_KEY_VALUE_FULLINFO]; // ptr to static buffer
  2274. BOOL IfAlloc = FALSE; // if buffer was allocated
  2275. ULONG rc = 0L; // return code
  2276. //
  2277. // Make sure there is a sort id.
  2278. //
  2279. if (!SORTIDFROMLCID(Locale))
  2280. {
  2281. return (TRUE);
  2282. }
  2283. //
  2284. // Open the Alternate Sorts registry key.
  2285. //
  2286. OPEN_ALT_SORTS_KEY(FALSE);
  2287. //
  2288. // Convert locale value to Unicode string.
  2289. //
  2290. if (NlsConvertIntegerToString(Locale, 16, 8, pTmpBuf, MAX_PATH))
  2291. {
  2292. return (FALSE);
  2293. }
  2294. //
  2295. // Query the registry for the value.
  2296. //
  2297. pKeyValueFull = (PKEY_VALUE_FULL_INFORMATION)pStatic;
  2298. if (rc = QueryRegValue( hAltSortsKey,
  2299. pTmpBuf,
  2300. &pKeyValueFull,
  2301. MAX_KEY_VALUE_FULLINFO,
  2302. &IfAlloc ))
  2303. {
  2304. return (FALSE);
  2305. }
  2306. //
  2307. // Free the buffer used for the query.
  2308. //
  2309. if (IfAlloc)
  2310. {
  2311. NLS_FREE_MEM(pKeyValueFull);
  2312. }
  2313. //
  2314. // Return success.
  2315. //
  2316. return (TRUE);
  2317. }
  2318. ////////////////////////////////////////////////////////////////////////////
  2319. //
  2320. // GetLanguageExceptionInfo
  2321. //
  2322. // Opens and Maps a view of the section for the language exception file.
  2323. // It then fills in the appropriate fields of the global table pointers
  2324. // structure.
  2325. //
  2326. // 08-30-95 JulieB Created.
  2327. ////////////////////////////////////////////////////////////////////////////
  2328. ULONG GetLanguageExceptionInfo()
  2329. {
  2330. HANDLE hSec = (HANDLE)0; // section handle
  2331. LPWORD pBaseAddr; // ptr to base address of section
  2332. DWORD Num; // number of entries in table
  2333. ULONG rc = 0L; // return code
  2334. //
  2335. // Make sure we're in the critical section when entering this call.
  2336. //
  2337. ASSERT(NtCurrentTeb()->ClientId.UniqueThread == gcsTblPtrs.OwningThread);
  2338. //
  2339. // Make sure the table isn't already there.
  2340. //
  2341. if (pTblPtrs->pLangException != NULL)
  2342. {
  2343. return (NO_ERROR);
  2344. }
  2345. //
  2346. // Create and map the section, and then save the pointer.
  2347. //
  2348. if ((rc = CsrBasepNlsCreateSection( NLS_CREATE_SECTION_LANG_EXCEPT, 0,
  2349. &hSec )) == NO_ERROR)
  2350. {
  2351. //
  2352. // Map a View of the Section.
  2353. //
  2354. if ((rc = MapSection( hSec,
  2355. &pBaseAddr,
  2356. PAGE_READONLY,
  2357. TRUE )) != NO_ERROR)
  2358. {
  2359. return (rc);
  2360. }
  2361. }
  2362. else
  2363. {
  2364. return (rc);
  2365. }
  2366. //
  2367. // Save the pointers to the exception information.
  2368. //
  2369. Num = *((LPDWORD)pBaseAddr);
  2370. if (Num > 0)
  2371. {
  2372. pTblPtrs->NumLangException = Num;
  2373. pTblPtrs->pLangExceptHdr = (PL_EXCEPT_HDR)(pBaseAddr +
  2374. L_EXCEPT_HDR_OFFSET);
  2375. pTblPtrs->pLangException = (PL_EXCEPT)(pBaseAddr +
  2376. L_EXCEPT_HDR_OFFSET +
  2377. (Num * (sizeof(L_EXCEPT_HDR) /
  2378. sizeof(WORD))));
  2379. }
  2380. //
  2381. // Return success.
  2382. //
  2383. return (NO_ERROR);
  2384. }
  2385. ////////////////////////////////////////////////////////////////////////////
  2386. //
  2387. // GetLinguisticLanguageInfo
  2388. //
  2389. // Opens and Maps a view of the section for the default linguistic language
  2390. // table. It then stores the pointer to the table in the global pointer
  2391. // table.
  2392. //
  2393. // 08-30-95 JulieB Created.
  2394. ////////////////////////////////////////////////////////////////////////////
  2395. LPWORD GetLinguisticLanguageInfo(
  2396. LCID Locale)
  2397. {
  2398. HANDLE hSec = (HANDLE)0; // section handle
  2399. UNICODE_STRING ObSecName; // section name
  2400. LPWORD pBaseAddr; // ptr to base address of section
  2401. ULONG rc = 0L; // return code
  2402. SECTION_BASIC_INFORMATION SecInfo; // section information - query
  2403. //
  2404. // Make sure we're in the critical section when entering this call.
  2405. //
  2406. ASSERT(NtCurrentTeb()->ClientId.UniqueThread == gcsTblPtrs.OwningThread);
  2407. //
  2408. // Create/Open and Map a view of the section if it hasn't been done yet.
  2409. //
  2410. if (pTblPtrs->pLinguistLanguage == NULL)
  2411. {
  2412. //
  2413. // See if we can simply open the section.
  2414. //
  2415. RtlInitUnicodeString(&ObSecName, NLS_SECTION_LANG_INTL);
  2416. if (rc = OpenSection( &hSec,
  2417. &ObSecName,
  2418. (PVOID *)&pBaseAddr,
  2419. SECTION_MAP_READ,
  2420. TRUE ))
  2421. {
  2422. //
  2423. // Need to create the default linguistic language section.
  2424. //
  2425. if (CreateAndCopyLanguageExceptions(0L, &pBaseAddr))
  2426. {
  2427. return (NULL);
  2428. }
  2429. }
  2430. //
  2431. // Get Default Linguistic Language Information.
  2432. //
  2433. pTblPtrs->pLinguistLanguage = (P844_TABLE)(pBaseAddr);
  2434. }
  2435. //
  2436. // Now see if there are any exceptions for the given locale.
  2437. //
  2438. if (CreateAndCopyLanguageExceptions(Locale, &pBaseAddr))
  2439. {
  2440. return (pTblPtrs->pLinguistLanguage);
  2441. }
  2442. //
  2443. // Return success.
  2444. //
  2445. return (pBaseAddr);
  2446. }
  2447. ////////////////////////////////////////////////////////////////////////////
  2448. //
  2449. // CreateAndCopyLanguageExceptions
  2450. //
  2451. // Creates the section for the new language table (if necessary) and then
  2452. // copies the exceptions to the table.
  2453. //
  2454. // 08-30-95 JulieB Created.
  2455. ////////////////////////////////////////////////////////////////////////////
  2456. ULONG CreateAndCopyLanguageExceptions(
  2457. LCID Locale,
  2458. LPWORD *ppBaseAddr)
  2459. {
  2460. HANDLE hSec = (HANDLE)0; // section handle
  2461. UNICODE_STRING ObSecName; // section name
  2462. LPWORD pBaseAddr; // ptr to base address of section
  2463. P844_TABLE pLangDefault; // ptr to default table to copy from
  2464. ULONG rc = 0L; // return code
  2465. PL_EXCEPT_HDR pExceptHdr; // ptr to exception header
  2466. PL_EXCEPT pExceptTbl; // ptr to exception table
  2467. WCHAR wszSecName[MAX_SMALL_BUF_LEN]; // Place for the section name string \
  2468. NTSTATUS Status;
  2469. //
  2470. // Make sure we're in the critical section when entering this call.
  2471. //
  2472. ASSERT(NtCurrentTeb()->ClientId.UniqueThread == gcsTblPtrs.OwningThread);
  2473. if (Locale == 0)
  2474. {
  2475. //
  2476. // Creating the default section.
  2477. //
  2478. RtlInitUnicodeString(&ObSecName, NLS_SECTION_LANG_INTL);
  2479. pLangDefault = pTblPtrs->pDefaultLanguage;
  2480. }
  2481. else
  2482. {
  2483. GET_LANG_SECTION_NAME(Locale, wszSecName, MAX_SMALL_BUF_LEN, &ObSecName);
  2484. pLangDefault = pTblPtrs->pLinguistLanguage;
  2485. }
  2486. //
  2487. // Try to Open and Map a view of the section (read only).
  2488. //
  2489. if (rc = OpenSection( &hSec,
  2490. &ObSecName,
  2491. (PVOID *)&pBaseAddr,
  2492. SECTION_MAP_READ,
  2493. FALSE ))
  2494. {
  2495. //
  2496. // Open failed.
  2497. // See if any exceptions exist for given Locale ID.
  2498. //
  2499. if (!FindLanguageExceptionPointers( Locale,
  2500. &pExceptHdr,
  2501. &pExceptTbl ) &&
  2502. (Locale != 0))
  2503. {
  2504. //
  2505. // No exceptions for locale and we're not trying to create
  2506. // the default table, so return the pointer to the default
  2507. // table (which should always be created at this point).
  2508. //
  2509. *ppBaseAddr = pTblPtrs->pLinguistLanguage;
  2510. return (NO_ERROR);
  2511. }
  2512. else
  2513. {
  2514. //
  2515. // Exceptions exist for the given locale. Need to create the
  2516. // new section (and call the server to make it permanent).
  2517. //
  2518. Status = CsrBasepNlsCreateSection( NLS_CREATE_LANG_EXCEPTION_SECTION,
  2519. Locale,
  2520. &hSec );
  2521. //
  2522. // Check return from server call.
  2523. //
  2524. rc = (ULONG)Status;
  2525. if (!NT_SUCCESS(rc))
  2526. {
  2527. if (hSec != NULL)
  2528. {
  2529. NtClose(hSec);
  2530. }
  2531. return (rc);
  2532. }
  2533. //
  2534. // Map the section for ReadWrite.
  2535. //
  2536. if (rc = MapSection( hSec,
  2537. (PVOID *)&pBaseAddr,
  2538. PAGE_READWRITE,
  2539. FALSE ))
  2540. {
  2541. NtClose(hSec);
  2542. return (rc);
  2543. }
  2544. //
  2545. // Put a 0 in the semaphore part to denote that the file
  2546. // is not ready yet.
  2547. //
  2548. *pBaseAddr = 0;
  2549. //
  2550. // Copy the Default Language Table to the New Section.
  2551. //
  2552. RtlMoveMemory( (PVOID)((LPWORD)pBaseAddr + LANG_HEADER),
  2553. (PVOID)((LPWORD)(pLangDefault) + LANG_HEADER),
  2554. (ULONG)(pTblPtrs->LinguistLangSize.LowPart -
  2555. (LANG_HEADER * sizeof(WORD))) );
  2556. //
  2557. // Copy exception information to the table.
  2558. //
  2559. CopyLanguageExceptionInfo( pBaseAddr,
  2560. pExceptHdr,
  2561. pExceptTbl );
  2562. //
  2563. // Write a 1 to the WORD semaphore (table may now be read).
  2564. //
  2565. *pBaseAddr = 1;
  2566. //
  2567. // Unmap the section for Write and remap it for Read.
  2568. //
  2569. if ((rc = UnMapSection(pBaseAddr)) ||
  2570. (rc = MapSection( hSec,
  2571. (PVOID *)&pBaseAddr,
  2572. PAGE_READONLY,
  2573. FALSE )))
  2574. {
  2575. NtClose(hSec);
  2576. return (rc);
  2577. }
  2578. }
  2579. }
  2580. //
  2581. // Close the section handle.
  2582. //
  2583. NtClose(hSec);
  2584. //
  2585. // Check semaphore bit in file. Make sure that the open
  2586. // succeeded AFTER all exceptions were added to the memory
  2587. // mapped section.
  2588. //
  2589. if (*pBaseAddr == 0)
  2590. {
  2591. //
  2592. // Another process is still adding the appropriate exception
  2593. // information. Must wait for its completion.
  2594. //
  2595. if (rc = WaitOnEvent(pBaseAddr))
  2596. {
  2597. UnMapSection(pBaseAddr);
  2598. return (rc);
  2599. }
  2600. }
  2601. //
  2602. // Return the pointer to the section.
  2603. //
  2604. *ppBaseAddr = pBaseAddr;
  2605. //
  2606. // Return success.
  2607. //
  2608. return (NO_ERROR);
  2609. }
  2610. ////////////////////////////////////////////////////////////////////////////
  2611. //
  2612. // FindLanguageExceptionPointers
  2613. //
  2614. // Checks to see if any exceptions exist for the given locale id. If
  2615. // exceptions exist, then TRUE is returned and the pointer to the exception
  2616. // header and the pointer to the exception table are stored in the given
  2617. // parameters.
  2618. //
  2619. // 08-30-95 JulieB Created.
  2620. ////////////////////////////////////////////////////////////////////////////
  2621. BOOL FASTCALL FindLanguageExceptionPointers(
  2622. LCID Locale,
  2623. PL_EXCEPT_HDR *ppExceptHdr,
  2624. PL_EXCEPT *ppExceptTbl)
  2625. {
  2626. DWORD ctr; // loop counter
  2627. PL_EXCEPT_HDR pHdr; // ptr to exception header
  2628. BOOL rc = FALSE; // return value
  2629. //
  2630. // Make sure we're in the critical section when entering this call.
  2631. //
  2632. ASSERT(NtCurrentTeb()->ClientId.UniqueThread == gcsTblPtrs.OwningThread);
  2633. //
  2634. // Initialize pointers.
  2635. //
  2636. *ppExceptHdr = NULL;
  2637. *ppExceptTbl = NULL;
  2638. //
  2639. // Need to search down the exception header for the given locale.
  2640. //
  2641. pHdr = pTblPtrs->pLangExceptHdr;
  2642. for (ctr = pTblPtrs->NumLangException; ctr > 0; ctr--, pHdr++)
  2643. {
  2644. if (pHdr->Locale == (DWORD)Locale)
  2645. {
  2646. //
  2647. // Found the locale id, so set the pointers.
  2648. //
  2649. *ppExceptHdr = pHdr;
  2650. *ppExceptTbl = (PL_EXCEPT)(((LPWORD)(pTblPtrs->pLangException)) +
  2651. pHdr->Offset);
  2652. //
  2653. // Set the return code for success.
  2654. //
  2655. rc = TRUE;
  2656. break;
  2657. }
  2658. }
  2659. //
  2660. // Return the value in rc.
  2661. //
  2662. return (rc);
  2663. }
  2664. ////////////////////////////////////////////////////////////////////////////
  2665. //
  2666. // CopyLanguageExceptionInfo
  2667. //
  2668. // Copies the language exception information to the given language table.
  2669. //
  2670. // 08-30-95 JulieB Created.
  2671. ////////////////////////////////////////////////////////////////////////////
  2672. void FASTCALL CopyLanguageExceptionInfo(
  2673. LPWORD pBaseAddr,
  2674. PL_EXCEPT_HDR pExceptHdr,
  2675. PL_EXCEPT pExceptTbl)
  2676. {
  2677. DWORD ctr; // loop counter
  2678. P844_TABLE pUpCase; // ptr to upper case table
  2679. P844_TABLE pLoCase; // ptr to lower case table
  2680. //
  2681. // Make sure we're in the critical section when entering this call.
  2682. //
  2683. ASSERT(NtCurrentTeb()->ClientId.UniqueThread == gcsTblPtrs.OwningThread);
  2684. if (pExceptTbl)
  2685. {
  2686. //
  2687. // Get the pointers to the upper and lower case table.
  2688. //
  2689. pUpCase = pBaseAddr + LANG_HEADER + UP_HEADER;
  2690. pLoCase = pBaseAddr + LANG_HEADER + pBaseAddr[LANG_HEADER] + LO_HEADER;
  2691. //
  2692. // For each entry in the exception table, copy the information to the
  2693. // sortkey table.
  2694. //
  2695. for (ctr = pExceptHdr->NumUpEntries; ctr > 0; ctr--, pExceptTbl++)
  2696. {
  2697. TRAVERSE_844_W(pUpCase, pExceptTbl->UCP) = pExceptTbl->AddAmount;
  2698. }
  2699. for (ctr = pExceptHdr->NumLoEntries; ctr > 0; ctr--, pExceptTbl++)
  2700. {
  2701. TRAVERSE_844_W(pLoCase, pExceptTbl->UCP) = pExceptTbl->AddAmount;
  2702. }
  2703. }
  2704. }
  2705. ////////////////////////////////////////////////////////////////////////////
  2706. //
  2707. // FindExceptionPointers
  2708. //
  2709. // Checks to see if any exceptions exist for the given locale id. If
  2710. // exceptions exist, then TRUE is returned and the pointer to the exception
  2711. // header and the pointer to the exception table are stored in the given
  2712. // parameters.
  2713. //
  2714. // 05-31-91 JulieB Created.
  2715. ////////////////////////////////////////////////////////////////////////////
  2716. BOOL FASTCALL FindExceptionPointers(
  2717. LCID Locale,
  2718. PEXCEPT_HDR *ppExceptHdr,
  2719. PEXCEPT *ppExceptTbl,
  2720. PVOID *ppIdeograph,
  2721. PULONG pReturn)
  2722. {
  2723. HANDLE hSec = (HANDLE)0; // section handle
  2724. DWORD ctr; // loop counter
  2725. PEXCEPT_HDR pHdr; // ptr to exception header
  2726. BOOL bFound = FALSE; // if an exception is found
  2727. PIDEOGRAPH_LCID pIdeoLcid; // ptr to ideograph lcid entry
  2728. PVOID pBaseAddr; // ptr to base address of section
  2729. //
  2730. // Make sure we're in the critical section when entering this call.
  2731. //
  2732. ASSERT(NtCurrentTeb()->ClientId.UniqueThread == gcsTblPtrs.OwningThread);
  2733. //
  2734. // Initialize pointers.
  2735. //
  2736. *ppExceptHdr = NULL;
  2737. *ppExceptTbl = NULL;
  2738. *ppIdeograph = NULL;
  2739. *pReturn = NO_ERROR;
  2740. //
  2741. // Need to search down the exception header for the given locale.
  2742. //
  2743. pHdr = pTblPtrs->pExceptHdr;
  2744. for (ctr = pTblPtrs->NumException; ctr > 0; ctr--, pHdr++)
  2745. {
  2746. if (pHdr->Locale == (DWORD)Locale)
  2747. {
  2748. //
  2749. // Found the locale id, so set the pointers.
  2750. //
  2751. *ppExceptHdr = pHdr;
  2752. *ppExceptTbl = (PEXCEPT)(((LPWORD)(pTblPtrs->pException)) +
  2753. pHdr->Offset);
  2754. //
  2755. // Set the return code to show that an exception has been
  2756. // found.
  2757. //
  2758. bFound = TRUE;
  2759. break;
  2760. }
  2761. }
  2762. //
  2763. // Need to search down the ideograph lcid exception list for the
  2764. // given locale.
  2765. //
  2766. pIdeoLcid = pTblPtrs->pIdeographLcid;
  2767. for (ctr = pTblPtrs->NumIdeographLcid; ctr > 0; ctr--, pIdeoLcid++)
  2768. {
  2769. if (pIdeoLcid->Locale == (DWORD)Locale)
  2770. {
  2771. //
  2772. // Found the locale id, so create/open and map the section
  2773. // for the appropriate file.
  2774. //
  2775. if (*pReturn = CreateSectionTemp(&hSec, pIdeoLcid->pFileName))
  2776. {
  2777. //
  2778. // Ideograph file section could not be created, so return
  2779. // the error.
  2780. //
  2781. return (TRUE);
  2782. }
  2783. if (*pReturn = MapSection(hSec, &pBaseAddr, PAGE_READONLY, TRUE))
  2784. {
  2785. //
  2786. // Ideograph file section could not be mapped, so close
  2787. // the created section and return the error.
  2788. //
  2789. NtClose(hSec);
  2790. return (TRUE);
  2791. }
  2792. //
  2793. // Set the pointer to the ideograph information.
  2794. //
  2795. *ppIdeograph = pBaseAddr;
  2796. //
  2797. // Set the return code to show that an exception has been
  2798. // found.
  2799. //
  2800. bFound = TRUE;
  2801. break;
  2802. }
  2803. }
  2804. //
  2805. // Return the appropriate value.
  2806. //
  2807. return (bFound);
  2808. }
  2809. ////////////////////////////////////////////////////////////////////////////
  2810. //
  2811. // CopyExceptionInfo
  2812. //
  2813. // Copies the exception information to the given sortkey table.
  2814. //
  2815. // 05-31-91 JulieB Created.
  2816. ////////////////////////////////////////////////////////////////////////////
  2817. void FASTCALL CopyExceptionInfo(
  2818. PSORTKEY pSortkey,
  2819. PEXCEPT_HDR pExceptHdr,
  2820. PEXCEPT pExceptTbl,
  2821. PVOID pIdeograph)
  2822. {
  2823. DWORD ctr; // loop counter
  2824. PIDEOGRAPH_EXCEPT_HDR pHdrIG; // ptr to ideograph exception header
  2825. PIDEOGRAPH_EXCEPT pEntryIG; // ptr to ideograph exception entry
  2826. PEXCEPT pEntryIGEx; // ptr to ideograph exception entry ex
  2827. //
  2828. // Make sure we're in the critical section when entering this call.
  2829. //
  2830. ASSERT(NtCurrentTeb()->ClientId.UniqueThread == gcsTblPtrs.OwningThread);
  2831. //
  2832. // For each entry in the exception table, copy the information to the
  2833. // sortkey table.
  2834. //
  2835. if (pExceptTbl)
  2836. {
  2837. for (ctr = pExceptHdr->NumEntries; ctr > 0; ctr--, pExceptTbl++)
  2838. {
  2839. (pSortkey[pExceptTbl->UCP]).UW.Unicode = pExceptTbl->Unicode;
  2840. (pSortkey[pExceptTbl->UCP]).Diacritic = pExceptTbl->Diacritic;
  2841. (pSortkey[pExceptTbl->UCP]).Case = pExceptTbl->Case;
  2842. }
  2843. }
  2844. //
  2845. // For each entry in the ideograph exception table, copy the
  2846. // information to the sortkey table.
  2847. //
  2848. if (pIdeograph)
  2849. {
  2850. pHdrIG = (PIDEOGRAPH_EXCEPT_HDR)pIdeograph;
  2851. ctr = pHdrIG->NumEntries;
  2852. if (pHdrIG->NumColumns == 2)
  2853. {
  2854. pEntryIG = (PIDEOGRAPH_EXCEPT)( ((LPBYTE)pIdeograph) +
  2855. sizeof(IDEOGRAPH_EXCEPT_HDR) );
  2856. for (; ctr > 0; ctr--, pEntryIG++)
  2857. {
  2858. (pSortkey[pEntryIG->UCP]).UW.Unicode = pEntryIG->Unicode;
  2859. }
  2860. }
  2861. else
  2862. {
  2863. pEntryIGEx = (PEXCEPT)( ((LPBYTE)pIdeograph) +
  2864. sizeof(IDEOGRAPH_EXCEPT_HDR) );
  2865. for (; ctr > 0; ctr--, pEntryIGEx++)
  2866. {
  2867. (pSortkey[pEntryIGEx->UCP]).UW.Unicode = pEntryIGEx->Unicode;
  2868. (pSortkey[pEntryIGEx->UCP]).Diacritic = pEntryIGEx->Diacritic;
  2869. (pSortkey[pEntryIGEx->UCP]).Case = pEntryIGEx->Case;
  2870. }
  2871. }
  2872. //
  2873. // Unmap and Close the ideograph section.
  2874. //
  2875. UnMapSection(pIdeograph);
  2876. }
  2877. }
  2878. ////////////////////////////////////////////////////////////////////////////
  2879. //
  2880. // WaitOnEvent
  2881. //
  2882. // Waits (via timeout) for the semaphore dword to be set to a non-zero
  2883. // value.
  2884. //
  2885. // 05-31-91 JulieB Created.
  2886. ////////////////////////////////////////////////////////////////////////////
  2887. ULONG WaitOnEvent(
  2888. LPWORD pSem)
  2889. {
  2890. TIME TimeOut; // ptr to timeout
  2891. //
  2892. // Set up the TIME structure.
  2893. //
  2894. TimeOut.QuadPart = -100000;
  2895. //
  2896. // Wait on the event until the semaphore is set to non-zero.
  2897. // Use a timeout on the wait.
  2898. //
  2899. do
  2900. {
  2901. NtDelayExecution(FALSE, &TimeOut);
  2902. } while (*pSem == 0);
  2903. //
  2904. // Return success.
  2905. //
  2906. return (NO_ERROR);
  2907. }