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