aboutsummaryrefslogtreecommitdiffstats
path: root/subprojects/exess/test/test_double.c
diff options
context:
space:
mode:
Diffstat (limited to 'subprojects/exess/test/test_double.c')
-rw-r--r--subprojects/exess/test/test_double.c246
1 files changed, 246 insertions, 0 deletions
diff --git a/subprojects/exess/test/test_double.c b/subprojects/exess/test/test_double.c
new file mode 100644
index 00000000..d4901c5a
--- /dev/null
+++ b/subprojects/exess/test/test_double.c
@@ -0,0 +1,246 @@
+/*
+ Copyright 2011-2021 David Robillard <d@drobilla.net>
+
+ Permission to use, copy, modify, and/or distribute this software for any
+ purpose with or without fee is hereby granted, provided that the above
+ copyright notice and this permission notice appear in all copies.
+
+ THIS SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
+ WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
+ MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
+ ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
+ WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
+ ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
+ OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
+*/
+
+#undef NDEBUG
+
+#include "float_test_data.h"
+#include "int_test_data.h"
+#include "num_test_utils.h"
+#include "string_utils.h"
+
+#include "exess/exess.h"
+
+#include <assert.h>
+#include <float.h>
+#include <math.h>
+#include <stdint.h>
+#include <stdio.h>
+#include <string.h>
+
+static void
+check_read(const char* const string,
+ const ExessStatus expected_status,
+ const double expected_value,
+ const size_t expected_count)
+{
+ double value = (double)NAN;
+ const ExessResult r = exess_read_double(&value, string);
+
+ assert(r.status == expected_status);
+ assert(r.count == expected_count);
+ assert(double_matches(value, expected_value));
+}
+
+static void
+test_read_double(void)
+{
+ // Limits
+ check_read("-1.7976931348623157E308", EXESS_SUCCESS, -DBL_MAX, 23);
+ check_read("-2.2250738585072014E-308", EXESS_SUCCESS, -DBL_MIN, 24);
+ check_read("2.2250738585072014E-308", EXESS_SUCCESS, DBL_MIN, 23);
+ check_read("1.7976931348623157E308", EXESS_SUCCESS, DBL_MAX, 22);
+
+ // Beyond limits
+ check_read("1e-326", EXESS_SUCCESS, 0.0, 6);
+ check_read("12345678901234567123", EXESS_SUCCESS, 12345678901234567000.0, 20);
+ check_read("1e309", EXESS_SUCCESS, (double)INFINITY, 5);
+
+ // Non-canonical form
+ check_read("+1E3", EXESS_SUCCESS, 1e3, 4);
+ check_read("1E+3", EXESS_SUCCESS, 1e3, 4);
+ check_read("+1.5E3", EXESS_SUCCESS, 1.5e3, 6);
+ check_read(".5E3", EXESS_SUCCESS, 0.5e3, 4);
+ check_read("+.5E3", EXESS_SUCCESS, 0.5e3, 5);
+ check_read("-.5E3", EXESS_SUCCESS, -0.5e3, 5);
+ check_read("1.E3", EXESS_SUCCESS, 1e3, 4);
+ check_read("+1.E3", EXESS_SUCCESS, 1e3, 5);
+ check_read("-1.E3", EXESS_SUCCESS, -1e3, 5);
+
+ // Special values
+ check_read("NaN", EXESS_SUCCESS, (double)NAN, 3);
+ check_read("-INF", EXESS_SUCCESS, (double)-INFINITY, 4);
+ check_read("-0.0E0", EXESS_SUCCESS, -0.0, 6);
+ check_read("0.0E0", EXESS_SUCCESS, 0.0, 5);
+ check_read("+0.0E0", EXESS_SUCCESS, 0.0, 6);
+ check_read("INF", EXESS_SUCCESS, (double)INFINITY, 3);
+ check_read("+INF", EXESS_SUCCESS, (double)INFINITY, 4);
+
+ // No exponent
+ check_read("1", EXESS_SUCCESS, 1.0, 1);
+ check_read("2.3", EXESS_SUCCESS, 2.3, 3);
+ check_read("-4.5", EXESS_SUCCESS, -4.5, 4);
+
+ // Leading whitespace
+ check_read(" \f\n\r\t\v1.2", EXESS_SUCCESS, 1.2, 9);
+
+ // Garbage
+ check_read("true", EXESS_EXPECTED_DIGIT, (double)NAN, 0);
+ check_read("+true", EXESS_EXPECTED_DIGIT, (double)NAN, 1);
+ check_read("-false", EXESS_EXPECTED_DIGIT, (double)NAN, 1);
+ check_read("1.0eX", EXESS_EXPECTED_DIGIT, (double)NAN, 4);
+ check_read("1.0EX", EXESS_EXPECTED_DIGIT, (double)NAN, 4);
+}
+
+/// Check that `str` is a canonical xsd:double string
+static void
+check_canonical(const char* const str)
+{
+ if (!strcmp(str, "NaN") || !strcmp(str, "-INF") || !strcmp(str, "INF")) {
+ return;
+ }
+
+ assert(strlen(str) > 4); // Shortest possible is something like 1.2E3
+ assert(str[0] == '-' || is_digit(str[0]));
+
+ const int first_digit = str[0] == '-' ? 1 : 0;
+ assert(is_digit(str[first_digit]));
+ assert(str[first_digit + 1] == '.');
+ assert(is_digit(str[first_digit + 2]));
+
+ const char* const e = strchr(str, 'E');
+ assert(e);
+ assert(*e == 'E');
+ assert(*(e + 1) == '-' || is_digit(*(e + 1)));
+}
+
+static void
+check_write(const double value,
+ const ExessStatus expected_status,
+ const size_t buf_size,
+ const char* const expected_string)
+{
+ char buf[EXESS_MAX_DOUBLE_LENGTH + 1] = {
+ 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, //
+ 11, 12, 13, 14, 15, 16, 17, 18, 19, 20, //
+ 21, 22, 23};
+
+ assert(buf_size <= sizeof(buf));
+
+ const ExessResult r = exess_write_double(value, buf_size, buf);
+ assert(r.status == expected_status);
+ assert(r.count == strlen(buf));
+ assert(!strcmp(buf, expected_string));
+ assert((r.status && r.status != EXESS_NO_SPACE) ||
+ exess_write_double(value, 0, NULL).count == r.count);
+
+ if (!r.status) {
+ check_canonical(buf);
+ }
+}
+
+static void
+test_write_double(void)
+{
+ check_write((double)NAN, EXESS_SUCCESS, 4, "NaN");
+ check_write(DBL_MIN, EXESS_SUCCESS, 24, "2.2250738585072014E-308");
+ check_write(-0.0, EXESS_SUCCESS, 7, "-0.0E0");
+ check_write(0.0, EXESS_SUCCESS, 6, "0.0E0");
+ check_write(DBL_MAX, EXESS_SUCCESS, 23, "1.7976931348623157E308");
+
+ /* check_write((double)NAN, EXESS_NO_SPACE, 3, ""); */
+ /* check_write(DBL_MIN, EXESS_SUCCESS, 24, "2.2250738585072014E-308"); */
+ /* check_write(-0.0, EXESS_SUCCESS, 7, "-0.0E0"); */
+ /* check_write(0.0, EXESS_SUCCESS, 6, "0.0E0"); */
+ /* check_write(DBL_MAX, EXESS_SUCCESS, 23, "1.7976931348623157E308"); */
+}
+
+static void
+check_round_trip(const double value)
+{
+ double parsed_value = 0.0;
+
+ char buf[EXESS_MAX_DOUBLE_LENGTH + 1] = {
+ 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, //
+ 11, 12, 13, 14, 15, 16, 17, 18, 19, 20, //
+ 21, 22, 23};
+
+ assert(!exess_write_double(value, sizeof(buf), buf).status);
+ assert(!exess_read_double(&parsed_value, buf).status);
+ assert(double_matches(parsed_value, value));
+}
+
+static void
+test_round_trip(const ExessNumTestOptions opts)
+{
+ check_round_trip((double)NAN);
+ check_round_trip(-(double)INFINITY);
+ check_round_trip(DBL_MIN);
+ check_round_trip(-0.0);
+ check_round_trip(0.0);
+ check_round_trip(DBL_MAX);
+ check_round_trip((double)INFINITY);
+
+ check_round_trip(5.0);
+ check_round_trip(50.0);
+ check_round_trip(500000000000000000000.0);
+ check_round_trip(-0.5);
+ check_round_trip(0.5);
+ check_round_trip(0.05);
+ check_round_trip(0.005);
+ check_round_trip(0.00000000000000000005);
+
+ // Normal limits
+ check_round_trip(nextafter(DBL_MIN, (double)INFINITY));
+ check_round_trip(nextafter(DBL_EPSILON, (double)INFINITY));
+ check_round_trip(nextafter(DBL_MAX, -(double)INFINITY));
+
+ // Subnormals
+ check_round_trip(nextafter(0.0, 1.0));
+ check_round_trip(nextafter(nextafter(0.0, 1.0), 2.0));
+ check_round_trip(nextafter(0.0, -1.0));
+ check_round_trip(nextafter(nextafter(0.0, -1.0), -2.0));
+
+ // Various tricky cases
+ check_round_trip(1e23);
+ check_round_trip(6.02951420360127e-309);
+ check_round_trip(9.17857104364115e+288);
+ check_round_trip(2.68248422823759e+22);
+
+ // Powers of two (where the lower boundary is closer)
+ for (int i = -1023; i <= 1023; ++i) {
+ check_round_trip(pow(2, i));
+ }
+
+ fprintf(stderr, "Testing xsd:double randomly with seed %u\n", opts.seed);
+
+ uint64_t rep = opts.seed;
+ for (uint64_t i = 0; i < opts.n_tests; ++i) {
+ rep = lcg64(rep);
+
+ const double value = double_from_rep(rep);
+
+ check_round_trip(nextafter(value, -(double)INFINITY));
+ check_round_trip(value);
+ check_round_trip(nextafter(value, (double)INFINITY));
+
+ print_num_test_progress(i, opts.n_tests);
+ }
+}
+
+int
+main(int argc, char** argv)
+{
+ const ExessNumTestOptions opts = parse_num_test_options(argc, argv);
+ if (opts.error) {
+ return 1;
+ }
+
+ test_read_double();
+ test_write_double();
+ test_round_trip(opts);
+
+ return 0;
+}