Coverage Report

Created: 2026-02-05 06:34

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/icu/icu4c/source/i18n/collationweights.cpp
Line
Count
Source
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) 1999-2015, International Business Machines
7
*   Corporation and others.  All Rights Reserved.
8
*
9
*******************************************************************************
10
*   file name:  collationweights.cpp
11
*   encoding:   UTF-8
12
*   tab size:   8 (not used)
13
*   indentation:4
14
*
15
*   created on: 2001mar08 as ucol_wgt.cpp
16
*   created by: Markus W. Scherer
17
*
18
*   This file contains code for allocating n collation element weights
19
*   between two exclusive limits.
20
*   It is used only internally by the collation tailoring builder.
21
*/
22
23
#include "unicode/utypes.h"
24
25
#if !UCONFIG_NO_COLLATION
26
27
#include "cmemory.h"
28
#include "collation.h"
29
#include "collationweights.h"
30
#include "uarrsort.h"
31
#include "uassert.h"
32
33
#ifdef UCOL_DEBUG
34
#   include <stdio.h>
35
#endif
36
37
U_NAMESPACE_BEGIN
38
39
/* collation element weight allocation -------------------------------------- */
40
41
/* helper functions for CE weights */
42
43
static inline uint32_t
44
1.93M
getWeightTrail(uint32_t weight, int32_t length) {
45
1.93M
    return (weight >> (8 * (4 - length))) & 0xff;
46
1.93M
}
47
48
static inline uint32_t
49
23.3k
setWeightTrail(uint32_t weight, int32_t length, uint32_t trail) {
50
23.3k
    length=8*(4-length);
51
23.3k
    return static_cast<uint32_t>((weight & (0xffffff00 << length)) | (trail << length));
52
23.3k
}
53
54
static inline uint32_t
55
1.90M
getWeightByte(uint32_t weight, int32_t idx) {
56
1.90M
    return getWeightTrail(weight, idx); /* same calculation */
57
1.90M
}
58
59
static inline uint32_t
60
1.90M
setWeightByte(uint32_t weight, int32_t idx, uint32_t byte) {
61
1.90M
    uint32_t mask; /* 0xffffffff except a 00 "hole" for the index-th byte */
62
63
1.90M
    idx*=8;
64
1.90M
    if(idx<32) {
65
1.10M
        mask = (static_cast<uint32_t>(0xffffffff)) >> idx;
66
1.10M
    } else {
67
        // Do not use uint32_t>>32 because on some platforms that does not shift at all
68
        // while we need it to become 0.
69
        // PowerPC: 0xffffffff>>32 = 0           (wanted)
70
        // x86:     0xffffffff>>32 = 0xffffffff  (not wanted)
71
        //
72
        // ANSI C99 6.5.7 Bitwise shift operators:
73
        // "If the value of the right operand is negative
74
        // or is greater than or equal to the width of the promoted left operand,
75
        // the behavior is undefined."
76
796k
        mask=0;
77
796k
    }
78
1.90M
    idx=32-idx;
79
1.90M
    mask|=0xffffff00<<idx;
80
1.90M
    return ((weight & mask) | (byte << idx));
81
1.90M
}
82
83
static inline uint32_t
84
19.7k
truncateWeight(uint32_t weight, int32_t length) {
85
19.7k
    return static_cast<uint32_t>(weight & (0xffffffff << (8 * (4 - length))));
86
19.7k
}
87
88
static inline uint32_t
89
28.6k
incWeightTrail(uint32_t weight, int32_t length) {
90
28.6k
    return static_cast<uint32_t>(weight + (1UL << (8 * (4 - length))));
91
28.6k
}
92
93
static inline uint32_t
94
28.5k
decWeightTrail(uint32_t weight, int32_t length) {
95
28.5k
    return static_cast<uint32_t>(weight - (1UL << (8 * (4 - length))));
96
28.5k
}
97
98
CollationWeights::CollationWeights()
99
11.4k
        : middleLength(0), rangeIndex(0), rangeCount(0) {
100
68.9k
    for(int32_t i = 0; i < 5; ++i) {
101
57.4k
        minBytes[i] = maxBytes[i] = 0;
102
57.4k
    }
103
11.4k
}
104
105
void
106
7.33k
CollationWeights::initForPrimary(UBool compressible) {
107
7.33k
    middleLength=1;
108
7.33k
    minBytes[1] = Collation::MERGE_SEPARATOR_BYTE + 1;
109
7.33k
    maxBytes[1] = Collation::TRAIL_WEIGHT_BYTE;
110
7.33k
    if(compressible) {
111
2.96k
        minBytes[2] = Collation::PRIMARY_COMPRESSION_LOW_BYTE + 1;
112
2.96k
        maxBytes[2] = Collation::PRIMARY_COMPRESSION_HIGH_BYTE - 1;
113
4.36k
    } else {
114
4.36k
        minBytes[2] = 2;
115
4.36k
        maxBytes[2] = 0xff;
116
4.36k
    }
117
7.33k
    minBytes[3] = 2;
118
7.33k
    maxBytes[3] = 0xff;
119
7.33k
    minBytes[4] = 2;
120
7.33k
    maxBytes[4] = 0xff;
121
7.33k
}
122
123
void
124
5.58k
CollationWeights::initForSecondary() {
125
    // We use only the lower 16 bits for secondary weights.
126
5.58k
    middleLength=3;
127
5.58k
    minBytes[1] = 0;
128
5.58k
    maxBytes[1] = 0;
129
5.58k
    minBytes[2] = 0;
130
5.58k
    maxBytes[2] = 0;
131
5.58k
    minBytes[3] = Collation::LEVEL_SEPARATOR_BYTE + 1;
132
5.58k
    maxBytes[3] = 0xff;
133
5.58k
    minBytes[4] = 2;
134
5.58k
    maxBytes[4] = 0xff;
135
5.58k
}
136
137
void
138
6.87k
CollationWeights::initForTertiary() {
139
    // We use only the lower 16 bits for tertiary weights.
140
6.87k
    middleLength=3;
141
6.87k
    minBytes[1] = 0;
142
6.87k
    maxBytes[1] = 0;
143
6.87k
    minBytes[2] = 0;
144
6.87k
    maxBytes[2] = 0;
145
    // We use only 6 bits per byte.
146
    // The other bits are used for case & quaternary weights.
147
6.87k
    minBytes[3] = Collation::LEVEL_SEPARATOR_BYTE + 1;
148
6.87k
    maxBytes[3] = 0x3f;
149
6.87k
    minBytes[4] = 2;
150
6.87k
    maxBytes[4] = 0x3f;
151
6.87k
}
152
153
uint32_t
154
1.89M
CollationWeights::incWeight(uint32_t weight, int32_t length) const {
155
1.89M
    for(;;) {
156
1.89M
        uint32_t byte=getWeightByte(weight, length);
157
1.89M
        if(byte<maxBytes[length]) {
158
1.89M
            return setWeightByte(weight, length, byte+1);
159
1.89M
        } else {
160
            // Roll over, set this byte to the minimum and increment the previous one.
161
7.75k
            weight=setWeightByte(weight, length, minBytes[length]);
162
7.75k
            --length;
163
7.75k
            U_ASSERT(length > 0);
164
7.75k
        }
165
1.89M
    }
166
1.89M
}
167
168
uint32_t
169
869
CollationWeights::incWeightByOffset(uint32_t weight, int32_t length, int32_t offset) const {
170
910
    for(;;) {
171
910
        offset += getWeightByte(weight, length);
172
910
        if (static_cast<uint32_t>(offset) <= maxBytes[length]) {
173
869
            return setWeightByte(weight, length, offset);
174
869
        } else {
175
            // Split the offset between this byte and the previous one.
176
41
            offset -= minBytes[length];
177
41
            weight = setWeightByte(weight, length, minBytes[length] + offset % countBytes(length));
178
41
            offset /= countBytes(length);
179
41
            --length;
180
41
            U_ASSERT(length > 0);
181
41
        }
182
910
    }
183
869
}
184
185
void
186
2.89k
CollationWeights::lengthenRange(WeightRange &range) const {
187
2.89k
    int32_t length=range.length+1;
188
2.89k
    range.start=setWeightTrail(range.start, length, minBytes[length]);
189
2.89k
    range.end=setWeightTrail(range.end, length, maxBytes[length]);
190
2.89k
    range.count*=countBytes(length);
191
2.89k
    range.length=length;
192
2.89k
}
193
194
/* for uprv_sortArray: sort ranges in weight order */
195
static int32_t U_CALLCONV
196
67
compareRanges(const void * /*context*/, const void *left, const void *right) {
197
67
    uint32_t l, r;
198
199
67
    l = static_cast<const CollationWeights::WeightRange*>(left)->start;
200
67
    r = static_cast<const CollationWeights::WeightRange*>(right)->start;
201
67
    if(l<r) {
202
67
        return -1;
203
67
    } else if(l>r) {
204
0
        return 1;
205
0
    } else {
206
0
        return 0;
207
0
    }
208
67
}
209
210
UBool
211
19.7k
CollationWeights::getWeightRanges(uint32_t lowerLimit, uint32_t upperLimit) {
212
19.7k
    U_ASSERT(lowerLimit != 0);
213
19.7k
    U_ASSERT(upperLimit != 0);
214
215
    /* get the lengths of the limits */
216
19.7k
    int32_t lowerLength=lengthOfWeight(lowerLimit);
217
19.7k
    int32_t upperLength=lengthOfWeight(upperLimit);
218
219
#ifdef UCOL_DEBUG
220
    printf("length of lower limit 0x%08lx is %ld\n", lowerLimit, lowerLength);
221
    printf("length of upper limit 0x%08lx is %ld\n", upperLimit, upperLength);
222
#endif
223
19.7k
    U_ASSERT(lowerLength>=middleLength);
224
    // Permit upperLength<middleLength: The upper limit for secondaries is 0x10000.
225
226
19.7k
    if(lowerLimit>=upperLimit) {
227
#ifdef UCOL_DEBUG
228
        printf("error: no space between lower & upper limits\n");
229
#endif
230
0
        return false;
231
0
    }
232
233
    /* check that neither is a prefix of the other */
234
19.7k
    if(lowerLength<upperLength) {
235
1.35k
        if(lowerLimit==truncateWeight(upperLimit, lowerLength)) {
236
#ifdef UCOL_DEBUG
237
            printf("error: lower limit 0x%08lx is a prefix of upper limit 0x%08lx\n", lowerLimit, upperLimit);
238
#endif
239
0
            return false;
240
0
        }
241
1.35k
    }
242
    /* if the upper limit is a prefix of the lower limit then the earlier test lowerLimit>=upperLimit has caught it */
243
244
19.7k
    WeightRange lower[5], middle, upper[5]; /* [0] and [1] are not used - this simplifies indexing */
245
19.7k
    uprv_memset(lower, 0, sizeof(lower));
246
19.7k
    uprv_memset(&middle, 0, sizeof(middle));
247
19.7k
    uprv_memset(upper, 0, sizeof(upper));
248
249
    /*
250
     * With the limit lengths of 1..4, there are up to 7 ranges for allocation:
251
     * range     minimum length
252
     * lower[4]  4
253
     * lower[3]  3
254
     * lower[2]  2
255
     * middle    1
256
     * upper[2]  2
257
     * upper[3]  3
258
     * upper[4]  4
259
     *
260
     * We are now going to calculate up to 7 ranges.
261
     * Some of them will typically overlap, so we will then have to merge and eliminate ranges.
262
     */
263
19.7k
    uint32_t weight=lowerLimit;
264
28.6k
    for(int32_t length=lowerLength; length>middleLength; --length) {
265
8.85k
        uint32_t trail=getWeightTrail(weight, length);
266
8.85k
        if(trail<maxBytes[length]) {
267
8.81k
            lower[length].start=incWeightTrail(weight, length);
268
8.81k
            lower[length].end=setWeightTrail(weight, length, maxBytes[length]);
269
8.81k
            lower[length].length=length;
270
8.81k
            lower[length].count=maxBytes[length]-trail;
271
8.81k
        }
272
8.85k
        weight=truncateWeight(weight, length-1);
273
8.85k
    }
274
19.7k
    if(weight<0xff000000) {
275
19.7k
        middle.start=incWeightTrail(weight, middleLength);
276
19.7k
    } else {
277
        // Prevent overflow for primary lead byte FF
278
        // which would yield a middle range starting at 0.
279
0
        middle.start=0xffffffff;  // no middle range
280
0
    }
281
282
19.7k
    weight=upperLimit;
283
29.3k
    for(int32_t length=upperLength; length>middleLength; --length) {
284
9.51k
        uint32_t trail=getWeightTrail(weight, length);
285
9.51k
        if(trail>minBytes[length]) {
286
8.77k
            upper[length].start=setWeightTrail(weight, length, minBytes[length]);
287
8.77k
            upper[length].end=decWeightTrail(weight, length);
288
8.77k
            upper[length].length=length;
289
8.77k
            upper[length].count=trail-minBytes[length];
290
8.77k
        }
291
9.51k
        weight=truncateWeight(weight, length-1);
292
9.51k
    }
293
19.7k
    middle.end=decWeightTrail(weight, middleLength);
294
295
    /* set the middle range */
296
19.7k
    middle.length=middleLength;
297
19.7k
    if(middle.end>=middle.start) {
298
12.0k
        middle.count = static_cast<int32_t>((middle.end - middle.start) >> (8 * (4 - middleLength))) + 1;
299
12.0k
    } else {
300
        /* no middle range, eliminate overlaps */
301
21.8k
        for(int32_t length=4; length>middleLength; --length) {
302
20.4k
            if(lower[length].count>0 && upper[length].count>0) {
303
                // Note: The lowerEnd and upperStart weights are versions of
304
                // lowerLimit and upperLimit (which are lowerLimit<upperLimit),
305
                // truncated (still less-or-equal)
306
                // and then with their last bytes changed to the
307
                // maxByte (for lowerEnd) or minByte (for upperStart).
308
6.37k
                const uint32_t lowerEnd=lower[length].end;
309
6.37k
                const uint32_t upperStart=upper[length].start;
310
6.37k
                UBool merged=false;
311
312
6.37k
                if(lowerEnd>upperStart) {
313
                    // These two lower and upper ranges collide.
314
                    // Since lowerLimit<upperLimit and lowerEnd and upperStart
315
                    // are versions with only their last bytes modified
316
                    // (and following ones removed/reset to 0),
317
                    // lowerEnd>upperStart is only possible
318
                    // if the leading bytes are equal
319
                    // and lastByte(lowerEnd)>lastByte(upperStart).
320
5.94k
                    U_ASSERT(truncateWeight(lowerEnd, length-1)==
321
5.94k
                            truncateWeight(upperStart, length-1));
322
                    // Intersect these two ranges.
323
5.94k
                    lower[length].end=upper[length].end;
324
5.94k
                    lower[length].count=
325
5.94k
                            static_cast<int32_t>(getWeightTrail(lower[length].end, length)) -
326
5.94k
                            static_cast<int32_t>(getWeightTrail(lower[length].start, length)) + 1;
327
                    // count might be <=0 in which case there is no room,
328
                    // and the range-collecting code below will ignore this range.
329
5.94k
                    merged=true;
330
5.94k
                } else if(lowerEnd==upperStart) {
331
                    // Not possible, unless minByte==maxByte which is not allowed.
332
0
                    U_ASSERT(minBytes[length]<maxBytes[length]);
333
430
                } else /* lowerEnd<upperStart */ {
334
430
                    if(incWeight(lowerEnd, length)==upperStart) {
335
                        // Merge adjacent ranges.
336
430
                        lower[length].end=upper[length].end;
337
430
                        lower[length].count+=upper[length].count;  // might be >countBytes
338
430
                        merged=true;
339
430
                    }
340
430
                }
341
6.37k
                if(merged) {
342
                    // Remove all shorter ranges.
343
                    // There was no room available for them between the ranges we just merged.
344
6.37k
                    upper[length].count=0;
345
8.07k
                    while(--length>middleLength) {
346
1.69k
                        lower[length].count=upper[length].count=0;
347
1.69k
                    }
348
6.37k
                    break;
349
6.37k
                }
350
6.37k
            }
351
20.4k
        }
352
7.77k
    }
353
354
#ifdef UCOL_DEBUG
355
    /* print ranges */
356
    for(int32_t length=4; length>=2; --length) {
357
        if(lower[length].count>0) {
358
            printf("lower[%ld] .start=0x%08lx .end=0x%08lx .count=%ld\n", length, lower[length].start, lower[length].end, lower[length].count);
359
        }
360
    }
361
    if(middle.count>0) {
362
        printf("middle   .start=0x%08lx .end=0x%08lx .count=%ld\n", middle.start, middle.end, middle.count);
363
    }
364
    for(int32_t length=2; length<=4; ++length) {
365
        if(upper[length].count>0) {
366
            printf("upper[%ld] .start=0x%08lx .end=0x%08lx .count=%ld\n", length, upper[length].start, upper[length].end, upper[length].count);
367
        }
368
    }
369
#endif
370
371
    /* copy the ranges, shortest first, into the result array */
372
19.7k
    rangeCount=0;
373
19.7k
    if(middle.count>0) {
374
12.0k
        uprv_memcpy(ranges, &middle, sizeof(WeightRange));
375
12.0k
        rangeCount=1;
376
12.0k
    }
377
54.2k
    for(int32_t length=middleLength+1; length<=4; ++length) {
378
        /* copy upper first so that later the middle range is more likely the first one to use */
379
34.4k
        if(upper[length].count>0) {
380
777
            uprv_memcpy(ranges+rangeCount, upper+length, sizeof(WeightRange));
381
777
            ++rangeCount;
382
777
        }
383
34.4k
        if(lower[length].count>0) {
384
7.13k
            uprv_memcpy(ranges+rangeCount, lower+length, sizeof(WeightRange));
385
7.13k
            ++rangeCount;
386
7.13k
        }
387
34.4k
    }
388
19.7k
    return rangeCount>0;
389
19.7k
}
390
391
UBool
392
20.0k
CollationWeights::allocWeightsInShortRanges(int32_t n, int32_t minLength) {
393
    // See if the first few minLength and minLength+1 ranges have enough weights.
394
23.0k
    for(int32_t i = 0; i < rangeCount && ranges[i].length <= (minLength + 1); ++i) {
395
20.1k
        if(n <= ranges[i].count) {
396
            // Use the first few minLength and minLength+1 ranges.
397
17.1k
            if(ranges[i].length > minLength) {
398
                // Reduce the number of weights from the last minLength+1 range
399
                // which might sort before some minLength ranges,
400
                // so that we use all weights in the minLength ranges.
401
52
                ranges[i].count = n;
402
52
            }
403
17.1k
            rangeCount = i + 1;
404
#ifdef UCOL_DEBUG
405
            printf("take first %ld ranges\n", rangeCount);
406
#endif
407
408
17.1k
            if(rangeCount>1) {
409
                /* sort the ranges by weight values */
410
67
                UErrorCode errorCode=U_ZERO_ERROR;
411
67
                uprv_sortArray(ranges, rangeCount, sizeof(WeightRange),
412
67
                               compareRanges, nullptr, false, &errorCode);
413
                /* ignore error code: we know that the internal sort function will not fail here */
414
67
            }
415
17.1k
            return true;
416
17.1k
        }
417
3.05k
        n -= ranges[i].count;  // still >0
418
3.05k
    }
419
2.91k
    return false;
420
20.0k
}
421
422
UBool
423
2.89k
CollationWeights::allocWeightsInMinLengthRanges(int32_t n, int32_t minLength) {
424
    // See if the minLength ranges have enough weights
425
    // when we split one and lengthen the following ones.
426
2.89k
    int32_t count = 0;
427
2.89k
    int32_t minLengthRangeCount;
428
2.89k
    for(minLengthRangeCount = 0;
429
5.79k
            minLengthRangeCount < rangeCount &&
430
2.96k
                ranges[minLengthRangeCount].length == minLength;
431
2.90k
            ++minLengthRangeCount) {
432
2.90k
        count += ranges[minLengthRangeCount].count;
433
2.90k
    }
434
435
2.89k
    int32_t nextCountBytes = countBytes(minLength + 1);
436
2.89k
    if(n > count * nextCountBytes) { return false; }
437
438
    // Use the minLength ranges. Merge them, and then split again as necessary.
439
2.65k
    uint32_t start = ranges[0].start;
440
2.65k
    uint32_t end = ranges[0].end;
441
2.66k
    for(int32_t i = 1; i < minLengthRangeCount; ++i) {
442
10
        if(ranges[i].start < start) { start = ranges[i].start; }
443
10
        if(ranges[i].end > end) { end = ranges[i].end; }
444
10
    }
445
446
    // Calculate how to split the range between minLength (count1) and minLength+1 (count2).
447
    // Goal:
448
    //   count1 + count2 * nextCountBytes = n
449
    //   count1 + count2 = count
450
    // These turn into
451
    //   (count - count2) + count2 * nextCountBytes = n
452
    // and then into the following count1 & count2 computations.
453
2.65k
    int32_t count2 = (n - count) / (nextCountBytes - 1);  // number of weights to be lengthened
454
2.65k
    int32_t count1 = count - count2;  // number of minLength weights
455
2.65k
    if(count2 == 0 || (count1 + count2 * nextCountBytes) < n) {
456
        // round up
457
2.62k
        ++count2;
458
2.62k
        --count1;
459
2.62k
        U_ASSERT((count1 + count2 * nextCountBytes) >= n);
460
2.62k
    }
461
462
2.65k
    ranges[0].start = start;
463
464
2.65k
    if(count1 == 0) {
465
        // Make one long range.
466
1.78k
        ranges[0].end = end;
467
1.78k
        ranges[0].count = count;
468
1.78k
        lengthenRange(ranges[0]);
469
1.78k
        rangeCount = 1;
470
1.78k
    } else {
471
        // Split the range, lengthen the second part.
472
#ifdef UCOL_DEBUG
473
        printf("split the range number %ld (out of %ld minLength ranges) by %ld:%ld\n",
474
               splitRange, rangeCount, count1, count2);
475
#endif
476
477
        // Next start = start + count1. First end = 1 before that.
478
869
        ranges[0].end = incWeightByOffset(start, minLength, count1 - 1);
479
869
        ranges[0].count = count1;
480
481
869
        ranges[1].start = incWeight(ranges[0].end, minLength);
482
869
        ranges[1].end = end;
483
869
        ranges[1].length = minLength;  // +1 when lengthened
484
869
        ranges[1].count = count2;  // *countBytes when lengthened
485
869
        lengthenRange(ranges[1]);
486
869
        rangeCount = 2;
487
869
    }
488
2.65k
    return true;
489
2.89k
}
490
491
/*
492
 * call getWeightRanges and then determine heuristically
493
 * which ranges to use for a given number of weights between (excluding)
494
 * two limits
495
 */
496
UBool
497
19.7k
CollationWeights::allocWeights(uint32_t lowerLimit, uint32_t upperLimit, int32_t n) {
498
#ifdef UCOL_DEBUG
499
    puts("");
500
#endif
501
502
19.7k
    if(!getWeightRanges(lowerLimit, upperLimit)) {
503
#ifdef UCOL_DEBUG
504
        printf("error: unable to get Weight ranges\n");
505
#endif
506
0
        return false;
507
0
    }
508
509
    /* try until we find suitably large ranges */
510
20.0k
    for(;;) {
511
        /* get the smallest number of bytes in a range */
512
20.0k
        int32_t minLength=ranges[0].length;
513
514
20.0k
        if(allocWeightsInShortRanges(n, minLength)) { break; }
515
516
2.91k
        if(minLength == 4) {
517
#ifdef UCOL_DEBUG
518
            printf("error: the maximum number of %ld weights is insufficient for n=%ld\n",
519
                   minLengthCount, n);
520
#endif
521
23
            return false;
522
23
        }
523
524
2.89k
        if(allocWeightsInMinLengthRanges(n, minLength)) { break; }
525
526
        /* no good match, lengthen all minLength ranges and iterate */
527
#ifdef UCOL_DEBUG
528
        printf("lengthen the short ranges from %ld bytes to %ld and iterate\n", minLength, minLength+1);
529
#endif
530
482
        for(int32_t i=0; i<rangeCount && ranges[i].length==minLength; ++i) {
531
241
            lengthenRange(ranges[i]);
532
241
        }
533
241
    }
534
535
#ifdef UCOL_DEBUG
536
    puts("final ranges:");
537
    for(int32_t i=0; i<rangeCount; ++i) {
538
        printf("ranges[%ld] .start=0x%08lx .end=0x%08lx .length=%ld .count=%ld\n",
539
               i, ranges[i].start, ranges[i].end, ranges[i].length, ranges[i].count);
540
    }
541
#endif
542
543
19.7k
    rangeIndex = 0;
544
19.7k
    return true;
545
19.7k
}
546
547
uint32_t
548
1.89M
CollationWeights::nextWeight() {
549
1.89M
    if(rangeIndex >= rangeCount) {
550
0
        return 0xffffffff;
551
1.89M
    } else {
552
        /* get the next weight */
553
1.89M
        WeightRange &range = ranges[rangeIndex];
554
1.89M
        uint32_t weight = range.start;
555
1.89M
        if(--range.count == 0) {
556
            /* this range is finished */
557
5.04k
            ++rangeIndex;
558
1.89M
        } else {
559
            /* increment the weight for the next value */
560
1.89M
            range.start = incWeight(weight, range.length);
561
1.89M
            U_ASSERT(range.start <= range.end);
562
1.89M
        }
563
564
1.89M
        return weight;
565
1.89M
    }
566
1.89M
}
567
568
U_NAMESPACE_END
569
570
#endif /* #if !UCONFIG_NO_COLLATION */