| 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-2014, International Business Machines | 
|---|
| 6 | *   Corporation and others.  All Rights Reserved. | 
|---|
| 7 | ******************************************************************************* | 
|---|
| 8 | *   file name:  bocsu.h | 
|---|
| 9 | *   encoding:   UTF-8 | 
|---|
| 10 | *   tab size:   8 (not used) | 
|---|
| 11 | *   indentation:4 | 
|---|
| 12 | * | 
|---|
| 13 | *   Author: Markus W. Scherer | 
|---|
| 14 | * | 
|---|
| 15 | *   Modification history: | 
|---|
| 16 | *   05/18/2001  weiv    Made into separate module | 
|---|
| 17 | */ | 
|---|
| 18 |  | 
|---|
| 19 | #ifndef BOCSU_H | 
|---|
| 20 | #define BOCSU_H | 
|---|
| 21 |  | 
|---|
| 22 | #include "unicode/utypes.h" | 
|---|
| 23 |  | 
|---|
| 24 | #if !UCONFIG_NO_COLLATION | 
|---|
| 25 |  | 
|---|
| 26 | U_NAMESPACE_BEGIN | 
|---|
| 27 |  | 
|---|
| 28 | class ByteSink; | 
|---|
| 29 |  | 
|---|
| 30 | U_NAMESPACE_END | 
|---|
| 31 |  | 
|---|
| 32 | /* | 
|---|
| 33 | * "BOCSU" | 
|---|
| 34 | * Binary Ordered Compression Scheme for Unicode | 
|---|
| 35 | * | 
|---|
| 36 | * Specific application: | 
|---|
| 37 | * | 
|---|
| 38 | * Encode a Unicode string for the identical level of a sort key. | 
|---|
| 39 | * Restrictions: | 
|---|
| 40 | * - byte stream (unsigned 8-bit bytes) | 
|---|
| 41 | * - lexical order of the identical-level run must be | 
|---|
| 42 | *   the same as code point order for the string | 
|---|
| 43 | * - avoid byte values 0, 1, 2 | 
|---|
| 44 | * | 
|---|
| 45 | * Method: Slope Detection | 
|---|
| 46 | * Remember the previous code point (initial 0). | 
|---|
| 47 | * For each cp in the string, encode the difference to the previous one. | 
|---|
| 48 | * | 
|---|
| 49 | * With a compact encoding of differences, this yields good results for | 
|---|
| 50 | * small scripts and UTF-like results otherwise. | 
|---|
| 51 | * | 
|---|
| 52 | * Encoding of differences: | 
|---|
| 53 | * - Similar to a UTF, encoding the length of the byte sequence in the lead bytes. | 
|---|
| 54 | * - Does not need to be friendly for decoding or random access | 
|---|
| 55 | *   (trail byte values may overlap with lead/single byte values). | 
|---|
| 56 | * - The signedness must be encoded as the most significant part. | 
|---|
| 57 | * | 
|---|
| 58 | * We encode differences with few bytes if their absolute values are small. | 
|---|
| 59 | * For correct ordering, we must treat the entire value range -10ffff..+10ffff | 
|---|
| 60 | * in ascending order, which forbids encoding the sign and the absolute value separately. | 
|---|
| 61 | * Instead, we split the lead byte range in the middle and encode non-negative values | 
|---|
| 62 | * going up and negative values going down. | 
|---|
| 63 | * | 
|---|
| 64 | * For very small absolute values, the difference is added to a middle byte value | 
|---|
| 65 | * for single-byte encoded differences. | 
|---|
| 66 | * For somewhat larger absolute values, the difference is divided by the number | 
|---|
| 67 | * of byte values available, the modulo is used for one trail byte, and the remainder | 
|---|
| 68 | * is added to a lead byte avoiding the single-byte range. | 
|---|
| 69 | * For large absolute values, the difference is similarly encoded in three bytes. | 
|---|
| 70 | * | 
|---|
| 71 | * This encoding does not use byte values 0, 1, 2, but uses all other byte values | 
|---|
| 72 | * for lead/single bytes so that the middle range of single bytes is as large | 
|---|
| 73 | * as possible. | 
|---|
| 74 | * Note that the lead byte ranges overlap some, but that the sequences as a whole | 
|---|
| 75 | * are well ordered. I.e., even if the lead byte is the same for sequences of different | 
|---|
| 76 | * lengths, the trail bytes establish correct order. | 
|---|
| 77 | * It would be possible to encode slightly larger ranges for each length (>1) by | 
|---|
| 78 | * subtracting the lower bound of the range. However, that would also slow down the | 
|---|
| 79 | * calculation. | 
|---|
| 80 | * | 
|---|
| 81 | * For the actual string encoding, an optimization moves the previous code point value | 
|---|
| 82 | * to the middle of its Unicode script block to minimize the differences in | 
|---|
| 83 | * same-script text runs. | 
|---|
| 84 | */ | 
|---|
| 85 |  | 
|---|
| 86 | /* Do not use byte values 0, 1, 2 because they are separators in sort keys. */ | 
|---|
| 87 | #define SLOPE_MIN           3 | 
|---|
| 88 | #define SLOPE_MAX           0xff | 
|---|
| 89 | #define SLOPE_MIDDLE        0x81 | 
|---|
| 90 |  | 
|---|
| 91 | #define SLOPE_TAIL_COUNT    (SLOPE_MAX-SLOPE_MIN+1) | 
|---|
| 92 |  | 
|---|
| 93 | #define SLOPE_MAX_BYTES     4 | 
|---|
| 94 |  | 
|---|
| 95 | /* | 
|---|
| 96 | * Number of lead bytes: | 
|---|
| 97 | * 1        middle byte for 0 | 
|---|
| 98 | * 2*80=160 single bytes for !=0 | 
|---|
| 99 | * 2*42=84  for double-byte values | 
|---|
| 100 | * 2*3=6    for 3-byte values | 
|---|
| 101 | * 2*1=2    for 4-byte values | 
|---|
| 102 | * | 
|---|
| 103 | * The sum must be <=SLOPE_TAIL_COUNT. | 
|---|
| 104 | * | 
|---|
| 105 | * Why these numbers? | 
|---|
| 106 | * - There should be >=128 single-byte values to cover 128-blocks | 
|---|
| 107 | *   with small scripts. | 
|---|
| 108 | * - There should be >=20902 single/double-byte values to cover Unihan. | 
|---|
| 109 | * - It helps CJK Extension B some if there are 3-byte values that cover | 
|---|
| 110 | *   the distance between them and Unihan. | 
|---|
| 111 | *   This also helps to jump among distant places in the BMP. | 
|---|
| 112 | * - Four-byte values are necessary to cover the rest of Unicode. | 
|---|
| 113 | * | 
|---|
| 114 | * Symmetrical lead byte counts are for convenience. | 
|---|
| 115 | * With an equal distribution of even and odd differences there is also | 
|---|
| 116 | * no advantage to asymmetrical lead byte counts. | 
|---|
| 117 | */ | 
|---|
| 118 | #define SLOPE_SINGLE        80 | 
|---|
| 119 | #define SLOPE_LEAD_2        42 | 
|---|
| 120 | #define SLOPE_LEAD_3        3 | 
|---|
| 121 | #define SLOPE_LEAD_4        1 | 
|---|
| 122 |  | 
|---|
| 123 | /* The difference value range for single-byters. */ | 
|---|
| 124 | #define SLOPE_REACH_POS_1   SLOPE_SINGLE | 
|---|
| 125 | #define SLOPE_REACH_NEG_1   (-SLOPE_SINGLE) | 
|---|
| 126 |  | 
|---|
| 127 | /* The difference value range for double-byters. */ | 
|---|
| 128 | #define SLOPE_REACH_POS_2   (SLOPE_LEAD_2*SLOPE_TAIL_COUNT+(SLOPE_LEAD_2-1)) | 
|---|
| 129 | #define SLOPE_REACH_NEG_2   (-SLOPE_REACH_POS_2-1) | 
|---|
| 130 |  | 
|---|
| 131 | /* The difference value range for 3-byters. */ | 
|---|
| 132 | #define SLOPE_REACH_POS_3   (SLOPE_LEAD_3*SLOPE_TAIL_COUNT*SLOPE_TAIL_COUNT+(SLOPE_LEAD_3-1)*SLOPE_TAIL_COUNT+(SLOPE_TAIL_COUNT-1)) | 
|---|
| 133 | #define SLOPE_REACH_NEG_3   (-SLOPE_REACH_POS_3-1) | 
|---|
| 134 |  | 
|---|
| 135 | /* The lead byte start values. */ | 
|---|
| 136 | #define SLOPE_START_POS_2   (SLOPE_MIDDLE+SLOPE_SINGLE+1) | 
|---|
| 137 | #define SLOPE_START_POS_3   (SLOPE_START_POS_2+SLOPE_LEAD_2) | 
|---|
| 138 |  | 
|---|
| 139 | #define SLOPE_START_NEG_2   (SLOPE_MIDDLE+SLOPE_REACH_NEG_1) | 
|---|
| 140 | #define SLOPE_START_NEG_3   (SLOPE_START_NEG_2-SLOPE_LEAD_2) | 
|---|
| 141 |  | 
|---|
| 142 | /* | 
|---|
| 143 | * Integer division and modulo with negative numerators | 
|---|
| 144 | * yields negative modulo results and quotients that are one more than | 
|---|
| 145 | * what we need here. | 
|---|
| 146 | */ | 
|---|
| 147 | #define NEGDIVMOD(n, d, m) UPRV_BLOCK_MACRO_BEGIN { \ | 
|---|
| 148 | (m)=(n)%(d); \ | 
|---|
| 149 | (n)/=(d); \ | 
|---|
| 150 | if((m)<0) { \ | 
|---|
| 151 | --(n); \ | 
|---|
| 152 | (m)+=(d); \ | 
|---|
| 153 | } \ | 
|---|
| 154 | } UPRV_BLOCK_MACRO_END | 
|---|
| 155 |  | 
|---|
| 156 | U_CFUNC UChar32 | 
|---|
| 157 | u_writeIdenticalLevelRun(UChar32 prev, const UChar *s, int32_t length, icu::ByteSink &sink); | 
|---|
| 158 |  | 
|---|
| 159 | #endif /* #if !UCONFIG_NO_COLLATION */ | 
|---|
| 160 |  | 
|---|
| 161 | #endif | 
|---|
| 162 |  | 
|---|