ce7e060d50
This makes fixes in order to run the icu4c tests (intltest, cintltst, iotest, and icuinfo) cleanly under valgrind with --leak-check=full.
368 lines
11 KiB
C++
368 lines
11 KiB
C++
// © 2016 and later: Unicode, Inc. and others.
|
|
// License & terms of use: http://www.unicode.org/copyright.html
|
|
/*
|
|
*******************************************************************************
|
|
* Copyright (C) 2010-2015, International Business Machines Corporation and
|
|
* others. All Rights Reserved.
|
|
*******************************************************************************
|
|
*
|
|
*
|
|
* File NUMSYS.CPP
|
|
*
|
|
* Modification History:*
|
|
* Date Name Description
|
|
*
|
|
********************************************************************************
|
|
*/
|
|
|
|
#include "unicode/utypes.h"
|
|
#include "unicode/localpointer.h"
|
|
#include "unicode/uchar.h"
|
|
#include "unicode/unistr.h"
|
|
#include "unicode/ures.h"
|
|
#include "unicode/ustring.h"
|
|
#include "unicode/uloc.h"
|
|
#include "unicode/schriter.h"
|
|
#include "unicode/numsys.h"
|
|
#include "cstring.h"
|
|
#include "uassert.h"
|
|
#include "ucln_in.h"
|
|
#include "umutex.h"
|
|
#include "uresimp.h"
|
|
#include "numsys_impl.h"
|
|
|
|
#if !UCONFIG_NO_FORMATTING
|
|
|
|
U_NAMESPACE_BEGIN
|
|
|
|
// Useful constants
|
|
|
|
#define DEFAULT_DIGITS UNICODE_STRING_SIMPLE("0123456789")
|
|
static const char gNumberingSystems[] = "numberingSystems";
|
|
static const char gNumberElements[] = "NumberElements";
|
|
static const char gDefault[] = "default";
|
|
static const char gNative[] = "native";
|
|
static const char gTraditional[] = "traditional";
|
|
static const char gFinance[] = "finance";
|
|
static const char gDesc[] = "desc";
|
|
static const char gRadix[] = "radix";
|
|
static const char gAlgorithmic[] = "algorithmic";
|
|
static const char gLatn[] = "latn";
|
|
|
|
|
|
UOBJECT_DEFINE_RTTI_IMPLEMENTATION(NumberingSystem)
|
|
UOBJECT_DEFINE_RTTI_IMPLEMENTATION(NumsysNameEnumeration)
|
|
|
|
/**
|
|
* Default Constructor.
|
|
*
|
|
* @draft ICU 4.2
|
|
*/
|
|
|
|
NumberingSystem::NumberingSystem() {
|
|
radix = 10;
|
|
algorithmic = FALSE;
|
|
UnicodeString defaultDigits = DEFAULT_DIGITS;
|
|
desc.setTo(defaultDigits);
|
|
uprv_strcpy(name,gLatn);
|
|
}
|
|
|
|
/**
|
|
* Copy constructor.
|
|
* @draft ICU 4.2
|
|
*/
|
|
|
|
NumberingSystem::NumberingSystem(const NumberingSystem& other)
|
|
: UObject(other) {
|
|
*this=other;
|
|
}
|
|
|
|
NumberingSystem* U_EXPORT2
|
|
NumberingSystem::createInstance(int32_t radix_in, UBool isAlgorithmic_in, const UnicodeString & desc_in, UErrorCode &status) {
|
|
|
|
if (U_FAILURE(status)) {
|
|
return nullptr;
|
|
}
|
|
|
|
if ( radix_in < 2 ) {
|
|
status = U_ILLEGAL_ARGUMENT_ERROR;
|
|
return nullptr;
|
|
}
|
|
|
|
if ( !isAlgorithmic_in ) {
|
|
if ( desc_in.countChar32() != radix_in ) {
|
|
status = U_ILLEGAL_ARGUMENT_ERROR;
|
|
return nullptr;
|
|
}
|
|
}
|
|
|
|
LocalPointer<NumberingSystem> ns(new NumberingSystem(), status);
|
|
if (U_FAILURE(status)) {
|
|
return nullptr;
|
|
}
|
|
|
|
ns->setRadix(radix_in);
|
|
ns->setDesc(desc_in);
|
|
ns->setAlgorithmic(isAlgorithmic_in);
|
|
ns->setName(nullptr);
|
|
|
|
return ns.orphan();
|
|
}
|
|
|
|
NumberingSystem* U_EXPORT2
|
|
NumberingSystem::createInstance(const Locale & inLocale, UErrorCode& status) {
|
|
|
|
if (U_FAILURE(status)) {
|
|
return nullptr;
|
|
}
|
|
|
|
UBool nsResolved = TRUE;
|
|
UBool usingFallback = FALSE;
|
|
char buffer[ULOC_KEYWORDS_CAPACITY] = "";
|
|
int32_t count = inLocale.getKeywordValue("numbers", buffer, sizeof(buffer), status);
|
|
if (U_FAILURE(status) || status == U_STRING_NOT_TERMINATED_WARNING) {
|
|
// the "numbers" keyword exceeds ULOC_KEYWORDS_CAPACITY; ignore and use default.
|
|
count = 0;
|
|
status = U_ZERO_ERROR;
|
|
}
|
|
if ( count > 0 ) { // @numbers keyword was specified in the locale
|
|
U_ASSERT(count < ULOC_KEYWORDS_CAPACITY);
|
|
buffer[count] = '\0'; // Make sure it is null terminated.
|
|
if ( !uprv_strcmp(buffer,gDefault) || !uprv_strcmp(buffer,gNative) ||
|
|
!uprv_strcmp(buffer,gTraditional) || !uprv_strcmp(buffer,gFinance)) {
|
|
nsResolved = FALSE;
|
|
}
|
|
} else {
|
|
uprv_strcpy(buffer, gDefault);
|
|
nsResolved = FALSE;
|
|
}
|
|
|
|
if (!nsResolved) { // Resolve the numbering system ( default, native, traditional or finance ) into a "real" numbering system
|
|
UErrorCode localStatus = U_ZERO_ERROR;
|
|
LocalUResourceBundlePointer resource(ures_open(nullptr, inLocale.getName(), &localStatus));
|
|
LocalUResourceBundlePointer numberElementsRes(ures_getByKey(resource.getAlias(), gNumberElements, nullptr, &localStatus));
|
|
// Don't stomp on the catastrophic failure of OOM.
|
|
if (localStatus == U_MEMORY_ALLOCATION_ERROR) {
|
|
status = U_MEMORY_ALLOCATION_ERROR;
|
|
return nullptr;
|
|
}
|
|
while (!nsResolved) {
|
|
localStatus = U_ZERO_ERROR;
|
|
count = 0;
|
|
const UChar *nsName = ures_getStringByKeyWithFallback(numberElementsRes.getAlias(), buffer, &count, &localStatus);
|
|
// Don't stomp on the catastrophic failure of OOM.
|
|
if (localStatus == U_MEMORY_ALLOCATION_ERROR) {
|
|
status = U_MEMORY_ALLOCATION_ERROR;
|
|
return nullptr;
|
|
}
|
|
if ( count > 0 && count < ULOC_KEYWORDS_CAPACITY ) { // numbering system found
|
|
u_UCharsToChars(nsName, buffer, count);
|
|
buffer[count] = '\0'; // Make sure it is null terminated.
|
|
nsResolved = TRUE;
|
|
}
|
|
|
|
if (!nsResolved) { // Fallback behavior per TR35 - traditional falls back to native, finance and native fall back to default
|
|
if (!uprv_strcmp(buffer,gNative) || !uprv_strcmp(buffer,gFinance)) {
|
|
uprv_strcpy(buffer,gDefault);
|
|
} else if (!uprv_strcmp(buffer,gTraditional)) {
|
|
uprv_strcpy(buffer,gNative);
|
|
} else { // If we get here we couldn't find even the default numbering system
|
|
usingFallback = TRUE;
|
|
nsResolved = TRUE;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if (usingFallback) {
|
|
status = U_USING_FALLBACK_WARNING;
|
|
NumberingSystem *ns = new NumberingSystem();
|
|
if (ns == nullptr) {
|
|
status = U_MEMORY_ALLOCATION_ERROR;
|
|
}
|
|
return ns;
|
|
} else {
|
|
return NumberingSystem::createInstanceByName(buffer, status);
|
|
}
|
|
}
|
|
|
|
NumberingSystem* U_EXPORT2
|
|
NumberingSystem::createInstance(UErrorCode& status) {
|
|
return NumberingSystem::createInstance(Locale::getDefault(), status);
|
|
}
|
|
|
|
NumberingSystem* U_EXPORT2
|
|
NumberingSystem::createInstanceByName(const char *name, UErrorCode& status) {
|
|
int32_t radix = 10;
|
|
int32_t algorithmic = 0;
|
|
|
|
LocalUResourceBundlePointer numberingSystemsInfo(ures_openDirect(nullptr, gNumberingSystems, &status));
|
|
LocalUResourceBundlePointer nsCurrent(ures_getByKey(numberingSystemsInfo.getAlias(), gNumberingSystems, nullptr, &status));
|
|
LocalUResourceBundlePointer nsTop(ures_getByKey(nsCurrent.getAlias(), name, nullptr, &status));
|
|
|
|
UnicodeString nsd = ures_getUnicodeStringByKey(nsTop.getAlias(), gDesc, &status);
|
|
|
|
ures_getByKey(nsTop.getAlias(), gRadix, nsCurrent.getAlias(), &status);
|
|
radix = ures_getInt(nsCurrent.getAlias(), &status);
|
|
|
|
ures_getByKey(nsTop.getAlias(), gAlgorithmic, nsCurrent.getAlias(), &status);
|
|
algorithmic = ures_getInt(nsCurrent.getAlias(), &status);
|
|
|
|
UBool isAlgorithmic = ( algorithmic == 1 );
|
|
|
|
if (U_FAILURE(status)) {
|
|
// Don't stomp on the catastrophic failure of OOM.
|
|
if (status != U_MEMORY_ALLOCATION_ERROR) {
|
|
status = U_UNSUPPORTED_ERROR;
|
|
}
|
|
return nullptr;
|
|
}
|
|
|
|
LocalPointer<NumberingSystem> ns(NumberingSystem::createInstance(radix, isAlgorithmic, nsd, status), status);
|
|
if (U_FAILURE(status)) {
|
|
return nullptr;
|
|
}
|
|
ns->setName(name);
|
|
return ns.orphan();
|
|
}
|
|
|
|
/**
|
|
* Destructor.
|
|
* @draft ICU 4.2
|
|
*/
|
|
NumberingSystem::~NumberingSystem() {
|
|
}
|
|
|
|
int32_t NumberingSystem::getRadix() const {
|
|
return radix;
|
|
}
|
|
|
|
UnicodeString NumberingSystem::getDescription() const {
|
|
return desc;
|
|
}
|
|
|
|
const char * NumberingSystem::getName() const {
|
|
return name;
|
|
}
|
|
|
|
void NumberingSystem::setRadix(int32_t r) {
|
|
radix = r;
|
|
}
|
|
|
|
void NumberingSystem::setAlgorithmic(UBool c) {
|
|
algorithmic = c;
|
|
}
|
|
|
|
void NumberingSystem::setDesc(const UnicodeString &d) {
|
|
desc.setTo(d);
|
|
}
|
|
void NumberingSystem::setName(const char *n) {
|
|
if ( n == nullptr ) {
|
|
name[0] = (char) 0;
|
|
} else {
|
|
uprv_strncpy(name,n,kInternalNumSysNameCapacity);
|
|
name[kInternalNumSysNameCapacity] = '\0'; // Make sure it is null terminated.
|
|
}
|
|
}
|
|
UBool NumberingSystem::isAlgorithmic() const {
|
|
return ( algorithmic );
|
|
}
|
|
|
|
namespace {
|
|
|
|
UVector* gNumsysNames = nullptr;
|
|
UInitOnce gNumSysInitOnce = U_INITONCE_INITIALIZER;
|
|
|
|
U_CFUNC UBool U_CALLCONV numSysCleanup() {
|
|
delete gNumsysNames;
|
|
gNumsysNames = nullptr;
|
|
gNumSysInitOnce.reset();
|
|
return true;
|
|
}
|
|
|
|
U_CFUNC void initNumsysNames(UErrorCode &status) {
|
|
U_ASSERT(gNumsysNames == nullptr);
|
|
ucln_i18n_registerCleanup(UCLN_I18N_NUMSYS, numSysCleanup);
|
|
|
|
// TODO: Simple array of UnicodeString objects, based on length of table resource?
|
|
LocalPointer<UVector> numsysNames(new UVector(uprv_deleteUObject, nullptr, status), status);
|
|
if (U_FAILURE(status)) {
|
|
return;
|
|
}
|
|
|
|
UErrorCode rbstatus = U_ZERO_ERROR;
|
|
UResourceBundle *numberingSystemsInfo = ures_openDirect(nullptr, "numberingSystems", &rbstatus);
|
|
numberingSystemsInfo =
|
|
ures_getByKey(numberingSystemsInfo, "numberingSystems", numberingSystemsInfo, &rbstatus);
|
|
if (U_FAILURE(rbstatus)) {
|
|
// Don't stomp on the catastrophic failure of OOM.
|
|
if (rbstatus == U_MEMORY_ALLOCATION_ERROR) {
|
|
status = rbstatus;
|
|
} else {
|
|
status = U_MISSING_RESOURCE_ERROR;
|
|
}
|
|
ures_close(numberingSystemsInfo);
|
|
return;
|
|
}
|
|
|
|
while ( ures_hasNext(numberingSystemsInfo) && U_SUCCESS(status) ) {
|
|
LocalUResourceBundlePointer nsCurrent(ures_getNextResource(numberingSystemsInfo, nullptr, &rbstatus));
|
|
if (rbstatus == U_MEMORY_ALLOCATION_ERROR) {
|
|
status = rbstatus; // we want to report OOM failure back to the caller.
|
|
break;
|
|
}
|
|
const char *nsName = ures_getKey(nsCurrent.getAlias());
|
|
LocalPointer<UnicodeString> newElem(new UnicodeString(nsName, -1, US_INV), status);
|
|
if (U_SUCCESS(status)) {
|
|
numsysNames->addElement(newElem.getAlias(), status);
|
|
if (U_SUCCESS(status)) {
|
|
newElem.orphan(); // on success, the numsysNames vector owns newElem.
|
|
}
|
|
}
|
|
}
|
|
|
|
ures_close(numberingSystemsInfo);
|
|
if (U_SUCCESS(status)) {
|
|
gNumsysNames = numsysNames.orphan();
|
|
}
|
|
return;
|
|
}
|
|
|
|
} // end anonymous namespace
|
|
|
|
StringEnumeration* NumberingSystem::getAvailableNames(UErrorCode &status) {
|
|
umtx_initOnce(gNumSysInitOnce, &initNumsysNames, status);
|
|
LocalPointer<StringEnumeration> result(new NumsysNameEnumeration(status), status);
|
|
return result.orphan();
|
|
}
|
|
|
|
NumsysNameEnumeration::NumsysNameEnumeration(UErrorCode& status) : pos(0) {
|
|
(void)status;
|
|
}
|
|
|
|
const UnicodeString*
|
|
NumsysNameEnumeration::snext(UErrorCode& status) {
|
|
if (U_SUCCESS(status) && (gNumsysNames != nullptr) && (pos < gNumsysNames->size())) {
|
|
return (const UnicodeString*)gNumsysNames->elementAt(pos++);
|
|
}
|
|
return nullptr;
|
|
}
|
|
|
|
void
|
|
NumsysNameEnumeration::reset(UErrorCode& /*status*/) {
|
|
pos=0;
|
|
}
|
|
|
|
int32_t
|
|
NumsysNameEnumeration::count(UErrorCode& /*status*/) const {
|
|
return (gNumsysNames==nullptr) ? 0 : gNumsysNames->size();
|
|
}
|
|
|
|
NumsysNameEnumeration::~NumsysNameEnumeration() {
|
|
}
|
|
U_NAMESPACE_END
|
|
|
|
#endif /* #if !UCONFIG_NO_FORMATTING */
|
|
|
|
//eof
|