libcore_icu_ICU.cpp revision 679cf68b607e9b4a3beb8bcdee06868ae583386f
1adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project/*
2adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project * Copyright (C) 2008 The Android Open Source Project
3adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project *
4adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project * Licensed under the Apache License, Version 2.0 (the "License");
5adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project * you may not use this file except in compliance with the License.
6adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project * You may obtain a copy of the License at
7adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project *
8adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project *      http://www.apache.org/licenses/LICENSE-2.0
9adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project *
10adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project * Unless required by applicable law or agreed to in writing, software
11adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project * distributed under the License is distributed on an "AS IS" BASIS,
12adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project * See the License for the specific language governing permissions and
14adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project * limitations under the License.
15adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project */
16adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project
17757a7942eed2b0aa457f8517a0259d2ac82c5b18Elliott Hughes#define LOG_TAG "ICU"
18c08f9fb2a3be82bb1a3f477ca1524ddcf7a1d4b8Elliott Hughes
19adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project#include "JNIHelp.h"
20a9f5c16a864ff63ba63f810410f8a27c086d5d52Elliott Hughes#include "JniConstants.h"
21bef9ec33e1368f57c731fce63b6a8c61628c64b0Elliott Hughes#include "JniException.h"
22566bbb1d5a0ffadfc07fa58176587d4bd0d0be13Elliott Hughes#include "ScopedFd.h"
234e3714f7fe8ac7f6b20896038e491d378f4e6464Elliott Hughes#include "ScopedJavaUnicodeString.h"
24f281667712baf8e0721ceb2cc60e7eef19c2d859Elliott Hughes#include "ScopedLocalRef.h"
2552b310afcffe8b6aed6fa0c1e9e8b0ade6f0a672Elliott Hughes#include "ScopedStringChars.h"
269de899cc3ffd3aa3f8f827201cbe14120609018bElliott Hughes#include "ScopedUtfChars.h"
27757a7942eed2b0aa457f8517a0259d2ac82c5b18Elliott Hughes#include "UniquePtr.h"
28f2d5062b67e57ef00ee81fec67480f0d58d66b50Elliott Hughes#include "cutils/log.h"
29ddafeb1d73cfe504720d10a2634b5858fc4cc413Elliott Hughes#include "toStringArray.h"
30adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project#include "unicode/calendar.h"
31adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project#include "unicode/datefmt.h"
32adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project#include "unicode/dcfmtsym.h"
3394782d07dd2d65b1a37fddca68eb9a9ac81ada4aElliott Hughes#include "unicode/decimfmt.h"
3494782d07dd2d65b1a37fddca68eb9a9ac81ada4aElliott Hughes#include "unicode/dtfmtsym.h"
3594782d07dd2d65b1a37fddca68eb9a9ac81ada4aElliott Hughes#include "unicode/gregocal.h"
3694782d07dd2d65b1a37fddca68eb9a9ac81ada4aElliott Hughes#include "unicode/locid.h"
3794782d07dd2d65b1a37fddca68eb9a9ac81ada4aElliott Hughes#include "unicode/numfmt.h"
38adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project#include "unicode/strenum.h"
3994782d07dd2d65b1a37fddca68eb9a9ac81ada4aElliott Hughes#include "unicode/ubrk.h"
4094782d07dd2d65b1a37fddca68eb9a9ac81ada4aElliott Hughes#include "unicode/ucal.h"
4194782d07dd2d65b1a37fddca68eb9a9ac81ada4aElliott Hughes#include "unicode/uclean.h"
4294782d07dd2d65b1a37fddca68eb9a9ac81ada4aElliott Hughes#include "unicode/ucol.h"
4394782d07dd2d65b1a37fddca68eb9a9ac81ada4aElliott Hughes#include "unicode/ucurr.h"
4494782d07dd2d65b1a37fddca68eb9a9ac81ada4aElliott Hughes#include "unicode/udat.h"
450452e9adf6fb133b6d3e59ebae2135aa90ad0d5eFabrice Di Meglio#include "unicode/uloc.h"
4694782d07dd2d65b1a37fddca68eb9a9ac81ada4aElliott Hughes#include "unicode/ustring.h"
47947eeb80f985827209c9500851e288504d58ec2eclaireho#include "ureslocs.h"
4894782d07dd2d65b1a37fddca68eb9a9ac81ada4aElliott Hughes#include "valueOf.h"
49ddafeb1d73cfe504720d10a2634b5858fc4cc413Elliott Hughes
50566bbb1d5a0ffadfc07fa58176587d4bd0d0be13Elliott Hughes#include <errno.h>
51566bbb1d5a0ffadfc07fa58176587d4bd0d0be13Elliott Hughes#include <fcntl.h>
52adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project#include <stdlib.h>
53adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project#include <string.h>
54ddafeb1d73cfe504720d10a2634b5858fc4cc413Elliott Hughes#include <string>
55566bbb1d5a0ffadfc07fa58176587d4bd0d0be13Elliott Hughes#include <sys/mman.h>
56566bbb1d5a0ffadfc07fa58176587d4bd0d0be13Elliott Hughes#include <sys/stat.h>
57adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project#include <sys/time.h>
58566bbb1d5a0ffadfc07fa58176587d4bd0d0be13Elliott Hughes#include <sys/types.h>
59566bbb1d5a0ffadfc07fa58176587d4bd0d0be13Elliott Hughes#include <time.h>
60566bbb1d5a0ffadfc07fa58176587d4bd0d0be13Elliott Hughes#include <unistd.h>
61adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project
6270fa193185427473999f1cda823c1ec408acd2baElliott Hughesclass ScopedResourceBundle {
6370fa193185427473999f1cda823c1ec408acd2baElliott Hughespublic:
6470fa193185427473999f1cda823c1ec408acd2baElliott Hughes    ScopedResourceBundle(UResourceBundle* bundle) : mBundle(bundle) {
6570fa193185427473999f1cda823c1ec408acd2baElliott Hughes    }
66adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project
6770fa193185427473999f1cda823c1ec408acd2baElliott Hughes    ~ScopedResourceBundle() {
6870fa193185427473999f1cda823c1ec408acd2baElliott Hughes        if (mBundle != NULL) {
6970fa193185427473999f1cda823c1ec408acd2baElliott Hughes            ures_close(mBundle);
7070fa193185427473999f1cda823c1ec408acd2baElliott Hughes        }
7170fa193185427473999f1cda823c1ec408acd2baElliott Hughes    }
72adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project
7370fa193185427473999f1cda823c1ec408acd2baElliott Hughes    UResourceBundle* get() {
7470fa193185427473999f1cda823c1ec408acd2baElliott Hughes        return mBundle;
75adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project    }
7670fa193185427473999f1cda823c1ec408acd2baElliott Hughes
7770fa193185427473999f1cda823c1ec408acd2baElliott Hughesprivate:
7870fa193185427473999f1cda823c1ec408acd2baElliott Hughes    UResourceBundle* mBundle;
797ca6fd0dca02f7abdd8808db78357743bbdd23a5Elliott Hughes
807ca6fd0dca02f7abdd8808db78357743bbdd23a5Elliott Hughes    // Disallow copy and assignment.
817ca6fd0dca02f7abdd8808db78357743bbdd23a5Elliott Hughes    ScopedResourceBundle(const ScopedResourceBundle&);
827ca6fd0dca02f7abdd8808db78357743bbdd23a5Elliott Hughes    void operator=(const ScopedResourceBundle&);
8370fa193185427473999f1cda823c1ec408acd2baElliott Hughes};
8470fa193185427473999f1cda823c1ec408acd2baElliott Hughes
8521557bb6a8f35a2f9889adba449cac950c9d41b9Elliott HughesLocale getLocale(JNIEnv* env, jstring localeName) {
860808cae1a2616ba9c708c7cc4489723b4060178eElliott Hughes    return Locale::createFromName(ScopedUtfChars(env, localeName).c_str());
87adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project}
88adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project
890452e9adf6fb133b6d3e59ebae2135aa90ad0d5eFabrice Di Megliostatic jstring ICU_addLikelySubtags(JNIEnv* env, jclass, jstring javaLocale) {
900452e9adf6fb133b6d3e59ebae2135aa90ad0d5eFabrice Di Meglio    UErrorCode status = U_ZERO_ERROR;
910452e9adf6fb133b6d3e59ebae2135aa90ad0d5eFabrice Di Meglio    ScopedUtfChars localeID(env, javaLocale);
920452e9adf6fb133b6d3e59ebae2135aa90ad0d5eFabrice Di Meglio    char maximizedLocaleID[ULOC_FULLNAME_CAPACITY];
930452e9adf6fb133b6d3e59ebae2135aa90ad0d5eFabrice Di Meglio    uloc_addLikelySubtags(localeID.c_str(), maximizedLocaleID, sizeof(maximizedLocaleID), &status);
940452e9adf6fb133b6d3e59ebae2135aa90ad0d5eFabrice Di Meglio    if (U_FAILURE(status)) {
950452e9adf6fb133b6d3e59ebae2135aa90ad0d5eFabrice Di Meglio        return javaLocale;
960452e9adf6fb133b6d3e59ebae2135aa90ad0d5eFabrice Di Meglio    }
970452e9adf6fb133b6d3e59ebae2135aa90ad0d5eFabrice Di Meglio    return env->NewStringUTF(maximizedLocaleID);
980452e9adf6fb133b6d3e59ebae2135aa90ad0d5eFabrice Di Meglio}
990452e9adf6fb133b6d3e59ebae2135aa90ad0d5eFabrice Di Meglio
1004c94a6128d1fcfaa75539b36c9f50d0c75df6144Fabrice Di Megliostatic jstring ICU_getScript(JNIEnv* env, jclass, jstring javaLocale) {
1014c94a6128d1fcfaa75539b36c9f50d0c75df6144Fabrice Di Meglio    UErrorCode status = U_ZERO_ERROR;
1024c94a6128d1fcfaa75539b36c9f50d0c75df6144Fabrice Di Meglio    ScopedUtfChars localeID(env, javaLocale);
1034c94a6128d1fcfaa75539b36c9f50d0c75df6144Fabrice Di Meglio    char script[ULOC_SCRIPT_CAPACITY];
1044c94a6128d1fcfaa75539b36c9f50d0c75df6144Fabrice Di Meglio    uloc_getScript(localeID.c_str(), script, sizeof(script), &status);
1054c94a6128d1fcfaa75539b36c9f50d0c75df6144Fabrice Di Meglio    if (U_FAILURE(status)) {
1064c94a6128d1fcfaa75539b36c9f50d0c75df6144Fabrice Di Meglio        return NULL;
1074c94a6128d1fcfaa75539b36c9f50d0c75df6144Fabrice Di Meglio    }
1084c94a6128d1fcfaa75539b36c9f50d0c75df6144Fabrice Di Meglio    return env->NewStringUTF(script);
1094c94a6128d1fcfaa75539b36c9f50d0c75df6144Fabrice Di Meglio}
1104c94a6128d1fcfaa75539b36c9f50d0c75df6144Fabrice Di Meglio
11152b310afcffe8b6aed6fa0c1e9e8b0ade6f0a672Elliott Hughesstatic jint ICU_getCurrencyFractionDigits(JNIEnv* env, jclass, jstring javaCurrencyCode) {
1124e3714f7fe8ac7f6b20896038e491d378f4e6464Elliott Hughes    ScopedJavaUnicodeString currencyCode(env, javaCurrencyCode);
11352b310afcffe8b6aed6fa0c1e9e8b0ade6f0a672Elliott Hughes    UnicodeString icuCurrencyCode(currencyCode.unicodeString());
11452b310afcffe8b6aed6fa0c1e9e8b0ade6f0a672Elliott Hughes    UErrorCode status = U_ZERO_ERROR;
11552b310afcffe8b6aed6fa0c1e9e8b0ade6f0a672Elliott Hughes    return ucurr_getDefaultFractionDigits(icuCurrencyCode.getTerminatedBuffer(), &status);
116adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project}
117adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project
11852b310afcffe8b6aed6fa0c1e9e8b0ade6f0a672Elliott Hughesstatic jstring ICU_getCurrencyCode(JNIEnv* env, jclass, jstring javaCountryCode) {
119adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project    UErrorCode status = U_ZERO_ERROR;
120947eeb80f985827209c9500851e288504d58ec2eclaireho    ScopedResourceBundle supplData(ures_openDirect(U_ICUDATA_CURR, "supplementalData", &status));
12170fa193185427473999f1cda823c1ec408acd2baElliott Hughes    if (U_FAILURE(status)) {
122adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project        return NULL;
123adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project    }
124adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project
12570fa193185427473999f1cda823c1ec408acd2baElliott Hughes    ScopedResourceBundle currencyMap(ures_getByKey(supplData.get(), "CurrencyMap", NULL, &status));
12670fa193185427473999f1cda823c1ec408acd2baElliott Hughes    if (U_FAILURE(status)) {
127adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project        return NULL;
128adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project    }
129adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project
13052b310afcffe8b6aed6fa0c1e9e8b0ade6f0a672Elliott Hughes    ScopedUtfChars countryCode(env, javaCountryCode);
13152b310afcffe8b6aed6fa0c1e9e8b0ade6f0a672Elliott Hughes    ScopedResourceBundle currency(ures_getByKey(currencyMap.get(), countryCode.c_str(), NULL, &status));
13270fa193185427473999f1cda823c1ec408acd2baElliott Hughes    if (U_FAILURE(status)) {
133adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project        return NULL;
134adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project    }
135adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project
13670fa193185427473999f1cda823c1ec408acd2baElliott Hughes    ScopedResourceBundle currencyElem(ures_getByIndex(currency.get(), 0, NULL, &status));
13770fa193185427473999f1cda823c1ec408acd2baElliott Hughes    if (U_FAILURE(status)) {
138adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project        return env->NewStringUTF("None");
139adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project    }
140adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project
14152b310afcffe8b6aed6fa0c1e9e8b0ade6f0a672Elliott Hughes    // Check if there's a 'to' date. If there is, the currency isn't used anymore.
14267081ff5bcb8a2b3f7db2940f6e0cb3e459ab4b4Elliott Hughes    ScopedResourceBundle currencyTo(ures_getByKey(currencyElem.get(), "to", NULL, &status));
14367081ff5bcb8a2b3f7db2940f6e0cb3e459ab4b4Elliott Hughes    if (!U_FAILURE(status)) {
14467081ff5bcb8a2b3f7db2940f6e0cb3e459ab4b4Elliott Hughes        return NULL;
145adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project    }
14652b310afcffe8b6aed6fa0c1e9e8b0ade6f0a672Elliott Hughes    // Ignore the failure to find a 'to' date.
14767081ff5bcb8a2b3f7db2940f6e0cb3e459ab4b4Elliott Hughes    status = U_ZERO_ERROR;
148adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project
14970fa193185427473999f1cda823c1ec408acd2baElliott Hughes    ScopedResourceBundle currencyId(ures_getByKey(currencyElem.get(), "id", NULL, &status));
15070fa193185427473999f1cda823c1ec408acd2baElliott Hughes    if (U_FAILURE(status)) {
151adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project        // No id defined for this country
152adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project        return env->NewStringUTF("None");
153adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project    }
154adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project
15552b310afcffe8b6aed6fa0c1e9e8b0ade6f0a672Elliott Hughes    int32_t charCount;
15652b310afcffe8b6aed6fa0c1e9e8b0ade6f0a672Elliott Hughes    const jchar* chars = ures_getString(currencyId.get(), &charCount, &status);
15752b310afcffe8b6aed6fa0c1e9e8b0ade6f0a672Elliott Hughes    return (charCount == 0) ? env->NewStringUTF("None") : env->NewString(chars, charCount);
15852b310afcffe8b6aed6fa0c1e9e8b0ade6f0a672Elliott Hughes}
15952b310afcffe8b6aed6fa0c1e9e8b0ade6f0a672Elliott Hughes
16052b310afcffe8b6aed6fa0c1e9e8b0ade6f0a672Elliott Hughesstatic jstring ICU_getCurrencyDisplayName(JNIEnv* env, jclass, jstring javaLocaleName, jstring javaCurrencyCode) {
16152b310afcffe8b6aed6fa0c1e9e8b0ade6f0a672Elliott Hughes    ScopedUtfChars localeName(env, javaLocaleName);
16252b310afcffe8b6aed6fa0c1e9e8b0ade6f0a672Elliott Hughes    ScopedJavaUnicodeString currencyCode(env, javaCurrencyCode);
16352b310afcffe8b6aed6fa0c1e9e8b0ade6f0a672Elliott Hughes    UnicodeString icuCurrencyCode(currencyCode.unicodeString());
16452b310afcffe8b6aed6fa0c1e9e8b0ade6f0a672Elliott Hughes    UErrorCode status = U_ZERO_ERROR;
16552b310afcffe8b6aed6fa0c1e9e8b0ade6f0a672Elliott Hughes    UBool isChoiceFormat;
16652b310afcffe8b6aed6fa0c1e9e8b0ade6f0a672Elliott Hughes    int32_t charCount;
16752b310afcffe8b6aed6fa0c1e9e8b0ade6f0a672Elliott Hughes    const UChar* chars = ucurr_getName(icuCurrencyCode.getTerminatedBuffer(), localeName.c_str(),
16852b310afcffe8b6aed6fa0c1e9e8b0ade6f0a672Elliott Hughes            UCURR_LONG_NAME, &isChoiceFormat, &charCount, &status);
16952b310afcffe8b6aed6fa0c1e9e8b0ade6f0a672Elliott Hughes    if (status == U_USING_DEFAULT_WARNING) {
17052b310afcffe8b6aed6fa0c1e9e8b0ade6f0a672Elliott Hughes        // ICU's default is English. We want the ISO 4217 currency code instead.
17152b310afcffe8b6aed6fa0c1e9e8b0ade6f0a672Elliott Hughes        chars = icuCurrencyCode.getBuffer();
17252b310afcffe8b6aed6fa0c1e9e8b0ade6f0a672Elliott Hughes        charCount = icuCurrencyCode.length();
173adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project    }
17452b310afcffe8b6aed6fa0c1e9e8b0ade6f0a672Elliott Hughes    return (charCount == 0) ? NULL : env->NewString(chars, charCount);
175adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project}
176adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project
17752b310afcffe8b6aed6fa0c1e9e8b0ade6f0a672Elliott Hughesstatic jstring ICU_getCurrencySymbol(JNIEnv* env, jclass, jstring locale, jstring currencyCode) {
17852b310afcffe8b6aed6fa0c1e9e8b0ade6f0a672Elliott Hughes    // We can't use ucurr_getName because it doesn't distinguish between using data root from
17952b310afcffe8b6aed6fa0c1e9e8b0ade6f0a672Elliott Hughes    // the root locale and parroting back the input because it's never heard of the currency code.
18005960876dff6a5b686821eed8f7ae7cef5af4f50Elliott Hughes    ScopedUtfChars localeName(env, locale);
181adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project    UErrorCode status = U_ZERO_ERROR;
182947eeb80f985827209c9500851e288504d58ec2eclaireho    ScopedResourceBundle currLoc(ures_open(U_ICUDATA_CURR, localeName.c_str(), &status));
18370fa193185427473999f1cda823c1ec408acd2baElliott Hughes    if (U_FAILURE(status)) {
184adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project        return NULL;
185adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project    }
186adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project
187947eeb80f985827209c9500851e288504d58ec2eclaireho    ScopedResourceBundle currencies(ures_getByKey(currLoc.get(), "Currencies", NULL, &status));
18870fa193185427473999f1cda823c1ec408acd2baElliott Hughes    if (U_FAILURE(status)) {
189adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project        return NULL;
190adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project    }
191adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project
19205960876dff6a5b686821eed8f7ae7cef5af4f50Elliott Hughes    ScopedUtfChars currency(env, currencyCode);
19305960876dff6a5b686821eed8f7ae7cef5af4f50Elliott Hughes    ScopedResourceBundle currencyElems(ures_getByKey(currencies.get(), currency.c_str(), NULL, &status));
19470fa193185427473999f1cda823c1ec408acd2baElliott Hughes    if (U_FAILURE(status)) {
195adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project        return NULL;
196adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project    }
197adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project
19852b310afcffe8b6aed6fa0c1e9e8b0ade6f0a672Elliott Hughes    int32_t charCount;
19952b310afcffe8b6aed6fa0c1e9e8b0ade6f0a672Elliott Hughes    const jchar* chars = ures_getStringByIndex(currencyElems.get(), 0, &charCount, &status);
20070fa193185427473999f1cda823c1ec408acd2baElliott Hughes    if (U_FAILURE(status)) {
201adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project        return NULL;
202adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project    }
20352b310afcffe8b6aed6fa0c1e9e8b0ade6f0a672Elliott Hughes    return (charCount == 0) ? NULL : env->NewString(chars, charCount);
204adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project}
205adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project
206e22935d3c7040c22b48d53bd18878844f381287cElliott Hughesstatic jstring ICU_getDisplayCountryNative(JNIEnv* env, jclass, jstring targetLocale, jstring locale) {
207adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project    Locale loc = getLocale(env, locale);
208adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project    Locale targetLoc = getLocale(env, targetLocale);
20970fa193185427473999f1cda823c1ec408acd2baElliott Hughes    UnicodeString str;
21070fa193185427473999f1cda823c1ec408acd2baElliott Hughes    targetLoc.getDisplayCountry(loc, str);
21170fa193185427473999f1cda823c1ec408acd2baElliott Hughes    return env->NewString(str.getBuffer(), str.length());
212adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project}
213adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project
214e22935d3c7040c22b48d53bd18878844f381287cElliott Hughesstatic jstring ICU_getDisplayLanguageNative(JNIEnv* env, jclass, jstring targetLocale, jstring locale) {
215adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project    Locale loc = getLocale(env, locale);
216adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project    Locale targetLoc = getLocale(env, targetLocale);
21770fa193185427473999f1cda823c1ec408acd2baElliott Hughes    UnicodeString str;
21870fa193185427473999f1cda823c1ec408acd2baElliott Hughes    targetLoc.getDisplayLanguage(loc, str);
21970fa193185427473999f1cda823c1ec408acd2baElliott Hughes    return env->NewString(str.getBuffer(), str.length());
220adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project}
221adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project
222e22935d3c7040c22b48d53bd18878844f381287cElliott Hughesstatic jstring ICU_getDisplayVariantNative(JNIEnv* env, jclass, jstring targetLocale, jstring locale) {
223adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project    Locale loc = getLocale(env, locale);
224adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project    Locale targetLoc = getLocale(env, targetLocale);
22570fa193185427473999f1cda823c1ec408acd2baElliott Hughes    UnicodeString str;
22670fa193185427473999f1cda823c1ec408acd2baElliott Hughes    targetLoc.getDisplayVariant(loc, str);
22770fa193185427473999f1cda823c1ec408acd2baElliott Hughes    return env->NewString(str.getBuffer(), str.length());
228adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project}
229adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project
230e22935d3c7040c22b48d53bd18878844f381287cElliott Hughesstatic jstring ICU_getISO3CountryNative(JNIEnv* env, jclass, jstring locale) {
231adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project    Locale loc = getLocale(env, locale);
2322e3a41defb42a97b463194d859d2d4088a600fd8Elliott Hughes    return env->NewStringUTF(loc.getISO3Country());
233adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project}
234adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project
235e22935d3c7040c22b48d53bd18878844f381287cElliott Hughesstatic jstring ICU_getISO3LanguageNative(JNIEnv* env, jclass, jstring locale) {
236adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project    Locale loc = getLocale(env, locale);
2372e3a41defb42a97b463194d859d2d4088a600fd8Elliott Hughes    return env->NewStringUTF(loc.getISO3Language());
238adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project}
239adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project
240e22935d3c7040c22b48d53bd18878844f381287cElliott Hughesstatic jobjectArray ICU_getISOCountriesNative(JNIEnv* env, jclass) {
2412e3a41defb42a97b463194d859d2d4088a600fd8Elliott Hughes    return toStringArray(env, Locale::getISOCountries());
2422e3a41defb42a97b463194d859d2d4088a600fd8Elliott Hughes}
243adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project
244e22935d3c7040c22b48d53bd18878844f381287cElliott Hughesstatic jobjectArray ICU_getISOLanguagesNative(JNIEnv* env, jclass) {
2452e3a41defb42a97b463194d859d2d4088a600fd8Elliott Hughes    return toStringArray(env, Locale::getISOLanguages());
246adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project}
247adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project
248e22935d3c7040c22b48d53bd18878844f381287cElliott Hughesstatic jobjectArray ICU_getAvailableLocalesNative(JNIEnv* env, jclass) {
249ddafeb1d73cfe504720d10a2634b5858fc4cc413Elliott Hughes    return toStringArray(env, uloc_countAvailable, uloc_getAvailable);
250f9157eaea53923d3dbe6a521b29427819052f176Elliott Hughes}
251f9157eaea53923d3dbe6a521b29427819052f176Elliott Hughes
252e22935d3c7040c22b48d53bd18878844f381287cElliott Hughesstatic jobjectArray ICU_getAvailableBreakIteratorLocalesNative(JNIEnv* env, jclass) {
253ddafeb1d73cfe504720d10a2634b5858fc4cc413Elliott Hughes    return toStringArray(env, ubrk_countAvailable, ubrk_getAvailable);
254f9157eaea53923d3dbe6a521b29427819052f176Elliott Hughes}
255f9157eaea53923d3dbe6a521b29427819052f176Elliott Hughes
256e22935d3c7040c22b48d53bd18878844f381287cElliott Hughesstatic jobjectArray ICU_getAvailableCalendarLocalesNative(JNIEnv* env, jclass) {
257ddafeb1d73cfe504720d10a2634b5858fc4cc413Elliott Hughes    return toStringArray(env, ucal_countAvailable, ucal_getAvailable);
258f9157eaea53923d3dbe6a521b29427819052f176Elliott Hughes}
259f9157eaea53923d3dbe6a521b29427819052f176Elliott Hughes
260e22935d3c7040c22b48d53bd18878844f381287cElliott Hughesstatic jobjectArray ICU_getAvailableCollatorLocalesNative(JNIEnv* env, jclass) {
261ddafeb1d73cfe504720d10a2634b5858fc4cc413Elliott Hughes    return toStringArray(env, ucol_countAvailable, ucol_getAvailable);
262f9157eaea53923d3dbe6a521b29427819052f176Elliott Hughes}
263f9157eaea53923d3dbe6a521b29427819052f176Elliott Hughes
264e22935d3c7040c22b48d53bd18878844f381287cElliott Hughesstatic jobjectArray ICU_getAvailableDateFormatLocalesNative(JNIEnv* env, jclass) {
265ddafeb1d73cfe504720d10a2634b5858fc4cc413Elliott Hughes    return toStringArray(env, udat_countAvailable, udat_getAvailable);
266f9157eaea53923d3dbe6a521b29427819052f176Elliott Hughes}
267f9157eaea53923d3dbe6a521b29427819052f176Elliott Hughes
268e22935d3c7040c22b48d53bd18878844f381287cElliott Hughesstatic jobjectArray ICU_getAvailableNumberFormatLocalesNative(JNIEnv* env, jclass) {
269ddafeb1d73cfe504720d10a2634b5858fc4cc413Elliott Hughes    return toStringArray(env, unum_countAvailable, unum_getAvailable);
270f9157eaea53923d3dbe6a521b29427819052f176Elliott Hughes}
271f9157eaea53923d3dbe6a521b29427819052f176Elliott Hughes
27244e0e560c92338110953ce806df475fedcdf926eBrian Carlstromstatic bool getDayIntVector(JNIEnv*, UResourceBundle* gregorian, int* values) {
273adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project    // get the First day of week and the minimal days in first week numbers
2742e3a41defb42a97b463194d859d2d4088a600fd8Elliott Hughes    UErrorCode status = U_ZERO_ERROR;
27570fa193185427473999f1cda823c1ec408acd2baElliott Hughes    ScopedResourceBundle gregorianElems(ures_getByKey(gregorian, "DateTimeElements", NULL, &status));
2762e3a41defb42a97b463194d859d2d4088a600fd8Elliott Hughes    if (U_FAILURE(status)) {
27733aa6eb602478e7f51ac16f30c88db3566022886Elliott Hughes        return false;
278adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project    }
279adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project
280adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project    int intVectSize;
28170fa193185427473999f1cda823c1ec408acd2baElliott Hughes    const int* result = ures_getIntVector(gregorianElems.get(), &intVectSize, &status);
28270fa193185427473999f1cda823c1ec408acd2baElliott Hughes    if (U_FAILURE(status) || intVectSize != 2) {
28333aa6eb602478e7f51ac16f30c88db3566022886Elliott Hughes        return false;
284adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project    }
28533aa6eb602478e7f51ac16f30c88db3566022886Elliott Hughes
28670fa193185427473999f1cda823c1ec408acd2baElliott Hughes    values[0] = result[0];
28770fa193185427473999f1cda823c1ec408acd2baElliott Hughes    values[1] = result[1];
28833aa6eb602478e7f51ac16f30c88db3566022886Elliott Hughes    return true;
289adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project}
290adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project
291ddafeb1d73cfe504720d10a2634b5858fc4cc413Elliott Hughes// This allows you to leave extra space at the beginning or end of the array to support the
292ddafeb1d73cfe504720d10a2634b5858fc4cc413Elliott Hughes// month names and day names arrays.
293ddafeb1d73cfe504720d10a2634b5858fc4cc413Elliott Hughesstatic jobjectArray toStringArray(JNIEnv* env, UResourceBundle* rb, size_t size, int capacity, size_t offset) {
294ddafeb1d73cfe504720d10a2634b5858fc4cc413Elliott Hughes    if (capacity == -1) {
295ddafeb1d73cfe504720d10a2634b5858fc4cc413Elliott Hughes        capacity = size;
296ddafeb1d73cfe504720d10a2634b5858fc4cc413Elliott Hughes    }
297ddafeb1d73cfe504720d10a2634b5858fc4cc413Elliott Hughes    jobjectArray result = env->NewObjectArray(capacity, JniConstants::stringClass, NULL);
298ddafeb1d73cfe504720d10a2634b5858fc4cc413Elliott Hughes    if (result == NULL) {
299adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project        return NULL;
300adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project    }
301ddafeb1d73cfe504720d10a2634b5858fc4cc413Elliott Hughes    UErrorCode status = U_ZERO_ERROR;
302ddafeb1d73cfe504720d10a2634b5858fc4cc413Elliott Hughes    for (size_t i = 0; i < size; ++i) {
303ddafeb1d73cfe504720d10a2634b5858fc4cc413Elliott Hughes        int charCount;
304ddafeb1d73cfe504720d10a2634b5858fc4cc413Elliott Hughes        const jchar* chars = ures_getStringByIndex(rb, i, &charCount, &status);
305ddafeb1d73cfe504720d10a2634b5858fc4cc413Elliott Hughes        if (U_FAILURE(status)) {
306ddafeb1d73cfe504720d10a2634b5858fc4cc413Elliott Hughes            return NULL;
307ddafeb1d73cfe504720d10a2634b5858fc4cc413Elliott Hughes        }
308ddafeb1d73cfe504720d10a2634b5858fc4cc413Elliott Hughes        ScopedLocalRef<jstring> s(env, env->NewString(chars, charCount));
309ddafeb1d73cfe504720d10a2634b5858fc4cc413Elliott Hughes        if (env->ExceptionCheck()) {
310ddafeb1d73cfe504720d10a2634b5858fc4cc413Elliott Hughes            return NULL;
311ddafeb1d73cfe504720d10a2634b5858fc4cc413Elliott Hughes        }
312ddafeb1d73cfe504720d10a2634b5858fc4cc413Elliott Hughes        env->SetObjectArrayElement(result, offset + i, s.get());
313ddafeb1d73cfe504720d10a2634b5858fc4cc413Elliott Hughes        if (env->ExceptionCheck()) {
314ddafeb1d73cfe504720d10a2634b5858fc4cc413Elliott Hughes            return NULL;
315ddafeb1d73cfe504720d10a2634b5858fc4cc413Elliott Hughes        }
316ddafeb1d73cfe504720d10a2634b5858fc4cc413Elliott Hughes    }
317ddafeb1d73cfe504720d10a2634b5858fc4cc413Elliott Hughes    return result;
318ddafeb1d73cfe504720d10a2634b5858fc4cc413Elliott Hughes}
319adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project
320ddafeb1d73cfe504720d10a2634b5858fc4cc413Elliott Hughesstatic jobjectArray getAmPmMarkers(JNIEnv* env, UResourceBundle* gregorian) {
321ddafeb1d73cfe504720d10a2634b5858fc4cc413Elliott Hughes    UErrorCode status = U_ZERO_ERROR;
322ddafeb1d73cfe504720d10a2634b5858fc4cc413Elliott Hughes    ScopedResourceBundle amPmMarkers(ures_getByKey(gregorian, "AmPmMarkers", NULL, &status));
3232e3a41defb42a97b463194d859d2d4088a600fd8Elliott Hughes    if (U_FAILURE(status)) {
324adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project        return NULL;
325adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project    }
326ddafeb1d73cfe504720d10a2634b5858fc4cc413Elliott Hughes    return toStringArray(env, amPmMarkers.get(), ures_getSize(amPmMarkers.get()), -1, 0);
327adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project}
328adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project
32970fa193185427473999f1cda823c1ec408acd2baElliott Hughesstatic jobjectArray getEras(JNIEnv* env, UResourceBundle* gregorian) {
330adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project    UErrorCode status = U_ZERO_ERROR;
331ddafeb1d73cfe504720d10a2634b5858fc4cc413Elliott Hughes    ScopedResourceBundle eras(ures_getByKey(gregorian, "eras", NULL, &status));
33270fa193185427473999f1cda823c1ec408acd2baElliott Hughes    if (U_FAILURE(status)) {
333adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project        return NULL;
334adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project    }
335ddafeb1d73cfe504720d10a2634b5858fc4cc413Elliott Hughes    ScopedResourceBundle abbreviatedEras(ures_getByKey(eras.get(), "abbreviated", NULL, &status));
33670fa193185427473999f1cda823c1ec408acd2baElliott Hughes    if (U_FAILURE(status)) {
337adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project        return NULL;
338adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project    }
339ddafeb1d73cfe504720d10a2634b5858fc4cc413Elliott Hughes    return toStringArray(env, abbreviatedEras.get(), ures_getSize(abbreviatedEras.get()), -1, 0);
340adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project}
341adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project
342143e8c9cf91cfc01c3c91c8e93cad661ec7554eeElliott Hughesenum NameType { REGULAR, STAND_ALONE };
343143e8c9cf91cfc01c3c91c8e93cad661ec7554eeElliott Hughesenum NameWidth { LONG, SHORT };
344143e8c9cf91cfc01c3c91c8e93cad661ec7554eeElliott Hughesstatic jobjectArray getNames(JNIEnv* env, UResourceBundle* namesBundle, bool months, NameType type, NameWidth width) {
345143e8c9cf91cfc01c3c91c8e93cad661ec7554eeElliott Hughes    const char* typeKey = (type == REGULAR) ? "format" : "stand-alone";
346143e8c9cf91cfc01c3c91c8e93cad661ec7554eeElliott Hughes    const char* widthKey = (width == LONG) ? "wide" : "abbreviated";
347adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project    UErrorCode status = U_ZERO_ERROR;
348143e8c9cf91cfc01c3c91c8e93cad661ec7554eeElliott Hughes    ScopedResourceBundle formatBundle(ures_getByKey(namesBundle, typeKey, NULL, &status));
349143e8c9cf91cfc01c3c91c8e93cad661ec7554eeElliott Hughes    ScopedResourceBundle valuesBundle(ures_getByKey(formatBundle.get(), widthKey, NULL, &status));
35070fa193185427473999f1cda823c1ec408acd2baElliott Hughes    if (U_FAILURE(status)) {
351adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project        return NULL;
352adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project    }
3538044bf6b446c93cd29c5753544246316f269064fElliott Hughes
354143e8c9cf91cfc01c3c91c8e93cad661ec7554eeElliott Hughes    // The months array has a trailing empty string. The days array has a leading empty string.
355143e8c9cf91cfc01c3c91c8e93cad661ec7554eeElliott Hughes    int count = ures_getSize(valuesBundle.get());
356ddafeb1d73cfe504720d10a2634b5858fc4cc413Elliott Hughes    int offset = months ? 0 : 1;
357ddafeb1d73cfe504720d10a2634b5858fc4cc413Elliott Hughes    jobjectArray result = toStringArray(env, valuesBundle.get(), count, count + 1, offset);
358e2377cdd707b830d07a5708216834f7ac76ee3e1Elliott Hughes    ScopedLocalRef<jstring> emptyString(env, env->NewStringUTF(""));
359e2377cdd707b830d07a5708216834f7ac76ee3e1Elliott Hughes    env->SetObjectArrayElement(result, months ? count : 0, emptyString.get());
360143e8c9cf91cfc01c3c91c8e93cad661ec7554eeElliott Hughes    return result;
361adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project}
362adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project
36333aa6eb602478e7f51ac16f30c88db3566022886Elliott Hughesstatic void setIntegerField(JNIEnv* env, jobject obj, const char* fieldName, int value) {
36494782d07dd2d65b1a37fddca68eb9a9ac81ada4aElliott Hughes    ScopedLocalRef<jobject> integerValue(env, integerValueOf(env, value));
365a9f5c16a864ff63ba63f810410f8a27c086d5d52Elliott Hughes    jfieldID fid = env->GetFieldID(JniConstants::localeDataClass, fieldName, "Ljava/lang/Integer;");
36694782d07dd2d65b1a37fddca68eb9a9ac81ada4aElliott Hughes    env->SetObjectField(obj, fid, integerValue.get());
36733aa6eb602478e7f51ac16f30c88db3566022886Elliott Hughes}
368adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project
36933aa6eb602478e7f51ac16f30c88db3566022886Elliott Hughesstatic void setStringField(JNIEnv* env, jobject obj, const char* fieldName, jstring value) {
370a9f5c16a864ff63ba63f810410f8a27c086d5d52Elliott Hughes    jfieldID fid = env->GetFieldID(JniConstants::localeDataClass, fieldName, "Ljava/lang/String;");
37133aa6eb602478e7f51ac16f30c88db3566022886Elliott Hughes    env->SetObjectField(obj, fid, value);
372e2377cdd707b830d07a5708216834f7ac76ee3e1Elliott Hughes    env->DeleteLocalRef(value);
37333aa6eb602478e7f51ac16f30c88db3566022886Elliott Hughes}
374adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project
37533aa6eb602478e7f51ac16f30c88db3566022886Elliott Hughesstatic void setStringArrayField(JNIEnv* env, jobject obj, const char* fieldName, jobjectArray value) {
376a9f5c16a864ff63ba63f810410f8a27c086d5d52Elliott Hughes    jfieldID fid = env->GetFieldID(JniConstants::localeDataClass, fieldName, "[Ljava/lang/String;");
37733aa6eb602478e7f51ac16f30c88db3566022886Elliott Hughes    env->SetObjectField(obj, fid, value);
37833aa6eb602478e7f51ac16f30c88db3566022886Elliott Hughes}
379adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project
38033aa6eb602478e7f51ac16f30c88db3566022886Elliott Hughesstatic void setStringField(JNIEnv* env, jobject obj, const char* fieldName, UResourceBundle* bundle, int index) {
38133aa6eb602478e7f51ac16f30c88db3566022886Elliott Hughes    UErrorCode status = U_ZERO_ERROR;
38233aa6eb602478e7f51ac16f30c88db3566022886Elliott Hughes    int charCount;
38333aa6eb602478e7f51ac16f30c88db3566022886Elliott Hughes    const UChar* chars = ures_getStringByIndex(bundle, index, &charCount, &status);
38433aa6eb602478e7f51ac16f30c88db3566022886Elliott Hughes    if (U_SUCCESS(status)) {
38533aa6eb602478e7f51ac16f30c88db3566022886Elliott Hughes        setStringField(env, obj, fieldName, env->NewString(chars, charCount));
38690aa512eb7b126deb8d752b7474c30d3f73507b2Elliott Hughes    } else {
387679cf68b607e9b4a3beb8bcdee06868ae583386fSteve Block        ALOGE("Error setting String field %s from ICU resource: %s", fieldName, u_errorName(status));
388b7e820b92c7345cdc0cd4fea50954289ae66eb67Elliott Hughes    }
389b7e820b92c7345cdc0cd4fea50954289ae66eb67Elliott Hughes}
390b7e820b92c7345cdc0cd4fea50954289ae66eb67Elliott Hughes
3913be1277a4644a149c158f99febbab890fc7b708fclairehostatic bool setStringField(JNIEnv* env, jobject obj, const char* key, const char* fieldName, UResourceBundle* bundle) {
3923be1277a4644a149c158f99febbab890fc7b708fclaireho    if (bundle == NULL) {
3933be1277a4644a149c158f99febbab890fc7b708fclaireho        return false;
3943be1277a4644a149c158f99febbab890fc7b708fclaireho    }
395b7e820b92c7345cdc0cd4fea50954289ae66eb67Elliott Hughes    UErrorCode status = U_ZERO_ERROR;
396b7e820b92c7345cdc0cd4fea50954289ae66eb67Elliott Hughes    int charCount;
3973be1277a4644a149c158f99febbab890fc7b708fclaireho    const UChar* chars = ures_getStringByKey(bundle, key, &charCount, &status);
3983be1277a4644a149c158f99febbab890fc7b708fclaireho    if (U_SUCCESS(status)) {
3993be1277a4644a149c158f99febbab890fc7b708fclaireho        setStringField(env, obj, fieldName, env->NewString(chars, charCount));
4003be1277a4644a149c158f99febbab890fc7b708fclaireho        return true;
4013be1277a4644a149c158f99febbab890fc7b708fclaireho    } else {
4023be1277a4644a149c158f99febbab890fc7b708fclaireho        // Missing item in current resource bundle but not an error.
4033be1277a4644a149c158f99febbab890fc7b708fclaireho        return false;
4043be1277a4644a149c158f99febbab890fc7b708fclaireho    }
4053be1277a4644a149c158f99febbab890fc7b708fclaireho}
4063be1277a4644a149c158f99febbab890fc7b708fclaireho
4073be1277a4644a149c158f99febbab890fc7b708fclairehostatic void setStringField(JNIEnv* env, jobject obj, const char* key, const char* fieldName,
4083be1277a4644a149c158f99febbab890fc7b708fclaireho    UResourceBundle* bundle, UResourceBundle* fallbackBundle) {
4093be1277a4644a149c158f99febbab890fc7b708fclaireho    if (!setStringField(env, obj, key, fieldName, bundle) && fallbackBundle != NULL) {
4103be1277a4644a149c158f99febbab890fc7b708fclaireho        setStringField(env, obj, key, fieldName, fallbackBundle);
4113be1277a4644a149c158f99febbab890fc7b708fclaireho    }
4123be1277a4644a149c158f99febbab890fc7b708fclaireho}
4133be1277a4644a149c158f99febbab890fc7b708fclaireho
4143be1277a4644a149c158f99febbab890fc7b708fclairehostatic bool setCharField(JNIEnv* env, jobject obj, const char* key, const char* fieldName,
4153be1277a4644a149c158f99febbab890fc7b708fclaireho    UResourceBundle* bundle) {
4163be1277a4644a149c158f99febbab890fc7b708fclaireho    if (bundle == NULL) {
4173be1277a4644a149c158f99febbab890fc7b708fclaireho        return false;
4183be1277a4644a149c158f99febbab890fc7b708fclaireho    }
4193be1277a4644a149c158f99febbab890fc7b708fclaireho    UErrorCode status = U_ZERO_ERROR;
4203be1277a4644a149c158f99febbab890fc7b708fclaireho    int charCount;
4213be1277a4644a149c158f99febbab890fc7b708fclaireho    const UChar* chars = ures_getStringByKey(bundle, key, &charCount, &status);
422b7e820b92c7345cdc0cd4fea50954289ae66eb67Elliott Hughes    if (U_SUCCESS(status)) {
423a9f5c16a864ff63ba63f810410f8a27c086d5d52Elliott Hughes        jfieldID fid = env->GetFieldID(JniConstants::localeDataClass, fieldName, "C");
424b7e820b92c7345cdc0cd4fea50954289ae66eb67Elliott Hughes        env->SetCharField(obj, fid, chars[0]);
4253be1277a4644a149c158f99febbab890fc7b708fclaireho        return true;
426b7e820b92c7345cdc0cd4fea50954289ae66eb67Elliott Hughes    } else {
4273be1277a4644a149c158f99febbab890fc7b708fclaireho        // Missing item in current resource bundle but not an error.
4283be1277a4644a149c158f99febbab890fc7b708fclaireho        return false;
4293be1277a4644a149c158f99febbab890fc7b708fclaireho    }
4303be1277a4644a149c158f99febbab890fc7b708fclaireho}
4313be1277a4644a149c158f99febbab890fc7b708fclaireho
4323be1277a4644a149c158f99febbab890fc7b708fclairehostatic void setCharField(JNIEnv* env, jobject obj, const char* key, const char* fieldName,
4333be1277a4644a149c158f99febbab890fc7b708fclaireho    UResourceBundle* bundle, UResourceBundle* fallbackBundle) {
4343be1277a4644a149c158f99febbab890fc7b708fclaireho    if (!setCharField(env, obj, key, fieldName, bundle) && fallbackBundle != NULL) {
4353be1277a4644a149c158f99febbab890fc7b708fclaireho        setCharField(env, obj, key, fieldName, fallbackBundle);
4363be1277a4644a149c158f99febbab890fc7b708fclaireho    }
4373be1277a4644a149c158f99febbab890fc7b708fclaireho}
4383be1277a4644a149c158f99febbab890fc7b708fclaireho
439e2377cdd707b830d07a5708216834f7ac76ee3e1Elliott Hughesstatic void setNumberSymbols(JNIEnv* env, jobject obj, UResourceBundle* numberSymbols, UResourceBundle* fallbackNumberSymbols) {
4403be1277a4644a149c158f99febbab890fc7b708fclaireho    setCharField(env, obj, "decimal", "decimalSeparator", numberSymbols, fallbackNumberSymbols);
4413be1277a4644a149c158f99febbab890fc7b708fclaireho    setCharField(env, obj, "group", "groupingSeparator", numberSymbols, fallbackNumberSymbols);
4423be1277a4644a149c158f99febbab890fc7b708fclaireho    setCharField(env, obj, "list", "patternSeparator", numberSymbols, fallbackNumberSymbols);
4433be1277a4644a149c158f99febbab890fc7b708fclaireho    setCharField(env, obj, "percentSign", "percent", numberSymbols, fallbackNumberSymbols);
4443be1277a4644a149c158f99febbab890fc7b708fclaireho    setCharField(env, obj, "perMille", "perMill", numberSymbols, fallbackNumberSymbols);
4453be1277a4644a149c158f99febbab890fc7b708fclaireho    setCharField(env, obj, "decimal", "monetarySeparator", numberSymbols, fallbackNumberSymbols);
4463be1277a4644a149c158f99febbab890fc7b708fclaireho    setCharField(env, obj, "minusSign", "minusSign", numberSymbols, fallbackNumberSymbols);
4473be1277a4644a149c158f99febbab890fc7b708fclaireho    setStringField(env, obj, "exponential", "exponentSeparator", numberSymbols, fallbackNumberSymbols);
4483be1277a4644a149c158f99febbab890fc7b708fclaireho    setStringField(env, obj, "infinity", "infinity", numberSymbols, fallbackNumberSymbols);
4493be1277a4644a149c158f99febbab890fc7b708fclaireho    setStringField(env, obj, "nan", "NaN", numberSymbols, fallbackNumberSymbols);
4503be1277a4644a149c158f99febbab890fc7b708fclaireho}
4513be1277a4644a149c158f99febbab890fc7b708fclaireho
4523be1277a4644a149c158f99febbab890fc7b708fclairehostatic void setZeroDigitToDefault(JNIEnv* env, jobject obj) {
453e2377cdd707b830d07a5708216834f7ac76ee3e1Elliott Hughes    static jfieldID fid = env->GetFieldID(JniConstants::localeDataClass, "zeroDigit", "C");
4543be1277a4644a149c158f99febbab890fc7b708fclaireho    env->SetCharField(obj, fid, '0');
4553be1277a4644a149c158f99febbab890fc7b708fclaireho}
4563be1277a4644a149c158f99febbab890fc7b708fclaireho
4573be1277a4644a149c158f99febbab890fc7b708fclairehostatic void setZeroDigit(JNIEnv* env, jobject obj, bool isLatn, char* buffer) {
4583be1277a4644a149c158f99febbab890fc7b708fclaireho    if (isLatn || buffer == NULL || buffer[0] == '\0') {
4593be1277a4644a149c158f99febbab890fc7b708fclaireho        return setZeroDigitToDefault(env, obj);
4603be1277a4644a149c158f99febbab890fc7b708fclaireho    }
4613be1277a4644a149c158f99febbab890fc7b708fclaireho    UErrorCode status = U_ZERO_ERROR;
4623be1277a4644a149c158f99febbab890fc7b708fclaireho    ScopedResourceBundle numSystemRoot(ures_openDirect(NULL, "numberingSystems", &status));
4633be1277a4644a149c158f99febbab890fc7b708fclaireho    if (U_FAILURE(status)) {
4643be1277a4644a149c158f99febbab890fc7b708fclaireho        return setZeroDigitToDefault(env, obj);
4653be1277a4644a149c158f99febbab890fc7b708fclaireho    }
4663be1277a4644a149c158f99febbab890fc7b708fclaireho    ScopedResourceBundle numSystem(ures_getByKey(numSystemRoot.get(), "numberingSystems", NULL, &status));
4673be1277a4644a149c158f99febbab890fc7b708fclaireho    if (U_FAILURE(status)) {
4683be1277a4644a149c158f99febbab890fc7b708fclaireho        return setZeroDigitToDefault(env, obj);
4693be1277a4644a149c158f99febbab890fc7b708fclaireho    }
4703be1277a4644a149c158f99febbab890fc7b708fclaireho    ScopedResourceBundle nonLatnSystem(ures_getByKey(numSystem.get(), buffer, NULL, &status));
4713be1277a4644a149c158f99febbab890fc7b708fclaireho    if (U_FAILURE(status)) {
4723be1277a4644a149c158f99febbab890fc7b708fclaireho        return setZeroDigitToDefault(env, obj);
4733be1277a4644a149c158f99febbab890fc7b708fclaireho    }
4743be1277a4644a149c158f99febbab890fc7b708fclaireho    int32_t charCount = 0;
4753be1277a4644a149c158f99febbab890fc7b708fclaireho    const UChar* chars = ures_getStringByKey(nonLatnSystem.get(), "desc", &charCount, &status);
4763be1277a4644a149c158f99febbab890fc7b708fclaireho    if (charCount == 0) {
4773be1277a4644a149c158f99febbab890fc7b708fclaireho        setZeroDigitToDefault(env, obj);
4783be1277a4644a149c158f99febbab890fc7b708fclaireho    } else {
479e2377cdd707b830d07a5708216834f7ac76ee3e1Elliott Hughes        static jfieldID fid = env->GetFieldID(JniConstants::localeDataClass, "zeroDigit", "C");
4803be1277a4644a149c158f99febbab890fc7b708fclaireho        env->SetCharField(obj, fid, chars[0]);
4813be1277a4644a149c158f99febbab890fc7b708fclaireho    }
4823be1277a4644a149c158f99febbab890fc7b708fclaireho}
4833be1277a4644a149c158f99febbab890fc7b708fclaireho
4843be1277a4644a149c158f99febbab890fc7b708fclairehostatic void setNumberElements(JNIEnv* env, jobject obj, UResourceBundle* numberElements) {
4853be1277a4644a149c158f99febbab890fc7b708fclaireho    UErrorCode status = U_ZERO_ERROR;
4863be1277a4644a149c158f99febbab890fc7b708fclaireho    ScopedResourceBundle latnNumberRB(ures_getByKey(numberElements, "latn", NULL, &status));
4873be1277a4644a149c158f99febbab890fc7b708fclaireho    if (U_FAILURE(status)) {
48806f039b4c39bee1b0618f40d4d4af0d0f2bef2faSteve Block        ALOGW("Error getting ICU latn number elements system value: %s", u_errorName(status));
4893be1277a4644a149c158f99febbab890fc7b708fclaireho        return;
4903be1277a4644a149c158f99febbab890fc7b708fclaireho    }
4913be1277a4644a149c158f99febbab890fc7b708fclaireho    ScopedResourceBundle patternsRB(ures_getByKey(latnNumberRB.get(), "patterns", NULL, &status));
4923be1277a4644a149c158f99febbab890fc7b708fclaireho    if (U_FAILURE(status)) {
49306f039b4c39bee1b0618f40d4d4af0d0f2bef2faSteve Block        ALOGW("Error getting ICU latn number patterns value: %s", u_errorName(status));
4943be1277a4644a149c158f99febbab890fc7b708fclaireho        return;
4953be1277a4644a149c158f99febbab890fc7b708fclaireho    }
4963be1277a4644a149c158f99febbab890fc7b708fclaireho    // Get the patterns from the 'latn' numberElements
4973be1277a4644a149c158f99febbab890fc7b708fclaireho    // This is a temporary workaround for ICU ticket#8611.
4983be1277a4644a149c158f99febbab890fc7b708fclaireho    UResourceBundle* bundle = patternsRB.get();
4993be1277a4644a149c158f99febbab890fc7b708fclaireho    setStringField(env, obj, "currencyFormat", "currencyPattern", bundle);
5003be1277a4644a149c158f99febbab890fc7b708fclaireho    setStringField(env, obj, "decimalFormat", "numberPattern", bundle);
5013be1277a4644a149c158f99febbab890fc7b708fclaireho    setStringField(env, obj, "percentFormat", "percentPattern", bundle);
5023be1277a4644a149c158f99febbab890fc7b708fclaireho
5033be1277a4644a149c158f99febbab890fc7b708fclaireho    status = U_ZERO_ERROR;
5043be1277a4644a149c158f99febbab890fc7b708fclaireho    bool isLatn = false;
5053be1277a4644a149c158f99febbab890fc7b708fclaireho    char buffer[256];
5063be1277a4644a149c158f99febbab890fc7b708fclaireho    buffer[0] = '\0';
5073be1277a4644a149c158f99febbab890fc7b708fclaireho    ScopedResourceBundle defaultNumberElem(ures_getByKey(numberElements, "default", NULL, &status));
5083be1277a4644a149c158f99febbab890fc7b708fclaireho    if (U_SUCCESS(status)) {
5093be1277a4644a149c158f99febbab890fc7b708fclaireho        int32_t charCount = 256;
5103be1277a4644a149c158f99febbab890fc7b708fclaireho        ures_getUTF8String(defaultNumberElem.get(), buffer, &charCount, true, &status);
5113be1277a4644a149c158f99febbab890fc7b708fclaireho        buffer[charCount] = '\0';
5123be1277a4644a149c158f99febbab890fc7b708fclaireho        if (U_FAILURE(status)) {
51306f039b4c39bee1b0618f40d4d4af0d0f2bef2faSteve Block            ALOGW("Error getting ICU default number element system value: %s", u_errorName(status));
5143be1277a4644a149c158f99febbab890fc7b708fclaireho            // Use latn number symbols instead.
5153be1277a4644a149c158f99febbab890fc7b708fclaireho            isLatn = true;
5163be1277a4644a149c158f99febbab890fc7b708fclaireho        } else {
5173be1277a4644a149c158f99febbab890fc7b708fclaireho            isLatn = (strcmp(buffer, "latn") == 0);
5183be1277a4644a149c158f99febbab890fc7b708fclaireho        }
5193be1277a4644a149c158f99febbab890fc7b708fclaireho    } else {
5203be1277a4644a149c158f99febbab890fc7b708fclaireho        // Not default data, fallback to latn number elements.
5213be1277a4644a149c158f99febbab890fc7b708fclaireho        isLatn = true;
5223be1277a4644a149c158f99febbab890fc7b708fclaireho    }
5233be1277a4644a149c158f99febbab890fc7b708fclaireho
5243be1277a4644a149c158f99febbab890fc7b708fclaireho    status = U_ZERO_ERROR;
5253be1277a4644a149c158f99febbab890fc7b708fclaireho    setZeroDigit(env, obj, isLatn, buffer);
5263be1277a4644a149c158f99febbab890fc7b708fclaireho    if (isLatn) {
5273be1277a4644a149c158f99febbab890fc7b708fclaireho        ScopedResourceBundle symbolsRB(ures_getByKey(latnNumberRB.get(), "symbols", NULL, &status));
5283be1277a4644a149c158f99febbab890fc7b708fclaireho        if (U_SUCCESS(status)) {
5293be1277a4644a149c158f99febbab890fc7b708fclaireho            setNumberSymbols(env, obj, symbolsRB.get(), NULL);
5303be1277a4644a149c158f99febbab890fc7b708fclaireho        } else {
53106f039b4c39bee1b0618f40d4d4af0d0f2bef2faSteve Block            ALOGW("Missing ICU latn symbols system value: %s", u_errorName(status));
5323be1277a4644a149c158f99febbab890fc7b708fclaireho        }
5333be1277a4644a149c158f99febbab890fc7b708fclaireho    } else {
5343be1277a4644a149c158f99febbab890fc7b708fclaireho        // Get every symbol item from default numbering system first. If it does not
5353be1277a4644a149c158f99febbab890fc7b708fclaireho        // exist, get the symbol from latn numbering system.
5363be1277a4644a149c158f99febbab890fc7b708fclaireho        ScopedResourceBundle defaultNumberRB(ures_getByKey(numberElements, (const char*)buffer, NULL, &status));
5373be1277a4644a149c158f99febbab890fc7b708fclaireho        ScopedResourceBundle defaultSymbolsRB(ures_getByKey(defaultNumberRB.get(), "symbols", NULL, &status));
5383be1277a4644a149c158f99febbab890fc7b708fclaireho        if (U_FAILURE(status)) {
53906f039b4c39bee1b0618f40d4d4af0d0f2bef2faSteve Block            ALOGW("Missing ICU %s symbols system value: %s", buffer, u_errorName(status));
5403be1277a4644a149c158f99febbab890fc7b708fclaireho            isLatn = true;  // Fallback to latn symbols.
5413be1277a4644a149c158f99febbab890fc7b708fclaireho            status = U_ZERO_ERROR;
5423be1277a4644a149c158f99febbab890fc7b708fclaireho        }
5433be1277a4644a149c158f99febbab890fc7b708fclaireho        ScopedResourceBundle latnSymbolsRB(ures_getByKey(latnNumberRB.get(), "symbols", NULL, &status));
5443be1277a4644a149c158f99febbab890fc7b708fclaireho        if (isLatn && U_FAILURE(status)) {
5453be1277a4644a149c158f99febbab890fc7b708fclaireho            return;
5463be1277a4644a149c158f99febbab890fc7b708fclaireho        }
5473be1277a4644a149c158f99febbab890fc7b708fclaireho        setNumberSymbols(env, obj, defaultSymbolsRB.get(), latnSymbolsRB.get());
548adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project    }
54933aa6eb602478e7f51ac16f30c88db3566022886Elliott Hughes}
550adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project
551e22935d3c7040c22b48d53bd18878844f381287cElliott Hughesstatic jboolean ICU_initLocaleDataImpl(JNIEnv* env, jclass, jstring locale, jobject localeData) {
55205960876dff6a5b686821eed8f7ae7cef5af4f50Elliott Hughes    ScopedUtfChars localeName(env, locale);
553e2377cdd707b830d07a5708216834f7ac76ee3e1Elliott Hughes    if (localeName.c_str() == NULL) {
554e2377cdd707b830d07a5708216834f7ac76ee3e1Elliott Hughes        return JNI_FALSE;
555e2377cdd707b830d07a5708216834f7ac76ee3e1Elliott Hughes    }
556e2377cdd707b830d07a5708216834f7ac76ee3e1Elliott Hughes
55733aa6eb602478e7f51ac16f30c88db3566022886Elliott Hughes    UErrorCode status = U_ZERO_ERROR;
55821557bb6a8f35a2f9889adba449cac950c9d41b9Elliott Hughes    ScopedResourceBundle root(ures_open(NULL, localeName.c_str(), &status));
55933aa6eb602478e7f51ac16f30c88db3566022886Elliott Hughes    if (U_FAILURE(status)) {
560679cf68b607e9b4a3beb8bcdee06868ae583386fSteve Block        ALOGE("Error getting ICU resource bundle: %s", u_errorName(status));
56133aa6eb602478e7f51ac16f30c88db3566022886Elliott Hughes        return JNI_FALSE;
562adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project    }
563adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project
56433aa6eb602478e7f51ac16f30c88db3566022886Elliott Hughes    ScopedResourceBundle calendar(ures_getByKey(root.get(), "calendar", NULL, &status));
56533aa6eb602478e7f51ac16f30c88db3566022886Elliott Hughes    if (U_FAILURE(status)) {
566679cf68b607e9b4a3beb8bcdee06868ae583386fSteve Block        ALOGE("Error getting ICU calendar resource bundle: %s", u_errorName(status));
56733aa6eb602478e7f51ac16f30c88db3566022886Elliott Hughes        return JNI_FALSE;
568adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project    }
569adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project
57033aa6eb602478e7f51ac16f30c88db3566022886Elliott Hughes    ScopedResourceBundle gregorian(ures_getByKey(calendar.get(), "gregorian", NULL, &status));
57133aa6eb602478e7f51ac16f30c88db3566022886Elliott Hughes    if (U_FAILURE(status)) {
572679cf68b607e9b4a3beb8bcdee06868ae583386fSteve Block        ALOGE("Error getting ICU gregorian resource bundle: %s", u_errorName(status));
57333aa6eb602478e7f51ac16f30c88db3566022886Elliott Hughes        return JNI_FALSE;
574adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project    }
575adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project
576e22935d3c7040c22b48d53bd18878844f381287cElliott Hughes    int firstDayVals[] = { 0, 0 };
57733aa6eb602478e7f51ac16f30c88db3566022886Elliott Hughes    if (getDayIntVector(env, gregorian.get(), firstDayVals)) {
57833aa6eb602478e7f51ac16f30c88db3566022886Elliott Hughes        setIntegerField(env, localeData, "firstDayOfWeek", firstDayVals[0]);
57933aa6eb602478e7f51ac16f30c88db3566022886Elliott Hughes        setIntegerField(env, localeData, "minimalDaysInFirstWeek", firstDayVals[1]);
580adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project    }
581adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project
582e2377cdd707b830d07a5708216834f7ac76ee3e1Elliott Hughes    jobjectArray amPmMarkers = getAmPmMarkers(env, gregorian.get());
583e2377cdd707b830d07a5708216834f7ac76ee3e1Elliott Hughes    setStringArrayField(env, localeData, "amPm", amPmMarkers);
584e2377cdd707b830d07a5708216834f7ac76ee3e1Elliott Hughes    env->DeleteLocalRef(amPmMarkers);
585e2377cdd707b830d07a5708216834f7ac76ee3e1Elliott Hughes
586e2377cdd707b830d07a5708216834f7ac76ee3e1Elliott Hughes    jobjectArray eras = getEras(env, gregorian.get());
587e2377cdd707b830d07a5708216834f7ac76ee3e1Elliott Hughes    setStringArrayField(env, localeData, "eras", eras);
588e2377cdd707b830d07a5708216834f7ac76ee3e1Elliott Hughes    env->DeleteLocalRef(eras);
589adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project
590143e8c9cf91cfc01c3c91c8e93cad661ec7554eeElliott Hughes    ScopedResourceBundle dayNames(ures_getByKey(gregorian.get(), "dayNames", NULL, &status));
591143e8c9cf91cfc01c3c91c8e93cad661ec7554eeElliott Hughes    ScopedResourceBundle monthNames(ures_getByKey(gregorian.get(), "monthNames", NULL, &status));
592143e8c9cf91cfc01c3c91c8e93cad661ec7554eeElliott Hughes
593143e8c9cf91cfc01c3c91c8e93cad661ec7554eeElliott Hughes    // Get the regular month and weekday names.
594143e8c9cf91cfc01c3c91c8e93cad661ec7554eeElliott Hughes    jobjectArray longMonthNames = getNames(env, monthNames.get(), true, REGULAR, LONG);
595143e8c9cf91cfc01c3c91c8e93cad661ec7554eeElliott Hughes    jobjectArray shortMonthNames = getNames(env, monthNames.get(), true, REGULAR, SHORT);
596143e8c9cf91cfc01c3c91c8e93cad661ec7554eeElliott Hughes    jobjectArray longWeekdayNames = getNames(env, dayNames.get(), false, REGULAR, LONG);
597143e8c9cf91cfc01c3c91c8e93cad661ec7554eeElliott Hughes    jobjectArray shortWeekdayNames = getNames(env, dayNames.get(), false, REGULAR, SHORT);
598143e8c9cf91cfc01c3c91c8e93cad661ec7554eeElliott Hughes    setStringArrayField(env, localeData, "longMonthNames", longMonthNames);
599143e8c9cf91cfc01c3c91c8e93cad661ec7554eeElliott Hughes    setStringArrayField(env, localeData, "shortMonthNames", shortMonthNames);
600143e8c9cf91cfc01c3c91c8e93cad661ec7554eeElliott Hughes    setStringArrayField(env, localeData, "longWeekdayNames", longWeekdayNames);
601143e8c9cf91cfc01c3c91c8e93cad661ec7554eeElliott Hughes    setStringArrayField(env, localeData, "shortWeekdayNames", shortWeekdayNames);
602143e8c9cf91cfc01c3c91c8e93cad661ec7554eeElliott Hughes
603143e8c9cf91cfc01c3c91c8e93cad661ec7554eeElliott Hughes    // Get the stand-alone month and weekday names. If they're not available (as they aren't for
604143e8c9cf91cfc01c3c91c8e93cad661ec7554eeElliott Hughes    // English), we reuse the regular names. If we returned null to Java, the usual fallback
605143e8c9cf91cfc01c3c91c8e93cad661ec7554eeElliott Hughes    // mechanisms would come into play and we'd end up with the bogus stand-alone names from the
606143e8c9cf91cfc01c3c91c8e93cad661ec7554eeElliott Hughes    // root locale ("1" for January, and so on).
607143e8c9cf91cfc01c3c91c8e93cad661ec7554eeElliott Hughes    jobjectArray longStandAloneMonthNames = getNames(env, monthNames.get(), true, STAND_ALONE, LONG);
608143e8c9cf91cfc01c3c91c8e93cad661ec7554eeElliott Hughes    if (longStandAloneMonthNames == NULL) {
609143e8c9cf91cfc01c3c91c8e93cad661ec7554eeElliott Hughes        longStandAloneMonthNames = longMonthNames;
610143e8c9cf91cfc01c3c91c8e93cad661ec7554eeElliott Hughes    }
611143e8c9cf91cfc01c3c91c8e93cad661ec7554eeElliott Hughes    jobjectArray shortStandAloneMonthNames = getNames(env, monthNames.get(), true, STAND_ALONE, SHORT);
612143e8c9cf91cfc01c3c91c8e93cad661ec7554eeElliott Hughes    if (shortStandAloneMonthNames == NULL) {
613143e8c9cf91cfc01c3c91c8e93cad661ec7554eeElliott Hughes        shortStandAloneMonthNames = shortMonthNames;
614143e8c9cf91cfc01c3c91c8e93cad661ec7554eeElliott Hughes    }
615143e8c9cf91cfc01c3c91c8e93cad661ec7554eeElliott Hughes    jobjectArray longStandAloneWeekdayNames = getNames(env, dayNames.get(), false, STAND_ALONE, LONG);
616143e8c9cf91cfc01c3c91c8e93cad661ec7554eeElliott Hughes    if (longStandAloneWeekdayNames == NULL) {
617143e8c9cf91cfc01c3c91c8e93cad661ec7554eeElliott Hughes        longStandAloneWeekdayNames = longWeekdayNames;
618143e8c9cf91cfc01c3c91c8e93cad661ec7554eeElliott Hughes    }
619143e8c9cf91cfc01c3c91c8e93cad661ec7554eeElliott Hughes    jobjectArray shortStandAloneWeekdayNames = getNames(env, dayNames.get(), false, STAND_ALONE, SHORT);
620143e8c9cf91cfc01c3c91c8e93cad661ec7554eeElliott Hughes    if (shortStandAloneWeekdayNames == NULL) {
621143e8c9cf91cfc01c3c91c8e93cad661ec7554eeElliott Hughes        shortStandAloneWeekdayNames = shortWeekdayNames;
622143e8c9cf91cfc01c3c91c8e93cad661ec7554eeElliott Hughes    }
623143e8c9cf91cfc01c3c91c8e93cad661ec7554eeElliott Hughes    setStringArrayField(env, localeData, "longStandAloneMonthNames", longStandAloneMonthNames);
624143e8c9cf91cfc01c3c91c8e93cad661ec7554eeElliott Hughes    setStringArrayField(env, localeData, "shortStandAloneMonthNames", shortStandAloneMonthNames);
625143e8c9cf91cfc01c3c91c8e93cad661ec7554eeElliott Hughes    setStringArrayField(env, localeData, "longStandAloneWeekdayNames", longStandAloneWeekdayNames);
626143e8c9cf91cfc01c3c91c8e93cad661ec7554eeElliott Hughes    setStringArrayField(env, localeData, "shortStandAloneWeekdayNames", shortStandAloneWeekdayNames);
627143e8c9cf91cfc01c3c91c8e93cad661ec7554eeElliott Hughes
628143e8c9cf91cfc01c3c91c8e93cad661ec7554eeElliott Hughes    ScopedResourceBundle dateTimePatterns(ures_getByKey(gregorian.get(), "DateTimePatterns", NULL, &status));
62933aa6eb602478e7f51ac16f30c88db3566022886Elliott Hughes    if (U_SUCCESS(status)) {
630143e8c9cf91cfc01c3c91c8e93cad661ec7554eeElliott Hughes        setStringField(env, localeData, "fullTimeFormat", dateTimePatterns.get(), 0);
631143e8c9cf91cfc01c3c91c8e93cad661ec7554eeElliott Hughes        setStringField(env, localeData, "longTimeFormat", dateTimePatterns.get(), 1);
632143e8c9cf91cfc01c3c91c8e93cad661ec7554eeElliott Hughes        setStringField(env, localeData, "mediumTimeFormat", dateTimePatterns.get(), 2);
633143e8c9cf91cfc01c3c91c8e93cad661ec7554eeElliott Hughes        setStringField(env, localeData, "shortTimeFormat", dateTimePatterns.get(), 3);
634143e8c9cf91cfc01c3c91c8e93cad661ec7554eeElliott Hughes        setStringField(env, localeData, "fullDateFormat", dateTimePatterns.get(), 4);
635143e8c9cf91cfc01c3c91c8e93cad661ec7554eeElliott Hughes        setStringField(env, localeData, "longDateFormat", dateTimePatterns.get(), 5);
636143e8c9cf91cfc01c3c91c8e93cad661ec7554eeElliott Hughes        setStringField(env, localeData, "mediumDateFormat", dateTimePatterns.get(), 6);
637143e8c9cf91cfc01c3c91c8e93cad661ec7554eeElliott Hughes        setStringField(env, localeData, "shortDateFormat", dateTimePatterns.get(), 7);
638adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project    }
63933aa6eb602478e7f51ac16f30c88db3566022886Elliott Hughes    status = U_ZERO_ERROR;
640adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project
6413be1277a4644a149c158f99febbab890fc7b708fclaireho    // For numberPatterns and symbols.
64233aa6eb602478e7f51ac16f30c88db3566022886Elliott Hughes    ScopedResourceBundle numberElements(ures_getByKey(root.get(), "NumberElements", NULL, &status));
6433be1277a4644a149c158f99febbab890fc7b708fclaireho    if (U_SUCCESS(status)) {
6443be1277a4644a149c158f99febbab890fc7b708fclaireho        setNumberElements(env, localeData, numberElements.get());
645adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project    }
64633aa6eb602478e7f51ac16f30c88db3566022886Elliott Hughes    status = U_ZERO_ERROR;
647adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project
6487ff7010c01a090dcbf992c7223fbc8fde54b3018Elliott Hughes    jstring countryCode = env->NewStringUTF(Locale::createFromName(localeName.c_str()).getCountry());
64952b310afcffe8b6aed6fa0c1e9e8b0ade6f0a672Elliott Hughes    jstring internationalCurrencySymbol = ICU_getCurrencyCode(env, NULL, countryCode);
650e2377cdd707b830d07a5708216834f7ac76ee3e1Elliott Hughes    env->DeleteLocalRef(countryCode);
651e2377cdd707b830d07a5708216834f7ac76ee3e1Elliott Hughes    countryCode = NULL;
652e2377cdd707b830d07a5708216834f7ac76ee3e1Elliott Hughes
65333aa6eb602478e7f51ac16f30c88db3566022886Elliott Hughes    jstring currencySymbol = NULL;
65433aa6eb602478e7f51ac16f30c88db3566022886Elliott Hughes    if (internationalCurrencySymbol != NULL) {
65552b310afcffe8b6aed6fa0c1e9e8b0ade6f0a672Elliott Hughes        currencySymbol = ICU_getCurrencySymbol(env, NULL, locale, internationalCurrencySymbol);
656adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project    } else {
65733aa6eb602478e7f51ac16f30c88db3566022886Elliott Hughes        internationalCurrencySymbol = env->NewStringUTF("XXX");
658adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project    }
65933aa6eb602478e7f51ac16f30c88db3566022886Elliott Hughes    if (currencySymbol == NULL) {
66033aa6eb602478e7f51ac16f30c88db3566022886Elliott Hughes        // This is the UTF-8 encoding of U+00A4 (CURRENCY SIGN).
661ad884a7b8cae7a1c3d973c8032d4f78ef7c96b80Hao Feng        currencySymbol = env->NewStringUTF("\xc2\xa4");
662adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project    }
66333aa6eb602478e7f51ac16f30c88db3566022886Elliott Hughes    setStringField(env, localeData, "currencySymbol", currencySymbol);
66433aa6eb602478e7f51ac16f30c88db3566022886Elliott Hughes    setStringField(env, localeData, "internationalCurrencySymbol", internationalCurrencySymbol);
665adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project
66633aa6eb602478e7f51ac16f30c88db3566022886Elliott Hughes    return JNI_TRUE;
667adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project}
668adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project
669dbbdffce5ac97a0e93ef495adaacca3660b9ab21Elliott Hughesstatic jstring ICU_toLowerCase(JNIEnv* env, jclass, jstring javaString, jstring localeName) {
670dbbdffce5ac97a0e93ef495adaacca3660b9ab21Elliott Hughes    ScopedJavaUnicodeString scopedString(env, javaString);
671dbbdffce5ac97a0e93ef495adaacca3660b9ab21Elliott Hughes    UnicodeString& s(scopedString.unicodeString());
672dbbdffce5ac97a0e93ef495adaacca3660b9ab21Elliott Hughes    UnicodeString original(s);
673dbbdffce5ac97a0e93ef495adaacca3660b9ab21Elliott Hughes    s.toLower(Locale::createFromName(ScopedUtfChars(env, localeName).c_str()));
674dbbdffce5ac97a0e93ef495adaacca3660b9ab21Elliott Hughes    return s == original ? javaString : env->NewString(s.getBuffer(), s.length());
675dbbdffce5ac97a0e93ef495adaacca3660b9ab21Elliott Hughes}
676dbbdffce5ac97a0e93ef495adaacca3660b9ab21Elliott Hughes
677dbbdffce5ac97a0e93ef495adaacca3660b9ab21Elliott Hughesstatic jstring ICU_toUpperCase(JNIEnv* env, jclass, jstring javaString, jstring localeName) {
678dbbdffce5ac97a0e93ef495adaacca3660b9ab21Elliott Hughes    ScopedJavaUnicodeString scopedString(env, javaString);
679dbbdffce5ac97a0e93ef495adaacca3660b9ab21Elliott Hughes    UnicodeString& s(scopedString.unicodeString());
680dbbdffce5ac97a0e93ef495adaacca3660b9ab21Elliott Hughes    UnicodeString original(s);
681dbbdffce5ac97a0e93ef495adaacca3660b9ab21Elliott Hughes    s.toUpper(Locale::createFromName(ScopedUtfChars(env, localeName).c_str()));
682dbbdffce5ac97a0e93ef495adaacca3660b9ab21Elliott Hughes    return s == original ? javaString : env->NewString(s.getBuffer(), s.length());
683dbbdffce5ac97a0e93ef495adaacca3660b9ab21Elliott Hughes}
684dbbdffce5ac97a0e93ef495adaacca3660b9ab21Elliott Hughes
685e0e567287e4392bebc5f5826b8ef3b1bd8ca166eElliott Hughesstatic jstring versionString(JNIEnv* env, const UVersionInfo& version) {
686e0e567287e4392bebc5f5826b8ef3b1bd8ca166eElliott Hughes    char versionString[U_MAX_VERSION_STRING_LENGTH];
687e0e567287e4392bebc5f5826b8ef3b1bd8ca166eElliott Hughes    u_versionToString(const_cast<UVersionInfo&>(version), &versionString[0]);
688e0e567287e4392bebc5f5826b8ef3b1bd8ca166eElliott Hughes    return env->NewStringUTF(versionString);
689e0e567287e4392bebc5f5826b8ef3b1bd8ca166eElliott Hughes}
690e0e567287e4392bebc5f5826b8ef3b1bd8ca166eElliott Hughes
691e0e567287e4392bebc5f5826b8ef3b1bd8ca166eElliott Hughesstatic jstring ICU_getIcuVersion(JNIEnv* env, jclass) {
692e0e567287e4392bebc5f5826b8ef3b1bd8ca166eElliott Hughes    UVersionInfo icuVersion;
693e0e567287e4392bebc5f5826b8ef3b1bd8ca166eElliott Hughes    u_getVersion(icuVersion);
694e0e567287e4392bebc5f5826b8ef3b1bd8ca166eElliott Hughes    return versionString(env, icuVersion);
695e0e567287e4392bebc5f5826b8ef3b1bd8ca166eElliott Hughes}
696e0e567287e4392bebc5f5826b8ef3b1bd8ca166eElliott Hughes
697e0e567287e4392bebc5f5826b8ef3b1bd8ca166eElliott Hughesstatic jstring ICU_getUnicodeVersion(JNIEnv* env, jclass) {
698e0e567287e4392bebc5f5826b8ef3b1bd8ca166eElliott Hughes    UVersionInfo unicodeVersion;
699e0e567287e4392bebc5f5826b8ef3b1bd8ca166eElliott Hughes    u_getUnicodeVersion(unicodeVersion);
700e0e567287e4392bebc5f5826b8ef3b1bd8ca166eElliott Hughes    return versionString(env, unicodeVersion);
701e0e567287e4392bebc5f5826b8ef3b1bd8ca166eElliott Hughes}
702e0e567287e4392bebc5f5826b8ef3b1bd8ca166eElliott Hughes
70352b310afcffe8b6aed6fa0c1e9e8b0ade6f0a672Elliott Hughes
70452b310afcffe8b6aed6fa0c1e9e8b0ade6f0a672Elliott Hughesstruct EnumerationCounter {
70552b310afcffe8b6aed6fa0c1e9e8b0ade6f0a672Elliott Hughes    const size_t count;
70652b310afcffe8b6aed6fa0c1e9e8b0ade6f0a672Elliott Hughes    EnumerationCounter(size_t count) : count(count) {}
70752b310afcffe8b6aed6fa0c1e9e8b0ade6f0a672Elliott Hughes    size_t operator()() { return count; }
70852b310afcffe8b6aed6fa0c1e9e8b0ade6f0a672Elliott Hughes};
70952b310afcffe8b6aed6fa0c1e9e8b0ade6f0a672Elliott Hughesstruct EnumerationGetter {
71052b310afcffe8b6aed6fa0c1e9e8b0ade6f0a672Elliott Hughes    UEnumeration* e;
71152b310afcffe8b6aed6fa0c1e9e8b0ade6f0a672Elliott Hughes    UErrorCode* status;
71252b310afcffe8b6aed6fa0c1e9e8b0ade6f0a672Elliott Hughes    EnumerationGetter(UEnumeration* e, UErrorCode* status) : e(e), status(status) {}
71352b310afcffe8b6aed6fa0c1e9e8b0ade6f0a672Elliott Hughes    const UChar* operator()(int32_t* charCount) { return uenum_unext(e, charCount, status); }
71452b310afcffe8b6aed6fa0c1e9e8b0ade6f0a672Elliott Hughes};
71552b310afcffe8b6aed6fa0c1e9e8b0ade6f0a672Elliott Hughesstatic jobject ICU_getAvailableCurrencyCodes(JNIEnv* env, jclass) {
71652b310afcffe8b6aed6fa0c1e9e8b0ade6f0a672Elliott Hughes    UErrorCode status = U_ZERO_ERROR;
71752b310afcffe8b6aed6fa0c1e9e8b0ade6f0a672Elliott Hughes    UEnumeration* e(ucurr_openISOCurrencies(UCURR_COMMON|UCURR_NON_DEPRECATED, &status));
71852b310afcffe8b6aed6fa0c1e9e8b0ade6f0a672Elliott Hughes    EnumerationCounter counter(uenum_count(e, &status));
71952b310afcffe8b6aed6fa0c1e9e8b0ade6f0a672Elliott Hughes    EnumerationGetter getter(e, &status);
72052b310afcffe8b6aed6fa0c1e9e8b0ade6f0a672Elliott Hughes    jobject result = toStringArray16(env, &counter, &getter);
721bef9ec33e1368f57c731fce63b6a8c61628c64b0Elliott Hughes    maybeThrowIcuException(env, status);
72252b310afcffe8b6aed6fa0c1e9e8b0ade6f0a672Elliott Hughes    uenum_close(e);
72352b310afcffe8b6aed6fa0c1e9e8b0ade6f0a672Elliott Hughes    return result;
72452b310afcffe8b6aed6fa0c1e9e8b0ade6f0a672Elliott Hughes}
72552b310afcffe8b6aed6fa0c1e9e8b0ade6f0a672Elliott Hughes
726adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Projectstatic JNINativeMethod gMethods[] = {
7270452e9adf6fb133b6d3e59ebae2135aa90ad0d5eFabrice Di Meglio    NATIVE_METHOD(ICU, addLikelySubtags, "(Ljava/lang/String;)Ljava/lang/String;"),
728e22935d3c7040c22b48d53bd18878844f381287cElliott Hughes    NATIVE_METHOD(ICU, getAvailableBreakIteratorLocalesNative, "()[Ljava/lang/String;"),
729e22935d3c7040c22b48d53bd18878844f381287cElliott Hughes    NATIVE_METHOD(ICU, getAvailableCalendarLocalesNative, "()[Ljava/lang/String;"),
730e22935d3c7040c22b48d53bd18878844f381287cElliott Hughes    NATIVE_METHOD(ICU, getAvailableCollatorLocalesNative, "()[Ljava/lang/String;"),
73152b310afcffe8b6aed6fa0c1e9e8b0ade6f0a672Elliott Hughes    NATIVE_METHOD(ICU, getAvailableCurrencyCodes, "()[Ljava/lang/String;"),
732e22935d3c7040c22b48d53bd18878844f381287cElliott Hughes    NATIVE_METHOD(ICU, getAvailableDateFormatLocalesNative, "()[Ljava/lang/String;"),
733e22935d3c7040c22b48d53bd18878844f381287cElliott Hughes    NATIVE_METHOD(ICU, getAvailableLocalesNative, "()[Ljava/lang/String;"),
734e22935d3c7040c22b48d53bd18878844f381287cElliott Hughes    NATIVE_METHOD(ICU, getAvailableNumberFormatLocalesNative, "()[Ljava/lang/String;"),
73552b310afcffe8b6aed6fa0c1e9e8b0ade6f0a672Elliott Hughes    NATIVE_METHOD(ICU, getCurrencyCode, "(Ljava/lang/String;)Ljava/lang/String;"),
73652b310afcffe8b6aed6fa0c1e9e8b0ade6f0a672Elliott Hughes    NATIVE_METHOD(ICU, getCurrencyDisplayName, "(Ljava/lang/String;Ljava/lang/String;)Ljava/lang/String;"),
73752b310afcffe8b6aed6fa0c1e9e8b0ade6f0a672Elliott Hughes    NATIVE_METHOD(ICU, getCurrencyFractionDigits, "(Ljava/lang/String;)I"),
73852b310afcffe8b6aed6fa0c1e9e8b0ade6f0a672Elliott Hughes    NATIVE_METHOD(ICU, getCurrencySymbol, "(Ljava/lang/String;Ljava/lang/String;)Ljava/lang/String;"),
739e22935d3c7040c22b48d53bd18878844f381287cElliott Hughes    NATIVE_METHOD(ICU, getDisplayCountryNative, "(Ljava/lang/String;Ljava/lang/String;)Ljava/lang/String;"),
740e22935d3c7040c22b48d53bd18878844f381287cElliott Hughes    NATIVE_METHOD(ICU, getDisplayLanguageNative, "(Ljava/lang/String;Ljava/lang/String;)Ljava/lang/String;"),
741e22935d3c7040c22b48d53bd18878844f381287cElliott Hughes    NATIVE_METHOD(ICU, getDisplayVariantNative, "(Ljava/lang/String;Ljava/lang/String;)Ljava/lang/String;"),
742e22935d3c7040c22b48d53bd18878844f381287cElliott Hughes    NATIVE_METHOD(ICU, getISO3CountryNative, "(Ljava/lang/String;)Ljava/lang/String;"),
743e22935d3c7040c22b48d53bd18878844f381287cElliott Hughes    NATIVE_METHOD(ICU, getISO3LanguageNative, "(Ljava/lang/String;)Ljava/lang/String;"),
744e22935d3c7040c22b48d53bd18878844f381287cElliott Hughes    NATIVE_METHOD(ICU, getISOCountriesNative, "()[Ljava/lang/String;"),
745e22935d3c7040c22b48d53bd18878844f381287cElliott Hughes    NATIVE_METHOD(ICU, getISOLanguagesNative, "()[Ljava/lang/String;"),
746e0e567287e4392bebc5f5826b8ef3b1bd8ca166eElliott Hughes    NATIVE_METHOD(ICU, getIcuVersion, "()Ljava/lang/String;"),
7474c94a6128d1fcfaa75539b36c9f50d0c75df6144Fabrice Di Meglio    NATIVE_METHOD(ICU, getScript, "(Ljava/lang/String;)Ljava/lang/String;"),
748e0e567287e4392bebc5f5826b8ef3b1bd8ca166eElliott Hughes    NATIVE_METHOD(ICU, getUnicodeVersion, "()Ljava/lang/String;"),
749c27a366a89e470690e99374b15270e7b9169ade1Elliott Hughes    NATIVE_METHOD(ICU, initLocaleDataImpl, "(Ljava/lang/String;Llibcore/icu/LocaleData;)Z"),
750e22935d3c7040c22b48d53bd18878844f381287cElliott Hughes    NATIVE_METHOD(ICU, toLowerCase, "(Ljava/lang/String;Ljava/lang/String;)Ljava/lang/String;"),
751e22935d3c7040c22b48d53bd18878844f381287cElliott Hughes    NATIVE_METHOD(ICU, toUpperCase, "(Ljava/lang/String;Ljava/lang/String;)Ljava/lang/String;"),
752adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project};
753c27a366a89e470690e99374b15270e7b9169ade1Elliott Hughesint register_libcore_icu_ICU(JNIEnv* env) {
754566bbb1d5a0ffadfc07fa58176587d4bd0d0be13Elliott Hughes    std::string path;
755566bbb1d5a0ffadfc07fa58176587d4bd0d0be13Elliott Hughes    path = u_getDataDirectory();
756566bbb1d5a0ffadfc07fa58176587d4bd0d0be13Elliott Hughes    path += "/";
757566bbb1d5a0ffadfc07fa58176587d4bd0d0be13Elliott Hughes    path += U_ICUDATA_NAME;
758566bbb1d5a0ffadfc07fa58176587d4bd0d0be13Elliott Hughes    path += ".dat";
7599f557fae5a751ba8de8c0bc9ba689ab23ad405f6Elliott Hughes
7609f557fae5a751ba8de8c0bc9ba689ab23ad405f6Elliott Hughes    #define FAIL_WITH_STRERROR(s) \
761679cf68b607e9b4a3beb8bcdee06868ae583386fSteve Block        ALOGE("Couldn't " s " '%s': %s", path.c_str(), strerror(errno)); \
7629f557fae5a751ba8de8c0bc9ba689ab23ad405f6Elliott Hughes        return -1;
7639f557fae5a751ba8de8c0bc9ba689ab23ad405f6Elliott Hughes    #define MAYBE_FAIL_WITH_ICU_ERROR(s) \
7649f557fae5a751ba8de8c0bc9ba689ab23ad405f6Elliott Hughes        if (status != U_ZERO_ERROR) {\
765679cf68b607e9b4a3beb8bcdee06868ae583386fSteve Block            ALOGE("Couldn't initialize ICU (" s "): %s (%s)", u_errorName(status), path.c_str()); \
7669f557fae5a751ba8de8c0bc9ba689ab23ad405f6Elliott Hughes            return -1; \
7679f557fae5a751ba8de8c0bc9ba689ab23ad405f6Elliott Hughes        }
768566bbb1d5a0ffadfc07fa58176587d4bd0d0be13Elliott Hughes
769566bbb1d5a0ffadfc07fa58176587d4bd0d0be13Elliott Hughes    // Open the file and get its length.
770566bbb1d5a0ffadfc07fa58176587d4bd0d0be13Elliott Hughes    ScopedFd fd(open(path.c_str(), O_RDONLY));
771566bbb1d5a0ffadfc07fa58176587d4bd0d0be13Elliott Hughes    if (fd.get() == -1) {
7729f557fae5a751ba8de8c0bc9ba689ab23ad405f6Elliott Hughes        FAIL_WITH_STRERROR("open");
773566bbb1d5a0ffadfc07fa58176587d4bd0d0be13Elliott Hughes    }
774566bbb1d5a0ffadfc07fa58176587d4bd0d0be13Elliott Hughes    struct stat sb;
775566bbb1d5a0ffadfc07fa58176587d4bd0d0be13Elliott Hughes    if (fstat(fd.get(), &sb) == -1) {
7769f557fae5a751ba8de8c0bc9ba689ab23ad405f6Elliott Hughes        FAIL_WITH_STRERROR("stat");
777566bbb1d5a0ffadfc07fa58176587d4bd0d0be13Elliott Hughes    }
778566bbb1d5a0ffadfc07fa58176587d4bd0d0be13Elliott Hughes
779566bbb1d5a0ffadfc07fa58176587d4bd0d0be13Elliott Hughes    // Map it.
780566bbb1d5a0ffadfc07fa58176587d4bd0d0be13Elliott Hughes    void* data = mmap(NULL, sb.st_size, PROT_READ, MAP_SHARED, fd.get(), 0);
781566bbb1d5a0ffadfc07fa58176587d4bd0d0be13Elliott Hughes    if (data == MAP_FAILED) {
7829f557fae5a751ba8de8c0bc9ba689ab23ad405f6Elliott Hughes        FAIL_WITH_STRERROR("mmap");
783566bbb1d5a0ffadfc07fa58176587d4bd0d0be13Elliott Hughes    }
784566bbb1d5a0ffadfc07fa58176587d4bd0d0be13Elliott Hughes
785566bbb1d5a0ffadfc07fa58176587d4bd0d0be13Elliott Hughes    // Tell the kernel that accesses are likely to be random rather than sequential.
786566bbb1d5a0ffadfc07fa58176587d4bd0d0be13Elliott Hughes    if (madvise(data, sb.st_size, MADV_RANDOM) == -1) {
7879f557fae5a751ba8de8c0bc9ba689ab23ad405f6Elliott Hughes        FAIL_WITH_STRERROR("madvise(MADV_RANDOM)");
788566bbb1d5a0ffadfc07fa58176587d4bd0d0be13Elliott Hughes    }
789566bbb1d5a0ffadfc07fa58176587d4bd0d0be13Elliott Hughes
790566bbb1d5a0ffadfc07fa58176587d4bd0d0be13Elliott Hughes    // Tell ICU to use our memory-mapped data.
791566bbb1d5a0ffadfc07fa58176587d4bd0d0be13Elliott Hughes    UErrorCode status = U_ZERO_ERROR;
792566bbb1d5a0ffadfc07fa58176587d4bd0d0be13Elliott Hughes    udata_setCommonData(data, &status);
7939f557fae5a751ba8de8c0bc9ba689ab23ad405f6Elliott Hughes    MAYBE_FAIL_WITH_ICU_ERROR("udata_setCommonData");
794566bbb1d5a0ffadfc07fa58176587d4bd0d0be13Elliott Hughes    // Tell ICU it can *only* use our memory-mapped data.
795566bbb1d5a0ffadfc07fa58176587d4bd0d0be13Elliott Hughes    udata_setFileAccess(UDATA_NO_FILES, &status);
7969f557fae5a751ba8de8c0bc9ba689ab23ad405f6Elliott Hughes    MAYBE_FAIL_WITH_ICU_ERROR("udata_setFileAccess");
797566bbb1d5a0ffadfc07fa58176587d4bd0d0be13Elliott Hughes
798972d9be2dbbf6d467728b8ad44625f6574068bf8Elliott Hughes    // Failures to find the ICU data tend to be somewhat obscure because ICU loads its data on first
799972d9be2dbbf6d467728b8ad44625f6574068bf8Elliott Hughes    // use, which can be anywhere. Force initialization up front so we can report a nice clear error
800972d9be2dbbf6d467728b8ad44625f6574068bf8Elliott Hughes    // and bail.
801972d9be2dbbf6d467728b8ad44625f6574068bf8Elliott Hughes    u_init(&status);
8029f557fae5a751ba8de8c0bc9ba689ab23ad405f6Elliott Hughes    MAYBE_FAIL_WITH_ICU_ERROR("u_init");
803c27a366a89e470690e99374b15270e7b9169ade1Elliott Hughes    return jniRegisterNativeMethods(env, "libcore/icu/ICU", gMethods, NELEM(gMethods));
804adc854b798c1cfe3bfd4c27d68d5cee38ca617daThe Android Open Source Project}
805