]>
Commit | Line | Data |
---|---|---|
f3c0d7a5 A |
1 | // © 2016 and later: Unicode, Inc. and others. |
2 | // License & terms of use: http://www.unicode.org/copyright.html | |
46f4442e A |
3 | /* |
4 | ******************************************************************************* | |
57a6839d A |
5 | * Copyright (C) 2007-2014, International Business Machines Corporation and |
6 | * others. All Rights Reserved. | |
46f4442e A |
7 | ******************************************************************************* |
8 | */ | |
9 | ||
10 | #include "unicode/utypes.h" | |
11 | ||
12 | #if !UCONFIG_NO_FORMATTING | |
13 | ||
14 | #include "zonemeta.h" | |
15 | ||
16 | #include "unicode/timezone.h" | |
17 | #include "unicode/ustring.h" | |
18 | #include "unicode/putil.h" | |
4388f060 | 19 | #include "unicode/simpletz.h" |
f3c0d7a5 | 20 | #include "unicode/strenum.h" |
46f4442e A |
21 | #include "umutex.h" |
22 | #include "uvector.h" | |
23 | #include "cmemory.h" | |
24 | #include "gregoimp.h" | |
25 | #include "cstring.h" | |
26 | #include "ucln_in.h" | |
729e4ab9 | 27 | #include "uassert.h" |
4388f060 A |
28 | #include "uresimp.h" |
29 | #include "uhash.h" | |
30 | #include "olsontz.h" | |
f3c0d7a5 | 31 | #include "uinvchar.h" |
46f4442e | 32 | |
340931cb | 33 | static icu::UMutex gZoneMetaLock; |
729e4ab9 | 34 | |
4388f060 A |
35 | // CLDR Canonical ID mapping table |
36 | static UHashtable *gCanonicalIDCache = NULL; | |
57a6839d | 37 | static icu::UInitOnce gCanonicalIDCacheInitOnce = U_INITONCE_INITIALIZER; |
4388f060 | 38 | |
729e4ab9 | 39 | // Metazone mapping table |
46f4442e | 40 | static UHashtable *gOlsonToMeta = NULL; |
57a6839d | 41 | static icu::UInitOnce gOlsonToMetaInitOnce = U_INITONCE_INITIALIZER; |
46f4442e | 42 | |
4388f060 A |
43 | // Available metazone IDs vector and table |
44 | static icu::UVector *gMetaZoneIDs = NULL; | |
45 | static UHashtable *gMetaZoneIDTable = NULL; | |
57a6839d | 46 | static icu::UInitOnce gMetaZoneIDsInitOnce = U_INITONCE_INITIALIZER; |
4388f060 | 47 | |
729e4ab9 | 48 | // Country info vectors |
4388f060 A |
49 | static icu::UVector *gSingleZoneCountries = NULL; |
50 | static icu::UVector *gMultiZonesCountries = NULL; | |
57a6839d | 51 | static icu::UInitOnce gCountryInfoVectorsInitOnce = U_INITONCE_INITIALIZER; |
46f4442e | 52 | |
729e4ab9 | 53 | U_CDECL_BEGIN |
46f4442e | 54 | |
46f4442e A |
55 | /** |
56 | * Cleanup callback func | |
57 | */ | |
58 | static UBool U_CALLCONV zoneMeta_cleanup(void) | |
59 | { | |
4388f060 A |
60 | if (gCanonicalIDCache != NULL) { |
61 | uhash_close(gCanonicalIDCache); | |
62 | gCanonicalIDCache = NULL; | |
63 | } | |
57a6839d | 64 | gCanonicalIDCacheInitOnce.reset(); |
46f4442e | 65 | |
46f4442e A |
66 | if (gOlsonToMeta != NULL) { |
67 | uhash_close(gOlsonToMeta); | |
68 | gOlsonToMeta = NULL; | |
69 | } | |
57a6839d | 70 | gOlsonToMetaInitOnce.reset(); |
46f4442e | 71 | |
4388f060 A |
72 | if (gMetaZoneIDTable != NULL) { |
73 | uhash_close(gMetaZoneIDTable); | |
57a6839d | 74 | gMetaZoneIDTable = NULL; |
4388f060 A |
75 | } |
76 | // delete after closing gMetaZoneIDTable, because it holds | |
77 | // value objects held by the hashtable | |
78 | delete gMetaZoneIDs; | |
57a6839d A |
79 | gMetaZoneIDs = NULL; |
80 | gMetaZoneIDsInitOnce.reset(); | |
4388f060 | 81 | |
729e4ab9 | 82 | delete gSingleZoneCountries; |
57a6839d | 83 | gSingleZoneCountries = NULL; |
729e4ab9 | 84 | delete gMultiZonesCountries; |
57a6839d A |
85 | gMultiZonesCountries = NULL; |
86 | gCountryInfoVectorsInitOnce.reset(); | |
46f4442e A |
87 | |
88 | return TRUE; | |
89 | } | |
90 | ||
91 | /** | |
92 | * Deleter for UChar* string | |
93 | */ | |
94 | static void U_CALLCONV | |
95 | deleteUCharString(void *obj) { | |
96 | UChar *entry = (UChar*)obj; | |
97 | uprv_free(entry); | |
98 | } | |
99 | ||
100 | /** | |
101 | * Deleter for UVector | |
102 | */ | |
103 | static void U_CALLCONV | |
104 | deleteUVector(void *obj) { | |
4388f060 | 105 | delete (icu::UVector*) obj; |
46f4442e A |
106 | } |
107 | ||
46f4442e A |
108 | /** |
109 | * Deleter for OlsonToMetaMappingEntry | |
110 | */ | |
111 | static void U_CALLCONV | |
112 | deleteOlsonToMetaMappingEntry(void *obj) { | |
4388f060 | 113 | icu::OlsonToMetaMappingEntry *entry = (icu::OlsonToMetaMappingEntry*)obj; |
46f4442e A |
114 | uprv_free(entry); |
115 | } | |
116 | ||
46f4442e A |
117 | U_CDECL_END |
118 | ||
119 | U_NAMESPACE_BEGIN | |
120 | ||
121 | #define ZID_KEY_MAX 128 | |
46f4442e | 122 | |
729e4ab9 A |
123 | static const char gMetaZones[] = "metaZones"; |
124 | static const char gMetazoneInfo[] = "metazoneInfo"; | |
46f4442e | 125 | static const char gMapTimezonesTag[] = "mapTimezones"; |
46f4442e | 126 | |
51004dcb | 127 | static const char gKeyTypeData[] = "keyTypeData"; |
729e4ab9 A |
128 | static const char gTypeAliasTag[] = "typeAlias"; |
129 | static const char gTypeMapTag[] = "typeMap"; | |
130 | static const char gTimezoneTag[] = "timezone"; | |
46f4442e | 131 | |
51004dcb A |
132 | static const char gPrimaryZonesTag[] = "primaryZones"; |
133 | ||
729e4ab9 | 134 | static const char gWorldTag[] = "001"; |
46f4442e A |
135 | |
136 | static const UChar gWorld[] = {0x30, 0x30, 0x31, 0x00}; // "001" | |
137 | ||
729e4ab9 A |
138 | static const UChar gDefaultFrom[] = {0x31, 0x39, 0x37, 0x30, 0x2D, 0x30, 0x31, 0x2D, 0x30, 0x31, |
139 | 0x20, 0x30, 0x30, 0x3A, 0x30, 0x30, 0x00}; // "1970-01-01 00:00" | |
140 | static const UChar gDefaultTo[] = {0x39, 0x39, 0x39, 0x39, 0x2D, 0x31, 0x32, 0x2D, 0x33, 0x31, | |
141 | 0x20, 0x32, 0x33, 0x3A, 0x35, 0x39, 0x00}; // "9999-12-31 23:59" | |
142 | ||
4388f060 A |
143 | static const UChar gCustomTzPrefix[] = {0x47, 0x4D, 0x54, 0}; // "GMT" |
144 | ||
46f4442e A |
145 | #define ASCII_DIGIT(c) (((c)>=0x30 && (c)<=0x39) ? (c)-0x30 : -1) |
146 | ||
147 | /* | |
148 | * Convert a date string used by metazone mappings to UDate. | |
149 | * The format used by CLDR metazone mapping is "yyyy-MM-dd HH:mm". | |
150 | */ | |
151 | static UDate | |
152 | parseDate (const UChar *text, UErrorCode &status) { | |
153 | if (U_FAILURE(status)) { | |
154 | return 0; | |
155 | } | |
156 | int32_t len = u_strlen(text); | |
157 | if (len != 16 && len != 10) { | |
158 | // It must be yyyy-MM-dd HH:mm (length 16) or yyyy-MM-dd (length 10) | |
159 | status = U_INVALID_FORMAT_ERROR; | |
160 | return 0; | |
161 | } | |
162 | ||
163 | int32_t year = 0, month = 0, day = 0, hour = 0, min = 0, n; | |
164 | int32_t idx; | |
165 | ||
166 | // "yyyy" (0 - 3) | |
167 | for (idx = 0; idx <= 3 && U_SUCCESS(status); idx++) { | |
168 | n = ASCII_DIGIT((int32_t)text[idx]); | |
169 | if (n >= 0) { | |
170 | year = 10*year + n; | |
171 | } else { | |
172 | status = U_INVALID_FORMAT_ERROR; | |
173 | } | |
174 | } | |
175 | // "MM" (5 - 6) | |
176 | for (idx = 5; idx <= 6 && U_SUCCESS(status); idx++) { | |
177 | n = ASCII_DIGIT((int32_t)text[idx]); | |
178 | if (n >= 0) { | |
179 | month = 10*month + n; | |
180 | } else { | |
181 | status = U_INVALID_FORMAT_ERROR; | |
182 | } | |
183 | } | |
184 | // "dd" (8 - 9) | |
185 | for (idx = 8; idx <= 9 && U_SUCCESS(status); idx++) { | |
186 | n = ASCII_DIGIT((int32_t)text[idx]); | |
187 | if (n >= 0) { | |
188 | day = 10*day + n; | |
189 | } else { | |
190 | status = U_INVALID_FORMAT_ERROR; | |
191 | } | |
192 | } | |
193 | if (len == 16) { | |
194 | // "HH" (11 - 12) | |
195 | for (idx = 11; idx <= 12 && U_SUCCESS(status); idx++) { | |
196 | n = ASCII_DIGIT((int32_t)text[idx]); | |
197 | if (n >= 0) { | |
198 | hour = 10*hour + n; | |
199 | } else { | |
200 | status = U_INVALID_FORMAT_ERROR; | |
201 | } | |
202 | } | |
203 | // "mm" (14 - 15) | |
204 | for (idx = 14; idx <= 15 && U_SUCCESS(status); idx++) { | |
205 | n = ASCII_DIGIT((int32_t)text[idx]); | |
206 | if (n >= 0) { | |
207 | min = 10*min + n; | |
208 | } else { | |
209 | status = U_INVALID_FORMAT_ERROR; | |
210 | } | |
211 | } | |
212 | } | |
213 | ||
214 | if (U_SUCCESS(status)) { | |
215 | UDate date = Grego::fieldsToDay(year, month - 1, day) * U_MILLIS_PER_DAY | |
216 | + hour * U_MILLIS_PER_HOUR + min * U_MILLIS_PER_MINUTE; | |
217 | return date; | |
218 | } | |
219 | return 0; | |
220 | } | |
221 | ||
57a6839d A |
222 | static void U_CALLCONV initCanonicalIDCache(UErrorCode &status) { |
223 | gCanonicalIDCache = uhash_open(uhash_hashUChars, uhash_compareUChars, NULL, &status); | |
224 | if (gCanonicalIDCache == NULL) { | |
225 | status = U_MEMORY_ALLOCATION_ERROR; | |
226 | } | |
227 | if (U_FAILURE(status)) { | |
228 | gCanonicalIDCache = NULL; | |
229 | } | |
230 | // No key/value deleters - keys/values are from a resource bundle | |
231 | ucln_i18n_registerCleanup(UCLN_I18N_ZONEMETA, zoneMeta_cleanup); | |
232 | } | |
233 | ||
234 | ||
4388f060 A |
235 | const UChar* U_EXPORT2 |
236 | ZoneMeta::getCanonicalCLDRID(const UnicodeString &tzid, UErrorCode& status) { | |
237 | if (U_FAILURE(status)) { | |
238 | return NULL; | |
239 | } | |
240 | ||
57a6839d | 241 | if (tzid.isBogus() || tzid.length() > ZID_KEY_MAX) { |
729e4ab9 | 242 | status = U_ILLEGAL_ARGUMENT_ERROR; |
4388f060 | 243 | return NULL; |
46f4442e | 244 | } |
46f4442e | 245 | |
4388f060 | 246 | // Checking the cached results |
57a6839d A |
247 | umtx_initOnce(gCanonicalIDCacheInitOnce, &initCanonicalIDCache, status); |
248 | if (U_FAILURE(status)) { | |
249 | return NULL; | |
4388f060 A |
250 | } |
251 | ||
252 | const UChar *canonicalID = NULL; | |
253 | ||
254 | UErrorCode tmpStatus = U_ZERO_ERROR; | |
255 | UChar utzid[ZID_KEY_MAX + 1]; | |
256 | tzid.extract(utzid, ZID_KEY_MAX + 1, tmpStatus); | |
257 | U_ASSERT(tmpStatus == U_ZERO_ERROR); // we checked the length of tzid already | |
258 | ||
f3c0d7a5 A |
259 | if (!uprv_isInvariantUString(utzid, -1)) { |
260 | // All of known tz IDs are only containing ASCII invariant characters. | |
261 | status = U_ILLEGAL_ARGUMENT_ERROR; | |
262 | return NULL; | |
263 | } | |
264 | ||
4388f060 | 265 | // Check if it was already cached |
340931cb | 266 | umtx_lock(&gZoneMetaLock); |
4388f060 A |
267 | { |
268 | canonicalID = (const UChar *)uhash_get(gCanonicalIDCache, utzid); | |
269 | } | |
340931cb | 270 | umtx_unlock(&gZoneMetaLock); |
4388f060 A |
271 | |
272 | if (canonicalID != NULL) { | |
273 | return canonicalID; | |
274 | } | |
275 | ||
276 | // If not, resolve CLDR canonical ID with resource data | |
277 | UBool isInputCanonical = FALSE; | |
278 | char id[ZID_KEY_MAX + 1]; | |
b331163b | 279 | tzid.extract(0, 0x7fffffff, id, UPRV_LENGTHOF(id), US_INV); |
46f4442e | 280 | |
729e4ab9 A |
281 | // replace '/' with ':' |
282 | char *p = id; | |
283 | while (*p++) { | |
284 | if (*p == '/') { | |
285 | *p = ':'; | |
46f4442e | 286 | } |
729e4ab9 | 287 | } |
46f4442e | 288 | |
51004dcb | 289 | UResourceBundle *top = ures_openDirect(NULL, gKeyTypeData, &tmpStatus); |
729e4ab9 A |
290 | UResourceBundle *rb = ures_getByKey(top, gTypeMapTag, NULL, &tmpStatus); |
291 | ures_getByKey(rb, gTimezoneTag, rb, &tmpStatus); | |
292 | ures_getByKey(rb, id, rb, &tmpStatus); | |
293 | if (U_SUCCESS(tmpStatus)) { | |
4388f060 A |
294 | // type entry (canonical) found |
295 | // the input is the canonical ID. resolve to const UChar* | |
296 | canonicalID = TimeZone::findID(tzid); | |
297 | isInputCanonical = TRUE; | |
729e4ab9 | 298 | } |
46f4442e | 299 | |
4388f060 A |
300 | if (canonicalID == NULL) { |
301 | // If a map element not found, then look for an alias | |
302 | tmpStatus = U_ZERO_ERROR; | |
303 | ures_getByKey(top, gTypeAliasTag, rb, &tmpStatus); | |
304 | ures_getByKey(rb, gTimezoneTag, rb, &tmpStatus); | |
305 | const UChar *canonical = ures_getStringByKey(rb,id,NULL,&tmpStatus); | |
306 | if (U_SUCCESS(tmpStatus)) { | |
307 | // canonical map found | |
308 | canonicalID = canonical; | |
309 | } | |
46f4442e | 310 | |
4388f060 A |
311 | if (canonicalID == NULL) { |
312 | // Dereference the input ID using the tz data | |
313 | const UChar *derefer = TimeZone::dereferOlsonLink(tzid); | |
314 | if (derefer == NULL) { | |
315 | status = U_ILLEGAL_ARGUMENT_ERROR; | |
316 | } else { | |
57a6839d | 317 | int32_t len = u_strlen(derefer); |
4388f060 A |
318 | u_UCharsToChars(derefer,id,len); |
319 | id[len] = (char) 0; // Make sure it is null terminated. | |
320 | ||
321 | // replace '/' with ':' | |
3d1f044b A |
322 | char *q = id; |
323 | while (*q++) { | |
324 | if (*q == '/') { | |
325 | *q = ':'; | |
4388f060 A |
326 | } |
327 | } | |
46f4442e | 328 | |
4388f060 A |
329 | // If a dereference turned something up then look for an alias. |
330 | // rb still points to the alias table, so we don't have to go looking | |
331 | // for it. | |
332 | tmpStatus = U_ZERO_ERROR; | |
333 | canonical = ures_getStringByKey(rb,id,NULL,&tmpStatus); | |
334 | if (U_SUCCESS(tmpStatus)) { | |
335 | // canonical map for the dereferenced ID found | |
336 | canonicalID = canonical; | |
337 | } else { | |
338 | canonicalID = derefer; | |
339 | isInputCanonical = TRUE; | |
340 | } | |
46f4442e A |
341 | } |
342 | } | |
4388f060 A |
343 | } |
344 | ures_close(rb); | |
345 | ures_close(top); | |
46f4442e | 346 | |
4388f060 A |
347 | if (U_SUCCESS(status)) { |
348 | U_ASSERT(canonicalID != NULL); // canocanilD must be non-NULL here | |
349 | ||
350 | // Put the resolved canonical ID to the cache | |
340931cb | 351 | umtx_lock(&gZoneMetaLock); |
4388f060 A |
352 | { |
353 | const UChar* idInCache = (const UChar *)uhash_get(gCanonicalIDCache, utzid); | |
354 | if (idInCache == NULL) { | |
355 | const UChar* key = ZoneMeta::findTimeZoneID(tzid); | |
356 | U_ASSERT(key != NULL); | |
357 | if (key != NULL) { | |
358 | idInCache = (const UChar *)uhash_put(gCanonicalIDCache, (void *)key, (void *)canonicalID, &status); | |
359 | U_ASSERT(idInCache == NULL); | |
360 | } | |
361 | } | |
362 | if (U_SUCCESS(status) && isInputCanonical) { | |
363 | // Also put canonical ID itself into the cache if not exist | |
364 | const UChar *canonicalInCache = (const UChar*)uhash_get(gCanonicalIDCache, canonicalID); | |
365 | if (canonicalInCache == NULL) { | |
366 | canonicalInCache = (const UChar *)uhash_put(gCanonicalIDCache, (void *)canonicalID, (void *)canonicalID, &status); | |
367 | U_ASSERT(canonicalInCache == NULL); | |
368 | } | |
369 | } | |
46f4442e | 370 | } |
340931cb | 371 | umtx_unlock(&gZoneMetaLock); |
46f4442e A |
372 | } |
373 | ||
4388f060 | 374 | return canonicalID; |
46f4442e A |
375 | } |
376 | ||
4388f060 A |
377 | UnicodeString& U_EXPORT2 |
378 | ZoneMeta::getCanonicalCLDRID(const UnicodeString &tzid, UnicodeString &systemID, UErrorCode& status) { | |
379 | const UChar *canonicalID = getCanonicalCLDRID(tzid, status); | |
380 | if (U_FAILURE(status) || canonicalID == NULL) { | |
381 | systemID.setToBogus(); | |
382 | return systemID; | |
383 | } | |
384 | systemID.setTo(TRUE, canonicalID, -1); | |
385 | return systemID; | |
386 | } | |
387 | ||
388 | const UChar* U_EXPORT2 | |
389 | ZoneMeta::getCanonicalCLDRID(const TimeZone& tz) { | |
390 | if (dynamic_cast<const OlsonTimeZone *>(&tz) != NULL) { | |
391 | // short cut for OlsonTimeZone | |
392 | const OlsonTimeZone *otz = (const OlsonTimeZone*)&tz; | |
393 | return otz->getCanonicalID(); | |
394 | } | |
395 | UErrorCode status = U_ZERO_ERROR; | |
396 | UnicodeString tzID; | |
397 | return getCanonicalCLDRID(tz.getID(tzID), status); | |
398 | } | |
399 | ||
57a6839d A |
400 | static void U_CALLCONV countryInfoVectorsInit(UErrorCode &status) { |
401 | // Create empty vectors | |
402 | // No deleters for these UVectors, it's a reference to a resource bundle string. | |
403 | gSingleZoneCountries = new UVector(NULL, uhash_compareUChars, status); | |
404 | if (gSingleZoneCountries == NULL) { | |
405 | status = U_MEMORY_ALLOCATION_ERROR; | |
406 | } | |
407 | gMultiZonesCountries = new UVector(NULL, uhash_compareUChars, status); | |
408 | if (gMultiZonesCountries == NULL) { | |
409 | status = U_MEMORY_ALLOCATION_ERROR; | |
410 | } | |
411 | ||
412 | if (U_FAILURE(status)) { | |
413 | delete gSingleZoneCountries; | |
414 | delete gMultiZonesCountries; | |
415 | gSingleZoneCountries = NULL; | |
416 | gMultiZonesCountries = NULL; | |
417 | } | |
418 | ucln_i18n_registerCleanup(UCLN_I18N_ZONEMETA, zoneMeta_cleanup); | |
419 | } | |
420 | ||
421 | ||
729e4ab9 | 422 | UnicodeString& U_EXPORT2 |
51004dcb A |
423 | ZoneMeta::getCanonicalCountry(const UnicodeString &tzid, UnicodeString &country, UBool *isPrimary /* = NULL */) { |
424 | if (isPrimary != NULL) { | |
425 | *isPrimary = FALSE; | |
46f4442e | 426 | } |
46f4442e | 427 | |
729e4ab9 | 428 | const UChar *region = TimeZone::getRegion(tzid); |
51004dcb A |
429 | if (region != NULL && u_strcmp(gWorld, region) != 0) { |
430 | country.setTo(region, -1); | |
431 | } else { | |
4388f060 | 432 | country.setToBogus(); |
729e4ab9 | 433 | return country; |
46f4442e A |
434 | } |
435 | ||
51004dcb A |
436 | if (isPrimary != NULL) { |
437 | char regionBuf[] = {0, 0, 0}; | |
438 | ||
439 | // Checking the cached results | |
440 | UErrorCode status = U_ZERO_ERROR; | |
57a6839d A |
441 | umtx_initOnce(gCountryInfoVectorsInitOnce, &countryInfoVectorsInit, status); |
442 | if (U_FAILURE(status)) { | |
443 | return country; | |
46f4442e A |
444 | } |
445 | ||
51004dcb A |
446 | // Check if it was already cached |
447 | UBool cached = FALSE; | |
448 | UBool singleZone = FALSE; | |
340931cb | 449 | umtx_lock(&gZoneMetaLock); |
51004dcb A |
450 | { |
451 | singleZone = cached = gSingleZoneCountries->contains((void*)region); | |
452 | if (!cached) { | |
453 | cached = gMultiZonesCountries->contains((void*)region); | |
454 | } | |
729e4ab9 | 455 | } |
340931cb | 456 | umtx_unlock(&gZoneMetaLock); |
46f4442e | 457 | |
51004dcb A |
458 | if (!cached) { |
459 | // We need to go through all zones associated with the region. | |
460 | // This is relatively heavy operation. | |
729e4ab9 | 461 | |
51004dcb | 462 | U_ASSERT(u_strlen(region) == 2); |
729e4ab9 | 463 | |
51004dcb | 464 | u_UCharsToChars(region, regionBuf, 2); |
729e4ab9 | 465 | |
51004dcb A |
466 | StringEnumeration *ids = TimeZone::createTimeZoneIDEnumeration(UCAL_ZONE_TYPE_CANONICAL_LOCATION, regionBuf, NULL, status); |
467 | int32_t idsLen = ids->count(status); | |
468 | if (U_SUCCESS(status) && idsLen == 1) { | |
469 | // only the single zone is available for the region | |
470 | singleZone = TRUE; | |
471 | } | |
472 | delete ids; | |
473 | ||
474 | // Cache the result | |
340931cb | 475 | umtx_lock(&gZoneMetaLock); |
51004dcb A |
476 | { |
477 | UErrorCode ec = U_ZERO_ERROR; | |
478 | if (singleZone) { | |
479 | if (!gSingleZoneCountries->contains((void*)region)) { | |
480 | gSingleZoneCountries->addElement((void*)region, ec); | |
481 | } | |
482 | } else { | |
483 | if (!gMultiZonesCountries->contains((void*)region)) { | |
484 | gMultiZonesCountries->addElement((void*)region, ec); | |
485 | } | |
486 | } | |
487 | } | |
340931cb | 488 | umtx_unlock(&gZoneMetaLock); |
729e4ab9 | 489 | } |
729e4ab9 | 490 | |
51004dcb A |
491 | if (singleZone) { |
492 | *isPrimary = TRUE; | |
493 | } else { | |
494 | // Note: We may cache the primary zone map in future. | |
495 | ||
496 | // Even a country has multiple zones, one of them might be | |
497 | // dominant and treated as a primary zone | |
498 | int32_t idLen = 0; | |
499 | if (regionBuf[0] == 0) { | |
500 | u_UCharsToChars(region, regionBuf, 2); | |
501 | } | |
502 | ||
503 | UResourceBundle *rb = ures_openDirect(NULL, gMetaZones, &status); | |
504 | ures_getByKey(rb, gPrimaryZonesTag, rb, &status); | |
505 | const UChar *primaryZone = ures_getStringByKey(rb, regionBuf, &idLen, &status); | |
506 | if (U_SUCCESS(status)) { | |
507 | if (tzid.compare(primaryZone, idLen) == 0) { | |
508 | *isPrimary = TRUE; | |
509 | } else { | |
510 | // The given ID might not be a canonical ID | |
511 | UnicodeString canonicalID; | |
512 | TimeZone::getCanonicalID(tzid, canonicalID, status); | |
513 | if (U_SUCCESS(status) && canonicalID.compare(primaryZone, idLen) == 0) { | |
514 | *isPrimary = TRUE; | |
515 | } | |
729e4ab9 | 516 | } |
46f4442e | 517 | } |
51004dcb | 518 | ures_close(rb); |
46f4442e A |
519 | } |
520 | } | |
521 | ||
729e4ab9 A |
522 | return country; |
523 | } | |
524 | ||
525 | UnicodeString& U_EXPORT2 | |
526 | ZoneMeta::getMetazoneID(const UnicodeString &tzid, UDate date, UnicodeString &result) { | |
527 | UBool isSet = FALSE; | |
528 | const UVector *mappings = getMetazoneMappings(tzid); | |
529 | if (mappings != NULL) { | |
530 | for (int32_t i = 0; i < mappings->size(); i++) { | |
531 | OlsonToMetaMappingEntry *mzm = (OlsonToMetaMappingEntry*)mappings->elementAt(i); | |
532 | if (mzm->from <= date && mzm->to > date) { | |
533 | result.setTo(mzm->mzid, -1); | |
534 | isSet = TRUE; | |
535 | break; | |
536 | } | |
537 | } | |
46f4442e | 538 | } |
729e4ab9 | 539 | if (!isSet) { |
4388f060 | 540 | result.setToBogus(); |
46f4442e | 541 | } |
729e4ab9 | 542 | return result; |
46f4442e A |
543 | } |
544 | ||
57a6839d A |
545 | static void U_CALLCONV olsonToMetaInit(UErrorCode &status) { |
546 | U_ASSERT(gOlsonToMeta == NULL); | |
547 | ucln_i18n_registerCleanup(UCLN_I18N_ZONEMETA, zoneMeta_cleanup); | |
548 | gOlsonToMeta = uhash_open(uhash_hashUChars, uhash_compareUChars, NULL, &status); | |
549 | if (U_FAILURE(status)) { | |
550 | gOlsonToMeta = NULL; | |
551 | } else { | |
552 | uhash_setKeyDeleter(gOlsonToMeta, deleteUCharString); | |
553 | uhash_setValueDeleter(gOlsonToMeta, deleteUVector); | |
554 | } | |
555 | } | |
556 | ||
557 | ||
729e4ab9 A |
558 | const UVector* U_EXPORT2 |
559 | ZoneMeta::getMetazoneMappings(const UnicodeString &tzid) { | |
46f4442e | 560 | UErrorCode status = U_ZERO_ERROR; |
4388f060 A |
561 | UChar tzidUChars[ZID_KEY_MAX + 1]; |
562 | tzid.extract(tzidUChars, ZID_KEY_MAX + 1, status); | |
729e4ab9 | 563 | if (U_FAILURE(status) || status == U_STRING_NOT_TERMINATED_WARNING) { |
46f4442e A |
564 | return NULL; |
565 | } | |
46f4442e | 566 | |
57a6839d A |
567 | umtx_initOnce(gOlsonToMetaInitOnce, &olsonToMetaInit, status); |
568 | if (U_FAILURE(status)) { | |
569 | return NULL; | |
46f4442e | 570 | } |
46f4442e | 571 | |
729e4ab9 A |
572 | // get the mapping from cache |
573 | const UVector *result = NULL; | |
46f4442e | 574 | |
340931cb | 575 | umtx_lock(&gZoneMetaLock); |
729e4ab9 A |
576 | { |
577 | result = (UVector*) uhash_get(gOlsonToMeta, tzidUChars); | |
46f4442e | 578 | } |
340931cb | 579 | umtx_unlock(&gZoneMetaLock); |
46f4442e | 580 | |
729e4ab9 A |
581 | if (result != NULL) { |
582 | return result; | |
46f4442e | 583 | } |
46f4442e | 584 | |
729e4ab9 A |
585 | // miss the cache - create new one |
586 | UVector *tmpResult = createMetazoneMappings(tzid); | |
587 | if (tmpResult == NULL) { | |
588 | // not available | |
589 | return NULL; | |
46f4442e | 590 | } |
46f4442e | 591 | |
729e4ab9 | 592 | // put the new one into the cache |
340931cb | 593 | umtx_lock(&gZoneMetaLock); |
729e4ab9 A |
594 | { |
595 | // make sure it's already created | |
596 | result = (UVector*) uhash_get(gOlsonToMeta, tzidUChars); | |
597 | if (result == NULL) { | |
598 | // add the one just created | |
599 | int32_t tzidLen = tzid.length() + 1; | |
600 | UChar *key = (UChar*)uprv_malloc(tzidLen * sizeof(UChar)); | |
601 | if (key == NULL) { | |
602 | // memory allocation error.. just return NULL | |
603 | result = NULL; | |
604 | delete tmpResult; | |
605 | } else { | |
606 | tzid.extract(key, tzidLen, status); | |
607 | uhash_put(gOlsonToMeta, key, tmpResult, &status); | |
608 | if (U_FAILURE(status)) { | |
609 | // delete the mapping | |
610 | result = NULL; | |
611 | delete tmpResult; | |
612 | } else { | |
613 | result = tmpResult; | |
614 | } | |
615 | } | |
616 | } else { | |
617 | // another thread already put the one | |
618 | delete tmpResult; | |
619 | } | |
46f4442e | 620 | } |
340931cb | 621 | umtx_unlock(&gZoneMetaLock); |
46f4442e | 622 | |
729e4ab9 | 623 | return result; |
46f4442e A |
624 | } |
625 | ||
729e4ab9 A |
626 | UVector* |
627 | ZoneMeta::createMetazoneMappings(const UnicodeString &tzid) { | |
628 | UVector *mzMappings = NULL; | |
629 | UErrorCode status = U_ZERO_ERROR; | |
46f4442e | 630 | |
729e4ab9 A |
631 | UnicodeString canonicalID; |
632 | UResourceBundle *rb = ures_openDirect(NULL, gMetaZones, &status); | |
633 | ures_getByKey(rb, gMetazoneInfo, rb, &status); | |
4388f060 | 634 | getCanonicalCLDRID(tzid, canonicalID, status); |
729e4ab9 A |
635 | |
636 | if (U_SUCCESS(status)) { | |
4388f060 A |
637 | char tzKey[ZID_KEY_MAX + 1]; |
638 | int32_t tzKeyLen = canonicalID.extract(0, canonicalID.length(), tzKey, sizeof(tzKey), US_INV); | |
639 | tzKey[tzKeyLen] = 0; | |
46f4442e | 640 | |
729e4ab9 A |
641 | // tzid keys are using ':' as separators |
642 | char *p = tzKey; | |
643 | while (*p) { | |
644 | if (*p == '/') { | |
645 | *p = ':'; | |
646 | } | |
647 | p++; | |
46f4442e | 648 | } |
46f4442e | 649 | |
729e4ab9 | 650 | ures_getByKey(rb, tzKey, rb, &status); |
46f4442e | 651 | |
729e4ab9 A |
652 | if (U_SUCCESS(status)) { |
653 | UResourceBundle *mz = NULL; | |
654 | while (ures_hasNext(rb)) { | |
655 | mz = ures_getNextResource(rb, mz, &status); | |
46f4442e | 656 | |
729e4ab9 A |
657 | const UChar *mz_name = ures_getStringByIndex(mz, 0, NULL, &status); |
658 | const UChar *mz_from = gDefaultFrom; | |
659 | const UChar *mz_to = gDefaultTo; | |
46f4442e | 660 | |
729e4ab9 A |
661 | if (ures_getSize(mz) == 3) { |
662 | mz_from = ures_getStringByIndex(mz, 1, NULL, &status); | |
663 | mz_to = ures_getStringByIndex(mz, 2, NULL, &status); | |
664 | } | |
665 | ||
666 | if(U_FAILURE(status)){ | |
667 | status = U_ZERO_ERROR; | |
668 | continue; | |
669 | } | |
670 | // We do not want to use SimpleDateformat to parse boundary dates, | |
671 | // because this code could be triggered by the initialization code | |
672 | // used by SimpleDateFormat. | |
673 | UDate from = parseDate(mz_from, status); | |
674 | UDate to = parseDate(mz_to, status); | |
675 | if (U_FAILURE(status)) { | |
676 | status = U_ZERO_ERROR; | |
677 | continue; | |
678 | } | |
679 | ||
680 | OlsonToMetaMappingEntry *entry = (OlsonToMetaMappingEntry*)uprv_malloc(sizeof(OlsonToMetaMappingEntry)); | |
681 | if (entry == NULL) { | |
682 | status = U_MEMORY_ALLOCATION_ERROR; | |
46f4442e A |
683 | break; |
684 | } | |
729e4ab9 A |
685 | entry->mzid = mz_name; |
686 | entry->from = from; | |
687 | entry->to = to; | |
688 | ||
689 | if (mzMappings == NULL) { | |
690 | mzMappings = new UVector(deleteOlsonToMetaMappingEntry, NULL, status); | |
691 | if (U_FAILURE(status)) { | |
692 | delete mzMappings; | |
0f5d89e8 | 693 | mzMappings = NULL; |
729e4ab9 A |
694 | uprv_free(entry); |
695 | break; | |
696 | } | |
697 | } | |
698 | ||
699 | mzMappings->addElement(entry, status); | |
700 | if (U_FAILURE(status)) { | |
701 | break; | |
702 | } | |
703 | } | |
704 | ures_close(mz); | |
705 | if (U_FAILURE(status)) { | |
706 | if (mzMappings != NULL) { | |
707 | delete mzMappings; | |
708 | mzMappings = NULL; | |
709 | } | |
46f4442e A |
710 | } |
711 | } | |
46f4442e | 712 | } |
729e4ab9 A |
713 | ures_close(rb); |
714 | return mzMappings; | |
46f4442e A |
715 | } |
716 | ||
717 | UnicodeString& U_EXPORT2 | |
729e4ab9 A |
718 | ZoneMeta::getZoneIdByMetazone(const UnicodeString &mzid, const UnicodeString ®ion, UnicodeString &result) { |
719 | UErrorCode status = U_ZERO_ERROR; | |
720 | const UChar *tzid = NULL; | |
721 | int32_t tzidLen = 0; | |
722 | char keyBuf[ZID_KEY_MAX + 1]; | |
723 | int32_t keyLen = 0; | |
724 | ||
57a6839d | 725 | if (mzid.isBogus() || mzid.length() > ZID_KEY_MAX) { |
4388f060 | 726 | result.setToBogus(); |
729e4ab9 | 727 | return result; |
46f4442e | 728 | } |
46f4442e | 729 | |
4388f060 A |
730 | keyLen = mzid.extract(0, mzid.length(), keyBuf, ZID_KEY_MAX + 1, US_INV); |
731 | keyBuf[keyLen] = 0; | |
46f4442e | 732 | |
729e4ab9 A |
733 | UResourceBundle *rb = ures_openDirect(NULL, gMetaZones, &status); |
734 | ures_getByKey(rb, gMapTimezonesTag, rb, &status); | |
735 | ures_getByKey(rb, keyBuf, rb, &status); | |
736 | ||
737 | if (U_SUCCESS(status)) { | |
738 | // check region mapping | |
739 | if (region.length() == 2 || region.length() == 3) { | |
4388f060 A |
740 | keyLen = region.extract(0, region.length(), keyBuf, ZID_KEY_MAX + 1, US_INV); |
741 | keyBuf[keyLen] = 0; | |
729e4ab9 A |
742 | tzid = ures_getStringByKey(rb, keyBuf, &tzidLen, &status); |
743 | if (status == U_MISSING_RESOURCE_ERROR) { | |
744 | status = U_ZERO_ERROR; | |
46f4442e A |
745 | } |
746 | } | |
729e4ab9 A |
747 | if (U_SUCCESS(status) && tzid == NULL) { |
748 | // try "001" | |
749 | tzid = ures_getStringByKey(rb, gWorldTag, &tzidLen, &status); | |
750 | } | |
46f4442e | 751 | } |
729e4ab9 A |
752 | ures_close(rb); |
753 | ||
754 | if (tzid == NULL) { | |
4388f060 | 755 | result.setToBogus(); |
729e4ab9 A |
756 | } else { |
757 | result.setTo(tzid, tzidLen); | |
46f4442e | 758 | } |
729e4ab9 | 759 | |
46f4442e A |
760 | return result; |
761 | } | |
762 | ||
57a6839d A |
763 | static void U_CALLCONV initAvailableMetaZoneIDs () { |
764 | U_ASSERT(gMetaZoneIDs == NULL); | |
765 | U_ASSERT(gMetaZoneIDTable == NULL); | |
766 | ucln_i18n_registerCleanup(UCLN_I18N_ZONEMETA, zoneMeta_cleanup); | |
767 | ||
768 | UErrorCode status = U_ZERO_ERROR; | |
340931cb | 769 | gMetaZoneIDTable = uhash_open(uhash_hashUChars, uhash_compareUChars, NULL, &status); |
57a6839d A |
770 | if (U_FAILURE(status) || gMetaZoneIDTable == NULL) { |
771 | gMetaZoneIDTable = NULL; | |
772 | return; | |
773 | } | |
57a6839d A |
774 | gMetaZoneIDs = new UVector(NULL, uhash_compareUChars, status); |
775 | if (U_FAILURE(status) || gMetaZoneIDs == NULL) { | |
776 | gMetaZoneIDs = NULL; | |
777 | uhash_close(gMetaZoneIDTable); | |
778 | gMetaZoneIDTable = NULL; | |
779 | return; | |
780 | } | |
781 | gMetaZoneIDs->setDeleter(uprv_free); | |
782 | ||
783 | UResourceBundle *rb = ures_openDirect(NULL, gMetaZones, &status); | |
784 | UResourceBundle *bundle = ures_getByKey(rb, gMapTimezonesTag, NULL, &status); | |
3d1f044b | 785 | StackUResourceBundle res; |
57a6839d | 786 | while (U_SUCCESS(status) && ures_hasNext(bundle)) { |
3d1f044b | 787 | ures_getNextResource(bundle, res.getAlias(), &status); |
57a6839d A |
788 | if (U_FAILURE(status)) { |
789 | break; | |
4388f060 | 790 | } |
3d1f044b | 791 | const char *mzID = ures_getKey(res.getAlias()); |
0f5d89e8 | 792 | int32_t len = static_cast<int32_t>(uprv_strlen(mzID)); |
57a6839d A |
793 | UChar *uMzID = (UChar*)uprv_malloc(sizeof(UChar) * (len + 1)); |
794 | if (uMzID == NULL) { | |
795 | status = U_MEMORY_ALLOCATION_ERROR; | |
796 | break; | |
797 | } | |
798 | u_charsToUChars(mzID, uMzID, len); | |
799 | uMzID[len] = 0; | |
340931cb | 800 | if (uhash_get(gMetaZoneIDTable, uMzID) == NULL) { |
57a6839d | 801 | gMetaZoneIDs->addElement((void *)uMzID, status); |
340931cb | 802 | uhash_put(gMetaZoneIDTable, (void *)uMzID, (void *)uMzID, &status); |
57a6839d A |
803 | } else { |
804 | uprv_free(uMzID); | |
57a6839d A |
805 | } |
806 | } | |
57a6839d A |
807 | ures_close(bundle); |
808 | ures_close(rb); | |
809 | ||
810 | if (U_FAILURE(status)) { | |
811 | uhash_close(gMetaZoneIDTable); | |
812 | delete gMetaZoneIDs; | |
813 | gMetaZoneIDTable = NULL; | |
814 | gMetaZoneIDs = NULL; | |
4388f060 A |
815 | } |
816 | } | |
817 | ||
818 | const UVector* | |
819 | ZoneMeta::getAvailableMetazoneIDs() { | |
57a6839d | 820 | umtx_initOnce(gMetaZoneIDsInitOnce, &initAvailableMetaZoneIDs); |
4388f060 A |
821 | return gMetaZoneIDs; |
822 | } | |
823 | ||
824 | const UChar* | |
825 | ZoneMeta::findMetaZoneID(const UnicodeString& mzid) { | |
57a6839d A |
826 | umtx_initOnce(gMetaZoneIDsInitOnce, &initAvailableMetaZoneIDs); |
827 | if (gMetaZoneIDTable == NULL) { | |
828 | return NULL; | |
829 | } | |
340931cb A |
830 | UnicodeString localMzid = mzid; |
831 | return (const UChar*)uhash_get(gMetaZoneIDTable, localMzid.getTerminatedBuffer()); | |
4388f060 A |
832 | } |
833 | ||
834 | const UChar* | |
835 | ZoneMeta::findTimeZoneID(const UnicodeString& tzid) { | |
836 | return TimeZone::findID(tzid); | |
837 | } | |
838 | ||
839 | ||
840 | TimeZone* | |
841 | ZoneMeta::createCustomTimeZone(int32_t offset) { | |
842 | UBool negative = FALSE; | |
843 | int32_t tmp = offset; | |
844 | if (offset < 0) { | |
845 | negative = TRUE; | |
846 | tmp = -offset; | |
847 | } | |
3d1f044b | 848 | uint8_t hour, min, sec; |
4388f060 A |
849 | |
850 | tmp /= 1000; | |
3d1f044b | 851 | sec = static_cast<uint8_t>(tmp % 60); |
4388f060 | 852 | tmp /= 60; |
3d1f044b A |
853 | min = static_cast<uint8_t>(tmp % 60); |
854 | hour = static_cast<uint8_t>(tmp / 60); | |
4388f060 A |
855 | |
856 | UnicodeString zid; | |
857 | formatCustomID(hour, min, sec, negative, zid); | |
858 | return new SimpleTimeZone(offset, zid); | |
859 | } | |
860 | ||
861 | UnicodeString& | |
862 | ZoneMeta::formatCustomID(uint8_t hour, uint8_t min, uint8_t sec, UBool negative, UnicodeString& id) { | |
863 | // Create normalized time zone ID - GMT[+|-]HH:mm[:ss] | |
864 | id.setTo(gCustomTzPrefix, -1); | |
865 | if (hour != 0 || min != 0) { | |
866 | if (negative) { | |
51004dcb | 867 | id.append((UChar)0x2D); // '-' |
4388f060 | 868 | } else { |
51004dcb | 869 | id.append((UChar)0x2B); // '+' |
4388f060 A |
870 | } |
871 | // Always use US-ASCII digits | |
51004dcb A |
872 | id.append((UChar)(0x30 + (hour%100)/10)); |
873 | id.append((UChar)(0x30 + (hour%10))); | |
874 | id.append((UChar)0x3A); // ':' | |
875 | id.append((UChar)(0x30 + (min%100)/10)); | |
876 | id.append((UChar)(0x30 + (min%10))); | |
4388f060 | 877 | if (sec != 0) { |
51004dcb A |
878 | id.append((UChar)0x3A); // ':' |
879 | id.append((UChar)(0x30 + (sec%100)/10)); | |
880 | id.append((UChar)(0x30 + (sec%10))); | |
4388f060 A |
881 | } |
882 | } | |
883 | return id; | |
884 | } | |
885 | ||
51004dcb A |
886 | const UChar* |
887 | ZoneMeta::getShortID(const TimeZone& tz) { | |
888 | const UChar* canonicalID = NULL; | |
889 | if (dynamic_cast<const OlsonTimeZone *>(&tz) != NULL) { | |
890 | // short cut for OlsonTimeZone | |
891 | const OlsonTimeZone *otz = (const OlsonTimeZone*)&tz; | |
892 | canonicalID = otz->getCanonicalID(); | |
893 | } | |
894 | if (canonicalID == NULL) { | |
895 | return NULL; | |
896 | } | |
897 | return getShortIDFromCanonical(canonicalID); | |
898 | } | |
899 | ||
900 | const UChar* | |
901 | ZoneMeta::getShortID(const UnicodeString& id) { | |
902 | UErrorCode status = U_ZERO_ERROR; | |
903 | const UChar* canonicalID = ZoneMeta::getCanonicalCLDRID(id, status); | |
904 | if (U_FAILURE(status) || canonicalID == NULL) { | |
905 | return NULL; | |
906 | } | |
907 | return ZoneMeta::getShortIDFromCanonical(canonicalID); | |
908 | } | |
909 | ||
910 | const UChar* | |
911 | ZoneMeta::getShortIDFromCanonical(const UChar* canonicalID) { | |
912 | const UChar* shortID = NULL; | |
913 | int32_t len = u_strlen(canonicalID); | |
914 | char tzidKey[ZID_KEY_MAX + 1]; | |
915 | ||
916 | u_UCharsToChars(canonicalID, tzidKey, len); | |
917 | tzidKey[len] = (char) 0; // Make sure it is null terminated. | |
918 | ||
919 | // replace '/' with ':' | |
920 | char *p = tzidKey; | |
921 | while (*p++) { | |
922 | if (*p == '/') { | |
923 | *p = ':'; | |
924 | } | |
925 | } | |
926 | ||
927 | UErrorCode status = U_ZERO_ERROR; | |
928 | UResourceBundle *rb = ures_openDirect(NULL, gKeyTypeData, &status); | |
929 | ures_getByKey(rb, gTypeMapTag, rb, &status); | |
930 | ures_getByKey(rb, gTimezoneTag, rb, &status); | |
931 | shortID = ures_getStringByKey(rb, tzidKey, NULL, &status); | |
932 | ures_close(rb); | |
933 | ||
934 | return shortID; | |
935 | } | |
4388f060 | 936 | |
46f4442e A |
937 | U_NAMESPACE_END |
938 | ||
939 | #endif /* #if !UCONFIG_NO_FORMATTING */ |