Bitcoin ABC 0.30.5
P2P Digital Currency
univalue_utffilter.h
Go to the documentation of this file.
1// Copyright 2016 Wladimir J. van der Laan
2// Distributed under the MIT software license, see the accompanying
3// file COPYING or https://opensource.org/licenses/mit-license.php.
4#ifndef BITCOIN_UNIVALUE_INCLUDE_UNIVALUE_UTFFILTER_H
5#define BITCOIN_UNIVALUE_INCLUDE_UNIVALUE_UTFFILTER_H
6
7#include <string>
8
14public:
15 explicit JSONUTF8StringFilter(std::string &s)
16 : str(s), is_valid(true), codepoint(0), state(0), surpair(0) {}
17 // Write single 8-bit char (may be part of UTF-8 sequence)
18 void push_back(uint8_t ch) {
19 if (state == 0) {
20 if (ch < 0x80) // 7-bit ASCII, fast direct pass-through
21 str.push_back(ch);
22 else if (ch < 0xc0) // Mid-sequence character, invalid in this state
23 is_valid = false;
24 else if (ch < 0xe0) { // Start of 2-byte sequence
25 codepoint = (ch & 0x1f) << 6;
26 state = 6;
27 } else if (ch < 0xf0) { // Start of 3-byte sequence
28 codepoint = (ch & 0x0f) << 12;
29 state = 12;
30 } else if (ch < 0xf8) { // Start of 4-byte sequence
31 codepoint = (ch & 0x07) << 18;
32 state = 18;
33 } else // Reserved, invalid
34 is_valid = false;
35 } else {
36 if ((ch & 0xc0) != 0x80) // Not a continuation, invalid
37 is_valid = false;
38 state -= 6;
39 codepoint |= (ch & 0x3f) << state;
40 if (state == 0) push_back_u(codepoint);
41 }
42 }
43 // Write codepoint directly, possibly collating surrogate pairs
44 void push_back_u(unsigned int codepoint_) {
45 if (state) // Only accept full codepoints in open state
46 is_valid = false;
47 if (codepoint_ >= 0xD800 &&
48 codepoint_ < 0xDC00) { // First half of surrogate pair
49 if (surpair) // Two subsequent surrogate pair openers - fail
50 is_valid = false;
51 else
52 surpair = codepoint_;
53 } else if (codepoint_ >= 0xDC00 &&
54 codepoint_ < 0xE000) { // Second half of surrogate pair
55 if (surpair) { // Open surrogate pair, expect second half
56 // Compute code point from UTF-16 surrogate pair
57 append_codepoint(0x10000 | ((surpair - 0xD800) << 10) |
58 (codepoint_ - 0xDC00));
59 surpair = 0;
60 } else // Second half doesn't follow a first half - fail
61 is_valid = false;
62 } else {
63 if (surpair) // First half of surrogate pair not followed by second
64 // - fail
65 is_valid = false;
66 else
67 append_codepoint(codepoint_);
68 }
69 }
70 // Check that we're in a state where the string can be ended
71 // No open sequences, no open surrogate pairs, etc
72 bool finalize() {
73 if (state || surpair) is_valid = false;
74 return is_valid;
75 }
76
77private:
78 std::string &str;
80 // Current UTF-8 decoding state
81 unsigned int codepoint;
82 int state; // Top bit to be filled in for next UTF-8 byte, or 0
83
84 // Keep track of the following state to handle the following section of
85 // RFC4627:
86 //
87 // To escape an extended character that is not in the Basic Multilingual
88 // Plane, the character is represented as a twelve-character sequence,
89 // encoding the UTF-16 surrogate pair. So, for example, a string
90 // containing only the G clef character (U+1D11E) may be represented as
91 // "\uD834\uDD1E".
92 //
93 // Two subsequent \u.... may have to be replaced with one actual codepoint.
94 unsigned int surpair; // First half of open UTF-16 surrogate pair, or 0
95
96 void append_codepoint(unsigned int codepoint_) {
97 if (codepoint_ <= 0x7f)
98 str.push_back((char)codepoint_);
99 else if (codepoint_ <= 0x7FF) {
100 str.push_back((char)(0xC0 | (codepoint_ >> 6)));
101 str.push_back((char)(0x80 | (codepoint_ & 0x3F)));
102 } else if (codepoint_ <= 0xFFFF) {
103 str.push_back((char)(0xE0 | (codepoint_ >> 12)));
104 str.push_back((char)(0x80 | ((codepoint_ >> 6) & 0x3F)));
105 str.push_back((char)(0x80 | (codepoint_ & 0x3F)));
106 } else if (codepoint_ <= 0x1FFFFF) {
107 str.push_back((char)(0xF0 | (codepoint_ >> 18)));
108 str.push_back((char)(0x80 | ((codepoint_ >> 12) & 0x3F)));
109 str.push_back((char)(0x80 | ((codepoint_ >> 6) & 0x3F)));
110 str.push_back((char)(0x80 | (codepoint_ & 0x3F)));
111 }
112 }
113};
114
115#endif // BITCOIN_UNIVALUE_INCLUDE_UNIVALUE_UTFFILTER_H
Filter that generates and validates UTF-8, as well as collates UTF-16 surrogate pairs as specified in...
void push_back_u(unsigned int codepoint_)
void append_codepoint(unsigned int codepoint_)
void push_back(uint8_t ch)
JSONUTF8StringFilter(std::string &s)