2022-04-07 18:33:05 +02:00

159 lines
4.1 KiB
Scheme

;; In Emacs, open this file in -*- Scheme -*- mode.
;; Integers
(def int? (fun (o) (@int? o)))
;; Basic arithmetic
(def +@2 (fun (x y) (@+ x y)))
(def +@3 (fun (x y z) (@+ x (@+ y z))))
(def +@4 (fun (x y z t) (@+ (@+ x y) (@+ z t))))
(def + +@2)
(def -@1 (fun (x) (@- 0 x)))
(def -@2 (fun (x y) (@- x y)))
(def - -@2)
(def *@2 (fun (x y) (@* x y)))
(def *@3 (fun (x y z) (@* x (@* y z))))
(def *@4 (fun (x y z t) (@* (@* x y) (@* z t))))
(def * *@2)
;; Truncated division
(def /t (fun (x y) (@/ x y)))
(def %t (fun (x y) (@% x y)))
;; Comparisons
(def <@2 (fun (x y) (@< x y)))
(def <@3 (fun (x y z) (and (@< x y) (@< y z))))
(def < <@2)
(def <=@2 (fun (x y) (@<= x y)))
(def <=@3 (fun (x y z) (and (@<= x y) (@<= y z))))
(def <= <=@2)
(def >@2 (fun (x y) (@< y x)))
(def >@3 (fun (x y z) (and (> x y) (> y z))))
(def > >@2)
(def >=@2 (fun (x y) (@<= y x)))
(def >=@3 (fun (x y z) (and (>= x y) (>= y z))))
(def >= >=@2)
(def = (fun (x y) (@= x y)))
(def != (fun (x y) (not (@= x y))))
(def int-min (fun (x y) (if (<= x y) x y)))
(def int-max (fun (x y) (if (>= x y) x y)))
;; Bitwise operations
(def int-shift-left (fun (x y) (if (>= y 31) 0 (@shift-left x y))))
(def int-shift-right (fun (x y) (if (>= y 31) 0 (@shift-right x y))))
(def int-bitwise-and (fun (x y) (@and x y)))
(def int-bitwise-or@2 (fun (x y) (@or x y)))
(def int-bitwise-or@3 (fun (x y z) (@or (@or x y) z)))
(def int-bitwise-or@4 (fun (x y z t) (@or (@or x y) (@or z t))))
(def int-bitwise-or int-bitwise-or@2)
(def int-bitwise-xor (fun (x y) (@xor x y)))
(def int-bitwise-not (fun (x) (@xor x #x7FFFFFFF)))
;; Return a mask for the n least-significant bits
(def int-mask
(fun (n) (- (int-shift-left 1 n) 1)))
;; Extract n bits from x, starting from bit s
(def int-extract
(fun (x s n)
(int-bitwise-and (int-shift-right x s) (int-mask n))))
(def int-even?
(fun (i)
(= 0 (int-bitwise-and 1 i))))
(def int-odd?
(fun (i)
(not (int-even? i))))
(def int-abs
(fun (i)
(if (< i 0) (- i) i)))
(def int-signum
(fun (i)
(cond ((< i 0) -1)
((= i 0) 0)
(#t 1))))
;; Adapted from "Hacker's Delight" by Henry Warren (2nd ed.)
(def int-count-leading-zeros
(fun (i)
(if (< i 0)
0
(rec loop ((i i) (n 31) (c 16))
(if (= 0 c)
(- n i)
(let ((y (@shift-right i c)))
(if (= 0 y)
(loop i n (@shift-right c 1))
(loop y (- n c) (@shift-right c 1)))))))))
;; Floored division
(def /f
(fun (x y)
(let ((qt (/t x y)) (rt (%t x y)))
(if (= (int-signum rt) (- (int-signum y)))
(- qt 1)
qt))))
(def %f
(fun (x y)
(let ((rt (%t x y)))
(if (= (int-signum rt) (- (int-signum y)))
(+ rt y)
rt))))
(def int-gcd
(fun (x y)
(rec loop ((x (int-abs x))
(y (int-abs y)))
(if (= 0 y)
x
(loop y (%t x y))))))
(def int-pow
(fun (x y)
(rec loop ((x x) (y y))
(cond ((= 0 y)
1)
((int-even? y)
(let ((t (loop x (/t y 2))))
(* t t)))
(#t
(* x (loop x (- y 1))))))))
(def int-read
(letrec ((loop
(fun (acc-f acc)
(let ((c (char-read)))
(if (char-digit? c)
(loop acc-f (acc-f (* acc 10) (char-digit->int c)))
acc)))))
(fun ()
(let ((c (char-read)))
(cond ((= c '-') (loop - 0))
((char-digit? c) (loop + (char-digit->int c)))
(#t 0))))))
(def int-print
(fun (i)
(if (< i 0) (char-print '-'))
(rec loop ((i (int-abs i)))
(if (> i 9) (loop (/t i 10)))
(char-print (int->char-digit (%t i 10))))))
(def int->char
(fun (i)
(@int->char i)))