VariantData.hpp 8.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347
  1. // ArduinoJson - https://arduinojson.org
  2. // Copyright © 2014-2022, Benoit BLANCHON
  3. // MIT License
  4. #pragma once
  5. #include <ArduinoJson/Memory/MemoryPool.hpp>
  6. #include <ArduinoJson/Misc/SerializedValue.hpp>
  7. #include <ArduinoJson/Numbers/convertNumber.hpp>
  8. #include <ArduinoJson/Strings/String.hpp>
  9. #include <ArduinoJson/Strings/StringAdapters.hpp>
  10. #include <ArduinoJson/Variant/VariantContent.hpp>
  11. // VariantData can't have a constructor (to be a POD), so we have no way to fix
  12. // this warning
  13. #if defined(__GNUC__)
  14. # if __GNUC__ >= 7
  15. # pragma GCC diagnostic push
  16. # pragma GCC diagnostic ignored "-Wmaybe-uninitialized"
  17. # pragma GCC diagnostic ignored "-Wuninitialized"
  18. # endif
  19. #endif
  20. namespace ARDUINOJSON_NAMESPACE {
  21. class VariantData {
  22. VariantContent _content; // must be first to allow cast from array to variant
  23. uint8_t _flags;
  24. public:
  25. // Must be a POD!
  26. // - no constructor
  27. // - no destructor
  28. // - no virtual
  29. // - no inheritance
  30. void init() {
  31. _flags = VALUE_IS_NULL;
  32. }
  33. void operator=(const VariantData& src) {
  34. _content = src._content;
  35. _flags = uint8_t((_flags & OWNED_KEY_BIT) | (src._flags & ~OWNED_KEY_BIT));
  36. }
  37. template <typename TVisitor>
  38. typename TVisitor::result_type accept(TVisitor& visitor) const {
  39. switch (type()) {
  40. case VALUE_IS_FLOAT:
  41. return visitor.visitFloat(_content.asFloat);
  42. case VALUE_IS_ARRAY:
  43. return visitor.visitArray(_content.asCollection);
  44. case VALUE_IS_OBJECT:
  45. return visitor.visitObject(_content.asCollection);
  46. case VALUE_IS_LINKED_STRING:
  47. case VALUE_IS_OWNED_STRING:
  48. return visitor.visitString(_content.asString.data,
  49. _content.asString.size);
  50. case VALUE_IS_OWNED_RAW:
  51. case VALUE_IS_LINKED_RAW:
  52. return visitor.visitRawJson(_content.asString.data,
  53. _content.asString.size);
  54. case VALUE_IS_SIGNED_INTEGER:
  55. return visitor.visitSignedInteger(_content.asSignedInteger);
  56. case VALUE_IS_UNSIGNED_INTEGER:
  57. return visitor.visitUnsignedInteger(_content.asUnsignedInteger);
  58. case VALUE_IS_BOOLEAN:
  59. return visitor.visitBoolean(_content.asBoolean != 0);
  60. default:
  61. return visitor.visitNull();
  62. }
  63. }
  64. template <typename T>
  65. T asIntegral() const;
  66. template <typename T>
  67. T asFloat() const;
  68. String asString() const;
  69. bool asBoolean() const;
  70. CollectionData* asArray() {
  71. return isArray() ? &_content.asCollection : 0;
  72. }
  73. const CollectionData* asArray() const {
  74. return const_cast<VariantData*>(this)->asArray();
  75. }
  76. const CollectionData* asCollection() const {
  77. return isCollection() ? &_content.asCollection : 0;
  78. }
  79. CollectionData* asObject() {
  80. return isObject() ? &_content.asCollection : 0;
  81. }
  82. const CollectionData* asObject() const {
  83. return const_cast<VariantData*>(this)->asObject();
  84. }
  85. bool copyFrom(const VariantData& src, MemoryPool* pool);
  86. bool isArray() const {
  87. return (_flags & VALUE_IS_ARRAY) != 0;
  88. }
  89. bool isBoolean() const {
  90. return type() == VALUE_IS_BOOLEAN;
  91. }
  92. bool isCollection() const {
  93. return (_flags & COLLECTION_MASK) != 0;
  94. }
  95. template <typename T>
  96. bool isInteger() const {
  97. switch (type()) {
  98. case VALUE_IS_UNSIGNED_INTEGER:
  99. return canConvertNumber<T>(_content.asUnsignedInteger);
  100. case VALUE_IS_SIGNED_INTEGER:
  101. return canConvertNumber<T>(_content.asSignedInteger);
  102. default:
  103. return false;
  104. }
  105. }
  106. bool isFloat() const {
  107. return (_flags & NUMBER_BIT) != 0;
  108. }
  109. bool isString() const {
  110. return type() == VALUE_IS_LINKED_STRING || type() == VALUE_IS_OWNED_STRING;
  111. }
  112. bool isObject() const {
  113. return (_flags & VALUE_IS_OBJECT) != 0;
  114. }
  115. bool isNull() const {
  116. return type() == VALUE_IS_NULL;
  117. }
  118. bool isEnclosed() const {
  119. return !isFloat();
  120. }
  121. void remove(size_t index) {
  122. if (isArray())
  123. _content.asCollection.removeElement(index);
  124. }
  125. template <typename TAdaptedString>
  126. void remove(TAdaptedString key) {
  127. if (isObject())
  128. _content.asCollection.removeMember(key);
  129. }
  130. void setBoolean(bool value) {
  131. setType(VALUE_IS_BOOLEAN);
  132. _content.asBoolean = value;
  133. }
  134. void setFloat(Float value) {
  135. setType(VALUE_IS_FLOAT);
  136. _content.asFloat = value;
  137. }
  138. void setLinkedRaw(SerializedValue<const char*> value) {
  139. if (value.data()) {
  140. setType(VALUE_IS_LINKED_RAW);
  141. _content.asString.data = value.data();
  142. _content.asString.size = value.size();
  143. } else {
  144. setType(VALUE_IS_NULL);
  145. }
  146. }
  147. template <typename T>
  148. bool storeOwnedRaw(SerializedValue<T> value, MemoryPool* pool) {
  149. const char* dup = pool->saveString(adaptString(value.data(), value.size()));
  150. if (dup) {
  151. setType(VALUE_IS_OWNED_RAW);
  152. _content.asString.data = dup;
  153. _content.asString.size = value.size();
  154. return true;
  155. } else {
  156. setType(VALUE_IS_NULL);
  157. return false;
  158. }
  159. }
  160. template <typename T>
  161. typename enable_if<is_unsigned<T>::value>::type setInteger(T value) {
  162. setType(VALUE_IS_UNSIGNED_INTEGER);
  163. _content.asUnsignedInteger = static_cast<UInt>(value);
  164. }
  165. template <typename T>
  166. typename enable_if<is_signed<T>::value>::type setInteger(T value) {
  167. setType(VALUE_IS_SIGNED_INTEGER);
  168. _content.asSignedInteger = value;
  169. }
  170. void setNull() {
  171. setType(VALUE_IS_NULL);
  172. }
  173. void setString(String s) {
  174. ARDUINOJSON_ASSERT(s);
  175. if (s.isLinked())
  176. setType(VALUE_IS_LINKED_STRING);
  177. else
  178. setType(VALUE_IS_OWNED_STRING);
  179. _content.asString.data = s.c_str();
  180. _content.asString.size = s.size();
  181. }
  182. CollectionData& toArray() {
  183. setType(VALUE_IS_ARRAY);
  184. _content.asCollection.clear();
  185. return _content.asCollection;
  186. }
  187. CollectionData& toObject() {
  188. setType(VALUE_IS_OBJECT);
  189. _content.asCollection.clear();
  190. return _content.asCollection;
  191. }
  192. size_t memoryUsage() const {
  193. switch (type()) {
  194. case VALUE_IS_OWNED_STRING:
  195. case VALUE_IS_OWNED_RAW:
  196. // We always add a zero at the end: the deduplication function uses it
  197. // to detect the beginning of the next string.
  198. return _content.asString.size + 1;
  199. case VALUE_IS_OBJECT:
  200. case VALUE_IS_ARRAY:
  201. return _content.asCollection.memoryUsage();
  202. default:
  203. return 0;
  204. }
  205. }
  206. size_t size() const {
  207. return isCollection() ? _content.asCollection.size() : 0;
  208. }
  209. VariantData* addElement(MemoryPool* pool) {
  210. if (isNull())
  211. toArray();
  212. if (!isArray())
  213. return 0;
  214. return _content.asCollection.addElement(pool);
  215. }
  216. VariantData* getElement(size_t index) const {
  217. const CollectionData* col = asArray();
  218. return col ? col->getElement(index) : 0;
  219. }
  220. VariantData* getOrAddElement(size_t index, MemoryPool* pool) {
  221. if (isNull())
  222. toArray();
  223. if (!isArray())
  224. return 0;
  225. return _content.asCollection.getOrAddElement(index, pool);
  226. }
  227. template <typename TAdaptedString>
  228. VariantData* getMember(TAdaptedString key) const {
  229. const CollectionData* col = asObject();
  230. return col ? col->getMember(key) : 0;
  231. }
  232. template <typename TAdaptedString, typename TStoragePolicy>
  233. VariantData* getOrAddMember(TAdaptedString key, MemoryPool* pool,
  234. TStoragePolicy storage_policy) {
  235. if (isNull())
  236. toObject();
  237. if (!isObject())
  238. return 0;
  239. return _content.asCollection.getOrAddMember(key, pool, storage_policy);
  240. }
  241. void movePointers(ptrdiff_t stringDistance, ptrdiff_t variantDistance) {
  242. if (_flags & OWNED_VALUE_BIT)
  243. _content.asString.data += stringDistance;
  244. if (_flags & COLLECTION_MASK)
  245. _content.asCollection.movePointers(stringDistance, variantDistance);
  246. }
  247. uint8_t type() const {
  248. return _flags & VALUE_MASK;
  249. }
  250. template <typename TAdaptedString, typename TStoragePolicy>
  251. inline bool storeString(TAdaptedString value, MemoryPool* pool,
  252. TStoragePolicy storage) {
  253. if (value.isNull()) {
  254. setNull();
  255. return true;
  256. }
  257. return storage.store(value, pool, VariantStringSetter(this));
  258. }
  259. private:
  260. void setType(uint8_t t) {
  261. _flags &= OWNED_KEY_BIT;
  262. _flags |= t;
  263. }
  264. struct VariantStringSetter {
  265. VariantStringSetter(VariantData* instance) : _instance(instance) {}
  266. template <typename TStoredString>
  267. void operator()(TStoredString s) {
  268. if (s)
  269. _instance->setString(s);
  270. else
  271. _instance->setNull();
  272. }
  273. VariantData* _instance;
  274. };
  275. };
  276. } // namespace ARDUINOJSON_NAMESPACE
  277. #if defined(__GNUC__)
  278. # if __GNUC__ >= 8
  279. # pragma GCC diagnostic pop
  280. # endif
  281. #endif