// © 2016 and later: Unicode, Inc. and others. // License & terms of use: http://www.unicode.org/copyright.html /* ******************************************************************************* * * Copyright (C) 2009-2014, International Business Machines * Corporation and others. All Rights Reserved. * ******************************************************************************* * file name: normalizer2impl.h * encoding: UTF-8 * tab size: 8 (not used) * indentation:4 * * created on: 2009nov22 * created by: Markus W. Scherer */ #ifndef __NORMALIZER2IMPL_H__ #define __NORMALIZER2IMPL_H__ #include "unicode/utypes.h" #if !UCONFIG_NO_NORMALIZATION #include "unicode/normalizer2.h" #include "unicode/ucptrie.h" #include "unicode/unistr.h" #include "unicode/unorm.h" #include "unicode/utf.h" #include "unicode/utf16.h" #include "mutex.h" #include "udataswp.h" #include "uset_imp.h" // When the nfc.nrm data is *not* hardcoded into the common library // (with this constant set to 0), // then it needs to be built into the data package: // Add nfc.nrm to icu4c/source/data/Makefile.in DAT_FILES_SHORT #define NORM2_HARDCODE_NFC_DATA … U_NAMESPACE_BEGIN struct CanonIterData; class ByteSink; class Edits; class InitCanonIterData; class LcccContext; class U_COMMON_API Hangul { … }; class Normalizer2Impl; class U_COMMON_API ReorderingBuffer : public UMemory { … }; /** * Low-level implementation of the Unicode Normalization Algorithm. * For the data structure and details see the documentation at the end of * this normalizer2impl.h and in the design doc at * https://unicode-org.github.io/icu/design/normalization/custom.html */ class U_COMMON_API Normalizer2Impl : public UObject { … }; // bits in canonIterData #define CANON_NOT_SEGMENT_STARTER … #define CANON_HAS_COMPOSITIONS … #define CANON_HAS_SET … #define CANON_VALUE_MASK … /** * ICU-internal shortcut for quick access to standard Unicode normalization. */ class U_COMMON_API Normalizer2Factory { … }; U_NAMESPACE_END U_CAPI int32_t U_EXPORT2 unorm2_swap(const UDataSwapper *ds, const void *inData, int32_t length, void *outData, UErrorCode *pErrorCode); /** * Get the NF*_QC property for a code point, for u_getIntPropertyValue(). * @internal */ U_CFUNC UNormalizationCheckResult unorm_getQuickCheck(UChar32 c, UNormalizationMode mode); /** * Gets the 16-bit FCD value (lead & trail CCs) for a code point, for u_getIntPropertyValue(). * @internal */ U_CFUNC uint16_t unorm_getFCD16(UChar32 c); /** * Format of Normalizer2 .nrm data files. * Format version 5.0. * * Normalizer2 .nrm data files provide data for the Unicode Normalization algorithms. * ICU ships with data files for standard Unicode Normalization Forms * NFC and NFD (nfc.nrm), NFKC and NFKD (nfkc.nrm), * NFKC_Casefold (nfkc_cf.nrm) and NFKC_Simple_Casefold (nfkc_scf.nrm). * Custom (application-specific) data can be built into additional .nrm files * with the gennorm2 build tool. * ICU ships with one such file, uts46.nrm, for the implementation of UTS #46. * * Normalizer2.getInstance() causes a .nrm file to be loaded, unless it has been * cached already. Internally, Normalizer2Impl.load() reads the .nrm file. * * A .nrm file begins with a standard ICU data file header * (DataHeader, see ucmndata.h and unicode/udata.h). * The UDataInfo.dataVersion field usually contains the Unicode version * for which the data was generated. * * After the header, the file contains the following parts. * Constants are defined as enum values of the Normalizer2Impl class. * * Many details of the data structures are described in the design doc * which is at https://unicode-org.github.io/icu/design/normalization/custom.html * * int32_t indexes[indexesLength]; -- indexesLength=indexes[IX_NORM_TRIE_OFFSET]/4; * * The first eight indexes are byte offsets in ascending order. * Each byte offset marks the start of the next part in the data file, * and the end of the previous one. * When two consecutive byte offsets are the same, then the corresponding part is empty. * Byte offsets are offsets from after the header, * that is, from the beginning of the indexes[]. * Each part starts at an offset with proper alignment for its data. * If necessary, the previous part may include padding bytes to achieve this alignment. * * minDecompNoCP=indexes[IX_MIN_DECOMP_NO_CP] is the lowest code point * with a decomposition mapping, that is, with NF*D_QC=No. * minCompNoMaybeCP=indexes[IX_MIN_COMP_NO_MAYBE_CP] is the lowest code point * with NF*C_QC=No (has a one-way mapping) or Maybe (combines backward). * minLcccCP=indexes[IX_MIN_LCCC_CP] (index 18, new in formatVersion 3) * is the lowest code point with lccc!=0. * * The next eight indexes are thresholds of 16-bit trie values for ranges of * values indicating multiple normalization properties. * Format version 5 adds the two minMaybeNo* threshold indexes. * The thresholds are listed here in threshold order, * not in the order they are stored in the indexes. * minYesNo=indexes[IX_MIN_YES_NO]; * minYesNoMappingsOnly=indexes[IX_MIN_YES_NO_MAPPINGS_ONLY]; * minNoNo=indexes[IX_MIN_NO_NO]; * minNoNoCompBoundaryBefore=indexes[IX_MIN_NO_NO_COMP_BOUNDARY_BEFORE]; * minNoNoCompNoMaybeCC=indexes[IX_MIN_NO_NO_COMP_NO_MAYBE_CC]; * minNoNoEmpty=indexes[IX_MIN_NO_NO_EMPTY]; * limitNoNo=indexes[IX_LIMIT_NO_NO]; * minMaybeNo=indexes[IX_MIN_MAYBE_NO]; * minMaybeNoCombinesFwd=indexes[IX_MIN_MAYBE_NO_COMBINES_FWD]; * minMaybeYes=indexes[IX_MIN_MAYBE_YES]; * See the normTrie description below and the design doc for details. * * UCPTrie normTrie; -- see ucptrie_impl.h and ucptrie.h, same as Java CodePointTrie * * The trie holds the main normalization data. Each code point is mapped to a 16-bit value. * Rather than using independent bits in the value (which would require more than 16 bits), * information is extracted primarily via range checks. * Except, format version 3+ uses bit 0 for hasCompBoundaryAfter(). * For example, a 16-bit value norm16 in the range minYesNo<=norm16<minNoNo * means that the character has NF*C_QC=Yes and NF*D_QC=No properties, * which means it has a two-way (round-trip) decomposition mapping. * Values in the ranges 2<=norm16<limitNoNo and minMaybeNo<=norm16<minMaybeYes * are also directly indexes into the extraData * pointing to mappings, compositions lists, or both. * Value norm16==INERT (0 in versions 1 & 2, 1 in version 3+) * means that the character is normalization-inert, that is, * it does not have a mapping, does not participate in composition, has a zero * canonical combining class, and forms a boundary where text before it and after it * can be normalized independently. * For details about how multiple properties are encoded in 16-bit values * see the design doc. * Note that the encoding cannot express all combinations of the properties involved; * it only supports those combinations that are allowed by * the Unicode Normalization algorithms. Details are in the design doc as well. * The gennorm2 tool only builds .nrm files for data that conforms to the limitations. * * The trie has a value for each lead surrogate code unit representing the "worst case" * properties of the 1024 supplementary characters whose UTF-16 form starts with * the lead surrogate. If all of the 1024 supplementary characters are normalization-inert, * then their lead surrogate code unit has the trie value INERT. * When the lead surrogate unit's value exceeds the quick check minimum during processing, * the properties for the full supplementary code point need to be looked up. * * uint16_t extraData[]; * * The extraData array contains many per-character data sections. * Each section contains mappings and/or composition lists. * The norm16 value of each character that has such data is directly an index to * a section of the extraData array. * * In version 3+, the norm16 values must be shifted right by OFFSET_SHIFT * for accessing extraData. * * The data structures for compositions lists and mappings are described in the design doc. * * In version 4 and below, the composition lists for MaybeYes characters were stored before * the data for other characters. * This sub-array had a length of MIN_NORMAL_MAYBE_YES-minMaybeYes. * In version 3 & 4, the difference must be shifted right by OFFSET_SHIFT. * * In version 5, the data for MaybeNo and MaybeYes characters is stored after * the data for other characters. * * If there are no MaybeNo and no MaybeYes characters, * then minMaybeYes==minMaybeNo==MIN_NORMAL_MAYBE_YES. * If there are such characters, then minMaybeNo is subtracted from their norm16 values * to get the index into the extraData. * In version 4 and below, the data index for Yes* and No* characters needs to be * offset by the length of the MaybeYes data. * In version 5, the data index for Maybe* characters needs to be offset by limitNoNo. * * Version 5 is the first to support MaybeNo characters, and * adds the minMaybeNo and minMaybeNoCombinesFwd thresholds and * the corresponding sections of the extraData. * * uint8_t smallFCD[0x100]; -- new in format version 2 * * This is a bit set to help speed up FCD value lookups in the absence of a full * UTrie2 or other large data structure with the full FCD value mapping. * * Each smallFCD bit is set if any of the corresponding 32 BMP code points * has a non-zero FCD value (lccc!=0 or tccc!=0). * Bit 0 of smallFCD[0] is for U+0000..U+001F. Bit 7 of smallFCD[0xff] is for U+FFE0..U+FFFF. * A bit for 32 lead surrogates is set if any of the 32k corresponding * _supplementary_ code points has a non-zero FCD value. * * This bit set is most useful for the large blocks of CJK characters with FCD=0. * * Changes from format version 1 to format version 2 --------------------------- * * - Addition of data for raw (not recursively decomposed) mappings. * + The MAPPING_NO_COMP_BOUNDARY_AFTER bit in the extraData is now also set when * the mapping is to an empty string or when the character combines-forward. * This subsumes the one actual use of the MAPPING_PLUS_COMPOSITION_LIST bit which * is then repurposed for the MAPPING_HAS_RAW_MAPPING bit. * + For details see the design doc. * - Addition of indexes[IX_MIN_YES_NO_MAPPINGS_ONLY] and separation of the yesNo extraData into * distinct ranges (combines-forward vs. not) * so that a range check can be used to find out if there is a compositions list. * This is fully equivalent with formatVersion 1's MAPPING_PLUS_COMPOSITION_LIST flag. * It is needed for the new (in ICU 49) composePair(), not for other normalization. * - Addition of the smallFCD[] bit set. * * Changes from format version 2 to format version 3 (ICU 60) ------------------ * * - norm16 bit 0 indicates hasCompBoundaryAfter(), * except that for contiguous composition (FCC) the tccc must be checked as well. * Data indexes and ccc values are shifted left by one (OFFSET_SHIFT). * Thresholds like minNoNo are tested before shifting. * * - Algorithmic mapping deltas are shifted left by two more bits (total DELTA_SHIFT), * to make room for two bits (three values) indicating whether the tccc is 0, 1, or greater. * See DELTA_TCCC_MASK etc. * This helps with fetching tccc/FCD values and FCC hasCompBoundaryAfter(). * minMaybeNo is 8-aligned so that the DELTA_TCCC_MASK bits can be tested directly. * * - Algorithmic mappings are only used for mapping to "comp yes and ccc=0" characters, * and ASCII characters are mapped algorithmically only to other ASCII characters. * This helps with hasCompBoundaryBefore() and compose() fast paths. * It is never necessary any more to loop for algorithmic mappings. * * - Addition of indexes[IX_MIN_NO_NO_COMP_BOUNDARY_BEFORE], * indexes[IX_MIN_NO_NO_COMP_NO_MAYBE_CC], and indexes[IX_MIN_NO_NO_EMPTY], * and separation of the noNo extraData into distinct ranges. * With this, the noNo norm16 value indicates whether the mapping is * compose-normalized, not normalized but hasCompBoundaryBefore(), * not even that, or maps to an empty string. * hasCompBoundaryBefore() can be determined solely from the norm16 value. * * - The norm16 value for Hangul LVT is now different from that for Hangul LV, * so that hasCompBoundaryAfter() need not check for the syllable type. * For Hangul LV, minYesNo continues to be used (no comp-boundary-after). * For Hangul LVT, minYesNoMappingsOnly|HAS_COMP_BOUNDARY_AFTER is used. * The extraData units at these indexes are set to firstUnit=2 and firstUnit=3, respectively, * to simplify some code. * * - The extraData firstUnit bit 5 is no longer necessary * (norm16 bit 0 used instead of firstUnit MAPPING_NO_COMP_BOUNDARY_AFTER), * is reserved again, and always set to 0. * * - Addition of indexes[IX_MIN_LCCC_CP], the first code point where lccc!=0. * This used to be hardcoded to U+0300, but in data like NFKC_Casefold it is lower: * U+00AD Soft Hyphen maps to an empty string, * which is artificially assigned "worst case" values lccc=1 and tccc=255. * * - A mapping to an empty string has explicit lccc=1 and tccc=255 values. * * Changes from format version 3 to format version 4 (ICU 63) ------------------ * * Switched from UTrie2 to UCPTrie/CodePointTrie. * * The new trie no longer stores different values for surrogate code *units* vs. * surrogate code *points*. * Lead surrogates still have values for optimized UTF-16 string processing. * When looking up code point properties, the code now checks for lead surrogates and * treats them as inert. * * gennorm2 now has to reject mappings for surrogate code points. * UTS #46 maps unpaired surrogates to U+FFFD in code rather than via its * custom normalization data file. * * Changes from format version 4 to format version 5 (ICU 76) ------------------ * * Unicode 16 adds the first MaybeYes characters which combine both backward and forward, * taking this formerly theoretical data structure into reality. * * Unicode 16 also adds the first characters that have two-way mappings whose first characters * combine backward. In order for normalization and the quick check to work properly, * these composite characters also must be marked as NFC_QC=Maybe, * corresponding to "combines back", although the composites themselves do not combine backward. * Format version 5 adds two new ranges between "algorithmic NoNo" and MaybeYes, * with thresholds minMaybeNo and minMaybeNoCombinesFwd, * and indexes[IX_MIN_MAYBE_NO] and indexes[IX_MIN_MAYBE_NO_COMBINES_FWD], * and corresponding mappings and composition lists in the extraData. * * Format version 5 moves the data for Maybe* characters from the start of the extraData array * to its end. */ #endif /* !UCONFIG_NO_NORMALIZATION */ #endif /* __NORMALIZER2IMPL_H__ */