mirror of
https://github.com/unicode-org/icu.git
synced 2025-04-06 22:15:31 +00:00
ICU-21705 ICU-21706 Fix crash if ICU's default locale has BCP47 Unicode Extensions, and fix ures_openDirect crash with NULL locale ID.
Add test case for ures_openDirect with NULL locale ID.
This commit is contained in:
parent
3534d337c6
commit
9ddda243d7
2 changed files with 26 additions and 8 deletions
|
@ -461,11 +461,10 @@ getPoolEntry(const char *path, UErrorCode *status) {
|
|||
/* INTERNAL: */
|
||||
/* CAUTION: resbMutex must be locked when calling this function! */
|
||||
static UResourceDataEntry *
|
||||
findFirstExisting(const char* path, char* name,
|
||||
findFirstExisting(const char* path, char* name, const char* defaultLocale,
|
||||
UBool *isRoot, UBool *hasChopped, UBool *isDefault, UErrorCode* status) {
|
||||
UResourceDataEntry *r = NULL;
|
||||
UBool hasRealData = FALSE;
|
||||
const char *defaultLoc = uloc_getDefault();
|
||||
*hasChopped = TRUE; /* we're starting with a fresh name */
|
||||
|
||||
while(*hasChopped && !hasRealData) {
|
||||
|
@ -474,7 +473,7 @@ findFirstExisting(const char* path, char* name,
|
|||
if (U_FAILURE(*status)) {
|
||||
return NULL;
|
||||
}
|
||||
*isDefault = (UBool)(uprv_strncmp(name, defaultLoc, uprv_strlen(name)) == 0);
|
||||
*isDefault = (UBool)(uprv_strncmp(name, defaultLocale, uprv_strlen(name)) == 0);
|
||||
hasRealData = (UBool)(r->fBogus == U_ZERO_ERROR);
|
||||
if(!hasRealData) {
|
||||
/* this entry is not real. We will discard it. */
|
||||
|
@ -669,10 +668,13 @@ static UResourceDataEntry *entryOpen(const char* path, const char* localeID,
|
|||
}
|
||||
}
|
||||
|
||||
// Note: We need to query the default locale *before* locking resbMutex.
|
||||
const char *defaultLocale = uloc_getDefault();
|
||||
|
||||
Mutex lock(&resbMutex); // Lock resbMutex until the end of this function.
|
||||
|
||||
/* We're going to skip all the locales that do not have any data */
|
||||
r = findFirstExisting(path, name, &isRoot, &hasChopped, &isDefault, &intStatus);
|
||||
r = findFirstExisting(path, name, defaultLocale, &isRoot, &hasChopped, &isDefault, &intStatus);
|
||||
|
||||
// If we failed due to out-of-memory, report the failure and exit early.
|
||||
if (intStatus == U_MEMORY_ALLOCATION_ERROR) {
|
||||
|
@ -712,8 +714,8 @@ static UResourceDataEntry *entryOpen(const char* path, const char* localeID,
|
|||
/* if that is the case, we need to chain in the default locale */
|
||||
if(r==NULL && openType == URES_OPEN_LOCALE_DEFAULT_ROOT && !isDefault && !isRoot) {
|
||||
/* insert default locale */
|
||||
uprv_strcpy(name, uloc_getDefault());
|
||||
r = findFirstExisting(path, name, &isRoot, &hasChopped, &isDefault, &intStatus);
|
||||
uprv_strcpy(name, defaultLocale);
|
||||
r = findFirstExisting(path, name, defaultLocale, &isRoot, &hasChopped, &isDefault, &intStatus);
|
||||
// If we failed due to out-of-memory, report the failure and exit early.
|
||||
if (intStatus == U_MEMORY_ALLOCATION_ERROR) {
|
||||
*status = intStatus;
|
||||
|
@ -737,7 +739,7 @@ static UResourceDataEntry *entryOpen(const char* path, const char* localeID,
|
|||
/* present */
|
||||
if(r == NULL) {
|
||||
uprv_strcpy(name, kRootLocaleName);
|
||||
r = findFirstExisting(path, name, &isRoot, &hasChopped, &isDefault, &intStatus);
|
||||
r = findFirstExisting(path, name, defaultLocale, &isRoot, &hasChopped, &isDefault, &intStatus);
|
||||
// If we failed due to out-of-memory, report the failure and exit early.
|
||||
if (intStatus == U_MEMORY_ALLOCATION_ERROR) {
|
||||
*status = intStatus;
|
||||
|
@ -791,7 +793,17 @@ entryOpenDirect(const char* path, const char* localeID, UErrorCode* status) {
|
|||
return NULL;
|
||||
}
|
||||
|
||||
// Note: We need to query the default locale *before* locking resbMutex.
|
||||
// If the localeID is NULL, then we want to use the default locale.
|
||||
if (localeID == NULL) {
|
||||
localeID = uloc_getDefault();
|
||||
} else if (*localeID == 0) {
|
||||
// If the localeID is "", then we want to use the root locale.
|
||||
localeID = kRootLocaleName;
|
||||
}
|
||||
|
||||
Mutex lock(&resbMutex);
|
||||
|
||||
// findFirstExisting() without fallbacks.
|
||||
UResourceDataEntry *r = init_entry(localeID, path, status);
|
||||
if(U_SUCCESS(*status)) {
|
||||
|
|
|
@ -496,7 +496,7 @@ static void TestFallback()
|
|||
|
||||
static void
|
||||
TestOpenDirect(void) {
|
||||
UResourceBundle *idna_rules, *casing, *te_IN, *ne, *item;
|
||||
UResourceBundle *idna_rules, *casing, *te_IN, *ne, *item, *defaultLocale;
|
||||
UErrorCode errorCode;
|
||||
|
||||
/*
|
||||
|
@ -641,6 +641,12 @@ TestOpenDirect(void) {
|
|||
ures_close(item);
|
||||
}
|
||||
ures_close(te_IN);
|
||||
|
||||
// ICU-21705
|
||||
// Verify that calling ures_openDirect() with a NULL localeID doesn't crash or assert.
|
||||
errorCode = U_ZERO_ERROR;
|
||||
defaultLocale = ures_openDirect(NULL, NULL, &errorCode);
|
||||
ures_close(defaultLocale);
|
||||
}
|
||||
|
||||
static void
|
||||
|
|
Loading…
Add table
Reference in a new issue