Browse Source

big: Add `test_pow` and some more switches.

Jeroen van Rijn 4 years ago
parent
commit
f12672727d
2 changed files with 128 additions and 74 deletions
  1. 44 24
      core/math/big/test.odin
  2. 84 50
      core/math/big/test.py

+ 44 - 24
core/math/big/test.odin

@@ -26,53 +26,53 @@ PyRes :: struct {
 	return strings.clone_to_cstring(es[err], context.temp_allocator);
 }
 
-@export test_add_two :: proc "c" (a, b: cstring, radix := int(10)) -> (res: PyRes) {
+@export test_add_two :: proc "c" (a, b: cstring) -> (res: PyRes) {
 	context = runtime.default_context();
 	err: Error;
 
 	aa, bb, sum := &Int{}, &Int{}, &Int{};
 	defer destroy(aa, bb, sum);
 
-	if err = atoi(aa, string(a), i8(radix)); err != .None { return PyRes{res=":add_two:atoi(a):", err=err}; }
-	if err = atoi(bb, string(b), i8(radix)); err != .None { return PyRes{res=":add_two:atoi(b):", err=err}; }
-	if err = add(sum, aa, bb);               err != .None { return PyRes{res=":add_two:add(sum,a,b):", err=err}; }
+	if err = atoi(aa, string(a), 10); err != .None { return PyRes{res=":add_two:atoi(a):", err=err}; }
+	if err = atoi(bb, string(b), 10); err != .None { return PyRes{res=":add_two:atoi(b):", err=err}; }
+	if err = add(sum, aa, bb);        err != .None { return PyRes{res=":add_two:add(sum,a,b):", err=err}; }
 
 	r: cstring;
-	r, err = int_itoa_cstring(sum, i8(radix), context.temp_allocator);
+	r, err = int_itoa_cstring(sum, 10, context.temp_allocator);
 	if err != .None { return PyRes{res=":add_two:itoa(sum):", err=err}; }
 	return PyRes{res = r, err = .None};
 }
 
-@export test_sub_two :: proc "c" (a, b: cstring, radix := int(10)) -> (res: PyRes) {
+@export test_sub_two :: proc "c" (a, b: cstring) -> (res: PyRes) {
 	context = runtime.default_context();
 	err: Error;
 
 	aa, bb, sum := &Int{}, &Int{}, &Int{};
 	defer destroy(aa, bb, sum);
 
-	if err = atoi(aa, string(a), i8(radix)); err != .None { return PyRes{res=":sub_two:atoi(a):", err=err}; }
-	if err = atoi(bb, string(b), i8(radix)); err != .None { return PyRes{res=":sub_two:atoi(b):", err=err}; }
-	if err = sub(sum, aa, bb);               err != .None { return PyRes{res=":sub_two:sub(sum,a,b):", err=err}; }
+	if err = atoi(aa, string(a), 10); err != .None { return PyRes{res=":sub_two:atoi(a):", err=err}; }
+	if err = atoi(bb, string(b), 10); err != .None { return PyRes{res=":sub_two:atoi(b):", err=err}; }
+	if err = sub(sum, aa, bb);        err != .None { return PyRes{res=":sub_two:sub(sum,a,b):", err=err}; }
 
 	r: cstring;
-	r, err = int_itoa_cstring(sum, i8(radix), context.temp_allocator);
+	r, err = int_itoa_cstring(sum, 10, context.temp_allocator);
 	if err != .None { return PyRes{res=":sub_two:itoa(sum):", err=err}; }
 	return PyRes{res = r, err = .None};
 }
 
-@export test_mul_two :: proc "c" (a, b: cstring, radix := int(10)) -> (res: PyRes) {
+@export test_mul_two :: proc "c" (a, b: cstring) -> (res: PyRes) {
 	context = runtime.default_context();
 	err: Error;
 
 	aa, bb, product := &Int{}, &Int{}, &Int{};
 	defer destroy(aa, bb, product);
 
-	if err = atoi(aa, string(a), i8(radix)); err != .None { return PyRes{res=":mul_two:atoi(a):", err=err}; }
-	if err = atoi(bb, string(b), i8(radix)); err != .None { return PyRes{res=":mul_two:atoi(b):", err=err}; }
-	if err = mul(product, aa, bb);           err != .None { return PyRes{res=":mul_two:mul(product,a,b):", err=err}; }
+	if err = atoi(aa, string(a), 10); err != .None { return PyRes{res=":mul_two:atoi(a):", err=err}; }
+	if err = atoi(bb, string(b), 10); err != .None { return PyRes{res=":mul_two:atoi(b):", err=err}; }
+	if err = mul(product, aa, bb);    err != .None { return PyRes{res=":mul_two:mul(product,a,b):", err=err}; }
 
 	r: cstring;
-	r, err = int_itoa_cstring(product, i8(radix), context.temp_allocator);
+	r, err = int_itoa_cstring(product, 10, context.temp_allocator);
 	if err != .None { return PyRes{res=":mul_two:itoa(product):", err=err}; }
 	return PyRes{res = r, err = .None};
 }
@@ -80,19 +80,19 @@ PyRes :: struct {
 /*
 	NOTE(Jeroen): For simplicity, we don't return the quotient and the remainder, just the quotient.
 */
-@export test_div_two :: proc "c" (a, b: cstring, radix := int(10)) -> (res: PyRes) {
+@export test_div_two :: proc "c" (a, b: cstring) -> (res: PyRes) {
 	context = runtime.default_context();
 	err: Error;
 
 	aa, bb, quotient := &Int{}, &Int{}, &Int{};
 	defer destroy(aa, bb, quotient);
 
-	if err = atoi(aa, string(a), i8(radix)); err != .None { return PyRes{res=":div_two:atoi(a):", err=err}; }
-	if err = atoi(bb, string(b), i8(radix)); err != .None { return PyRes{res=":div_two:atoi(b):", err=err}; }
-	if err = div(quotient, aa, bb);          err != .None { return PyRes{res=":div_two:div(quotient,a,b):", err=err}; }
+	if err = atoi(aa, string(a), 10); err != .None { return PyRes{res=":div_two:atoi(a):", err=err}; }
+	if err = atoi(bb, string(b), 10); err != .None { return PyRes{res=":div_two:atoi(b):", err=err}; }
+	if err = div(quotient, aa, bb);   err != .None { return PyRes{res=":div_two:div(quotient,a,b):", err=err}; }
 
 	r: cstring;
-	r, err = int_itoa_cstring(quotient, i8(radix), context.temp_allocator);
+	r, err = int_itoa_cstring(quotient, 10, context.temp_allocator);
 	if err != .None { return PyRes{res=":div_two:itoa(quotient):", err=err}; }
 	return PyRes{res = r, err = .None};
 }
@@ -101,7 +101,7 @@ PyRes :: struct {
 /*
 	res = log(a, base)
 */
-@export test_log :: proc "c" (a: cstring, base := DIGIT(2), radix := int(10)) -> (res: PyRes) {
+@export test_log :: proc "c" (a: cstring, base := DIGIT(2)) -> (res: PyRes) {
 	context = runtime.default_context();
 	err: Error;
 	l: int;
@@ -109,8 +109,8 @@ PyRes :: struct {
 	aa := &Int{};
 	defer destroy(aa);
 
-	if err = atoi(aa, string(a), i8(radix)); err != .None { return PyRes{res=":log:atoi(a):", err=err}; }
-	if l, err = log(aa, base);               err != .None { return PyRes{res=":log:log(a, base):", err=err}; }
+	if err = atoi(aa, string(a), 10); err != .None { return PyRes{res=":log:atoi(a):", err=err}; }
+	if l, err = log(aa, base);        err != .None { return PyRes{res=":log:log(a, base):", err=err}; }
 
 	zero(aa);
 	aa.digit[0] = DIGIT(l)  & _MASK;
@@ -119,7 +119,27 @@ PyRes :: struct {
 	clamp(aa);
 
 	r: cstring;
-	r, err = int_itoa_cstring(aa, i8(radix), context.temp_allocator);
+	r, err = int_itoa_cstring(aa, 10, context.temp_allocator);
 	if err != .None { return PyRes{res=":log:itoa(res):", err=err}; }
 	return PyRes{res = r, err = .None};
 }
+
+/*
+	dest = base^power
+*/
+@export test_pow :: proc "c" (base: cstring, power := int(2)) -> (res: PyRes) {
+	context = runtime.default_context();
+	err: Error;
+	l: int;
+
+	dest, bb := &Int{}, &Int{};
+	defer destroy(dest, bb);
+
+	if err = atoi(bb, string(base), 10); err != .None { return PyRes{res=":pow:atoi(base):", err=err}; }
+	if err = pow(dest, bb, power);       err != .None { return PyRes{res=":pow:pow(dest, base, power):", err=err}; }
+
+	r: cstring;
+	r, err = int_itoa_cstring(dest, 10, context.temp_allocator);
+	if err != .None { return PyRes{res=":log:itoa(res):", err=err}; }
+	return PyRes{res = r, err = .None};
+}

+ 84 - 50
core/math/big/test.py

@@ -7,14 +7,21 @@ import time
 from enum import Enum
 
 #
-# How many iterations of each random test do we want to run?
+# Normally, we report the number of passes and fails.
+# With EXIT_ON_FAIL set, we exit at the first fail.
 #
-BITS_AND_ITERATIONS = [
-	(   120, 10_000),
-	( 1_200,  1_000),
-	( 4_096,    100),
-	(12_000,     10),
-]
+EXIT_ON_FAIL = False
+
+#
+# We skip randomized tests altogether if NO_RANDOM_TESTS is set.
+#
+NO_RANDOM_TESTS = False
+
+#
+# If TIMED_TESTS == False and FAST_TESTS == True, we cut down the number of iterations.
+# See below.
+#
+FAST_TESTS = True
 
 #
 # For timed tests we budget a second per `n` bits and iterate until we hit that time.
@@ -24,16 +31,23 @@ TIMED_TESTS = False
 TIMED_BITS_PER_SECOND = 20_000
 
 #
-# If TIMED_TESTS == False and FAST_TESTS == True, we cut down the number of iterations.
-# See below.
+# How many iterations of each random test do we want to run?
 #
-FAST_TESTS = True
+BITS_AND_ITERATIONS = [
+	(   120, 10_000),
+	( 1_200,  1_000),
+	( 4_096,    100),
+	(12_000,     10),
+]
 
 if 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 NO_RANDOM_TESTS:
+	BITS_AND_ITERATIONS = []
+
 #
 # Where is the DLL? If missing, build using: `odin build . -build-mode:shared`
 #
@@ -99,13 +113,13 @@ class Res(Structure):
 
 error_string = load(l.test_error_string, [c_byte], c_char_p)
 
-add_two = load(l.test_add_two, [c_char_p, c_char_p, c_longlong], Res)
-sub_two = load(l.test_sub_two, [c_char_p, c_char_p, c_longlong], Res)
-mul_two = load(l.test_mul_two, [c_char_p, c_char_p, c_longlong], Res)
-div_two = load(l.test_div_two, [c_char_p, c_char_p, c_longlong], Res)
-
-int_log = load(l.test_log, [c_char_p, c_longlong, c_longlong], Res)
+add_two = load(l.test_add_two, [c_char_p, c_char_p], Res)
+sub_two = load(l.test_sub_two, [c_char_p, c_char_p], Res)
+mul_two = load(l.test_mul_two, [c_char_p, c_char_p], Res)
+div_two = load(l.test_div_two, [c_char_p, c_char_p], Res)
 
+int_log = load(l.test_log, [c_char_p, c_longlong], Res)
+int_pow = load(l.test_pow, [c_char_p, c_longlong], Res)
 
 def test(test_name: "", res: Res, param=[], expected_error = Error.Okay, expected_result = ""):
 	passed = True
@@ -137,46 +151,45 @@ def test(test_name: "", res: Res, param=[], expected_error = Error.Okay, expecte
 			print(error, flush=True)
 			passed = False
 
-	if not passed:
-		exit()
+	if EXIT_ON_FAIL and not passed: exit(res.err)
 
 	return passed
 
 
-def test_add_two(a = 0, b = 0, radix = 10, expected_error = Error.Okay):
-	args = [str(a), str(b), radix]
+def test_add_two(a = 0, b = 0, expected_error = Error.Okay):
+	args = [str(a), str(b)]
 	sa_c, sb_c = args[0].encode('utf-8'), args[1].encode('utf-8')
-	res  = add_two(sa_c, sb_c, radix)
+	res  = add_two(sa_c, sb_c)
 	expected_result = None
 	if expected_error == Error.Okay:
 		expected_result = a + b
 	return test("test_add_two", res, args, expected_error, expected_result)
 
-def test_sub_two(a = 0, b = 0, radix = 10, expected_error = Error.Okay):
+def test_sub_two(a = 0, b = 0, expected_error = Error.Okay):
 	sa,     sb = str(a), str(b)
 	sa_c, sb_c = sa.encode('utf-8'), sb.encode('utf-8')
-	res  = sub_two(sa_c, sb_c, radix)
+	res  = sub_two(sa_c, sb_c)
 	expected_result = None
 	if expected_error == Error.Okay:
 		expected_result = a - b
-	return test("test_sub_two", res, [sa_c, sb_c, radix], expected_error, expected_result)
+	return test("test_sub_two", res, [sa_c, sb_c], expected_error, expected_result)
 
-def test_mul_two(a = 0, b = 0, radix = 10, expected_error = Error.Okay):
+def test_mul_two(a = 0, b = 0, expected_error = Error.Okay):
 	sa,     sb = str(a), str(b)
 	sa_c, sb_c = sa.encode('utf-8'), sb.encode('utf-8')
-	res  = mul_two(sa_c, sb_c, radix)
+	res  = mul_two(sa_c, sb_c)
 	expected_result = None
 	if expected_error == Error.Okay:
 		expected_result = a * b
-	return test("test_mul_two", res, [sa_c, sb_c, radix], expected_error, expected_result)
+	return test("test_mul_two", res, [sa_c, sb_c], expected_error, expected_result)
 
-def test_div_two(a = 0, b = 0, radix = 10, expected_error = Error.Okay):
+def test_div_two(a = 0, b = 0, expected_error = Error.Okay):
 	sa,     sb = str(a), str(b)
 	sa_c, sb_c = sa.encode('utf-8'), sb.encode('utf-8')
 	try:
-		res  = div_two(sa_c, sb_c, radix)
+		res  = div_two(sa_c, sb_c)
 	except:
-		print("Exception with arguments:", a, b, radix)
+		print("Exception with arguments:", a, b)
 		return False
 	expected_result = None
 	if expected_error == Error.Okay:
@@ -189,19 +202,31 @@ def test_div_two(a = 0, b = 0, radix = 10, expected_error = Error.Okay):
 			expected_result = int(-(a / abs((b))))
 		else:
 			expected_result = a // b if b != 0 else None
-	return test("test_div_two", res, [sa_c, sb_c, radix], expected_error, expected_result)
+	return test("test_div_two", res, [sa_c, sb_c], expected_error, expected_result)
 
 
-def test_log(a = 0, base = 0, radix = 10, expected_error = Error.Okay):
-	args  = [str(a), base, radix]
+def test_log(a = 0, base = 0, expected_error = Error.Okay):
+	args  = [str(a), base]
 	sa_c  = args[0].encode('utf-8')
-	res   = int_log(sa_c, base, radix)
+	res   = int_log(sa_c, base)
 
 	expected_result = None
 	if expected_error == Error.Okay:
 		expected_result = int(log(a, base))
 	return test("test_log", res, args, expected_error, expected_result)
 
+def test_pow(base = 0, power = 0, expected_error = Error.Okay):
+	args  = [str(base), power]
+	sa_c  = args[0].encode('utf-8')
+	res   = int_pow(sa_c, power)
+
+	expected_result = None
+	if expected_error == Error.Okay:
+		if power < 0:
+			expected_result = 0
+		else:
+			expected_result = pow(base, power)
+	return test("test_pow", res, args, expected_error, expected_result)
 
 # TODO(Jeroen): Make sure tests cover edge cases, fast paths, and so on.
 #
@@ -213,25 +238,33 @@ def test_log(a = 0, base = 0, radix = 10, expected_error = Error.Okay):
 
 TESTS = {
 	test_add_two: [
-		[ 1234,   5432,    10, ],
-		[ 1234,   5432,   110, Error.Invalid_Argument],
+		[ 1234,   5432],
 	],
 	test_sub_two: [
-		[ 1234,   5432,    10, ],
+		[ 1234,   5432],
 	],
 	test_mul_two: [
-		[ 1234,   5432,    10, ],
-		[ 0xd3b4e926aaba3040e1c12b5ea553b5, 0x1a821e41257ed9281bee5bc7789ea7, 10, ]
+		[ 1234,   5432],
+		[ 0xd3b4e926aaba3040e1c12b5ea553b5, 0x1a821e41257ed9281bee5bc7789ea7]
 	],
 	test_div_two: [
-		[ 54321,	12345,		10, ],
-		[ 55431,		0,		10,		Error.Division_by_Zero],
+		[ 54321,	12345],
+		[ 55431,		0, Error.Division_by_Zero],
 	],
 	test_log: [
-		[ 3192,			1,		10,		Error.Invalid_Argument],
-		[ -1234,		2,		10,		Error.Math_Domain_Error],
-		[ 0,			2,		10,		Error.Math_Domain_Error],
-		[ 1024,			2,		10, ],
+		[ 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
 	],
 }
 
@@ -240,12 +273,13 @@ RANDOM_TESTS = [test_add_two, test_sub_two, test_mul_two, test_div_two, test_log
 total_passes   = 0
 total_failures = 0
 
+# Untimed warmup.
+for test_proc in TESTS:
+	for t in TESTS[test_proc]:
+		res   = test_proc(*t)
 
 if __name__ == '__main__':
-
-	test_log(1234, 2, 10)
-
-	print("---- core:math/big tests ----")
+	print("---- math/big tests ----")
 	print()
 
 	for test_proc in TESTS:
@@ -274,7 +308,7 @@ if __name__ == '__main__':
 
 	for BITS, ITERATIONS in BITS_AND_ITERATIONS:
 		print()		
-		print("---- core:math/big with two random {bits:,} bit numbers ----".format(bits=BITS))
+		print("---- math/big with two random {bits:,} bit numbers ----".format(bits=BITS))
 		print()
 
 		for test_proc in RANDOM_TESTS: