1 // © 2016 and later: Unicode, Inc. and others.
2 // License & terms of use: http://www.unicode.org/copyright.html
3 /*
4 *******************************************************************************
5 * Copyright (C) 2011-2016, International Business Machines Corporation and
6 * others. All Rights Reserved.
7 *******************************************************************************
8 *
9 * File TZNAMES_IMPL.CPP
10 *
11 *******************************************************************************
12 */
13
14 #include "unicode/utypes.h"
15
16 #if !UCONFIG_NO_FORMATTING
17
18 #include "unicode/strenum.h"
19 #include "unicode/ustring.h"
20 #include "unicode/timezone.h"
21 #include "unicode/utf16.h"
22
23 #include "tznames_impl.h"
24 #include "cmemory.h"
25 #include "cstring.h"
26 #include "uassert.h"
27 #include "mutex.h"
28 #include "resource.h"
29 #include "uresimp.h"
30 #include "ureslocs.h"
31 #include "zonemeta.h"
32 #include "ucln_in.h"
33 #include "uvector.h"
34 #include "olsontz.h"
35
36 U_NAMESPACE_BEGIN
37
38 #define ZID_KEY_MAX 128
39 #define MZ_PREFIX_LEN 5
40
41 static const char gZoneStrings[] = "zoneStrings";
42 static const char gMZPrefix[] = "meta:";
43
44 static const char EMPTY[] = "<empty>"; // place holder for empty ZNames
45 static const char DUMMY_LOADER[] = "<dummy>"; // place holder for dummy ZNamesLoader
46 static const UChar NO_NAME[] = { 0 }; // for empty no-fallback time zone names
47
48 // stuff for TZDBTimeZoneNames
49 static const char* TZDBNAMES_KEYS[] = {"ss", "sd"};
50 static const int32_t TZDBNAMES_KEYS_SIZE = UPRV_LENGTHOF(TZDBNAMES_KEYS);
51
52 static UMutex gTZDBNamesMapLock = U_MUTEX_INITIALIZER;
53 static UMutex gDataMutex = U_MUTEX_INITIALIZER;
54
55 static UHashtable* gTZDBNamesMap = NULL;
56 static icu::UInitOnce gTZDBNamesMapInitOnce = U_INITONCE_INITIALIZER;
57
58 static TextTrieMap* gTZDBNamesTrie = NULL;
59 static icu::UInitOnce gTZDBNamesTrieInitOnce = U_INITONCE_INITIALIZER;
60
61 // The order in which strings are stored may be different than the order in the public enum.
62 enum UTimeZoneNameTypeIndex {
63 UTZNM_INDEX_UNKNOWN = -1,
64 UTZNM_INDEX_EXEMPLAR_LOCATION,
65 UTZNM_INDEX_LONG_GENERIC,
66 UTZNM_INDEX_LONG_STANDARD,
67 UTZNM_INDEX_LONG_DAYLIGHT,
68 UTZNM_INDEX_SHORT_GENERIC,
69 UTZNM_INDEX_SHORT_STANDARD,
70 UTZNM_INDEX_SHORT_DAYLIGHT,
71 UTZNM_INDEX_COUNT
72 };
73 static const UChar* const EMPTY_NAMES[UTZNM_INDEX_COUNT] = {0,0,0,0,0,0,0};
74
75 U_CDECL_BEGIN
tzdbTimeZoneNames_cleanup(void)76 static UBool U_CALLCONV tzdbTimeZoneNames_cleanup(void) {
77 if (gTZDBNamesMap != NULL) {
78 uhash_close(gTZDBNamesMap);
79 gTZDBNamesMap = NULL;
80 }
81 gTZDBNamesMapInitOnce.reset();
82
83 if (gTZDBNamesTrie != NULL) {
84 delete gTZDBNamesTrie;
85 gTZDBNamesTrie = NULL;
86 }
87 gTZDBNamesTrieInitOnce.reset();
88
89 return TRUE;
90 }
91 U_CDECL_END
92
93 /**
94 * ZNameInfo stores zone name information in the trie
95 */
96 struct ZNameInfo {
97 UTimeZoneNameType type;
98 const UChar* tzID;
99 const UChar* mzID;
100 };
101
102 /**
103 * ZMatchInfo stores zone name match information used by find method
104 */
105 struct ZMatchInfo {
106 const ZNameInfo* znameInfo;
107 int32_t matchLength;
108 };
109
110 // Helper functions
111 static void mergeTimeZoneKey(const UnicodeString& mzID, char* result);
112
113 #define DEFAULT_CHARACTERNODE_CAPACITY 1
114
115 // ---------------------------------------------------
116 // CharacterNode class implementation
117 // ---------------------------------------------------
clear()118 void CharacterNode::clear() {
119 uprv_memset(this, 0, sizeof(*this));
120 }
121
deleteValues(UObjectDeleter * valueDeleter)122 void CharacterNode::deleteValues(UObjectDeleter *valueDeleter) {
123 if (fValues == NULL) {
124 // Do nothing.
125 } else if (!fHasValuesVector) {
126 if (valueDeleter) {
127 valueDeleter(fValues);
128 }
129 } else {
130 delete (UVector *)fValues;
131 }
132 }
133
134 void
addValue(void * value,UObjectDeleter * valueDeleter,UErrorCode & status)135 CharacterNode::addValue(void *value, UObjectDeleter *valueDeleter, UErrorCode &status) {
136 if (U_FAILURE(status)) {
137 if (valueDeleter) {
138 valueDeleter(value);
139 }
140 return;
141 }
142 if (fValues == NULL) {
143 fValues = value;
144 } else {
145 // At least one value already.
146 if (!fHasValuesVector) {
147 // There is only one value so far, and not in a vector yet.
148 // Create a vector and add the old value.
149 UVector *values = new UVector(valueDeleter, NULL, DEFAULT_CHARACTERNODE_CAPACITY, status);
150 if (U_FAILURE(status)) {
151 if (valueDeleter) {
152 valueDeleter(value);
153 }
154 return;
155 }
156 values->addElement(fValues, status);
157 fValues = values;
158 fHasValuesVector = TRUE;
159 }
160 // Add the new value.
161 ((UVector *)fValues)->addElement(value, status);
162 }
163 }
164
165 // ---------------------------------------------------
166 // TextTrieMapSearchResultHandler class implementation
167 // ---------------------------------------------------
~TextTrieMapSearchResultHandler()168 TextTrieMapSearchResultHandler::~TextTrieMapSearchResultHandler(){
169 }
170
171 // ---------------------------------------------------
172 // TextTrieMap class implementation
173 // ---------------------------------------------------
TextTrieMap(UBool ignoreCase,UObjectDeleter * valueDeleter)174 TextTrieMap::TextTrieMap(UBool ignoreCase, UObjectDeleter *valueDeleter)
175 : fIgnoreCase(ignoreCase), fNodes(NULL), fNodesCapacity(0), fNodesCount(0),
176 fLazyContents(NULL), fIsEmpty(TRUE), fValueDeleter(valueDeleter) {
177 }
178
~TextTrieMap()179 TextTrieMap::~TextTrieMap() {
180 int32_t index;
181 for (index = 0; index < fNodesCount; ++index) {
182 fNodes[index].deleteValues(fValueDeleter);
183 }
184 uprv_free(fNodes);
185 if (fLazyContents != NULL) {
186 for (int32_t i=0; i<fLazyContents->size(); i+=2) {
187 if (fValueDeleter) {
188 fValueDeleter(fLazyContents->elementAt(i+1));
189 }
190 }
191 delete fLazyContents;
192 }
193 }
194
isEmpty() const195 int32_t TextTrieMap::isEmpty() const {
196 // Use a separate field for fIsEmpty because it will remain unchanged once the
197 // Trie is built, while fNodes and fLazyContents change with the lazy init
198 // of the nodes structure. Trying to test the changing fields has
199 // thread safety complications.
200 return fIsEmpty;
201 }
202
203
204 // We defer actually building the TextTrieMap node structure until the first time a
205 // search is performed. put() simply saves the parameters in case we do
206 // eventually need to build it.
207 //
208 void
put(const UnicodeString & key,void * value,ZNStringPool & sp,UErrorCode & status)209 TextTrieMap::put(const UnicodeString &key, void *value, ZNStringPool &sp, UErrorCode &status) {
210 const UChar *s = sp.get(key, status);
211 put(s, value, status);
212 }
213
214 // This method is designed for a persistent key, such as string key stored in
215 // resource bundle.
216 void
put(const UChar * key,void * value,UErrorCode & status)217 TextTrieMap::put(const UChar *key, void *value, UErrorCode &status) {
218 fIsEmpty = FALSE;
219 if (fLazyContents == NULL) {
220 fLazyContents = new UVector(status);
221 if (fLazyContents == NULL) {
222 status = U_MEMORY_ALLOCATION_ERROR;
223 }
224 }
225 if (U_FAILURE(status)) {
226 if (fValueDeleter) {
227 fValueDeleter((void*) key);
228 }
229 return;
230 }
231 U_ASSERT(fLazyContents != NULL);
232
233 UChar *s = const_cast<UChar *>(key);
234 fLazyContents->addElement(s, status);
235 if (U_FAILURE(status)) {
236 if (fValueDeleter) {
237 fValueDeleter((void*) key);
238 }
239 return;
240 }
241
242 fLazyContents->addElement(value, status);
243 }
244
245 void
putImpl(const UnicodeString & key,void * value,UErrorCode & status)246 TextTrieMap::putImpl(const UnicodeString &key, void *value, UErrorCode &status) {
247 if (fNodes == NULL) {
248 fNodesCapacity = 512;
249 fNodes = (CharacterNode *)uprv_malloc(fNodesCapacity * sizeof(CharacterNode));
250 if (fNodes == NULL) {
251 status = U_MEMORY_ALLOCATION_ERROR;
252 return;
253 }
254 fNodes[0].clear(); // Init root node.
255 fNodesCount = 1;
256 }
257
258 UnicodeString foldedKey;
259 const UChar *keyBuffer;
260 int32_t keyLength;
261 if (fIgnoreCase) {
262 // Ok to use fastCopyFrom() because we discard the copy when we return.
263 foldedKey.fastCopyFrom(key).foldCase();
264 keyBuffer = foldedKey.getBuffer();
265 keyLength = foldedKey.length();
266 } else {
267 keyBuffer = key.getBuffer();
268 keyLength = key.length();
269 }
270
271 CharacterNode *node = fNodes;
272 int32_t index;
273 for (index = 0; index < keyLength; ++index) {
274 node = addChildNode(node, keyBuffer[index], status);
275 }
276 node->addValue(value, fValueDeleter, status);
277 }
278
279 UBool
growNodes()280 TextTrieMap::growNodes() {
281 if (fNodesCapacity == 0xffff) {
282 return FALSE; // We use 16-bit node indexes.
283 }
284 int32_t newCapacity = fNodesCapacity + 1000;
285 if (newCapacity > 0xffff) {
286 newCapacity = 0xffff;
287 }
288 CharacterNode *newNodes = (CharacterNode *)uprv_malloc(newCapacity * sizeof(CharacterNode));
289 if (newNodes == NULL) {
290 return FALSE;
291 }
292 uprv_memcpy(newNodes, fNodes, fNodesCount * sizeof(CharacterNode));
293 uprv_free(fNodes);
294 fNodes = newNodes;
295 fNodesCapacity = newCapacity;
296 return TRUE;
297 }
298
299 CharacterNode*
addChildNode(CharacterNode * parent,UChar c,UErrorCode & status)300 TextTrieMap::addChildNode(CharacterNode *parent, UChar c, UErrorCode &status) {
301 if (U_FAILURE(status)) {
302 return NULL;
303 }
304 // Linear search of the sorted list of children.
305 uint16_t prevIndex = 0;
306 uint16_t nodeIndex = parent->fFirstChild;
307 while (nodeIndex > 0) {
308 CharacterNode *current = fNodes + nodeIndex;
309 UChar childCharacter = current->fCharacter;
310 if (childCharacter == c) {
311 return current;
312 } else if (childCharacter > c) {
313 break;
314 }
315 prevIndex = nodeIndex;
316 nodeIndex = current->fNextSibling;
317 }
318
319 // Ensure capacity. Grow fNodes[] if needed.
320 if (fNodesCount == fNodesCapacity) {
321 int32_t parentIndex = (int32_t)(parent - fNodes);
322 if (!growNodes()) {
323 status = U_MEMORY_ALLOCATION_ERROR;
324 return NULL;
325 }
326 parent = fNodes + parentIndex;
327 }
328
329 // Insert a new child node with c in sorted order.
330 CharacterNode *node = fNodes + fNodesCount;
331 node->clear();
332 node->fCharacter = c;
333 node->fNextSibling = nodeIndex;
334 if (prevIndex == 0) {
335 parent->fFirstChild = (uint16_t)fNodesCount;
336 } else {
337 fNodes[prevIndex].fNextSibling = (uint16_t)fNodesCount;
338 }
339 ++fNodesCount;
340 return node;
341 }
342
343 CharacterNode*
getChildNode(CharacterNode * parent,UChar c) const344 TextTrieMap::getChildNode(CharacterNode *parent, UChar c) const {
345 // Linear search of the sorted list of children.
346 uint16_t nodeIndex = parent->fFirstChild;
347 while (nodeIndex > 0) {
348 CharacterNode *current = fNodes + nodeIndex;
349 UChar childCharacter = current->fCharacter;
350 if (childCharacter == c) {
351 return current;
352 } else if (childCharacter > c) {
353 break;
354 }
355 nodeIndex = current->fNextSibling;
356 }
357 return NULL;
358 }
359
360 // Mutex for protecting the lazy creation of the Trie node structure on the first call to search().
361 static UMutex TextTrieMutex = U_MUTEX_INITIALIZER;
362
363 // buildTrie() - The Trie node structure is needed. Create it from the data that was
364 // saved at the time the ZoneStringFormatter was created. The Trie is only
365 // needed for parsing operations, which are less common than formatting,
366 // and the Trie is big, which is why its creation is deferred until first use.
buildTrie(UErrorCode & status)367 void TextTrieMap::buildTrie(UErrorCode &status) {
368 if (fLazyContents != NULL) {
369 for (int32_t i=0; i<fLazyContents->size(); i+=2) {
370 const UChar *key = (UChar *)fLazyContents->elementAt(i);
371 void *val = fLazyContents->elementAt(i+1);
372 UnicodeString keyString(TRUE, key, -1); // Aliasing UnicodeString constructor.
373 putImpl(keyString, val, status);
374 }
375 delete fLazyContents;
376 fLazyContents = NULL;
377 }
378 }
379
380 void
search(const UnicodeString & text,int32_t start,TextTrieMapSearchResultHandler * handler,UErrorCode & status) const381 TextTrieMap::search(const UnicodeString &text, int32_t start,
382 TextTrieMapSearchResultHandler *handler, UErrorCode &status) const {
383 {
384 // TODO: if locking the mutex for each check proves to be a performance problem,
385 // add a flag of type atomic_int32_t to class TextTrieMap, and use only
386 // the ICU atomic safe functions for assigning and testing.
387 // Don't test the pointer fLazyContents.
388 // Don't do unless it's really required.
389 Mutex lock(&TextTrieMutex);
390 if (fLazyContents != NULL) {
391 TextTrieMap *nonConstThis = const_cast<TextTrieMap *>(this);
392 nonConstThis->buildTrie(status);
393 }
394 }
395 if (fNodes == NULL) {
396 return;
397 }
398 search(fNodes, text, start, start, handler, status);
399 }
400
401 void
search(CharacterNode * node,const UnicodeString & text,int32_t start,int32_t index,TextTrieMapSearchResultHandler * handler,UErrorCode & status) const402 TextTrieMap::search(CharacterNode *node, const UnicodeString &text, int32_t start,
403 int32_t index, TextTrieMapSearchResultHandler *handler, UErrorCode &status) const {
404 if (U_FAILURE(status)) {
405 return;
406 }
407 if (node->hasValues()) {
408 if (!handler->handleMatch(index - start, node, status)) {
409 return;
410 }
411 if (U_FAILURE(status)) {
412 return;
413 }
414 }
415 if (fIgnoreCase) {
416 // for folding we need to get a complete code point.
417 // size of character may grow after fold operation;
418 // then we need to get result as UTF16 code units.
419 UChar32 c32 = text.char32At(index);
420 index += U16_LENGTH(c32);
421 UnicodeString tmp(c32);
422 tmp.foldCase();
423 int32_t tmpidx = 0;
424 while (tmpidx < tmp.length()) {
425 UChar c = tmp.charAt(tmpidx++);
426 node = getChildNode(node, c);
427 if (node == NULL) {
428 break;
429 }
430 }
431 } else {
432 // here we just get the next UTF16 code unit
433 UChar c = text.charAt(index++);
434 node = getChildNode(node, c);
435 }
436 if (node != NULL) {
437 search(node, text, start, index, handler, status);
438 }
439 }
440
441 // ---------------------------------------------------
442 // ZNStringPool class implementation
443 // ---------------------------------------------------
444 static const int32_t POOL_CHUNK_SIZE = 2000;
445 struct ZNStringPoolChunk: public UMemory {
446 ZNStringPoolChunk *fNext; // Ptr to next pool chunk
447 int32_t fLimit; // Index to start of unused area at end of fStrings
448 UChar fStrings[POOL_CHUNK_SIZE]; // Strings array
449 ZNStringPoolChunk();
450 };
451
ZNStringPoolChunk()452 ZNStringPoolChunk::ZNStringPoolChunk() {
453 fNext = NULL;
454 fLimit = 0;
455 }
456
ZNStringPool(UErrorCode & status)457 ZNStringPool::ZNStringPool(UErrorCode &status) {
458 fChunks = NULL;
459 fHash = NULL;
460 if (U_FAILURE(status)) {
461 return;
462 }
463 fChunks = new ZNStringPoolChunk;
464 if (fChunks == NULL) {
465 status = U_MEMORY_ALLOCATION_ERROR;
466 return;
467 }
468
469 fHash = uhash_open(uhash_hashUChars /* keyHash */,
470 uhash_compareUChars /* keyComp */,
471 uhash_compareUChars /* valueComp */,
472 &status);
473 if (U_FAILURE(status)) {
474 return;
475 }
476 }
477
~ZNStringPool()478 ZNStringPool::~ZNStringPool() {
479 if (fHash != NULL) {
480 uhash_close(fHash);
481 fHash = NULL;
482 }
483
484 while (fChunks != NULL) {
485 ZNStringPoolChunk *nextChunk = fChunks->fNext;
486 delete fChunks;
487 fChunks = nextChunk;
488 }
489 }
490
491 static const UChar EmptyString = 0;
492
get(const UChar * s,UErrorCode & status)493 const UChar *ZNStringPool::get(const UChar *s, UErrorCode &status) {
494 const UChar *pooledString;
495 if (U_FAILURE(status)) {
496 return &EmptyString;
497 }
498
499 pooledString = static_cast<UChar *>(uhash_get(fHash, s));
500 if (pooledString != NULL) {
501 return pooledString;
502 }
503
504 int32_t length = u_strlen(s);
505 int32_t remainingLength = POOL_CHUNK_SIZE - fChunks->fLimit;
506 if (remainingLength <= length) {
507 U_ASSERT(length < POOL_CHUNK_SIZE);
508 if (length >= POOL_CHUNK_SIZE) {
509 status = U_INTERNAL_PROGRAM_ERROR;
510 return &EmptyString;
511 }
512 ZNStringPoolChunk *oldChunk = fChunks;
513 fChunks = new ZNStringPoolChunk;
514 if (fChunks == NULL) {
515 status = U_MEMORY_ALLOCATION_ERROR;
516 return &EmptyString;
517 }
518 fChunks->fNext = oldChunk;
519 }
520
521 UChar *destString = &fChunks->fStrings[fChunks->fLimit];
522 u_strcpy(destString, s);
523 fChunks->fLimit += (length + 1);
524 uhash_put(fHash, destString, destString, &status);
525 return destString;
526 }
527
528
529 //
530 // ZNStringPool::adopt() Put a string into the hash, but do not copy the string data
531 // into the pool's storage. Used for strings from resource bundles,
532 // which will perisist for the life of the zone string formatter, and
533 // therefore can be used directly without copying.
adopt(const UChar * s,UErrorCode & status)534 const UChar *ZNStringPool::adopt(const UChar * s, UErrorCode &status) {
535 const UChar *pooledString;
536 if (U_FAILURE(status)) {
537 return &EmptyString;
538 }
539 if (s != NULL) {
540 pooledString = static_cast<UChar *>(uhash_get(fHash, s));
541 if (pooledString == NULL) {
542 UChar *ncs = const_cast<UChar *>(s);
543 uhash_put(fHash, ncs, ncs, &status);
544 }
545 }
546 return s;
547 }
548
549
get(const UnicodeString & s,UErrorCode & status)550 const UChar *ZNStringPool::get(const UnicodeString &s, UErrorCode &status) {
551 UnicodeString &nonConstStr = const_cast<UnicodeString &>(s);
552 return this->get(nonConstStr.getTerminatedBuffer(), status);
553 }
554
555 /*
556 * freeze(). Close the hash table that maps to the pooled strings.
557 * After freezing, the pool can not be searched or added to,
558 * but all existing references to pooled strings remain valid.
559 *
560 * The main purpose is to recover the storage used for the hash.
561 */
freeze()562 void ZNStringPool::freeze() {
563 uhash_close(fHash);
564 fHash = NULL;
565 }
566
567
568 /**
569 * This class stores name data for a meta zone or time zone.
570 */
571 class ZNames : public UMemory {
572 private:
573 friend class TimeZoneNamesImpl;
574
getTZNameTypeIndex(UTimeZoneNameType type)575 static UTimeZoneNameTypeIndex getTZNameTypeIndex(UTimeZoneNameType type) {
576 switch(type) {
577 case UTZNM_EXEMPLAR_LOCATION: return UTZNM_INDEX_EXEMPLAR_LOCATION;
578 case UTZNM_LONG_GENERIC: return UTZNM_INDEX_LONG_GENERIC;
579 case UTZNM_LONG_STANDARD: return UTZNM_INDEX_LONG_STANDARD;
580 case UTZNM_LONG_DAYLIGHT: return UTZNM_INDEX_LONG_DAYLIGHT;
581 case UTZNM_SHORT_GENERIC: return UTZNM_INDEX_SHORT_GENERIC;
582 case UTZNM_SHORT_STANDARD: return UTZNM_INDEX_SHORT_STANDARD;
583 case UTZNM_SHORT_DAYLIGHT: return UTZNM_INDEX_SHORT_DAYLIGHT;
584 default: return UTZNM_INDEX_UNKNOWN;
585 }
586 }
getTZNameType(UTimeZoneNameTypeIndex index)587 static UTimeZoneNameType getTZNameType(UTimeZoneNameTypeIndex index) {
588 switch(index) {
589 case UTZNM_INDEX_EXEMPLAR_LOCATION: return UTZNM_EXEMPLAR_LOCATION;
590 case UTZNM_INDEX_LONG_GENERIC: return UTZNM_LONG_GENERIC;
591 case UTZNM_INDEX_LONG_STANDARD: return UTZNM_LONG_STANDARD;
592 case UTZNM_INDEX_LONG_DAYLIGHT: return UTZNM_LONG_DAYLIGHT;
593 case UTZNM_INDEX_SHORT_GENERIC: return UTZNM_SHORT_GENERIC;
594 case UTZNM_INDEX_SHORT_STANDARD: return UTZNM_SHORT_STANDARD;
595 case UTZNM_INDEX_SHORT_DAYLIGHT: return UTZNM_SHORT_DAYLIGHT;
596 default: return UTZNM_UNKNOWN;
597 }
598 }
599
600 const UChar* fNames[UTZNM_INDEX_COUNT];
601 UBool fDidAddIntoTrie;
602
603 // Whether we own the location string, if computed rather than loaded from a bundle.
604 // A meta zone names instance never has an exemplar location string.
605 UBool fOwnsLocationName;
606
ZNames(const UChar * names[],const UChar * locationName)607 ZNames(const UChar* names[], const UChar* locationName)
608 : fDidAddIntoTrie(FALSE) {
609 uprv_memcpy(fNames, names, sizeof(fNames));
610 if (locationName != NULL) {
611 fOwnsLocationName = TRUE;
612 fNames[UTZNM_INDEX_EXEMPLAR_LOCATION] = locationName;
613 } else {
614 fOwnsLocationName = FALSE;
615 }
616 }
617
618 public:
~ZNames()619 ~ZNames() {
620 if (fOwnsLocationName) {
621 const UChar* locationName = fNames[UTZNM_INDEX_EXEMPLAR_LOCATION];
622 U_ASSERT(locationName != NULL);
623 uprv_free((void*) locationName);
624 }
625 }
626
627 private:
createMetaZoneAndPutInCache(UHashtable * cache,const UChar * names[],const UnicodeString & mzID,UErrorCode & status)628 static void* createMetaZoneAndPutInCache(UHashtable* cache, const UChar* names[],
629 const UnicodeString& mzID, UErrorCode& status) {
630 if (U_FAILURE(status)) { return NULL; }
631 U_ASSERT(names != NULL);
632
633 // Use the persistent ID as the resource key, so we can
634 // avoid duplications.
635 // TODO: Is there a more efficient way, like intern() in Java?
636 void* key = (void*) ZoneMeta::findMetaZoneID(mzID);
637 void* value;
638 if (uprv_memcmp(names, EMPTY_NAMES, sizeof(EMPTY_NAMES)) == 0) {
639 value = (void*) EMPTY;
640 } else {
641 value = (void*) (new ZNames(names, NULL));
642 if (value == NULL) {
643 status = U_MEMORY_ALLOCATION_ERROR;
644 return NULL;
645 }
646 }
647 uhash_put(cache, key, value, &status);
648 return value;
649 }
650
createTimeZoneAndPutInCache(UHashtable * cache,const UChar * names[],const UnicodeString & tzID,UErrorCode & status)651 static void* createTimeZoneAndPutInCache(UHashtable* cache, const UChar* names[],
652 const UnicodeString& tzID, UErrorCode& status) {
653 if (U_FAILURE(status)) { return NULL; }
654 U_ASSERT(names != NULL);
655
656 // If necessary, compute the location name from the time zone name.
657 UChar* locationName = NULL;
658 if (names[UTZNM_INDEX_EXEMPLAR_LOCATION] == NULL) {
659 UnicodeString locationNameUniStr;
660 TimeZoneNamesImpl::getDefaultExemplarLocationName(tzID, locationNameUniStr);
661
662 // Copy the computed location name to the heap
663 if (locationNameUniStr.length() > 0) {
664 const UChar* buff = locationNameUniStr.getTerminatedBuffer();
665 int32_t len = sizeof(UChar) * (locationNameUniStr.length() + 1);
666 locationName = (UChar*) uprv_malloc(len);
667 if (locationName == NULL) {
668 status = U_MEMORY_ALLOCATION_ERROR;
669 return NULL;
670 }
671 uprv_memcpy(locationName, buff, len);
672 }
673 }
674
675 // Use the persistent ID as the resource key, so we can
676 // avoid duplications.
677 // TODO: Is there a more efficient way, like intern() in Java?
678 void* key = (void*) ZoneMeta::findTimeZoneID(tzID);
679 void* value = (void*) (new ZNames(names, locationName));
680 if (value == NULL) {
681 status = U_MEMORY_ALLOCATION_ERROR;
682 return NULL;
683 }
684 uhash_put(cache, key, value, &status);
685 return value;
686 }
687
getName(UTimeZoneNameType type) const688 const UChar* getName(UTimeZoneNameType type) const {
689 UTimeZoneNameTypeIndex index = getTZNameTypeIndex(type);
690 return index >= 0 ? fNames[index] : NULL;
691 }
692
addAsMetaZoneIntoTrie(const UChar * mzID,TextTrieMap & trie,UErrorCode & status)693 void addAsMetaZoneIntoTrie(const UChar* mzID, TextTrieMap& trie, UErrorCode& status) {
694 addNamesIntoTrie(mzID, NULL, trie, status);
695 }
addAsTimeZoneIntoTrie(const UChar * tzID,TextTrieMap & trie,UErrorCode & status)696 void addAsTimeZoneIntoTrie(const UChar* tzID, TextTrieMap& trie, UErrorCode& status) {
697 addNamesIntoTrie(NULL, tzID, trie, status);
698 }
699
addNamesIntoTrie(const UChar * mzID,const UChar * tzID,TextTrieMap & trie,UErrorCode & status)700 void addNamesIntoTrie(const UChar* mzID, const UChar* tzID, TextTrieMap& trie,
701 UErrorCode& status) {
702 if (U_FAILURE(status)) { return; }
703 if (fDidAddIntoTrie) { return; }
704 fDidAddIntoTrie = TRUE;
705
706 for (int32_t i = 0; i < UTZNM_INDEX_COUNT; i++) {
707 const UChar* name = fNames[i];
708 if (name != NULL) {
709 ZNameInfo *nameinfo = (ZNameInfo *)uprv_malloc(sizeof(ZNameInfo));
710 if (nameinfo == NULL) {
711 status = U_MEMORY_ALLOCATION_ERROR;
712 return;
713 }
714 nameinfo->mzID = mzID;
715 nameinfo->tzID = tzID;
716 nameinfo->type = getTZNameType((UTimeZoneNameTypeIndex)i);
717 trie.put(name, nameinfo, status); // trie.put() takes ownership of the key
718 if (U_FAILURE(status)) {
719 return;
720 }
721 }
722 }
723 }
724
725 public:
726 struct ZNamesLoader;
727 };
728
729 struct ZNames::ZNamesLoader : public ResourceSink {
730 const UChar *names[UTZNM_INDEX_COUNT];
731
ZNamesLoaderZNames::ZNamesLoader732 ZNamesLoader() {
733 clear();
734 }
735 virtual ~ZNamesLoader();
736
737 /** Reset for loading another set of names. */
clearZNames::ZNamesLoader738 void clear() {
739 uprv_memcpy(names, EMPTY_NAMES, sizeof(names));
740 }
741
loadMetaZoneZNames::ZNamesLoader742 void loadMetaZone(const UResourceBundle* zoneStrings, const UnicodeString& mzID, UErrorCode& errorCode) {
743 if (U_FAILURE(errorCode)) { return; }
744
745 char key[ZID_KEY_MAX + 1];
746 mergeTimeZoneKey(mzID, key);
747
748 loadNames(zoneStrings, key, errorCode);
749 }
750
loadTimeZoneZNames::ZNamesLoader751 void loadTimeZone(const UResourceBundle* zoneStrings, const UnicodeString& tzID, UErrorCode& errorCode) {
752 // Replace "/" with ":".
753 UnicodeString uKey(tzID);
754 for (int32_t i = 0; i < uKey.length(); i++) {
755 if (uKey.charAt(i) == (UChar)0x2F) {
756 uKey.setCharAt(i, (UChar)0x3A);
757 }
758 }
759
760 char key[ZID_KEY_MAX + 1];
761 uKey.extract(0, uKey.length(), key, sizeof(key), US_INV);
762
763 loadNames(zoneStrings, key, errorCode);
764 }
765
loadNamesZNames::ZNamesLoader766 void loadNames(const UResourceBundle* zoneStrings, const char* key, UErrorCode& errorCode) {
767 U_ASSERT(zoneStrings != NULL);
768 U_ASSERT(key != NULL);
769 U_ASSERT(key[0] != '\0');
770
771 UErrorCode localStatus = U_ZERO_ERROR;
772 clear();
773 ures_getAllItemsWithFallback(zoneStrings, key, *this, localStatus);
774
775 // Ignore errors, but propogate possible warnings.
776 if (U_SUCCESS(localStatus)) {
777 errorCode = localStatus;
778 }
779 }
780
setNameIfEmptyZNames::ZNamesLoader781 void setNameIfEmpty(const char* key, const ResourceValue* value, UErrorCode& errorCode) {
782 UTimeZoneNameTypeIndex type = nameTypeFromKey(key);
783 if (type == UTZNM_INDEX_UNKNOWN) { return; }
784 if (names[type] == NULL) {
785 int32_t length;
786 // 'NO_NAME' indicates internally that this field should remain empty. It will be
787 // replaced by 'NULL' in getNames()
788 names[type] = (value == NULL) ? NO_NAME : value->getString(length, errorCode);
789 }
790 }
791
putZNames::ZNamesLoader792 virtual void put(const char* key, ResourceValue& value, UBool /*noFallback*/,
793 UErrorCode &errorCode) {
794 ResourceTable namesTable = value.getTable(errorCode);
795 if (U_FAILURE(errorCode)) { return; }
796 for (int32_t i = 0; namesTable.getKeyAndValue(i, key, value); ++i) {
797 if (value.isNoInheritanceMarker()) {
798 setNameIfEmpty(key, NULL, errorCode);
799 } else {
800 setNameIfEmpty(key, &value, errorCode);
801 }
802 }
803 }
804
nameTypeFromKeyZNames::ZNamesLoader805 static UTimeZoneNameTypeIndex nameTypeFromKey(const char *key) {
806 char c0, c1;
807 if ((c0 = key[0]) == 0 || (c1 = key[1]) == 0 || key[2] != 0) {
808 return UTZNM_INDEX_UNKNOWN;
809 }
810 if (c0 == 'l') {
811 return c1 == 'g' ? UTZNM_INDEX_LONG_GENERIC :
812 c1 == 's' ? UTZNM_INDEX_LONG_STANDARD :
813 c1 == 'd' ? UTZNM_INDEX_LONG_DAYLIGHT : UTZNM_INDEX_UNKNOWN;
814 } else if (c0 == 's') {
815 return c1 == 'g' ? UTZNM_INDEX_SHORT_GENERIC :
816 c1 == 's' ? UTZNM_INDEX_SHORT_STANDARD :
817 c1 == 'd' ? UTZNM_INDEX_SHORT_DAYLIGHT : UTZNM_INDEX_UNKNOWN;
818 } else if (c0 == 'e' && c1 == 'c') {
819 return UTZNM_INDEX_EXEMPLAR_LOCATION;
820 }
821 return UTZNM_INDEX_UNKNOWN;
822 }
823
824 /**
825 * Returns an array of names. It is the caller's responsibility to copy the data into a
826 * permanent location, as the returned array is owned by the loader instance and may be
827 * cleared or leave scope.
828 *
829 * This is different than Java, where the array will no longer be modified and null
830 * may be returned.
831 */
getNamesZNames::ZNamesLoader832 const UChar** getNames() {
833 // Remove 'NO_NAME' references in the array and replace with 'NULL'
834 for (int32_t i = 0; i < UTZNM_INDEX_COUNT; ++i) {
835 if (names[i] == NO_NAME) {
836 names[i] = NULL;
837 }
838 }
839 return names;
840 }
841 };
842
~ZNamesLoader()843 ZNames::ZNamesLoader::~ZNamesLoader() {}
844
845
846 // ---------------------------------------------------
847 // The meta zone ID enumeration class
848 // ---------------------------------------------------
849 class MetaZoneIDsEnumeration : public StringEnumeration {
850 public:
851 MetaZoneIDsEnumeration();
852 MetaZoneIDsEnumeration(const UVector& mzIDs);
853 MetaZoneIDsEnumeration(UVector* mzIDs);
854 virtual ~MetaZoneIDsEnumeration();
855 static UClassID U_EXPORT2 getStaticClassID(void);
856 virtual UClassID getDynamicClassID(void) const;
857 virtual const UnicodeString* snext(UErrorCode& status);
858 virtual void reset(UErrorCode& status);
859 virtual int32_t count(UErrorCode& status) const;
860 private:
861 int32_t fLen;
862 int32_t fPos;
863 const UVector* fMetaZoneIDs;
864 UVector *fLocalVector;
865 };
866
UOBJECT_DEFINE_RTTI_IMPLEMENTATION(MetaZoneIDsEnumeration)867 UOBJECT_DEFINE_RTTI_IMPLEMENTATION(MetaZoneIDsEnumeration)
868
869 MetaZoneIDsEnumeration::MetaZoneIDsEnumeration()
870 : fLen(0), fPos(0), fMetaZoneIDs(NULL), fLocalVector(NULL) {
871 }
872
MetaZoneIDsEnumeration(const UVector & mzIDs)873 MetaZoneIDsEnumeration::MetaZoneIDsEnumeration(const UVector& mzIDs)
874 : fPos(0), fMetaZoneIDs(&mzIDs), fLocalVector(NULL) {
875 fLen = fMetaZoneIDs->size();
876 }
877
MetaZoneIDsEnumeration(UVector * mzIDs)878 MetaZoneIDsEnumeration::MetaZoneIDsEnumeration(UVector *mzIDs)
879 : fLen(0), fPos(0), fMetaZoneIDs(mzIDs), fLocalVector(mzIDs) {
880 if (fMetaZoneIDs) {
881 fLen = fMetaZoneIDs->size();
882 }
883 }
884
885 const UnicodeString*
snext(UErrorCode & status)886 MetaZoneIDsEnumeration::snext(UErrorCode& status) {
887 if (U_SUCCESS(status) && fMetaZoneIDs != NULL && fPos < fLen) {
888 unistr.setTo((const UChar*)fMetaZoneIDs->elementAt(fPos++), -1);
889 return &unistr;
890 }
891 return NULL;
892 }
893
894 void
reset(UErrorCode &)895 MetaZoneIDsEnumeration::reset(UErrorCode& /*status*/) {
896 fPos = 0;
897 }
898
899 int32_t
count(UErrorCode &) const900 MetaZoneIDsEnumeration::count(UErrorCode& /*status*/) const {
901 return fLen;
902 }
903
~MetaZoneIDsEnumeration()904 MetaZoneIDsEnumeration::~MetaZoneIDsEnumeration() {
905 if (fLocalVector) {
906 delete fLocalVector;
907 }
908 }
909
910
911 // ---------------------------------------------------
912 // ZNameSearchHandler
913 // ---------------------------------------------------
914 class ZNameSearchHandler : public TextTrieMapSearchResultHandler {
915 public:
916 ZNameSearchHandler(uint32_t types);
917 virtual ~ZNameSearchHandler();
918
919 UBool handleMatch(int32_t matchLength, const CharacterNode *node, UErrorCode &status);
920 TimeZoneNames::MatchInfoCollection* getMatches(int32_t& maxMatchLen);
921
922 private:
923 uint32_t fTypes;
924 int32_t fMaxMatchLen;
925 TimeZoneNames::MatchInfoCollection* fResults;
926 };
927
ZNameSearchHandler(uint32_t types)928 ZNameSearchHandler::ZNameSearchHandler(uint32_t types)
929 : fTypes(types), fMaxMatchLen(0), fResults(NULL) {
930 }
931
~ZNameSearchHandler()932 ZNameSearchHandler::~ZNameSearchHandler() {
933 if (fResults != NULL) {
934 delete fResults;
935 }
936 }
937
938 UBool
handleMatch(int32_t matchLength,const CharacterNode * node,UErrorCode & status)939 ZNameSearchHandler::handleMatch(int32_t matchLength, const CharacterNode *node, UErrorCode &status) {
940 if (U_FAILURE(status)) {
941 return FALSE;
942 }
943 if (node->hasValues()) {
944 int32_t valuesCount = node->countValues();
945 for (int32_t i = 0; i < valuesCount; i++) {
946 ZNameInfo *nameinfo = (ZNameInfo *)node->getValue(i);
947 if (nameinfo == NULL) {
948 continue;
949 }
950 if ((nameinfo->type & fTypes) != 0) {
951 // matches a requested type
952 if (fResults == NULL) {
953 fResults = new TimeZoneNames::MatchInfoCollection();
954 if (fResults == NULL) {
955 status = U_MEMORY_ALLOCATION_ERROR;
956 }
957 }
958 if (U_SUCCESS(status)) {
959 U_ASSERT(fResults != NULL);
960 if (nameinfo->tzID) {
961 fResults->addZone(nameinfo->type, matchLength, UnicodeString(nameinfo->tzID, -1), status);
962 } else {
963 U_ASSERT(nameinfo->mzID);
964 fResults->addMetaZone(nameinfo->type, matchLength, UnicodeString(nameinfo->mzID, -1), status);
965 }
966 if (U_SUCCESS(status) && matchLength > fMaxMatchLen) {
967 fMaxMatchLen = matchLength;
968 }
969 }
970 }
971 }
972 }
973 return TRUE;
974 }
975
976 TimeZoneNames::MatchInfoCollection*
getMatches(int32_t & maxMatchLen)977 ZNameSearchHandler::getMatches(int32_t& maxMatchLen) {
978 // give the ownership to the caller
979 TimeZoneNames::MatchInfoCollection* results = fResults;
980 maxMatchLen = fMaxMatchLen;
981
982 // reset
983 fResults = NULL;
984 fMaxMatchLen = 0;
985 return results;
986 }
987
988 // ---------------------------------------------------
989 // TimeZoneNamesImpl
990 //
991 // TimeZoneNames implementation class. This is the main
992 // part of this module.
993 // ---------------------------------------------------
994
995 U_CDECL_BEGIN
996 /**
997 * Deleter for ZNames
998 */
999 static void U_CALLCONV
deleteZNames(void * obj)1000 deleteZNames(void *obj) {
1001 if (obj != EMPTY) {
1002 delete (ZNames*) obj;
1003 }
1004 }
1005
1006 /**
1007 * Deleter for ZNameInfo
1008 */
1009 static void U_CALLCONV
deleteZNameInfo(void * obj)1010 deleteZNameInfo(void *obj) {
1011 uprv_free(obj);
1012 }
1013
1014 U_CDECL_END
1015
TimeZoneNamesImpl(const Locale & locale,UErrorCode & status)1016 TimeZoneNamesImpl::TimeZoneNamesImpl(const Locale& locale, UErrorCode& status)
1017 : fLocale(locale),
1018 fZoneStrings(NULL),
1019 fTZNamesMap(NULL),
1020 fMZNamesMap(NULL),
1021 fNamesTrieFullyLoaded(FALSE),
1022 fNamesFullyLoaded(FALSE),
1023 fNamesTrie(TRUE, deleteZNameInfo) {
1024 initialize(locale, status);
1025 }
1026
1027 void
initialize(const Locale & locale,UErrorCode & status)1028 TimeZoneNamesImpl::initialize(const Locale& locale, UErrorCode& status) {
1029 if (U_FAILURE(status)) {
1030 return;
1031 }
1032
1033 // Load zoneStrings bundle
1034 UErrorCode tmpsts = U_ZERO_ERROR; // OK with fallback warning..
1035 fZoneStrings = ures_open(U_ICUDATA_ZONE, locale.getName(), &tmpsts);
1036 fZoneStrings = ures_getByKeyWithFallback(fZoneStrings, gZoneStrings, fZoneStrings, &tmpsts);
1037 if (U_FAILURE(tmpsts)) {
1038 status = tmpsts;
1039 cleanup();
1040 return;
1041 }
1042
1043 // Initialize hashtables holding time zone/meta zone names
1044 fMZNamesMap = uhash_open(uhash_hashUChars, uhash_compareUChars, NULL, &status);
1045 fTZNamesMap = uhash_open(uhash_hashUChars, uhash_compareUChars, NULL, &status);
1046 if (U_FAILURE(status)) {
1047 cleanup();
1048 return;
1049 }
1050
1051 uhash_setValueDeleter(fMZNamesMap, deleteZNames);
1052 uhash_setValueDeleter(fTZNamesMap, deleteZNames);
1053 // no key deleters for name maps
1054
1055 // preload zone strings for the default zone
1056 TimeZone *tz = TimeZone::createDefault();
1057 const UChar *tzID = ZoneMeta::getCanonicalCLDRID(*tz);
1058 if (tzID != NULL) {
1059 loadStrings(UnicodeString(tzID), status);
1060 }
1061 delete tz;
1062
1063 return;
1064 }
1065
1066 /*
1067 * This method updates the cache and must be called with a lock,
1068 * except initializer.
1069 */
1070 void
loadStrings(const UnicodeString & tzCanonicalID,UErrorCode & status)1071 TimeZoneNamesImpl::loadStrings(const UnicodeString& tzCanonicalID, UErrorCode& status) {
1072 loadTimeZoneNames(tzCanonicalID, status);
1073 LocalPointer<StringEnumeration> mzIDs(getAvailableMetaZoneIDs(tzCanonicalID, status));
1074 if (U_FAILURE(status)) { return; }
1075 U_ASSERT(!mzIDs.isNull());
1076
1077 const UnicodeString *mzID;
1078 while (((mzID = mzIDs->snext(status)) != NULL) && U_SUCCESS(status)) {
1079 loadMetaZoneNames(*mzID, status);
1080 }
1081 }
1082
~TimeZoneNamesImpl()1083 TimeZoneNamesImpl::~TimeZoneNamesImpl() {
1084 cleanup();
1085 }
1086
1087 void
cleanup()1088 TimeZoneNamesImpl::cleanup() {
1089 if (fZoneStrings != NULL) {
1090 ures_close(fZoneStrings);
1091 fZoneStrings = NULL;
1092 }
1093 if (fMZNamesMap != NULL) {
1094 uhash_close(fMZNamesMap);
1095 fMZNamesMap = NULL;
1096 }
1097 if (fTZNamesMap != NULL) {
1098 uhash_close(fTZNamesMap);
1099 fTZNamesMap = NULL;
1100 }
1101 }
1102
1103 UBool
operator ==(const TimeZoneNames & other) const1104 TimeZoneNamesImpl::operator==(const TimeZoneNames& other) const {
1105 if (this == &other) {
1106 return TRUE;
1107 }
1108 // No implementation for now
1109 return FALSE;
1110 }
1111
1112 TimeZoneNames*
clone() const1113 TimeZoneNamesImpl::clone() const {
1114 UErrorCode status = U_ZERO_ERROR;
1115 return new TimeZoneNamesImpl(fLocale, status);
1116 }
1117
1118 StringEnumeration*
getAvailableMetaZoneIDs(UErrorCode & status) const1119 TimeZoneNamesImpl::getAvailableMetaZoneIDs(UErrorCode& status) const {
1120 return TimeZoneNamesImpl::_getAvailableMetaZoneIDs(status);
1121 }
1122
1123 // static implementation of getAvailableMetaZoneIDs(UErrorCode&)
1124 StringEnumeration*
_getAvailableMetaZoneIDs(UErrorCode & status)1125 TimeZoneNamesImpl::_getAvailableMetaZoneIDs(UErrorCode& status) {
1126 if (U_FAILURE(status)) {
1127 return NULL;
1128 }
1129 const UVector* mzIDs = ZoneMeta::getAvailableMetazoneIDs();
1130 if (mzIDs == NULL) {
1131 return new MetaZoneIDsEnumeration();
1132 }
1133 return new MetaZoneIDsEnumeration(*mzIDs);
1134 }
1135
1136 StringEnumeration*
getAvailableMetaZoneIDs(const UnicodeString & tzID,UErrorCode & status) const1137 TimeZoneNamesImpl::getAvailableMetaZoneIDs(const UnicodeString& tzID, UErrorCode& status) const {
1138 return TimeZoneNamesImpl::_getAvailableMetaZoneIDs(tzID, status);
1139 }
1140
1141 // static implementation of getAvailableMetaZoneIDs(const UnicodeString&, UErrorCode&)
1142 StringEnumeration*
_getAvailableMetaZoneIDs(const UnicodeString & tzID,UErrorCode & status)1143 TimeZoneNamesImpl::_getAvailableMetaZoneIDs(const UnicodeString& tzID, UErrorCode& status) {
1144 if (U_FAILURE(status)) {
1145 return NULL;
1146 }
1147 const UVector* mappings = ZoneMeta::getMetazoneMappings(tzID);
1148 if (mappings == NULL) {
1149 return new MetaZoneIDsEnumeration();
1150 }
1151
1152 MetaZoneIDsEnumeration *senum = NULL;
1153 UVector* mzIDs = new UVector(NULL, uhash_compareUChars, status);
1154 if (mzIDs == NULL) {
1155 status = U_MEMORY_ALLOCATION_ERROR;
1156 }
1157 if (U_SUCCESS(status)) {
1158 U_ASSERT(mzIDs != NULL);
1159 for (int32_t i = 0; U_SUCCESS(status) && i < mappings->size(); i++) {
1160
1161 OlsonToMetaMappingEntry *map = (OlsonToMetaMappingEntry *)mappings->elementAt(i);
1162 const UChar *mzID = map->mzid;
1163 if (!mzIDs->contains((void *)mzID)) {
1164 mzIDs->addElement((void *)mzID, status);
1165 }
1166 }
1167 if (U_SUCCESS(status)) {
1168 senum = new MetaZoneIDsEnumeration(mzIDs);
1169 } else {
1170 delete mzIDs;
1171 }
1172 }
1173 return senum;
1174 }
1175
1176 UnicodeString&
getMetaZoneID(const UnicodeString & tzID,UDate date,UnicodeString & mzID) const1177 TimeZoneNamesImpl::getMetaZoneID(const UnicodeString& tzID, UDate date, UnicodeString& mzID) const {
1178 return TimeZoneNamesImpl::_getMetaZoneID(tzID, date, mzID);
1179 }
1180
1181 // static implementation of getMetaZoneID
1182 UnicodeString&
_getMetaZoneID(const UnicodeString & tzID,UDate date,UnicodeString & mzID)1183 TimeZoneNamesImpl::_getMetaZoneID(const UnicodeString& tzID, UDate date, UnicodeString& mzID) {
1184 ZoneMeta::getMetazoneID(tzID, date, mzID);
1185 return mzID;
1186 }
1187
1188 UnicodeString&
getReferenceZoneID(const UnicodeString & mzID,const char * region,UnicodeString & tzID) const1189 TimeZoneNamesImpl::getReferenceZoneID(const UnicodeString& mzID, const char* region, UnicodeString& tzID) const {
1190 return TimeZoneNamesImpl::_getReferenceZoneID(mzID, region, tzID);
1191 }
1192
1193 // static implementaion of getReferenceZoneID
1194 UnicodeString&
_getReferenceZoneID(const UnicodeString & mzID,const char * region,UnicodeString & tzID)1195 TimeZoneNamesImpl::_getReferenceZoneID(const UnicodeString& mzID, const char* region, UnicodeString& tzID) {
1196 ZoneMeta::getZoneIdByMetazone(mzID, UnicodeString(region, -1, US_INV), tzID);
1197 return tzID;
1198 }
1199
1200 UnicodeString&
getMetaZoneDisplayName(const UnicodeString & mzID,UTimeZoneNameType type,UnicodeString & name) const1201 TimeZoneNamesImpl::getMetaZoneDisplayName(const UnicodeString& mzID,
1202 UTimeZoneNameType type,
1203 UnicodeString& name) const {
1204 name.setToBogus(); // cleanup result.
1205 if (mzID.isEmpty()) {
1206 return name;
1207 }
1208
1209 ZNames *znames = NULL;
1210 TimeZoneNamesImpl *nonConstThis = const_cast<TimeZoneNamesImpl *>(this);
1211
1212 {
1213 Mutex lock(&gDataMutex);
1214 UErrorCode status = U_ZERO_ERROR;
1215 znames = nonConstThis->loadMetaZoneNames(mzID, status);
1216 if (U_FAILURE(status)) { return name; }
1217 }
1218
1219 if (znames != NULL) {
1220 const UChar* s = znames->getName(type);
1221 if (s != NULL) {
1222 name.setTo(TRUE, s, -1);
1223 }
1224 }
1225 return name;
1226 }
1227
1228 UnicodeString&
getTimeZoneDisplayName(const UnicodeString & tzID,UTimeZoneNameType type,UnicodeString & name) const1229 TimeZoneNamesImpl::getTimeZoneDisplayName(const UnicodeString& tzID, UTimeZoneNameType type, UnicodeString& name) const {
1230 name.setToBogus(); // cleanup result.
1231 if (tzID.isEmpty()) {
1232 return name;
1233 }
1234
1235 ZNames *tznames = NULL;
1236 TimeZoneNamesImpl *nonConstThis = const_cast<TimeZoneNamesImpl *>(this);
1237
1238 {
1239 Mutex lock(&gDataMutex);
1240 UErrorCode status = U_ZERO_ERROR;
1241 tznames = nonConstThis->loadTimeZoneNames(tzID, status);
1242 if (U_FAILURE(status)) { return name; }
1243 }
1244
1245 if (tznames != NULL) {
1246 const UChar *s = tznames->getName(type);
1247 if (s != NULL) {
1248 name.setTo(TRUE, s, -1);
1249 }
1250 }
1251 return name;
1252 }
1253
1254 UnicodeString&
getExemplarLocationName(const UnicodeString & tzID,UnicodeString & name) const1255 TimeZoneNamesImpl::getExemplarLocationName(const UnicodeString& tzID, UnicodeString& name) const {
1256 name.setToBogus(); // cleanup result.
1257 const UChar* locName = NULL;
1258 ZNames *tznames = NULL;
1259 TimeZoneNamesImpl *nonConstThis = const_cast<TimeZoneNamesImpl *>(this);
1260
1261 {
1262 Mutex lock(&gDataMutex);
1263 UErrorCode status = U_ZERO_ERROR;
1264 tznames = nonConstThis->loadTimeZoneNames(tzID, status);
1265 if (U_FAILURE(status)) { return name; }
1266 }
1267
1268 if (tznames != NULL) {
1269 locName = tznames->getName(UTZNM_EXEMPLAR_LOCATION);
1270 }
1271 if (locName != NULL) {
1272 name.setTo(TRUE, locName, -1);
1273 }
1274
1275 return name;
1276 }
1277
1278
1279 // Merge the MZ_PREFIX and mzId
mergeTimeZoneKey(const UnicodeString & mzID,char * result)1280 static void mergeTimeZoneKey(const UnicodeString& mzID, char* result) {
1281 if (mzID.isEmpty()) {
1282 result[0] = '\0';
1283 return;
1284 }
1285
1286 char mzIdChar[ZID_KEY_MAX + 1];
1287 int32_t keyLen;
1288 int32_t prefixLen = uprv_strlen(gMZPrefix);
1289 keyLen = mzID.extract(0, mzID.length(), mzIdChar, ZID_KEY_MAX + 1, US_INV);
1290 uprv_memcpy((void *)result, (void *)gMZPrefix, prefixLen);
1291 uprv_memcpy((void *)(result + prefixLen), (void *)mzIdChar, keyLen);
1292 result[keyLen + prefixLen] = '\0';
1293 }
1294
1295 /*
1296 * This method updates the cache and must be called with a lock
1297 */
1298 ZNames*
loadMetaZoneNames(const UnicodeString & mzID,UErrorCode & status)1299 TimeZoneNamesImpl::loadMetaZoneNames(const UnicodeString& mzID, UErrorCode& status) {
1300 if (U_FAILURE(status)) { return NULL; }
1301 U_ASSERT(mzID.length() <= ZID_KEY_MAX - MZ_PREFIX_LEN);
1302
1303 UChar mzIDKey[ZID_KEY_MAX + 1];
1304 mzID.extract(mzIDKey, ZID_KEY_MAX + 1, status);
1305 U_ASSERT(U_SUCCESS(status)); // already checked length above
1306 mzIDKey[mzID.length()] = 0;
1307
1308 void* mznames = uhash_get(fMZNamesMap, mzIDKey);
1309 if (mznames == NULL) {
1310 ZNames::ZNamesLoader loader;
1311 loader.loadMetaZone(fZoneStrings, mzID, status);
1312 mznames = ZNames::createMetaZoneAndPutInCache(fMZNamesMap, loader.getNames(), mzID, status);
1313 if (U_FAILURE(status)) { return NULL; }
1314 }
1315
1316 if (mznames != EMPTY) {
1317 return (ZNames*)mznames;
1318 } else {
1319 return NULL;
1320 }
1321 }
1322
1323 /*
1324 * This method updates the cache and must be called with a lock
1325 */
1326 ZNames*
loadTimeZoneNames(const UnicodeString & tzID,UErrorCode & status)1327 TimeZoneNamesImpl::loadTimeZoneNames(const UnicodeString& tzID, UErrorCode& status) {
1328 if (U_FAILURE(status)) { return NULL; }
1329 U_ASSERT(tzID.length() <= ZID_KEY_MAX);
1330
1331 UChar tzIDKey[ZID_KEY_MAX + 1];
1332 int32_t tzIDKeyLen = tzID.extract(tzIDKey, ZID_KEY_MAX + 1, status);
1333 U_ASSERT(U_SUCCESS(status)); // already checked length above
1334 tzIDKey[tzIDKeyLen] = 0;
1335
1336 void *tznames = uhash_get(fTZNamesMap, tzIDKey);
1337 if (tznames == NULL) {
1338 ZNames::ZNamesLoader loader;
1339 loader.loadTimeZone(fZoneStrings, tzID, status);
1340 tznames = ZNames::createTimeZoneAndPutInCache(fTZNamesMap, loader.getNames(), tzID, status);
1341 if (U_FAILURE(status)) { return NULL; }
1342 }
1343
1344 // tznames is never EMPTY
1345 return (ZNames*)tznames;
1346 }
1347
1348 TimeZoneNames::MatchInfoCollection*
find(const UnicodeString & text,int32_t start,uint32_t types,UErrorCode & status) const1349 TimeZoneNamesImpl::find(const UnicodeString& text, int32_t start, uint32_t types, UErrorCode& status) const {
1350 ZNameSearchHandler handler(types);
1351 TimeZoneNames::MatchInfoCollection* matches;
1352 TimeZoneNamesImpl* nonConstThis = const_cast<TimeZoneNamesImpl*>(this);
1353
1354 // Synchronize so that data is not loaded multiple times.
1355 // TODO: Consider more fine-grained synchronization.
1356 {
1357 Mutex lock(&gDataMutex);
1358
1359 // First try of lookup.
1360 matches = doFind(handler, text, start, status);
1361 if (U_FAILURE(status)) { return NULL; }
1362 if (matches != NULL) {
1363 return matches;
1364 }
1365
1366 // All names are not yet loaded into the trie.
1367 // We may have loaded names for formatting several time zones,
1368 // and might be parsing one of those.
1369 // Populate the parsing trie from all of the already-loaded names.
1370 nonConstThis->addAllNamesIntoTrie(status);
1371
1372 // Second try of lookup.
1373 matches = doFind(handler, text, start, status);
1374 if (U_FAILURE(status)) { return NULL; }
1375 if (matches != NULL) {
1376 return matches;
1377 }
1378
1379 // There are still some names we haven't loaded into the trie yet.
1380 // Load everything now.
1381 nonConstThis->internalLoadAllDisplayNames(status);
1382 nonConstThis->addAllNamesIntoTrie(status);
1383 nonConstThis->fNamesTrieFullyLoaded = TRUE;
1384 if (U_FAILURE(status)) { return NULL; }
1385
1386 // Third try: we must return this one.
1387 return doFind(handler, text, start, status);
1388 }
1389 }
1390
1391 TimeZoneNames::MatchInfoCollection*
doFind(ZNameSearchHandler & handler,const UnicodeString & text,int32_t start,UErrorCode & status) const1392 TimeZoneNamesImpl::doFind(ZNameSearchHandler& handler,
1393 const UnicodeString& text, int32_t start, UErrorCode& status) const {
1394
1395 fNamesTrie.search(text, start, (TextTrieMapSearchResultHandler *)&handler, status);
1396 if (U_FAILURE(status)) { return NULL; }
1397
1398 int32_t maxLen = 0;
1399 TimeZoneNames::MatchInfoCollection* matches = handler.getMatches(maxLen);
1400 if (matches != NULL && ((maxLen == (text.length() - start)) || fNamesTrieFullyLoaded)) {
1401 // perfect match, or no more names available
1402 return matches;
1403 }
1404 delete matches;
1405 return NULL;
1406 }
1407
1408 // Caller must synchronize.
addAllNamesIntoTrie(UErrorCode & status)1409 void TimeZoneNamesImpl::addAllNamesIntoTrie(UErrorCode& status) {
1410 if (U_FAILURE(status)) return;
1411 int32_t pos;
1412 const UHashElement* element;
1413
1414 pos = UHASH_FIRST;
1415 while ((element = uhash_nextElement(fMZNamesMap, &pos)) != NULL) {
1416 if (element->value.pointer == EMPTY) { continue; }
1417 UChar* mzID = (UChar*) element->key.pointer;
1418 ZNames* znames = (ZNames*) element->value.pointer;
1419 znames->addAsMetaZoneIntoTrie(mzID, fNamesTrie, status);
1420 if (U_FAILURE(status)) { return; }
1421 }
1422
1423 pos = UHASH_FIRST;
1424 while ((element = uhash_nextElement(fTZNamesMap, &pos)) != NULL) {
1425 if (element->value.pointer == EMPTY) { continue; }
1426 UChar* tzID = (UChar*) element->key.pointer;
1427 ZNames* znames = (ZNames*) element->value.pointer;
1428 znames->addAsTimeZoneIntoTrie(tzID, fNamesTrie, status);
1429 if (U_FAILURE(status)) { return; }
1430 }
1431 }
1432
1433 U_CDECL_BEGIN
1434 static void U_CALLCONV
deleteZNamesLoader(void * obj)1435 deleteZNamesLoader(void* obj) {
1436 if (obj == DUMMY_LOADER) { return; }
1437 const ZNames::ZNamesLoader* loader = (const ZNames::ZNamesLoader*) obj;
1438 delete loader;
1439 }
1440 U_CDECL_END
1441
1442 struct TimeZoneNamesImpl::ZoneStringsLoader : public ResourceSink {
1443 TimeZoneNamesImpl& tzn;
1444 UHashtable* keyToLoader;
1445
ZoneStringsLoaderTimeZoneNamesImpl::ZoneStringsLoader1446 ZoneStringsLoader(TimeZoneNamesImpl& _tzn, UErrorCode& status)
1447 : tzn(_tzn) {
1448 keyToLoader = uhash_open(uhash_hashChars, uhash_compareChars, NULL, &status);
1449 if (U_FAILURE(status)) { return; }
1450 uhash_setKeyDeleter(keyToLoader, uprv_free);
1451 uhash_setValueDeleter(keyToLoader, deleteZNamesLoader);
1452 }
1453 virtual ~ZoneStringsLoader();
1454
createKeyTimeZoneNamesImpl::ZoneStringsLoader1455 void* createKey(const char* key, UErrorCode& status) {
1456 int32_t len = sizeof(char) * (uprv_strlen(key) + 1);
1457 char* newKey = (char*) uprv_malloc(len);
1458 if (newKey == NULL) {
1459 status = U_MEMORY_ALLOCATION_ERROR;
1460 return NULL;
1461 }
1462 uprv_memcpy(newKey, key, len);
1463 newKey[len-1] = '\0';
1464 return (void*) newKey;
1465 }
1466
isMetaZoneTimeZoneNamesImpl::ZoneStringsLoader1467 UBool isMetaZone(const char* key) {
1468 return (uprv_strlen(key) >= MZ_PREFIX_LEN && uprv_memcmp(key, gMZPrefix, MZ_PREFIX_LEN) == 0);
1469 }
1470
mzIDFromKeyTimeZoneNamesImpl::ZoneStringsLoader1471 UnicodeString mzIDFromKey(const char* key) {
1472 return UnicodeString(key + MZ_PREFIX_LEN, uprv_strlen(key) - MZ_PREFIX_LEN, US_INV);
1473 }
1474
tzIDFromKeyTimeZoneNamesImpl::ZoneStringsLoader1475 UnicodeString tzIDFromKey(const char* key) {
1476 UnicodeString tzID(key, -1, US_INV);
1477 // Replace all colons ':' with slashes '/'
1478 for (int i=0; i<tzID.length(); i++) {
1479 if (tzID.charAt(i) == 0x003A) {
1480 tzID.setCharAt(i, 0x002F);
1481 }
1482 }
1483 return tzID;
1484 }
1485
loadTimeZoneNamesImpl::ZoneStringsLoader1486 void load(UErrorCode& status) {
1487 ures_getAllItemsWithFallback(tzn.fZoneStrings, "", *this, status);
1488 if (U_FAILURE(status)) { return; }
1489
1490 int32_t pos = UHASH_FIRST;
1491 const UHashElement* element;
1492 while ((element = uhash_nextElement(keyToLoader, &pos)) != NULL) {
1493 if (element->value.pointer == DUMMY_LOADER) { continue; }
1494 ZNames::ZNamesLoader* loader = (ZNames::ZNamesLoader*) element->value.pointer;
1495 char* key = (char*) element->key.pointer;
1496
1497 if (isMetaZone(key)) {
1498 UnicodeString mzID = mzIDFromKey(key);
1499 ZNames::createMetaZoneAndPutInCache(tzn.fMZNamesMap, loader->getNames(), mzID, status);
1500 } else {
1501 UnicodeString tzID = tzIDFromKey(key);
1502 ZNames::createTimeZoneAndPutInCache(tzn.fTZNamesMap, loader->getNames(), tzID, status);
1503 }
1504 if (U_FAILURE(status)) { return; }
1505 }
1506 }
1507
consumeNamesTableTimeZoneNamesImpl::ZoneStringsLoader1508 void consumeNamesTable(const char *key, ResourceValue &value, UBool noFallback,
1509 UErrorCode &status) {
1510 if (U_FAILURE(status)) { return; }
1511
1512 void* loader = uhash_get(keyToLoader, key);
1513 if (loader == NULL) {
1514 if (isMetaZone(key)) {
1515 UnicodeString mzID = mzIDFromKey(key);
1516 void* cacheVal = uhash_get(tzn.fMZNamesMap, mzID.getTerminatedBuffer());
1517 if (cacheVal != NULL) {
1518 // We have already loaded the names for this meta zone.
1519 loader = (void*) DUMMY_LOADER;
1520 } else {
1521 loader = (void*) new ZNames::ZNamesLoader();
1522 if (loader == NULL) {
1523 status = U_MEMORY_ALLOCATION_ERROR;
1524 return;
1525 }
1526 }
1527 } else {
1528 UnicodeString tzID = tzIDFromKey(key);
1529 void* cacheVal = uhash_get(tzn.fTZNamesMap, tzID.getTerminatedBuffer());
1530 if (cacheVal != NULL) {
1531 // We have already loaded the names for this time zone.
1532 loader = (void*) DUMMY_LOADER;
1533 } else {
1534 loader = (void*) new ZNames::ZNamesLoader();
1535 if (loader == NULL) {
1536 status = U_MEMORY_ALLOCATION_ERROR;
1537 return;
1538 }
1539 }
1540 }
1541
1542 void* newKey = createKey(key, status);
1543 if (U_FAILURE(status)) {
1544 deleteZNamesLoader(loader);
1545 return;
1546 }
1547
1548 uhash_put(keyToLoader, newKey, loader, &status);
1549 if (U_FAILURE(status)) { return; }
1550 }
1551
1552 if (loader != DUMMY_LOADER) {
1553 // Let the ZNamesLoader consume the names table.
1554 ((ZNames::ZNamesLoader*)loader)->put(key, value, noFallback, status);
1555 }
1556 }
1557
putTimeZoneNamesImpl::ZoneStringsLoader1558 virtual void put(const char *key, ResourceValue &value, UBool noFallback,
1559 UErrorCode &status) {
1560 ResourceTable timeZonesTable = value.getTable(status);
1561 if (U_FAILURE(status)) { return; }
1562 for (int32_t i = 0; timeZonesTable.getKeyAndValue(i, key, value); ++i) {
1563 U_ASSERT(!value.isNoInheritanceMarker());
1564 if (value.getType() == URES_TABLE) {
1565 consumeNamesTable(key, value, noFallback, status);
1566 } else {
1567 // Ignore fields that aren't tables (e.g., fallbackFormat and regionFormatStandard).
1568 // All time zone fields are tables.
1569 }
1570 if (U_FAILURE(status)) { return; }
1571 }
1572 }
1573 };
1574
1575 // Virtual destructors must be defined out of line.
~ZoneStringsLoader()1576 TimeZoneNamesImpl::ZoneStringsLoader::~ZoneStringsLoader() {
1577 uhash_close(keyToLoader);
1578 }
1579
loadAllDisplayNames(UErrorCode & status)1580 void TimeZoneNamesImpl::loadAllDisplayNames(UErrorCode& status) {
1581 if (U_FAILURE(status)) return;
1582
1583 {
1584 Mutex lock(&gDataMutex);
1585 internalLoadAllDisplayNames(status);
1586 }
1587 }
1588
getDisplayNames(const UnicodeString & tzID,const UTimeZoneNameType types[],int32_t numTypes,UDate date,UnicodeString dest[],UErrorCode & status) const1589 void TimeZoneNamesImpl::getDisplayNames(const UnicodeString& tzID,
1590 const UTimeZoneNameType types[], int32_t numTypes,
1591 UDate date, UnicodeString dest[], UErrorCode& status) const {
1592 if (U_FAILURE(status)) return;
1593
1594 if (tzID.isEmpty()) { return; }
1595 void* tznames = NULL;
1596 void* mznames = NULL;
1597 TimeZoneNamesImpl *nonConstThis = const_cast<TimeZoneNamesImpl*>(this);
1598
1599 // Load the time zone strings
1600 {
1601 Mutex lock(&gDataMutex);
1602 tznames = (void*) nonConstThis->loadTimeZoneNames(tzID, status);
1603 if (U_FAILURE(status)) { return; }
1604 }
1605 U_ASSERT(tznames != NULL);
1606
1607 // Load the values into the dest array
1608 for (int i = 0; i < numTypes; i++) {
1609 UTimeZoneNameType type = types[i];
1610 const UChar* name = ((ZNames*)tznames)->getName(type);
1611 if (name == NULL) {
1612 if (mznames == NULL) {
1613 // Load the meta zone name
1614 UnicodeString mzID;
1615 getMetaZoneID(tzID, date, mzID);
1616 if (mzID.isEmpty()) {
1617 mznames = (void*) EMPTY;
1618 } else {
1619 // Load the meta zone strings
1620 // Mutex is scoped to the "else" statement
1621 Mutex lock(&gDataMutex);
1622 mznames = (void*) nonConstThis->loadMetaZoneNames(mzID, status);
1623 if (U_FAILURE(status)) { return; }
1624 // Note: when the metazone doesn't exist, in Java, loadMetaZoneNames returns
1625 // a dummy object instead of NULL.
1626 if (mznames == NULL) {
1627 mznames = (void*) EMPTY;
1628 }
1629 }
1630 }
1631 U_ASSERT(mznames != NULL);
1632 if (mznames != EMPTY) {
1633 name = ((ZNames*)mznames)->getName(type);
1634 }
1635 }
1636 if (name != NULL) {
1637 dest[i].setTo(TRUE, name, -1);
1638 } else {
1639 dest[i].setToBogus();
1640 }
1641 }
1642 }
1643
1644 // Caller must synchronize.
internalLoadAllDisplayNames(UErrorCode & status)1645 void TimeZoneNamesImpl::internalLoadAllDisplayNames(UErrorCode& status) {
1646 if (!fNamesFullyLoaded) {
1647 fNamesFullyLoaded = TRUE;
1648
1649 ZoneStringsLoader loader(*this, status);
1650 loader.load(status);
1651 if (U_FAILURE(status)) { return; }
1652
1653 const UnicodeString *id;
1654
1655 // load strings for all zones
1656 StringEnumeration *tzIDs = TimeZone::createTimeZoneIDEnumeration(
1657 UCAL_ZONE_TYPE_CANONICAL, NULL, NULL, status);
1658 if (U_SUCCESS(status)) {
1659 while ((id = tzIDs->snext(status)) != NULL) {
1660 if (U_FAILURE(status)) {
1661 break;
1662 }
1663 UnicodeString copy(*id);
1664 void* value = uhash_get(fTZNamesMap, copy.getTerminatedBuffer());
1665 if (value == NULL) {
1666 // loadStrings also loads related metazone strings
1667 loadStrings(*id, status);
1668 }
1669 }
1670 }
1671 if (tzIDs != NULL) {
1672 delete tzIDs;
1673 }
1674 }
1675 }
1676
1677
1678
1679 static const UChar gEtcPrefix[] = { 0x45, 0x74, 0x63, 0x2F }; // "Etc/"
1680 static const int32_t gEtcPrefixLen = 4;
1681 static const UChar gSystemVPrefix[] = { 0x53, 0x79, 0x73, 0x74, 0x65, 0x6D, 0x56, 0x2F }; // "SystemV/
1682 static const int32_t gSystemVPrefixLen = 8;
1683 static const UChar gRiyadh8[] = { 0x52, 0x69, 0x79, 0x61, 0x64, 0x68, 0x38 }; // "Riyadh8"
1684 static const int32_t gRiyadh8Len = 7;
1685
1686 UnicodeString& U_EXPORT2
getDefaultExemplarLocationName(const UnicodeString & tzID,UnicodeString & name)1687 TimeZoneNamesImpl::getDefaultExemplarLocationName(const UnicodeString& tzID, UnicodeString& name) {
1688 if (tzID.isEmpty() || tzID.startsWith(gEtcPrefix, gEtcPrefixLen)
1689 || tzID.startsWith(gSystemVPrefix, gSystemVPrefixLen) || tzID.indexOf(gRiyadh8, gRiyadh8Len, 0) > 0) {
1690 name.setToBogus();
1691 return name;
1692 }
1693
1694 int32_t sep = tzID.lastIndexOf((UChar)0x2F /* '/' */);
1695 if (sep > 0 && sep + 1 < tzID.length()) {
1696 name.setTo(tzID, sep + 1);
1697 name.findAndReplace(UnicodeString((UChar)0x5f /* _ */),
1698 UnicodeString((UChar)0x20 /* space */));
1699 } else {
1700 name.setToBogus();
1701 }
1702 return name;
1703 }
1704
1705 // ---------------------------------------------------
1706 // TZDBTimeZoneNames and its supporting classes
1707 //
1708 // TZDBTimeZoneNames is an implementation class of
1709 // TimeZoneNames holding the IANA tz database abbreviations.
1710 // ---------------------------------------------------
1711
1712 class TZDBNames : public UMemory {
1713 public:
1714 virtual ~TZDBNames();
1715
1716 static TZDBNames* createInstance(UResourceBundle* rb, const char* key);
1717 const UChar* getName(UTimeZoneNameType type) const;
1718 const char** getParseRegions(int32_t& numRegions) const;
1719
1720 protected:
1721 TZDBNames(const UChar** names, char** regions, int32_t numRegions);
1722
1723 private:
1724 const UChar** fNames;
1725 char** fRegions;
1726 int32_t fNumRegions;
1727 };
1728
TZDBNames(const UChar ** names,char ** regions,int32_t numRegions)1729 TZDBNames::TZDBNames(const UChar** names, char** regions, int32_t numRegions)
1730 : fNames(names),
1731 fRegions(regions),
1732 fNumRegions(numRegions) {
1733 }
1734
~TZDBNames()1735 TZDBNames::~TZDBNames() {
1736 if (fNames != NULL) {
1737 uprv_free(fNames);
1738 }
1739 if (fRegions != NULL) {
1740 char **p = fRegions;
1741 for (int32_t i = 0; i < fNumRegions; p++, i++) {
1742 uprv_free(*p);
1743 }
1744 uprv_free(fRegions);
1745 }
1746 }
1747
1748 TZDBNames*
createInstance(UResourceBundle * rb,const char * key)1749 TZDBNames::createInstance(UResourceBundle* rb, const char* key) {
1750 if (rb == NULL || key == NULL || *key == 0) {
1751 return NULL;
1752 }
1753
1754 UErrorCode status = U_ZERO_ERROR;
1755
1756 const UChar **names = NULL;
1757 char** regions = NULL;
1758 int32_t numRegions = 0;
1759
1760 int32_t len = 0;
1761
1762 UResourceBundle* rbTable = NULL;
1763 rbTable = ures_getByKey(rb, key, rbTable, &status);
1764 if (U_FAILURE(status)) {
1765 return NULL;
1766 }
1767
1768 names = (const UChar **)uprv_malloc(sizeof(const UChar*) * TZDBNAMES_KEYS_SIZE);
1769 UBool isEmpty = TRUE;
1770 if (names != NULL) {
1771 for (int32_t i = 0; i < TZDBNAMES_KEYS_SIZE; i++) {
1772 status = U_ZERO_ERROR;
1773 const UChar *value = ures_getStringByKey(rbTable, TZDBNAMES_KEYS[i], &len, &status);
1774 if (U_FAILURE(status) || len == 0) {
1775 names[i] = NULL;
1776 } else {
1777 names[i] = value;
1778 isEmpty = FALSE;
1779 }
1780 }
1781 }
1782
1783 if (isEmpty) {
1784 if (names != NULL) {
1785 uprv_free(names);
1786 }
1787 return NULL;
1788 }
1789
1790 UResourceBundle *regionsRes = ures_getByKey(rbTable, "parseRegions", NULL, &status);
1791 UBool regionError = FALSE;
1792 if (U_SUCCESS(status)) {
1793 numRegions = ures_getSize(regionsRes);
1794 if (numRegions > 0) {
1795 regions = (char**)uprv_malloc(sizeof(char*) * numRegions);
1796 if (regions != NULL) {
1797 char **pRegion = regions;
1798 for (int32_t i = 0; i < numRegions; i++, pRegion++) {
1799 *pRegion = NULL;
1800 }
1801 // filling regions
1802 pRegion = regions;
1803 for (int32_t i = 0; i < numRegions; i++, pRegion++) {
1804 status = U_ZERO_ERROR;
1805 const UChar *uregion = ures_getStringByIndex(regionsRes, i, &len, &status);
1806 if (U_FAILURE(status)) {
1807 regionError = TRUE;
1808 break;
1809 }
1810 *pRegion = (char*)uprv_malloc(sizeof(char) * (len + 1));
1811 if (*pRegion == NULL) {
1812 regionError = TRUE;
1813 break;
1814 }
1815 u_UCharsToChars(uregion, *pRegion, len);
1816 (*pRegion)[len] = 0;
1817 }
1818 }
1819 }
1820 }
1821 ures_close(regionsRes);
1822 ures_close(rbTable);
1823
1824 if (regionError) {
1825 if (names != NULL) {
1826 uprv_free(names);
1827 }
1828 if (regions != NULL) {
1829 char **p = regions;
1830 for (int32_t i = 0; i < numRegions; p++, i++) {
1831 uprv_free(*p);
1832 }
1833 uprv_free(regions);
1834 }
1835 return NULL;
1836 }
1837
1838 return new TZDBNames(names, regions, numRegions);
1839 }
1840
1841 const UChar*
getName(UTimeZoneNameType type) const1842 TZDBNames::getName(UTimeZoneNameType type) const {
1843 if (fNames == NULL) {
1844 return NULL;
1845 }
1846 const UChar *name = NULL;
1847 switch(type) {
1848 case UTZNM_SHORT_STANDARD:
1849 name = fNames[0];
1850 break;
1851 case UTZNM_SHORT_DAYLIGHT:
1852 name = fNames[1];
1853 break;
1854 default:
1855 name = NULL;
1856 }
1857 return name;
1858 }
1859
1860 const char**
getParseRegions(int32_t & numRegions) const1861 TZDBNames::getParseRegions(int32_t& numRegions) const {
1862 if (fRegions == NULL) {
1863 numRegions = 0;
1864 } else {
1865 numRegions = fNumRegions;
1866 }
1867 return (const char**)fRegions;
1868 }
1869
1870 U_CDECL_BEGIN
1871 /**
1872 * TZDBNameInfo stores metazone name information for the IANA abbreviations
1873 * in the trie
1874 */
1875 typedef struct TZDBNameInfo {
1876 const UChar* mzID;
1877 UTimeZoneNameType type;
1878 UBool ambiguousType;
1879 const char** parseRegions;
1880 int32_t nRegions;
1881 } TZDBNameInfo;
1882 U_CDECL_END
1883
1884
1885 class TZDBNameSearchHandler : public TextTrieMapSearchResultHandler {
1886 public:
1887 TZDBNameSearchHandler(uint32_t types, const char* region);
1888 virtual ~TZDBNameSearchHandler();
1889
1890 UBool handleMatch(int32_t matchLength, const CharacterNode *node, UErrorCode &status);
1891 TimeZoneNames::MatchInfoCollection* getMatches(int32_t& maxMatchLen);
1892
1893 private:
1894 uint32_t fTypes;
1895 int32_t fMaxMatchLen;
1896 TimeZoneNames::MatchInfoCollection* fResults;
1897 const char* fRegion;
1898 };
1899
TZDBNameSearchHandler(uint32_t types,const char * region)1900 TZDBNameSearchHandler::TZDBNameSearchHandler(uint32_t types, const char* region)
1901 : fTypes(types), fMaxMatchLen(0), fResults(NULL), fRegion(region) {
1902 }
1903
~TZDBNameSearchHandler()1904 TZDBNameSearchHandler::~TZDBNameSearchHandler() {
1905 if (fResults != NULL) {
1906 delete fResults;
1907 }
1908 }
1909
1910 UBool
handleMatch(int32_t matchLength,const CharacterNode * node,UErrorCode & status)1911 TZDBNameSearchHandler::handleMatch(int32_t matchLength, const CharacterNode *node, UErrorCode &status) {
1912 if (U_FAILURE(status)) {
1913 return FALSE;
1914 }
1915
1916 TZDBNameInfo *match = NULL;
1917 TZDBNameInfo *defaultRegionMatch = NULL;
1918
1919 if (node->hasValues()) {
1920 int32_t valuesCount = node->countValues();
1921 for (int32_t i = 0; i < valuesCount; i++) {
1922 TZDBNameInfo *ninfo = (TZDBNameInfo *)node->getValue(i);
1923 if (ninfo == NULL) {
1924 continue;
1925 }
1926 if ((ninfo->type & fTypes) != 0) {
1927 // Some tz database abbreviations are ambiguous. For example,
1928 // CST means either Central Standard Time or China Standard Time.
1929 // Unlike CLDR time zone display names, this implementation
1930 // does not use unique names. And TimeZoneFormat does not expect
1931 // multiple results returned for the same time zone type.
1932 // For this reason, this implementation resolve one among same
1933 // zone type with a same name at this level.
1934 if (ninfo->parseRegions == NULL) {
1935 // parseRegions == null means this is the default metazone
1936 // mapping for the abbreviation.
1937 if (defaultRegionMatch == NULL) {
1938 match = defaultRegionMatch = ninfo;
1939 }
1940 } else {
1941 UBool matchRegion = FALSE;
1942 // non-default metazone mapping for an abbreviation
1943 // comes with applicable regions. For example, the default
1944 // metazone mapping for "CST" is America_Central,
1945 // but if region is one of CN/MO/TW, "CST" is parsed
1946 // as metazone China (China Standard Time).
1947 for (int32_t i = 0; i < ninfo->nRegions; i++) {
1948 const char *region = ninfo->parseRegions[i];
1949 if (uprv_strcmp(fRegion, region) == 0) {
1950 match = ninfo;
1951 matchRegion = TRUE;
1952 break;
1953 }
1954 }
1955 if (matchRegion) {
1956 break;
1957 }
1958 if (match == NULL) {
1959 match = ninfo;
1960 }
1961 }
1962 }
1963 }
1964
1965 if (match != NULL) {
1966 UTimeZoneNameType ntype = match->type;
1967 // Note: Workaround for duplicated standard/daylight names
1968 // The tz database contains a few zones sharing a
1969 // same name for both standard time and daylight saving
1970 // time. For example, Australia/Sydney observes DST,
1971 // but "EST" is used for both standard and daylight.
1972 // When both SHORT_STANDARD and SHORT_DAYLIGHT are included
1973 // in the find operation, we cannot tell which one was
1974 // actually matched.
1975 // TimeZoneFormat#parse returns a matched name type (standard
1976 // or daylight) and DateFormat implementation uses the info to
1977 // to adjust actual time. To avoid false type information,
1978 // this implementation replaces the name type with SHORT_GENERIC.
1979 if (match->ambiguousType
1980 && (ntype == UTZNM_SHORT_STANDARD || ntype == UTZNM_SHORT_DAYLIGHT)
1981 && (fTypes & UTZNM_SHORT_STANDARD) != 0
1982 && (fTypes & UTZNM_SHORT_DAYLIGHT) != 0) {
1983 ntype = UTZNM_SHORT_GENERIC;
1984 }
1985
1986 if (fResults == NULL) {
1987 fResults = new TimeZoneNames::MatchInfoCollection();
1988 if (fResults == NULL) {
1989 status = U_MEMORY_ALLOCATION_ERROR;
1990 }
1991 }
1992 if (U_SUCCESS(status)) {
1993 U_ASSERT(fResults != NULL);
1994 U_ASSERT(match->mzID != NULL);
1995 fResults->addMetaZone(ntype, matchLength, UnicodeString(match->mzID, -1), status);
1996 if (U_SUCCESS(status) && matchLength > fMaxMatchLen) {
1997 fMaxMatchLen = matchLength;
1998 }
1999 }
2000 }
2001 }
2002 return TRUE;
2003 }
2004
2005 TimeZoneNames::MatchInfoCollection*
getMatches(int32_t & maxMatchLen)2006 TZDBNameSearchHandler::getMatches(int32_t& maxMatchLen) {
2007 // give the ownership to the caller
2008 TimeZoneNames::MatchInfoCollection* results = fResults;
2009 maxMatchLen = fMaxMatchLen;
2010
2011 // reset
2012 fResults = NULL;
2013 fMaxMatchLen = 0;
2014 return results;
2015 }
2016
2017 U_CDECL_BEGIN
2018 /**
2019 * Deleter for TZDBNames
2020 */
2021 static void U_CALLCONV
deleteTZDBNames(void * obj)2022 deleteTZDBNames(void *obj) {
2023 if (obj != EMPTY) {
2024 delete (TZDBNames *)obj;
2025 }
2026 }
2027
initTZDBNamesMap(UErrorCode & status)2028 static void U_CALLCONV initTZDBNamesMap(UErrorCode &status) {
2029 gTZDBNamesMap = uhash_open(uhash_hashUChars, uhash_compareUChars, NULL, &status);
2030 if (U_FAILURE(status)) {
2031 gTZDBNamesMap = NULL;
2032 return;
2033 }
2034 // no key deleters for tzdb name maps
2035 uhash_setValueDeleter(gTZDBNamesMap, deleteTZDBNames);
2036 ucln_i18n_registerCleanup(UCLN_I18N_TZDBTIMEZONENAMES, tzdbTimeZoneNames_cleanup);
2037 }
2038
2039 /**
2040 * Deleter for TZDBNameInfo
2041 */
2042 static void U_CALLCONV
deleteTZDBNameInfo(void * obj)2043 deleteTZDBNameInfo(void *obj) {
2044 if (obj != NULL) {
2045 uprv_free(obj);
2046 }
2047 }
2048
prepareFind(UErrorCode & status)2049 static void U_CALLCONV prepareFind(UErrorCode &status) {
2050 if (U_FAILURE(status)) {
2051 return;
2052 }
2053 gTZDBNamesTrie = new TextTrieMap(TRUE, deleteTZDBNameInfo);
2054 if (gTZDBNamesTrie == NULL) {
2055 status = U_MEMORY_ALLOCATION_ERROR;
2056 return;
2057 }
2058
2059 const UnicodeString *mzID;
2060 StringEnumeration *mzIDs = TimeZoneNamesImpl::_getAvailableMetaZoneIDs(status);
2061 if (U_SUCCESS(status)) {
2062 while ((mzID = mzIDs->snext(status)) && U_SUCCESS(status)) {
2063 const TZDBNames *names = TZDBTimeZoneNames::getMetaZoneNames(*mzID, status);
2064 if (U_FAILURE(status)) {
2065 break;
2066 }
2067 if (names == NULL) {
2068 continue;
2069 }
2070 const UChar *std = names->getName(UTZNM_SHORT_STANDARD);
2071 const UChar *dst = names->getName(UTZNM_SHORT_DAYLIGHT);
2072 if (std == NULL && dst == NULL) {
2073 continue;
2074 }
2075 int32_t numRegions = 0;
2076 const char **parseRegions = names->getParseRegions(numRegions);
2077
2078 // The tz database contains a few zones sharing a
2079 // same name for both standard time and daylight saving
2080 // time. For example, Australia/Sydney observes DST,
2081 // but "EST" is used for both standard and daylight.
2082 // we need to store the information for later processing.
2083 UBool ambiguousType = (std != NULL && dst != NULL && u_strcmp(std, dst) == 0);
2084
2085 const UChar *uMzID = ZoneMeta::findMetaZoneID(*mzID);
2086 if (std != NULL) {
2087 TZDBNameInfo *stdInf = (TZDBNameInfo *)uprv_malloc(sizeof(TZDBNameInfo));
2088 if (stdInf == NULL) {
2089 status = U_MEMORY_ALLOCATION_ERROR;
2090 break;
2091 }
2092 stdInf->mzID = uMzID;
2093 stdInf->type = UTZNM_SHORT_STANDARD;
2094 stdInf->ambiguousType = ambiguousType;
2095 stdInf->parseRegions = parseRegions;
2096 stdInf->nRegions = numRegions;
2097 gTZDBNamesTrie->put(std, stdInf, status);
2098 }
2099 if (U_SUCCESS(status) && dst != NULL) {
2100 TZDBNameInfo *dstInf = (TZDBNameInfo *)uprv_malloc(sizeof(TZDBNameInfo));
2101 if (dstInf == NULL) {
2102 status = U_MEMORY_ALLOCATION_ERROR;
2103 break;
2104 }
2105 dstInf->mzID = uMzID;
2106 dstInf->type = UTZNM_SHORT_DAYLIGHT;
2107 dstInf->ambiguousType = ambiguousType;
2108 dstInf->parseRegions = parseRegions;
2109 dstInf->nRegions = numRegions;
2110 gTZDBNamesTrie->put(dst, dstInf, status);
2111 }
2112 }
2113 }
2114 delete mzIDs;
2115
2116 if (U_FAILURE(status)) {
2117 delete gTZDBNamesTrie;
2118 gTZDBNamesTrie = NULL;
2119 return;
2120 }
2121
2122 ucln_i18n_registerCleanup(UCLN_I18N_TZDBTIMEZONENAMES, tzdbTimeZoneNames_cleanup);
2123 }
2124
2125 U_CDECL_END
2126
TZDBTimeZoneNames(const Locale & locale)2127 TZDBTimeZoneNames::TZDBTimeZoneNames(const Locale& locale)
2128 : fLocale(locale) {
2129 UBool useWorld = TRUE;
2130 const char* region = fLocale.getCountry();
2131 int32_t regionLen = uprv_strlen(region);
2132 if (regionLen == 0) {
2133 UErrorCode status = U_ZERO_ERROR;
2134 char loc[ULOC_FULLNAME_CAPACITY];
2135 uloc_addLikelySubtags(fLocale.getName(), loc, sizeof(loc), &status);
2136 regionLen = uloc_getCountry(loc, fRegion, sizeof(fRegion), &status);
2137 if (U_SUCCESS(status) && regionLen < (int32_t)sizeof(fRegion)) {
2138 useWorld = FALSE;
2139 }
2140 } else if (regionLen < (int32_t)sizeof(fRegion)) {
2141 uprv_strcpy(fRegion, region);
2142 useWorld = FALSE;
2143 }
2144 if (useWorld) {
2145 uprv_strcpy(fRegion, "001");
2146 }
2147 }
2148
~TZDBTimeZoneNames()2149 TZDBTimeZoneNames::~TZDBTimeZoneNames() {
2150 }
2151
2152 UBool
operator ==(const TimeZoneNames & other) const2153 TZDBTimeZoneNames::operator==(const TimeZoneNames& other) const {
2154 if (this == &other) {
2155 return TRUE;
2156 }
2157 // No implementation for now
2158 return FALSE;
2159 }
2160
2161 TimeZoneNames*
clone() const2162 TZDBTimeZoneNames::clone() const {
2163 return new TZDBTimeZoneNames(fLocale);
2164 }
2165
2166 StringEnumeration*
getAvailableMetaZoneIDs(UErrorCode & status) const2167 TZDBTimeZoneNames::getAvailableMetaZoneIDs(UErrorCode& status) const {
2168 return TimeZoneNamesImpl::_getAvailableMetaZoneIDs(status);
2169 }
2170
2171 StringEnumeration*
getAvailableMetaZoneIDs(const UnicodeString & tzID,UErrorCode & status) const2172 TZDBTimeZoneNames::getAvailableMetaZoneIDs(const UnicodeString& tzID, UErrorCode& status) const {
2173 return TimeZoneNamesImpl::_getAvailableMetaZoneIDs(tzID, status);
2174 }
2175
2176 UnicodeString&
getMetaZoneID(const UnicodeString & tzID,UDate date,UnicodeString & mzID) const2177 TZDBTimeZoneNames::getMetaZoneID(const UnicodeString& tzID, UDate date, UnicodeString& mzID) const {
2178 return TimeZoneNamesImpl::_getMetaZoneID(tzID, date, mzID);
2179 }
2180
2181 UnicodeString&
getReferenceZoneID(const UnicodeString & mzID,const char * region,UnicodeString & tzID) const2182 TZDBTimeZoneNames::getReferenceZoneID(const UnicodeString& mzID, const char* region, UnicodeString& tzID) const {
2183 return TimeZoneNamesImpl::_getReferenceZoneID(mzID, region, tzID);
2184 }
2185
2186 UnicodeString&
getMetaZoneDisplayName(const UnicodeString & mzID,UTimeZoneNameType type,UnicodeString & name) const2187 TZDBTimeZoneNames::getMetaZoneDisplayName(const UnicodeString& mzID,
2188 UTimeZoneNameType type,
2189 UnicodeString& name) const {
2190 name.setToBogus();
2191 if (mzID.isEmpty()) {
2192 return name;
2193 }
2194
2195 UErrorCode status = U_ZERO_ERROR;
2196 const TZDBNames *tzdbNames = TZDBTimeZoneNames::getMetaZoneNames(mzID, status);
2197 if (U_SUCCESS(status)) {
2198 if (tzdbNames != NULL) {
2199 const UChar *s = tzdbNames->getName(type);
2200 if (s != NULL) {
2201 name.setTo(TRUE, s, -1);
2202 }
2203 }
2204 }
2205
2206 return name;
2207 }
2208
2209 UnicodeString&
getTimeZoneDisplayName(const UnicodeString &,UTimeZoneNameType,UnicodeString & name) const2210 TZDBTimeZoneNames::getTimeZoneDisplayName(const UnicodeString& /* tzID */, UTimeZoneNameType /* type */, UnicodeString& name) const {
2211 // No abbreviations associated a zone directly for now.
2212 name.setToBogus();
2213 return name;
2214 }
2215
2216 TZDBTimeZoneNames::MatchInfoCollection*
find(const UnicodeString & text,int32_t start,uint32_t types,UErrorCode & status) const2217 TZDBTimeZoneNames::find(const UnicodeString& text, int32_t start, uint32_t types, UErrorCode& status) const {
2218 umtx_initOnce(gTZDBNamesTrieInitOnce, &prepareFind, status);
2219 if (U_FAILURE(status)) {
2220 return NULL;
2221 }
2222
2223 TZDBNameSearchHandler handler(types, fRegion);
2224 gTZDBNamesTrie->search(text, start, (TextTrieMapSearchResultHandler *)&handler, status);
2225 if (U_FAILURE(status)) {
2226 return NULL;
2227 }
2228 int32_t maxLen = 0;
2229 return handler.getMatches(maxLen);
2230 }
2231
2232 const TZDBNames*
getMetaZoneNames(const UnicodeString & mzID,UErrorCode & status)2233 TZDBTimeZoneNames::getMetaZoneNames(const UnicodeString& mzID, UErrorCode& status) {
2234 umtx_initOnce(gTZDBNamesMapInitOnce, &initTZDBNamesMap, status);
2235 if (U_FAILURE(status)) {
2236 return NULL;
2237 }
2238
2239 TZDBNames* tzdbNames = NULL;
2240
2241 UChar mzIDKey[ZID_KEY_MAX + 1];
2242 mzID.extract(mzIDKey, ZID_KEY_MAX + 1, status);
2243 U_ASSERT(status == U_ZERO_ERROR); // already checked length above
2244 mzIDKey[mzID.length()] = 0;
2245
2246 umtx_lock(&gTZDBNamesMapLock);
2247 {
2248 void *cacheVal = uhash_get(gTZDBNamesMap, mzIDKey);
2249 if (cacheVal == NULL) {
2250 UResourceBundle *zoneStringsRes = ures_openDirect(U_ICUDATA_ZONE, "tzdbNames", &status);
2251 zoneStringsRes = ures_getByKey(zoneStringsRes, gZoneStrings, zoneStringsRes, &status);
2252 if (U_SUCCESS(status)) {
2253 char key[ZID_KEY_MAX + 1];
2254 mergeTimeZoneKey(mzID, key);
2255 tzdbNames = TZDBNames::createInstance(zoneStringsRes, key);
2256
2257 if (tzdbNames == NULL) {
2258 cacheVal = (void *)EMPTY;
2259 } else {
2260 cacheVal = tzdbNames;
2261 }
2262 // Use the persistent ID as the resource key, so we can
2263 // avoid duplications.
2264 // TODO: Is there a more efficient way, like intern() in Java?
2265 void* newKey = (void*) ZoneMeta::findMetaZoneID(mzID);
2266 if (newKey != NULL) {
2267 uhash_put(gTZDBNamesMap, newKey, cacheVal, &status);
2268 if (U_FAILURE(status)) {
2269 if (tzdbNames != NULL) {
2270 delete tzdbNames;
2271 tzdbNames = NULL;
2272 }
2273 }
2274 } else {
2275 // Should never happen with a valid input
2276 if (tzdbNames != NULL) {
2277 // It's not possible that we get a valid tzdbNames with unknown ID.
2278 // But just in case..
2279 delete tzdbNames;
2280 tzdbNames = NULL;
2281 }
2282 }
2283 }
2284 ures_close(zoneStringsRes);
2285 } else if (cacheVal != EMPTY) {
2286 tzdbNames = (TZDBNames *)cacheVal;
2287 }
2288 }
2289 umtx_unlock(&gTZDBNamesMapLock);
2290
2291 return tzdbNames;
2292 }
2293
2294 U_NAMESPACE_END
2295
2296
2297 #endif /* #if !UCONFIG_NO_FORMATTING */
2298
2299 //eof
2300