aboutsummaryrefslogtreecommitdiff
path: root/tests
diff options
context:
space:
mode:
authorJeroen van Rijn <Kelimion@users.noreply.github.com>2021-09-07 19:47:37 +0200
committerJeroen van Rijn <Kelimion@users.noreply.github.com>2021-09-07 19:51:56 +0200
commitf7601a759b0bd9830fae772bcdab9210097f68bb (patch)
treeca60212798b3fcff6f9387649f3182c1895e60cb /tests
parent49011f5198290e740263b4d54d041b333dc8bc05 (diff)
Move math/big tests under `tests/`.
Diffstat (limited to 'tests')
-rw-r--r--tests/core/build.bat25
-rw-r--r--tests/core/math/big/test.odin390
-rw-r--r--tests/core/math/big/test.py760
3 files changed, 1173 insertions, 2 deletions
diff --git a/tests/core/build.bat b/tests/core/build.bat
index f2a810cac..cda0b078f 100644
--- a/tests/core/build.bat
+++ b/tests/core/build.bat
@@ -1,6 +1,27 @@
@echo off
set COMMON=-show-timings -no-bounds-check -vet -strict-style
set PATH_TO_ODIN==..\..\odin
-python3 download_assets.py
+:python3 download_assets.py
+echo ---
+echo Running core:image tests
+echo ---
%PATH_TO_ODIN% test image %COMMON%
-%PATH_TO_ODIN% test compress %COMMON% \ No newline at end of file
+del image.exe
+echo ---
+echo Running core:compress tests
+echo ---
+%PATH_TO_ODIN% test compress %COMMON%
+del compress.exe
+
+rem math/big tests
+set TEST_ARGS=-fast-tests
+set TEST_ARGS=
+set OUT_NAME=math_big_test_library
+set COMMON=-build-mode:shared -show-timings -no-bounds-check -define:MATH_BIG_EXE=false -vet -strict-style
+echo ---
+echo Running core:math/big tests
+echo ---
+
+%PATH_TO_ODIN% build math/big %COMMON% -o:speed -out:%OUT_NAME%
+python3 math/big/test.py %TEST_ARGS%
+del %OUT_NAME%.* \ No newline at end of file
diff --git a/tests/core/math/big/test.odin b/tests/core/math/big/test.odin
new file mode 100644
index 000000000..baf7b3177
--- /dev/null
+++ b/tests/core/math/big/test.odin
@@ -0,0 +1,390 @@
+//+ignore
+/*
+ Copyright 2021 Jeroen van Rijn <nom@duclavier.com>.
+ Made available under Odin's BSD-3 license.
+
+ An arbitrary precision mathematics implementation in Odin.
+ For the theoretical underpinnings, see Knuth's The Art of Computer Programming, Volume 2, section 4.3.
+ The code started out as an idiomatic source port of libTomMath, which is in the public domain, with thanks.
+
+ This file exports procedures for use with the test.py test suite.
+*/
+package math_big_tests
+
+/*
+ TODO: Write tests for `internal_*` and test reusing parameters with the public implementations.
+*/
+
+import "core:runtime"
+import "core:strings"
+import "core:math/big"
+
+PyRes :: struct {
+ res: cstring,
+ err: big.Error,
+}
+
+@export test_initialize_constants :: proc "c" () -> (res: u64) {
+ context = runtime.default_context()
+ res = u64(big.initialize_constants())
+ //assert(MUL_KARATSUBA_CUTOFF >= 40);
+ return res
+}
+
+@export test_error_string :: proc "c" (err: big.Error) -> (res: cstring) {
+ context = runtime.default_context()
+ es := big.Error_String
+ return strings.clone_to_cstring(es[err], context.temp_allocator)
+}
+
+@export test_add :: proc "c" (a, b: cstring) -> (res: PyRes) {
+ context = runtime.default_context()
+ err: big.Error
+
+ aa, bb, sum := &big.Int{}, &big.Int{}, &big.Int{}
+ defer big.internal_destroy(aa, bb, sum)
+
+ if err = big.atoi(aa, string(a), 16); err != nil { return PyRes{res=":add:atoi(a):", err=err} }
+ if err = big.atoi(bb, string(b), 16); err != nil { return PyRes{res=":add:atoi(b):", err=err} }
+ if bb.used == 1 {
+ if err = #force_inline big.internal_add(sum, aa, bb.digit[0]); err != nil { return PyRes{res=":add:add(sum,a,b):", err=err} }
+ } else {
+ if err = #force_inline big.internal_add(sum, aa, bb); err != nil { return PyRes{res=":add:add(sum,a,b):", err=err} }
+ }
+
+ r: cstring
+ r, err = big.int_itoa_cstring(sum, 16, context.temp_allocator)
+ if err != nil { return PyRes{res=":add:itoa(sum):", err=err} }
+ return PyRes{res = r, err = nil}
+}
+
+@export test_sub :: proc "c" (a, b: cstring) -> (res: PyRes) {
+ context = runtime.default_context()
+ err: big.Error
+
+ aa, bb, sum := &big.Int{}, &big.Int{}, &big.Int{}
+ defer big.internal_destroy(aa, bb, sum)
+
+ if err = big.atoi(aa, string(a), 16); err != nil { return PyRes{res=":sub:atoi(a):", err=err} }
+ if err = big.atoi(bb, string(b), 16); err != nil { return PyRes{res=":sub:atoi(b):", err=err} }
+ if bb.used == 1 {
+ if err = #force_inline big.internal_sub(sum, aa, bb.digit[0]); err != nil { return PyRes{res=":sub:sub(sum,a,b):", err=err} }
+ } else {
+ if err = #force_inline big.internal_sub(sum, aa, bb); err != nil { return PyRes{res=":sub:sub(sum,a,b):", err=err} }
+ }
+
+ r: cstring
+ r, err = big.int_itoa_cstring(sum, 16, context.temp_allocator)
+ if err != nil { return PyRes{res=":sub:itoa(sum):", err=err} }
+ return PyRes{res = r, err = nil}
+}
+
+@export test_mul :: proc "c" (a, b: cstring) -> (res: PyRes) {
+ context = runtime.default_context()
+ err: big.Error
+
+ aa, bb, product := &big.Int{}, &big.Int{}, &big.Int{}
+ defer big.internal_destroy(aa, bb, product)
+
+ if err = big.atoi(aa, string(a), 16); err != nil { return PyRes{res=":mul:atoi(a):", err=err} }
+ if err = big.atoi(bb, string(b), 16); err != nil { return PyRes{res=":mul:atoi(b):", err=err} }
+ if err = #force_inline big.internal_mul(product, aa, bb); err != nil { return PyRes{res=":mul:mul(product,a,b):", err=err} }
+
+ r: cstring
+ r, err = big.int_itoa_cstring(product, 16, context.temp_allocator)
+ if err != nil { return PyRes{res=":mul:itoa(product):", err=err} }
+ return PyRes{res = r, err = nil}
+}
+
+@export test_sqr :: proc "c" (a: cstring) -> (res: PyRes) {
+ context = runtime.default_context()
+ err: big.Error
+
+ aa, square := &big.Int{}, &big.Int{}
+ defer big.internal_destroy(aa, square)
+
+ if err = big.atoi(aa, string(a), 16); err != nil { return PyRes{res=":sqr:atoi(a):", err=err} }
+ if err = #force_inline big.internal_sqr(square, aa); err != nil { return PyRes{res=":sqr:sqr(square,a):", err=err} }
+
+ r: cstring
+ r, err = big.int_itoa_cstring(square, 16, context.temp_allocator)
+ if err != nil { return PyRes{res=":sqr:itoa(square):", err=err} }
+ return PyRes{res = r, err = nil}
+}
+
+/*
+ NOTE(Jeroen): For simplicity, we don't return the quotient and the remainder, just the quotient.
+*/
+@export test_div :: proc "c" (a, b: cstring) -> (res: PyRes) {
+ context = runtime.default_context()
+ err: big.Error
+
+ aa, bb, quotient := &big.Int{}, &big.Int{}, &big.Int{}
+ defer big.internal_destroy(aa, bb, quotient)
+
+ if err = big.atoi(aa, string(a), 16); err != nil { return PyRes{res=":div:atoi(a):", err=err} }
+ if err = big.atoi(bb, string(b), 16); err != nil { return PyRes{res=":div:atoi(b):", err=err} }
+ if err = #force_inline big.internal_div(quotient, aa, bb); err != nil { return PyRes{res=":div:div(quotient,a,b):", err=err} }
+
+ r: cstring
+ r, err = big.int_itoa_cstring(quotient, 16, context.temp_allocator)
+ if err != nil { return PyRes{res=":div:itoa(quotient):", err=err} }
+ return PyRes{res = r, err = nil}
+}
+
+
+/*
+ res = log(a, base)
+*/
+@export test_log :: proc "c" (a: cstring, base := big.DIGIT(2)) -> (res: PyRes) {
+ context = runtime.default_context()
+ err: big.Error
+ l: int
+
+ aa := &big.Int{}
+ defer big.internal_destroy(aa)
+
+ if err = big.atoi(aa, string(a), 16); err != nil { return PyRes{res=":log:atoi(a):", err=err} }
+ if l, err = #force_inline big.internal_log(aa, base); err != nil { return PyRes{res=":log:log(a, base):", err=err} }
+
+ #force_inline big.internal_zero(aa)
+ aa.digit[0] = big.DIGIT(l) & big._MASK
+ aa.digit[1] = big.DIGIT(l) >> big._DIGIT_BITS
+ aa.used = 2
+ big.clamp(aa)
+
+ r: cstring
+ r, err = big.int_itoa_cstring(aa, 16, context.temp_allocator)
+ if err != nil { return PyRes{res=":log:itoa(res):", err=err} }
+ return PyRes{res = r, err = nil}
+}
+
+/*
+ dest = base^power
+*/
+@export test_pow :: proc "c" (base: cstring, power := int(2)) -> (res: PyRes) {
+ context = runtime.default_context()
+ err: big.Error
+
+ dest, bb := &big.Int{}, &big.Int{}
+ defer big.internal_destroy(dest, bb)
+
+ if err = big.atoi(bb, string(base), 16); err != nil { return PyRes{res=":pow:atoi(base):", err=err} }
+ if err = #force_inline big.internal_pow(dest, bb, power); err != nil { return PyRes{res=":pow:pow(dest, base, power):", err=err} }
+
+ r: cstring
+ r, err = big.int_itoa_cstring(dest, 16, context.temp_allocator)
+ if err != nil { return PyRes{res=":log:itoa(res):", err=err} }
+ return PyRes{res = r, err = nil}
+}
+
+/*
+ dest = sqrt(src)
+*/
+@export test_sqrt :: proc "c" (source: cstring) -> (res: PyRes) {
+ context = runtime.default_context()
+ err: big.Error
+
+ src := &big.Int{}
+ defer big.internal_destroy(src)
+
+ if err = big.atoi(src, string(source), 16); err != nil { return PyRes{res=":sqrt:atoi(src):", err=err} }
+ if err = #force_inline big.internal_sqrt(src, src); err != nil { return PyRes{res=":sqrt:sqrt(src):", err=err} }
+
+ r: cstring
+ r, err = big.int_itoa_cstring(src, 16, context.temp_allocator)
+ if err != nil { return PyRes{res=":log:itoa(res):", err=err} }
+ return PyRes{res = r, err = nil}
+}
+
+/*
+ dest = root_n(src, power)
+*/
+@export test_root_n :: proc "c" (source: cstring, power: int) -> (res: PyRes) {
+ context = runtime.default_context()
+ err: big.Error
+
+ src := &big.Int{}
+ defer big.internal_destroy(src)
+
+ if err = big.atoi(src, string(source), 16); err != nil { return PyRes{res=":root_n:atoi(src):", err=err} }
+ if err = #force_inline big.internal_root_n(src, src, power); err != nil { return PyRes{res=":root_n:root_n(src):", err=err} }
+
+ r: cstring
+ r, err = big.int_itoa_cstring(src, 16, context.temp_allocator)
+ if err != nil { return PyRes{res=":root_n:itoa(res):", err=err} }
+ return PyRes{res = r, err = nil}
+}
+
+/*
+ dest = shr_digit(src, digits)
+*/
+@export test_shr_digit :: proc "c" (source: cstring, digits: int) -> (res: PyRes) {
+ context = runtime.default_context()
+ err: big.Error
+
+ src := &big.Int{}
+ defer big.internal_destroy(src)
+
+ if err = big.atoi(src, string(source), 16); err != nil { return PyRes{res=":shr_digit:atoi(src):", err=err} }
+ if err = #force_inline big.internal_shr_digit(src, digits); err != nil { return PyRes{res=":shr_digit:shr_digit(src):", err=err} }
+
+ r: cstring
+ r, err = big.int_itoa_cstring(src, 16, context.temp_allocator)
+ if err != nil { return PyRes{res=":shr_digit:itoa(res):", err=err} }
+ return PyRes{res = r, err = nil}
+}
+
+/*
+ dest = shl_digit(src, digits)
+*/
+@export test_shl_digit :: proc "c" (source: cstring, digits: int) -> (res: PyRes) {
+ context = runtime.default_context()
+ err: big.Error
+
+ src := &big.Int{}
+ defer big.internal_destroy(src)
+
+ if err = big.atoi(src, string(source), 16); err != nil { return PyRes{res=":shl_digit:atoi(src):", err=err} }
+ if err = #force_inline big.internal_shl_digit(src, digits); err != nil { return PyRes{res=":shl_digit:shr_digit(src):", err=err} }
+
+ r: cstring
+ r, err = big.int_itoa_cstring(src, 16, context.temp_allocator)
+ if err != nil { return PyRes{res=":shl_digit:itoa(res):", err=err} }
+ return PyRes{res = r, err = nil}
+}
+
+/*
+ dest = shr(src, bits)
+*/
+@export test_shr :: proc "c" (source: cstring, bits: int) -> (res: PyRes) {
+ context = runtime.default_context()
+ err: big.Error
+
+ src := &big.Int{}
+ defer big.internal_destroy(src)
+
+ if err = big.atoi(src, string(source), 16); err != nil { return PyRes{res=":shr:atoi(src):", err=err} }
+ if err = #force_inline big.internal_shr(src, src, bits); err != nil { return PyRes{res=":shr:shr(src, bits):", err=err} }
+
+ r: cstring
+ r, err = big.int_itoa_cstring(src, 16, context.temp_allocator)
+ if err != nil { return PyRes{res=":shr:itoa(res):", err=err} }
+ return PyRes{res = r, err = nil}
+}
+
+/*
+ dest = shr_signed(src, bits)
+*/
+@export test_shr_signed :: proc "c" (source: cstring, bits: int) -> (res: PyRes) {
+ context = runtime.default_context()
+ err: big.Error
+
+ src := &big.Int{}
+ defer big.internal_destroy(src)
+
+ if err = big.atoi(src, string(source), 16); err != nil { return PyRes{res=":shr_signed:atoi(src):", err=err} }
+ if err = #force_inline big.internal_shr_signed(src, src, bits); err != nil { return PyRes{res=":shr_signed:shr_signed(src, bits):", err=err} }
+
+ r: cstring
+ r, err = big.int_itoa_cstring(src, 16, context.temp_allocator)
+ if err != nil { return PyRes{res=":shr_signed:itoa(res):", err=err} }
+ return PyRes{res = r, err = nil}
+}
+
+/*
+ dest = shl(src, bits)
+*/
+@export test_shl :: proc "c" (source: cstring, bits: int) -> (res: PyRes) {
+ context = runtime.default_context()
+ err: big.Error
+
+ src := &big.Int{}
+ defer big.internal_destroy(src)
+
+ if err = big.atoi(src, string(source), 16); err != nil { return PyRes{res=":shl:atoi(src):", err=err} }
+ if err = #force_inline big.internal_shl(src, src, bits); err != nil { return PyRes{res=":shl:shl(src, bits):", err=err} }
+
+ r: cstring
+ r, err = big.int_itoa_cstring(src, 16, context.temp_allocator)
+ if err != nil { return PyRes{res=":shl:itoa(res):", err=err} }
+ return PyRes{res = r, err = nil}
+}
+
+/*
+ dest = factorial(n)
+*/
+@export test_factorial :: proc "c" (n: int) -> (res: PyRes) {
+ context = runtime.default_context()
+ err: big.Error
+
+ dest := &big.Int{}
+ defer big.internal_destroy(dest)
+
+ if err = #force_inline big.internal_int_factorial(dest, n); err != nil { return PyRes{res=":factorial:factorial(n):", err=err} }
+
+ r: cstring
+ r, err = big.int_itoa_cstring(dest, 16, context.temp_allocator)
+ if err != nil { return PyRes{res=":factorial:itoa(res):", err=err} }
+ return PyRes{res = r, err = nil}
+}
+
+/*
+ dest = gcd(a, b)
+*/
+@export test_gcd :: proc "c" (a, b: cstring) -> (res: PyRes) {
+ context = runtime.default_context()
+ err: big.Error
+
+ ai, bi, dest := &big.Int{}, &big.Int{}, &big.Int{}
+ defer big.internal_destroy(ai, bi, dest)
+
+ if err = big.atoi(ai, string(a), 16); err != nil { return PyRes{res=":gcd:atoi(a):", err=err} }
+ if err = big.atoi(bi, string(b), 16); err != nil { return PyRes{res=":gcd:atoi(b):", err=err} }
+ if err = #force_inline big.internal_int_gcd_lcm(dest, nil, ai, bi); err != nil { return PyRes{res=":gcd:gcd(a, b):", err=err} }
+
+ r: cstring
+ r, err = big.int_itoa_cstring(dest, 16, context.temp_allocator)
+ if err != nil { return PyRes{res=":gcd:itoa(res):", err=err} }
+ return PyRes{res = r, err = nil}
+}
+
+/*
+ dest = lcm(a, b)
+*/
+@export test_lcm :: proc "c" (a, b: cstring) -> (res: PyRes) {
+ context = runtime.default_context()
+ err: big.Error
+
+ ai, bi, dest := &big.Int{}, &big.Int{}, &big.Int{}
+ defer big.internal_destroy(ai, bi, dest)
+
+ if err = big.atoi(ai, string(a), 16); err != nil { return PyRes{res=":lcm:atoi(a):", err=err} }
+ if err = big.atoi(bi, string(b), 16); err != nil { return PyRes{res=":lcm:atoi(b):", err=err} }
+ if err = #force_inline big.internal_int_gcd_lcm(nil, dest, ai, bi); err != nil { return PyRes{res=":lcm:lcm(a, b):", err=err} }
+
+ r: cstring
+ r, err = big.int_itoa_cstring(dest, 16, context.temp_allocator)
+ if err != nil { return PyRes{res=":lcm:itoa(res):", err=err} }
+ return PyRes{res = r, err = nil}
+}
+
+/*
+ dest = lcm(a, b)
+*/
+@export test_is_square :: proc "c" (a: cstring) -> (res: PyRes) {
+ context = runtime.default_context()
+ err: big.Error
+ square: bool
+
+ ai := &big.Int{}
+ defer big.internal_destroy(ai)
+
+ if err = big.atoi(ai, string(a), 16); err != nil { return PyRes{res=":is_square:atoi(a):", err=err} }
+ if square, err = #force_inline big.internal_int_is_square(ai); err != nil { return PyRes{res=":is_square:is_square(a):", err=err} }
+
+ if square {
+ return PyRes{"True", nil}
+ }
+ return PyRes{"False", nil}
+} \ No newline at end of file
diff --git a/tests/core/math/big/test.py b/tests/core/math/big/test.py
new file mode 100644
index 000000000..b819e268e
--- /dev/null
+++ b/tests/core/math/big/test.py
@@ -0,0 +1,760 @@
+#
+# Copyright 2021 Jeroen van Rijn <nom@duclavier.com>.
+# Made available under Odin's BSD-3 license.
+#
+# A BigInt implementation in Odin.
+# For the theoretical underpinnings, see Knuth's The Art of Computer Programming, Volume 2, section 4.3.
+# The code started out as an idiomatic source port of libTomMath, which is in the public domain, with thanks.
+#
+
+from ctypes import *
+from random import *
+import math
+import os
+import platform
+import time
+import gc
+from enum import Enum
+import argparse
+
+
+parser = argparse.ArgumentParser(
+ description = "Odin core:math/big test suite",
+ epilog = "By default we run regression and random tests with preset parameters.",
+ formatter_class = argparse.ArgumentDefaultsHelpFormatter,
+)
+
+#
+# Normally, we report the number of passes and fails. With this option set, we exit at first fail.
+#
+parser.add_argument(
+ "-exit-on-fail",
+ help = "Exit when a test fails",
+ action = "store_true",
+)
+
+#
+# We skip randomized tests altogether if this is set.
+#
+no_random = parser.add_mutually_exclusive_group()
+
+no_random.add_argument(
+ "-no-random",
+ help = "No random tests",
+ action = "store_true",
+)
+
+#
+# Normally we run a given number of cycles on each test.
+# Timed tests budget 1 second per 20_000 bits instead.
+#
+# For timed tests we budget a second per `n` bits and iterate until we hit that time.
+#
+timed_or_fast = no_random.add_mutually_exclusive_group()
+
+timed_or_fast.add_argument(
+ "-timed",
+ type = bool,
+ default = False,
+ help = "Timed tests instead of a preset number of iterations.",
+)
+parser.add_argument(
+ "-timed-bits",
+ type = int,
+ metavar = "BITS",
+ default = 20_000,
+ help = "Timed tests. Every `BITS` worth of input is given a second of running time.",
+)
+
+#
+# For normal tests (non-timed), `-fast-tests` cuts down on the number of iterations.
+#
+timed_or_fast.add_argument(
+ "-fast-tests",
+ help = "Cut down on the number of iterations of each test",
+ action = "store_true",
+)
+
+args = parser.parse_args()
+
+EXIT_ON_FAIL = args.exit_on_fail
+
+#
+# How many iterations of each random test do we want to run?
+#
+BITS_AND_ITERATIONS = [
+ ( 120, 10_000),
+ ( 1_200, 1_000),
+ ( 4_096, 100),
+ (12_000, 10),
+]
+
+if args.fast_tests:
+ for k in range(len(BITS_AND_ITERATIONS)):
+ b, i = BITS_AND_ITERATIONS[k]
+ BITS_AND_ITERATIONS[k] = (b, i // 10 if i >= 100 else 5)
+
+if args.no_random:
+ BITS_AND_ITERATIONS = []
+
+#
+# Where is the DLL? If missing, build using: `odin build . -build-mode:shared`
+#
+if platform.system() == "Windows":
+ LIB_PATH = os.getcwd() + os.sep + "math_big_test_library.dll"
+elif platform.system() == "Linux":
+ LIB_PATH = os.getcwd() + os.sep + "math_big_test_library.so"
+elif platform.system() == "Darwin":
+ LIB_PATH = os.getcwd() + os.sep + "math_big_test_library.dylib"
+else:
+ print("Platform is unsupported.")
+ exit(1)
+
+
+TOTAL_TIME = 0
+UNTIL_TIME = 0
+UNTIL_ITERS = 0
+
+def we_iterate():
+ if args.timed:
+ return TOTAL_TIME < UNTIL_TIME
+ else:
+ global UNTIL_ITERS
+ UNTIL_ITERS -= 1
+ return UNTIL_ITERS != -1
+
+#
+# Error enum values
+#
+class Error(Enum):
+ Okay = 0
+ Out_Of_Memory = 1
+ Invalid_Pointer = 2
+ Invalid_Argument = 3
+ Unknown_Error = 4
+ Max_Iterations_Reached = 5
+ Buffer_Overflow = 6
+ Integer_Overflow = 7
+ Division_by_Zero = 8
+ Math_Domain_Error = 9
+ Unimplemented = 127
+
+#
+# Disable garbage collection
+#
+gc.disable()
+
+#
+# Set up exported procedures
+#
+try:
+ l = cdll.LoadLibrary(LIB_PATH)
+except:
+ print("Couldn't find or load " + LIB_PATH + ".")
+ exit(1)
+
+def load(export_name, args, res):
+ export_name.argtypes = args
+ export_name.restype = res
+ return export_name
+
+#
+# Result values will be passed in a struct { res: cstring, err: Error }
+#
+class Res(Structure):
+ _fields_ = [("res", c_char_p), ("err", c_uint64)]
+
+initialize_constants = load(l.test_initialize_constants, [], c_uint64)
+print("initialize_constants: ", initialize_constants())
+
+error_string = load(l.test_error_string, [c_byte], c_char_p)
+
+add = load(l.test_add, [c_char_p, c_char_p ], Res)
+sub = load(l.test_sub, [c_char_p, c_char_p ], Res)
+mul = load(l.test_mul, [c_char_p, c_char_p ], Res)
+sqr = load(l.test_sqr, [c_char_p ], Res)
+div = load(l.test_div, [c_char_p, c_char_p ], Res)
+
+# Powers and such
+int_log = load(l.test_log, [c_char_p, c_longlong], Res)
+int_pow = load(l.test_pow, [c_char_p, c_longlong], Res)
+int_sqrt = load(l.test_sqrt, [c_char_p ], Res)
+int_root_n = load(l.test_root_n, [c_char_p, c_longlong], Res)
+
+# Logical operations
+int_shl_digit = load(l.test_shl_digit, [c_char_p, c_longlong], Res)
+int_shr_digit = load(l.test_shr_digit, [c_char_p, c_longlong], Res)
+int_shl = load(l.test_shl, [c_char_p, c_longlong], Res)
+int_shr = load(l.test_shr, [c_char_p, c_longlong], Res)
+int_shr_signed = load(l.test_shr_signed, [c_char_p, c_longlong], Res)
+
+int_factorial = load(l.test_factorial, [c_uint64 ], Res)
+int_gcd = load(l.test_gcd, [c_char_p, c_char_p ], Res)
+int_lcm = load(l.test_lcm, [c_char_p, c_char_p ], Res)
+
+is_square = load(l.test_is_square, [c_char_p ], Res)
+
+def test(test_name: "", res: Res, param=[], expected_error = Error.Okay, expected_result = "", radix=16):
+ passed = True
+ r = None
+ err = Error(res.err)
+
+ if err != expected_error:
+ error_loc = res.res.decode('utf-8')
+ error = "{}: {} in '{}'".format(test_name, err, error_loc)
+
+ if len(param):
+ error += " with params {}".format(param)
+
+ print(error, flush=True)
+ passed = False
+ elif err == Error.Okay:
+ r = None
+ try:
+ r = res.res.decode('utf-8')
+ r = int(res.res, radix)
+ except:
+ pass
+
+ if r != expected_result:
+ error = "{}: Result was '{}', expected '{}'".format(test_name, r, expected_result)
+ if len(param):
+ error += " with params {}".format(param)
+
+ print(error, flush=True)
+ passed = False
+
+ if EXIT_ON_FAIL and not passed: exit(res.err)
+
+ return passed
+
+def arg_to_odin(a):
+ if a >= 0:
+ s = hex(a)[2:]
+ else:
+ s = '-' + hex(a)[3:]
+ return s.encode('utf-8')
+
+
+def big_integer_sqrt(src):
+ # The Python version on Github's CI doesn't offer math.isqrt.
+ # We implement our own
+ count = src.bit_length()
+ a, b = count >> 1, count & 1
+
+ x = 1 << (a + b)
+
+ while True:
+ # y = (x + n // x) // 2
+ t1 = src // x
+ t2 = t1 + x
+ y = t2 >> 1
+
+ if y >= x:
+ return x
+
+ x, y = y, x
+
+def big_integer_lcm(a, b):
+ # Computes least common multiple as `|a*b|/gcd(a,b)`
+ # Divide the smallest by the GCD.
+
+ if a == 0 or b == 0:
+ return 0
+
+ if abs(a) < abs(b):
+ # Store quotient in `t2` such that `t2 * b` is the LCM.
+ lcm = a // math.gcd(a, b)
+ return abs(b * lcm)
+ else:
+ # Store quotient in `t2` such that `t2 * a` is the LCM.
+ lcm = b // math.gcd(a, b)
+ return abs(a * lcm)
+
+def test_add(a = 0, b = 0, expected_error = Error.Okay):
+ args = [arg_to_odin(a), arg_to_odin(b)]
+ res = add(*args)
+ expected_result = None
+ if expected_error == Error.Okay:
+ expected_result = a + b
+ return test("test_add", res, [a, b], expected_error, expected_result)
+
+def test_sub(a = 0, b = 0, expected_error = Error.Okay):
+ args = [arg_to_odin(a), arg_to_odin(b)]
+ res = sub(*args)
+ expected_result = None
+ if expected_error == Error.Okay:
+ expected_result = a - b
+ return test("test_sub", res, [a, b], expected_error, expected_result)
+
+def test_mul(a = 0, b = 0, expected_error = Error.Okay):
+ args = [arg_to_odin(a), arg_to_odin(b)]
+ try:
+ res = mul(*args)
+ except OSError as e:
+ print("{} while trying to multiply {} x {}.".format(e, a, b))
+ if EXIT_ON_FAIL: exit(3)
+ return False
+
+ expected_result = None
+ if expected_error == Error.Okay:
+ expected_result = a * b
+ return test("test_mul", res, [a, b], expected_error, expected_result)
+
+def test_sqr(a = 0, b = 0, expected_error = Error.Okay):
+ args = [arg_to_odin(a)]
+ try:
+ res = sqr(*args)
+ except OSError as e:
+ print("{} while trying to square {}.".format(e, a))
+ if EXIT_ON_FAIL: exit(3)
+ return False
+
+ expected_result = None
+ if expected_error == Error.Okay:
+ expected_result = a * a
+ return test("test_sqr", res, [a], expected_error, expected_result)
+
+def test_div(a = 0, b = 0, expected_error = Error.Okay):
+ args = [arg_to_odin(a), arg_to_odin(b)]
+ try:
+ res = div(*args)
+ except OSError as e:
+ print("{} while trying divide to {} / {}.".format(e, a, b))
+ if EXIT_ON_FAIL: exit(3)
+ return False
+ expected_result = None
+ if expected_error == Error.Okay:
+ #
+ # We don't round the division results, so if one component is negative, we're off by one.
+ #
+ if a < 0 and b > 0:
+ expected_result = int(-(abs(a) // b))
+ elif b < 0 and a > 0:
+ expected_result = int(-(a // abs((b))))
+ else:
+ expected_result = a // b if b != 0 else None
+ return test("test_div", res, [a, b], expected_error, expected_result)
+
+
+def test_log(a = 0, base = 0, expected_error = Error.Okay):
+ args = [arg_to_odin(a), base]
+ res = int_log(*args)
+
+ expected_result = None
+ if expected_error == Error.Okay:
+ expected_result = int(math.log(a, base))
+ return test("test_log", res, [a, base], expected_error, expected_result)
+
+def test_pow(base = 0, power = 0, expected_error = Error.Okay):
+ args = [arg_to_odin(base), power]
+ res = int_pow(*args)
+
+ expected_result = None
+ if expected_error == Error.Okay:
+ if power < 0:
+ expected_result = 0
+ else:
+ # NOTE(Jeroen): Don't use `math.pow`, it's a floating point approximation.
+ # Use built-in `pow` or `a**b` instead.
+ expected_result = pow(base, power)
+ return test("test_pow", res, [base, power], expected_error, expected_result)
+
+def test_sqrt(number = 0, expected_error = Error.Okay):
+ args = [arg_to_odin(number)]
+ try:
+ res = int_sqrt(*args)
+ except OSError as e:
+ print("{} while trying to sqrt {}.".format(e, number))
+ if EXIT_ON_FAIL: exit(3)
+ return False
+
+ expected_result = None
+ if expected_error == Error.Okay:
+ if number < 0:
+ expected_result = 0
+ else:
+ expected_result = big_integer_sqrt(number)
+ return test("test_sqrt", res, [number], expected_error, expected_result)
+
+def root_n(number, root):
+ u, s = number, number + 1
+ while u < s:
+ s = u
+ t = (root-1) * s + number // pow(s, root - 1)
+ u = t // root
+ return s
+
+def test_root_n(number = 0, root = 0, expected_error = Error.Okay):
+ args = [arg_to_odin(number), root]
+ res = int_root_n(*args)
+ expected_result = None
+ if expected_error == Error.Okay:
+ if number < 0:
+ expected_result = 0
+ else:
+ expected_result = root_n(number, root)
+
+ return test("test_root_n", res, [number, root], expected_error, expected_result)
+
+def test_shl_digit(a = 0, digits = 0, expected_error = Error.Okay):
+ args = [arg_to_odin(a), digits]
+ res = int_shl_digit(*args)
+ expected_result = None
+ if expected_error == Error.Okay:
+ expected_result = a << (digits * 60)
+ return test("test_shl_digit", res, [a, digits], expected_error, expected_result)
+
+def test_shr_digit(a = 0, digits = 0, expected_error = Error.Okay):
+ args = [arg_to_odin(a), digits]
+ res = int_shr_digit(*args)
+ expected_result = None
+ if expected_error == Error.Okay:
+ if a < 0:
+ # Don't pass negative numbers. We have a shr_signed.
+ return False
+ else:
+ expected_result = a >> (digits * 60)
+
+ return test("test_shr_digit", res, [a, digits], expected_error, expected_result)
+
+def test_shl(a = 0, bits = 0, expected_error = Error.Okay):
+ args = [arg_to_odin(a), bits]
+ res = int_shl(*args)
+ expected_result = None
+ if expected_error == Error.Okay:
+ expected_result = a << bits
+ return test("test_shl", res, [a, bits], expected_error, expected_result)
+
+def test_shr(a = 0, bits = 0, expected_error = Error.Okay):
+ args = [arg_to_odin(a), bits]
+ res = int_shr(*args)
+ expected_result = None
+ if expected_error == Error.Okay:
+ if a < 0:
+ # Don't pass negative numbers. We have a shr_signed.
+ return False
+ else:
+ expected_result = a >> bits
+
+ return test("test_shr", res, [a, bits], expected_error, expected_result)
+
+def test_shr_signed(a = 0, bits = 0, expected_error = Error.Okay):
+ args = [arg_to_odin(a), bits]
+ res = int_shr_signed(*args)
+ expected_result = None
+ if expected_error == Error.Okay:
+ expected_result = a >> bits
+
+ return test("test_shr_signed", res, [a, bits], expected_error, expected_result)
+
+def test_factorial(number = 0, expected_error = Error.Okay):
+ args = [number]
+ try:
+ res = int_factorial(*args)
+ except OSError as e:
+ print("{} while trying to factorial {}.".format(e, number))
+ if EXIT_ON_FAIL: exit(3)
+ return False
+
+ expected_result = None
+ if expected_error == Error.Okay:
+ expected_result = math.factorial(number)
+
+ return test("test_factorial", res, [number], expected_error, expected_result)
+
+def test_gcd(a = 0, b = 0, expected_error = Error.Okay):
+ args = [arg_to_odin(a), arg_to_odin(b)]
+ res = int_gcd(*args)
+ expected_result = None
+ if expected_error == Error.Okay:
+ expected_result = math.gcd(a, b)
+
+ return test("test_gcd", res, [a, b], expected_error, expected_result)
+
+def test_lcm(a = 0, b = 0, expected_error = Error.Okay):
+ args = [arg_to_odin(a), arg_to_odin(b)]
+ res = int_lcm(*args)
+ expected_result = None
+ if expected_error == Error.Okay:
+ expected_result = big_integer_lcm(a, b)
+
+ return test("test_lcm", res, [a, b], expected_error, expected_result)
+
+def test_is_square(a = 0, b = 0, expected_error = Error.Okay):
+ args = [arg_to_odin(a)]
+ res = is_square(*args)
+ expected_result = None
+ if expected_error == Error.Okay:
+ expected_result = str(big_integer_sqrt(a) ** 2 == a) if a > 0 else "False"
+
+ return test("test_is_square", res, [a], expected_error, expected_result)
+
+# TODO(Jeroen): Make sure tests cover edge cases, fast paths, and so on.
+#
+# The last two arguments in tests are the expected error and expected result.
+#
+# The expected error defaults to None.
+# By default the Odin implementation will be tested against the Python one.
+# You can override that by supplying an expected result as the last argument instead.
+
+TESTS = {
+ test_add: [
+ [ 1234, 5432],
+ ],
+ test_sub: [
+ [ 1234, 5432],
+ ],
+ test_mul: [
+ [ 1234, 5432],
+ [ 0xd3b4e926aaba3040e1c12b5ea553b5, 0x1a821e41257ed9281bee5bc7789ea7 ],
+ [ 1 << 21_105, 1 << 21_501 ],
+ ],
+ test_sqr: [
+ [ 5432],
+ [ 0xd3b4e926aaba3040e1c12b5ea553b5 ],
+ ],
+ test_div: [
+ [ 54321, 12345],
+ [ 55431, 0, Error.Division_by_Zero],
+ [ 12980742146337069150589594264770969721, 4611686018427387904 ],
+ [ 831956404029821402159719858789932422, 243087903122332132 ],
+ ],
+ test_log: [
+ [ 3192, 1, Error.Invalid_Argument],
+ [ -1234, 2, Error.Math_Domain_Error],
+ [ 0, 2, Error.Math_Domain_Error],
+ [ 1024, 2],
+ ],
+ test_pow: [
+ [ 0, -1, Error.Math_Domain_Error ], # Math
+ [ 0, 0 ], # 1
+ [ 0, 2 ], # 0
+ [ 42, -1,], # 0
+ [ 42, 1 ], # 1
+ [ 42, 0 ], # 42
+ [ 42, 2 ], # 42*42
+ ],
+ test_sqrt: [
+ [ -1, Error.Invalid_Argument, ],
+ [ 42, Error.Okay, ],
+ [ 12345678901234567890, Error.Okay, ],
+ [ 1298074214633706907132624082305024, Error.Okay, ],
+ [ 686885735734829009541949746871140768343076607029752932751182108475420900392874228486622313727012705619148037570309621219533087263900443932890792804879473795673302686046941536636874184361869252299636701671980034458333859202703255467709267777184095435235980845369829397344182319113372092844648570818726316581751114346501124871729572474923695509057166373026411194094493240101036672016770945150422252961487398124677567028263059046193391737576836378376192651849283925197438927999526058932679219572030021792914065825542626400207956134072247020690107136531852625253942429167557531123651471221455967386267137846791963149859804549891438562641323068751514370656287452006867713758971418043865298618635213551059471668293725548570452377976322899027050925842868079489675596835389444833567439058609775325447891875359487104691935576723532407937236505941186660707032433807075470656782452889754501872408562496805517394619388777930253411467941214807849472083814447498068636264021405175653742244368865090604940094889189800007448083930490871954101880815781177612910234741529950538835837693870921008635195545246771593130784786737543736434086434015200264933536294884482218945403958647118802574342840790536176272341586020230110889699633073513016344826709214, Error.Okay, ],
+ ],
+ test_root_n: [
+ [ 1298074214633706907132624082305024, 2, Error.Okay, ],
+ ],
+ test_shl_digit: [
+ [ 3192, 1 ],
+ [ 1298074214633706907132624082305024, 2 ],
+ [ 1024, 3 ],
+ ],
+ test_shr_digit: [
+ [ 3680125442705055547392, 1 ],
+ [ 1725436586697640946858688965569256363112777243042596638790631055949824, 2 ],
+ [ 219504133884436710204395031992179571, 2 ],
+ ],
+ test_shl: [
+ [ 3192, 1 ],
+ [ 1298074214633706907132624082305024, 2 ],
+ [ 1024, 3 ],
+ ],
+ test_shr: [
+ [ 3680125442705055547392, 1 ],
+ [ 1725436586697640946858688965569256363112777243042596638790631055949824, 2 ],
+ [ 219504133884436710204395031992179571, 2 ],
+ ],
+ test_shr_signed: [
+ [ -611105530635358368578155082258244262, 12 ],
+ [ -149195686190273039203651143129455, 12 ],
+ [ 611105530635358368578155082258244262, 12 ],
+ [ 149195686190273039203651143129455, 12 ],
+ ],
+ test_factorial: [
+ [ 6_000 ], # Regular factorial, see cutoff in common.odin.
+ [ 12_345 ], # Binary split factorial
+ ],
+ test_gcd: [
+ [ 23, 25, ],
+ [ 125, 25, ],
+ [ 125, 0, ],
+ [ 0, 0, ],
+ [ 0, 125,],
+ ],
+ test_lcm: [
+ [ 23, 25,],
+ [ 125, 25, ],
+ [ 125, 0, ],
+ [ 0, 0, ],
+ [ 0, 125,],
+ ],
+ test_is_square: [
+ [ 12, ],
+ [ 92232459121502451677697058974826760244863271517919321608054113675118660929276431348516553336313179167211015633639725554914519355444316239500734169769447134357534241879421978647995614218985202290368055757891124109355450669008628757662409138767505519391883751112010824030579849970582074544353971308266211776494228299586414907715854328360867232691292422194412634523666770452490676515117702116926803826546868467146319938818238521874072436856528051486567230096290549225463582766830777324099589751817442141036031904145041055454639783559905920619197290800070679733841430619962318433709503256637256772215111521321630777950145713049902839937043785039344243357384899099910837463164007565230287809026956254332260375327814271845678201, ]
+ ],
+}
+
+if not args.fast_tests:
+ TESTS[test_factorial].append(
+ # This one on its own takes around 800ms, so we exclude it for FAST_TESTS
+ [ 10_000 ],
+ )
+
+total_passes = 0
+total_failures = 0
+
+#
+# test_shr_signed also tests shr, so we're not going to test shr randomly.
+#
+RANDOM_TESTS = [
+ test_add, test_sub, test_mul, test_sqr, test_div,
+ test_log, test_pow, test_sqrt, test_root_n,
+ test_shl_digit, test_shr_digit, test_shl, test_shr_signed,
+ test_gcd, test_lcm, test_is_square,
+]
+SKIP_LARGE = [
+ test_pow, test_root_n, # test_gcd,
+]
+SKIP_LARGEST = []
+
+# Untimed warmup.
+for test_proc in TESTS:
+ for t in TESTS[test_proc]:
+ res = test_proc(*t)
+
+if __name__ == '__main__':
+ print("\n---- math/big tests ----")
+ print()
+
+ max_name = 0
+ for test_proc in TESTS:
+ max_name = max(max_name, len(test_proc.__name__))
+
+ fmt_string = "{name:>{max_name}}: {count_pass:7,} passes and {count_fail:7,} failures in {timing:9.3f} ms."
+ fmt_string = fmt_string.replace("{max_name}", str(max_name))
+
+ for test_proc in TESTS:
+ count_pass = 0
+ count_fail = 0
+ TIMINGS = {}
+ for t in TESTS[test_proc]:
+ start = time.perf_counter()
+ res = test_proc(*t)
+ diff = time.perf_counter() - start
+ TOTAL_TIME += diff
+
+ if test_proc not in TIMINGS:
+ TIMINGS[test_proc] = diff
+ else:
+ TIMINGS[test_proc] += diff
+
+ if res:
+ count_pass += 1
+ total_passes += 1
+ else:
+ count_fail += 1
+ total_failures += 1
+
+ print(fmt_string.format(name=test_proc.__name__, count_pass=count_pass, count_fail=count_fail, timing=TIMINGS[test_proc] * 1_000))
+
+ for BITS, ITERATIONS in BITS_AND_ITERATIONS:
+ print()
+ print("---- math/big with two random {bits:,} bit numbers ----".format(bits=BITS))
+ print()
+
+ #
+ # We've already tested up to the 10th root.
+ #
+ TEST_ROOT_N_PARAMS = [2, 3, 4, 5, 6]
+
+ for test_proc in RANDOM_TESTS:
+ if BITS > 1_200 and test_proc in SKIP_LARGE: continue
+ if BITS > 4_096 and test_proc in SKIP_LARGEST: continue
+
+ count_pass = 0
+ count_fail = 0
+ TIMINGS = {}
+
+ UNTIL_ITERS = ITERATIONS
+ if test_proc == test_root_n and BITS == 1_200:
+ UNTIL_ITERS /= 10
+
+ UNTIL_TIME = TOTAL_TIME + BITS / args.timed_bits
+ # We run each test for a second per 20k bits
+
+ index = 0
+
+ while we_iterate():
+ a = randint(-(1 << BITS), 1 << BITS)
+ b = randint(-(1 << BITS), 1 << BITS)
+
+ if test_proc == test_div:
+ # We've already tested division by zero above.
+ bits = int(BITS * 0.6)
+ b = randint(-(1 << bits), 1 << bits)
+ if b == 0:
+ b == 42
+ elif test_proc == test_log:
+ # We've already tested log's domain errors.
+ a = randint(1, 1 << BITS)
+ b = randint(2, 1 << 60)
+ elif test_proc == test_pow:
+ b = randint(1, 10)
+ elif test_proc == test_sqrt:
+ a = randint(1, 1 << BITS)
+ b = Error.Okay
+ elif test_proc == test_root_n:
+ a = randint(1, 1 << BITS)
+ b = TEST_ROOT_N_PARAMS[index]
+ index = (index + 1) % len(TEST_ROOT_N_PARAMS)
+ elif test_proc == test_shl_digit:
+ b = randint(0, 10);
+ elif test_proc == test_shr_digit:
+ a = abs(a)
+ b = randint(0, 10);
+ elif test_proc == test_shl:
+ b = randint(0, min(BITS, 120))
+ elif test_proc == test_shr_signed:
+ b = randint(0, min(BITS, 120))
+ elif test_proc == test_is_square:
+ a = randint(0, 1 << BITS)
+ elif test_proc == test_lcm:
+ smallest = min(a, b)
+ biggest = max(a, b)
+
+ # Randomly swap biggest and smallest
+ if randint(1, 11) % 2 == 0:
+ smallest, biggest = biggest, smallest
+
+ a, b = smallest, biggest
+ else:
+ b = randint(0, 1 << BITS)
+
+ res = None
+
+ start = time.perf_counter()
+ res = test_proc(a, b)
+ diff = time.perf_counter() - start
+
+ TOTAL_TIME += diff
+
+ if test_proc not in TIMINGS:
+ TIMINGS[test_proc] = diff
+ else:
+ TIMINGS[test_proc] += diff
+
+ if res:
+ count_pass += 1; total_passes += 1
+ else:
+ count_fail += 1; total_failures += 1
+
+ print(fmt_string.format(name=test_proc.__name__, count_pass=count_pass, count_fail=count_fail, timing=TIMINGS[test_proc] * 1_000))
+
+ print()
+ print("---- THE END ----")
+ print()
+ print(fmt_string.format(name="total", count_pass=total_passes, count_fail=total_failures, timing=TOTAL_TIME * 1_000))
+
+ if total_failures:
+ exit(1) \ No newline at end of file