rat.odin 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540
  1. package math_big
  2. import "core:builtin"
  3. import "core:intrinsics"
  4. import "core:math"
  5. Rat :: struct {
  6. a, b: Int,
  7. }
  8. rat_set_f64 :: proc(dst: ^Rat, f: f64, allocator := context.allocator) -> (err: Error) {
  9. assert_if_nil(dst)
  10. context.allocator = allocator
  11. EXP_MASK :: 1<<11 - 1
  12. bits := transmute(u64)f
  13. mantissa := bits & (1<<52 - 1)
  14. exp := int((bits>>52) & EXP_MASK)
  15. int_set_from_integer(&dst.b, 1) or_return
  16. switch exp {
  17. case EXP_MASK:
  18. dst.a.flags += {.Inf}
  19. return
  20. case 0:
  21. exp -= 1022
  22. case:
  23. mantissa |= 1<<52
  24. exp -= 1023
  25. }
  26. shift := 52 - exp
  27. for mantissa&1 == 0 && shift > 0 {
  28. mantissa >>= 1
  29. shift -= 1
  30. }
  31. int_set_from_integer(&dst.a, mantissa) or_return
  32. dst.a.sign = .Negative if f < 0 else .Zero_or_Positive
  33. if shift > 0 {
  34. internal_int_shl(&dst.b, &dst.b, shift) or_return
  35. } else {
  36. internal_int_shl(&dst.a, &dst.a, -shift) or_return
  37. }
  38. return internal_rat_norm(dst)
  39. }
  40. rat_set_f32 :: proc(dst: ^Rat, f: f32, allocator := context.allocator) -> (err: Error) {
  41. return rat_set_f64(dst, f64(f), allocator)
  42. }
  43. rat_set_f16 :: proc(dst: ^Rat, f: f16, allocator := context.allocator) -> (err: Error) {
  44. return rat_set_f64(dst, f64(f), allocator)
  45. }
  46. rat_set_frac :: proc{rat_set_frac_digit, rat_set_frac_int}
  47. rat_set_frac_digit :: proc(dst: ^Rat, a, b: DIGIT, allocator := context.allocator) -> (err: Error) {
  48. assert_if_nil(dst)
  49. if b == 0 {
  50. return .Division_by_Zero
  51. }
  52. context.allocator = allocator
  53. internal_set(&dst.a, a) or_return
  54. internal_set(&dst.b, b) or_return
  55. return internal_rat_norm(dst)
  56. }
  57. rat_set_frac_int :: proc(dst: ^Rat, a, b: ^Int, allocator := context.allocator) -> (err: Error) {
  58. assert_if_nil(dst)
  59. assert_if_nil(a, b)
  60. if internal_is_zero(b) {
  61. return .Division_by_Zero
  62. }
  63. context.allocator = allocator
  64. internal_set(&dst.a, a) or_return
  65. internal_set(&dst.b, b) or_return
  66. return internal_rat_norm(dst)
  67. }
  68. rat_set_int :: proc(dst: ^Rat, a: ^Int, allocator := context.allocator) -> (err: Error) {
  69. assert_if_nil(dst)
  70. assert_if_nil(a)
  71. context.allocator = allocator
  72. internal_set(&dst.a, a) or_return
  73. internal_set(&dst.b, 1) or_return
  74. return
  75. }
  76. rat_set_digit :: proc(dst: ^Rat, a: DIGIT, allocator := context.allocator) -> (err: Error) {
  77. assert_if_nil(dst)
  78. context.allocator = allocator
  79. internal_set(&dst.a, a) or_return
  80. internal_set(&dst.b, 1) or_return
  81. return
  82. }
  83. rat_set_rat :: proc(dst, x: ^Rat, allocator := context.allocator) -> (err: Error) {
  84. assert_if_nil(dst, x)
  85. context.allocator = allocator
  86. internal_set(&dst.a, &x.a) or_return
  87. internal_set(&dst.b, &x.b) or_return
  88. return
  89. }
  90. rat_set_u64 :: proc(dst: ^Rat, x: u64, allocator := context.allocator) -> (err: Error) {
  91. assert_if_nil(dst)
  92. context.allocator = allocator
  93. internal_set(&dst.a, x) or_return
  94. internal_set(&dst.b, 1) or_return
  95. return
  96. }
  97. rat_set_i64 :: proc(dst: ^Rat, x: i64, allocator := context.allocator) -> (err: Error) {
  98. assert_if_nil(dst)
  99. context.allocator = allocator
  100. internal_set(&dst.a, x) or_return
  101. internal_set(&dst.b, 1) or_return
  102. return
  103. }
  104. rat_copy :: proc(dst, src: ^Rat, minimize := false, allocator := context.allocator) -> (err: Error) {
  105. if (dst == src) { return nil }
  106. assert_if_nil(dst, src)
  107. context.allocator = allocator
  108. int_copy(&dst.a, &src.a, minimize, allocator) or_return
  109. int_copy(&dst.b, &src.b, minimize, allocator) or_return
  110. internal_rat_norm(dst) or_return
  111. return nil
  112. }
  113. internal_rat_destroy :: proc(rationals: ..^Rat) {
  114. rationals := rationals
  115. for z in &rationals {
  116. internal_int_destroy(&z.a, &z.b)
  117. }
  118. }
  119. internal_rat_norm :: proc(z: ^Rat, allocator := context.allocator) -> (err: Error) {
  120. assert_if_nil(z)
  121. context.allocator = allocator
  122. switch {
  123. case internal_is_zero(&z.a):
  124. z.a.sign = .Zero_or_Positive
  125. fallthrough
  126. case internal_is_zero(&z.b):
  127. int_set_from_integer(&z.b, 1) or_return
  128. case:
  129. sign := z.a.sign
  130. z.a.sign = .Zero_or_Positive
  131. z.b.sign = .Zero_or_Positive
  132. f := &Int{}
  133. internal_int_gcd(f, &z.a, &z.b) or_return
  134. if !internal_int_equals_digit(f, 1) {
  135. f.sign = .Zero_or_Positive
  136. internal_int_div(&z.a, &z.a, f) or_return
  137. internal_int_div(&z.b, &z.b, f) or_return
  138. }
  139. z.a.sign = sign
  140. }
  141. return
  142. }
  143. rat_swap :: proc(a, b: ^Rat) {
  144. assert_if_nil(a, b)
  145. #force_inline internal_swap(a, b)
  146. }
  147. internal_rat_swap :: #force_inline proc(a, b: ^Rat) {
  148. internal_int_swap(&a.a, &b.a)
  149. internal_int_swap(&a.b, &b.b)
  150. }
  151. rat_sign :: proc(z: ^Rat) -> Sign {
  152. if z == nil {
  153. return .Zero_or_Positive
  154. }
  155. return z.a.sign
  156. }
  157. rat_is_int :: proc(z: ^Rat) -> bool {
  158. assert_if_nil(z)
  159. return internal_is_zero(&z.a) || internal_int_equals_digit(&z.b, 1)
  160. }
  161. rat_is_zero :: proc(z: ^Rat) -> bool {
  162. return internal_rat_is_zero(z)
  163. }
  164. internal_rat_is_zero :: #force_inline proc(z: ^Rat) -> bool {
  165. assert_if_nil(z)
  166. return internal_is_zero(&z.a)
  167. }
  168. internal_int_mul_denom :: proc(dst, x, y: ^Int, allocator := context.allocator) -> (err: Error) {
  169. assert_if_nil(dst, x, y)
  170. context.allocator = allocator
  171. switch {
  172. case internal_is_zero(x) && internal_is_zero(y):
  173. return internal_set(dst, 1)
  174. case internal_is_zero(x):
  175. return internal_set(dst, y)
  176. case internal_is_zero(y):
  177. return internal_set(dst, x)
  178. }
  179. return int_mul(dst, x, y)
  180. }
  181. internal_int_scale_denom :: proc(dst, x, y: ^Int, allocator := context.allocator) -> (err: Error) {
  182. assert_if_nil(dst, x, y)
  183. if internal_is_zero(y) {
  184. return internal_set(dst, x)
  185. }
  186. int_mul(dst, x, y) or_return
  187. dst.sign = x.sign
  188. return
  189. }
  190. rat_add_rat :: proc(dst, x, y: ^Rat, allocator := context.allocator) -> (err: Error) {
  191. assert_if_nil(dst, x, y)
  192. context.allocator = allocator
  193. a1, a2: Int
  194. defer internal_destroy(&a1, &a2)
  195. internal_int_scale_denom(&a1, &x.a, &y.b) or_return
  196. internal_int_scale_denom(&a2, &y.a, &x.b) or_return
  197. int_add(&dst.a, &a1, &a2) or_return
  198. internal_int_mul_denom(&dst.b, &x.b, &y.b) or_return
  199. return internal_rat_norm(dst)
  200. }
  201. rat_sub_rat :: proc(dst, x, y: ^Rat, allocator := context.allocator) -> (err: Error) {
  202. assert_if_nil(dst, x, y)
  203. context.allocator = allocator
  204. a1, a2 := &Int{}, &Int{}
  205. defer internal_destroy(a1, a2)
  206. internal_int_scale_denom(a1, &x.a, &y.b) or_return
  207. internal_int_scale_denom(a2, &y.a, &x.b) or_return
  208. int_sub(&dst.a, a1, a2) or_return
  209. internal_int_mul_denom(&dst.b, &x.b, &y.b) or_return
  210. return internal_rat_norm(dst)
  211. }
  212. rat_mul_rat :: proc(dst, x, y: ^Rat, allocator := context.allocator) -> (err: Error) {
  213. assert_if_nil(dst, x, y)
  214. context.allocator = allocator
  215. if x == y {
  216. internal_sqr(&dst.a, &x.a) or_return
  217. if internal_is_zero(&x.b) {
  218. internal_set(&dst.b, 1) or_return
  219. } else {
  220. internal_sqr(&dst.a, &x.b) or_return
  221. }
  222. return
  223. }
  224. int_mul(&dst.a, &x.a, &y.a) or_return
  225. internal_int_mul_denom(&dst.b, &x.b, &y.b) or_return
  226. return internal_rat_norm(dst)
  227. }
  228. rat_div_rat :: proc(dst, x, y: ^Rat, allocator := context.allocator) -> (err: Error) {
  229. if internal_rat_is_zero(y) {
  230. return .Division_by_Zero
  231. }
  232. context.allocator = allocator
  233. a, b := &Int{}, &Int{}
  234. defer internal_destroy(a, b)
  235. internal_int_scale_denom(a, &x.a, &y.b) or_return
  236. internal_int_scale_denom(b, &y.a, &x.b) or_return
  237. internal_set(&dst.a, a) or_return
  238. internal_set(&dst.b, b) or_return
  239. internal_int_abs(&dst.a, &dst.a)
  240. internal_int_abs(&dst.b, &dst.b)
  241. dst.a.sign = .Negative if a.sign != b.sign else .Zero_or_Positive
  242. return internal_rat_norm(dst)
  243. }
  244. rat_abs :: proc(dst, x: ^Rat, allocator := context.allocator) -> (err: Error) {
  245. rat_set_rat(dst, x, allocator) or_return
  246. internal_abs(&dst.a, &dst.a, allocator) or_return
  247. return
  248. }
  249. rat_neg :: proc(dst, x: ^Rat, allocator := context.allocator) -> (err: Error) {
  250. rat_set_rat(dst, x, allocator) or_return
  251. internal_neg(&dst.a, &dst.a, allocator) or_return
  252. return
  253. }
  254. rat_is_positive :: proc(z: ^Rat, allocator := context.allocator) -> (ok: bool, err: Error) {
  255. assert_if_nil(z)
  256. a := int_is_positive(&z.a, allocator) or_return
  257. b := int_is_positive(&z.b, allocator) or_return
  258. return !(a ~ b), nil
  259. }
  260. rat_is_negative :: proc(z: ^Rat, allocator := context.allocator) -> (ok: bool, err: Error) {
  261. assert_if_nil(z)
  262. a := int_is_positive(&z.a, allocator) or_return
  263. b := int_is_positive(&z.b, allocator) or_return
  264. return (a ~ b), nil
  265. }
  266. rat_is_even :: proc(z: ^Rat, allocator := context.allocator) -> (ok: bool, err: Error) {
  267. assert_if_nil(z)
  268. if rat_is_int(z) {
  269. return int_is_even(&z.a, allocator)
  270. }
  271. return false, nil
  272. }
  273. rat_is_odd :: proc(z: ^Rat, allocator := context.allocator) -> (ok: bool, err: Error) {
  274. assert_if_nil(z)
  275. if rat_is_int(z) {
  276. return int_is_odd(&z.a, allocator)
  277. }
  278. return false, nil
  279. }
  280. rat_to_f16 :: proc(z: ^Rat, allocator := context.allocator) -> (f: f16, exact: bool, err: Error) {
  281. assert_if_nil(z)
  282. return internal_rat_to_float(f16, z, allocator)
  283. }
  284. rat_to_f32 :: proc(z: ^Rat, allocator := context.allocator) -> (f: f32, exact: bool, err: Error) {
  285. assert_if_nil(z)
  286. return internal_rat_to_float(f32, z, allocator)
  287. }
  288. rat_to_f64 :: proc(z: ^Rat, allocator := context.allocator) -> (f: f64, exact: bool, err: Error) {
  289. assert_if_nil(z)
  290. return internal_rat_to_float(f64, z, allocator)
  291. }
  292. internal_rat_to_float :: proc($T: typeid, z: ^Rat, allocator := context.allocator) -> (f: T, exact: bool, err: Error) where intrinsics.type_is_float(T) {
  293. FSIZE :: 8*size_of(T)
  294. when FSIZE == 16 {
  295. MSIZE :: 10
  296. } else when FSIZE == 32 {
  297. MSIZE :: 23
  298. } else when FSIZE == 64 {
  299. MSIZE :: 52
  300. } else {
  301. #panic("unsupported float type")
  302. }
  303. MSIZE1 :: MSIZE+1
  304. MSIZE2 :: MSIZE+2
  305. ESIZE :: FSIZE - MSIZE1
  306. EBIAS :: 1<<(ESIZE-1) - 1
  307. EMIN :: 1 - EBIAS
  308. EMAX :: EBIAS
  309. assert_if_nil(z)
  310. a, b := &z.a, &z.b
  311. context.allocator = allocator
  312. alen := internal_count_bits(a)
  313. if alen == 0 {
  314. return 0, true, nil
  315. }
  316. blen := internal_count_bits(b)
  317. if blen == 0 {
  318. return T(math.nan_f64()), false, .Division_by_Zero
  319. }
  320. has_sign := a.sign != b.sign
  321. defer if has_sign {
  322. f = -builtin.abs(f)
  323. }
  324. exp := alen - blen
  325. a2, b2 := &Int{}, &Int{}
  326. defer internal_destroy(a2, b2)
  327. internal_int_abs(a2, a) or_return
  328. internal_int_abs(b2, b) or_return
  329. if shift := MSIZE2 - exp; shift > 0 {
  330. internal_int_shl(a2, a2, shift) or_return
  331. } else if shift < 0 {
  332. internal_int_shl(b2, b2, -shift) or_return
  333. }
  334. q, r := &Int{}, &Int{}
  335. defer internal_destroy(q, r)
  336. internal_int_divmod(q, r, a2, b2) or_return
  337. has_rem := !internal_is_zero(r)
  338. mantissa := internal_int_get_u64(q) or_return
  339. if mantissa>>MSIZE2 == 1 {
  340. if mantissa&1 == 1 {
  341. has_rem = true
  342. }
  343. mantissa >>= 1
  344. exp += 1
  345. }
  346. assert(mantissa>>MSIZE1 == 1, "invalid bit result")
  347. if EMIN-MSIZE <= exp && exp <= EMIN {
  348. shift := uint(EMIN - (exp - 1))
  349. lost_bits := mantissa & (1<<shift - 1)
  350. has_rem ||= lost_bits != 0
  351. mantissa >>= shift
  352. exp = 2 - EBIAS // exp + shift
  353. }
  354. exact = !has_rem
  355. if mantissa&1 != 0 {
  356. exact = false
  357. if has_rem || mantissa&2 != 0 {
  358. mantissa += 1
  359. if mantissa >= 1<<MSIZE2 {
  360. mantissa >>= 1
  361. exp += 1
  362. }
  363. }
  364. }
  365. mantissa >>= 1
  366. f = T(math.ldexp(f64(mantissa), exp-MSIZE1))
  367. if math.is_inf(f, 0) {
  368. exact = false
  369. }
  370. return
  371. }
  372. rat_compare :: proc(x, y: ^Rat, allocator := context.allocator) -> (comparison: int, error: Error) {
  373. assert_if_nil(x, y)
  374. context.allocator = allocator
  375. a, b: Int
  376. internal_init_multi(&a, &b) or_return
  377. defer internal_destroy(&a, &b)
  378. internal_int_scale_denom(&a, &x.a, &y.b) or_return
  379. internal_int_scale_denom(&b, &y.a, &x.b) or_return
  380. return int_compare(&a, &b)
  381. }
  382. rat_add_int :: proc(dst, x: ^Rat, y: ^Int, allocator := context.allocator) -> (err: Error) {
  383. assert_if_nil(dst, x)
  384. assert_if_nil(y)
  385. z: Rat
  386. rat_set_int(&z, y, allocator) or_return
  387. defer internal_destroy(&z)
  388. return rat_add_rat(dst, x, &z, allocator)
  389. }
  390. rat_sub_int :: proc(dst, x: ^Rat, y: ^Int, allocator := context.allocator) -> (err: Error) {
  391. assert_if_nil(dst, x)
  392. assert_if_nil(y)
  393. z: Rat
  394. rat_set_int(&z, y, allocator) or_return
  395. defer internal_destroy(&z)
  396. return rat_sub_rat(dst, x, &z, allocator)
  397. }
  398. rat_mul_int :: proc(dst, x: ^Rat, y: ^Int, allocator := context.allocator) -> (err: Error) {
  399. assert_if_nil(dst, x)
  400. assert_if_nil(y)
  401. z: Rat
  402. rat_set_int(&z, y, allocator) or_return
  403. defer internal_destroy(&z)
  404. return rat_mul_rat(dst, x, &z, allocator)
  405. }
  406. rat_div_int :: proc(dst, x: ^Rat, y: ^Int, allocator := context.allocator) -> (err: Error) {
  407. if internal_is_zero(y) {
  408. return .Division_by_Zero
  409. }
  410. z: Rat
  411. rat_set_int(&z, y, allocator) or_return
  412. defer internal_destroy(&z)
  413. return rat_div_rat(dst, x, &z, allocator)
  414. }
  415. int_add_rat :: proc(dst: ^Rat, x: ^Int, y: ^Rat, allocator := context.allocator) -> (err: Error) {
  416. assert_if_nil(x)
  417. assert_if_nil(dst, y)
  418. w: Rat
  419. rat_set_int(&w, x, allocator) or_return
  420. defer internal_destroy(&w)
  421. return rat_add_rat(dst, &w, y, allocator)
  422. }
  423. int_sub_rat :: proc(dst: ^Rat, x: ^Int, y: ^Rat, allocator := context.allocator) -> (err: Error) {
  424. assert_if_nil(x)
  425. assert_if_nil(dst, y)
  426. w: Rat
  427. rat_set_int(&w, x, allocator) or_return
  428. defer internal_destroy(&w)
  429. return rat_sub_rat(dst, &w, y, allocator)
  430. }
  431. int_mul_rat :: proc(dst: ^Rat, x: ^Int, y: ^Rat, allocator := context.allocator) -> (err: Error) {
  432. assert_if_nil(x)
  433. assert_if_nil(dst, y)
  434. w: Rat
  435. rat_set_int(&w, x, allocator) or_return
  436. defer internal_destroy(&w)
  437. return rat_mul_rat(dst, &w, y, allocator)
  438. }
  439. int_div_rat :: proc(dst: ^Rat, x: ^Int, y: ^Rat, allocator := context.allocator) -> (err: Error) {
  440. if internal_is_zero(y) {
  441. return .Division_by_Zero
  442. }
  443. w: Rat
  444. rat_set_int(&w, x, allocator) or_return
  445. defer internal_destroy(&w)
  446. return rat_div_rat(dst, &w, y, allocator)
  447. }