example.odin 5.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232
  1. //+ignore
  2. package math_big
  3. /*
  4. Copyright 2021 Jeroen van Rijn <[email protected]>.
  5. Made available under Odin's BSD-3 license.
  6. A BigInt implementation in Odin.
  7. For the theoretical underpinnings, see Knuth's The Art of Computer Programming, Volume 2, section 4.3.
  8. The code started out as an idiomatic source port of libTomMath, which is in the public domain, with thanks.
  9. */
  10. import "core:fmt"
  11. import "core:mem"
  12. print_configation :: proc() {
  13. fmt.printf(
  14. `
  15. Configuration:
  16. _DIGIT_BITS %v
  17. _MIN_DIGIT_COUNT %v
  18. _MAX_DIGIT_COUNT %v
  19. _DEFAULT_DIGIT_COUNT %v
  20. _MAX_COMBA %v
  21. _WARRAY %v
  22. Runtime tunable:
  23. MUL_KARATSUBA_CUTOFF %v
  24. SQR_KARATSUBA_CUTOFF %v
  25. MUL_TOOM_CUTOFF %v
  26. SQR_TOOM_CUTOFF %v
  27. MAX_ITERATIONS_ROOT_N %v
  28. FACTORIAL_MAX_N %v
  29. FACTORIAL_BINARY_SPLIT_CUTOFF %v
  30. FACTORIAL_BINARY_SPLIT_MAX_RECURSIONS %v
  31. `, _DIGIT_BITS,
  32. _MIN_DIGIT_COUNT,
  33. _MAX_DIGIT_COUNT,
  34. _DEFAULT_DIGIT_COUNT,
  35. _MAX_COMBA,
  36. _WARRAY,
  37. MUL_KARATSUBA_CUTOFF,
  38. SQR_KARATSUBA_CUTOFF,
  39. MUL_TOOM_CUTOFF,
  40. SQR_TOOM_CUTOFF,
  41. MAX_ITERATIONS_ROOT_N,
  42. FACTORIAL_MAX_N,
  43. FACTORIAL_BINARY_SPLIT_CUTOFF,
  44. FACTORIAL_BINARY_SPLIT_MAX_RECURSIONS,
  45. );
  46. }
  47. print :: proc(name: string, a: ^Int, base := i8(10), print_name := true, newline := true, print_extra_info := false) {
  48. assert_if_nil(a);
  49. as, err := itoa(a, base);
  50. defer delete(as);
  51. cb := internal_count_bits(a);
  52. if print_name {
  53. fmt.printf("%v", name);
  54. }
  55. if err != nil {
  56. fmt.printf("%v (error: %v | %v)", name, err, a);
  57. }
  58. fmt.printf("%v", as);
  59. if print_extra_info {
  60. fmt.printf(" (base: %v, bits: %v (digits: %v), flags: %v)", base, cb, a.used, a.flags);
  61. }
  62. if newline {
  63. fmt.println();
  64. }
  65. }
  66. int_to_byte :: proc(v: ^Int) {
  67. err: Error;
  68. size: int;
  69. print("v: ", v);
  70. fmt.println();
  71. t := &Int{};
  72. defer destroy(t);
  73. if size, err = int_to_bytes_size(v); err != nil {
  74. fmt.printf("int_to_bytes_size returned: %v\n", err);
  75. return;
  76. }
  77. b1 := make([]u8, size, context.temp_allocator);
  78. err = int_to_bytes_big(v, b1);
  79. int_from_bytes_big(t, b1);
  80. fmt.printf("big: %v | err: %v\n", b1, err);
  81. int_from_bytes_big(t, b1);
  82. if internal_cmp_mag(t, v) != 0 {
  83. print("\tError parsing t: ", t);
  84. }
  85. if size, err = int_to_bytes_size(v); err != nil {
  86. fmt.printf("int_to_bytes_size returned: %v\n", err);
  87. return;
  88. }
  89. b2 := make([]u8, size, context.temp_allocator);
  90. err = int_to_bytes_big_python(v, b2);
  91. fmt.printf("big python: %v | err: %v\n", b2, err);
  92. if err == nil {
  93. int_from_bytes_big_python(t, b2);
  94. if internal_cmp_mag(t, v) != 0 {
  95. print("\tError parsing t: ", t);
  96. }
  97. }
  98. if size, err = int_to_bytes_size(v, true); err != nil {
  99. fmt.printf("int_to_bytes_size returned: %v\n", err);
  100. return;
  101. }
  102. b3 := make([]u8, size, context.temp_allocator);
  103. err = int_to_bytes_big(v, b3, true);
  104. fmt.printf("big signed: %v | err: %v\n", b3, err);
  105. int_from_bytes_big(t, b3, true);
  106. if internal_cmp(t, v) != 0 {
  107. print("\tError parsing t: ", t);
  108. }
  109. if size, err = int_to_bytes_size(v, true); err != nil {
  110. fmt.printf("int_to_bytes_size returned: %v\n", err);
  111. return;
  112. }
  113. b4 := make([]u8, size, context.temp_allocator);
  114. err = int_to_bytes_big_python(v, b4, true);
  115. fmt.printf("big signed python: %v | err: %v\n", b4, err);
  116. int_from_bytes_big_python(t, b4, true);
  117. if internal_cmp(t, v) != 0 {
  118. print("\tError parsing t: ", t);
  119. }
  120. }
  121. int_to_byte_little :: proc(v: ^Int) {
  122. err: Error;
  123. size: int;
  124. print("v: ", v);
  125. fmt.println();
  126. t := &Int{};
  127. defer destroy(t);
  128. if size, err = int_to_bytes_size(v); err != nil {
  129. fmt.printf("int_to_bytes_size returned: %v\n", err);
  130. return;
  131. }
  132. b1 := make([]u8, size, context.temp_allocator);
  133. err = int_to_bytes_little(v, b1);
  134. fmt.printf("little: %v | err: %v\n", b1, err);
  135. int_from_bytes_little(t, b1);
  136. if internal_cmp_mag(t, v) != 0 {
  137. print("\tError parsing t: ", t);
  138. }
  139. if size, err = int_to_bytes_size(v); err != nil {
  140. fmt.printf("int_to_bytes_size returned: %v\n", err);
  141. return;
  142. }
  143. b2 := make([]u8, size, context.temp_allocator);
  144. err = int_to_bytes_little_python(v, b2);
  145. fmt.printf("little python: %v | err: %v\n", b2, err);
  146. if err == nil {
  147. int_from_bytes_little_python(t, b2);
  148. if internal_cmp_mag(t, v) != 0 {
  149. print("\tError parsing t: ", t);
  150. }
  151. }
  152. if size, err = int_to_bytes_size(v, true); err != nil {
  153. fmt.printf("int_to_bytes_size returned: %v\n", err);
  154. return;
  155. }
  156. b3 := make([]u8, size, context.temp_allocator);
  157. err = int_to_bytes_little(v, b3, true);
  158. fmt.printf("little signed: %v | err: %v\n", b3, err);
  159. int_from_bytes_little(t, b3, true);
  160. if internal_cmp(t, v) != 0 {
  161. print("\tError parsing t: ", t);
  162. }
  163. if size, err = int_to_bytes_size(v, true); err != nil {
  164. fmt.printf("int_to_bytes_size returned: %v\n", err);
  165. return;
  166. }
  167. b4 := make([]u8, size, context.temp_allocator);
  168. err = int_to_bytes_little_python(v, b4, true);
  169. fmt.printf("little signed python: %v | err: %v\n", b4, err);
  170. int_from_bytes_little_python(t, b4, true);
  171. if internal_cmp(t, v) != 0 {
  172. print("\tError parsing t: ", t);
  173. }
  174. }
  175. demo :: proc() {
  176. a, b, c, d, e, f := &Int{}, &Int{}, &Int{}, &Int{}, &Int{}, &Int{};
  177. defer destroy(a, b, c, d, e, f);
  178. }
  179. main :: proc() {
  180. ta := mem.Tracking_Allocator{};
  181. mem.tracking_allocator_init(&ta, context.allocator);
  182. context.allocator = mem.tracking_allocator(&ta);
  183. demo();
  184. print_configation();
  185. print_timings();
  186. if len(ta.allocation_map) > 0 {
  187. for _, v in ta.allocation_map {
  188. fmt.printf("Leaked %v bytes @ %v\n", v.size, v.location);
  189. }
  190. }
  191. if len(ta.bad_free_array) > 0 {
  192. fmt.println("Bad frees:");
  193. for v in ta.bad_free_array {
  194. fmt.println(v);
  195. }
  196. }
  197. }