compact_str/
unicode_data.rs

1//! Adapted from
2//! <https://doc.rust-lang.org/nightly/src/core/unicode/unicode_data.rs.html>
3
4#[inline(always)]
5fn skip_search<const SOR: usize, const OFFSETS: usize>(
6    needle: u32,
7    short_offset_runs: &[u32; SOR],
8    offsets: &[u8; OFFSETS],
9) -> bool {
10    // Note that this *cannot* be past the end of the array, as the last
11    // element is greater than char::MAX (the largest possible needle).
12    //
13    // So, we cannot have found it (i.e. Ok(idx) + 1 != length) and the correct
14    // location cannot be past it, so Err(idx) != length either.
15    //
16    // This means that we can avoid bounds checking for the accesses below, too.
17    let last_idx =
18        match short_offset_runs.binary_search_by_key(&(needle << 11), |header| header << 11) {
19            Ok(idx) => idx + 1,
20            Err(idx) => idx,
21        };
22
23    let mut offset_idx = decode_length(short_offset_runs[last_idx]);
24    let length = if let Some(next) = short_offset_runs.get(last_idx + 1) {
25        decode_length(*next) - offset_idx
26    } else {
27        offsets.len() - offset_idx
28    };
29    let prev = last_idx
30        .checked_sub(1)
31        .map(|prev| decode_prefix_sum(short_offset_runs[prev]))
32        .unwrap_or(0);
33
34    let total = needle - prev;
35    let mut prefix_sum = 0;
36    for _ in 0..(length - 1) {
37        let offset = offsets[offset_idx];
38        prefix_sum += offset as u32;
39        if prefix_sum > total {
40            break;
41        }
42        offset_idx += 1;
43    }
44    offset_idx % 2 == 1
45}
46
47#[inline(always)]
48const fn decode_prefix_sum(short_offset_run_header: u32) -> u32 {
49    short_offset_run_header & ((1 << 21) - 1)
50}
51
52#[inline(always)]
53const fn decode_length(short_offset_run_header: u32) -> usize {
54    (short_offset_run_header >> 21) as usize
55}
56
57#[rustfmt::skip]
58pub mod case_ignorable {
59    static SHORT_OFFSET_RUNS: [u32; 35] = [
60        688, 44045149, 572528402, 576724925, 807414908, 878718981, 903913493, 929080568, 933275148,
61        937491230, 1138818560, 1147208189, 1210124160, 1222707713, 1235291428, 1260457643,
62        1264654383, 1499535675, 1507925040, 1566646003, 1629566000, 1650551536, 1658941263,
63        1671540720, 1688321181, 1700908800, 1709298023, 1717688832, 1738661888, 1763828398,
64        1797383403, 1805773008, 1809970171, 1819148289, 1824457200,
65    ];
66    static OFFSETS: [u8; 875] = [
67        39, 1, 6, 1, 11, 1, 35, 1, 1, 1, 71, 1, 4, 1, 1, 1, 4, 1, 2, 2, 0, 192, 4, 2, 4, 1, 9, 2,
68        1, 1, 251, 7, 207, 1, 5, 1, 49, 45, 1, 1, 1, 2, 1, 2, 1, 1, 44, 1, 11, 6, 10, 11, 1, 1, 35,
69        1, 10, 21, 16, 1, 101, 8, 1, 10, 1, 4, 33, 1, 1, 1, 30, 27, 91, 11, 58, 11, 4, 1, 2, 1, 24,
70        24, 43, 3, 44, 1, 7, 2, 6, 8, 41, 58, 55, 1, 1, 1, 4, 8, 4, 1, 3, 7, 10, 2, 13, 1, 15, 1,
71        58, 1, 4, 4, 8, 1, 20, 2, 26, 1, 2, 2, 57, 1, 4, 2, 4, 2, 2, 3, 3, 1, 30, 2, 3, 1, 11, 2,
72        57, 1, 4, 5, 1, 2, 4, 1, 20, 2, 22, 6, 1, 1, 58, 1, 2, 1, 1, 4, 8, 1, 7, 2, 11, 2, 30, 1,
73        61, 1, 12, 1, 50, 1, 3, 1, 55, 1, 1, 3, 5, 3, 1, 4, 7, 2, 11, 2, 29, 1, 58, 1, 2, 1, 6, 1,
74        5, 2, 20, 2, 28, 2, 57, 2, 4, 4, 8, 1, 20, 2, 29, 1, 72, 1, 7, 3, 1, 1, 90, 1, 2, 7, 11, 9,
75        98, 1, 2, 9, 9, 1, 1, 7, 73, 2, 27, 1, 1, 1, 1, 1, 55, 14, 1, 5, 1, 2, 5, 11, 1, 36, 9, 1,
76        102, 4, 1, 6, 1, 2, 2, 2, 25, 2, 4, 3, 16, 4, 13, 1, 2, 2, 6, 1, 15, 1, 94, 1, 0, 3, 0, 3,
77        29, 2, 30, 2, 30, 2, 64, 2, 1, 7, 8, 1, 2, 11, 3, 1, 5, 1, 45, 5, 51, 1, 65, 2, 34, 1, 118,
78        3, 4, 2, 9, 1, 6, 3, 219, 2, 2, 1, 58, 1, 1, 7, 1, 1, 1, 1, 2, 8, 6, 10, 2, 1, 39, 1, 8, 31,
79        49, 4, 48, 1, 1, 5, 1, 1, 5, 1, 40, 9, 12, 2, 32, 4, 2, 2, 1, 3, 56, 1, 1, 2, 3, 1, 1, 3,
80        58, 8, 2, 2, 64, 6, 82, 3, 1, 13, 1, 7, 4, 1, 6, 1, 3, 2, 50, 63, 13, 1, 34, 101, 0, 1, 1,
81        3, 11, 3, 13, 3, 13, 3, 13, 2, 12, 5, 8, 2, 10, 1, 2, 1, 2, 5, 49, 5, 1, 10, 1, 1, 13, 1,
82        16, 13, 51, 33, 0, 2, 113, 3, 125, 1, 15, 1, 96, 32, 47, 1, 0, 1, 36, 4, 3, 5, 5, 1, 93, 6,
83        93, 3, 0, 1, 0, 6, 0, 1, 98, 4, 1, 10, 1, 1, 28, 4, 80, 2, 14, 34, 78, 1, 23, 3, 103, 3, 3,
84        2, 8, 1, 3, 1, 4, 1, 25, 2, 5, 1, 151, 2, 26, 18, 13, 1, 38, 8, 25, 11, 46, 3, 48, 1, 2, 4,
85        2, 2, 17, 1, 21, 2, 66, 6, 2, 2, 2, 2, 12, 1, 8, 1, 35, 1, 11, 1, 51, 1, 1, 3, 2, 2, 5, 2,
86        1, 1, 27, 1, 14, 2, 5, 2, 1, 1, 100, 5, 9, 3, 121, 1, 2, 1, 4, 1, 0, 1, 147, 17, 0, 16, 3,
87        1, 12, 16, 34, 1, 2, 1, 169, 1, 7, 1, 6, 1, 11, 1, 35, 1, 1, 1, 47, 1, 45, 2, 67, 1, 21, 3,
88        0, 1, 226, 1, 149, 5, 0, 6, 1, 42, 1, 9, 0, 3, 1, 2, 5, 4, 40, 3, 4, 1, 165, 2, 0, 4, 0, 2,
89        80, 3, 70, 11, 49, 4, 123, 1, 54, 15, 41, 1, 2, 2, 10, 3, 49, 4, 2, 2, 2, 1, 4, 1, 10, 1,
90        50, 3, 36, 5, 1, 8, 62, 1, 12, 2, 52, 9, 10, 4, 2, 1, 95, 3, 2, 1, 1, 2, 6, 1, 2, 1, 157, 1,
91        3, 8, 21, 2, 57, 2, 3, 1, 37, 7, 3, 5, 195, 8, 2, 3, 1, 1, 23, 1, 84, 6, 1, 1, 4, 2, 1, 2,
92        238, 4, 6, 2, 1, 2, 27, 2, 85, 8, 2, 1, 1, 2, 106, 1, 1, 1, 2, 6, 1, 1, 101, 3, 2, 4, 1, 5,
93        0, 9, 1, 2, 0, 2, 1, 1, 4, 1, 144, 4, 2, 2, 4, 1, 32, 10, 40, 6, 2, 4, 8, 1, 9, 6, 2, 3, 46,
94        13, 1, 2, 0, 7, 1, 6, 1, 1, 82, 22, 2, 7, 1, 2, 1, 2, 122, 6, 3, 1, 1, 2, 1, 7, 1, 1, 72, 2,
95        3, 1, 1, 1, 0, 2, 11, 2, 52, 5, 5, 1, 1, 1, 0, 17, 6, 15, 0, 5, 59, 7, 9, 4, 0, 1, 63, 17,
96        64, 2, 1, 2, 0, 4, 1, 7, 1, 2, 0, 2, 1, 4, 0, 46, 2, 23, 0, 3, 9, 16, 2, 7, 30, 4, 148, 3,
97        0, 55, 4, 50, 8, 1, 14, 1, 22, 5, 1, 15, 0, 7, 1, 17, 2, 7, 1, 2, 1, 5, 5, 62, 33, 1, 160,
98        14, 0, 1, 61, 4, 0, 5, 0, 7, 109, 8, 0, 5, 0, 1, 30, 96, 128, 240, 0,
99    ];
100    #[inline(always)]
101    pub fn lookup(c: char) -> bool {
102        super::skip_search(
103            c as u32,
104            &SHORT_OFFSET_RUNS,
105            &OFFSETS,
106        )
107    }
108}
109
110#[rustfmt::skip]
111pub mod cased {
112    static SHORT_OFFSET_RUNS: [u32; 22] = [
113        4256, 115348384, 136322176, 144711446, 163587254, 320875520, 325101120, 350268208,
114        392231680, 404815649, 413205504, 421595008, 467733632, 484513952, 492924480, 497144832,
115        501339814, 578936576, 627171376, 639756544, 643952944, 649261450,
116    ];
117    static OFFSETS: [u8; 315] = [
118        65, 26, 6, 26, 47, 1, 10, 1, 4, 1, 5, 23, 1, 31, 1, 195, 1, 4, 4, 208, 1, 36, 7, 2, 30, 5,
119        96, 1, 42, 4, 2, 2, 2, 4, 1, 1, 6, 1, 1, 3, 1, 1, 1, 20, 1, 83, 1, 139, 8, 166, 1, 38, 9,
120        41, 0, 38, 1, 1, 5, 1, 2, 43, 1, 4, 0, 86, 2, 6, 0, 9, 7, 43, 2, 3, 64, 192, 64, 0, 2, 6, 2,
121        38, 2, 6, 2, 8, 1, 1, 1, 1, 1, 1, 1, 31, 2, 53, 1, 7, 1, 1, 3, 3, 1, 7, 3, 4, 2, 6, 4, 13,
122        5, 3, 1, 7, 116, 1, 13, 1, 16, 13, 101, 1, 4, 1, 2, 10, 1, 1, 3, 5, 6, 1, 1, 1, 1, 1, 1, 4,
123        1, 6, 4, 1, 2, 4, 5, 5, 4, 1, 17, 32, 3, 2, 0, 52, 0, 229, 6, 4, 3, 2, 12, 38, 1, 1, 5, 1,
124        0, 46, 18, 30, 132, 102, 3, 4, 1, 59, 5, 2, 1, 1, 1, 5, 24, 5, 1, 3, 0, 43, 1, 14, 6, 80, 0,
125        7, 12, 5, 0, 26, 6, 26, 0, 80, 96, 36, 4, 36, 116, 11, 1, 15, 1, 7, 1, 2, 1, 11, 1, 15, 1,
126        7, 1, 2, 0, 1, 2, 3, 1, 42, 1, 9, 0, 51, 13, 51, 0, 64, 0, 64, 0, 85, 1, 71, 1, 2, 2, 1, 2,
127        2, 2, 4, 1, 12, 1, 1, 1, 7, 1, 65, 1, 4, 2, 8, 1, 7, 1, 28, 1, 4, 1, 5, 1, 1, 3, 7, 1, 0, 2,
128        25, 1, 25, 1, 31, 1, 25, 1, 31, 1, 25, 1, 31, 1, 25, 1, 31, 1, 25, 1, 8, 0, 10, 1, 20, 6, 6,
129        0, 62, 0, 68, 0, 26, 6, 26, 6, 26, 0,
130    ];
131    #[inline(always)]
132    pub fn lookup(c: char) -> bool {
133        super::skip_search(
134            c as u32,
135            &SHORT_OFFSET_RUNS,
136            &OFFSETS,
137        )
138    }
139}