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) 2002-2012, International Business Machines
7 *   Corporation and others.  All Rights Reserved.
8 *
9 *******************************************************************************
10 *   file name:  uenum.c
11 *   encoding:   US-ASCII
12 *   tab size:   8 (not used)
13 *   indentation:2
14 *
15 *   created on: 2002jul08
16 *   created by: Vladimir Weinstein
17 */
18 
19 #include "unicode/putil.h"
20 #include "uenumimp.h"
21 #include "cmemory.h"
22 
23 /* Layout of the baseContext buffer. */
24 typedef struct {
25     int32_t len;  /* number of bytes available starting at 'data' */
26     char    data; /* actual data starts here */
27 } _UEnumBuffer;
28 
29 /* Extra bytes to allocate in the baseContext buffer. */
30 static const int32_t PAD = 8;
31 
32 /* Return a pointer to the baseContext buffer, possibly allocating
33    or reallocating it if at least 'capacity' bytes are not available. */
_getBuffer(UEnumeration * en,int32_t capacity)34 static void* _getBuffer(UEnumeration* en, int32_t capacity) {
35 
36     if (en->baseContext != NULL) {
37         if (((_UEnumBuffer*) en->baseContext)->len < capacity) {
38             capacity += PAD;
39             en->baseContext = uprv_realloc(en->baseContext,
40                                            sizeof(int32_t) + capacity);
41             if (en->baseContext == NULL) {
42                 return NULL;
43             }
44             ((_UEnumBuffer*) en->baseContext)->len = capacity;
45         }
46     } else {
47         capacity += PAD;
48         en->baseContext = uprv_malloc(sizeof(int32_t) + capacity);
49         if (en->baseContext == NULL) {
50             return NULL;
51         }
52         ((_UEnumBuffer*) en->baseContext)->len = capacity;
53     }
54 
55     return (void*) & ((_UEnumBuffer*) en->baseContext)->data;
56 }
57 
58 U_CAPI void U_EXPORT2
uenum_close(UEnumeration * en)59 uenum_close(UEnumeration* en)
60 {
61     if (en) {
62         if (en->close != NULL) {
63             if (en->baseContext) {
64                 uprv_free(en->baseContext);
65             }
66             en->close(en);
67         } else { /* this seems dangerous, but we better kill the object */
68             uprv_free(en);
69         }
70     }
71 }
72 
73 U_CAPI int32_t U_EXPORT2
uenum_count(UEnumeration * en,UErrorCode * status)74 uenum_count(UEnumeration* en, UErrorCode* status)
75 {
76     if (!en || U_FAILURE(*status)) {
77         return -1;
78     }
79     if (en->count != NULL) {
80         return en->count(en, status);
81     } else {
82         *status = U_UNSUPPORTED_ERROR;
83         return -1;
84     }
85 }
86 
87 /* Don't call this directly. Only uenum_unext should be calling this. */
88 U_CAPI const UChar* U_EXPORT2
uenum_unextDefault(UEnumeration * en,int32_t * resultLength,UErrorCode * status)89 uenum_unextDefault(UEnumeration* en,
90             int32_t* resultLength,
91             UErrorCode* status)
92 {
93     UChar *ustr = NULL;
94     int32_t len = 0;
95     if (en->next != NULL) {
96         const char *cstr = en->next(en, &len, status);
97         if (cstr != NULL) {
98             ustr = (UChar*) _getBuffer(en, (len+1) * sizeof(UChar));
99             if (ustr == NULL) {
100                 *status = U_MEMORY_ALLOCATION_ERROR;
101             } else {
102                 u_charsToUChars(cstr, ustr, len+1);
103             }
104         }
105     } else {
106         *status = U_UNSUPPORTED_ERROR;
107     }
108     if (resultLength) {
109         *resultLength = len;
110     }
111     return ustr;
112 }
113 
114 /* Don't call this directly. Only uenum_next should be calling this. */
115 U_CAPI const char* U_EXPORT2
uenum_nextDefault(UEnumeration * en,int32_t * resultLength,UErrorCode * status)116 uenum_nextDefault(UEnumeration* en,
117             int32_t* resultLength,
118             UErrorCode* status)
119 {
120     if (en->uNext != NULL) {
121         char *tempCharVal;
122         const UChar *tempUCharVal = en->uNext(en, resultLength, status);
123         if (tempUCharVal == NULL) {
124             return NULL;
125         }
126         tempCharVal = (char*)
127             _getBuffer(en, (*resultLength+1) * sizeof(char));
128         if (!tempCharVal) {
129             *status = U_MEMORY_ALLOCATION_ERROR;
130             return NULL;
131         }
132         u_UCharsToChars(tempUCharVal, tempCharVal, *resultLength + 1);
133         return tempCharVal;
134     } else {
135         *status = U_UNSUPPORTED_ERROR;
136         return NULL;
137     }
138 }
139 
140 U_CAPI const UChar* U_EXPORT2
uenum_unext(UEnumeration * en,int32_t * resultLength,UErrorCode * status)141 uenum_unext(UEnumeration* en,
142             int32_t* resultLength,
143             UErrorCode* status)
144 {
145     if (!en || U_FAILURE(*status)) {
146         return NULL;
147     }
148     if (en->uNext != NULL) {
149         return en->uNext(en, resultLength, status);
150     } else {
151         *status = U_UNSUPPORTED_ERROR;
152         return NULL;
153     }
154 }
155 
156 U_CAPI const char* U_EXPORT2
uenum_next(UEnumeration * en,int32_t * resultLength,UErrorCode * status)157 uenum_next(UEnumeration* en,
158           int32_t* resultLength,
159           UErrorCode* status)
160 {
161     if (!en || U_FAILURE(*status)) {
162         return NULL;
163     }
164     if (en->next != NULL) {
165         if (resultLength != NULL) {
166             return en->next(en, resultLength, status);
167         }
168         else {
169             int32_t dummyLength=0;
170             return en->next(en, &dummyLength, status);
171         }
172     } else {
173         *status = U_UNSUPPORTED_ERROR;
174         return NULL;
175     }
176 }
177 
178 U_CAPI void U_EXPORT2
uenum_reset(UEnumeration * en,UErrorCode * status)179 uenum_reset(UEnumeration* en, UErrorCode* status)
180 {
181     if (!en || U_FAILURE(*status)) {
182         return;
183     }
184     if (en->reset != NULL) {
185         en->reset(en, status);
186     } else {
187         *status = U_UNSUPPORTED_ERROR;
188     }
189 }
190