1 /*
2 *******************************************************************************
3 * Copyright (C) 2012-2014, International Business Machines
4 * Corporation and others. All Rights Reserved.
5 *******************************************************************************
6 * collationroot.cpp
7 *
8 * created on: 2012dec17
9 * created by: Markus W. Scherer
10 */
11
12 #include "unicode/utypes.h"
13
14 #if !UCONFIG_NO_COLLATION
15
16 #include "unicode/coll.h"
17 #include "unicode/udata.h"
18 #include "collation.h"
19 #include "collationdata.h"
20 #include "collationdatareader.h"
21 #include "collationroot.h"
22 #include "collationsettings.h"
23 #include "collationtailoring.h"
24 #include "normalizer2impl.h"
25 #include "ucln_in.h"
26 #include "udatamem.h"
27 #include "umutex.h"
28
29 U_NAMESPACE_BEGIN
30
31 namespace {
32
33 static const CollationCacheEntry *rootSingleton = NULL;
34 static UInitOnce initOnce = U_INITONCE_INITIALIZER;
35
36 } // namespace
37
38 U_CDECL_BEGIN
39
uprv_collation_root_cleanup()40 static UBool U_CALLCONV uprv_collation_root_cleanup() {
41 SharedObject::clearPtr(rootSingleton);
42 initOnce.reset();
43 return TRUE;
44 }
45
46 U_CDECL_END
47
48 void
load(UErrorCode & errorCode)49 CollationRoot::load(UErrorCode &errorCode) {
50 if(U_FAILURE(errorCode)) { return; }
51 LocalPointer<CollationTailoring> t(new CollationTailoring(NULL));
52 if(t.isNull() || t->isBogus()) {
53 errorCode = U_MEMORY_ALLOCATION_ERROR;
54 return;
55 }
56 t->memory = udata_openChoice(U_ICUDATA_NAME U_TREE_SEPARATOR_STRING "coll",
57 "icu", "ucadata",
58 CollationDataReader::isAcceptable, t->version, &errorCode);
59 if(U_FAILURE(errorCode)) { return; }
60 const uint8_t *inBytes = static_cast<const uint8_t *>(udata_getMemory(t->memory));
61 CollationDataReader::read(NULL, inBytes, udata_getLength(t->memory), *t, errorCode);
62 if(U_FAILURE(errorCode)) { return; }
63 ucln_i18n_registerCleanup(UCLN_I18N_COLLATION_ROOT, uprv_collation_root_cleanup);
64 CollationCacheEntry *entry = new CollationCacheEntry(Locale::getRoot(), t.getAlias());
65 if(entry != NULL) {
66 t.orphan(); // The rootSingleton took ownership of the tailoring.
67 entry->addRef();
68 rootSingleton = entry;
69 }
70 }
71
72 const CollationCacheEntry *
getRootCacheEntry(UErrorCode & errorCode)73 CollationRoot::getRootCacheEntry(UErrorCode &errorCode) {
74 umtx_initOnce(initOnce, CollationRoot::load, errorCode);
75 if(U_FAILURE(errorCode)) { return NULL; }
76 return rootSingleton;
77 }
78
79 const CollationTailoring *
getRoot(UErrorCode & errorCode)80 CollationRoot::getRoot(UErrorCode &errorCode) {
81 umtx_initOnce(initOnce, CollationRoot::load, errorCode);
82 if(U_FAILURE(errorCode)) { return NULL; }
83 return rootSingleton->tailoring;
84 }
85
86 const CollationData *
getData(UErrorCode & errorCode)87 CollationRoot::getData(UErrorCode &errorCode) {
88 const CollationTailoring *root = getRoot(errorCode);
89 if(U_FAILURE(errorCode)) { return NULL; }
90 return root->data;
91 }
92
93 const CollationSettings *
getSettings(UErrorCode & errorCode)94 CollationRoot::getSettings(UErrorCode &errorCode) {
95 const CollationTailoring *root = getRoot(errorCode);
96 if(U_FAILURE(errorCode)) { return NULL; }
97 return root->settings;
98 }
99
100 U_NAMESPACE_END
101
102 #endif // !UCONFIG_NO_COLLATION
103