1 // Copyright (C) 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
locbund_cleanup(void)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;
copyInvariantFormatter(ULocaleBundle * result,UNumberFormatStyle style)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 *
u_locbund_init(ULocaleBundle * result,const char * loc)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
u_locbund_close(ULocaleBundle * bundle)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 *
u_locbund_getNumberFormat(ULocaleBundle * bundle,UNumberFormatStyle style)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