123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390 |
- //+ignore
- package math_big
- /*
- Copyright 2021 Jeroen van Rijn <[email protected]>.
- 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.
- */
- /*
- TODO: Write tests for `internal_*` and test reusing parameters with the public implementations.
- */
- import "core:runtime"
- import "core:strings"
- PyRes :: struct {
- res: cstring,
- err: Error,
- }
- @export test_initialize_constants :: proc "c" () -> (res: u64) {
- context = runtime.default_context();
- res = u64(initialize_constants());
- //assert(MUL_KARATSUBA_CUTOFF >= 40);
- return res;
- }
- @export test_error_string :: proc "c" (err: Error) -> (res: cstring) {
- context = runtime.default_context();
- es := 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: Error;
- aa, bb, sum := &Int{}, &Int{}, &Int{};
- defer internal_destroy(aa, bb, sum);
- if err = atoi(aa, string(a), 16); err != nil { return PyRes{res=":add:atoi(a):", err=err}; }
- if err = atoi(bb, string(b), 16); err != nil { return PyRes{res=":add:atoi(b):", err=err}; }
- if bb.used == 1 {
- if err = #force_inline internal_add(sum, aa, bb.digit[0]); err != nil { return PyRes{res=":add:add(sum,a,b):", err=err}; }
- } else {
- if err = #force_inline internal_add(sum, aa, bb); err != nil { return PyRes{res=":add:add(sum,a,b):", err=err}; }
- }
- r: cstring;
- r, err = 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: Error;
- aa, bb, sum := &Int{}, &Int{}, &Int{};
- defer internal_destroy(aa, bb, sum);
- if err = atoi(aa, string(a), 16); err != nil { return PyRes{res=":sub:atoi(a):", err=err}; }
- if err = atoi(bb, string(b), 16); err != nil { return PyRes{res=":sub:atoi(b):", err=err}; }
- if bb.used == 1 {
- if err = #force_inline internal_sub(sum, aa, bb.digit[0]); err != nil { return PyRes{res=":sub:sub(sum,a,b):", err=err}; }
- } else {
- if err = #force_inline internal_sub(sum, aa, bb); err != nil { return PyRes{res=":sub:sub(sum,a,b):", err=err}; }
- }
- r: cstring;
- r, err = 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: Error;
- aa, bb, product := &Int{}, &Int{}, &Int{};
- defer internal_destroy(aa, bb, product);
- if err = atoi(aa, string(a), 16); err != nil { return PyRes{res=":mul:atoi(a):", err=err}; }
- if err = atoi(bb, string(b), 16); err != nil { return PyRes{res=":mul:atoi(b):", err=err}; }
- if err = #force_inline internal_mul(product, aa, bb); err != nil { return PyRes{res=":mul:mul(product,a,b):", err=err}; }
- r: cstring;
- r, err = 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: Error;
- aa, square := &Int{}, &Int{};
- defer internal_destroy(aa, square);
- if err = atoi(aa, string(a), 16); err != nil { return PyRes{res=":sqr:atoi(a):", err=err}; }
- if err = #force_inline internal_sqr(square, aa); err != nil { return PyRes{res=":sqr:sqr(square,a):", err=err}; }
- r: cstring;
- r, err = 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: Error;
- aa, bb, quotient := &Int{}, &Int{}, &Int{};
- defer internal_destroy(aa, bb, quotient);
- if err = atoi(aa, string(a), 16); err != nil { return PyRes{res=":div:atoi(a):", err=err}; }
- if err = atoi(bb, string(b), 16); err != nil { return PyRes{res=":div:atoi(b):", err=err}; }
- if err = #force_inline internal_div(quotient, aa, bb); err != nil { return PyRes{res=":div:div(quotient,a,b):", err=err}; }
- r: cstring;
- r, err = 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 := DIGIT(2)) -> (res: PyRes) {
- context = runtime.default_context();
- err: Error;
- l: int;
- aa := &Int{};
- defer internal_destroy(aa);
- if err = atoi(aa, string(a), 16); err != nil { return PyRes{res=":log:atoi(a):", err=err}; }
- if l, err = #force_inline internal_log(aa, base); err != nil { return PyRes{res=":log:log(a, base):", err=err}; }
- #force_inline internal_zero(aa);
- aa.digit[0] = DIGIT(l) & _MASK;
- aa.digit[1] = DIGIT(l) >> _DIGIT_BITS;
- aa.used = 2;
- clamp(aa);
- r: cstring;
- r, err = 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: Error;
- dest, bb := &Int{}, &Int{};
- defer internal_destroy(dest, bb);
- if err = atoi(bb, string(base), 16); err != nil { return PyRes{res=":pow:atoi(base):", err=err}; }
- if err = #force_inline internal_pow(dest, bb, power); err != nil { return PyRes{res=":pow:pow(dest, base, power):", err=err}; }
- r: cstring;
- r, err = 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: Error;
- src := &Int{};
- defer internal_destroy(src);
- if err = atoi(src, string(source), 16); err != nil { return PyRes{res=":sqrt:atoi(src):", err=err}; }
- if err = #force_inline internal_sqrt(src, src); err != nil { return PyRes{res=":sqrt:sqrt(src):", err=err}; }
- r: cstring;
- r, err = 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: Error;
- src := &Int{};
- defer internal_destroy(src);
- if err = atoi(src, string(source), 16); err != nil { return PyRes{res=":root_n:atoi(src):", err=err}; }
- if err = #force_inline internal_root_n(src, src, power); err != nil { return PyRes{res=":root_n:root_n(src):", err=err}; }
- r: cstring;
- r, err = 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: Error;
- src := &Int{};
- defer internal_destroy(src);
- if err = atoi(src, string(source), 16); err != nil { return PyRes{res=":shr_digit:atoi(src):", err=err}; }
- if err = #force_inline internal_shr_digit(src, digits); err != nil { return PyRes{res=":shr_digit:shr_digit(src):", err=err}; }
- r: cstring;
- r, err = 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: Error;
- src := &Int{};
- defer internal_destroy(src);
- if err = atoi(src, string(source), 16); err != nil { return PyRes{res=":shl_digit:atoi(src):", err=err}; }
- if err = #force_inline internal_shl_digit(src, digits); err != nil { return PyRes{res=":shl_digit:shr_digit(src):", err=err}; }
- r: cstring;
- r, err = 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: Error;
- src := &Int{};
- defer internal_destroy(src);
- if err = atoi(src, string(source), 16); err != nil { return PyRes{res=":shr:atoi(src):", err=err}; }
- if err = #force_inline internal_shr(src, src, bits); err != nil { return PyRes{res=":shr:shr(src, bits):", err=err}; }
- r: cstring;
- r, err = 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: Error;
- src := &Int{};
- defer internal_destroy(src);
- if err = atoi(src, string(source), 16); err != nil { return PyRes{res=":shr_signed:atoi(src):", err=err}; }
- if err = #force_inline internal_shr_signed(src, src, bits); err != nil { return PyRes{res=":shr_signed:shr_signed(src, bits):", err=err}; }
- r: cstring;
- r, err = 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: Error;
- src := &Int{};
- defer internal_destroy(src);
- if err = atoi(src, string(source), 16); err != nil { return PyRes{res=":shl:atoi(src):", err=err}; }
- if err = #force_inline internal_shl(src, src, bits); err != nil { return PyRes{res=":shl:shl(src, bits):", err=err}; }
- r: cstring;
- r, err = 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: Error;
- dest := &Int{};
- defer internal_destroy(dest);
- if err = #force_inline internal_int_factorial(dest, n); err != nil { return PyRes{res=":factorial:factorial(n):", err=err}; }
- r: cstring;
- r, err = 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: Error;
- ai, bi, dest := &Int{}, &Int{}, &Int{};
- defer internal_destroy(ai, bi, dest);
- if err = atoi(ai, string(a), 16); err != nil { return PyRes{res=":gcd:atoi(a):", err=err}; }
- if err = atoi(bi, string(b), 16); err != nil { return PyRes{res=":gcd:atoi(b):", err=err}; }
- if err = #force_inline internal_int_gcd_lcm(dest, nil, ai, bi); err != nil { return PyRes{res=":gcd:gcd(a, b):", err=err}; }
- r: cstring;
- r, err = 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: Error;
- ai, bi, dest := &Int{}, &Int{}, &Int{};
- defer internal_destroy(ai, bi, dest);
- if err = atoi(ai, string(a), 16); err != nil { return PyRes{res=":lcm:atoi(a):", err=err}; }
- if err = atoi(bi, string(b), 16); err != nil { return PyRes{res=":lcm:atoi(b):", err=err}; }
- if err = #force_inline internal_int_gcd_lcm(nil, dest, ai, bi); err != nil { return PyRes{res=":lcm:lcm(a, b):", err=err}; }
- r: cstring;
- r, err = 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: Error;
- square: bool;
- ai := &Int{};
- defer internal_destroy(ai);
- if err = atoi(ai, string(a), 16); err != nil { return PyRes{res=":is_square:atoi(a):", err=err}; }
- if square, err = #force_inline 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};
- }
|