]>
Commit | Line | Data |
---|---|---|
1 | // © 2016 and later: Unicode, Inc. and others. | |
2 | // License & terms of use: http://www.unicode.org/copyright.html | |
3 | /* | |
4 | ******************************************************************************* | |
5 | * Copyright (C) 2013-2015, International Business Machines | |
6 | * Corporation and others. All Rights Reserved. | |
7 | ******************************************************************************* | |
8 | * collationfastlatin.h | |
9 | * | |
10 | * created on: 2013aug09 | |
11 | * created by: Markus W. Scherer | |
12 | */ | |
13 | ||
14 | #ifndef __COLLATIONFASTLATIN_H__ | |
15 | #define __COLLATIONFASTLATIN_H__ | |
16 | ||
17 | #include "unicode/utypes.h" | |
18 | ||
19 | #if !UCONFIG_NO_COLLATION | |
20 | ||
21 | U_NAMESPACE_BEGIN | |
22 | ||
23 | struct CollationData; | |
24 | struct CollationSettings; | |
25 | ||
26 | class U_I18N_API CollationFastLatin /* all static */ { | |
27 | public: | |
28 | /** | |
29 | * Fast Latin format version (one byte 1..FF). | |
30 | * Must be incremented for any runtime-incompatible changes, | |
31 | * in particular, for changes to any of the following constants. | |
32 | * | |
33 | * When the major version number of the main data format changes, | |
34 | * we can reset this fast Latin version to 1. | |
35 | */ | |
36 | static const uint16_t VERSION = 2; | |
37 | ||
38 | static const int32_t LATIN_MAX = 0x17f; | |
39 | static const int32_t LATIN_LIMIT = LATIN_MAX + 1; | |
40 | ||
41 | static const int32_t LATIN_MAX_UTF8_LEAD = 0xc5; // UTF-8 lead byte of LATIN_MAX | |
42 | ||
43 | static const int32_t PUNCT_START = 0x2000; | |
44 | static const int32_t PUNCT_LIMIT = 0x2040; | |
45 | ||
46 | // excludes U+FFFE & U+FFFF | |
47 | static const int32_t NUM_FAST_CHARS = LATIN_LIMIT + (PUNCT_LIMIT - PUNCT_START); | |
48 | ||
49 | // Note on the supported weight ranges: | |
50 | // Analysis of UCA 6.3 and CLDR 23 non-search tailorings shows that | |
51 | // the CEs for characters in the above ranges, excluding expansions with length >2, | |
52 | // excluding contractions of >2 characters, and other restrictions | |
53 | // (see the builder's getCEsFromCE32()), | |
54 | // use at most about 150 primary weights, | |
55 | // where about 94 primary weights are possibly-variable (space/punct/symbol/currency), | |
56 | // at most 4 secondary before-common weights, | |
57 | // at most 4 secondary after-common weights, | |
58 | // at most 16 secondary high weights (in secondary CEs), and | |
59 | // at most 4 tertiary after-common weights. | |
60 | // The following ranges are designed to support slightly more weights than that. | |
61 | // (en_US_POSIX is unusual: It creates about 64 variable + 116 Latin primaries.) | |
62 | ||
63 | // Digits may use long primaries (preserving more short ones) | |
64 | // or short primaries (faster) without changing this data structure. | |
65 | // (If we supported numeric collation, then digits would have to have long primaries | |
66 | // so that special handling does not affect the fast path.) | |
67 | ||
68 | static const uint32_t SHORT_PRIMARY_MASK = 0xfc00; // bits 15..10 | |
69 | static const uint32_t INDEX_MASK = 0x3ff; // bits 9..0 for expansions & contractions | |
70 | static const uint32_t SECONDARY_MASK = 0x3e0; // bits 9..5 | |
71 | static const uint32_t CASE_MASK = 0x18; // bits 4..3 | |
72 | static const uint32_t LONG_PRIMARY_MASK = 0xfff8; // bits 15..3 | |
73 | static const uint32_t TERTIARY_MASK = 7; // bits 2..0 | |
74 | static const uint32_t CASE_AND_TERTIARY_MASK = CASE_MASK | TERTIARY_MASK; | |
75 | ||
76 | static const uint32_t TWO_SHORT_PRIMARIES_MASK = | |
77 | (SHORT_PRIMARY_MASK << 16) | SHORT_PRIMARY_MASK; // 0xfc00fc00 | |
78 | static const uint32_t TWO_LONG_PRIMARIES_MASK = | |
79 | (LONG_PRIMARY_MASK << 16) | LONG_PRIMARY_MASK; // 0xfff8fff8 | |
80 | static const uint32_t TWO_SECONDARIES_MASK = | |
81 | (SECONDARY_MASK << 16) | SECONDARY_MASK; // 0x3e003e0 | |
82 | static const uint32_t TWO_CASES_MASK = | |
83 | (CASE_MASK << 16) | CASE_MASK; // 0x180018 | |
84 | static const uint32_t TWO_TERTIARIES_MASK = | |
85 | (TERTIARY_MASK << 16) | TERTIARY_MASK; // 0x70007 | |
86 | ||
87 | /** | |
88 | * Contraction with one fast Latin character. | |
89 | * Use INDEX_MASK to find the start of the contraction list after the fixed table. | |
90 | * The first entry contains the default mapping. | |
91 | * Otherwise use CONTR_CHAR_MASK for the contraction character index | |
92 | * (in ascending order). | |
93 | * Use CONTR_LENGTH_SHIFT for the length of the entry | |
94 | * (1=BAIL_OUT, 2=one CE, 3=two CEs). | |
95 | * | |
96 | * Also, U+0000 maps to a contraction entry, so that the fast path need not | |
97 | * check for NUL termination. | |
98 | * It usually maps to a contraction list with only the completely ignorable default value. | |
99 | */ | |
100 | static const uint32_t CONTRACTION = 0x400; | |
101 | /** | |
102 | * An expansion encodes two CEs. | |
103 | * Use INDEX_MASK to find the pair of CEs after the fixed table. | |
104 | * | |
105 | * The higher a mini CE value, the easier it is to process. | |
106 | * For expansions and higher, no context needs to be considered. | |
107 | */ | |
108 | static const uint32_t EXPANSION = 0x800; | |
109 | /** | |
110 | * Encodes one CE with a long/low mini primary (there are 128). | |
111 | * All potentially-variable primaries must be in this range, | |
112 | * to make the short-primary path as fast as possible. | |
113 | */ | |
114 | static const uint32_t MIN_LONG = 0xc00; | |
115 | static const uint32_t LONG_INC = 8; | |
116 | static const uint32_t MAX_LONG = 0xff8; | |
117 | /** | |
118 | * Encodes one CE with a short/high primary (there are 60), | |
119 | * plus a secondary CE if the secondary weight is high. | |
120 | * Fast handling: At least all letter primaries should be in this range. | |
121 | */ | |
122 | static const uint32_t MIN_SHORT = 0x1000; | |
123 | static const uint32_t SHORT_INC = 0x400; | |
124 | /** The highest primary weight is reserved for U+FFFF. */ | |
125 | static const uint32_t MAX_SHORT = SHORT_PRIMARY_MASK; | |
126 | ||
127 | static const uint32_t MIN_SEC_BEFORE = 0; // must add SEC_OFFSET | |
128 | static const uint32_t SEC_INC = 0x20; | |
129 | static const uint32_t MAX_SEC_BEFORE = MIN_SEC_BEFORE + 4 * SEC_INC; // 5 before common | |
130 | static const uint32_t COMMON_SEC = MAX_SEC_BEFORE + SEC_INC; | |
131 | static const uint32_t MIN_SEC_AFTER = COMMON_SEC + SEC_INC; | |
132 | static const uint32_t MAX_SEC_AFTER = MIN_SEC_AFTER + 5 * SEC_INC; // 6 after common | |
133 | static const uint32_t MIN_SEC_HIGH = MAX_SEC_AFTER + SEC_INC; // 20 high secondaries | |
134 | static const uint32_t MAX_SEC_HIGH = SECONDARY_MASK; | |
135 | ||
136 | /** | |
137 | * Lookup: Add this offset to secondary weights, except for completely ignorable CEs. | |
138 | * Must be greater than any special value, e.g., MERGE_WEIGHT. | |
139 | * The exact value is not relevant for the format version. | |
140 | */ | |
141 | static const uint32_t SEC_OFFSET = SEC_INC; | |
142 | static const uint32_t COMMON_SEC_PLUS_OFFSET = COMMON_SEC + SEC_OFFSET; | |
143 | ||
144 | static const uint32_t TWO_SEC_OFFSETS = | |
145 | (SEC_OFFSET << 16) | SEC_OFFSET; // 0x200020 | |
146 | static const uint32_t TWO_COMMON_SEC_PLUS_OFFSET = | |
147 | (COMMON_SEC_PLUS_OFFSET << 16) | COMMON_SEC_PLUS_OFFSET; | |
148 | ||
149 | static const uint32_t LOWER_CASE = 8; // case bits include this offset | |
150 | static const uint32_t TWO_LOWER_CASES = (LOWER_CASE << 16) | LOWER_CASE; // 0x80008 | |
151 | ||
152 | static const uint32_t COMMON_TER = 0; // must add TER_OFFSET | |
153 | static const uint32_t MAX_TER_AFTER = 7; // 7 after common | |
154 | ||
155 | /** | |
156 | * Lookup: Add this offset to tertiary weights, except for completely ignorable CEs. | |
157 | * Must be greater than any special value, e.g., MERGE_WEIGHT. | |
158 | * Must be greater than case bits as well, so that with combined case+tertiary weights | |
159 | * plus the offset the tertiary bits does not spill over into the case bits. | |
160 | * The exact value is not relevant for the format version. | |
161 | */ | |
162 | static const uint32_t TER_OFFSET = SEC_OFFSET; | |
163 | static const uint32_t COMMON_TER_PLUS_OFFSET = COMMON_TER + TER_OFFSET; | |
164 | ||
165 | static const uint32_t TWO_TER_OFFSETS = (TER_OFFSET << 16) | TER_OFFSET; | |
166 | static const uint32_t TWO_COMMON_TER_PLUS_OFFSET = | |
167 | (COMMON_TER_PLUS_OFFSET << 16) | COMMON_TER_PLUS_OFFSET; | |
168 | ||
169 | static const uint32_t MERGE_WEIGHT = 3; | |
170 | static const uint32_t EOS = 2; // end of string | |
171 | static const uint32_t BAIL_OUT = 1; | |
172 | ||
173 | /** | |
174 | * Contraction result first word bits 8..0 contain the | |
175 | * second contraction character, as a char index 0..NUM_FAST_CHARS-1. | |
176 | * Each contraction list is terminated with a word containing CONTR_CHAR_MASK. | |
177 | */ | |
178 | static const uint32_t CONTR_CHAR_MASK = 0x1ff; | |
179 | /** | |
180 | * Contraction result first word bits 10..9 contain the result length: | |
181 | * 1=bail out, 2=one mini CE, 3=two mini CEs | |
182 | */ | |
183 | static const uint32_t CONTR_LENGTH_SHIFT = 9; | |
184 | ||
185 | /** | |
186 | * Comparison return value when the regular comparison must be used. | |
187 | * The exact value is not relevant for the format version. | |
188 | */ | |
189 | static const int32_t BAIL_OUT_RESULT = -2; | |
190 | ||
191 | static inline int32_t getCharIndex(UChar c) { | |
192 | if(c <= LATIN_MAX) { | |
193 | return c; | |
194 | } else if(PUNCT_START <= c && c < PUNCT_LIMIT) { | |
195 | return c - (PUNCT_START - LATIN_LIMIT); | |
196 | } else { | |
197 | // Not a fast Latin character. | |
198 | // Note: U+FFFE & U+FFFF are forbidden in tailorings | |
199 | // and thus do not occur in any contractions. | |
200 | return -1; | |
201 | } | |
202 | } | |
203 | ||
204 | /** | |
205 | * Computes the options value for the compare functions | |
206 | * and writes the precomputed primary weights. | |
207 | * Returns -1 if the Latin fastpath is not supported for the data and settings. | |
208 | * The capacity must be LATIN_LIMIT. | |
209 | */ | |
210 | static int32_t getOptions(const CollationData *data, const CollationSettings &settings, | |
211 | uint16_t *primaries, int32_t capacity); | |
212 | ||
213 | static int32_t compareUTF16(const uint16_t *table, const uint16_t *primaries, int32_t options, | |
214 | const UChar *left, int32_t leftLength, | |
215 | const UChar *right, int32_t rightLength); | |
216 | ||
217 | static int32_t compareUTF8(const uint16_t *table, const uint16_t *primaries, int32_t options, | |
218 | const uint8_t *left, int32_t leftLength, | |
219 | const uint8_t *right, int32_t rightLength); | |
220 | ||
221 | private: | |
222 | static uint32_t lookup(const uint16_t *table, UChar32 c); | |
223 | static uint32_t lookupUTF8(const uint16_t *table, UChar32 c, | |
224 | const uint8_t *s8, int32_t &sIndex, int32_t sLength); | |
225 | static uint32_t lookupUTF8Unsafe(const uint16_t *table, UChar32 c, | |
226 | const uint8_t *s8, int32_t &sIndex); | |
227 | ||
228 | static uint32_t nextPair(const uint16_t *table, UChar32 c, uint32_t ce, | |
229 | const UChar *s16, const uint8_t *s8, int32_t &sIndex, int32_t &sLength); | |
230 | ||
231 | static inline uint32_t getPrimaries(uint32_t variableTop, uint32_t pair) { | |
232 | uint32_t ce = pair & 0xffff; | |
233 | if(ce >= MIN_SHORT) { return pair & TWO_SHORT_PRIMARIES_MASK; } | |
234 | if(ce > variableTop) { return pair & TWO_LONG_PRIMARIES_MASK; } | |
235 | if(ce >= MIN_LONG) { return 0; } // variable | |
236 | return pair; // special mini CE | |
237 | } | |
238 | static inline uint32_t getSecondariesFromOneShortCE(uint32_t ce) { | |
239 | ce &= SECONDARY_MASK; | |
240 | if(ce < MIN_SEC_HIGH) { | |
241 | return ce + SEC_OFFSET; | |
242 | } else { | |
243 | return ((ce + SEC_OFFSET) << 16) | COMMON_SEC_PLUS_OFFSET; | |
244 | } | |
245 | } | |
246 | static uint32_t getSecondaries(uint32_t variableTop, uint32_t pair); | |
247 | static uint32_t getCases(uint32_t variableTop, UBool strengthIsPrimary, uint32_t pair); | |
248 | static uint32_t getTertiaries(uint32_t variableTop, UBool withCaseBits, uint32_t pair); | |
249 | static uint32_t getQuaternaries(uint32_t variableTop, uint32_t pair); | |
250 | ||
251 | private: | |
252 | CollationFastLatin(); // no constructor | |
253 | }; | |
254 | ||
255 | /* | |
256 | * Format of the CollationFastLatin data table. | |
257 | * CollationFastLatin::VERSION = 2. | |
258 | * | |
259 | * This table contains data for a Latin-text collation fastpath. | |
260 | * The data is stored as an array of uint16_t which contains the following parts. | |
261 | * | |
262 | * uint16_t -- version & header length | |
263 | * Bits 15..8: version, must match the VERSION | |
264 | * 7..0: length of the header | |
265 | * | |
266 | * uint16_t varTops[header length - 1] | |
267 | * Version 2: | |
268 | * varTops[m] is the highest CollationFastLatin long-primary weight | |
269 | * of supported maxVariable group m | |
270 | * (special reorder group space, punct, symbol, currency). | |
271 | * | |
272 | * Version 1: | |
273 | * Each of these values maps the variable top lead byte of a supported maxVariable group | |
274 | * to the highest CollationFastLatin long-primary weight. | |
275 | * The values are stored in ascending order. | |
276 | * Bits 15..7: max fast-Latin long-primary weight (bits 11..3 shifted left by 4 bits) | |
277 | * 6..0: regular primary lead byte | |
278 | * | |
279 | * uint16_t miniCEs[0x1c0] | |
280 | * A mini collation element for each character U+0000..U+017F and U+2000..U+203F. | |
281 | * Each value encodes one or two mini CEs (two are possible if the first one | |
282 | * has a short mini primary and the second one is a secondary CE, i.e., primary == 0), | |
283 | * or points to an expansion or to a contraction table. | |
284 | * U+0000 always has a contraction entry, | |
285 | * so that NUL-termination need not be tested in the fastpath. | |
286 | * If the collation elements for a character or contraction cannot be encoded in this format, | |
287 | * then the BAIL_OUT value is stored. | |
288 | * For details see the comments for the class constants. | |
289 | * | |
290 | * uint16_t expansions[variable length]; | |
291 | * Expansion mini CEs contain an offset relative to just after the miniCEs table. | |
292 | * An expansions contains exactly 2 mini CEs. | |
293 | * | |
294 | * uint16_t contractions[variable length]; | |
295 | * Contraction mini CEs contain an offset relative to just after the miniCEs table. | |
296 | * It points to a list of tuples which map from a contraction suffix character to a result. | |
297 | * First uint16_t of each tuple: | |
298 | * Bits 10..9: Length of the result (1..3), see comments on CONTR_LENGTH_SHIFT. | |
299 | * Bits 8..0: Contraction character, see comments on CONTR_CHAR_MASK. | |
300 | * This is followed by 0, 1, or 2 uint16_t according to the length. | |
301 | * Each list is terminated by an entry with CONTR_CHAR_MASK. | |
302 | * Each list starts with such an entry which also contains the default result | |
303 | * for when there is no contraction match. | |
304 | * | |
305 | * ----------------- | |
306 | * Changes for version 2 (ICU 55) | |
307 | * | |
308 | * Special reorder groups do not necessarily start on whole primary lead bytes any more. | |
309 | * Therefore, the varTops data has a new format: | |
310 | * Version 1 stored the lead bytes of the highest root primaries for | |
311 | * the maxVariable-supported special reorder groups. | |
312 | * Now the top 16 bits would need to be stored, | |
313 | * and it is simpler to store only the fast-Latin weights. | |
314 | */ | |
315 | ||
316 | U_NAMESPACE_END | |
317 | ||
318 | #endif // !UCONFIG_NO_COLLATION | |
319 | #endif // __COLLATIONFASTLATIN_H__ |