1 // 2016 and later: Unicode, Inc. and others. 2 // License & terms of use: http://www.unicode.org/copyright.html 3 /* 4 ******************************************************************************* 5 * 6 * Copyright (C) 1998-2014, International Business Machines 7 * Corporation and others. All Rights Reserved. 8 * 9 ******************************************************************************* 10 * 11 * File locbund.cpp 12 * 13 * Modification History: 14 * 15 * Date Name Description 16 * 11/18/98 stephen Creation. 17 * 12/10/1999 bobbyr(at)optiosoftware.com Fix for memory leak + string allocation bugs 18 ******************************************************************************* 19 */ 20 21 #include "unicode/utypes.h" 22 23 #if !UCONFIG_NO_FORMATTING && !UCONFIG_NO_CONVERSION 24 25 #include "locbund.h" 26 27 #include "cmemory.h" 28 #include "cstring.h" 29 #include "ucln_io.h" 30 #include "mutex.h" 31 #include "umutex.h" 32 #include "unicode/ustring.h" 33 #include "unicode/uloc.h" 34 35 static UNumberFormat *gPosixNumberFormat[ULOCALEBUNDLE_NUMBERFORMAT_COUNT]; 36 37 U_CDECL_BEGIN 38 static UBool U_CALLCONV locbund_cleanup(void) { 39 int32_t style; 40 for (style = 0; style < ULOCALEBUNDLE_NUMBERFORMAT_COUNT; style++) { 41 unum_close(gPosixNumberFormat[style]); 42 gPosixNumberFormat[style] = NULL; 43 } 44 return TRUE; 45 } 46 U_CDECL_END 47 48 static UMutex gLock = U_MUTEX_INITIALIZER; 49 static inline UNumberFormat * copyInvariantFormatter(ULocaleBundle *result, UNumberFormatStyle style) { 50 U_NAMESPACE_USE 51 Mutex lock(&gLock); 52 if (result->fNumberFormat[style-1] == NULL) { 53 if (gPosixNumberFormat[style-1] == NULL) { 54 UErrorCode status = U_ZERO_ERROR; 55 UNumberFormat *formatAlias = unum_open(style, NULL, 0, "en_US_POSIX", NULL, &status); 56 if (U_SUCCESS(status)) { 57 gPosixNumberFormat[style-1] = formatAlias; 58 ucln_io_registerCleanup(UCLN_IO_LOCBUND, locbund_cleanup); 59 } 60 } 61 /* Copy the needed formatter. */ 62 if (gPosixNumberFormat[style-1] != NULL) { 63 UErrorCode status = U_ZERO_ERROR; 64 result->fNumberFormat[style-1] = unum_clone(gPosixNumberFormat[style-1], &status); 65 } 66 } 67 return result->fNumberFormat[style-1]; 68 } 69 70 U_CAPI ULocaleBundle * 71 u_locbund_init(ULocaleBundle *result, const char *loc) 72 { 73 int32_t len; 74 75 if(result == 0) 76 return 0; 77 78 if (loc == NULL) { 79 loc = uloc_getDefault(); 80 } 81 82 uprv_memset(result, 0, sizeof(ULocaleBundle)); 83 84 len = (int32_t)strlen(loc); 85 result->fLocale = (char*) uprv_malloc(len + 1); 86 if(result->fLocale == 0) { 87 return 0; 88 } 89 90 uprv_strcpy(result->fLocale, loc); 91 92 result->isInvariantLocale = uprv_strcmp(result->fLocale, "en_US_POSIX") == 0; 93 94 return result; 95 } 96 97 /*U_CAPI ULocaleBundle * 98 u_locbund_new(const char *loc) 99 { 100 ULocaleBundle *result = (ULocaleBundle*) uprv_malloc(sizeof(ULocaleBundle)); 101 return u_locbund_init(result, loc); 102 } 103 104 U_CAPI ULocaleBundle * 105 u_locbund_clone(const ULocaleBundle *bundle) 106 { 107 ULocaleBundle *result = (ULocaleBundle*)uprv_malloc(sizeof(ULocaleBundle)); 108 UErrorCode status = U_ZERO_ERROR; 109 int32_t styleIdx; 110 111 if(result == 0) 112 return 0; 113 114 result->fLocale = (char*) uprv_malloc(strlen(bundle->fLocale) + 1); 115 if(result->fLocale == 0) { 116 uprv_free(result); 117 return 0; 118 } 119 120 strcpy(result->fLocale, bundle->fLocale ); 121 122 for (styleIdx = 0; styleIdx < ULOCALEBUNDLE_NUMBERFORMAT_COUNT; styleIdx++) { 123 status = U_ZERO_ERROR; 124 if (result->fNumberFormat[styleIdx]) { 125 result->fNumberFormat[styleIdx] = unum_clone(bundle->fNumberFormat[styleIdx], &status); 126 if (U_FAILURE(status)) { 127 result->fNumberFormat[styleIdx] = NULL; 128 } 129 } 130 else { 131 result->fNumberFormat[styleIdx] = NULL; 132 } 133 } 134 result->fDateFormat = (bundle->fDateFormat == 0 ? 0 : 135 udat_clone(bundle->fDateFormat, &status)); 136 result->fTimeFormat = (bundle->fTimeFormat == 0 ? 0 : 137 udat_clone(bundle->fTimeFormat, &status)); 138 139 return result; 140 }*/ 141 142 U_CAPI void 143 u_locbund_close(ULocaleBundle *bundle) 144 { 145 int32_t styleIdx; 146 147 uprv_free(bundle->fLocale); 148 149 for (styleIdx = 0; styleIdx < ULOCALEBUNDLE_NUMBERFORMAT_COUNT; styleIdx++) { 150 if (bundle->fNumberFormat[styleIdx]) { 151 unum_close(bundle->fNumberFormat[styleIdx]); 152 } 153 } 154 155 uprv_memset(bundle, 0, sizeof(ULocaleBundle)); 156 /* uprv_free(bundle);*/ 157 } 158 159 U_CAPI UNumberFormat * 160 u_locbund_getNumberFormat(ULocaleBundle *bundle, UNumberFormatStyle style) 161 { 162 UNumberFormat *formatAlias = NULL; 163 if (style > UNUM_IGNORE) { 164 formatAlias = bundle->fNumberFormat[style-1]; 165 if (formatAlias == NULL) { 166 if (bundle->isInvariantLocale) { 167 formatAlias = copyInvariantFormatter(bundle, style); 168 } 169 else { 170 UErrorCode status = U_ZERO_ERROR; 171 formatAlias = unum_open(style, NULL, 0, bundle->fLocale, NULL, &status); 172 if (U_FAILURE(status)) { 173 unum_close(formatAlias); 174 formatAlias = NULL; 175 } 176 else { 177 bundle->fNumberFormat[style-1] = formatAlias; 178 } 179 } 180 } 181 } 182 return formatAlias; 183 } 184 185 #endif /* #if !UCONFIG_NO_FORMATTING */ 186