forked from Vector35/binaryninja-api
-
Notifications
You must be signed in to change notification settings - Fork 0
/
rapidjsonwrapper.h
145 lines (129 loc) · 3.45 KB
/
rapidjsonwrapper.h
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
#pragma once
#include <exception>
#include <stdexcept>
#if defined(__GNUC__) && __GNUC__ >= 8
// Disable warnings from rapidjson performance optimizations
#pragma GCC diagnostic push
#pragma GCC diagnostic ignored "-Wclass-memaccess"
#endif
struct GenericException;
struct ParseException;
struct GenericException : public std::exception
{
char text[0x200];
GenericException(const char* file, int line, const char* description) : std::exception() {
snprintf(text, sizeof(text), "%s:%d: %s", file, line, description);
}
virtual const char* what() const throw()
{ return text; }
};
#define RAPIDJSON_HAS_STDSTRING 0
#define RAPIDJSON_HAS_CXX11_NOEXCEPT 0
#define RAPIDJSON_ASSERT(x) \
do \
{ \
if (!(x)) \
throw GenericException(__FILE__, __LINE__, #x); \
} while (0);
#define RAPIDJSON_PARSE_ERROR_NORETURN(parseErrorCode, offset) \
throw ParseException(parseErrorCode, #parseErrorCode, offset)
#include "rapidjson/error/error.h"
struct ParseException : public std::runtime_error, public rapidjson::ParseResult
{
ParseException(rapidjson::ParseErrorCode code, const char* msg, size_t offset) :
std::runtime_error(msg), ParseResult(code, offset)
{}
};
#include "rapidjson/error/en.h"
#include "rapidjson/rapidjson.h"
#include "rapidjson/document.h"
#include "rapidjson/stringbuffer.h"
#include "rapidjson/writer.h"
#include "rapidjson/prettywriter.h"
inline size_t combine(size_t seed, size_t h) noexcept
{
seed ^= h + 0x9e3779b9 + (seed << 6U) + (seed >> 2U);
return seed;
}
inline size_t HashBytes(const void* const ptr, const size_t len)
{
// Taken from https://stackoverflow.com/questions/34597260/stdhash-value-on-char-value-and-not-on-memory-address
const char* cdata = static_cast<const char *>(ptr);
uint64_t acc = 0;
for (rapidjson::SizeType i = 0; i < len; ++i)
{
const size_t next = cdata[i];
acc = (acc ^ next) * 1099511628211;
}
return acc;
}
static inline uint64_t HashRapidValue(const rapidjson::Value& val)
{
const auto type = static_cast<std::size_t>(val.GetType());
switch (val.GetType())
{
case rapidjson::kNullType:
case rapidjson::kFalseType:
{
return combine(type, 0);
}
case rapidjson::kObjectType:
{
auto seed = combine(type, val.MemberCount());
for (const auto& element : val.GetObj())
{
const auto h = HashBytes(element.name.GetString(), element.name.GetStringLength());
seed = combine(seed, h);
seed = combine(seed, HashRapidValue(element.value));
}
return seed;
}
case rapidjson::kArrayType:
{
auto seed = combine(type, val.Size());
for (const auto& element : val.GetArray())
{
seed = combine(seed, HashRapidValue(element));
}
return seed;
}
case rapidjson::kStringType:
{
return combine(type, HashBytes(val.GetString(), val.GetStringLength()));
}
case rapidjson::kTrueType:
{
return combine(type, 1);
}
case rapidjson::kNumberType:
{
if (val.IsInt64())
{
return combine(type, static_cast<size_t>(val.GetInt64()));
}
else if (val.IsUint64())
{
return combine(type, val.GetUint64());
}
else if (val.IsUint())
{
return combine(type, static_cast<size_t>(val.GetUint()));
}
else if (val.IsInt())
{
return combine(type, static_cast<size_t>(val.GetInt()));
}
else
{
const double dVal = val.GetDouble();
return combine(type, HashBytes(&dVal, sizeof(dVal)));
}
}
default:
RAPIDJSON_ASSERT(false);
return 0;
}
}
#if defined(__GNUC__) && __GNUC__ >= 8
#pragma GCC diagnostic pop
#endif