Merge pull request #628 from simdjson/jkeiser/remove-simdjson-move-result
Remove simdjson_move_result and remove deprecation warnings
This commit is contained in:
commit
47859f3560
|
@ -7,6 +7,16 @@ executors:
|
||||||
- image: gcc:7
|
- image: gcc:7
|
||||||
environment:
|
environment:
|
||||||
CXX: g++
|
CXX: g++
|
||||||
|
gcc8:
|
||||||
|
docker:
|
||||||
|
- image: gcc:8
|
||||||
|
environment:
|
||||||
|
CXX: g++
|
||||||
|
gcc9:
|
||||||
|
docker:
|
||||||
|
- image: gcc:9
|
||||||
|
environment:
|
||||||
|
CXX: g++
|
||||||
clang6:
|
clang6:
|
||||||
docker:
|
docker:
|
||||||
- image: ubuntu:18.04
|
- image: ubuntu:18.04
|
||||||
|
@ -31,6 +41,14 @@ commands:
|
||||||
- run: ARCHFLAGS=-march=haswell make test # this breaks runtime dispatch, but see https://github.com/lemire/simdjson/issues/444... this is a code robustness test
|
- run: ARCHFLAGS=-march=haswell make test # this breaks runtime dispatch, but see https://github.com/lemire/simdjson/issues/444... this is a code robustness test
|
||||||
- run: make clean
|
- run: make clean
|
||||||
- run: EXTRAFLAGS=-DSIMDJSON_NO_COMPUTED_GOTO=true make test # this should run tests with computed gotos disabled
|
- run: EXTRAFLAGS=-DSIMDJSON_NO_COMPUTED_GOTO=true make test # this should run tests with computed gotos disabled
|
||||||
|
make_test_strict:
|
||||||
|
steps:
|
||||||
|
- checkout
|
||||||
|
- run: EXTRAFLAGS=-Werror make
|
||||||
|
- run: EXTRAFLAGS=-Werror make amalgamate
|
||||||
|
- run: ARCHFLAGS=-march=haswell make amalgamate # some users do this: https://github.com/lemire/simdjson/issues/444
|
||||||
|
- run: EXTRAFLAGS=-Werror make quicktests
|
||||||
|
- run: make clean
|
||||||
|
|
||||||
cmake_test:
|
cmake_test:
|
||||||
steps:
|
steps:
|
||||||
|
@ -65,6 +83,14 @@ jobs:
|
||||||
environment: { CMAKE_TEST_FLAGS: -DSIMDJSON_ENABLE_THREADS=ON }
|
environment: { CMAKE_TEST_FLAGS: -DSIMDJSON_ENABLE_THREADS=ON }
|
||||||
steps: [ init_clang6, cmake_test ]
|
steps: [ init_clang6, cmake_test ]
|
||||||
|
|
||||||
|
gcc9-avx:
|
||||||
|
description: Build, run tests and check performance on GCC 9 and AVX 2
|
||||||
|
executor: gcc9
|
||||||
|
steps: [ make_test_strict ]
|
||||||
|
gcc8-avx:
|
||||||
|
description: Build, run tests and check performance on GCC 8 and AVX 2
|
||||||
|
executor: gcc9
|
||||||
|
steps: [ make_test_strict ]
|
||||||
gcc-avx:
|
gcc-avx:
|
||||||
description: Build, run tests and check performance on GCC 7 and AVX 2
|
description: Build, run tests and check performance on GCC 7 and AVX 2
|
||||||
executor: gcc7
|
executor: gcc7
|
||||||
|
@ -161,6 +187,8 @@ workflows:
|
||||||
version: 2.1
|
version: 2.1
|
||||||
build_and_test:
|
build_and_test:
|
||||||
jobs:
|
jobs:
|
||||||
|
- gcc9-avx
|
||||||
|
- gcc8-avx
|
||||||
- gcc-avx
|
- gcc-avx
|
||||||
- gcc-avx-dynamic
|
- gcc-avx-dynamic
|
||||||
- gcc-avx-static
|
- gcc-avx-static
|
||||||
|
|
|
@ -73,8 +73,12 @@ constexpr size_t DEFAULT_MAX_DEPTH = 1024;
|
||||||
#define unlikely(x) x
|
#define unlikely(x) x
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
#define SIMDJSON_PUSH_DISABLE_WARNINGS __pragma(warning( push ))
|
||||||
|
#define SIMDJSON_DISABLE_VS_WARNING(WARNING_NUMBER) __pragma(warning( disable : WARNING_NUMBER ))
|
||||||
|
#define SIMDJSON_DISABLE_DEPRECATED_WARNING SIMDJSON_DISABLE_VS_WARNING(4996)
|
||||||
|
#define SIMDJSON_POP_DISABLE_WARNINGS __pragma(warning( pop ))
|
||||||
|
|
||||||
#else
|
#else // MSC_VER
|
||||||
|
|
||||||
|
|
||||||
#define really_inline inline __attribute__((always_inline, unused))
|
#define really_inline inline __attribute__((always_inline, unused))
|
||||||
|
@ -90,6 +94,12 @@ constexpr size_t DEFAULT_MAX_DEPTH = 1024;
|
||||||
#define unlikely(x) __builtin_expect(!!(x), 0)
|
#define unlikely(x) __builtin_expect(!!(x), 0)
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
#define SIMDJSON_PUSH_DISABLE_WARNINGS _Pragma("GCC diagnostic push")
|
||||||
|
#define SIMDJSON_PRAGMA(P) _Pragma(#P)
|
||||||
|
#define SIMDJSON_DISABLE_GCC_WARNING(WARNING) SIMDJSON_PRAGMA(GCC diagnostic ignored #WARNING)
|
||||||
|
#define SIMDJSON_DISABLE_DEPRECATED_WARNING SIMDJSON_DISABLE_GCC_WARNING(-Wdeprecated-declarations)
|
||||||
|
#define SIMDJSON_POP_DISABLE_WARNINGS _Pragma("GCC diagnostic pop")
|
||||||
|
|
||||||
#endif // MSC_VER
|
#endif // MSC_VER
|
||||||
|
|
||||||
#endif // SIMDJSON_COMMON_DEFS_H
|
#endif // SIMDJSON_COMMON_DEFS_H
|
||||||
|
|
|
@ -256,6 +256,11 @@ public:
|
||||||
*/
|
*/
|
||||||
inline array_result as_array() const noexcept;
|
inline array_result as_array() const noexcept;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get the root element of this document.
|
||||||
|
*/
|
||||||
|
inline element_result root() const noexcept;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get the value associated with the given JSON pointer.
|
* Get the value associated with the given JSON pointer.
|
||||||
*
|
*
|
||||||
|
@ -413,6 +418,11 @@ public:
|
||||||
really_inline bool is_number() const noexcept;
|
really_inline bool is_number() const noexcept;
|
||||||
/** Whether this is a JSON integer (e.g. 1 or -1, but *not* 1.0 or 1e2) */
|
/** Whether this is a JSON integer (e.g. 1 or -1, but *not* 1.0 or 1e2) */
|
||||||
really_inline bool is_integer() const noexcept;
|
really_inline bool is_integer() const noexcept;
|
||||||
|
/** Whether this is a JSON integer in [9223372036854775808, 18446744073709551616)
|
||||||
|
* that is, a value too large for a signed 64-bit integer, but that still fits
|
||||||
|
* in a 64-bit word. Note that is_integer() is true when is_unsigned_integer()
|
||||||
|
* is true.*/
|
||||||
|
really_inline bool is_unsigned_integer() const noexcept;
|
||||||
/** Whether this is a JSON number but not an integer */
|
/** Whether this is a JSON number but not an integer */
|
||||||
really_inline bool is_float() const noexcept;
|
really_inline bool is_float() const noexcept;
|
||||||
/** Whether this is a JSON string (e.g. "abc") */
|
/** Whether this is a JSON string (e.g. "abc") */
|
||||||
|
@ -889,17 +899,33 @@ public:
|
||||||
/**
|
/**
|
||||||
* Get the value associated with the given key.
|
* Get the value associated with the given key.
|
||||||
*
|
*
|
||||||
* Note: The key will be matched against **unescaped** JSON:
|
* Note: The key will be matched against **unescaped** JSON.
|
||||||
*
|
|
||||||
* document::parser parser;
|
|
||||||
* parser.parse(R"({ "a\n": 1 })")["a\n"].as_uint64_t().value == 1
|
|
||||||
* parser.parse(R"({ "a\n": 1 })")["a\\n"].as_uint64_t().error == NO_SUCH_FIELD
|
|
||||||
*
|
*
|
||||||
* @return The value associated with this field, or:
|
* @return The value associated with this field, or:
|
||||||
* - NO_SUCH_FIELD if the field does not exist in the object
|
* - NO_SUCH_FIELD if the field does not exist in the object
|
||||||
*/
|
*/
|
||||||
inline element_result at_key(const char *s) const noexcept;
|
inline element_result at_key(const char *s) const noexcept;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get the value associated with the given key, the provided key is
|
||||||
|
* considered to have length characters.
|
||||||
|
*
|
||||||
|
* Note: The key will be matched against **unescaped** JSON.
|
||||||
|
*
|
||||||
|
* @return The value associated with this field, or:
|
||||||
|
* - NO_SUCH_FIELD if the field does not exist in the object
|
||||||
|
*/
|
||||||
|
inline element_result at_key(const char *s, size_t length) const noexcept;
|
||||||
|
/**
|
||||||
|
* Get the value associated with the given key in a case-insensitive manner.
|
||||||
|
*
|
||||||
|
* Note: The key will be matched against **unescaped** JSON.
|
||||||
|
*
|
||||||
|
* @return The value associated with this field, or:
|
||||||
|
* - NO_SUCH_FIELD if the field does not exist in the object
|
||||||
|
*/
|
||||||
|
inline element_result at_key_case_insensitive(const char *s) const noexcept;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
really_inline object(const document *_doc, size_t _json_index) noexcept;
|
really_inline object(const document *_doc, size_t _json_index) noexcept;
|
||||||
friend class document::element;
|
friend class document::element;
|
||||||
|
@ -926,7 +952,7 @@ private:
|
||||||
class document::element_result : public simdjson_result<document::element> {
|
class document::element_result : public simdjson_result<document::element> {
|
||||||
public:
|
public:
|
||||||
really_inline element_result() noexcept;
|
really_inline element_result() noexcept;
|
||||||
really_inline element_result(element value) noexcept;
|
really_inline element_result(element &&value) noexcept;
|
||||||
really_inline element_result(error_code error) noexcept;
|
really_inline element_result(error_code error) noexcept;
|
||||||
|
|
||||||
/** Whether this is a JSON `null` */
|
/** Whether this is a JSON `null` */
|
||||||
|
|
|
@ -85,12 +85,12 @@ struct simdjson_result : public std::pair<T, error_code> {
|
||||||
/**
|
/**
|
||||||
* Move the value and the error to the provided variables.
|
* Move the value and the error to the provided variables.
|
||||||
*/
|
*/
|
||||||
void tie(T& t, error_code & e) {
|
void tie(T& t, error_code & e) && noexcept {
|
||||||
// on the clang compiler that comes with current macOS (Apple clang version 11.0.0),
|
// on the clang compiler that comes with current macOS (Apple clang version 11.0.0),
|
||||||
// tie(width, error) = size["w"].as_uint64_t();
|
// tie(width, error) = size["w"].as_uint64_t();
|
||||||
// fails with "error: no viable overloaded '='""
|
// fails with "error: no viable overloaded '='""
|
||||||
t = std::move(this->first);
|
t = std::forward<simdjson_result<T>>(*this).first;
|
||||||
e = std::move(this->second);
|
e = this->second;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -105,17 +105,29 @@ struct simdjson_result : public std::pair<T, error_code> {
|
||||||
*
|
*
|
||||||
* @throw simdjson_error if there was an error.
|
* @throw simdjson_error if there was an error.
|
||||||
*/
|
*/
|
||||||
T get() noexcept(false) {
|
T& get() noexcept(false) {
|
||||||
if (error()) { throw simdjson_error(error()); }
|
if (error()) { throw simdjson_error(error()); }
|
||||||
return this->first;
|
return this->first;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The value of the function.
|
||||||
|
*
|
||||||
|
* @throw simdjson_error if there was an error.
|
||||||
|
*/
|
||||||
|
T&& take() && {
|
||||||
|
if (error()) { throw simdjson_error(error()); }
|
||||||
|
return std::forward<T>(this->first);
|
||||||
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Cast to the value (will throw on error).
|
* Cast to the value (will throw on error).
|
||||||
*
|
*
|
||||||
* @throw simdjson_error if there was an error.
|
* @throw simdjson_error if there was an error.
|
||||||
*/
|
*/
|
||||||
operator T() noexcept(false) { return get(); }
|
operator T&&() && {
|
||||||
|
return std::forward<simdjson_result<T>>(*this).take();
|
||||||
|
}
|
||||||
|
|
||||||
#endif // SIMDJSON_EXCEPTIONS
|
#endif // SIMDJSON_EXCEPTIONS
|
||||||
|
|
||||||
|
@ -127,84 +139,17 @@ struct simdjson_result : public std::pair<T, error_code> {
|
||||||
/**
|
/**
|
||||||
* Create a new error result.
|
* Create a new error result.
|
||||||
*/
|
*/
|
||||||
simdjson_result(error_code _error) noexcept : std::pair<T, error_code>({}, _error) {}
|
simdjson_result(error_code error) noexcept : std::pair<T, error_code>(T{}, error) {}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Create a new successful result.
|
* Create a new successful result.
|
||||||
*/
|
*/
|
||||||
simdjson_result(T _value) noexcept : std::pair<T, error_code>(_value, SUCCESS) {}
|
simdjson_result(T &&value) noexcept : std::pair<T, error_code>(std::forward<T>(value), SUCCESS) {}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Create a new result with both things (use if you don't want to branch when creating the result).
|
* Create a new result with both things (use if you don't want to branch when creating the result).
|
||||||
*/
|
*/
|
||||||
simdjson_result(T value, error_code error) noexcept : std::pair<T, error_code>(value, error) {}
|
simdjson_result(T &&value, error_code error) noexcept : std::pair<T, error_code>(std::forward<T>(value), error) {}
|
||||||
};
|
|
||||||
|
|
||||||
/**
|
|
||||||
* The result of a simd operation that could fail.
|
|
||||||
*
|
|
||||||
* This class is for values that must be *moved*, like padded_string and document.
|
|
||||||
*
|
|
||||||
* Gives the option of reading error codes, or throwing an exception by casting to the desired result.
|
|
||||||
*/
|
|
||||||
template<typename T>
|
|
||||||
struct simdjson_move_result : std::pair<T, error_code> {
|
|
||||||
/**
|
|
||||||
* Move the value and the error to the provided variables.
|
|
||||||
*/
|
|
||||||
void tie(T& t, error_code & e) {
|
|
||||||
// on the clang compiler that comes with current macOS (Apple clang version 11.0.0),
|
|
||||||
// std::tie(this->json, error) = padded_string::load(filename);
|
|
||||||
// fails with "benchmark/benchmarker.h:266:33: error: no viable overloaded '='""
|
|
||||||
t = std::move(this->first);
|
|
||||||
e = std::move(this->second);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* The error.
|
|
||||||
*/
|
|
||||||
error_code error() const { return this->second; }
|
|
||||||
|
|
||||||
#if SIMDJSON_EXCEPTIONS
|
|
||||||
|
|
||||||
/**
|
|
||||||
* The value of the function.
|
|
||||||
*
|
|
||||||
* @throw simdjson_error if there was an error.
|
|
||||||
*/
|
|
||||||
T move() noexcept(false) {
|
|
||||||
if (error()) { throw simdjson_error(error()); }
|
|
||||||
return std::move(this->first);
|
|
||||||
};
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Cast to the value (will throw on error).
|
|
||||||
*
|
|
||||||
* @throw simdjson_error if there was an error.
|
|
||||||
*/
|
|
||||||
operator T() noexcept(false) { return move(); }
|
|
||||||
|
|
||||||
#endif
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Create a new empty result with error = UNINITIALIZED.
|
|
||||||
*/
|
|
||||||
simdjson_move_result() noexcept : simdjson_move_result(UNINITIALIZED) {}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Create a new error result.
|
|
||||||
*/
|
|
||||||
simdjson_move_result(error_code error) noexcept : std::pair<T, error_code>(T(), error) {}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Create a new successful result.
|
|
||||||
*/
|
|
||||||
simdjson_move_result(T value) noexcept : std::pair<T, error_code>(std::move(value), SUCCESS) {}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Create a new result with both things (use if you don't want to branch when creating the result).
|
|
||||||
*/
|
|
||||||
simdjson_move_result(T value, error_code error) noexcept : std::pair<T, error_code>(std::move(value), error) {}
|
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -17,7 +17,7 @@ namespace simdjson {
|
||||||
// element_result inline implementation
|
// element_result inline implementation
|
||||||
//
|
//
|
||||||
really_inline document::element_result::element_result() noexcept : simdjson_result<element>() {}
|
really_inline document::element_result::element_result() noexcept : simdjson_result<element>() {}
|
||||||
really_inline document::element_result::element_result(element value) noexcept : simdjson_result<element>(value) {}
|
really_inline document::element_result::element_result(element &&value) noexcept : simdjson_result<element>((element&&)value) {}
|
||||||
really_inline document::element_result::element_result(error_code error) noexcept : simdjson_result<element>(error) {}
|
really_inline document::element_result::element_result(error_code error) noexcept : simdjson_result<element>(error) {}
|
||||||
inline simdjson_result<bool> document::element_result::is_null() const noexcept {
|
inline simdjson_result<bool> document::element_result::is_null() const noexcept {
|
||||||
if (error()) { return error(); }
|
if (error()) { return error(); }
|
||||||
|
@ -113,7 +113,7 @@ inline document::element_result::operator document::object() const noexcept(fals
|
||||||
// array_result inline implementation
|
// array_result inline implementation
|
||||||
//
|
//
|
||||||
really_inline document::array_result::array_result() noexcept : simdjson_result<array>() {}
|
really_inline document::array_result::array_result() noexcept : simdjson_result<array>() {}
|
||||||
really_inline document::array_result::array_result(array value) noexcept : simdjson_result<array>(value) {}
|
really_inline document::array_result::array_result(array value) noexcept : simdjson_result<array>((array&&)value) {}
|
||||||
really_inline document::array_result::array_result(error_code error) noexcept : simdjson_result<array>(error) {}
|
really_inline document::array_result::array_result(error_code error) noexcept : simdjson_result<array>(error) {}
|
||||||
|
|
||||||
#if SIMDJSON_EXCEPTIONS
|
#if SIMDJSON_EXCEPTIONS
|
||||||
|
@ -149,7 +149,7 @@ inline document::element_result document::array_result::at(size_t index) const n
|
||||||
// object_result inline implementation
|
// object_result inline implementation
|
||||||
//
|
//
|
||||||
really_inline document::object_result::object_result() noexcept : simdjson_result<object>() {}
|
really_inline document::object_result::object_result() noexcept : simdjson_result<object>() {}
|
||||||
really_inline document::object_result::object_result(object value) noexcept : simdjson_result<object>(value) {}
|
really_inline document::object_result::object_result(object value) noexcept : simdjson_result<object>((object&&)value) {}
|
||||||
really_inline document::object_result::object_result(error_code error) noexcept : simdjson_result<object>(error) {}
|
really_inline document::object_result::object_result(error_code error) noexcept : simdjson_result<object>(error) {}
|
||||||
|
|
||||||
inline document::element_result document::object_result::operator[](std::string_view json_pointer) const noexcept {
|
inline document::element_result document::object_result::operator[](std::string_view json_pointer) const noexcept {
|
||||||
|
@ -359,12 +359,14 @@ inline bool document::dump_raw_tape(std::ostream &os) const noexcept {
|
||||||
inline document::doc_result::doc_result(document &doc, error_code error) noexcept : simdjson_result<document&>(doc, error) { }
|
inline document::doc_result::doc_result(document &doc, error_code error) noexcept : simdjson_result<document&>(doc, error) { }
|
||||||
|
|
||||||
inline document::array_result document::doc_result::as_array() const noexcept {
|
inline document::array_result document::doc_result::as_array() const noexcept {
|
||||||
if (error()) { return error(); }
|
return root().as_array();
|
||||||
return first.root().as_array();
|
|
||||||
}
|
}
|
||||||
inline document::object_result document::doc_result::as_object() const noexcept {
|
inline document::object_result document::doc_result::as_object() const noexcept {
|
||||||
|
return root().as_object();
|
||||||
|
}
|
||||||
|
inline document::element_result document::doc_result::root() const noexcept {
|
||||||
if (error()) { return error(); }
|
if (error()) { return error(); }
|
||||||
return first.root().as_object();
|
return first.root();
|
||||||
}
|
}
|
||||||
|
|
||||||
inline document::element_result document::doc_result::operator[](std::string_view key) const noexcept {
|
inline document::element_result document::doc_result::operator[](std::string_view key) const noexcept {
|
||||||
|
@ -783,6 +785,16 @@ inline document::element_result document::object::at(std::string_view json_point
|
||||||
|
|
||||||
return child;
|
return child;
|
||||||
}
|
}
|
||||||
|
inline document::element_result document::object::at_key(const char *key, size_t length) const noexcept {
|
||||||
|
iterator end_field = end();
|
||||||
|
for (iterator field = begin(); field != end_field; ++field) {
|
||||||
|
std::string_view v{field.key()};
|
||||||
|
if ((v.size() == length) && (!memcmp(v.data(), key, length))) {
|
||||||
|
return field.value();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return NO_SUCH_FIELD;
|
||||||
|
}
|
||||||
inline document::element_result document::object::at_key(std::string_view key) const noexcept {
|
inline document::element_result document::object::at_key(std::string_view key) const noexcept {
|
||||||
iterator end_field = end();
|
iterator end_field = end();
|
||||||
for (iterator field = begin(); field != end_field; ++field) {
|
for (iterator field = begin(); field != end_field; ++field) {
|
||||||
|
@ -801,7 +813,18 @@ inline document::element_result document::object::at_key(const char *key) const
|
||||||
}
|
}
|
||||||
return NO_SUCH_FIELD;
|
return NO_SUCH_FIELD;
|
||||||
}
|
}
|
||||||
|
// In case you wonder why we need this, please see
|
||||||
|
// https://github.com/simdjson/simdjson/issues/323
|
||||||
|
// People do seek keys in a case-insensitive manner.
|
||||||
|
inline document::element_result document::object::at_key_case_insensitive(const char *key) const noexcept {
|
||||||
|
iterator end_field = end();
|
||||||
|
for (iterator field = begin(); field != end_field; ++field) {
|
||||||
|
if (!simdjson_strcasecmp(key, field.key_c_str())) {
|
||||||
|
return field.value();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return NO_SUCH_FIELD;
|
||||||
|
}
|
||||||
//
|
//
|
||||||
// document::object::iterator inline implementation
|
// document::object::iterator inline implementation
|
||||||
//
|
//
|
||||||
|
@ -859,6 +882,9 @@ really_inline bool document::element::is_float() const noexcept {
|
||||||
really_inline bool document::element::is_integer() const noexcept {
|
really_inline bool document::element::is_integer() const noexcept {
|
||||||
return type() == internal::tape_type::UINT64 || type() == internal::tape_type::INT64;
|
return type() == internal::tape_type::UINT64 || type() == internal::tape_type::INT64;
|
||||||
}
|
}
|
||||||
|
really_inline bool document::element::is_unsigned_integer() const noexcept {
|
||||||
|
return type() == internal::tape_type::UINT64;
|
||||||
|
}
|
||||||
really_inline bool document::element::is_string() const noexcept {
|
really_inline bool document::element::is_string() const noexcept {
|
||||||
return type() == internal::tape_type::STRING;
|
return type() == internal::tape_type::STRING;
|
||||||
}
|
}
|
||||||
|
@ -951,7 +977,7 @@ inline simdjson_result<double> document::element::as_double() const noexcept {
|
||||||
if (result < 0) {
|
if (result < 0) {
|
||||||
return NUMBER_OUT_OF_RANGE;
|
return NUMBER_OUT_OF_RANGE;
|
||||||
}
|
}
|
||||||
return result;
|
return double(result);
|
||||||
}
|
}
|
||||||
case internal::tape_type::DOUBLE:
|
case internal::tape_type::DOUBLE:
|
||||||
return next_tape_value<double>();
|
return next_tape_value<double>();
|
||||||
|
|
|
@ -98,7 +98,9 @@ inline const char *padded_string::data() const noexcept { return data_ptr; }
|
||||||
|
|
||||||
inline char *padded_string::data() noexcept { return data_ptr; }
|
inline char *padded_string::data() noexcept { return data_ptr; }
|
||||||
|
|
||||||
inline simdjson_move_result<padded_string> padded_string::load(const std::string &filename) noexcept {
|
inline padded_string::operator std::string_view() const { return std::string_view(data(), length()); }
|
||||||
|
|
||||||
|
inline simdjson_result<padded_string> padded_string::load(const std::string &filename) noexcept {
|
||||||
// Open the file
|
// Open the file
|
||||||
std::FILE *fp = std::fopen(filename.c_str(), "rb");
|
std::FILE *fp = std::fopen(filename.c_str(), "rb");
|
||||||
if (fp == nullptr) {
|
if (fp == nullptr) {
|
||||||
|
@ -131,7 +133,7 @@ inline simdjson_move_result<padded_string> padded_string::load(const std::string
|
||||||
return IO_ERROR;
|
return IO_ERROR;
|
||||||
}
|
}
|
||||||
|
|
||||||
return std::move(s);
|
return s;
|
||||||
}
|
}
|
||||||
|
|
||||||
} // namespace simdjson
|
} // namespace simdjson
|
||||||
|
|
|
@ -89,12 +89,17 @@ struct padded_string final {
|
||||||
**/
|
**/
|
||||||
char *data() noexcept;
|
char *data() noexcept;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a std::string_view with the same content.
|
||||||
|
*/
|
||||||
|
operator std::string_view() const;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Load this padded string from a file.
|
* Load this padded string from a file.
|
||||||
*
|
*
|
||||||
* @param path the path to the file.
|
* @param path the path to the file.
|
||||||
**/
|
**/
|
||||||
inline static simdjson_move_result<padded_string> load(const std::string &path) noexcept;
|
inline static simdjson_result<padded_string> load(const std::string &path) noexcept;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
padded_string &operator=(const padded_string &o) = delete;
|
padded_string &operator=(const padded_string &o) = delete;
|
||||||
|
|
|
@ -2,10 +2,8 @@
|
||||||
|
|
||||||
TMPDIR1=$(mktemp -d -t simdjsonXXXXXXXX)
|
TMPDIR1=$(mktemp -d -t simdjsonXXXXXXXX)
|
||||||
TMPDIR2=$(mktemp -d -t simdjsonXXXXXXXX)
|
TMPDIR2=$(mktemp -d -t simdjsonXXXXXXXX)
|
||||||
TMPDIR3=$(mktemp -d -t simdjsonXXXXXXXX)
|
|
||||||
TMPDIR4=$(mktemp -d -t simdjsonXXXXXXXX)
|
|
||||||
trap "exit 1" HUP INT PIPE QUIT TERM
|
trap "exit 1" HUP INT PIPE QUIT TERM
|
||||||
trap "rm -rf $TMPDIR1 $TMPDIR2 $TMPDIR3 $TMPDIR4" EXIT
|
trap "rm -rf $TMPDIR1 $TMPDIR2" EXIT
|
||||||
|
|
||||||
function founderror() {
|
function founderror() {
|
||||||
echo "code is wrong"
|
echo "code is wrong"
|
||||||
|
@ -16,33 +14,19 @@ make minify json2json
|
||||||
for i in `cd jsonexamples && ls -1 *.json`; do
|
for i in `cd jsonexamples && ls -1 *.json`; do
|
||||||
echo $i
|
echo $i
|
||||||
./json2json jsonexamples/$i > $TMPDIR1/$i
|
./json2json jsonexamples/$i > $TMPDIR1/$i
|
||||||
./json2json -a jsonexamples/$i > $TMPDIR3/$i
|
|
||||||
./json2json $TMPDIR1/$i > $TMPDIR2/$i
|
./json2json $TMPDIR1/$i > $TMPDIR2/$i
|
||||||
./json2json -a $TMPDIR3/$i > $TMPDIR4/$i
|
|
||||||
cmp $TMPDIR1/$i $TMPDIR2/$i
|
cmp $TMPDIR1/$i $TMPDIR2/$i
|
||||||
retVal=$?
|
retVal=$?
|
||||||
if [ $retVal -ne 0 ]; then
|
if [ $retVal -ne 0 ]; then
|
||||||
founderror
|
founderror
|
||||||
fi
|
fi
|
||||||
cmp $TMPDIR3/$i $TMPDIR4/$i
|
|
||||||
retVal=$?
|
|
||||||
if [ $retVal -ne 0 ]; then
|
|
||||||
founderror
|
|
||||||
fi
|
|
||||||
./minify $TMPDIR1/$i > $TMPDIR1/minify$i
|
./minify $TMPDIR1/$i > $TMPDIR1/minify$i
|
||||||
./minify $TMPDIR2/$i > $TMPDIR2/minify$i
|
./minify $TMPDIR2/$i > $TMPDIR2/minify$i
|
||||||
./minify $TMPDIR3/$i > $TMPDIR3/minify$i
|
|
||||||
./minify $TMPDIR4/$i > $TMPDIR4/minify$i
|
|
||||||
cmp $TMPDIR1/minify$i $TMPDIR2/minify$i
|
cmp $TMPDIR1/minify$i $TMPDIR2/minify$i
|
||||||
retVal=$?
|
retVal=$?
|
||||||
if [ $retVal -ne 0 ]; then
|
if [ $retVal -ne 0 ]; then
|
||||||
founderror
|
founderror
|
||||||
fi
|
fi
|
||||||
cmp $TMPDIR3/minify$i $TMPDIR4/minify$i
|
|
||||||
retVal=$?
|
|
||||||
if [ $retVal -ne 0 ]; then
|
|
||||||
founderror
|
|
||||||
fi
|
|
||||||
./json2json $TMPDIR1/minify$i > $TMPDIR2/bisminify$i
|
./json2json $TMPDIR1/minify$i > $TMPDIR2/bisminify$i
|
||||||
cmp $TMPDIR1/$i $TMPDIR2/bisminify$i
|
cmp $TMPDIR1/$i $TMPDIR2/bisminify$i
|
||||||
retVal=$?
|
retVal=$?
|
||||||
|
@ -54,37 +38,22 @@ done
|
||||||
for i in `cd jsonchecker && ls -1 pass*.json`; do
|
for i in `cd jsonchecker && ls -1 pass*.json`; do
|
||||||
echo $i
|
echo $i
|
||||||
./json2json jsonchecker/$i > $TMPDIR1/$i
|
./json2json jsonchecker/$i > $TMPDIR1/$i
|
||||||
./json2json -a jsonchecker/$i > $TMPDIR3/$i
|
|
||||||
./json2json $TMPDIR1/$i > $TMPDIR2/$i
|
./json2json $TMPDIR1/$i > $TMPDIR2/$i
|
||||||
./json2json -a $TMPDIR3/$i > $TMPDIR4/$i
|
|
||||||
cmp $TMPDIR1/$i $TMPDIR2/$i
|
cmp $TMPDIR1/$i $TMPDIR2/$i
|
||||||
retVal=$?
|
retVal=$?
|
||||||
if [ $retVal -ne 0 ]; then
|
if [ $retVal -ne 0 ]; then
|
||||||
echo "reg failure"
|
echo "reg failure"
|
||||||
founderror
|
founderror
|
||||||
fi
|
fi
|
||||||
cmp $TMPDIR3/$i $TMPDIR4/$i
|
|
||||||
retVal=$?
|
|
||||||
if [ $retVal -ne 0 ]; then
|
|
||||||
echo "-a failure"
|
|
||||||
founderror
|
|
||||||
fi
|
|
||||||
./minify $TMPDIR1/$i > $TMPDIR1/minify$i
|
./minify $TMPDIR1/$i > $TMPDIR1/minify$i
|
||||||
./minify $TMPDIR2/$i > $TMPDIR2/minify$i
|
./minify $TMPDIR2/$i > $TMPDIR2/minify$i
|
||||||
./minify $TMPDIR3/$i > $TMPDIR3/minify$i
|
|
||||||
./minify $TMPDIR4/$i > $TMPDIR4/minify$i
|
|
||||||
cmp $TMPDIR1/minify$i $TMPDIR2/minify$i
|
cmp $TMPDIR1/minify$i $TMPDIR2/minify$i
|
||||||
retVal=$?
|
retVal=$?
|
||||||
if [ $retVal -ne 0 ]; then
|
if [ $retVal -ne 0 ]; then
|
||||||
echo "reg failure, step 2"
|
echo "reg failure, step 2"
|
||||||
founderror
|
founderror
|
||||||
fi
|
fi
|
||||||
cmp $TMPDIR3/minify$i $TMPDIR4/minify$i
|
|
||||||
retVal=$?
|
|
||||||
if [ $retVal -ne 0 ]; then
|
|
||||||
echo "-a failure, step 2"
|
|
||||||
founderror
|
|
||||||
fi
|
|
||||||
./json2json $TMPDIR1/minify$i > $TMPDIR2/bisminify$i
|
./json2json $TMPDIR1/minify$i > $TMPDIR2/bisminify$i
|
||||||
cmp $TMPDIR1/$i $TMPDIR2/bisminify$i
|
cmp $TMPDIR1/$i $TMPDIR2/bisminify$i
|
||||||
retVal=$?
|
retVal=$?
|
||||||
|
|
|
@ -74,15 +74,8 @@ int main(int argc, char *argv[]) {
|
||||||
std::cout << p.size() << " B ";
|
std::cout << p.size() << " B ";
|
||||||
std::cout << std::endl;
|
std::cout << std::endl;
|
||||||
}
|
}
|
||||||
simdjson::ParsedJson pj;
|
simdjson::document::parser parser;
|
||||||
size_t max_depth = 1024 * 4;
|
auto [doc, err] = parser.parse(p);
|
||||||
bool allocok = pj.allocate_capacity(p.size(), max_depth);
|
|
||||||
if (!allocok) {
|
|
||||||
std::cerr << "can't allocate memory" << std::endl;
|
|
||||||
return EXIT_FAILURE;
|
|
||||||
}
|
|
||||||
int oursreturn = json_parse(p, pj);
|
|
||||||
bool ours_correct = (oursreturn == 0); // returns 0 on success
|
|
||||||
|
|
||||||
rapidjson::Document d;
|
rapidjson::Document d;
|
||||||
|
|
||||||
|
@ -98,19 +91,19 @@ int main(int argc, char *argv[]) {
|
||||||
.is_valid();
|
.is_valid();
|
||||||
if (just_favorites) {
|
if (just_favorites) {
|
||||||
printf("our parser : %s \n",
|
printf("our parser : %s \n",
|
||||||
ours_correct ? "correct" : "invalid");
|
(err == simdjson::error_code::SUCCESS) ? "correct" : "invalid");
|
||||||
printf("rapid (check encoding) : %s \n",
|
printf("rapid (check encoding) : %s \n",
|
||||||
rapid_correct_checkencoding ? "correct" : "invalid");
|
rapid_correct_checkencoding ? "correct" : "invalid");
|
||||||
printf("sajson : %s \n",
|
printf("sajson : %s \n",
|
||||||
sajson_correct ? "correct" : "invalid");
|
sajson_correct ? "correct" : "invalid");
|
||||||
if (oursreturn == simdjson::DEPTH_ERROR) {
|
if (err == simdjson::DEPTH_ERROR) {
|
||||||
printf("simdjson encountered a DEPTH_ERROR, it was parametrized to "
|
printf("simdjson encountered a DEPTH_ERROR, it was parametrized to "
|
||||||
"reject documents with depth exceeding %zu.\n",
|
"reject documents with depth exceeding %zu.\n",
|
||||||
max_depth);
|
parser.max_depth());
|
||||||
}
|
}
|
||||||
if ((ours_correct != rapid_correct_checkencoding) ||
|
if (((err == simdjson::error_code::SUCCESS) != rapid_correct_checkencoding) ||
|
||||||
(rapid_correct_checkencoding != sajson_correct) ||
|
(rapid_correct_checkencoding != sajson_correct) ||
|
||||||
(ours_correct != sajson_correct)) {
|
((err == simdjson::SUCCESS) != sajson_correct)) {
|
||||||
printf("WARNING: THEY DISAGREE\n\n");
|
printf("WARNING: THEY DISAGREE\n\n");
|
||||||
return EXIT_FAILURE;
|
return EXIT_FAILURE;
|
||||||
}
|
}
|
||||||
|
@ -137,11 +130,11 @@ int main(int argc, char *argv[]) {
|
||||||
if (tokens == nullptr) {
|
if (tokens == nullptr) {
|
||||||
printf("Failed to alloc memory for jsmn\n");
|
printf("Failed to alloc memory for jsmn\n");
|
||||||
} else {
|
} else {
|
||||||
jsmn_parser parser;
|
jsmn_parser jsmnparser;
|
||||||
jsmn_init(&parser);
|
jsmn_init(&jsmnparser);
|
||||||
memcpy(buffer, p.data(), p.size());
|
memcpy(buffer, p.data(), p.size());
|
||||||
buffer[p.size()] = '\0';
|
buffer[p.size()] = '\0';
|
||||||
int r = jsmn_parse(&parser, buffer, p.size(), tokens.get(), p.size());
|
int r = jsmn_parse(&jsmnparser, buffer, p.size(), tokens.get(), p.size());
|
||||||
tokens = nullptr;
|
tokens = nullptr;
|
||||||
jsmn_correct = (r > 0);
|
jsmn_correct = (r > 0);
|
||||||
}
|
}
|
||||||
|
@ -163,7 +156,7 @@ int main(int argc, char *argv[]) {
|
||||||
delete json_cpp_reader;
|
delete json_cpp_reader;
|
||||||
|
|
||||||
printf("our parser : %s \n",
|
printf("our parser : %s \n",
|
||||||
ours_correct ? "correct" : "invalid");
|
(err == simdjson::error_code::SUCCESS) ? "correct" : "invalid");
|
||||||
printf("rapid : %s \n",
|
printf("rapid : %s \n",
|
||||||
rapid_correct ? "correct" : "invalid");
|
rapid_correct ? "correct" : "invalid");
|
||||||
printf("rapid (check encoding) : %s \n",
|
printf("rapid (check encoding) : %s \n",
|
||||||
|
|
|
@ -42,29 +42,17 @@ namespace number_tests {
|
||||||
bool small_integers() {
|
bool small_integers() {
|
||||||
std::cout << __func__ << std::endl;
|
std::cout << __func__ << std::endl;
|
||||||
char buf[1024];
|
char buf[1024];
|
||||||
simdjson::ParsedJson pj;
|
simdjson::document::parser parser;
|
||||||
for (int m = 10; m < 20; m++) {
|
for (int m = 10; m < 20; m++) {
|
||||||
for (int i = -1024; i < 1024; i++) {
|
for (int i = -1024; i < 1024; i++) {
|
||||||
auto n = sprintf(buf, "%*d", m, i);
|
auto n = sprintf(buf, "%*d", m, i);
|
||||||
buf[n] = '\0';
|
buf[n] = '\0';
|
||||||
fflush(NULL);
|
fflush(NULL);
|
||||||
auto error = simdjson::json_parse(buf, n, pj);
|
|
||||||
if (error) {
|
auto [actual, error] = parser.parse(buf, n).root().as_int64_t();
|
||||||
printf("Could not parse '%s': %s\n", buf, simdjson::error_message(error).c_str());
|
if (error) { std::cerr << error << std::endl; return false; }
|
||||||
return false;
|
if (actual != i) {
|
||||||
}
|
std::cerr << "JSON '" << buf << " parsed to " << actual << " instead of " << i << std::endl;
|
||||||
simdjson::ParsedJson::Iterator iter(pj);
|
|
||||||
if(!iter.is_number()) {
|
|
||||||
printf("Root should be number\n");
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
if(!iter.is_integer()) {
|
|
||||||
printf("Root should be an integer\n");
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
int64_t x = iter.get_integer();
|
|
||||||
if(x != i) {
|
|
||||||
printf("failed to parse %s. \n", buf);
|
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -77,62 +65,22 @@ namespace number_tests {
|
||||||
bool powers_of_two() {
|
bool powers_of_two() {
|
||||||
std::cout << __func__ << std::endl;
|
std::cout << __func__ << std::endl;
|
||||||
char buf[1024];
|
char buf[1024];
|
||||||
simdjson::ParsedJson pj;
|
simdjson::document::parser parser;
|
||||||
int maxulp = 0;
|
int maxulp = 0;
|
||||||
for (int i = -1075; i < 1024; ++i) {// large negative values should be zero.
|
for (int i = -1075; i < 1024; ++i) {// large negative values should be zero.
|
||||||
double expected = pow(2, i);
|
double expected = pow(2, i);
|
||||||
auto n = sprintf(buf, "%.*e", std::numeric_limits<double>::max_digits10 - 1, expected);
|
auto n = sprintf(buf, "%.*e", std::numeric_limits<double>::max_digits10 - 1, expected);
|
||||||
buf[n] = '\0';
|
buf[n] = '\0';
|
||||||
fflush(NULL);
|
fflush(NULL);
|
||||||
auto error = simdjson::json_parse(buf, n, pj);
|
|
||||||
if (error) {
|
auto [actual, error] = parser.parse(buf, n).root().as_double();
|
||||||
printf("Could not parse '%s': %s\n", buf, simdjson::error_message(error).c_str());
|
if (error) { std::cerr << error << std::endl; return false; }
|
||||||
|
int ulp = f64_ulp_dist(actual,expected);
|
||||||
|
if(ulp > maxulp) maxulp = ulp;
|
||||||
|
if(ulp > 0) {
|
||||||
|
std::cerr << "JSON '" << buf << " parsed to " << actual << " instead of " << expected << std::endl;
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
simdjson::ParsedJson::Iterator iter(pj);
|
|
||||||
if(!iter.is_number()) {
|
|
||||||
printf("Root should be number\n");
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
if(iter.is_integer()) {
|
|
||||||
int64_t x = iter.get_integer();
|
|
||||||
int power = 0;
|
|
||||||
while(x > 1) {
|
|
||||||
if((x % 2) != 0) {
|
|
||||||
printf("failed to parse %s. \n", buf);
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
x = x / 2;
|
|
||||||
power ++;
|
|
||||||
}
|
|
||||||
if(power != i) {
|
|
||||||
printf("failed to parse %s. \n", buf);
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
} else if(iter.is_unsigned_integer()) {
|
|
||||||
uint64_t x = iter.get_unsigned_integer();
|
|
||||||
int power = 0;
|
|
||||||
while(x > 1) {
|
|
||||||
if((x % 2) != 0) {
|
|
||||||
printf("failed to parse %s. \n", buf);
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
x = x / 2;
|
|
||||||
power ++;
|
|
||||||
}
|
|
||||||
if(power != i) {
|
|
||||||
printf("failed to parse %s. \n", buf);
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
double x = iter.get_double();
|
|
||||||
int ulp = f64_ulp_dist(x,expected);
|
|
||||||
if(ulp > maxulp) maxulp = ulp;
|
|
||||||
if(ulp > 0) {
|
|
||||||
printf("failed to parse %s. ULP = %d i = %d \n", buf, ulp, i);
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
printf("Powers of 2 can be parsed, maxulp = %d.\n", maxulp);
|
printf("Powers of 2 can be parsed, maxulp = %d.\n", maxulp);
|
||||||
return true;
|
return true;
|
||||||
|
@ -214,62 +162,20 @@ namespace number_tests {
|
||||||
bool powers_of_ten() {
|
bool powers_of_ten() {
|
||||||
std::cout << __func__ << std::endl;
|
std::cout << __func__ << std::endl;
|
||||||
char buf[1024];
|
char buf[1024];
|
||||||
simdjson::ParsedJson pj;
|
simdjson::document::parser parser;
|
||||||
for (int i = -1000000; i <= 308; ++i) {// large negative values should be zero.
|
for (int i = -1000000; i <= 308; ++i) {// large negative values should be zero.
|
||||||
auto n = sprintf(buf,"1e%d", i);
|
auto n = sprintf(buf,"1e%d", i);
|
||||||
buf[n] = '\0';
|
buf[n] = '\0';
|
||||||
fflush(NULL);
|
fflush(NULL);
|
||||||
auto error = simdjson::json_parse(buf, n, pj);
|
|
||||||
if (error) {
|
auto [actual, error] = parser.parse(buf, n).root().as_double();
|
||||||
printf("Could not parse '%s': %s\n", buf, simdjson::error_message(error).c_str());
|
if (error) { std::cerr << error << std::endl; return false; }
|
||||||
|
double expected = ((i >= -307) ? testing_power_of_ten[i + 307]: std::pow(10, i));
|
||||||
|
int ulp = (int) f64_ulp_dist(actual, expected);
|
||||||
|
if(ulp > 0) {
|
||||||
|
std::cerr << "JSON '" << buf << " parsed to " << actual << " instead of " << expected << std::endl;
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
simdjson::ParsedJson::Iterator iter(pj);
|
|
||||||
if(!iter.is_number()) {
|
|
||||||
printf("Root should be number\n");
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
if(iter.is_integer()) {
|
|
||||||
int64_t x = iter.get_integer();
|
|
||||||
int power = 0;
|
|
||||||
while(x > 1) {
|
|
||||||
if((x % 10) != 0) {
|
|
||||||
printf("failed to parse %s. \n", buf);
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
x = x / 10;
|
|
||||||
power ++;
|
|
||||||
}
|
|
||||||
if(power != i) {
|
|
||||||
printf("failed to parse %s. \n", buf);
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
} else if(iter.is_unsigned_integer()) {
|
|
||||||
uint64_t x = iter.get_unsigned_integer();
|
|
||||||
int power = 0;
|
|
||||||
while(x > 1) {
|
|
||||||
if((x % 10) != 0) {
|
|
||||||
printf("failed to parse %s. \n", buf);
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
x = x / 10;
|
|
||||||
power ++;
|
|
||||||
}
|
|
||||||
if(power != i) {
|
|
||||||
printf("failed to parse %s. \n", buf);
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
double x = iter.get_double();
|
|
||||||
double expected = ((i >= -307) ? testing_power_of_ten[i + 307]: std::pow(10, i));
|
|
||||||
int ulp = (int) f64_ulp_dist(x, expected);
|
|
||||||
if(ulp > 0) {
|
|
||||||
printf("failed to parse %s. \n", buf);
|
|
||||||
printf("actual: %.20g expected: %.20g \n", x, expected);
|
|
||||||
printf("ULP: %d \n", ulp);
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
printf("Powers of 10 can be parsed.\n");
|
printf("Powers of 10 can be parsed.\n");
|
||||||
return true;
|
return true;
|
||||||
|
@ -285,10 +191,11 @@ namespace document_tests {
|
||||||
// adversarial example that once triggred overruns, see https://github.com/lemire/simdjson/issues/345
|
// adversarial example that once triggred overruns, see https://github.com/lemire/simdjson/issues/345
|
||||||
bool bad_example() {
|
bool bad_example() {
|
||||||
std::cout << __func__ << std::endl;
|
std::cout << __func__ << std::endl;
|
||||||
std::string badjson = "[7,7,7,7,6,7,7,7,6,7,7,6,[7,7,7,7,6,7,7,7,6,7,7,6,7,7,7,7,7,7,6";
|
simdjson::padded_string badjson = "[7,7,7,7,6,7,7,7,6,7,7,6,[7,7,7,7,6,7,7,7,6,7,7,6,7,7,7,7,7,7,6"_padded;
|
||||||
simdjson::document::parser parser = simdjson::build_parsed_json(badjson);
|
simdjson::document::parser parser;
|
||||||
|
parser.parse(badjson);
|
||||||
if(parser.is_valid()) {
|
if(parser.is_valid()) {
|
||||||
printf("This json should not be valid %s.\n", badjson.c_str());
|
printf("This json should not be valid %s.\n", badjson.data());
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
return true;
|
return true;
|
||||||
|
@ -296,7 +203,7 @@ namespace document_tests {
|
||||||
// returns true if successful
|
// returns true if successful
|
||||||
bool stable_test() {
|
bool stable_test() {
|
||||||
std::cout << __func__ << std::endl;
|
std::cout << __func__ << std::endl;
|
||||||
std::string json = "{"
|
simdjson::padded_string json = "{"
|
||||||
"\"Image\":{"
|
"\"Image\":{"
|
||||||
"\"Width\":800,"
|
"\"Width\":800,"
|
||||||
"\"Height\":600,"
|
"\"Height\":600,"
|
||||||
|
@ -309,20 +216,17 @@ namespace document_tests {
|
||||||
"\"Animated\":false,"
|
"\"Animated\":false,"
|
||||||
"\"IDs\":[116,943.3,234,38793]"
|
"\"IDs\":[116,943.3,234,38793]"
|
||||||
"}"
|
"}"
|
||||||
"}";
|
"}"_padded;
|
||||||
simdjson::document::parser parser = simdjson::build_parsed_json(json);
|
simdjson::document::parser parser;
|
||||||
std::ostringstream myStream;
|
std::ostringstream myStream;
|
||||||
if( ! parser.print_json(myStream) ) {
|
myStream << parser.parse(json);
|
||||||
std::cout << "cannot print it out? " << std::endl;
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
std::string newjson = myStream.str();
|
std::string newjson = myStream.str();
|
||||||
if(json != newjson) {
|
if(static_cast<std::string>(json) != newjson) {
|
||||||
std::cout << "serialized json differs!" << std::endl;
|
std::cout << "serialized json differs!" << std::endl;
|
||||||
std::cout << json << std::endl;
|
std::cout << static_cast<std::string>(json) << std::endl;
|
||||||
std::cout << newjson << std::endl;
|
std::cout << newjson << std::endl;
|
||||||
}
|
}
|
||||||
return newjson == json;
|
return newjson == static_cast<std::string>(json);
|
||||||
}
|
}
|
||||||
// returns true if successful
|
// returns true if successful
|
||||||
bool skyprophet_test() {
|
bool skyprophet_test() {
|
||||||
|
@ -363,14 +267,16 @@ namespace document_tests {
|
||||||
fflush(NULL);
|
fflush(NULL);
|
||||||
}
|
}
|
||||||
counter++;
|
counter++;
|
||||||
auto ok1 = simdjson::json_parse(rec.c_str(), rec.length(), parser);
|
auto [doc1, res1] = parser.parse(rec.c_str(), rec.length());
|
||||||
if (ok1 != 0 || !parser.is_valid()) {
|
if (res1 != simdjson::error_code::SUCCESS) {
|
||||||
printf("Something is wrong in skyprophet_test: %s.\n", rec.c_str());
|
printf("Something is wrong in skyprophet_test: %s.\n", rec.c_str());
|
||||||
|
printf("Parsing failed. Error is %s\n", simdjson::error_message(res1));
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
auto ok2 = simdjson::json_parse(rec, parser);
|
auto [doc2, res2] = parser.parse(rec.c_str(), rec.length());
|
||||||
if (ok2 != 0 || !parser.is_valid()) {
|
if (res2 != simdjson::error_code::SUCCESS) {
|
||||||
printf("Something is wrong in skyprophet_test: %s.\n", rec.c_str());
|
printf("Something is wrong in skyprophet_test: %s.\n", rec.c_str());
|
||||||
|
printf("Parsing failed. Error is %s\n", simdjson::error_message(res2));
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -665,71 +571,64 @@ namespace dom_api_tests {
|
||||||
using namespace std;
|
using namespace std;
|
||||||
using namespace simdjson;
|
using namespace simdjson;
|
||||||
|
|
||||||
|
SIMDJSON_PUSH_DISABLE_WARNINGS
|
||||||
|
SIMDJSON_DISABLE_DEPRECATED_WARNING
|
||||||
// returns true if successful
|
// returns true if successful
|
||||||
bool document_iterator_test() {
|
bool ParsedJson_Iterator_test() {
|
||||||
std::cout << "Running " << __func__ << std::endl;
|
std::cout << "Running " << __func__ << std::endl;
|
||||||
std::string json = "{"
|
simdjson::padded_string json = R"({
|
||||||
"\"Image\": {"
|
"Image": {
|
||||||
"\"Width\": 800,"
|
"Width": 800,
|
||||||
"\"Height\": 600,"
|
"Height": 600,
|
||||||
"\"Title\": \"View from 15th Floor\","
|
"Title": "View from 15th Floor",
|
||||||
"\"Thumbnail\": {"
|
"Thumbnail": {
|
||||||
" \"Url\": \"http://www.example.com/image/481989943\","
|
"Url": "http://www.example.com/image/481989943",
|
||||||
" \"Height\": 125,"
|
"Height": 125,
|
||||||
" \"Width\": 100"
|
"Width": 100
|
||||||
"},"
|
},
|
||||||
"\"Animated\" : false,"
|
"Animated" : false,
|
||||||
"\"IDs\": [116, 943, 234, 38793]"
|
"IDs": [116, 943, 234, 38793]
|
||||||
"}"
|
}
|
||||||
"}";
|
})"_padded;
|
||||||
|
simdjson::ParsedJson pj = build_parsed_json(json);
|
||||||
ParsedJson pj = build_parsed_json(json);
|
|
||||||
if (pj.error) {
|
if (pj.error) {
|
||||||
printf("Could not parse '%s': %s\n", json.data(), simdjson::error_message(pj.error));
|
printf("Could not parse '%s': %s\n", json.data(), simdjson::error_message(pj.error));
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
simdjson::ParsedJson::Iterator iter(pj);
|
simdjson::ParsedJson::Iterator iter(pj);
|
||||||
if(!iter.is_object()) {
|
if (!iter.is_object()) {
|
||||||
printf("Root should be object\n");
|
printf("Root should be object\n");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
if(iter.move_to_key("bad key")) {
|
if (iter.move_to_key("bad key")) {
|
||||||
printf("We should not move to a non-existing key\n");
|
printf("We should not move to a non-existing key\n");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
if(!iter.is_object()) {
|
if (!iter.is_object()) {
|
||||||
printf("We should have remained at the object.\n");
|
printf("We should have remained at the object.\n");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
if(iter.move_to_key_insensitive("bad key")) {
|
if (iter.move_to_key_insensitive("bad key")) {
|
||||||
printf("We should not move to a non-existing key\n");
|
printf("We should not move to a non-existing key\n");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
if(!iter.is_object()) {
|
if (!iter.is_object()) {
|
||||||
printf("We should have remained at the object.\n");
|
printf("We should have remained at the object.\n");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
if(iter.move_to_key("bad key", 7)) {
|
if (!iter.down()) {
|
||||||
printf("We should not move to a non-existing key\n");
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
if(!iter.is_object()) {
|
|
||||||
printf("We should have remained at the object.\n");
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
if(!iter.down()) {
|
|
||||||
printf("Root should not be emtpy\n");
|
printf("Root should not be emtpy\n");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
if(!iter.is_string()) {
|
if (!iter.is_string()) {
|
||||||
printf("Object should start with string key\n");
|
printf("Object should start with string key\n");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
if(iter.prev()) {
|
if (iter.prev()) {
|
||||||
printf("We should not be able to go back from the start of the scope.\n");
|
printf("We should not be able to go back from the start of the scope.\n");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
if(strcmp(iter.get_string(),"Image")!=0) {
|
if (strcmp(iter.get_string(),"Image")!=0) {
|
||||||
printf("There should be a single key, image.\n");
|
printf("There should be a single key, image.\n");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
@ -750,31 +649,32 @@ namespace dom_api_tests {
|
||||||
printf("We should go back to the key.\n");
|
printf("We should go back to the key.\n");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
if(strcmp(iter.get_string(),"Width")!=0) {
|
if (strcmp(iter.get_string(),"Width")!=0) {
|
||||||
printf("There should be a key Width.\n");
|
printf("There should be a key Width.\n");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
if(!iter.up()) {
|
if (!iter.up()) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
if(!iter.move_to_key("IDs")) {
|
if (!iter.move_to_key("IDs")) {
|
||||||
printf("We should be able to move to an existing key\n");
|
printf("We should be able to move to an existing key\n");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
if(!iter.is_array()) {
|
if (!iter.is_array()) {
|
||||||
printf("Value of IDs should be array, it is %c \n", iter.get_type());
|
printf("Value of IDs should be array, it is %c \n", iter.get_type());
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
if(iter.move_to_index(4)) {
|
if (iter.move_to_index(4)) {
|
||||||
printf("We should not be able to move to a non-existing index\n");
|
printf("We should not be able to move to a non-existing index\n");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
if(!iter.is_array()) {
|
if (!iter.is_array()) {
|
||||||
printf("We should have remained at the array\n");
|
printf("We should have remained at the array\n");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
SIMDJSON_POP_DISABLE_WARNINGS
|
||||||
|
|
||||||
bool object_iterator() {
|
bool object_iterator() {
|
||||||
std::cout << "Running " << __func__ << std::endl;
|
std::cout << "Running " << __func__ << std::endl;
|
||||||
|
@ -1180,7 +1080,7 @@ namespace dom_api_tests {
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
bool run() {
|
bool run() {
|
||||||
return document_iterator_test() &&
|
return ParsedJson_Iterator_test() &&
|
||||||
object_iterator() &&
|
object_iterator() &&
|
||||||
array_iterator() &&
|
array_iterator() &&
|
||||||
object_iterator_empty() &&
|
object_iterator_empty() &&
|
||||||
|
|
|
@ -32,19 +32,17 @@ template <typename T>
|
||||||
static void parse_and_validate(const std::string src, T expected) {
|
static void parse_and_validate(const std::string src, T expected) {
|
||||||
std::cout << "src: " << src << ", ";
|
std::cout << "src: " << src << ", ";
|
||||||
const padded_string pstr{src};
|
const padded_string pstr{src};
|
||||||
auto json = build_parsed_json(pstr);
|
simdjson::document::parser parser;
|
||||||
|
|
||||||
ASSERT(json.is_valid());
|
|
||||||
ParsedJson::Iterator it{json};
|
|
||||||
ASSERT(it.down());
|
|
||||||
ASSERT(it.next());
|
|
||||||
bool result;
|
bool result;
|
||||||
if constexpr (std::is_same<int64_t, T>::value) {
|
if constexpr (std::is_same<int64_t, T>::value) {
|
||||||
const auto actual = it.get_integer();
|
auto [actual, error] = parser.parse(pstr).as_object()["key"].as_int64_t();
|
||||||
result = expected == actual;
|
if (error) { std::cerr << error << std::endl; abort(); }
|
||||||
|
result = (expected == actual);
|
||||||
} else {
|
} else {
|
||||||
const auto actual = it.get_unsigned_integer();
|
auto [actual, error] = parser.parse(pstr).as_object()["key"].as_uint64_t();
|
||||||
result = expected == actual;
|
if (error) { std::cerr << error << std::endl; abort(); }
|
||||||
|
result = (expected == actual);
|
||||||
}
|
}
|
||||||
std::cout << std::boolalpha << "test: " << result << std::endl;
|
std::cout << std::boolalpha << "test: " << result << std::endl;
|
||||||
if(!result) {
|
if(!result) {
|
||||||
|
@ -56,29 +54,21 @@ static void parse_and_validate(const std::string src, T expected) {
|
||||||
static bool parse_and_check_signed(const std::string src) {
|
static bool parse_and_check_signed(const std::string src) {
|
||||||
std::cout << "src: " << src << ", expecting signed" << std::endl;
|
std::cout << "src: " << src << ", expecting signed" << std::endl;
|
||||||
const padded_string pstr{src};
|
const padded_string pstr{src};
|
||||||
auto json = build_parsed_json(pstr);
|
simdjson::document::parser parser;
|
||||||
|
auto [value, error] = parser.parse(pstr).as_object()["key"];
|
||||||
ASSERT(json.is_valid());
|
if (error) { std::cerr << error << std::endl; abort(); }
|
||||||
ParsedJson::Iterator it{json};
|
return value.is_integer() && value.is_number();
|
||||||
ASSERT(it.down());
|
|
||||||
ASSERT(it.next());
|
|
||||||
return it.is_integer() && it.is_number();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static bool parse_and_check_unsigned(const std::string src) {
|
static bool parse_and_check_unsigned(const std::string src) {
|
||||||
std::cout << "src: " << src << ", expecting unsigned" << std::endl;
|
std::cout << "src: " << src << ", expecting signed" << std::endl;
|
||||||
const padded_string pstr{src};
|
const padded_string pstr{src};
|
||||||
auto json = build_parsed_json(pstr);
|
simdjson::document::parser parser;
|
||||||
|
auto [value, error] = parser.parse(pstr).as_object()["key"];
|
||||||
ASSERT(json.is_valid());
|
if (error) { std::cerr << error << std::endl; abort(); }
|
||||||
ParsedJson::Iterator it{json};
|
return value.is_unsigned_integer() && value.is_number();
|
||||||
ASSERT(it.down());
|
|
||||||
ASSERT(it.next());
|
|
||||||
return it.is_unsigned_integer() && it.is_number();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
int main() {
|
int main() {
|
||||||
using std::numeric_limits;
|
using std::numeric_limits;
|
||||||
constexpr auto int64_max = numeric_limits<int64_t>::max();
|
constexpr auto int64_max = numeric_limits<int64_t>::max();
|
||||||
|
|
|
@ -69,20 +69,20 @@ bool validate(const char *dirname) {
|
||||||
std::cerr << "Could not load the file " << fullpath << std::endl;
|
std::cerr << "Could not load the file " << fullpath << std::endl;
|
||||||
return EXIT_FAILURE;
|
return EXIT_FAILURE;
|
||||||
}
|
}
|
||||||
simdjson::ParsedJson pj;
|
simdjson::document::parser parser;
|
||||||
|
auto [doc, errorcode] = parser.parse(p);
|
||||||
++how_many;
|
++how_many;
|
||||||
const int parse_res = json_parse(p, pj);
|
printf("%s\n", errorcode == simdjson::error_code::SUCCESS ? "ok" : "invalid");
|
||||||
printf("%s\n", parse_res == 0 ? "ok" : "invalid");
|
|
||||||
if (contains("EXCLUDE", name)) {
|
if (contains("EXCLUDE", name)) {
|
||||||
// skipping
|
// skipping
|
||||||
how_many--;
|
how_many--;
|
||||||
} else if (starts_with("pass", name) && parse_res != 0) {
|
} else if (starts_with("pass", name) && errorcode != simdjson::error_code::SUCCESS) {
|
||||||
is_file_as_expected[i] = false;
|
is_file_as_expected[i] = false;
|
||||||
printf("warning: file %s should pass but it fails. Error is: %s\n",
|
printf("warning: file %s should pass but it fails. Error is: %s\n",
|
||||||
name, simdjson::error_message(parse_res).data());
|
name, simdjson::error_message(errorcode));
|
||||||
printf("size of file in bytes: %zu \n", p.size());
|
printf("size of file in bytes: %zu \n", p.size());
|
||||||
everything_fine = false;
|
everything_fine = false;
|
||||||
} else if (starts_with("fail", name) && parse_res == 0) {
|
} else if (starts_with("fail", name) && errorcode == simdjson::error_code::SUCCESS) {
|
||||||
is_file_as_expected[i] = false;
|
is_file_as_expected[i] = false;
|
||||||
printf("warning: file %s should fail but it passes.\n", name);
|
printf("warning: file %s should fail but it passes.\n", name);
|
||||||
printf("size of file in bytes: %zu \n", p.size());
|
printf("size of file in bytes: %zu \n", p.size());
|
||||||
|
|
|
@ -174,12 +174,13 @@ bool validate(const char *dirname) {
|
||||||
return EXIT_FAILURE;
|
return EXIT_FAILURE;
|
||||||
}
|
}
|
||||||
// terrible hack but just to get it working
|
// terrible hack but just to get it working
|
||||||
simdjson::ParsedJson pj;
|
|
||||||
float_count = 0;
|
float_count = 0;
|
||||||
int_count = 0;
|
int_count = 0;
|
||||||
invalid_count = 0;
|
invalid_count = 0;
|
||||||
total_count += float_count + int_count + invalid_count;
|
total_count += float_count + int_count + invalid_count;
|
||||||
bool isok = json_parse(p, pj);
|
simdjson::document::parser parser;
|
||||||
|
auto [doc, err] = parser.parse(p);
|
||||||
|
bool isok = (err == simdjson::error_code::SUCCESS);
|
||||||
if (int_count + float_count + invalid_count > 0) {
|
if (int_count + float_count + invalid_count > 0) {
|
||||||
printf("File %40s %s --- integers: %10zu floats: %10zu invalid: %10zu "
|
printf("File %40s %s --- integers: %10zu floats: %10zu invalid: %10zu "
|
||||||
"total numbers: %10zu \n",
|
"total numbers: %10zu \n",
|
||||||
|
|
|
@ -1,3 +1,4 @@
|
||||||
|
// This file is not part of our main, regular tests.
|
||||||
#include "../singleheader/simdjson.h"
|
#include "../singleheader/simdjson.h"
|
||||||
#include <iostream>
|
#include <iostream>
|
||||||
|
|
||||||
|
@ -6,13 +7,10 @@ using namespace simdjson;
|
||||||
int main() {
|
int main() {
|
||||||
const char *filename = JSON_TEST_PATH;
|
const char *filename = JSON_TEST_PATH;
|
||||||
padded_string p = get_corpus(filename);
|
padded_string p = get_corpus(filename);
|
||||||
ParsedJson pj = build_parsed_json(p); // do the parsing
|
document::parser parser;
|
||||||
if (!pj.is_valid()) {
|
auto [doc, error] = parser.parse(p);
|
||||||
return EXIT_FAILURE;
|
if(error) {
|
||||||
}
|
std::cerr << error << std::endl;
|
||||||
const int res = json_parse(p, pj);
|
|
||||||
if (res) {
|
|
||||||
std::cerr << error_message(res) << std::endl;
|
|
||||||
return EXIT_FAILURE;
|
return EXIT_FAILURE;
|
||||||
}
|
}
|
||||||
return EXIT_SUCCESS;
|
return EXIT_SUCCESS;
|
||||||
|
|
|
@ -338,7 +338,6 @@ bool validate(const char *dirname) {
|
||||||
std::cerr << "Could not load the file " << fullpath << std::endl;
|
std::cerr << "Could not load the file " << fullpath << std::endl;
|
||||||
return EXIT_FAILURE;
|
return EXIT_FAILURE;
|
||||||
}
|
}
|
||||||
simdjson::ParsedJson pj;
|
|
||||||
big_buffer = (char *)malloc(p.size());
|
big_buffer = (char *)malloc(p.size());
|
||||||
if (big_buffer == NULL) {
|
if (big_buffer == NULL) {
|
||||||
std::cerr << "can't allocate memory" << std::endl;
|
std::cerr << "can't allocate memory" << std::endl;
|
||||||
|
@ -348,7 +347,9 @@ bool validate(const char *dirname) {
|
||||||
good_string = 0;
|
good_string = 0;
|
||||||
total_string_length = 0;
|
total_string_length = 0;
|
||||||
empty_string = 0;
|
empty_string = 0;
|
||||||
bool isok = json_parse(p, pj);
|
simdjson::document::parser parser;
|
||||||
|
auto [doc, err] = parser.parse(p);
|
||||||
|
bool isok = (err == simdjson::error_code::SUCCESS);
|
||||||
free(big_buffer);
|
free(big_buffer);
|
||||||
if (good_string > 0) {
|
if (good_string > 0) {
|
||||||
printf("File %40s %s --- bad strings: %10zu \tgood strings: %10zu\t "
|
printf("File %40s %s --- bad strings: %10zu \tgood strings: %10zu\t "
|
||||||
|
|
|
@ -1,39 +1,6 @@
|
||||||
#include "simdjson.h"
|
#include "simdjson.h"
|
||||||
#include <iostream>
|
#include <iostream>
|
||||||
|
|
||||||
void compute_dump(simdjson::ParsedJson::Iterator &pjh) {
|
|
||||||
if (pjh.is_object()) {
|
|
||||||
std::cout << "{";
|
|
||||||
if (pjh.down()) {
|
|
||||||
pjh.print(std::cout); // must be a string
|
|
||||||
std::cout << ":";
|
|
||||||
pjh.next();
|
|
||||||
compute_dump(pjh); // let us recurse
|
|
||||||
while (pjh.next()) {
|
|
||||||
std::cout << ",";
|
|
||||||
pjh.print(std::cout);
|
|
||||||
std::cout << ":";
|
|
||||||
pjh.next();
|
|
||||||
compute_dump(pjh); // let us recurse
|
|
||||||
}
|
|
||||||
pjh.up();
|
|
||||||
}
|
|
||||||
std::cout << "}";
|
|
||||||
} else if (pjh.is_array()) {
|
|
||||||
std::cout << "[";
|
|
||||||
if (pjh.down()) {
|
|
||||||
compute_dump(pjh); // let us recurse
|
|
||||||
while (pjh.next()) {
|
|
||||||
std::cout << ",";
|
|
||||||
compute_dump(pjh); // let us recurse
|
|
||||||
}
|
|
||||||
pjh.up();
|
|
||||||
}
|
|
||||||
std::cout << "]";
|
|
||||||
} else {
|
|
||||||
pjh.print(std::cout); // just print the lone value
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
int main(int argc, char *argv[]) {
|
int main(int argc, char *argv[]) {
|
||||||
if (argc < 3) {
|
if (argc < 3) {
|
||||||
|
|
Loading…
Reference in New Issue