libcore_icu_ICU.cpp revision c5b1eb191102a20bc0626aea955aba417e337fbc
1/*
2 * Copyright (C) 2008 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 *      http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17#define LOG_TAG "ICU"
18
19#include "IcuUtilities.h"
20#include "JNIHelp.h"
21#include "JniConstants.h"
22#include "JniException.h"
23#include "ScopedFd.h"
24#include "ScopedJavaUnicodeString.h"
25#include "ScopedLocalRef.h"
26#include "ScopedUtfChars.h"
27#include "UniquePtr.h"
28#include "cutils/log.h"
29#include "toStringArray.h"
30#include "unicode/calendar.h"
31#include "unicode/datefmt.h"
32#include "unicode/dcfmtsym.h"
33#include "unicode/decimfmt.h"
34#include "unicode/dtfmtsym.h"
35#include "unicode/dtptngen.h"
36#include "unicode/gregocal.h"
37#include "unicode/locid.h"
38#include "unicode/numfmt.h"
39#include "unicode/strenum.h"
40#include "unicode/ubrk.h"
41#include "unicode/ucal.h"
42#include "unicode/uclean.h"
43#include "unicode/ucol.h"
44#include "unicode/ucurr.h"
45#include "unicode/udat.h"
46#include "unicode/uloc.h"
47#include "unicode/ulocdata.h"
48#include "unicode/ustring.h"
49#include "ureslocs.h"
50#include "valueOf.h"
51
52#include <errno.h>
53#include <fcntl.h>
54#include <stdlib.h>
55#include <string.h>
56#include <string>
57#include <sys/mman.h>
58#include <sys/stat.h>
59#include <sys/time.h>
60#include <sys/types.h>
61#include <time.h>
62#include <unistd.h>
63#include <vector>
64
65// TODO: put this in a header file and use it everywhere!
66// DISALLOW_COPY_AND_ASSIGN disallows the copy and operator= functions.
67// It goes in the private: declarations in a class.
68#define DISALLOW_COPY_AND_ASSIGN(TypeName) \
69    TypeName(const TypeName&); \
70    void operator=(const TypeName&)
71
72class ScopedResourceBundle {
73 public:
74  ScopedResourceBundle(UResourceBundle* bundle) : bundle_(bundle) {
75  }
76
77  ~ScopedResourceBundle() {
78    if (bundle_ != NULL) {
79      ures_close(bundle_);
80    }
81  }
82
83  UResourceBundle* get() {
84    return bundle_;
85  }
86
87  bool hasKey(const char* key) {
88    UErrorCode status = U_ZERO_ERROR;
89    ures_getStringByKey(bundle_, key, NULL, &status);
90    return U_SUCCESS(status);
91  }
92
93 private:
94  UResourceBundle* bundle_;
95  DISALLOW_COPY_AND_ASSIGN(ScopedResourceBundle);
96};
97
98static jstring ICU_addLikelySubtags(JNIEnv* env, jclass, jstring javaLocale) {
99    UErrorCode status = U_ZERO_ERROR;
100    ScopedUtfChars localeID(env, javaLocale);
101    char maximizedLocaleID[ULOC_FULLNAME_CAPACITY];
102    uloc_addLikelySubtags(localeID.c_str(), maximizedLocaleID, sizeof(maximizedLocaleID), &status);
103    if (U_FAILURE(status)) {
104        return javaLocale;
105    }
106    return env->NewStringUTF(maximizedLocaleID);
107}
108
109static jstring ICU_getScript(JNIEnv* env, jclass, jstring javaLocale) {
110    UErrorCode status = U_ZERO_ERROR;
111    ScopedUtfChars localeID(env, javaLocale);
112    char script[ULOC_SCRIPT_CAPACITY];
113    uloc_getScript(localeID.c_str(), script, sizeof(script), &status);
114    if (U_FAILURE(status)) {
115        return NULL;
116    }
117    return env->NewStringUTF(script);
118}
119
120static jstring ICU_localeForLanguageTag(JNIEnv* env, jclass, jstring languageTag, jboolean strict) {
121    ScopedUtfChars languageTagChars(env, languageTag);
122
123    // Naively assume that in the average case, the size of
124    // the normalized language tag will be very nearly the same as the
125    // size of the input. This is generally true for language
126    // tags that are "simple" language-region-variant combinations
127    // that don't contain any grandfathered tags.
128    const size_t initialBufferSize = languageTagChars.size() + 32;
129    std::vector<char> buffer(initialBufferSize);
130    int32_t parsedLength = 0;
131
132    UErrorCode status = U_ZERO_ERROR;
133    while (true) {
134        const size_t outputLength = uloc_forLanguageTag(languageTagChars.c_str(),
135               &buffer[0], buffer.size(), &parsedLength, &status);
136        if (U_FAILURE(status)) {
137            return NULL;
138        }
139
140        // Assume that we've run out of buffer space when this happens. Double
141        // the buffer size and try again. This should happen very infrequently.
142        if (outputLength == buffer.size()) {
143            buffer.resize(buffer.size() << 1);
144        } else {
145            break;
146        }
147    }
148
149    if (parsedLength < 0) {
150        return NULL;
151    }
152
153    // By default, ICU will ignore all subtags starting at the first unparseable
154    // or invalid subtag. Our "strict" mode is specified to throw an error if
155    // that happens.
156    //
157    // NOTE: The cast is safe because parsedLength can never be negative thanks
158    // to the check above. ICU does not document any negative return values for
159    // that field, but check for it anyway.
160    if ((strict == JNI_TRUE) &&
161        (static_cast<uint32_t>(parsedLength) != languageTagChars.size())) {
162        return NULL;
163    }
164
165    return env->NewStringUTF(&buffer[0]);
166}
167
168static jstring ICU_languageTagForLocale(JNIEnv* env, jclass, jstring javaLocaleId) {
169    ScopedUtfChars localeID(env, javaLocaleId);
170
171    // The conversion from an ICU locale ID to a BCP 47 tag will shrink
172    // the size of the string unless there's an invalid language or a bad
173    // parse (which will result in an x-lvariant private use subtag at
174    // the end of the input).
175    const size_t initialBufferSize = localeID.size();
176    std::vector<char> buffer(initialBufferSize);
177
178    UErrorCode status = U_ZERO_ERROR;
179    while (true) {
180        const size_t outputLength = uloc_toLanguageTag(localeID.c_str(),
181                &buffer[0], buffer.size(), false /* strict */, &status);
182        if (U_FAILURE(status)) {
183            return NULL;
184        }
185
186        if (outputLength == buffer.size()) {
187            buffer.resize(buffer.size() << 1);
188        } else {
189            break;
190        }
191    }
192
193    return env->NewStringUTF(&buffer[0]);
194}
195
196static jint ICU_getCurrencyFractionDigits(JNIEnv* env, jclass, jstring javaCurrencyCode) {
197  ScopedJavaUnicodeString currencyCode(env, javaCurrencyCode);
198  if (!currencyCode.valid()) {
199    return 0;
200  }
201  UnicodeString icuCurrencyCode(currencyCode.unicodeString());
202  UErrorCode status = U_ZERO_ERROR;
203  return ucurr_getDefaultFractionDigits(icuCurrencyCode.getTerminatedBuffer(), &status);
204}
205
206// TODO: rewrite this with int32_t ucurr_forLocale(const char* locale, UChar* buff, int32_t buffCapacity, UErrorCode* ec)...
207static jstring ICU_getCurrencyCode(JNIEnv* env, jclass, jstring javaCountryCode) {
208    UErrorCode status = U_ZERO_ERROR;
209    ScopedResourceBundle supplData(ures_openDirect(U_ICUDATA_CURR, "supplementalData", &status));
210    if (U_FAILURE(status)) {
211        return NULL;
212    }
213
214    ScopedResourceBundle currencyMap(ures_getByKey(supplData.get(), "CurrencyMap", NULL, &status));
215    if (U_FAILURE(status)) {
216        return NULL;
217    }
218
219    ScopedUtfChars countryCode(env, javaCountryCode);
220    ScopedResourceBundle currency(ures_getByKey(currencyMap.get(), countryCode.c_str(), NULL, &status));
221    if (U_FAILURE(status)) {
222        return NULL;
223    }
224
225    ScopedResourceBundle currencyElem(ures_getByIndex(currency.get(), 0, NULL, &status));
226    if (U_FAILURE(status)) {
227        return env->NewStringUTF("XXX");
228    }
229
230    // Check if there's a 'to' date. If there is, the currency isn't used anymore.
231    ScopedResourceBundle currencyTo(ures_getByKey(currencyElem.get(), "to", NULL, &status));
232    if (!U_FAILURE(status)) {
233        return NULL;
234    }
235    // Ignore the failure to find a 'to' date.
236    status = U_ZERO_ERROR;
237
238    ScopedResourceBundle currencyId(ures_getByKey(currencyElem.get(), "id", NULL, &status));
239    if (U_FAILURE(status)) {
240        // No id defined for this country
241        return env->NewStringUTF("XXX");
242    }
243
244    int32_t charCount;
245    const jchar* chars = ures_getString(currencyId.get(), &charCount, &status);
246    return (charCount == 0) ? env->NewStringUTF("XXX") : env->NewString(chars, charCount);
247}
248
249static jstring getCurrencyName(JNIEnv* env, jstring javaLocaleName, jstring javaCurrencyCode, UCurrNameStyle nameStyle) {
250  ScopedUtfChars localeName(env, javaLocaleName);
251  if (localeName.c_str() == NULL) {
252    return NULL;
253  }
254  ScopedJavaUnicodeString currencyCode(env, javaCurrencyCode);
255  if (!currencyCode.valid()) {
256    return NULL;
257  }
258  UnicodeString icuCurrencyCode(currencyCode.unicodeString());
259  UErrorCode status = U_ZERO_ERROR;
260  UBool isChoiceFormat = false;
261  int32_t charCount;
262  const UChar* chars = ucurr_getName(icuCurrencyCode.getTerminatedBuffer(), localeName.c_str(),
263                                     nameStyle, &isChoiceFormat, &charCount, &status);
264  if (status == U_USING_DEFAULT_WARNING) {
265    if (nameStyle == UCURR_SYMBOL_NAME) {
266      // ICU doesn't distinguish between falling back to the root locale and meeting a genuinely
267      // unknown currency. The Currency class does.
268      if (!ucurr_isAvailable(icuCurrencyCode.getTerminatedBuffer(), U_DATE_MIN, U_DATE_MAX, &status)) {
269        return NULL;
270      }
271    }
272    if (nameStyle == UCURR_LONG_NAME) {
273      // ICU's default is English. We want the ISO 4217 currency code instead.
274      chars = icuCurrencyCode.getBuffer();
275      charCount = icuCurrencyCode.length();
276    }
277  }
278  return (charCount == 0) ? NULL : env->NewString(chars, charCount);
279}
280
281static jstring ICU_getCurrencyDisplayName(JNIEnv* env, jclass, jstring javaLocaleName, jstring javaCurrencyCode) {
282  return getCurrencyName(env, javaLocaleName, javaCurrencyCode, UCURR_LONG_NAME);
283}
284
285static jstring ICU_getCurrencySymbol(JNIEnv* env, jclass, jstring javaLocaleName, jstring javaCurrencyCode) {
286  return getCurrencyName(env, javaLocaleName, javaCurrencyCode, UCURR_SYMBOL_NAME);
287}
288
289static jstring ICU_getDisplayCountryNative(JNIEnv* env, jclass, jstring targetLocale, jstring locale) {
290    Locale loc = getLocale(env, locale);
291    Locale targetLoc = getLocale(env, targetLocale);
292    UnicodeString str;
293    targetLoc.getDisplayCountry(loc, str);
294    return env->NewString(str.getBuffer(), str.length());
295}
296
297static jstring ICU_getDisplayLanguageNative(JNIEnv* env, jclass, jstring targetLocale, jstring locale) {
298    Locale loc = getLocale(env, locale);
299    Locale targetLoc = getLocale(env, targetLocale);
300    UnicodeString str;
301    targetLoc.getDisplayLanguage(loc, str);
302    return env->NewString(str.getBuffer(), str.length());
303}
304
305static jstring ICU_getDisplayScriptNative(JNIEnv* env, jclass, jstring targetLocale, jstring locale) {
306    Locale loc = getLocale(env, locale);
307    Locale targetLoc = getLocale(env, targetLocale);
308    UnicodeString str;
309    targetLoc.getDisplayScript(loc, str);
310    return env->NewString(str.getBuffer(), str.length());
311}
312
313static jstring ICU_getDisplayVariantNative(JNIEnv* env, jclass, jstring targetLocale, jstring locale) {
314    Locale loc = getLocale(env, locale);
315    Locale targetLoc = getLocale(env, targetLocale);
316    UnicodeString str;
317    targetLoc.getDisplayVariant(loc, str);
318    return env->NewString(str.getBuffer(), str.length());
319}
320
321static jstring ICU_getISO3CountryNative(JNIEnv* env, jclass, jstring locale) {
322    Locale loc = getLocale(env, locale);
323    return env->NewStringUTF(loc.getISO3Country());
324}
325
326static jstring ICU_getISO3LanguageNative(JNIEnv* env, jclass, jstring locale) {
327    Locale loc = getLocale(env, locale);
328    return env->NewStringUTF(loc.getISO3Language());
329}
330
331static jobjectArray ICU_getISOCountriesNative(JNIEnv* env, jclass) {
332    return toStringArray(env, Locale::getISOCountries());
333}
334
335static jobjectArray ICU_getISOLanguagesNative(JNIEnv* env, jclass) {
336    return toStringArray(env, Locale::getISOLanguages());
337}
338
339static jobjectArray ICU_getAvailableLocalesNative(JNIEnv* env, jclass) {
340    return toStringArray(env, uloc_countAvailable, uloc_getAvailable);
341}
342
343static jobjectArray ICU_getAvailableBreakIteratorLocalesNative(JNIEnv* env, jclass) {
344    return toStringArray(env, ubrk_countAvailable, ubrk_getAvailable);
345}
346
347static jobjectArray ICU_getAvailableCalendarLocalesNative(JNIEnv* env, jclass) {
348    return toStringArray(env, ucal_countAvailable, ucal_getAvailable);
349}
350
351static jobjectArray ICU_getAvailableCollatorLocalesNative(JNIEnv* env, jclass) {
352    return toStringArray(env, ucol_countAvailable, ucol_getAvailable);
353}
354
355static jobjectArray ICU_getAvailableDateFormatLocalesNative(JNIEnv* env, jclass) {
356    return toStringArray(env, udat_countAvailable, udat_getAvailable);
357}
358
359static jobjectArray ICU_getAvailableNumberFormatLocalesNative(JNIEnv* env, jclass) {
360    return toStringArray(env, unum_countAvailable, unum_getAvailable);
361}
362
363static void setIntegerField(JNIEnv* env, jobject obj, const char* fieldName, int value) {
364    ScopedLocalRef<jobject> integerValue(env, integerValueOf(env, value));
365    jfieldID fid = env->GetFieldID(JniConstants::localeDataClass, fieldName, "Ljava/lang/Integer;");
366    env->SetObjectField(obj, fid, integerValue.get());
367}
368
369static void setStringField(JNIEnv* env, jobject obj, const char* fieldName, jstring value) {
370    jfieldID fid = env->GetFieldID(JniConstants::localeDataClass, fieldName, "Ljava/lang/String;");
371    env->SetObjectField(obj, fid, value);
372    env->DeleteLocalRef(value);
373}
374
375static void setStringArrayField(JNIEnv* env, jobject obj, const char* fieldName, jobjectArray value) {
376    jfieldID fid = env->GetFieldID(JniConstants::localeDataClass, fieldName, "[Ljava/lang/String;");
377    env->SetObjectField(obj, fid, value);
378}
379
380static void setStringArrayField(JNIEnv* env, jobject obj, const char* fieldName, const UnicodeString* valueArray, int32_t size) {
381    ScopedLocalRef<jobjectArray> result(env, env->NewObjectArray(size, JniConstants::stringClass, NULL));
382    for (int32_t i = 0; i < size ; i++) {
383        ScopedLocalRef<jstring> s(env, env->NewString(valueArray[i].getBuffer(),valueArray[i].length()));
384        if (env->ExceptionCheck()) {
385            return;
386        }
387        env->SetObjectArrayElement(result.get(), i, s.get());
388        if (env->ExceptionCheck()) {
389            return;
390        }
391    }
392    setStringArrayField(env, obj, fieldName, result.get());
393}
394
395static void setStringField(JNIEnv* env, jobject obj, const char* fieldName, UResourceBundle* bundle, int index) {
396  UErrorCode status = U_ZERO_ERROR;
397  int charCount;
398  const UChar* chars = ures_getStringByIndex(bundle, index, &charCount, &status);
399  if (U_SUCCESS(status)) {
400    setStringField(env, obj, fieldName, env->NewString(chars, charCount));
401  } else {
402    ALOGE("Error setting String field %s from ICU resource (index %d): %s", fieldName, index, u_errorName(status));
403  }
404}
405
406static void setStringField(JNIEnv* env, jobject obj, const char* fieldName, UResourceBundle* bundle, const char* key) {
407  UErrorCode status = U_ZERO_ERROR;
408  int charCount;
409  const UChar* chars = ures_getStringByKey(bundle, key, &charCount, &status);
410  if (U_SUCCESS(status)) {
411    setStringField(env, obj, fieldName, env->NewString(chars, charCount));
412  } else {
413    ALOGE("Error setting String field %s from ICU resource (key %s): %s", fieldName, key, u_errorName(status));
414  }
415}
416
417static void setCharField(JNIEnv* env, jobject obj, const char* fieldName, const UnicodeString& value) {
418    if (value.length() == 0) {
419        return;
420    }
421    jfieldID fid = env->GetFieldID(JniConstants::localeDataClass, fieldName, "C");
422    env->SetCharField(obj, fid, value.charAt(0));
423}
424
425static void setStringField(JNIEnv* env, jobject obj, const char* fieldName, const UnicodeString& value) {
426    const UChar* chars = value.getBuffer();
427    setStringField(env, obj, fieldName, env->NewString(chars, value.length()));
428}
429
430static void setNumberPatterns(JNIEnv* env, jobject obj, Locale& locale) {
431    UErrorCode status = U_ZERO_ERROR;
432
433    UnicodeString pattern;
434    UniquePtr<DecimalFormat> fmt(static_cast<DecimalFormat*>(NumberFormat::createInstance(locale, UNUM_CURRENCY, status)));
435    pattern = fmt->toPattern(pattern.remove());
436    setStringField(env, obj, "currencyPattern", pattern);
437
438    fmt.reset(static_cast<DecimalFormat*>(NumberFormat::createInstance(locale, UNUM_DECIMAL, status)));
439    pattern = fmt->toPattern(pattern.remove());
440    setStringField(env, obj, "numberPattern", pattern);
441
442    fmt.reset(static_cast<DecimalFormat*>(NumberFormat::createInstance(locale, UNUM_PERCENT, status)));
443    pattern = fmt->toPattern(pattern.remove());
444    setStringField(env, obj, "percentPattern", pattern);
445}
446
447static void setDecimalFormatSymbolsData(JNIEnv* env, jobject obj, Locale& locale) {
448    UErrorCode status = U_ZERO_ERROR;
449    DecimalFormatSymbols dfs(locale, status);
450
451    setCharField(env, obj, "decimalSeparator", dfs.getSymbol(DecimalFormatSymbols::kDecimalSeparatorSymbol));
452    setCharField(env, obj, "groupingSeparator", dfs.getSymbol(DecimalFormatSymbols::kGroupingSeparatorSymbol));
453    setCharField(env, obj, "patternSeparator", dfs.getSymbol(DecimalFormatSymbols::kPatternSeparatorSymbol));
454    setCharField(env, obj, "percent", dfs.getSymbol(DecimalFormatSymbols::kPercentSymbol));
455    setCharField(env, obj, "perMill", dfs.getSymbol(DecimalFormatSymbols::kPerMillSymbol));
456    setCharField(env, obj, "monetarySeparator", dfs.getSymbol(DecimalFormatSymbols::kMonetarySeparatorSymbol));
457    setCharField(env, obj, "minusSign", dfs.getSymbol(DecimalFormatSymbols:: kMinusSignSymbol));
458    setStringField(env, obj, "exponentSeparator", dfs.getSymbol(DecimalFormatSymbols::kExponentialSymbol));
459    setStringField(env, obj, "infinity", dfs.getSymbol(DecimalFormatSymbols::kInfinitySymbol));
460    setStringField(env, obj, "NaN", dfs.getSymbol(DecimalFormatSymbols::kNaNSymbol));
461    setCharField(env, obj, "zeroDigit", dfs.getSymbol(DecimalFormatSymbols::kZeroDigitSymbol));
462}
463
464
465// Iterates up through the locale hierarchy. So "en_US" would return "en_US", "en", "".
466class LocaleNameIterator {
467 public:
468  LocaleNameIterator(const char* locale_name, UErrorCode& status) : status_(status), has_next_(true) {
469    strcpy(locale_name_, locale_name);
470    locale_name_length_ = strlen(locale_name_);
471  }
472
473  const char* Get() {
474      return locale_name_;
475  }
476
477  bool HasNext() {
478    return has_next_;
479  }
480
481  void Up() {
482    if (locale_name_length_ == 0) {
483      has_next_ = false;
484    } else {
485      locale_name_length_ = uloc_getParent(locale_name_, locale_name_, sizeof(locale_name_), &status_);
486    }
487  }
488
489 private:
490  UErrorCode& status_;
491  bool has_next_;
492  char locale_name_[ULOC_FULLNAME_CAPACITY];
493  int32_t locale_name_length_;
494
495  DISALLOW_COPY_AND_ASSIGN(LocaleNameIterator);
496};
497
498static bool getDateTimePatterns(JNIEnv* env, jobject localeData, const char* locale_name) {
499  UErrorCode status = U_ZERO_ERROR;
500  ScopedResourceBundle root(ures_open(NULL, locale_name, &status));
501  if (U_FAILURE(status)) {
502    return false;
503  }
504  ScopedResourceBundle calendar(ures_getByKey(root.get(), "calendar", NULL, &status));
505  if (U_FAILURE(status)) {
506    return false;
507  }
508  ScopedResourceBundle gregorian(ures_getByKey(calendar.get(), "gregorian", NULL, &status));
509  if (U_FAILURE(status)) {
510    return false;
511  }
512  ScopedResourceBundle dateTimePatterns(ures_getByKey(gregorian.get(), "DateTimePatterns", NULL, &status));
513  if (U_FAILURE(status)) {
514    return false;
515  }
516  setStringField(env, localeData, "fullTimeFormat", dateTimePatterns.get(), 0);
517  setStringField(env, localeData, "longTimeFormat", dateTimePatterns.get(), 1);
518  setStringField(env, localeData, "mediumTimeFormat", dateTimePatterns.get(), 2);
519  setStringField(env, localeData, "shortTimeFormat", dateTimePatterns.get(), 3);
520  setStringField(env, localeData, "fullDateFormat", dateTimePatterns.get(), 4);
521  setStringField(env, localeData, "longDateFormat", dateTimePatterns.get(), 5);
522  setStringField(env, localeData, "mediumDateFormat", dateTimePatterns.get(), 6);
523  setStringField(env, localeData, "shortDateFormat", dateTimePatterns.get(), 7);
524  return true;
525}
526
527static bool getYesterdayTodayAndTomorrow(JNIEnv* env, jobject localeData, const char* locale_name) {
528  UErrorCode status = U_ZERO_ERROR;
529  ScopedResourceBundle root(ures_open(NULL, locale_name, &status));
530  if (U_FAILURE(status)) {
531    return false;
532  }
533  ScopedResourceBundle fields(ures_getByKey(root.get(), "fields", NULL, &status));
534  if (U_FAILURE(status)) {
535    return false;
536  }
537  ScopedResourceBundle day(ures_getByKey(fields.get(), "day", NULL, &status));
538  if (U_FAILURE(status)) {
539    return false;
540  }
541  ScopedResourceBundle relative(ures_getByKey(day.get(), "relative", NULL, &status));
542  if (U_FAILURE(status)) {
543    return false;
544  }
545  // bn_BD only has a "-2" entry.
546  if (relative.hasKey("-1") && relative.hasKey("0") && relative.hasKey("1")) {
547    setStringField(env, localeData, "yesterday", relative.get(), "-1");
548    setStringField(env, localeData, "today", relative.get(), "0");
549    setStringField(env, localeData, "tomorrow", relative.get(), "1");
550    return true;
551  }
552  return false;
553}
554
555static jboolean ICU_initLocaleDataNative(JNIEnv* env, jclass, jstring javaLocaleName, jobject localeData) {
556    ScopedUtfChars localeName(env, javaLocaleName);
557    if (localeName.c_str() == NULL) {
558        return JNI_FALSE;
559    }
560    if (localeName.size() >= ULOC_FULLNAME_CAPACITY) {
561        return JNI_FALSE; // ICU has a fixed-length limit.
562    }
563
564    // Get the DateTimePatterns.
565    UErrorCode status = U_ZERO_ERROR;
566    bool foundDateTimePatterns = false;
567    for (LocaleNameIterator it(localeName.c_str(), status); it.HasNext(); it.Up()) {
568      if (getDateTimePatterns(env, localeData, it.Get())) {
569          foundDateTimePatterns = true;
570          break;
571      }
572    }
573    if (!foundDateTimePatterns) {
574        ALOGE("Couldn't find ICU DateTimePatterns for %s", localeName.c_str());
575        return JNI_FALSE;
576    }
577
578    // Get the "Yesterday", "Today", and "Tomorrow" strings.
579    bool foundYesterdayTodayAndTomorrow = false;
580    for (LocaleNameIterator it(localeName.c_str(), status); it.HasNext(); it.Up()) {
581      if (getYesterdayTodayAndTomorrow(env, localeData, it.Get())) {
582        foundYesterdayTodayAndTomorrow = true;
583        break;
584      }
585    }
586    if (!foundYesterdayTodayAndTomorrow) {
587      ALOGE("Couldn't find ICU yesterday/today/tomorrow for %s", localeName.c_str());
588      return JNI_FALSE;
589    }
590
591    status = U_ZERO_ERROR;
592    Locale locale = getLocale(env, javaLocaleName);
593    UniquePtr<Calendar> cal(Calendar::createInstance(locale, status));
594    if (U_FAILURE(status)) {
595        return JNI_FALSE;
596    }
597
598    setIntegerField(env, localeData, "firstDayOfWeek", cal->getFirstDayOfWeek());
599    setIntegerField(env, localeData, "minimalDaysInFirstWeek", cal->getMinimalDaysInFirstWeek());
600
601    // Get DateFormatSymbols.
602    status = U_ZERO_ERROR;
603    DateFormatSymbols dateFormatSym(locale, status);
604    if (U_FAILURE(status)) {
605        return JNI_FALSE;
606    }
607
608    // Get AM/PM and BC/AD.
609    int32_t count = 0;
610    const UnicodeString* amPmStrs = dateFormatSym.getAmPmStrings(count);
611    setStringArrayField(env, localeData, "amPm", amPmStrs, count);
612    const UnicodeString* erasStrs = dateFormatSym.getEras(count);
613    setStringArrayField(env, localeData, "eras", erasStrs, count);
614
615    const UnicodeString* longMonthNames =
616       dateFormatSym.getMonths(count, DateFormatSymbols::FORMAT, DateFormatSymbols::WIDE);
617    setStringArrayField(env, localeData, "longMonthNames", longMonthNames, count);
618    const UnicodeString* shortMonthNames =
619        dateFormatSym.getMonths(count, DateFormatSymbols::FORMAT, DateFormatSymbols::ABBREVIATED);
620    setStringArrayField(env, localeData, "shortMonthNames", shortMonthNames, count);
621    const UnicodeString* tinyMonthNames =
622        dateFormatSym.getMonths(count, DateFormatSymbols::FORMAT, DateFormatSymbols::NARROW);
623    setStringArrayField(env, localeData, "tinyMonthNames", tinyMonthNames, count);
624    const UnicodeString* longWeekdayNames =
625        dateFormatSym.getWeekdays(count, DateFormatSymbols::FORMAT, DateFormatSymbols::WIDE);
626    setStringArrayField(env, localeData, "longWeekdayNames", longWeekdayNames, count);
627    const UnicodeString* shortWeekdayNames =
628        dateFormatSym.getWeekdays(count, DateFormatSymbols::FORMAT, DateFormatSymbols::ABBREVIATED);
629    setStringArrayField(env, localeData, "shortWeekdayNames", shortWeekdayNames, count);
630    const UnicodeString* tinyWeekdayNames =
631        dateFormatSym.getWeekdays(count, DateFormatSymbols::FORMAT, DateFormatSymbols::NARROW);
632    setStringArrayField(env, localeData, "tinyWeekdayNames", tinyWeekdayNames, count);
633
634    const UnicodeString* longStandAloneMonthNames =
635        dateFormatSym.getMonths(count, DateFormatSymbols::STANDALONE, DateFormatSymbols::WIDE);
636    setStringArrayField(env, localeData, "longStandAloneMonthNames", longStandAloneMonthNames, count);
637    const UnicodeString* shortStandAloneMonthNames =
638        dateFormatSym.getMonths(count, DateFormatSymbols::STANDALONE, DateFormatSymbols::ABBREVIATED);
639    setStringArrayField(env, localeData, "shortStandAloneMonthNames", shortStandAloneMonthNames, count);
640    const UnicodeString* tinyStandAloneMonthNames =
641        dateFormatSym.getMonths(count, DateFormatSymbols::STANDALONE, DateFormatSymbols::NARROW);
642    setStringArrayField(env, localeData, "tinyStandAloneMonthNames", tinyStandAloneMonthNames, count);
643    const UnicodeString* longStandAloneWeekdayNames =
644        dateFormatSym.getWeekdays(count, DateFormatSymbols::STANDALONE, DateFormatSymbols::WIDE);
645    setStringArrayField(env, localeData, "longStandAloneWeekdayNames", longStandAloneWeekdayNames, count);
646    const UnicodeString* shortStandAloneWeekdayNames =
647        dateFormatSym.getWeekdays(count, DateFormatSymbols::STANDALONE, DateFormatSymbols::ABBREVIATED);
648    setStringArrayField(env, localeData, "shortStandAloneWeekdayNames", shortStandAloneWeekdayNames, count);
649    const UnicodeString* tinyStandAloneWeekdayNames =
650        dateFormatSym.getWeekdays(count, DateFormatSymbols::STANDALONE, DateFormatSymbols::NARROW);
651    setStringArrayField(env, localeData, "tinyStandAloneWeekdayNames", tinyStandAloneWeekdayNames, count);
652
653    status = U_ZERO_ERROR;
654
655    // For numberPatterns and symbols.
656    setNumberPatterns(env, localeData, locale);
657    setDecimalFormatSymbolsData(env, localeData, locale);
658
659    jstring countryCode = env->NewStringUTF(Locale::createFromName(localeName.c_str()).getCountry());
660    jstring internationalCurrencySymbol = ICU_getCurrencyCode(env, NULL, countryCode);
661    env->DeleteLocalRef(countryCode);
662    countryCode = NULL;
663
664    jstring currencySymbol = NULL;
665    if (internationalCurrencySymbol != NULL) {
666        currencySymbol = ICU_getCurrencySymbol(env, NULL, javaLocaleName, internationalCurrencySymbol);
667    } else {
668        internationalCurrencySymbol = env->NewStringUTF("XXX");
669    }
670    if (currencySymbol == NULL) {
671        // This is the UTF-8 encoding of U+00A4 (CURRENCY SIGN).
672        currencySymbol = env->NewStringUTF("\xc2\xa4");
673    }
674    setStringField(env, localeData, "currencySymbol", currencySymbol);
675    setStringField(env, localeData, "internationalCurrencySymbol", internationalCurrencySymbol);
676
677    return JNI_TRUE;
678}
679
680static jstring ICU_toLowerCase(JNIEnv* env, jclass, jstring javaString, jstring localeName) {
681  ScopedJavaUnicodeString scopedString(env, javaString);
682  if (!scopedString.valid()) {
683    return NULL;
684  }
685  UnicodeString& s(scopedString.unicodeString());
686  UnicodeString original(s);
687  s.toLower(getLocale(env, localeName));
688  return s == original ? javaString : env->NewString(s.getBuffer(), s.length());
689}
690
691static jstring ICU_toUpperCase(JNIEnv* env, jclass, jstring javaString, jstring localeName) {
692  ScopedJavaUnicodeString scopedString(env, javaString);
693  if (!scopedString.valid()) {
694    return NULL;
695  }
696  UnicodeString& s(scopedString.unicodeString());
697  UnicodeString original(s);
698  s.toUpper(getLocale(env, localeName));
699  return s == original ? javaString : env->NewString(s.getBuffer(), s.length());
700}
701
702static jstring versionString(JNIEnv* env, const UVersionInfo& version) {
703    char versionString[U_MAX_VERSION_STRING_LENGTH];
704    u_versionToString(const_cast<UVersionInfo&>(version), &versionString[0]);
705    return env->NewStringUTF(versionString);
706}
707
708static jstring ICU_getCldrVersion(JNIEnv* env, jclass) {
709  UErrorCode status = U_ZERO_ERROR;
710  UVersionInfo cldrVersion;
711  ulocdata_getCLDRVersion(cldrVersion, &status);
712  return versionString(env, cldrVersion);
713}
714
715static jstring ICU_getIcuVersion(JNIEnv* env, jclass) {
716    UVersionInfo icuVersion;
717    u_getVersion(icuVersion);
718    return versionString(env, icuVersion);
719}
720
721static jstring ICU_getUnicodeVersion(JNIEnv* env, jclass) {
722    UVersionInfo unicodeVersion;
723    u_getUnicodeVersion(unicodeVersion);
724    return versionString(env, unicodeVersion);
725}
726
727static jobject ICU_getAvailableCurrencyCodes(JNIEnv* env, jclass) {
728  UErrorCode status = U_ZERO_ERROR;
729  UStringEnumeration e(ucurr_openISOCurrencies(UCURR_COMMON|UCURR_NON_DEPRECATED, &status));
730  return fromStringEnumeration(env, status, "ucurr_openISOCurrencies", &e);
731}
732
733static jstring ICU_getBestDateTimePatternNative(JNIEnv* env, jclass, jstring javaSkeleton, jstring javaLocaleName) {
734  Locale locale = getLocale(env, javaLocaleName);
735  UErrorCode status = U_ZERO_ERROR;
736  UniquePtr<DateTimePatternGenerator> generator(DateTimePatternGenerator::createInstance(locale, status));
737  if (maybeThrowIcuException(env, "DateTimePatternGenerator::createInstance", status)) {
738    return NULL;
739  }
740
741  ScopedJavaUnicodeString skeletonHolder(env, javaSkeleton);
742  if (!skeletonHolder.valid()) {
743    return NULL;
744  }
745  UnicodeString result(generator->getBestPattern(skeletonHolder.unicodeString(), status));
746  if (maybeThrowIcuException(env, "DateTimePatternGenerator::getBestPattern", status)) {
747    return NULL;
748  }
749
750  return env->NewString(result.getBuffer(), result.length());
751}
752
753static JNINativeMethod gMethods[] = {
754    NATIVE_METHOD(ICU, addLikelySubtags, "(Ljava/lang/String;)Ljava/lang/String;"),
755    NATIVE_METHOD(ICU, getAvailableBreakIteratorLocalesNative, "()[Ljava/lang/String;"),
756    NATIVE_METHOD(ICU, getAvailableCalendarLocalesNative, "()[Ljava/lang/String;"),
757    NATIVE_METHOD(ICU, getAvailableCollatorLocalesNative, "()[Ljava/lang/String;"),
758    NATIVE_METHOD(ICU, getAvailableCurrencyCodes, "()[Ljava/lang/String;"),
759    NATIVE_METHOD(ICU, getAvailableDateFormatLocalesNative, "()[Ljava/lang/String;"),
760    NATIVE_METHOD(ICU, getAvailableLocalesNative, "()[Ljava/lang/String;"),
761    NATIVE_METHOD(ICU, getAvailableNumberFormatLocalesNative, "()[Ljava/lang/String;"),
762    NATIVE_METHOD(ICU, getBestDateTimePatternNative, "(Ljava/lang/String;Ljava/lang/String;)Ljava/lang/String;"),
763    NATIVE_METHOD(ICU, getCldrVersion, "()Ljava/lang/String;"),
764    NATIVE_METHOD(ICU, getCurrencyCode, "(Ljava/lang/String;)Ljava/lang/String;"),
765    NATIVE_METHOD(ICU, getCurrencyDisplayName, "(Ljava/lang/String;Ljava/lang/String;)Ljava/lang/String;"),
766    NATIVE_METHOD(ICU, getCurrencyFractionDigits, "(Ljava/lang/String;)I"),
767    NATIVE_METHOD(ICU, getCurrencySymbol, "(Ljava/lang/String;Ljava/lang/String;)Ljava/lang/String;"),
768    NATIVE_METHOD(ICU, getDisplayCountryNative, "(Ljava/lang/String;Ljava/lang/String;)Ljava/lang/String;"),
769    NATIVE_METHOD(ICU, getDisplayLanguageNative, "(Ljava/lang/String;Ljava/lang/String;)Ljava/lang/String;"),
770    NATIVE_METHOD(ICU, getDisplayScriptNative, "(Ljava/lang/String;Ljava/lang/String;)Ljava/lang/String;"),
771    NATIVE_METHOD(ICU, getDisplayVariantNative, "(Ljava/lang/String;Ljava/lang/String;)Ljava/lang/String;"),
772    NATIVE_METHOD(ICU, getISO3CountryNative, "(Ljava/lang/String;)Ljava/lang/String;"),
773    NATIVE_METHOD(ICU, getISO3LanguageNative, "(Ljava/lang/String;)Ljava/lang/String;"),
774    NATIVE_METHOD(ICU, getISOCountriesNative, "()[Ljava/lang/String;"),
775    NATIVE_METHOD(ICU, getISOLanguagesNative, "()[Ljava/lang/String;"),
776    NATIVE_METHOD(ICU, getIcuVersion, "()Ljava/lang/String;"),
777    NATIVE_METHOD(ICU, getScript, "(Ljava/lang/String;)Ljava/lang/String;"),
778    NATIVE_METHOD(ICU, getUnicodeVersion, "()Ljava/lang/String;"),
779    NATIVE_METHOD(ICU, languageTagForLocale, "(Ljava/lang/String;)Ljava/lang/String;"),
780    NATIVE_METHOD(ICU, localeForLanguageTag, "(Ljava/lang/String;Z)Ljava/lang/String;"),
781    NATIVE_METHOD(ICU, initLocaleDataNative, "(Ljava/lang/String;Llibcore/icu/LocaleData;)Z"),
782    NATIVE_METHOD(ICU, toLowerCase, "(Ljava/lang/String;Ljava/lang/String;)Ljava/lang/String;"),
783    NATIVE_METHOD(ICU, toUpperCase, "(Ljava/lang/String;Ljava/lang/String;)Ljava/lang/String;"),
784};
785void register_libcore_icu_ICU(JNIEnv* env) {
786    std::string path;
787    path = u_getDataDirectory();
788    path += "/";
789    path += U_ICUDATA_NAME;
790    path += ".dat";
791
792    #define FAIL_WITH_STRERROR(s) \
793        ALOGE("Couldn't " s " '%s': %s", path.c_str(), strerror(errno)); \
794        abort();
795    #define MAYBE_FAIL_WITH_ICU_ERROR(s) \
796        if (status != U_ZERO_ERROR) {\
797            ALOGE("Couldn't initialize ICU (" s "): %s (%s)", u_errorName(status), path.c_str()); \
798            abort(); \
799        }
800
801    // Open the file and get its length.
802    ScopedFd fd(open(path.c_str(), O_RDONLY));
803    if (fd.get() == -1) {
804        FAIL_WITH_STRERROR("open");
805    }
806    struct stat sb;
807    if (fstat(fd.get(), &sb) == -1) {
808        FAIL_WITH_STRERROR("stat");
809    }
810
811    // Map it.
812    void* data = mmap(NULL, sb.st_size, PROT_READ, MAP_SHARED, fd.get(), 0);
813    if (data == MAP_FAILED) {
814        FAIL_WITH_STRERROR("mmap");
815    }
816
817    // Tell the kernel that accesses are likely to be random rather than sequential.
818    if (madvise(data, sb.st_size, MADV_RANDOM) == -1) {
819        FAIL_WITH_STRERROR("madvise(MADV_RANDOM)");
820    }
821
822    // Tell ICU to use our memory-mapped data.
823    UErrorCode status = U_ZERO_ERROR;
824    udata_setCommonData(data, &status);
825    MAYBE_FAIL_WITH_ICU_ERROR("udata_setCommonData");
826    // Tell ICU it can *only* use our memory-mapped data.
827    udata_setFileAccess(UDATA_NO_FILES, &status);
828    MAYBE_FAIL_WITH_ICU_ERROR("udata_setFileAccess");
829
830    // Failures to find the ICU data tend to be somewhat obscure because ICU loads its data on first
831    // use, which can be anywhere. Force initialization up front so we can report a nice clear error
832    // and bail.
833    u_init(&status);
834    MAYBE_FAIL_WITH_ICU_ERROR("u_init");
835    jniRegisterNativeMethods(env, "libcore/icu/ICU", gMethods, NELEM(gMethods));
836}
837