~ chicken-core (chicken-5) f89539d7cd0df71361202687ca876a16f61781ed


commit f89539d7cd0df71361202687ca876a16f61781ed
Author:     Peter Bex <peter@more-magic.net>
AuthorDate: Sun Jan 25 20:07:11 2015 +0100
Commit:     Peter Bex <peter@more-magic.net>
CommitDate: Sun May 31 14:14:25 2015 +0200

    Make exact->inexact and inexact->exact aware of extended number types.
    Implement rounding operations: round, floor, ceiling, truncate.
    Add rationalize, remove ratnum restrictions from "Deviations from the standard".
    Update angle, log, exp, expt and the trig functions to accept extended numbers.
    
    After bootstrapping this with itself, numbers-string-conversion-tests works!

diff --git a/c-platform.scm b/c-platform.scm
index 383495a5..8808160e 100644
--- a/c-platform.scm
+++ b/c-platform.scm
@@ -127,7 +127,8 @@
     integer->char eof-object? vector-length string-length string-ref string-set! vector-ref 
     vector-set! char=? char<? char>? char>=? char<=? gcd lcm reverse symbol? string->symbol
     number? complex? real? integer? rational? odd? even? positive? negative? exact? inexact?
-    max min quotient remainder modulo floor ceiling truncate round exact->inexact inexact->exact
+    max min quotient remainder modulo floor ceiling truncate round rationalize
+    exact->inexact inexact->exact
     exp log sin expt sqrt cos tan asin acos atan number->string string->number char-ci=?
     char-ci<? char-ci>? char-ci>=? char-ci<=? char-alphabetic? char-whitespace? char-numeric?
     char-lower-case? char-upper-case? char-upcase char-downcase string? string=? string>? string<?
@@ -539,17 +540,6 @@
 (rewrite 'fpneg 16 1 "C_a_i_flonum_negate" #f words-per-flonum)
 (rewrite 'fpgcd 16 2 "C_a_i_flonum_gcd" #f words-per-flonum)
 
-(rewrite 'exp 16 1 "C_a_i_exp" #t words-per-flonum)
-(rewrite 'sin 16 1 "C_a_i_sin" #t words-per-flonum)
-(rewrite 'cos 16 1 "C_a_i_cos" #t words-per-flonum)
-(rewrite 'tan 16 1 "C_a_i_tan" #t words-per-flonum)
-(rewrite 'log 16 1 "C_a_i_log" #t words-per-flonum)
-(rewrite 'asin 16 1 "C_a_i_asin" #t words-per-flonum)
-(rewrite 'acos 16 1 "C_a_i_acos" #t words-per-flonum)
-(rewrite 'atan 16 1 "C_a_i_atan" #t words-per-flonum)
-(rewrite 'sqrt 16 1 "C_a_i_sqrt" #t words-per-flonum)
-(rewrite 'atan 16 2 "C_a_i_atan2" #t words-per-flonum)
-
 (rewrite 'zero? 5 "C_eqp" 0 'fixnum)
 (rewrite 'zero? 2 1 "C_i_zerop" #t)
 (rewrite 'zero? 2 1 "C_u_i_zerop" #f)
@@ -570,7 +560,6 @@
 (rewrite 'vector-length 2 1 "C_i_vector_length" #t)
 (rewrite '##sys#vector-length 2 1 "C_i_vector_length" #t)
 (rewrite 'string-length 2 1 "C_i_string_length" #t)
-(rewrite 'inexact->exact 2 1 "C_i_inexact_to_exact" #t)
 
 (rewrite '##sys#check-exact 2 1 "C_i_check_exact" #t)
 (rewrite '##sys#check-number 2 1 "C_i_check_number" #t)
@@ -625,8 +614,6 @@
 (rewrite 'lcm 18 1)
 (rewrite 'list 18 '())
 
-(rewrite 'exact->inexact 16 1 "C_a_i_exact_to_inexact" #t 4) ; words-per-flonum
-
 (rewrite '= 17 2 "C_i_nequalp")
 (rewrite '> 17 2 "C_i_greaterp")
 (rewrite '< 17 2 "C_i_lessp")
diff --git a/chicken.h b/chicken.h
index 00871212..1e7ce304 100644
--- a/chicken.h
+++ b/chicken.h
@@ -664,7 +664,6 @@ static inline int isinf_ld (long double x)
 #define C_BAD_ARGUMENT_TYPE_NO_REAL_ERROR             50
 #define C_BAD_ARGUMENT_TYPE_COMPLEX_NO_ORDERING_ERROR 51
 
-
 /* Platform information */
 #if defined(C_BIG_ENDIAN)
 # define C_MACHINE_BYTE_ORDER "big-endian"
@@ -1903,13 +1902,13 @@ C_fctexport void C_ccall C_u_integer_quotient(C_word c, C_word self, C_word k, C
 C_fctexport void C_ccall C_basic_remainder(C_word c, C_word self, C_word k, C_word x, C_word y) C_noret;
 C_fctexport void C_ccall C_u_integer_remainder(C_word c, C_word self, C_word k, C_word x, C_word y) C_noret;
 C_fctexport void C_ccall C_basic_divrem(C_word c, C_word self, C_word k, C_word x, C_word y) C_noret;
-void C_ccall C_u_integer_divrem(C_word c, C_word self, C_word k, C_word x, C_word y) C_noret;
+C_fctexport void C_ccall C_u_integer_divrem(C_word c, C_word self, C_word k, C_word x, C_word y) C_noret;
+C_fctexport void C_ccall C_u_flo_to_int(C_word c, C_word self, C_word k, C_word x) C_noret;
 C_fctexport void C_ccall C_nequalp(C_word c, C_word closure, C_word k, ...) C_noret;
 C_fctexport void C_ccall C_greaterp(C_word c, C_word closure, C_word k, ...) C_noret;
 C_fctexport void C_ccall C_lessp(C_word c, C_word closure, C_word k, ...) C_noret;
 C_fctexport void C_ccall C_greater_or_equal_p(C_word c, C_word closure, C_word k, ...) C_noret;
 C_fctexport void C_ccall C_less_or_equal_p(C_word c, C_word closure, C_word k, ...) C_noret;
-C_fctexport void C_ccall C_expt(C_word c, C_word closure, C_word k, C_word n1, C_word n2) C_noret;
 C_fctexport void C_ccall C_gc(C_word c, C_word closure, C_word k, ...) C_noret;
 C_fctexport void C_ccall C_open_file_port(C_word c, C_word closure, C_word k, C_word port, C_word channel, C_word mode) C_noret;
 C_fctexport void C_ccall C_allocate_vector(C_word c, C_word closure, C_word k, C_word size, C_word type, C_word init, C_word align8) C_noret;
@@ -2014,6 +2013,7 @@ C_fctexport C_word C_fcall C_i_memv(C_word x, C_word lst) C_regparm;
 C_fctexport C_word C_fcall C_i_member(C_word x, C_word lst) C_regparm;
 C_fctexport C_word C_fcall C_i_length(C_word lst) C_regparm;
 C_fctexport C_word C_fcall C_u_i_length(C_word lst) C_regparm;
+/* XXX TODO OBSOLETE: This can be removed after recompiling c-platform.scm */
 C_fctexport C_word C_fcall C_i_inexact_to_exact(C_word n) C_regparm;
 C_fctexport C_word C_fcall C_i_check_closure_2(C_word x, C_word loc) C_regparm;
 C_fctexport C_word C_fcall C_i_check_exact_2(C_word x, C_word loc) C_regparm;
@@ -2090,9 +2090,11 @@ C_fctexport double C_fcall C_milliseconds(void) C_regparm;
 C_fctexport double C_fcall C_cpu_milliseconds(void) C_regparm;
 C_fctexport double C_fcall C_bignum_to_double(C_word bignum) C_regparm;
 
+
 C_fctexport C_word C_fcall C_a_i_cpu_time(C_word **a, int c, C_word buf) C_regparm;
 /* XXX TODO OBSOLETE: This can be removed after recompiling c-platform.scm */
 C_fctexport C_word C_fcall C_a_i_string_to_number(C_word **a, int c, C_word str, C_word radix) C_regparm;
+/* XXX TODO OBSOLETE: This can be removed after recompiling c-platform.scm */
 C_fctexport C_word C_fcall C_a_i_exact_to_inexact(C_word **a, int c, C_word n) C_regparm;
 C_fctexport C_word C_fcall C_i_file_exists_p(C_word name, C_word file, C_word dir) C_regparm;
 
diff --git a/library.scm b/library.scm
index cdf65e04..347ec574 100644
--- a/library.scm
+++ b/library.scm
@@ -35,8 +35,8 @@
 	##sys#print-exit
 	##sys#format-here-doc-warning
 	exit-in-progress
-        maximal-string-length
-	make-complex ratnum rat+/-
+        maximal-string-length find-ratio-between find-ratio
+	make-complex flonum->ratnum ratnum rat+/- minimum-denorm-flonum-expt
 	+maximum-allowed-exponent+ mantexp->dbl ldexp round-quotient
 	##sys#string->compnum)
   (not inline ##sys#user-read-hook ##sys#error-hook ##sys#signal-hook ##sys#schedule
@@ -316,10 +316,8 @@ EOF
     (##sys#error-bad-integer x (and (pair? loc) (car loc))) ) )
 
 (define (##sys#check-real x . loc)
-  (unless (##core#inline "C_i_realp" x) 
-    (##sys#error-hook
-     (foreign-value "C_BAD_ARGUMENT_TYPE_NO_REAL_ERROR" int)
-     (and (pair? loc) (car loc)) x) ) )
+  (unless (##core#inline "C_i_realp" x)
+    (##sys#error-bad-real x (and (pair? loc) (car loc))) ) )
 
 (define (##sys#check-range i from to . loc)
   (##sys#check-exact i loc)
@@ -448,6 +446,14 @@ EOF
   (##sys#error-hook
    (foreign-value "C_BAD_ARGUMENT_TYPE_NO_INTEGER_ERROR" int) loc arg))
 
+(define (##sys#error-bad-inexact arg #!optional loc)
+  (##sys#error-hook
+   (foreign-value "C_CANT_REPRESENT_INEXACT_ERROR" int) loc arg))
+
+(define (##sys#error-bad-real arg #!optional loc)
+  (##sys#error-hook
+   (foreign-value "C_BAD_ARGUMENT_TYPE_NO_REAL_ERROR" int) loc arg))
+
 (define (##sys#error-bad-base arg #!optional loc)
   (##sys#error-hook
    (foreign-value "C_BAD_ARGUMENT_TYPE_BAD_BASE_ERROR" int) loc arg))
@@ -939,7 +945,6 @@ EOF
 (define (inexact? x) (##core#inline "C_i_inexactp" x))
 (define ##sys#exact? exact?)
 (define ##sys#inexact? inexact?)
-(define expt (##core#primitive "C_expt"))
 (define (##sys#fits-in-int? n) (##core#inline "C_fits_in_int_p" n))
 (define (##sys#fits-in-unsigned-int? n) (##core#inline "C_fits_in_unsigned_int_p" n))
 (define (##sys#flonum-in-fixnum-range? n) (##core#inline "C_flonum_in_fixnum_range_p" n))
@@ -985,7 +990,9 @@ EOF
 
 (define (angle n)
   (##sys#check-number n 'angle)
-  (if (< n 0) (fp* 2.0 (acos 0.0)) 0.0) )
+  (##core#inline_allocate ("C_a_i_atan2" 4)
+			  (exact->inexact (imag-part n))
+			  (exact->inexact (real-part n))))
 
 (define (magnitude x)
   (cond ((cplxnum? x)
@@ -1063,9 +1070,93 @@ EOF
 	((< n 0) (if (##sys#exact? n) -1 -1.0))
 	(else (if (##sys#exact? n) 0 0.0) ) ) )
 
-;; hooks for numbers
-(define (exact->inexact n) (##core#inline_allocate ("C_a_i_exact_to_inexact" 4) n))
-(define (inexact->exact n) (##core#inline "C_i_inexact_to_exact" n))
+(define (flonum->ratnum x)
+  ;; Try to multiply by two until we reach an integer
+  (define (float-fraction-length x)
+    (do ((x x (fp* x 2.0))
+         (i 0 (fx+ i 1)))
+        ((##core#inline "C_u_i_fpintegerp" x) i)))
+
+  (define (deliver y d)
+    (let* ((q (##sys#integer-power 2 (float-fraction-length y)))
+           (scaled-y (##sys#*-2 y (exact->inexact q))))
+      (if (finite? scaled-y)          ; Shouldn't this always be true?
+          (##sys#/-2 (##sys#/-2 ((##core#primitive "C_u_flo_to_int") scaled-y) q) d)
+          (##sys#error-bad-inexact x 'inexact->exact))))
+
+  (if (and (fp< x 1.0)         ; Watch out for denormalized numbers
+           (fp> x -1.0))       ; XXX: Needs a test, it seems pointless
+      (deliver (##sys#*-2 x (expt 2.0 flonum-precision))
+               ;; Can be bignum (is on 32-bit), so must wait until after init.
+               ;; We shouldn't need to calculate this every single time, tho..
+               (##sys#integer-power 2 flonum-precision))
+      (deliver x 1)))
+
+(define (inexact->exact x)
+  (cond ((exact? x) x)
+        ((##core#inline "C_i_flonump" x)
+         (cond ((##core#inline "C_u_i_fpintegerp" x)
+                ((##core#primitive "C_u_flo_to_int") x))
+               ((##core#inline "C_u_i_flonum_finitep" x)
+                (flonum->ratnum x))
+               (else (##sys#error-bad-inexact x 'inexact->exact))))
+        ((cplxnum? x)
+         (make-complex (inexact->exact (%cplxnum-real x))
+                       (inexact->exact (%cplxnum-imag x))))
+        (else (##sys#error-bad-number x 'inexact->exact))))
+
+;; Exponent of the lowest allowed flonum; if we get any lower we get zero.
+;; In other words, this is the first (smallest) flonum after 0.
+;; Equal to (expt 2.0 (- flonum-minimum-exponent flonum-precision))
+(define minimum-denorm-flonum-expt (fx- flonum-minimum-exponent flonum-precision))
+
+(define (exact->inexact x)
+  (cond ((##core#inline "C_fixnump" x)
+         (##core#inline_allocate ("C_a_i_fix_to_flo" 4) x))
+        ((##core#inline "C_i_flonump" x) x)
+        ((##core#inline "C_i_bignump" x)
+         (##core#inline_allocate ("C_a_u_i_big_to_flo" 4) x))
+        ((ratnum? x)
+         ;; This tries to keep the numbers within representable ranges
+         ;; and tries to drop as few significant digits as possible by
+         ;; bringing the two numbers to within the same powers of two.
+         ;; See algorithms M & N in Knuth, 4.2.1
+         (let* ((n1 (%ratnum-numerator x))
+                (an ((##core#primitive "C_u_integer_abs") n1))
+                (d1 (%ratnum-denominator x))
+                ;; Approximate distance between the numbers in powers
+                ;; of 2 ie, 2^e-1 < n/d < 2^e+1 (e is the *un*biased
+                ;; value of e_w in M2)
+                ;; XXX: What if b != 2 (ie, flonum-radix is not 2)?
+                (e (fx- (integer-length an) (integer-length d1)))
+                (rnd (lambda (n d e) ; Here, 1 <= n/d < 2  (normalized) [N5]
+                       ;; Cannot shift above the available precision,
+                       ;; and can't have an exponent that's below the
+                       ;; minimum flonum exponent.
+                       (let* ((s (min (fx- flonum-precision 1)
+                                      (fx- e minimum-denorm-flonum-expt)))
+                              (normalized (##sys#/-2 (arithmetic-shift n s) d))
+                              (r (round normalized))
+                              (fraction (exact->inexact r))
+                              (exp (fx- e s)))
+                         (let ((res (fp* fraction (expt 2.0 exp))))
+                           (if (negative? n1) (##sys#--2 0 res) res)))))
+                (scale (lambda (n d)  ; Here, 1/2 <= n/d < 2   [N3]
+                         (if (##sys#<-2 n d) ; n/d < 1?
+                             ;; Scale left [N3]; only needed once (see note in M3)
+                             (rnd (arithmetic-shift n 1) d (fx- e 1))
+                             ;; Already normalized
+                             (rnd n d e)))))
+           ;; After this step, which shifts the smaller number to
+           ;; align with the larger, "f" in algorithm N is represented
+           ;; in the procedures above by n/d.
+           (if (negative? e)
+               (scale (arithmetic-shift an (##sys#--2 0 e)) d1)
+               (scale an (arithmetic-shift d1 e)))))
+        ((cplxnum? x)
+         (make-complex (exact->inexact (%cplxnum-real x))
+                       (exact->inexact (%cplxnum-imag x))))
+        (else (##sys#error-bad-number x 'exact->inexact))))
 
 (define ##sys#exact->inexact exact->inexact)
 (define ##sys#inexact->exact inexact->exact)
@@ -1147,8 +1238,8 @@ EOF
          (%make-ratnum (##sys#integer-negate (%ratnum-numerator x))
                        (%ratnum-denominator x)))
         ((cplxnum? x)
-         (%make-complex (##sys#negate (compnum-real x))
-                        (##sys#negate (compnum-imag x))))
+         (make-complex (##sys#negate (%cplxnum-real x))
+		       (##sys#negate (%cplxnum-imag x))))
         (else (##sys#error-bad-number x '-)) ) ) ; loc?
 
 (define (##sys#extended-minus x y)
@@ -1289,28 +1380,66 @@ EOF
         (else (##sys#error-bad-number y '/))) )
 
 (define (floor x)
-  (##sys#check-number x 'floor)
-  (if (##core#inline "C_fixnump" x) 
-      x
-      (fpfloor x) ) )
+  (cond ((exact-integer? x) x)
+        ((##core#inline "C_i_flonump" x) (fpfloor x))
+        ;; (floor x) = greatest integer <= x
+        ((ratnum? x) (let* ((n (%ratnum-numerator x))
+                            (q (quotient n (%ratnum-denominator x))))
+                       (if (##sys#>=-2 n 0) q (##sys#--2 q 1))))
+        (else (##sys#error-bad-real x 'floor))))
 
 (define (ceiling x)
-  (##sys#check-number x 'ceiling)
-  (if (##core#inline "C_fixnump" x) 
-      x
-      (fpceiling x) ) )
+  (cond ((exact-integer? x) x)
+        ((##core#inline "C_i_flonump" x) (fpceiling x))
+        ;; (ceiling x) = smallest integer >= x
+        ((ratnum? x) (let* ((n (%ratnum-numerator x))
+                            (q (quotient n (%ratnum-denominator x))))
+                       (if (##sys#>=-2 n 0) (##sys#+-2 q 1) q)))
+        (else (##sys#error-bad-real x 'ceiling))) )
 
 (define (truncate x)
-  (##sys#check-number x 'truncate)
-  (if (##core#inline "C_fixnump" x) 
-      x
-      (fptruncate x) ) )
+  (cond ((exact-integer? x) x)
+        ((##core#inline "C_i_flonump" x) (fptruncate x))
+        ;; (rational-truncate x) = integer of largest magnitude <= (abs x)
+        ((ratnum? x) (quotient (%ratnum-numerator x)
+			       (%ratnum-denominator x)))
+        (else (##sys#error-bad-real x 'truncate))))
 
 (define (round x)
-  (##sys#check-number x 'round)
-  (if (##core#inline "C_fixnump" x) 
-      x
-      (##core#inline_allocate ("C_a_i_flonum_round_proper" 4) x)))
+  (cond ((exact-integer? x) x)
+        ((##core#inline "C_i_flonump" x)
+         (##core#inline_allocate ("C_a_i_flonum_round_proper" 4) x))
+        ((ratnum? x)
+	 (let* ((x+1/2 (##sys#+-2 x (%make-ratnum 1 2)))
+		(r (floor x+1/2)))
+	   (if (and (##sys#=-2 r x+1/2) (odd? r)) (##sys#--2 r 1) r)))
+        (else (##sys#error-bad-real x 'round))))
+
+(define (find-ratio-between x y)
+  (define (sr x y)
+    (let ((fx (inexact->exact (floor x))) 
+	  (fy (inexact->exact (floor y))))
+      (cond ((not (##sys#<-2 fx x)) (list fx 1))
+	    ((##sys#=-2 fx fy) 
+	     (let ((rat (sr (##sys#/-2 1 (##sys#--2 y fy))
+			    (##sys#/-2 1 (##sys#--2 x fx)))))
+	       (list (##sys#+-2 (cadr rat) (##sys#*-2 fx (car rat)))
+		     (car rat))))
+	    (else (list (##sys#+-2 1 fx) 1)))))
+  (cond ((##sys#<-2 y x) (find-ratio-between y x))
+	((not (##sys#<-2 x y)) (list x 1))
+	((positive? x) (sr x y))
+	((negative? y) (let ((rat (sr (##sys#--2 0 y) (##sys#--2 0 x))))
+                         (list (##sys#--2 0 (car rat)) (cadr rat))))
+	(else '(0 1))))
+
+(define (find-ratio x e) (find-ratio-between (##sys#--2 x e) (##sys#+-2 x e)))
+
+(define (rationalize x e)
+  (let ((result (apply ##sys#/-2 (find-ratio x e))))
+    (if (or (inexact? x) (inexact? e))
+        (exact->inexact result)
+        result)))
 
 (define quotient (##core#primitive "C_basic_quotient"))
 (define ##sys#integer-quotient (##core#primitive "C_u_integer_quotient"))
@@ -1361,34 +1490,113 @@ EOF
                 (##sys#slot xs 1)) ) ) )  )
 
 (define (exp n)
-  (##core#inline_allocate ("C_a_i_exp" 4) n) )
-
-(define (log n)
-  (##core#inline_allocate ("C_a_i_log" 4) n) )
+  (##sys#check-number n 'exp)
+  (if (cplxnum? n)
+      (##sys#*-2 (##core#inline_allocate ("C_a_i_exp" 4)
+					 (exact->inexact (%cplxnum-real n)))
+		 (let ((p (%cplxnum-imag n)))
+		   (make-complex
+		    (##core#inline_allocate
+		     ("C_a_i_cos" 4) (exact->inexact p))
+		    (##core#inline_allocate
+		     ("C_a_i_sin" 4) (exact->inexact p)) ) ) )
+      (##core#inline_allocate ("C_a_i_flonum_exp" 4) (exact->inexact n))))
+
+(define (##sys#log-1 x)		       ; log_e(x)
+  (cond
+   ((eq? x 0)			       ; Exact zero?  That's undefined
+    (##sys#signal-hook #:arithmetic-error 'log "log of exact 0 is undefined" x))
+   ;; avoid calling inexact->exact on X here (to avoid overflow?)
+   ((or (cplxnum? x) (negative? x)) ; General case
+    (##sys#+-2 (##sys#log-1 (magnitude x))
+	       (##sys#*-2 (make-complex 0 1) (angle x))))
+   (else ; Real number case (< already ensured the argument type is a number)
+    (##core#inline_allocate ("C_a_i_log" 4) (exact->inexact x)))))
+
+(define (log a #!optional b)
+  (if b (##sys#/-2 (##sys#log-1 a) (##sys#log-1 b)) (##sys#log-1 a)))
+
+;; OBSOLETE: These can be removed after integration into core and
+;; bootstrapping, when the compiler can write these objects natively.
+(define %i (make-complex 0 1))
+(define %-i (make-complex 0 -1))
+(define %i2 (make-complex 0 2))
 
 (define (sin n)
-  (##core#inline_allocate ("C_a_i_sin" 4) n) )
+  (##sys#check-number n 'sin)
+  (if (cplxnum? n)
+      (let ((in (##sys#*-2 %i n)))
+	(##sys#/-2 (##sys#--2 (exp in) (exp (##sys#--2 0 in))) %i2))
+      (##core#inline_allocate ("C_a_i_sin" 4) (exact->inexact n))))
 
 (define (cos n)
-  (##core#inline_allocate ("C_a_i_cos" 4) n) )
+  (##sys#check-number n 'cos)
+  (if (cplxnum? n)
+      (let ((in (##sys#*-2 %i n)))
+	(##sys#/-2 (##sys#+-2 (exp in) (exp (##sys#--2 0 in))) 2) )
+      (##core#inline_allocate ("C_a_i_cos" 4) (exact->inexact n))))
 
 (define (tan n)
-  (##core#inline_allocate ("C_a_i_tan" 4) n) )
+  (##sys#check-number n 'tan)
+  (if (cplxnum? n)
+      (##sys#/-2 (sin n) (cos n))
+      (##core#inline_allocate ("C_a_i_tan" 4) (exact->inexact n))))
 
+;; General case: sin^{-1}(z) = -i\ln(iz + \sqrt{1-z^2})
 (define (asin n)
-  (##core#inline_allocate ("C_a_i_asin" 4) n) )
-
-(define (acos n)
-  (##core#inline_allocate ("C_a_i_acos" 4) n) )
-
-(define (sqrt n)
-  (##core#inline_allocate ("C_a_i_sqrt" 4) n) )
-
-(define (atan n1 . n2)
-  (if (null? n2) 
-      (##core#inline_allocate ("C_a_i_atan" 4) n1)
-      (let ([n2 (car n2)])
-	(##core#inline_allocate ("C_a_i_atan2" 4) n1 n2) ) ) )
+  (##sys#check-number n 'asin)
+  (cond ((and (##core#inline "C_i_flonump" n) (fp>= n -1.0) (fp<= n 1.0))
+         (##core#inline_allocate ("C_a_i_asin" 4) n))
+        ((and (##core#inline "C_fixnump" n) (fx>= n -1) (fx<= n 1))
+         (##core#inline_allocate ("C_a_i_asin" 4)
+                                 (##core#inline_allocate
+                                  ("C_a_i_fix_to_flo" 4) n)))
+        ;; General definition can return compnums
+        (else (##sys#*-2 %-i
+			 (##sys#log-1
+			  (##sys#+-2 (##sys#*-2 %i n)
+				     (##sys#sqrt/loc
+				      'asin (##sys#--2 1 (##sys#*-2 n n)))))))))
+
+;; General case:
+;; cos^{-1}(z) = 1/2\pi + i\ln(iz + \sqrt{1-z^2}) = 1/2\pi - sin^{-1}(z) = sin(1) - sin(z)
+(define acos
+  (let ((asin1 (##core#inline_allocate ("C_a_i_asin" 4) 1)))
+    (lambda (n)
+      (##sys#check-number n 'acos)
+      (cond ((and (##core#inline "C_i_flonump" n) (fp>= n -1.0) (fp<= n 1.0))
+             (##core#inline_allocate ("C_a_i_acos" 4) n))
+            ((and (##core#inline "C_fixnump" n) (fx>= n -1) (fx<= n 1))
+             (##core#inline_allocate ("C_a_i_acos" 4)
+                                     (##core#inline_allocate
+                                      ("C_a_i_fix_to_flo" 4) n)))
+            ;; General definition can return compnums
+            (else (##sys#--2 asin1 (asin n)))))))
+
+(define (atan n #!optional b)
+  (##sys#check-number n 'atan)
+  (cond ((cplxnum? n)
+	 (if b
+	     (##sys#error-bad-real n 'atan)
+	     (let ((in (##sys#*-2 %i n)))
+	       (##sys#/-2 (##sys#--2 (##sys#log-1 (##sys#+-2 1 in))
+				     (##sys#log-1 (##sys#--2 1 in))) %i2))))
+        (b
+	 (##core#inline_allocate
+	  ("C_a_i_atan2" 4) (exact->inexact n) (exact->inexact b)))
+	(else
+	 (##core#inline_allocate
+	  ("C_a_i_atan" 4) (exact->inexact n)))))
+
+;; TODO: replace this with an actual sqrt implementation
+(define (##sys#sqrt/loc loc x)
+  (##core#inline_allocate ("C_a_i_sqrt" 4) n))
+
+(define (sqrt x) (##sys#sqrt/loc 'sqrt x))
+
+;; TODO: unimplemented
+(define (##sys#exact-integer-nth-root/loc loc k n)
+  (error "not yet implemented"))
 
 (define (##sys#integer-power base e)
   (define (square x) (##sys#*-2 x x))
@@ -1402,6 +1610,55 @@ EOF
          (else
           (lp (##sys#*-2 res base) (##sys#--2 e2 1)))))))
 
+(define (expt a b)
+  (define (log-expt a b)
+    (exp (##sys#*-2 b (##sys#log-1 a))))
+  (define (slow-expt a b)
+    (if (eq? 0 a)
+        (##sys#signal-hook
+	 #:arithmetic-error 'expt
+	 "exponent of exact 0 with complex argument is undefined" a b)
+        (exp (##sys#*-2 b (##sys#log-1 a)))))
+  (cond ((not (number? a)) (##sys#error-bad-number a 'expt))
+        ((not (number? b)) (##sys#error-bad-number b 'expt))
+        ((and (ratnum? a) (not (inexact? b)))
+         ;; (n*d)^b = n^b * d^b = n^b * x^{-b}  | x = 1/b
+         ;; Hopefully faster than integer-power
+         (##sys#*-2 (expt (%ratnum-numerator a) b)
+		    (expt (%ratnum-denominator a) (##sys#negate b))))
+        ((ratnum? b)
+         ;; x^{a/b} = (x^{1/b})^a
+         (cond
+          ((exact-integer? a)
+           (if (negative? a)
+               (log-expt (exact->inexact a) (exact->inexact b))
+               (receive (ds^n r)
+                   (##sys#exact-integer-nth-root/loc
+		    'expt a (%ratnum-denominator b))
+                 (if (eq? r 0)
+                     (##sys#integer-power ds^n (%ratnum-numerator b))
+                     (##core#inline_allocate ("C_a_i_flonum_expt" 4)
+					     (exact->inexact a)
+					     (exact->inexact b))))))
+          ((##core#inline "C_i_flonump" a)
+           (log-expt a (exact->inexact b)))
+          (else (slow-expt a b))))
+        ((or (cplxnum? b) (and (cplxnum? a) (not (integer? b))))
+         (slow-expt a b))
+        ((and (##core#inline "C_i_flonump" b)
+              (not (##core#inline "C_u_i_fpintegerp" b)))
+         (if (negative? a)
+             (log-expt (exact->inexact a) (exact->inexact b))
+             (##core#inline_allocate
+	      ("C_a_i_flonum_expt" 4) (exact->inexact a) b)))
+        ((##core#inline "C_i_flonump" a)
+	 (##core#inline_allocate ("C_a_i_flonum_expt" 4) a (exact->inexact b)))
+        ;; this doesn't work that well, yet...
+        ;; (XXX: What does this mean? why not? I do know this is ugly... :P)
+        (else (if (or (inexact? a) (inexact? b))
+                  (exact->inexact (##sys#integer-power a (inexact->exact b)))
+                  (##sys#integer-power a b)))) )
+
 (define (##sys#integer-gcd a b)
   ;; Currently this is only Euclidean GCD algorithm. TODO: Restore
   ;; Lehmer's algorithm when everything else has been implemented.
@@ -1444,7 +1701,7 @@ EOF
                     (##sys#slot next 1)) ) )  ) ) )
 
 (define (##sys#lcm x y)
-  (quotient (* x y) (##sys#internal-gcd 'lcm x y)) )
+  (abs (quotient (* x y) (##sys#internal-gcd 'lcm x y)) ))
 
 (define (lcm . ns)
   (if (null? ns)
diff --git a/manual/Deviations from the standard b/manual/Deviations from the standard
index 59570e4e..65dd11c6 100644
--- a/manual/Deviations from the standard	
+++ b/manual/Deviations from the standard	
@@ -15,20 +15,12 @@ is 120.  This is an implementation restriction that is unlikely
 to be lifted.
 
 
-=== {{numerator}}, {{denominator}} and {{rationalize}}
-
-The {{numerator}} and {{denominator}} procedures cannot be
-applied to inexact numbers, and the procedure {{rationalize}} is not
-implemented at all.
-
-
 === Numeric string-conversion considerations
 
-The runtime system uses the numerical string-conversion
-routines of the underlying C library and so does only understand
-standard (C-library) syntax for floating-point constants.  Consequently,
-the procedures {{string->number}}, {{read}}, {{write}}, and {{display}} do not obey
-read/write invariance to inexact numbers.
+In some cases the runtime system uses the numerical string-conversion
+routines of the underlying C library.  Consequently, the procedures
+{{string->number}}, {{read}}, {{write}}, and {{display}} do not obey
+read/write invariance for inexact numbers.
 
 
 === Environments and non-standard syntax
diff --git a/modules.scm b/modules.scm
index a9ee89f0..bf259a71 100644
--- a/modules.scm
+++ b/modules.scm
@@ -875,8 +875,9 @@
 	     member assq assv assoc symbol? symbol->string string->symbol number?
 	     integer? exact? real? complex? inexact? rational? zero? odd? even?
 	     positive? negative?  max min + - * / = > < >= <= quotient remainder
-	     modulo gcd lcm abs floor ceiling truncate round exact->inexact
-	     inexact->exact exp log expt sqrt sin cos tan asin acos atan
+	     modulo gcd lcm abs floor ceiling truncate round rationalize
+	     exact->inexact inexact->exact exp log expt sqrt
+	     sin cos tan asin acos atan
 	     number->string string->number char? char=? char>? char<? char>=?
 	     char<=? char-ci=? char-ci<? char-ci>?  char-ci>=? char-ci<=?
 	     char-alphabetic? char-whitespace? char-numeric? char-upper-case?
diff --git a/runtime.c b/runtime.c
index 0908c894..154e26cf 100644
--- a/runtime.c
+++ b/runtime.c
@@ -527,6 +527,7 @@ static C_regparm void bignum_divrem(C_word c, C_word self, C_word k, C_word x, C
 static void divrem_intflo_2(C_word c, C_word self, ...) C_noret;
 static void bignum_divrem_fixnum_2(C_word c, C_word self, C_word negated_big) C_noret;
 static C_word rat_cmp(C_word x, C_word y);
+static void flo_to_int_2(C_word c, C_word self, C_word result) C_noret;
 static void fabs_frexp_to_digits(C_uword exp, double sign, C_uword *start, C_uword *scan);
 static C_word flo_to_tmp_bignum(C_word x);
 static C_word int_flo_cmp(C_word intnum, C_word flonum);
@@ -542,6 +543,7 @@ static C_word C_fcall convert_string_to_number(C_char *str, int radix, C_word *f
 static void digits_to_integer_2(C_word c, C_word self, C_word result) C_noret;
 static C_regparm C_word str_to_bignum(C_word bignum, char *str, char *str_end, int radix);
 static void bignum_to_str_2(C_word c, C_word self, C_word string) C_noret;
+/* XXX TODO OBSOLETE: This can be removed after recompiling c-platform.scm */
 static C_word C_fcall maybe_inexact_to_exact(C_word n) C_regparm;
 static void C_fcall remark_system_globals(void) C_regparm;
 static void C_fcall really_remark(C_word *x) C_regparm;
@@ -836,7 +838,7 @@ static C_PTABLE_ENTRY *create_initial_ptable()
 {
   /* IMPORTANT: hardcoded table size -
      this must match the number of C_pte calls + 1 (NULL terminator)! */
-  C_PTABLE_ENTRY *pt = (C_PTABLE_ENTRY *)C_malloc(sizeof(C_PTABLE_ENTRY) * 74);
+  C_PTABLE_ENTRY *pt = (C_PTABLE_ENTRY *)C_malloc(sizeof(C_PTABLE_ENTRY) * 73);
   int i = 0;
 
   if(pt == NULL)
@@ -875,7 +877,6 @@ static C_PTABLE_ENTRY *create_initial_ptable()
   C_pte(C_greater_or_equal_p);
   C_pte(C_less_or_equal_p);
   C_pte(C_quotient);
-  C_pte(C_expt);
   C_pte(C_number_to_string);
   C_pte(C_make_symbol);
   C_pte(C_string_to_symbol);
@@ -5382,6 +5383,7 @@ C_regparm C_word C_fcall C_u_i_length(C_word lst)
   return C_fix(n);
 }
 
+/* XXX TODO OBSOLETE: This can be removed after recompiling c-platform.scm */
 C_regparm C_word maybe_inexact_to_exact(C_word n)
 {
   double m;
@@ -5396,6 +5398,7 @@ C_regparm C_word maybe_inexact_to_exact(C_word n)
   return C_SCHEME_FALSE;
 }
 
+/* XXX TODO OBSOLETE: This can be removed after recompiling c-platform.scm */
 C_regparm C_word C_fcall C_i_inexact_to_exact(C_word n)
 {
   C_word r;
@@ -7848,6 +7851,41 @@ C_regparm double C_fcall C_bignum_to_double(C_word bignum)
   return(C_bignum_negativep(bignum) ? -accumulator : accumulator);
 }
 
+void C_ccall C_u_flo_to_int(C_word c, C_word self, C_word k, C_word x)
+{
+  int exponent;
+  double significand = frexp(C_flonum_magnitude(x), &exponent);
+
+  assert(C_truep(C_u_i_fpintegerp(x)));
+
+  if (exponent <= 0) {
+    C_kontinue(k, C_fix(0));
+  } else if (exponent == 1) { /* TODO: check significand * 2^exp fits fixnum? */
+    C_kontinue(k, significand < 0.0 ? C_fix(-1) : C_fix(1));
+  } else {
+    C_word kab[C_SIZEOF_CLOSURE(4) + C_SIZEOF_FLONUM], *ka = kab, k2, size,
+           negp = C_mk_bool(C_flonum_magnitude(x) < 0.0),
+           sign = C_flonum(&ka, fabs(significand));
+
+    k2 = C_closure(&ka, 4, (C_word)flo_to_int_2, k, C_fix(exponent), sign);
+
+    size = C_fix(C_BIGNUM_BITS_TO_DIGITS(exponent));
+    C_allocate_bignum(5, (C_word)NULL, k2, size, negp, C_SCHEME_FALSE);
+  }
+}
+
+static void flo_to_int_2(C_word c, C_word self, C_word result)
+{
+  C_word k = C_block_item(self, 1);
+  C_uword exponent = C_unfix(C_block_item(self, 2)),
+          *start = C_bignum_digits(result),
+          *scan = start + C_bignum_size(result);
+  double significand = C_flonum_magnitude(C_block_item(self, 3));
+
+  fabs_frexp_to_digits(exponent, significand, start, scan);
+  C_kontinue(k, C_bignum_simplify(result));
+}
+
 static void
 fabs_frexp_to_digits(C_uword exp, double sign, C_uword *start, C_uword *scan)
 {
@@ -8440,34 +8478,6 @@ C_regparm C_word C_fcall C_i_integer_less_or_equalp(C_word x, C_word y)
   }
 }
 
-void C_ccall C_expt(C_word c, C_word closure, C_word k, C_word n1, C_word n2)
-{
-  double m1, m2;
-  C_word r;
-  C_alloc_flonum;
-
-  if(c != 4) C_bad_argc(c, 4);
-
-  if(n1 & C_FIXNUM_BIT) m1 = C_unfix(n1);
-  else if(!C_immediatep(n1) && C_block_header(n1) == C_FLONUM_TAG)
-    m1 = C_flonum_magnitude(n1);
-  else barf(C_BAD_ARGUMENT_TYPE_ERROR, "expt", n1);
-
-  if(n2 & C_FIXNUM_BIT) m2 = C_unfix(n2);
-  else if(!C_immediatep(n2) && C_block_header(n2) == C_FLONUM_TAG)
-    m2 = C_flonum_magnitude(n2);
-  else barf(C_BAD_ARGUMENT_TYPE_ERROR, "expt", n2);
-
-  m1 = pow(m1, m2);
-  r = (C_word)m1;
-
-  if(r == m1 && (n1 & C_FIXNUM_BIT) && (n2 & C_FIXNUM_BIT) && modf(m1, &m2) == 0.0 && C_fitsinfixnump(r))
-    C_kontinue(k, C_fix(r));
-
-  C_kontinue_flonum(k, m1);
-}
-
-
 void C_ccall C_gc(C_word c, C_word closure, C_word k, ...)
 {
   int f;
@@ -9055,6 +9065,7 @@ void C_ccall C_string_to_symbol(C_word c, C_word closure, C_word k, C_word strin
 }
 
 
+/* XXX TODO OBSOLETE: This can be removed after recompiling c-platform.scm */
 C_regparm C_word C_fcall 
 C_a_i_exact_to_inexact(C_word **a, int c, C_word n)
 {
diff --git a/tests/library-tests.scm b/tests/library-tests.scm
index 523cc163..f5dfdaeb 100644
--- a/tests/library-tests.scm
+++ b/tests/library-tests.scm
@@ -103,12 +103,12 @@
 ;;; A few denormalised numbers, cribbed from NetBSD ATF tests for ldexp():
 ;; On some machines/OSes these tests fail due to missing hardware support
 ;; and sometimes due to broken libc/libm support, so we have disabled them.
-(assert (equal? 1.0 (numerator 1.1125369292536006915451e-308)))
-(assert (equal? +inf.0 (denominator 1.1125369292536006915451e-308)))
-(assert (equal? -1.0 (numerator -5.5626846462680034577256e-309)))
-(assert (equal? +inf.0 (denominator -5.5626846462680034577256e-309)))
-(assert (equal? 1.0 (numerator 4.9406564584124654417657e-324)))
-(assert (equal? +inf.0 (denominator 4.9406564584124654417657e-324)))
+;(assert (equal? 1.0 (numerator 1.1125369292536006915451e-308)))
+;(assert (equal? +inf.0 (denominator 1.1125369292536006915451e-308)))
+;(assert (equal? -1.0 (numerator -5.5626846462680034577256e-309)))
+;(assert (equal? +inf.0 (denominator -5.5626846462680034577256e-309)))
+;(assert (equal? 1.0 (numerator 4.9406564584124654417657e-324)))
+;(assert (equal? +inf.0 (denominator 4.9406564584124654417657e-324)))
 
 (assert (equal? 4.0 (denominator -1.25)))
 (assert (equal? 1e10 (numerator 1e10)))
diff --git a/types.db b/types.db
index 5d08a3c7..ab1575c7 100644
--- a/types.db
+++ b/types.db
@@ -500,79 +500,86 @@
      ((float) (float) (##core#inline_allocate ("C_a_i_flonum_abs" 4) #(1)))
      ((integer) (integer) (##sys#integer-abs #(1))))
 
-(floor (#(procedure #:clean #:enforce #:foldable) floor (number) number)
+(floor (#(procedure #:clean #:enforce #:foldable) floor ((or integer ratnum float)) (or integer ratnum float))
        ((fixnum) (fixnum) #(1))
+       ((integer) (integer) #(1))
        ((float) (float)
 	(##core#inline_allocate ("C_a_i_flonum_floor" 4) #(1))))
 
-(ceiling (#(procedure #:clean #:enforce #:foldable) ceiling (number) number)
+(ceiling (#(procedure #:clean #:enforce #:foldable) ceiling ((or integer ratnum float)) (or integer ratnum float))
 	 ((fixnum) (fixnum) #(1))
+	 ((integer) (integer) #(1))
 	 ((float) (float)
 	  (##core#inline_allocate ("C_a_i_flonum_ceiling" 4) #(1))))
 
-(truncate (#(procedure #:clean #:enforce #:foldable) truncate (number) number)
+(truncate (#(procedure #:clean #:enforce #:foldable) truncate ((or integer ratnum float)) (or integer ratnum float))
 	  ((fixnum) (fixnum) #(1))
+	  ((integer) (integer) #(1))
 	  ((float) (float)
 	   (##core#inline_allocate ("C_a_i_flonum_truncate" 4) #(1))))
 
-(round (#(procedure #:clean #:enforce #:foldable) round (number) number)
+(round (#(procedure #:clean #:enforce #:foldable) round ((or integer ratnum float)) (or integer ratnum float))
        ((fixnum) (fixnum) #(1))
+       ((integer) (integer) #(1))
        ((float) (float)
 	(##core#inline_allocate ("C_a_i_flonum_round_proper" 4) #(1))))
 
-(exact->inexact (#(procedure #:clean #:enforce #:foldable) exact->inexact (number) float)
-		((float) #(1))
-		((fixnum) (##core#inline_allocate ("C_a_i_fix_to_flo" 4) #(1))))
+(exact->inexact (#(procedure #:clean #:enforce #:foldable) exact->inexact (number) (or float cplxnum))
+		((float) (float) #(1))
+		((fixnum) (float) (##core#inline_allocate ("C_a_i_fix_to_flo" 4) #(1))))
 
-(inexact->exact (#(procedure #:clean #:enforce #:foldable) inexact->exact (number) fixnum) ((fixnum) #(1)))
+(inexact->exact (#(procedure #:clean #:enforce #:foldable) inexact->exact (number) (or integer ratnum))
+		((fixnum) (fixnum) #(1))
+		((integer) (integer) #(1))
+		((ratnum) (ratnum) #(1))
+		(((or integer ratnum)) #(1)))
 
-(exp (#(procedure #:clean #:enforce #:foldable) exp (number) float)
-     ((float) (##core#inline_allocate ("C_a_i_flonum_exp" 4) #(1))))
+(exp (#(procedure #:clean #:enforce #:foldable) exp (number) (or float cplxnum))
+     ((float) (float) (##core#inline_allocate ("C_a_i_flonum_exp" 4) #(1))))
 
-(log (#(procedure #:clean #:enforce #:foldable) log (number) float)
-     ((float) (##core#inline_allocate ("C_a_i_flonum_log" 4) #(1))))
+(log (#(procedure #:clean #:enforce #:foldable) log (number) (or float cplxnum))
+     ;; Unfortunately this doesn't work when the argument is negative
+     ;;((float) (float) (##core#inline_allocate ("C_a_i_flonum_log" 4) #(1)))
+     ((number) (##sys#log-1 #(1))))
 
 (expt (#(procedure #:clean #:enforce #:foldable) expt (number number) number)
-      ((float float) (float)
-       (##core#inline_allocate ("C_a_i_flonum_expt" 4) #(1) #(2)))
-      ((float fixnum) (float)
+      ;; This breaks in some extreme edge cases... Worth disabling?
+      #;((float float) (float)
+      (##core#inline_allocate ("C_a_i_flonum_expt" 4) #(1) #(2)))
+      #;((float fixnum) (float)
        (##core#inline_allocate ("C_a_i_flonum_expt" 4) 
 			       #(1)
 			       (##core#inline_allocate ("C_a_i_fix_to_flo" 4) #(2))))
-      ((fixnum float) (float)
+      #;((fixnum float) (float)
        (##core#inline_allocate ("C_a_i_flonum_expt" 4)
 			       (##core#inline_allocate ("C_a_i_fix_to_flo" 4) #(1))
 			       #(2))))
 
-(sqrt (#(procedure #:clean #:enforce #:foldable) sqrt (number) float)
-      ((float) (##core#inline_allocate ("C_a_i_flonum_sqrt" 4) #(1))))
+(sqrt (#(procedure #:clean #:enforce #:foldable) sqrt (number) number)
+      ;; Unfortunately this doesn't work when the argument is negative
+      #;((float) (float) (##core#inline_allocate ("C_a_i_flonum_sqrt" 4) #(1))))
 
-(sin (#(procedure #:clean #:enforce #:foldable) sin (number) float)
-     ((float) (##core#inline_allocate ("C_a_i_flonum_sin" 4) #(1))))
+(sin (#(procedure #:clean #:enforce #:foldable) sin (number) (or float cplxnum))
+     ((float) (float) (##core#inline_allocate ("C_a_i_flonum_sin" 4) #(1))))
 
-(cos (#(procedure #:clean #:enforce #:foldable) cos (number) float)
-     ((float) (##core#inline_allocate ("C_a_i_flonum_cos" 4) #(1))))
+(cos (#(procedure #:clean #:enforce #:foldable) cos (number) (or float cplxnum))
+     ((float) (float) (##core#inline_allocate ("C_a_i_flonum_cos" 4) #(1))))
 
-(tan (#(procedure #:clean #:enforce #:foldable) tan (number) float)
-     ((float) (##core#inline_allocate ("C_a_i_flonum_tan" 4) #(1))))
+(tan (#(procedure #:clean #:enforce #:foldable) tan (number) (or float cplxnum))
+     ((float) (float) (##core#inline_allocate ("C_a_i_flonum_tan" 4) #(1))))
 
-(asin (#(procedure #:clean #:enforce #:foldable) asin (number) float)
-      ((float) (##core#inline_allocate ("C_a_i_flonum_asin" 4) #(1))))
+(asin (#(procedure #:clean #:enforce #:foldable) asin (number) (or float cplxnum))
+      ;; Unfortunately this doesn't work when the number is > 1.0 (returns compnum)
+      #;((float) (float) (##core#inline_allocate ("C_a_i_flonum_acos" 4) #(1))))
 
-(acos (#(procedure #:clean #:enforce #:foldable) acos (number) float)
-      ((float) (##core#inline_allocate ("C_a_i_flonum_acos" 4) #(1))))
+(acos (#(procedure #:clean #:enforce #:foldable) acos (number) (or float cplxnum))
+      ;; Unfortunately this doesn't work when the number is > 1.0 (returns compnum)
+      #;((float) (float) (##core#inline_allocate ("C_a_i_flonum_acos" 4) #(1))))
 
-(atan (#(procedure #:clean #:enforce #:foldable) atan (number #!optional number) float)
-      ((float) (##core#inline_allocate ("C_a_i_flonum_atan" 4) #(1)))
-      ((float fixnum)
-       (##core#inline_allocate ("C_a_i_flonum_atan2" 4) 
-			       #(1)
-			       (##core#inline_allocate ("C_a_i_fix_to_flo" 4) #(2))))
-      ((fixnum float)
-       (##core#inline_allocate ("C_a_i_flonum_atan2" 4) 
-			       (##core#inline_allocate ("C_a_i_fix_to_flo" 4) #(1))
-			       #(2)))
-      ((float float) (##core#inline_allocate ("C_a_i_flonum_atan2" 4) #(1) #(2))))
+(atan (#(procedure #:clean #:enforce #:foldable) atan (number #!optional number) (or float cplxnum))
+      ((float) (float) (##core#inline_allocate ("C_a_i_flonum_atan" 4) #(1)))
+      ((float float) (float)
+       (##core#inline_allocate ("C_a_i_flonum_atan2" 4) #(1) #(2))))
 
 (number->string (#(procedure #:clean #:enforce) number->string (number #!optional fixnum) string)
 		((fixnum fixnum) (##sys#fixnum->string #(1) #(2)))
@@ -789,6 +796,17 @@
 	   ((float) (float) (##core#inline_allocate ("C_a_i_flonum_abs" 4) #(1)))
 	   (((or fixnum float bignum ratnum)) (abs #(1))))
 
+(angle (#(procedure #:clean #:enforce #:foldable) angle (number) float)
+       ((float) (##core#inline_allocate ("C_a_i_flonum_atan2" 4) '0.0 #(1)))
+       ((fixnum) (##core#inline_allocate
+                          ("C_a_i_flonum_atan2" 4)
+                          '0.0
+                          (##core#inline_allocate ("C_a_i_fix_to_flo" 4) #(1))))
+       ((cplxnum) (##core#inline_allocate
+		   ("C_a_i_flonum_atan2" 4)
+		   (##sys#exact->inexact (##sys#slot #(1) '2))
+		   (##sys#exact->inexact (##sys#slot #(1) '1)))))
+
 (numerator (#(procedure #:clean #:enforce #:foldable) numerator ((or float integer ratnum)) (or float integer))
 	   ((fixnum) (fixnum) #(1))
 	   ((bignum) (bignum) #(1))
Trap