libcore_icu_ICU.cpp revision bef9ec33e1368f57c731fce63b6a8c61628c64b0
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 "JNIHelp.h" 20#include "JniConstants.h" 21#include "JniException.h" 22#include "ScopedFd.h" 23#include "ScopedJavaUnicodeString.h" 24#include "ScopedLocalRef.h" 25#include "ScopedStringChars.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/gregocal.h" 36#include "unicode/locid.h" 37#include "unicode/numfmt.h" 38#include "unicode/strenum.h" 39#include "unicode/ubrk.h" 40#include "unicode/ucal.h" 41#include "unicode/uclean.h" 42#include "unicode/ucol.h" 43#include "unicode/ucurr.h" 44#include "unicode/udat.h" 45#include "unicode/ustring.h" 46#include "ureslocs.h" 47#include "valueOf.h" 48 49#include <errno.h> 50#include <fcntl.h> 51#include <stdlib.h> 52#include <string.h> 53#include <string> 54#include <sys/mman.h> 55#include <sys/stat.h> 56#include <sys/time.h> 57#include <sys/types.h> 58#include <time.h> 59#include <unistd.h> 60 61class ScopedResourceBundle { 62public: 63 ScopedResourceBundle(UResourceBundle* bundle) : mBundle(bundle) { 64 } 65 66 ~ScopedResourceBundle() { 67 if (mBundle != NULL) { 68 ures_close(mBundle); 69 } 70 } 71 72 UResourceBundle* get() { 73 return mBundle; 74 } 75 76private: 77 UResourceBundle* mBundle; 78 79 // Disallow copy and assignment. 80 ScopedResourceBundle(const ScopedResourceBundle&); 81 void operator=(const ScopedResourceBundle&); 82}; 83 84Locale getLocale(JNIEnv* env, jstring localeName) { 85 return Locale::createFromName(ScopedUtfChars(env, localeName).c_str()); 86} 87 88static jint ICU_getCurrencyFractionDigits(JNIEnv* env, jclass, jstring javaCurrencyCode) { 89 ScopedJavaUnicodeString currencyCode(env, javaCurrencyCode); 90 UnicodeString icuCurrencyCode(currencyCode.unicodeString()); 91 UErrorCode status = U_ZERO_ERROR; 92 return ucurr_getDefaultFractionDigits(icuCurrencyCode.getTerminatedBuffer(), &status); 93} 94 95static jstring ICU_getCurrencyCode(JNIEnv* env, jclass, jstring javaCountryCode) { 96 UErrorCode status = U_ZERO_ERROR; 97 ScopedResourceBundle supplData(ures_openDirect(U_ICUDATA_CURR, "supplementalData", &status)); 98 if (U_FAILURE(status)) { 99 return NULL; 100 } 101 102 ScopedResourceBundle currencyMap(ures_getByKey(supplData.get(), "CurrencyMap", NULL, &status)); 103 if (U_FAILURE(status)) { 104 return NULL; 105 } 106 107 ScopedUtfChars countryCode(env, javaCountryCode); 108 ScopedResourceBundle currency(ures_getByKey(currencyMap.get(), countryCode.c_str(), NULL, &status)); 109 if (U_FAILURE(status)) { 110 return NULL; 111 } 112 113 ScopedResourceBundle currencyElem(ures_getByIndex(currency.get(), 0, NULL, &status)); 114 if (U_FAILURE(status)) { 115 return env->NewStringUTF("None"); 116 } 117 118 // Check if there's a 'to' date. If there is, the currency isn't used anymore. 119 ScopedResourceBundle currencyTo(ures_getByKey(currencyElem.get(), "to", NULL, &status)); 120 if (!U_FAILURE(status)) { 121 return NULL; 122 } 123 // Ignore the failure to find a 'to' date. 124 status = U_ZERO_ERROR; 125 126 ScopedResourceBundle currencyId(ures_getByKey(currencyElem.get(), "id", NULL, &status)); 127 if (U_FAILURE(status)) { 128 // No id defined for this country 129 return env->NewStringUTF("None"); 130 } 131 132 int32_t charCount; 133 const jchar* chars = ures_getString(currencyId.get(), &charCount, &status); 134 return (charCount == 0) ? env->NewStringUTF("None") : env->NewString(chars, charCount); 135} 136 137static jstring ICU_getCurrencyDisplayName(JNIEnv* env, jclass, jstring javaLocaleName, jstring javaCurrencyCode) { 138 ScopedUtfChars localeName(env, javaLocaleName); 139 ScopedJavaUnicodeString currencyCode(env, javaCurrencyCode); 140 UnicodeString icuCurrencyCode(currencyCode.unicodeString()); 141 UErrorCode status = U_ZERO_ERROR; 142 UBool isChoiceFormat; 143 int32_t charCount; 144 const UChar* chars = ucurr_getName(icuCurrencyCode.getTerminatedBuffer(), localeName.c_str(), 145 UCURR_LONG_NAME, &isChoiceFormat, &charCount, &status); 146 if (status == U_USING_DEFAULT_WARNING) { 147 // ICU's default is English. We want the ISO 4217 currency code instead. 148 chars = icuCurrencyCode.getBuffer(); 149 charCount = icuCurrencyCode.length(); 150 } 151 return (charCount == 0) ? NULL : env->NewString(chars, charCount); 152} 153 154static jstring ICU_getCurrencySymbol(JNIEnv* env, jclass, jstring locale, jstring currencyCode) { 155 // We can't use ucurr_getName because it doesn't distinguish between using data root from 156 // the root locale and parroting back the input because it's never heard of the currency code. 157 ScopedUtfChars localeName(env, locale); 158 UErrorCode status = U_ZERO_ERROR; 159 ScopedResourceBundle currLoc(ures_open(U_ICUDATA_CURR, localeName.c_str(), &status)); 160 if (U_FAILURE(status)) { 161 return NULL; 162 } 163 164 ScopedResourceBundle currencies(ures_getByKey(currLoc.get(), "Currencies", NULL, &status)); 165 if (U_FAILURE(status)) { 166 return NULL; 167 } 168 169 ScopedUtfChars currency(env, currencyCode); 170 ScopedResourceBundle currencyElems(ures_getByKey(currencies.get(), currency.c_str(), NULL, &status)); 171 if (U_FAILURE(status)) { 172 return NULL; 173 } 174 175 int32_t charCount; 176 const jchar* chars = ures_getStringByIndex(currencyElems.get(), 0, &charCount, &status); 177 if (U_FAILURE(status)) { 178 return NULL; 179 } 180 return (charCount == 0) ? NULL : env->NewString(chars, charCount); 181} 182 183static jstring ICU_getDisplayCountryNative(JNIEnv* env, jclass, jstring targetLocale, jstring locale) { 184 Locale loc = getLocale(env, locale); 185 Locale targetLoc = getLocale(env, targetLocale); 186 UnicodeString str; 187 targetLoc.getDisplayCountry(loc, str); 188 return env->NewString(str.getBuffer(), str.length()); 189} 190 191static jstring ICU_getDisplayLanguageNative(JNIEnv* env, jclass, jstring targetLocale, jstring locale) { 192 Locale loc = getLocale(env, locale); 193 Locale targetLoc = getLocale(env, targetLocale); 194 UnicodeString str; 195 targetLoc.getDisplayLanguage(loc, str); 196 return env->NewString(str.getBuffer(), str.length()); 197} 198 199static jstring ICU_getDisplayVariantNative(JNIEnv* env, jclass, jstring targetLocale, jstring locale) { 200 Locale loc = getLocale(env, locale); 201 Locale targetLoc = getLocale(env, targetLocale); 202 UnicodeString str; 203 targetLoc.getDisplayVariant(loc, str); 204 return env->NewString(str.getBuffer(), str.length()); 205} 206 207static jstring ICU_getISO3CountryNative(JNIEnv* env, jclass, jstring locale) { 208 Locale loc = getLocale(env, locale); 209 return env->NewStringUTF(loc.getISO3Country()); 210} 211 212static jstring ICU_getISO3LanguageNative(JNIEnv* env, jclass, jstring locale) { 213 Locale loc = getLocale(env, locale); 214 return env->NewStringUTF(loc.getISO3Language()); 215} 216 217static jobjectArray ICU_getISOCountriesNative(JNIEnv* env, jclass) { 218 return toStringArray(env, Locale::getISOCountries()); 219} 220 221static jobjectArray ICU_getISOLanguagesNative(JNIEnv* env, jclass) { 222 return toStringArray(env, Locale::getISOLanguages()); 223} 224 225static jobjectArray ICU_getAvailableLocalesNative(JNIEnv* env, jclass) { 226 return toStringArray(env, uloc_countAvailable, uloc_getAvailable); 227} 228 229static jobjectArray ICU_getAvailableBreakIteratorLocalesNative(JNIEnv* env, jclass) { 230 return toStringArray(env, ubrk_countAvailable, ubrk_getAvailable); 231} 232 233static jobjectArray ICU_getAvailableCalendarLocalesNative(JNIEnv* env, jclass) { 234 return toStringArray(env, ucal_countAvailable, ucal_getAvailable); 235} 236 237static jobjectArray ICU_getAvailableCollatorLocalesNative(JNIEnv* env, jclass) { 238 return toStringArray(env, ucol_countAvailable, ucol_getAvailable); 239} 240 241static jobjectArray ICU_getAvailableDateFormatLocalesNative(JNIEnv* env, jclass) { 242 return toStringArray(env, udat_countAvailable, udat_getAvailable); 243} 244 245static jobjectArray ICU_getAvailableNumberFormatLocalesNative(JNIEnv* env, jclass) { 246 return toStringArray(env, unum_countAvailable, unum_getAvailable); 247} 248 249static bool getDayIntVector(JNIEnv*, UResourceBundle* gregorian, int* values) { 250 // get the First day of week and the minimal days in first week numbers 251 UErrorCode status = U_ZERO_ERROR; 252 ScopedResourceBundle gregorianElems(ures_getByKey(gregorian, "DateTimeElements", NULL, &status)); 253 if (U_FAILURE(status)) { 254 return false; 255 } 256 257 int intVectSize; 258 const int* result = ures_getIntVector(gregorianElems.get(), &intVectSize, &status); 259 if (U_FAILURE(status) || intVectSize != 2) { 260 return false; 261 } 262 263 values[0] = result[0]; 264 values[1] = result[1]; 265 return true; 266} 267 268// This allows you to leave extra space at the beginning or end of the array to support the 269// month names and day names arrays. 270static jobjectArray toStringArray(JNIEnv* env, UResourceBundle* rb, size_t size, int capacity, size_t offset) { 271 if (capacity == -1) { 272 capacity = size; 273 } 274 jobjectArray result = env->NewObjectArray(capacity, JniConstants::stringClass, NULL); 275 if (result == NULL) { 276 return NULL; 277 } 278 UErrorCode status = U_ZERO_ERROR; 279 for (size_t i = 0; i < size; ++i) { 280 int charCount; 281 const jchar* chars = ures_getStringByIndex(rb, i, &charCount, &status); 282 if (U_FAILURE(status)) { 283 return NULL; 284 } 285 ScopedLocalRef<jstring> s(env, env->NewString(chars, charCount)); 286 if (env->ExceptionCheck()) { 287 return NULL; 288 } 289 env->SetObjectArrayElement(result, offset + i, s.get()); 290 if (env->ExceptionCheck()) { 291 return NULL; 292 } 293 } 294 return result; 295} 296 297static jobjectArray getAmPmMarkers(JNIEnv* env, UResourceBundle* gregorian) { 298 UErrorCode status = U_ZERO_ERROR; 299 ScopedResourceBundle amPmMarkers(ures_getByKey(gregorian, "AmPmMarkers", NULL, &status)); 300 if (U_FAILURE(status)) { 301 return NULL; 302 } 303 return toStringArray(env, amPmMarkers.get(), ures_getSize(amPmMarkers.get()), -1, 0); 304} 305 306static jobjectArray getEras(JNIEnv* env, UResourceBundle* gregorian) { 307 UErrorCode status = U_ZERO_ERROR; 308 ScopedResourceBundle eras(ures_getByKey(gregorian, "eras", NULL, &status)); 309 if (U_FAILURE(status)) { 310 return NULL; 311 } 312 ScopedResourceBundle abbreviatedEras(ures_getByKey(eras.get(), "abbreviated", NULL, &status)); 313 if (U_FAILURE(status)) { 314 return NULL; 315 } 316 return toStringArray(env, abbreviatedEras.get(), ures_getSize(abbreviatedEras.get()), -1, 0); 317} 318 319enum NameType { REGULAR, STAND_ALONE }; 320enum NameWidth { LONG, SHORT }; 321static jobjectArray getNames(JNIEnv* env, UResourceBundle* namesBundle, bool months, NameType type, NameWidth width) { 322 const char* typeKey = (type == REGULAR) ? "format" : "stand-alone"; 323 const char* widthKey = (width == LONG) ? "wide" : "abbreviated"; 324 UErrorCode status = U_ZERO_ERROR; 325 ScopedResourceBundle formatBundle(ures_getByKey(namesBundle, typeKey, NULL, &status)); 326 ScopedResourceBundle valuesBundle(ures_getByKey(formatBundle.get(), widthKey, NULL, &status)); 327 if (U_FAILURE(status)) { 328 return NULL; 329 } 330 331 // The months array has a trailing empty string. The days array has a leading empty string. 332 int count = ures_getSize(valuesBundle.get()); 333 int offset = months ? 0 : 1; 334 jobjectArray result = toStringArray(env, valuesBundle.get(), count, count + 1, offset); 335 env->SetObjectArrayElement(result, months ? count : 0, env->NewStringUTF("")); 336 return result; 337} 338 339static void setIntegerField(JNIEnv* env, jobject obj, const char* fieldName, int value) { 340 ScopedLocalRef<jobject> integerValue(env, integerValueOf(env, value)); 341 jfieldID fid = env->GetFieldID(JniConstants::localeDataClass, fieldName, "Ljava/lang/Integer;"); 342 env->SetObjectField(obj, fid, integerValue.get()); 343} 344 345static void setStringField(JNIEnv* env, jobject obj, const char* fieldName, jstring value) { 346 jfieldID fid = env->GetFieldID(JniConstants::localeDataClass, fieldName, "Ljava/lang/String;"); 347 env->SetObjectField(obj, fid, value); 348} 349 350static void setStringArrayField(JNIEnv* env, jobject obj, const char* fieldName, jobjectArray value) { 351 jfieldID fid = env->GetFieldID(JniConstants::localeDataClass, fieldName, "[Ljava/lang/String;"); 352 env->SetObjectField(obj, fid, value); 353} 354 355static void setStringField(JNIEnv* env, jobject obj, const char* fieldName, UResourceBundle* bundle, int index) { 356 UErrorCode status = U_ZERO_ERROR; 357 int charCount; 358 const UChar* chars = ures_getStringByIndex(bundle, index, &charCount, &status); 359 if (U_SUCCESS(status)) { 360 setStringField(env, obj, fieldName, env->NewString(chars, charCount)); 361 } else { 362 LOGE("Error setting String field %s from ICU resource: %s", fieldName, u_errorName(status)); 363 } 364} 365 366static void setCharField(JNIEnv* env, jobject obj, const char* fieldName, UResourceBundle* bundle, int index) { 367 UErrorCode status = U_ZERO_ERROR; 368 int charCount; 369 const UChar* chars = ures_getStringByIndex(bundle, index, &charCount, &status); 370 if (U_SUCCESS(status)) { 371 jfieldID fid = env->GetFieldID(JniConstants::localeDataClass, fieldName, "C"); 372 env->SetCharField(obj, fid, chars[0]); 373 } else { 374 LOGE("Error setting char field %s from ICU resource: %s", fieldName, u_errorName(status)); 375 } 376} 377 378static jboolean ICU_initLocaleDataImpl(JNIEnv* env, jclass, jstring locale, jobject localeData) { 379 ScopedUtfChars localeName(env, locale); 380 UErrorCode status = U_ZERO_ERROR; 381 ScopedResourceBundle root(ures_open(NULL, localeName.c_str(), &status)); 382 if (U_FAILURE(status)) { 383 LOGE("Error getting ICU resource bundle: %s", u_errorName(status)); 384 status = U_ZERO_ERROR; 385 return JNI_FALSE; 386 } 387 388 ScopedResourceBundle calendar(ures_getByKey(root.get(), "calendar", NULL, &status)); 389 if (U_FAILURE(status)) { 390 LOGE("Error getting ICU calendar resource bundle: %s", u_errorName(status)); 391 return JNI_FALSE; 392 } 393 394 ScopedResourceBundle gregorian(ures_getByKey(calendar.get(), "gregorian", NULL, &status)); 395 if (U_FAILURE(status)) { 396 LOGE("Error getting ICU gregorian resource bundle: %s", u_errorName(status)); 397 return JNI_FALSE; 398 } 399 400 int firstDayVals[] = { 0, 0 }; 401 if (getDayIntVector(env, gregorian.get(), firstDayVals)) { 402 setIntegerField(env, localeData, "firstDayOfWeek", firstDayVals[0]); 403 setIntegerField(env, localeData, "minimalDaysInFirstWeek", firstDayVals[1]); 404 } 405 406 setStringArrayField(env, localeData, "amPm", getAmPmMarkers(env, gregorian.get())); 407 setStringArrayField(env, localeData, "eras", getEras(env, gregorian.get())); 408 409 ScopedResourceBundle dayNames(ures_getByKey(gregorian.get(), "dayNames", NULL, &status)); 410 ScopedResourceBundle monthNames(ures_getByKey(gregorian.get(), "monthNames", NULL, &status)); 411 412 // Get the regular month and weekday names. 413 jobjectArray longMonthNames = getNames(env, monthNames.get(), true, REGULAR, LONG); 414 jobjectArray shortMonthNames = getNames(env, monthNames.get(), true, REGULAR, SHORT); 415 jobjectArray longWeekdayNames = getNames(env, dayNames.get(), false, REGULAR, LONG); 416 jobjectArray shortWeekdayNames = getNames(env, dayNames.get(), false, REGULAR, SHORT); 417 setStringArrayField(env, localeData, "longMonthNames", longMonthNames); 418 setStringArrayField(env, localeData, "shortMonthNames", shortMonthNames); 419 setStringArrayField(env, localeData, "longWeekdayNames", longWeekdayNames); 420 setStringArrayField(env, localeData, "shortWeekdayNames", shortWeekdayNames); 421 422 // Get the stand-alone month and weekday names. If they're not available (as they aren't for 423 // English), we reuse the regular names. If we returned null to Java, the usual fallback 424 // mechanisms would come into play and we'd end up with the bogus stand-alone names from the 425 // root locale ("1" for January, and so on). 426 jobjectArray longStandAloneMonthNames = getNames(env, monthNames.get(), true, STAND_ALONE, LONG); 427 if (longStandAloneMonthNames == NULL) { 428 longStandAloneMonthNames = longMonthNames; 429 } 430 jobjectArray shortStandAloneMonthNames = getNames(env, monthNames.get(), true, STAND_ALONE, SHORT); 431 if (shortStandAloneMonthNames == NULL) { 432 shortStandAloneMonthNames = shortMonthNames; 433 } 434 jobjectArray longStandAloneWeekdayNames = getNames(env, dayNames.get(), false, STAND_ALONE, LONG); 435 if (longStandAloneWeekdayNames == NULL) { 436 longStandAloneWeekdayNames = longWeekdayNames; 437 } 438 jobjectArray shortStandAloneWeekdayNames = getNames(env, dayNames.get(), false, STAND_ALONE, SHORT); 439 if (shortStandAloneWeekdayNames == NULL) { 440 shortStandAloneWeekdayNames = shortWeekdayNames; 441 } 442 setStringArrayField(env, localeData, "longStandAloneMonthNames", longStandAloneMonthNames); 443 setStringArrayField(env, localeData, "shortStandAloneMonthNames", shortStandAloneMonthNames); 444 setStringArrayField(env, localeData, "longStandAloneWeekdayNames", longStandAloneWeekdayNames); 445 setStringArrayField(env, localeData, "shortStandAloneWeekdayNames", shortStandAloneWeekdayNames); 446 447 ScopedResourceBundle dateTimePatterns(ures_getByKey(gregorian.get(), "DateTimePatterns", NULL, &status)); 448 if (U_SUCCESS(status)) { 449 setStringField(env, localeData, "fullTimeFormat", dateTimePatterns.get(), 0); 450 setStringField(env, localeData, "longTimeFormat", dateTimePatterns.get(), 1); 451 setStringField(env, localeData, "mediumTimeFormat", dateTimePatterns.get(), 2); 452 setStringField(env, localeData, "shortTimeFormat", dateTimePatterns.get(), 3); 453 setStringField(env, localeData, "fullDateFormat", dateTimePatterns.get(), 4); 454 setStringField(env, localeData, "longDateFormat", dateTimePatterns.get(), 5); 455 setStringField(env, localeData, "mediumDateFormat", dateTimePatterns.get(), 6); 456 setStringField(env, localeData, "shortDateFormat", dateTimePatterns.get(), 7); 457 } 458 status = U_ZERO_ERROR; 459 460 ScopedResourceBundle numberElements(ures_getByKey(root.get(), "NumberElements", NULL, &status)); 461 if (U_SUCCESS(status) && ures_getSize(numberElements.get()) >= 11) { 462 setCharField(env, localeData, "zeroDigit", numberElements.get(), 4); 463 setCharField(env, localeData, "digit", numberElements.get(), 5); 464 setCharField(env, localeData, "decimalSeparator", numberElements.get(), 0); 465 setCharField(env, localeData, "groupingSeparator", numberElements.get(), 1); 466 setCharField(env, localeData, "patternSeparator", numberElements.get(), 2); 467 setCharField(env, localeData, "percent", numberElements.get(), 3); 468 setCharField(env, localeData, "perMill", numberElements.get(), 8); 469 setCharField(env, localeData, "monetarySeparator", numberElements.get(), 0); 470 setCharField(env, localeData, "minusSign", numberElements.get(), 6); 471 setStringField(env, localeData, "exponentSeparator", numberElements.get(), 7); 472 setStringField(env, localeData, "infinity", numberElements.get(), 9); 473 setStringField(env, localeData, "NaN", numberElements.get(), 10); 474 } 475 status = U_ZERO_ERROR; 476 477 jstring countryCode = env->NewStringUTF(Locale::createFromName(localeName.c_str()).getCountry()); 478 jstring internationalCurrencySymbol = ICU_getCurrencyCode(env, NULL, countryCode); 479 jstring currencySymbol = NULL; 480 if (internationalCurrencySymbol != NULL) { 481 currencySymbol = ICU_getCurrencySymbol(env, NULL, locale, internationalCurrencySymbol); 482 } else { 483 internationalCurrencySymbol = env->NewStringUTF("XXX"); 484 } 485 if (currencySymbol == NULL) { 486 // This is the UTF-8 encoding of U+00A4 (CURRENCY SIGN). 487 currencySymbol = env->NewStringUTF("\xc2\xa4"); 488 } 489 setStringField(env, localeData, "currencySymbol", currencySymbol); 490 setStringField(env, localeData, "internationalCurrencySymbol", internationalCurrencySymbol); 491 492 ScopedResourceBundle numberPatterns(ures_getByKey(root.get(), "NumberPatterns", NULL, &status)); 493 if (U_SUCCESS(status) && ures_getSize(numberPatterns.get()) >= 3) { 494 setStringField(env, localeData, "numberPattern", numberPatterns.get(), 0); 495 setStringField(env, localeData, "currencyPattern", numberPatterns.get(), 1); 496 setStringField(env, localeData, "percentPattern", numberPatterns.get(), 2); 497 } 498 499 return JNI_TRUE; 500} 501 502static jstring ICU_toLowerCase(JNIEnv* env, jclass, jstring javaString, jstring localeName) { 503 ScopedJavaUnicodeString scopedString(env, javaString); 504 UnicodeString& s(scopedString.unicodeString()); 505 UnicodeString original(s); 506 s.toLower(Locale::createFromName(ScopedUtfChars(env, localeName).c_str())); 507 return s == original ? javaString : env->NewString(s.getBuffer(), s.length()); 508} 509 510static jstring ICU_toUpperCase(JNIEnv* env, jclass, jstring javaString, jstring localeName) { 511 ScopedJavaUnicodeString scopedString(env, javaString); 512 UnicodeString& s(scopedString.unicodeString()); 513 UnicodeString original(s); 514 s.toUpper(Locale::createFromName(ScopedUtfChars(env, localeName).c_str())); 515 return s == original ? javaString : env->NewString(s.getBuffer(), s.length()); 516} 517 518static jstring versionString(JNIEnv* env, const UVersionInfo& version) { 519 char versionString[U_MAX_VERSION_STRING_LENGTH]; 520 u_versionToString(const_cast<UVersionInfo&>(version), &versionString[0]); 521 return env->NewStringUTF(versionString); 522} 523 524static jstring ICU_getIcuVersion(JNIEnv* env, jclass) { 525 UVersionInfo icuVersion; 526 u_getVersion(icuVersion); 527 return versionString(env, icuVersion); 528} 529 530static jstring ICU_getUnicodeVersion(JNIEnv* env, jclass) { 531 UVersionInfo unicodeVersion; 532 u_getUnicodeVersion(unicodeVersion); 533 return versionString(env, unicodeVersion); 534} 535 536 537struct EnumerationCounter { 538 const size_t count; 539 EnumerationCounter(size_t count) : count(count) {} 540 size_t operator()() { return count; } 541}; 542struct EnumerationGetter { 543 UEnumeration* e; 544 UErrorCode* status; 545 EnumerationGetter(UEnumeration* e, UErrorCode* status) : e(e), status(status) {} 546 const UChar* operator()(int32_t* charCount) { return uenum_unext(e, charCount, status); } 547}; 548static jobject ICU_getAvailableCurrencyCodes(JNIEnv* env, jclass) { 549 UErrorCode status = U_ZERO_ERROR; 550 UEnumeration* e(ucurr_openISOCurrencies(UCURR_COMMON|UCURR_NON_DEPRECATED, &status)); 551 EnumerationCounter counter(uenum_count(e, &status)); 552 EnumerationGetter getter(e, &status); 553 jobject result = toStringArray16(env, &counter, &getter); 554 maybeThrowIcuException(env, status); 555 uenum_close(e); 556 return result; 557} 558 559static JNINativeMethod gMethods[] = { 560 NATIVE_METHOD(ICU, getAvailableBreakIteratorLocalesNative, "()[Ljava/lang/String;"), 561 NATIVE_METHOD(ICU, getAvailableCalendarLocalesNative, "()[Ljava/lang/String;"), 562 NATIVE_METHOD(ICU, getAvailableCollatorLocalesNative, "()[Ljava/lang/String;"), 563 NATIVE_METHOD(ICU, getAvailableCurrencyCodes, "()[Ljava/lang/String;"), 564 NATIVE_METHOD(ICU, getAvailableDateFormatLocalesNative, "()[Ljava/lang/String;"), 565 NATIVE_METHOD(ICU, getAvailableLocalesNative, "()[Ljava/lang/String;"), 566 NATIVE_METHOD(ICU, getAvailableNumberFormatLocalesNative, "()[Ljava/lang/String;"), 567 NATIVE_METHOD(ICU, getCurrencyCode, "(Ljava/lang/String;)Ljava/lang/String;"), 568 NATIVE_METHOD(ICU, getCurrencyDisplayName, "(Ljava/lang/String;Ljava/lang/String;)Ljava/lang/String;"), 569 NATIVE_METHOD(ICU, getCurrencyFractionDigits, "(Ljava/lang/String;)I"), 570 NATIVE_METHOD(ICU, getCurrencySymbol, "(Ljava/lang/String;Ljava/lang/String;)Ljava/lang/String;"), 571 NATIVE_METHOD(ICU, getDisplayCountryNative, "(Ljava/lang/String;Ljava/lang/String;)Ljava/lang/String;"), 572 NATIVE_METHOD(ICU, getDisplayLanguageNative, "(Ljava/lang/String;Ljava/lang/String;)Ljava/lang/String;"), 573 NATIVE_METHOD(ICU, getDisplayVariantNative, "(Ljava/lang/String;Ljava/lang/String;)Ljava/lang/String;"), 574 NATIVE_METHOD(ICU, getISO3CountryNative, "(Ljava/lang/String;)Ljava/lang/String;"), 575 NATIVE_METHOD(ICU, getISO3LanguageNative, "(Ljava/lang/String;)Ljava/lang/String;"), 576 NATIVE_METHOD(ICU, getISOCountriesNative, "()[Ljava/lang/String;"), 577 NATIVE_METHOD(ICU, getISOLanguagesNative, "()[Ljava/lang/String;"), 578 NATIVE_METHOD(ICU, getIcuVersion, "()Ljava/lang/String;"), 579 NATIVE_METHOD(ICU, getUnicodeVersion, "()Ljava/lang/String;"), 580 NATIVE_METHOD(ICU, initLocaleDataImpl, "(Ljava/lang/String;Llibcore/icu/LocaleData;)Z"), 581 NATIVE_METHOD(ICU, toLowerCase, "(Ljava/lang/String;Ljava/lang/String;)Ljava/lang/String;"), 582 NATIVE_METHOD(ICU, toUpperCase, "(Ljava/lang/String;Ljava/lang/String;)Ljava/lang/String;"), 583}; 584int register_libcore_icu_ICU(JNIEnv* env) { 585 std::string path; 586 path = u_getDataDirectory(); 587 path += "/"; 588 path += U_ICUDATA_NAME; 589 path += ".dat"; 590 591 #define FAIL_WITH_STRERROR(s) \ 592 LOGE("Couldn't " s " '%s': %s", path.c_str(), strerror(errno)); \ 593 return -1; 594 #define MAYBE_FAIL_WITH_ICU_ERROR(s) \ 595 if (status != U_ZERO_ERROR) {\ 596 LOGE("Couldn't initialize ICU (" s "): %s (%s)", u_errorName(status), path.c_str()); \ 597 return -1; \ 598 } 599 600 // Open the file and get its length. 601 ScopedFd fd(open(path.c_str(), O_RDONLY)); 602 if (fd.get() == -1) { 603 FAIL_WITH_STRERROR("open"); 604 } 605 struct stat sb; 606 if (fstat(fd.get(), &sb) == -1) { 607 FAIL_WITH_STRERROR("stat"); 608 } 609 610 // Map it. 611 void* data = mmap(NULL, sb.st_size, PROT_READ, MAP_SHARED, fd.get(), 0); 612 if (data == MAP_FAILED) { 613 FAIL_WITH_STRERROR("mmap"); 614 } 615 616 // Tell the kernel that accesses are likely to be random rather than sequential. 617 if (madvise(data, sb.st_size, MADV_RANDOM) == -1) { 618 FAIL_WITH_STRERROR("madvise(MADV_RANDOM)"); 619 } 620 621 // Tell ICU to use our memory-mapped data. 622 UErrorCode status = U_ZERO_ERROR; 623 udata_setCommonData(data, &status); 624 MAYBE_FAIL_WITH_ICU_ERROR("udata_setCommonData"); 625 // Tell ICU it can *only* use our memory-mapped data. 626 udata_setFileAccess(UDATA_NO_FILES, &status); 627 MAYBE_FAIL_WITH_ICU_ERROR("udata_setFileAccess"); 628 629 // Failures to find the ICU data tend to be somewhat obscure because ICU loads its data on first 630 // use, which can be anywhere. Force initialization up front so we can report a nice clear error 631 // and bail. 632 u_init(&status); 633 MAYBE_FAIL_WITH_ICU_ERROR("u_init"); 634 return jniRegisterNativeMethods(env, "libcore/icu/ICU", gMethods, NELEM(gMethods)); 635} 636