SimulationCraft
SimulationCraft is a tool to explore combat mechanics in the popular MMO RPG World of Warcraft (tm).
writer.h
1 // Tencent is pleased to support the open source community by making RapidJSON available.
2 //
3 // Copyright (C) 2015 THL A29 Limited, a Tencent company, and Milo Yip.
4 //
5 // Licensed under the MIT License (the "License"); you may not use this file except
6 // in compliance with the License. You may obtain a copy of the License at
7 //
8 // http://opensource.org/licenses/MIT
9 //
10 // Unless required by applicable law or agreed to in writing, software distributed
11 // under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR
12 // CONDITIONS OF ANY KIND, either express or implied. See the License for the
13 // specific language governing permissions and limitations under the License.
14 
15 #ifndef RAPIDJSON_WRITER_H_
16 #define RAPIDJSON_WRITER_H_
17 
18 #include "stream.h"
19 #include "internal/clzll.h"
20 #include "internal/meta.h"
21 #include "internal/stack.h"
22 #include "internal/strfunc.h"
23 #include "internal/dtoa.h"
24 #include "internal/itoa.h"
25 #include "stringbuffer.h"
26 #include <new> // placement new
27 
28 #if defined(RAPIDJSON_SIMD) && defined(_MSC_VER)
29 #include <intrin.h>
30 #pragma intrinsic(_BitScanForward)
31 #endif
32 #ifdef RAPIDJSON_SSE42
33 #include <nmmintrin.h>
34 #elif defined(RAPIDJSON_SSE2)
35 #include <emmintrin.h>
36 #elif defined(RAPIDJSON_NEON)
37 #include <arm_neon.h>
38 #endif
39 
40 #ifdef __clang__
41 RAPIDJSON_DIAG_PUSH
42 RAPIDJSON_DIAG_OFF(padded)
43 RAPIDJSON_DIAG_OFF(unreachable-code)
44 RAPIDJSON_DIAG_OFF(c++98-compat)
45 #elif defined(_MSC_VER)
46 RAPIDJSON_DIAG_PUSH
47 RAPIDJSON_DIAG_OFF(4127) // conditional expression is constant
48 #endif
49 
51 
53 // WriteFlag
54 
61 #ifndef RAPIDJSON_WRITE_DEFAULT_FLAGS
62 #define RAPIDJSON_WRITE_DEFAULT_FLAGS kWriteNoFlags
63 #endif
64 
66 enum WriteFlag {
67  kWriteNoFlags = 0,
68  kWriteValidateEncodingFlag = 1,
69  kWriteNanAndInfFlag = 2,
70  kWriteDefaultFlags = RAPIDJSON_WRITE_DEFAULT_FLAGS
71 };
72 
74 
89 template<typename OutputStream, typename SourceEncoding = UTF8<>, typename TargetEncoding = UTF8<>, typename StackAllocator = CrtAllocator, unsigned writeFlags = kWriteDefaultFlags>
90 class Writer {
91 public:
92  typedef typename SourceEncoding::Ch Ch;
93 
94  static const int kDefaultMaxDecimalPlaces = 324;
95 
97 
101  explicit
102  Writer(OutputStream& os, StackAllocator* stackAllocator = 0, size_t levelDepth = kDefaultLevelDepth) :
103  os_(&os), level_stack_(stackAllocator, levelDepth * sizeof(Level)), maxDecimalPlaces_(kDefaultMaxDecimalPlaces), hasRoot_(false) {}
104 
105  explicit
106  Writer(StackAllocator* allocator = 0, size_t levelDepth = kDefaultLevelDepth) :
107  os_(0), level_stack_(allocator, levelDepth * sizeof(Level)), maxDecimalPlaces_(kDefaultMaxDecimalPlaces), hasRoot_(false) {}
108 
109 #if RAPIDJSON_HAS_CXX11_RVALUE_REFS
110  Writer(Writer&& rhs) :
111  os_(rhs.os_), level_stack_(std::move(rhs.level_stack_)), maxDecimalPlaces_(rhs.maxDecimalPlaces_), hasRoot_(rhs.hasRoot_) {
112  rhs.os_ = 0;
113  }
114 #endif
115 
117 
134  void Reset(OutputStream& os) {
135  os_ = &os;
136  hasRoot_ = false;
137  level_stack_.Clear();
138  }
139 
141 
144  bool IsComplete() const {
145  return hasRoot_ && level_stack_.Empty();
146  }
147 
148  int GetMaxDecimalPlaces() const {
149  return maxDecimalPlaces_;
150  }
151 
153 
173  void SetMaxDecimalPlaces(int maxDecimalPlaces) {
174  maxDecimalPlaces_ = maxDecimalPlaces;
175  }
176 
181 
182  bool Null() { Prefix(kNullType); return EndValue(WriteNull()); }
183  bool Bool(bool b) { Prefix(b ? kTrueType : kFalseType); return EndValue(WriteBool(b)); }
184  bool Int(int i) { Prefix(kNumberType); return EndValue(WriteInt(i)); }
185  bool Uint(unsigned u) { Prefix(kNumberType); return EndValue(WriteUint(u)); }
186  bool Int64(int64_t i64) { Prefix(kNumberType); return EndValue(WriteInt64(i64)); }
187  bool Uint64(uint64_t u64) { Prefix(kNumberType); return EndValue(WriteUint64(u64)); }
188 
190 
194  bool Double(double d) { Prefix(kNumberType); return EndValue(WriteDouble(d)); }
195 
196  bool RawNumber(const Ch* str, SizeType length, bool copy = false) {
197  RAPIDJSON_ASSERT(str != 0);
198  (void)copy;
199  Prefix(kNumberType);
200  return EndValue(WriteString(str, length));
201  }
202 
203  bool String(const Ch* str, SizeType length, bool copy = false, bool raw = false) {
204  RAPIDJSON_ASSERT(str != 0);
205  (void)copy;
206  Prefix(kStringType);
207  return EndValue(WriteString(str, length, raw));
208  }
209 
210 #if RAPIDJSON_HAS_STDSTRING
211  bool String(const std::basic_string<Ch>& str) {
212  return String(str.data(), SizeType(str.size()));
213  }
214 #endif
215 
216  bool StartObject() {
217  Prefix(kObjectType);
218  new (level_stack_.template Push<Level>()) Level(false);
219  return WriteStartObject();
220  }
221 
222  bool Key(const Ch* str, SizeType length, bool copy = false) { return String(str, length, copy); }
223 
224 #if RAPIDJSON_HAS_STDSTRING
225  bool Key(const std::basic_string<Ch>& str)
226  {
227  return Key(str.data(), SizeType(str.size()));
228  }
229 #endif
230 
231  bool EndObject(SizeType memberCount = 0) {
232  (void)memberCount;
233  RAPIDJSON_ASSERT(level_stack_.GetSize() >= sizeof(Level)); // not inside an Object
234  RAPIDJSON_ASSERT(!level_stack_.template Top<Level>()->inArray); // currently inside an Array, not Object
235  RAPIDJSON_ASSERT(0 == level_stack_.template Top<Level>()->valueCount % 2); // Object has a Key without a Value
236  level_stack_.template Pop<Level>(1);
237  return EndValue(WriteEndObject());
238  }
239 
240  bool StartArray() {
241  Prefix(kArrayType);
242  new (level_stack_.template Push<Level>()) Level(true);
243  return WriteStartArray();
244  }
245 
246  bool EndArray(SizeType elementCount = 0) {
247  (void)elementCount;
248  RAPIDJSON_ASSERT(level_stack_.GetSize() >= sizeof(Level));
249  RAPIDJSON_ASSERT(level_stack_.template Top<Level>()->inArray);
250  level_stack_.template Pop<Level>(1);
251  return EndValue(WriteEndArray());
252  }
254 
257 
259  bool String(const Ch* const& str) { return String(str, internal::StrLen(str)); }
260  bool Key(const Ch* const& str) { return Key(str, internal::StrLen(str)); }
261 
263 
265 
272  bool RawValue(const Ch* json, size_t length, Type type) {
273  RAPIDJSON_ASSERT(json != 0);
274  Prefix(type);
275  return EndValue(WriteRawValue(json, length));
276  }
277 
279 
282  void Flush() {
283  os_->Flush();
284  }
285 
286  static const size_t kDefaultLevelDepth = 32;
287 
288 protected:
290  struct Level {
291  Level(bool inArray_) : valueCount(0), inArray(inArray_) {}
292  size_t valueCount;
293  bool inArray;
294  };
295 
296  bool WriteNull() {
297  PutReserve(*os_, 4);
298  PutUnsafe(*os_, 'n'); PutUnsafe(*os_, 'u'); PutUnsafe(*os_, 'l'); PutUnsafe(*os_, 'l'); return true;
299  }
300 
301  bool WriteBool(bool b) {
302  if (b) {
303  PutReserve(*os_, 4);
304  PutUnsafe(*os_, 't'); PutUnsafe(*os_, 'r'); PutUnsafe(*os_, 'u'); PutUnsafe(*os_, 'e');
305  }
306  else {
307  PutReserve(*os_, 5);
308  PutUnsafe(*os_, 'f'); PutUnsafe(*os_, 'a'); PutUnsafe(*os_, 'l'); PutUnsafe(*os_, 's'); PutUnsafe(*os_, 'e');
309  }
310  return true;
311  }
312 
313  bool WriteInt(int i) {
314  char buffer[11];
315  const char* end = internal::i32toa(i, buffer);
316  PutReserve(*os_, static_cast<size_t>(end - buffer));
317  for (const char* p = buffer; p != end; ++p)
318  PutUnsafe(*os_, static_cast<typename OutputStream::Ch>(*p));
319  return true;
320  }
321 
322  bool WriteUint(unsigned u) {
323  char buffer[10];
324  const char* end = internal::u32toa(u, buffer);
325  PutReserve(*os_, static_cast<size_t>(end - buffer));
326  for (const char* p = buffer; p != end; ++p)
327  PutUnsafe(*os_, static_cast<typename OutputStream::Ch>(*p));
328  return true;
329  }
330 
331  bool WriteInt64(int64_t i64) {
332  char buffer[21];
333  const char* end = internal::i64toa(i64, buffer);
334  PutReserve(*os_, static_cast<size_t>(end - buffer));
335  for (const char* p = buffer; p != end; ++p)
336  PutUnsafe(*os_, static_cast<typename OutputStream::Ch>(*p));
337  return true;
338  }
339 
340  bool WriteUint64(uint64_t u64) {
341  char buffer[20];
342  char* end = internal::u64toa(u64, buffer);
343  PutReserve(*os_, static_cast<size_t>(end - buffer));
344  for (char* p = buffer; p != end; ++p)
345  PutUnsafe(*os_, static_cast<typename OutputStream::Ch>(*p));
346  return true;
347  }
348 
349  bool WriteDouble(double d) {
350  if (internal::Double(d).IsNanOrInf()) {
351  if (!(writeFlags & kWriteNanAndInfFlag))
352  return false;
353  if (internal::Double(d).IsNan()) {
354  PutReserve(*os_, 3);
355  PutUnsafe(*os_, 'N'); PutUnsafe(*os_, 'a'); PutUnsafe(*os_, 'N');
356  return true;
357  }
358  if (internal::Double(d).Sign()) {
359  PutReserve(*os_, 9);
360  PutUnsafe(*os_, '-');
361  }
362  else
363  PutReserve(*os_, 8);
364  PutUnsafe(*os_, 'I'); PutUnsafe(*os_, 'n'); PutUnsafe(*os_, 'f');
365  PutUnsafe(*os_, 'i'); PutUnsafe(*os_, 'n'); PutUnsafe(*os_, 'i'); PutUnsafe(*os_, 't'); PutUnsafe(*os_, 'y');
366  return true;
367  }
368 
369  char buffer[25];
370  char* end = internal::dtoa(d, buffer, maxDecimalPlaces_);
371  PutReserve(*os_, static_cast<size_t>(end - buffer));
372  for (char* p = buffer; p != end; ++p)
373  PutUnsafe(*os_, static_cast<typename OutputStream::Ch>(*p));
374  return true;
375  }
376 
377  bool WriteString(const Ch* str, SizeType length, bool raw) {
378  static const typename OutputStream::Ch hexDigits[16] = { '0', '1', '2', '3', '4', '5', '6', '7', '8', '9', 'A', 'B', 'C', 'D', 'E', 'F' };
379  static const char escape[256] = {
380 #define Z16 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0
381  //0 1 2 3 4 5 6 7 8 9 A B C D E F
382  'u', 'u', 'u', 'u', 'u', 'u', 'u', 'u', 'b', 't', 'n', 'u', 'f', 'r', 'u', 'u', // 00
383  'u', 'u', 'u', 'u', 'u', 'u', 'u', 'u', 'u', 'u', 'u', 'u', 'u', 'u', 'u', 'u', // 10
384  0, 0, '"', 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, // 20
385  Z16, Z16, // 30~4F
386  0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,'\\', 0, 0, 0, // 50
387  Z16, Z16, Z16, Z16, Z16, Z16, Z16, Z16, Z16, Z16 // 60~FF
388 #undef Z16
389  };
390 
391  if (TargetEncoding::supportUnicode)
392  PutReserve(*os_, 2 + length * 6); // "\uxxxx..."
393  else
394  PutReserve(*os_, 2 + length * 12); // "\uxxxx\uyyyy..."
395 
396  if (!raw)
397  {
398  PutUnsafe(*os_, '\"');
399  }
401  while (ScanWriteUnescapedString(is, length)) {
402  const Ch c = is.Peek();
403  if (!TargetEncoding::supportUnicode && static_cast<unsigned>(c) >= 0x80) {
404  // Unicode escaping
405  unsigned codepoint;
406  if (RAPIDJSON_UNLIKELY(!SourceEncoding::Decode(is, &codepoint)))
407  return false;
408  PutUnsafe(*os_, '\\');
409  PutUnsafe(*os_, 'u');
410  if (codepoint <= 0xD7FF || (codepoint >= 0xE000 && codepoint <= 0xFFFF)) {
411  PutUnsafe(*os_, hexDigits[(codepoint >> 12) & 15]);
412  PutUnsafe(*os_, hexDigits[(codepoint >> 8) & 15]);
413  PutUnsafe(*os_, hexDigits[(codepoint >> 4) & 15]);
414  PutUnsafe(*os_, hexDigits[(codepoint ) & 15]);
415  }
416  else {
417  RAPIDJSON_ASSERT(codepoint >= 0x010000 && codepoint <= 0x10FFFF);
418  // Surrogate pair
419  unsigned s = codepoint - 0x010000;
420  unsigned lead = (s >> 10) + 0xD800;
421  unsigned trail = (s & 0x3FF) + 0xDC00;
422  PutUnsafe(*os_, hexDigits[(lead >> 12) & 15]);
423  PutUnsafe(*os_, hexDigits[(lead >> 8) & 15]);
424  PutUnsafe(*os_, hexDigits[(lead >> 4) & 15]);
425  PutUnsafe(*os_, hexDigits[(lead ) & 15]);
426  PutUnsafe(*os_, '\\');
427  PutUnsafe(*os_, 'u');
428  PutUnsafe(*os_, hexDigits[(trail >> 12) & 15]);
429  PutUnsafe(*os_, hexDigits[(trail >> 8) & 15]);
430  PutUnsafe(*os_, hexDigits[(trail >> 4) & 15]);
431  PutUnsafe(*os_, hexDigits[(trail ) & 15]);
432  }
433  }
434  else if ((sizeof(Ch) == 1 || static_cast<unsigned>(c) < 256) && RAPIDJSON_UNLIKELY(escape[static_cast<unsigned char>(c)])) {
435  is.Take();
436  PutUnsafe(*os_, '\\');
437  PutUnsafe(*os_, static_cast<typename OutputStream::Ch>(escape[static_cast<unsigned char>(c)]));
438  if (escape[static_cast<unsigned char>(c)] == 'u') {
439  PutUnsafe(*os_, '0');
440  PutUnsafe(*os_, '0');
441  PutUnsafe(*os_, hexDigits[static_cast<unsigned char>(c) >> 4]);
442  PutUnsafe(*os_, hexDigits[static_cast<unsigned char>(c) & 0xF]);
443  }
444  }
445  else if (RAPIDJSON_UNLIKELY(!(writeFlags & kWriteValidateEncodingFlag ?
448  return false;
449  }
450  if (!raw)
451  {
452  PutUnsafe(*os_, '\"');
453  }
454  return true;
455  }
456 
457  bool ScanWriteUnescapedString(GenericStringStream<SourceEncoding>& is, size_t length) {
458  return RAPIDJSON_LIKELY(is.Tell() < length);
459  }
460 
461  bool WriteStartObject() { os_->Put('{'); return true; }
462  bool WriteEndObject() { os_->Put('}'); return true; }
463  bool WriteStartArray() { os_->Put('['); return true; }
464  bool WriteEndArray() { os_->Put(']'); return true; }
465 
466  bool WriteRawValue(const Ch* json, size_t length) {
467  PutReserve(*os_, length);
469  while (RAPIDJSON_LIKELY(is.Tell() < length)) {
470  RAPIDJSON_ASSERT(is.Peek() != '\0');
471  if (RAPIDJSON_UNLIKELY(!(writeFlags & kWriteValidateEncodingFlag ?
474  return false;
475  }
476  return true;
477  }
478 
479  void Prefix(Type type) {
480  (void)type;
481  if (RAPIDJSON_LIKELY(level_stack_.GetSize() != 0)) { // this value is not at root
482  Level* level = level_stack_.template Top<Level>();
483  if (level->valueCount > 0) {
484  if (level->inArray)
485  os_->Put(','); // add comma if it is not the first element in array
486  else // in object
487  os_->Put((level->valueCount % 2 == 0) ? ',' : ':');
488  }
489  if (!level->inArray && level->valueCount % 2 == 0)
490  RAPIDJSON_ASSERT(type == kStringType); // if it's in object, then even number should be a name
491  level->valueCount++;
492  }
493  else {
494  RAPIDJSON_ASSERT(!hasRoot_); // Should only has one and only one root.
495  hasRoot_ = true;
496  }
497  }
498 
499  // Flush the value if it is the top level one.
500  bool EndValue(bool ret) {
501  if (RAPIDJSON_UNLIKELY(level_stack_.Empty())) // end of json text
502  Flush();
503  return ret;
504  }
505 
506  OutputStream* os_;
507  internal::Stack<StackAllocator> level_stack_;
508  int maxDecimalPlaces_;
509  bool hasRoot_;
510 
511 private:
512  // Prohibit copy constructor & assignment operator.
513  Writer(const Writer&);
514  Writer& operator=(const Writer&);
515 };
516 
517 // Full specialization for StringStream to prevent memory copying
518 
519 template<>
520 inline bool Writer<StringBuffer>::WriteInt(int i) {
521  char *buffer = os_->Push(11);
522  const char* end = internal::i32toa(i, buffer);
523  os_->Pop(static_cast<size_t>(11 - (end - buffer)));
524  return true;
525 }
526 
527 template<>
528 inline bool Writer<StringBuffer>::WriteUint(unsigned u) {
529  char *buffer = os_->Push(10);
530  const char* end = internal::u32toa(u, buffer);
531  os_->Pop(static_cast<size_t>(10 - (end - buffer)));
532  return true;
533 }
534 
535 template<>
536 inline bool Writer<StringBuffer>::WriteInt64(int64_t i64) {
537  char *buffer = os_->Push(21);
538  const char* end = internal::i64toa(i64, buffer);
539  os_->Pop(static_cast<size_t>(21 - (end - buffer)));
540  return true;
541 }
542 
543 template<>
544 inline bool Writer<StringBuffer>::WriteUint64(uint64_t u) {
545  char *buffer = os_->Push(20);
546  const char* end = internal::u64toa(u, buffer);
547  os_->Pop(static_cast<size_t>(20 - (end - buffer)));
548  return true;
549 }
550 
551 template<>
552 inline bool Writer<StringBuffer>::WriteDouble(double d) {
553  if (internal::Double(d).IsNanOrInf()) {
554  // Note: This code path can only be reached if (RAPIDJSON_WRITE_DEFAULT_FLAGS & kWriteNanAndInfFlag).
555  if (!(kWriteDefaultFlags & kWriteNanAndInfFlag))
556  return false;
557  if (internal::Double(d).IsNan()) {
558  PutReserve(*os_, 3);
559  PutUnsafe(*os_, 'N'); PutUnsafe(*os_, 'a'); PutUnsafe(*os_, 'N');
560  return true;
561  }
562  if (internal::Double(d).Sign()) {
563  PutReserve(*os_, 9);
564  PutUnsafe(*os_, '-');
565  }
566  else
567  PutReserve(*os_, 8);
568  PutUnsafe(*os_, 'I'); PutUnsafe(*os_, 'n'); PutUnsafe(*os_, 'f');
569  PutUnsafe(*os_, 'i'); PutUnsafe(*os_, 'n'); PutUnsafe(*os_, 'i'); PutUnsafe(*os_, 't'); PutUnsafe(*os_, 'y');
570  return true;
571  }
572 
573  char *buffer = os_->Push(25);
574  char* end = internal::dtoa(d, buffer, maxDecimalPlaces_);
575  os_->Pop(static_cast<size_t>(25 - (end - buffer)));
576  return true;
577 }
578 
579 #if defined(RAPIDJSON_SSE2) || defined(RAPIDJSON_SSE42)
580 template<>
581 inline bool Writer<StringBuffer>::ScanWriteUnescapedString(StringStream& is, size_t length) {
582  if (length < 16)
583  return RAPIDJSON_LIKELY(is.Tell() < length);
584 
585  if (!RAPIDJSON_LIKELY(is.Tell() < length))
586  return false;
587 
588  const char* p = is.src_;
589  const char* end = is.head_ + length;
590  const char* nextAligned = reinterpret_cast<const char*>((reinterpret_cast<size_t>(p) + 15) & static_cast<size_t>(~15));
591  const char* endAligned = reinterpret_cast<const char*>(reinterpret_cast<size_t>(end) & static_cast<size_t>(~15));
592  if (nextAligned > end)
593  return true;
594 
595  while (p != nextAligned)
596  if (*p < 0x20 || *p == '\"' || *p == '\\') {
597  is.src_ = p;
598  return RAPIDJSON_LIKELY(is.Tell() < length);
599  }
600  else
601  os_->PutUnsafe(*p++);
602 
603  // The rest of string using SIMD
604  static const char dquote[16] = { '\"', '\"', '\"', '\"', '\"', '\"', '\"', '\"', '\"', '\"', '\"', '\"', '\"', '\"', '\"', '\"' };
605  static const char bslash[16] = { '\\', '\\', '\\', '\\', '\\', '\\', '\\', '\\', '\\', '\\', '\\', '\\', '\\', '\\', '\\', '\\' };
606  static const char space[16] = { 0x1F, 0x1F, 0x1F, 0x1F, 0x1F, 0x1F, 0x1F, 0x1F, 0x1F, 0x1F, 0x1F, 0x1F, 0x1F, 0x1F, 0x1F, 0x1F };
607  const __m128i dq = _mm_loadu_si128(reinterpret_cast<const __m128i *>(&dquote[0]));
608  const __m128i bs = _mm_loadu_si128(reinterpret_cast<const __m128i *>(&bslash[0]));
609  const __m128i sp = _mm_loadu_si128(reinterpret_cast<const __m128i *>(&space[0]));
610 
611  for (; p != endAligned; p += 16) {
612  const __m128i s = _mm_load_si128(reinterpret_cast<const __m128i *>(p));
613  const __m128i t1 = _mm_cmpeq_epi8(s, dq);
614  const __m128i t2 = _mm_cmpeq_epi8(s, bs);
615  const __m128i t3 = _mm_cmpeq_epi8(_mm_max_epu8(s, sp), sp); // s < 0x20 <=> max(s, 0x1F) == 0x1F
616  const __m128i x = _mm_or_si128(_mm_or_si128(t1, t2), t3);
617  unsigned short r = static_cast<unsigned short>(_mm_movemask_epi8(x));
618  if (RAPIDJSON_UNLIKELY(r != 0)) { // some of characters is escaped
619  SizeType len;
620 #ifdef _MSC_VER // Find the index of first escaped
621  unsigned long offset;
622  _BitScanForward(&offset, r);
623  len = offset;
624 #else
625  len = static_cast<SizeType>(__builtin_ffs(r) - 1);
626 #endif
627  char* q = reinterpret_cast<char*>(os_->PushUnsafe(len));
628  for (size_t i = 0; i < len; i++)
629  q[i] = p[i];
630 
631  p += len;
632  break;
633  }
634  _mm_storeu_si128(reinterpret_cast<__m128i *>(os_->PushUnsafe(16)), s);
635  }
636 
637  is.src_ = p;
638  return RAPIDJSON_LIKELY(is.Tell() < length);
639 }
640 #elif defined(RAPIDJSON_NEON)
641 template<>
642 inline bool Writer<StringBuffer>::ScanWriteUnescapedString(StringStream& is, size_t length) {
643  if (length < 16)
644  return RAPIDJSON_LIKELY(is.Tell() < length);
645 
646  if (!RAPIDJSON_LIKELY(is.Tell() < length))
647  return false;
648 
649  const char* p = is.src_;
650  const char* end = is.head_ + length;
651  const char* nextAligned = reinterpret_cast<const char*>((reinterpret_cast<size_t>(p) + 15) & static_cast<size_t>(~15));
652  const char* endAligned = reinterpret_cast<const char*>(reinterpret_cast<size_t>(end) & static_cast<size_t>(~15));
653  if (nextAligned > end)
654  return true;
655 
656  while (p != nextAligned)
657  if (*p < 0x20 || *p == '\"' || *p == '\\') {
658  is.src_ = p;
659  return RAPIDJSON_LIKELY(is.Tell() < length);
660  }
661  else
662  os_->PutUnsafe(*p++);
663 
664  // The rest of string using SIMD
665  const uint8x16_t s0 = vmovq_n_u8('"');
666  const uint8x16_t s1 = vmovq_n_u8('\\');
667  const uint8x16_t s2 = vmovq_n_u8('\b');
668  const uint8x16_t s3 = vmovq_n_u8(32);
669 
670  for (; p != endAligned; p += 16) {
671  const uint8x16_t s = vld1q_u8(reinterpret_cast<const uint8_t *>(p));
672  uint8x16_t x = vceqq_u8(s, s0);
673  x = vorrq_u8(x, vceqq_u8(s, s1));
674  x = vorrq_u8(x, vceqq_u8(s, s2));
675  x = vorrq_u8(x, vcltq_u8(s, s3));
676 
677  x = vrev64q_u8(x); // Rev in 64
678  uint64_t low = vgetq_lane_u64(vreinterpretq_u64_u8(x), 0); // extract
679  uint64_t high = vgetq_lane_u64(vreinterpretq_u64_u8(x), 1); // extract
680 
681  SizeType len = 0;
682  bool escaped = false;
683  if (low == 0) {
684  if (high != 0) {
685  uint32_t lz = internal::clzll(high);
686  len = 8 + (lz >> 3);
687  escaped = true;
688  }
689  } else {
690  uint32_t lz = internal::clzll(low);
691  len = lz >> 3;
692  escaped = true;
693  }
694  if (RAPIDJSON_UNLIKELY(escaped)) { // some of characters is escaped
695  char* q = reinterpret_cast<char*>(os_->PushUnsafe(len));
696  for (size_t i = 0; i < len; i++)
697  q[i] = p[i];
698 
699  p += len;
700  break;
701  }
702  vst1q_u8(reinterpret_cast<uint8_t *>(os_->PushUnsafe(16)), s);
703  }
704 
705  is.src_ = p;
706  return RAPIDJSON_LIKELY(is.Tell() < length);
707 }
708 #endif // RAPIDJSON_NEON
709 
711 
712 #if defined(_MSC_VER) || defined(__clang__)
713 RAPIDJSON_DIAG_POP
714 #endif
715 
716 #endif // RAPIDJSON_RAPIDJSON_H_
size_t valueCount
number of values in this level
Definition: writer.h:292
Definition: ieee754.h:23
#define RAPIDJSON_NAMESPACE_BEGIN
provide custom rapidjson namespace (opening expression)
Definition: rapidjson.h:121
JSON writer.
Definition: fwd.h:95
void Flush()
Flush the output stream.
Definition: writer.h:282
#define RAPIDJSON_LIKELY(x)
Compiler branching hint for expression with high probability to be true.
Definition: rapidjson.h:494
Encoding conversion.
Definition: encodings.h:658
A contiguous memory buffer with an optional growing ability.
Definition: core.h:778
Writer(OutputStream &os, StackAllocator *stackAllocator=0, size_t levelDepth=kDefaultLevelDepth)
Constructor.
Definition: writer.h:102
const Ch * src_
Current read position.
Definition: stream.h:168
bool IsComplete() const
Checks whether the output is a complete JSON.
Definition: writer.h:144
Read-only string stream.
Definition: fwd.h:47
#define RAPIDJSON_UNLIKELY(x)
Compiler branching hint for expression with low probability to be true.
Definition: rapidjson.h:507
#define RAPIDJSON_NAMESPACE_END
provide custom rapidjson namespace (closing expression)
Definition: rapidjson.h:124
const Ch * head_
Original head of the string.
Definition: stream.h:169
Information for each nested level.
Definition: writer.h:290
void Reset(OutputStream &os)
Reset the writer with a new stream.
Definition: writer.h:134
void SetMaxDecimalPlaces(int maxDecimalPlaces)
Sets the maximum number of decimal places for double output.
Definition: writer.h:173
bool String(const Ch *const &str)
Simpler but slower overload.
Definition: writer.h:259
bool RawValue(const Ch *json, size_t length, Type type)
Write a raw JSON value.
Definition: writer.h:272
bool inArray
true if in array, otherwise in object
Definition: writer.h:293
#define RAPIDJSON_ASSERT(x)
Assertion.
Definition: rapidjson.h:437
bool Double(double d)
Writes the given double value to the stream.
Definition: writer.h:194