24 namespace json_details::string_quote {
25 template<std::
size_t N,
char c>
26 constexpr UInt8 test_at_byte( UInt64 b ) {
27 auto const lhs = b & ( 0xFF_u64 << ( N * 8U ) );
28 using rhs = daw::constant<to_uint64( static_cast<unsigned char>( c ) )
30 return to_uint8( not( lhs - rhs::value ) );
33 template<std::
size_t N,
char c>
34 DAW_ATTRIB_INLINE
constexpr UInt8 test_at_byte( UInt32 b ) {
35 auto const lhs = b & ( 0xFF_u32 << ( N * 8U ) );
36 using rhs = daw::constant<to_uint32( static_cast<unsigned char>( c ) )
38 return to_uint8( not( lhs - rhs::value ) );
41 constexpr void skip_to_first8( daw::not_null<char const *> &first,
42 daw::not_null<char const *>
const last ) {
43 bool keep_going = last - first >= 8;
45 auto buff = daw::to_uint64_buffer( first.get( ) );
46 auto const q7 = test_at_byte<7U,
'"'>( buff );
47 auto const q6 = test_at_byte<6U,
'"'>( buff );
48 auto const q5 = test_at_byte<5U,
'"'>( buff );
49 auto const q4 = test_at_byte<4U,
'"'>( buff );
50 auto const q3 = test_at_byte<3U,
'"'>( buff );
51 auto const q2 = test_at_byte<2U,
'"'>( buff );
52 auto const q1 = test_at_byte<1U,
'"'>( buff );
53 auto const q0 = test_at_byte<0U,
'"'>( buff );
54 auto const s7 = test_at_byte<7U, '\\'>( buff );
55 auto const s6 = test_at_byte<6U, '\\'>( buff );
56 auto const s5 = test_at_byte<5U, '\\'>( buff );
57 auto const s4 = test_at_byte<4U, '\\'>( buff );
58 auto const s3 = test_at_byte<3U, '\\'>( buff );
59 auto const s2 = test_at_byte<2U, '\\'>( buff );
60 auto const s1 = test_at_byte<1U, '\\'>( buff );
61 auto const s0 = test_at_byte<0U, '\\'>( buff );
63 keep_going = not( q0 | q1 | q2 | q3 | q4 | q5 | q6 | q7 | s0 | s1 |
64 s2 | s3 | s4 | s5 | s6 | s7 );
65 keep_going = keep_going &
static_cast<bool>( last - first >= 16 );
66 first +=
static_cast<int>( keep_going ) * 8;
68 first -= *( first - 1 ) ==
'\\' ? 1 : 0;
71 constexpr void skip_to_first4( daw::not_null<char const *> &first,
72 daw::not_null<char const *>
const last ) {
73 bool keep_going = last - first >= 4;
76 auto buff = daw::to_uint32_buffer( first.get( ) );
77 auto const q3 = test_at_byte<3U,
'"'>( buff );
78 auto const q2 = test_at_byte<2U,
'"'>( buff );
79 auto const q1 = test_at_byte<1U,
'"'>( buff );
80 auto const q0 = test_at_byte<0U,
'"'>( buff );
81 auto const s3 = test_at_byte<3U, '\\'>( buff );
82 auto const s2 = test_at_byte<2U, '\\'>( buff );
83 auto const s1 = test_at_byte<1U, '\\'>( buff );
84 auto const s0 = test_at_byte<0U, '\\'>( buff );
85 keep_going = not( q0 | q1 | q2 | q3 | s0 | s1 | s2 | s3 );
86 keep_going = keep_going &
static_cast<bool>( last - first >= 8 );
87 first +=
static_cast<int>( keep_going ) * 4;
89 first -= *( first - 1 ) ==
'\\' ? 1 : 0;
92 namespace string_quote_parser {
93 template<
typename ParseState>
94 [[nodiscard]]
static constexpr std::size_t
95 parse_nq_uncheck( ParseState &parse_state ) {
96 std::ptrdiff_t need_slow_path = -1;
98 daw::not_null<char const *>( daw::never_null, parse_state.first );
100 daw::not_null<char const *>( daw::never_null, parse_state.last );
104 if( json_details::use_constexpr_exec_mode<
105 typename ParseState::exec_tag_t>( ) ) {
107 mem_skip_until_end_of_string<
true,
108 typename ParseState::exec_tag_t>(
109 first, last, need_slow_path );
112 auto const sz = last - first;
114 skip_to_first8( first, last );
115 }
else if( sz >= 4 ) {
116 skip_to_first4( first, last );
119 while( *first !=
'"' ) {
121 return daw::nsc_and( c !=
'"', c !=
'\\' );
125 if( *first ==
'\\' ) {
126 if( need_slow_path < 0 ) {
127 need_slow_path = first - parse_state.first;
135 parse_state.first = first;
136 return static_cast<std::size_t
>( need_slow_path );
139 template<
typename ParseState>
140 [[nodiscard]]
static constexpr std::size_t
141 parse_nq_check( ParseState &parse_state ) {
143 std::ptrdiff_t need_slow_path = -1;
144 auto first = daw::not_null<char const *>( parse_state.first );
146 daw::not_null<char const *>( parse_state.class_last );
148 if( not json_details::use_constexpr_exec_mode<
149 typename ParseState::exec_tag_t>( ) ) {
151 mem_skip_until_end_of_string<
false,
152 typename ParseState::exec_tag_t>(
153 first, last, need_slow_path );
155 if constexpr( not ParseState::exclude_special_escapes ) {
157 daw::not_null<char const *>( parse_state.last );
159 skip_to_first8( first, l );
160 }
else if( last - first >= 4 ) {
161 skip_to_first4( first, l );
164 if constexpr( ParseState::is_zero_terminated_string ) {
165 if constexpr( ParseState::exclude_special_escapes ) {
166 while( *first !=
'\0' ) {
169 ErrorReason::InvalidString,
173 ErrorReason::InvalidString,
175 if( need_slow_path < 0 ) {
176 need_slow_path = first - parse_state.first;
194 }
else if( c ==
'"' ) {
200 while( daw::nsc_and( *first != 0, *first !=
'"' ) ) {
202 *first != 0, *first !=
'"', *first !=
'\\' ) ) {
206 if( daw::nsc_and( *first != 0, *first ==
'\\' ) ) {
207 if( need_slow_path < 0 ) {
208 need_slow_path = first - parse_state.first;
217 if constexpr( ParseState::exclude_special_escapes ) {
218 while( first < last ) {
221 ErrorReason::InvalidString,
225 ErrorReason::InvalidString,
227 if( need_slow_path < 0 ) {
228 need_slow_path = first - parse_state.first;
246 }
else if( c ==
'"' ) {
252 while( first < last and *first !=
'"' ) {
253 while( first < last and
254 daw::nsc_and( *first !=
'"', *first !=
'\\' ) ) {
258 if( first < last and *first ==
'\\' ) {
259 if( need_slow_path < 0 ) {
260 need_slow_path = first - parse_state.first;
270 if constexpr( ParseState::is_zero_terminated_string ) {
272 *first ==
'"', ErrorReason::InvalidString, parse_state );
275 ErrorReason::InvalidString,
278 parse_state.first = first;
279 return static_cast<std::size_t
>( need_slow_path );
282 template<
typename ParseState>
283 [[nodiscard]] DAW_ATTRIB_FLATTEN
static constexpr std::size_t
284 parse_nq( ParseState &parse_state ) {
285 if constexpr( ParseState::is_unchecked_input ) {
286 return parse_nq_uncheck( parse_state );
288 return parse_nq_check( parse_state );