• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 // © 2016 and later: Unicode, Inc. and others.
2 // License & terms of use: http://www.unicode.org/copyright.html
3 /*
4 ******************************************************************************
5 *   Copyright (C) 2001-2016, International Business Machines
6 *   Corporation and others.  All Rights Reserved.
7 ******************************************************************************
8 *
9 * File ucoleitr.cpp
10 *
11 * Modification History:
12 *
13 * Date        Name        Description
14 * 02/15/2001  synwee      Modified all methods to process its own function
15 *                         instead of calling the equivalent c++ api (coleitr.h)
16 * 2012-2014   markus      Rewritten in C++ again.
17 ******************************************************************************/
18 
19 #include "unicode/utypes.h"
20 
21 #if !UCONFIG_NO_COLLATION
22 
23 #include "unicode/coleitr.h"
24 #include "unicode/tblcoll.h"
25 #include "unicode/ucoleitr.h"
26 #include "unicode/ustring.h"
27 #include "unicode/sortkey.h"
28 #include "unicode/uobject.h"
29 #include "cmemory.h"
30 #include "usrchimp.h"
31 
32 U_NAMESPACE_USE
33 
34 #define BUFFER_LENGTH             100
35 
36 #define DEFAULT_BUFFER_SIZE 16
37 #define BUFFER_GROW 8
38 
39 #define ARRAY_COPY(dst, src, count) uprv_memcpy((void *) (dst), (void *) (src), (size_t)(count) * sizeof (src)[0])
40 
41 #define NEW_ARRAY(type, count) (type *) uprv_malloc((size_t)(count) * sizeof(type))
42 
43 #define DELETE_ARRAY(array) uprv_free((void *) (array))
44 
45 struct RCEI
46 {
47     uint32_t ce;
48     int32_t  low;
49     int32_t  high;
50 };
51 
52 U_NAMESPACE_BEGIN
53 
54 struct RCEBuffer
55 {
56     RCEI    defaultBuffer[DEFAULT_BUFFER_SIZE];
57     RCEI   *buffer;
58     int32_t bufferIndex;
59     int32_t bufferSize;
60 
61     RCEBuffer();
62     ~RCEBuffer();
63 
64     UBool isEmpty() const;
65     void  put(uint32_t ce, int32_t ixLow, int32_t ixHigh, UErrorCode &errorCode);
66     const RCEI *get();
67 };
68 
RCEBuffer()69 RCEBuffer::RCEBuffer()
70 {
71     buffer = defaultBuffer;
72     bufferIndex = 0;
73     bufferSize = UPRV_LENGTHOF(defaultBuffer);
74 }
75 
~RCEBuffer()76 RCEBuffer::~RCEBuffer()
77 {
78     if (buffer != defaultBuffer) {
79         DELETE_ARRAY(buffer);
80     }
81 }
82 
isEmpty() const83 UBool RCEBuffer::isEmpty() const
84 {
85     return bufferIndex <= 0;
86 }
87 
put(uint32_t ce,int32_t ixLow,int32_t ixHigh,UErrorCode & errorCode)88 void RCEBuffer::put(uint32_t ce, int32_t ixLow, int32_t ixHigh, UErrorCode &errorCode)
89 {
90     if (U_FAILURE(errorCode)) {
91         return;
92     }
93     if (bufferIndex >= bufferSize) {
94         RCEI *newBuffer = NEW_ARRAY(RCEI, bufferSize + BUFFER_GROW);
95         if (newBuffer == nullptr) {
96             errorCode = U_MEMORY_ALLOCATION_ERROR;
97             return;
98         }
99 
100         ARRAY_COPY(newBuffer, buffer, bufferSize);
101 
102         if (buffer != defaultBuffer) {
103             DELETE_ARRAY(buffer);
104         }
105 
106         buffer = newBuffer;
107         bufferSize += BUFFER_GROW;
108     }
109 
110     buffer[bufferIndex].ce   = ce;
111     buffer[bufferIndex].low  = ixLow;
112     buffer[bufferIndex].high = ixHigh;
113 
114     bufferIndex += 1;
115 }
116 
get()117 const RCEI *RCEBuffer::get()
118 {
119     if (bufferIndex > 0) {
120      return &buffer[--bufferIndex];
121     }
122 
123     return nullptr;
124 }
125 
PCEBuffer()126 PCEBuffer::PCEBuffer()
127 {
128     buffer = defaultBuffer;
129     bufferIndex = 0;
130     bufferSize = UPRV_LENGTHOF(defaultBuffer);
131 }
132 
~PCEBuffer()133 PCEBuffer::~PCEBuffer()
134 {
135     if (buffer != defaultBuffer) {
136         DELETE_ARRAY(buffer);
137     }
138 }
139 
reset()140 void PCEBuffer::reset()
141 {
142     bufferIndex = 0;
143 }
144 
isEmpty() const145 UBool PCEBuffer::isEmpty() const
146 {
147     return bufferIndex <= 0;
148 }
149 
put(uint64_t ce,int32_t ixLow,int32_t ixHigh,UErrorCode & errorCode)150 void PCEBuffer::put(uint64_t ce, int32_t ixLow, int32_t ixHigh, UErrorCode &errorCode)
151 {
152     if (U_FAILURE(errorCode)) {
153         return;
154     }
155     if (bufferIndex >= bufferSize) {
156         PCEI *newBuffer = NEW_ARRAY(PCEI, bufferSize + BUFFER_GROW);
157         if (newBuffer == nullptr) {
158             errorCode = U_MEMORY_ALLOCATION_ERROR;
159             return;
160         }
161 
162         ARRAY_COPY(newBuffer, buffer, bufferSize);
163 
164         if (buffer != defaultBuffer) {
165             DELETE_ARRAY(buffer);
166         }
167 
168         buffer = newBuffer;
169         bufferSize += BUFFER_GROW;
170     }
171 
172     buffer[bufferIndex].ce   = ce;
173     buffer[bufferIndex].low  = ixLow;
174     buffer[bufferIndex].high = ixHigh;
175 
176     bufferIndex += 1;
177 }
178 
get()179 const PCEI *PCEBuffer::get()
180 {
181     if (bufferIndex > 0) {
182      return &buffer[--bufferIndex];
183     }
184 
185     return nullptr;
186 }
187 
UCollationPCE(UCollationElements * elems)188 UCollationPCE::UCollationPCE(UCollationElements *elems) { init(elems); }
189 
UCollationPCE(CollationElementIterator * iter)190 UCollationPCE::UCollationPCE(CollationElementIterator *iter) { init(iter); }
191 
init(UCollationElements * elems)192 void UCollationPCE::init(UCollationElements *elems) {
193     init(CollationElementIterator::fromUCollationElements(elems));
194 }
195 
init(CollationElementIterator * iter)196 void UCollationPCE::init(CollationElementIterator *iter)
197 {
198     cei = iter;
199     init(*iter->rbc_);
200 }
201 
init(const Collator & coll)202 void UCollationPCE::init(const Collator &coll)
203 {
204     UErrorCode status = U_ZERO_ERROR;
205 
206     strength    = coll.getAttribute(UCOL_STRENGTH, status);
207     toShift     = coll.getAttribute(UCOL_ALTERNATE_HANDLING, status) == UCOL_SHIFTED;
208     isShifted   = false;
209     variableTop = coll.getVariableTop(status);
210 }
211 
~UCollationPCE()212 UCollationPCE::~UCollationPCE()
213 {
214     // nothing to do
215 }
216 
processCE(uint32_t ce)217 uint64_t UCollationPCE::processCE(uint32_t ce)
218 {
219     uint64_t primary = 0, secondary = 0, tertiary = 0, quaternary = 0;
220 
221     // This is clean, but somewhat slow...
222     // We could apply the mask to ce and then
223     // just get all three orders...
224     switch(strength) {
225     default:
226         tertiary = ucol_tertiaryOrder(ce);
227         U_FALLTHROUGH;
228 
229     case UCOL_SECONDARY:
230         secondary = ucol_secondaryOrder(ce);
231         U_FALLTHROUGH;
232 
233     case UCOL_PRIMARY:
234         primary = ucol_primaryOrder(ce);
235     }
236 
237     // **** This should probably handle continuations too.  ****
238     // **** That means that we need 24 bits for the primary ****
239     // **** instead of the 16 that we're currently using.   ****
240     // **** So we can lay out the 64 bits as: 24.12.12.16.  ****
241     // **** Another complication with continuations is that ****
242     // **** the *second* CE is marked as a continuation, so ****
243     // **** we always have to peek ahead to know how long   ****
244     // **** the primary is...                               ****
245     if ((toShift && variableTop > ce && primary != 0)
246                 || (isShifted && primary == 0)) {
247 
248         if (primary == 0) {
249             return UCOL_IGNORABLE;
250         }
251 
252         if (strength >= UCOL_QUATERNARY) {
253             quaternary = primary;
254         }
255 
256         primary = secondary = tertiary = 0;
257         isShifted = true;
258     } else {
259         if (strength >= UCOL_QUATERNARY) {
260             quaternary = 0xFFFF;
261         }
262 
263         isShifted = false;
264     }
265 
266     return primary << 48 | secondary << 32 | tertiary << 16 | quaternary;
267 }
268 
269 U_NAMESPACE_END
270 
271 /* public methods ---------------------------------------------------- */
272 
273 U_CAPI UCollationElements* U_EXPORT2
ucol_openElements(const UCollator * coll,const char16_t * text,int32_t textLength,UErrorCode * status)274 ucol_openElements(const UCollator  *coll,
275                   const char16_t   *text,
276                         int32_t    textLength,
277                         UErrorCode *status)
278 {
279     if (U_FAILURE(*status)) {
280         return nullptr;
281     }
282     if (coll == nullptr || (text == nullptr && textLength != 0)) {
283         *status = U_ILLEGAL_ARGUMENT_ERROR;
284         return nullptr;
285     }
286     const RuleBasedCollator *rbc = RuleBasedCollator::rbcFromUCollator(coll);
287     if (rbc == nullptr) {
288         *status = U_UNSUPPORTED_ERROR;  // coll is a Collator but not a RuleBasedCollator
289         return nullptr;
290     }
291 
292     UnicodeString s((UBool)(textLength < 0), text, textLength);
293     CollationElementIterator *cei = rbc->createCollationElementIterator(s);
294     if (cei == nullptr) {
295         *status = U_MEMORY_ALLOCATION_ERROR;
296         return nullptr;
297     }
298 
299     return cei->toUCollationElements();
300 }
301 
302 
303 U_CAPI void U_EXPORT2
ucol_closeElements(UCollationElements * elems)304 ucol_closeElements(UCollationElements *elems)
305 {
306     delete CollationElementIterator::fromUCollationElements(elems);
307 }
308 
309 U_CAPI void U_EXPORT2
ucol_reset(UCollationElements * elems)310 ucol_reset(UCollationElements *elems)
311 {
312     CollationElementIterator::fromUCollationElements(elems)->reset();
313 }
314 
315 U_CAPI int32_t U_EXPORT2
ucol_next(UCollationElements * elems,UErrorCode * status)316 ucol_next(UCollationElements *elems,
317           UErrorCode         *status)
318 {
319     if (U_FAILURE(*status)) {
320         return UCOL_NULLORDER;
321     }
322 
323     return CollationElementIterator::fromUCollationElements(elems)->next(*status);
324 }
325 
326 U_NAMESPACE_BEGIN
327 
328 int64_t
nextProcessed(int32_t * ixLow,int32_t * ixHigh,UErrorCode * status)329 UCollationPCE::nextProcessed(
330                    int32_t            *ixLow,
331                    int32_t            *ixHigh,
332                    UErrorCode         *status)
333 {
334     int64_t result = UCOL_IGNORABLE;
335     uint32_t low = 0, high = 0;
336 
337     if (U_FAILURE(*status)) {
338         return UCOL_PROCESSED_NULLORDER;
339     }
340 
341     pceBuffer.reset();
342 
343     do {
344         low = cei->getOffset();
345         int32_t ce = cei->next(*status);
346         high = cei->getOffset();
347 
348         if (ce == UCOL_NULLORDER) {
349              result = UCOL_PROCESSED_NULLORDER;
350              break;
351         }
352 
353         result = processCE((uint32_t)ce);
354     } while (result == UCOL_IGNORABLE);
355 
356     if (ixLow != nullptr) {
357         *ixLow = low;
358     }
359 
360     if (ixHigh != nullptr) {
361         *ixHigh = high;
362     }
363 
364     return result;
365 }
366 
367 U_NAMESPACE_END
368 
369 U_CAPI int32_t U_EXPORT2
ucol_previous(UCollationElements * elems,UErrorCode * status)370 ucol_previous(UCollationElements *elems,
371               UErrorCode         *status)
372 {
373     if(U_FAILURE(*status)) {
374         return UCOL_NULLORDER;
375     }
376     return CollationElementIterator::fromUCollationElements(elems)->previous(*status);
377 }
378 
379 U_NAMESPACE_BEGIN
380 
381 int64_t
previousProcessed(int32_t * ixLow,int32_t * ixHigh,UErrorCode * status)382 UCollationPCE::previousProcessed(
383                    int32_t            *ixLow,
384                    int32_t            *ixHigh,
385                    UErrorCode         *status)
386 {
387     int64_t result = UCOL_IGNORABLE;
388     int32_t  low = 0, high = 0;
389 
390     if (U_FAILURE(*status)) {
391         return UCOL_PROCESSED_NULLORDER;
392     }
393 
394     // pceBuffer.reset();
395 
396     while (pceBuffer.isEmpty()) {
397         // buffer raw CEs up to non-ignorable primary
398         RCEBuffer rceb;
399         int32_t ce;
400 
401         // **** do we need to reset rceb, or will it always be empty at this point ****
402         do {
403             high = cei->getOffset();
404             ce   = cei->previous(*status);
405             low  = cei->getOffset();
406 
407             if (ce == UCOL_NULLORDER) {
408                 if (!rceb.isEmpty()) {
409                     break;
410                 }
411 
412                 goto finish;
413             }
414 
415             rceb.put((uint32_t)ce, low, high, *status);
416         } while (U_SUCCESS(*status) && ((ce & UCOL_PRIMARYORDERMASK) == 0 || isContinuation(ce)));
417 
418         // process the raw CEs
419         while (U_SUCCESS(*status) && !rceb.isEmpty()) {
420             const RCEI *rcei = rceb.get();
421 
422             result = processCE(rcei->ce);
423 
424             if (result != UCOL_IGNORABLE) {
425                 pceBuffer.put(result, rcei->low, rcei->high, *status);
426             }
427         }
428         if (U_FAILURE(*status)) {
429             return UCOL_PROCESSED_NULLORDER;
430         }
431     }
432 
433 finish:
434     if (pceBuffer.isEmpty()) {
435         // **** Is -1 the right value for ixLow, ixHigh? ****
436     	if (ixLow != nullptr) {
437     		*ixLow = -1;
438     	}
439 
440     	if (ixHigh != nullptr) {
441     		*ixHigh = -1
442     		;
443     	}
444         return UCOL_PROCESSED_NULLORDER;
445     }
446 
447     const PCEI *pcei = pceBuffer.get();
448 
449     if (ixLow != nullptr) {
450         *ixLow = pcei->low;
451     }
452 
453     if (ixHigh != nullptr) {
454         *ixHigh = pcei->high;
455     }
456 
457     return pcei->ce;
458 }
459 
460 U_NAMESPACE_END
461 
462 U_CAPI int32_t U_EXPORT2
ucol_getMaxExpansion(const UCollationElements * elems,int32_t order)463 ucol_getMaxExpansion(const UCollationElements *elems,
464                            int32_t            order)
465 {
466     return CollationElementIterator::fromUCollationElements(elems)->getMaxExpansion(order);
467 
468     // TODO: The old code masked the order according to strength and then did a binary search.
469     // However this was probably at least partially broken because of the following comment.
470     // Still, it might have found a match when this version may not.
471 
472     // FIXME: with a masked search, there might be more than one hit,
473     // so we need to look forward and backward from the match to find all
474     // of the hits...
475 }
476 
477 U_CAPI void U_EXPORT2
ucol_setText(UCollationElements * elems,const char16_t * text,int32_t textLength,UErrorCode * status)478 ucol_setText(      UCollationElements *elems,
479              const char16_t           *text,
480                    int32_t            textLength,
481                    UErrorCode         *status)
482 {
483     if (U_FAILURE(*status)) {
484         return;
485     }
486 
487     if ((text == nullptr && textLength != 0)) {
488         *status = U_ILLEGAL_ARGUMENT_ERROR;
489         return;
490     }
491     UnicodeString s((UBool)(textLength < 0), text, textLength);
492     return CollationElementIterator::fromUCollationElements(elems)->setText(s, *status);
493 }
494 
495 U_CAPI int32_t U_EXPORT2
ucol_getOffset(const UCollationElements * elems)496 ucol_getOffset(const UCollationElements *elems)
497 {
498     return CollationElementIterator::fromUCollationElements(elems)->getOffset();
499 }
500 
501 U_CAPI void U_EXPORT2
ucol_setOffset(UCollationElements * elems,int32_t offset,UErrorCode * status)502 ucol_setOffset(UCollationElements    *elems,
503                int32_t           offset,
504                UErrorCode            *status)
505 {
506     if (U_FAILURE(*status)) {
507         return;
508     }
509 
510     CollationElementIterator::fromUCollationElements(elems)->setOffset(offset, *status);
511 }
512 
513 U_CAPI int32_t U_EXPORT2
ucol_primaryOrder(int32_t order)514 ucol_primaryOrder (int32_t order)
515 {
516     return (order >> 16) & 0xffff;
517 }
518 
519 U_CAPI int32_t U_EXPORT2
ucol_secondaryOrder(int32_t order)520 ucol_secondaryOrder (int32_t order)
521 {
522     return (order >> 8) & 0xff;
523 }
524 
525 U_CAPI int32_t U_EXPORT2
ucol_tertiaryOrder(int32_t order)526 ucol_tertiaryOrder (int32_t order)
527 {
528     return order & 0xff;
529 }
530 
531 #endif /* #if !UCONFIG_NO_COLLATION */
532