~ chicken-core (chicken-5) 3700b414240e87f0a6454949762b3240c458e1cb
commit 3700b414240e87f0a6454949762b3240c458e1cb Author: Peter Bex <peter@more-magic.net> AuthorDate: Sat Mar 21 10:19:41 2015 +0100 Commit: Peter Bex <peter@more-magic.net> CommitDate: Sun May 31 14:55:24 2015 +0200 Re-implement variadic + and - in C, in order to avoid consing up rest lists. Restore old-style compiler specialization rewrites for dyadic + and -. Also clean up library.scm a bit by removing the explicit calls to dyadic procedures: they're rewritten so it isn't necessary. diff --git a/c-platform.scm b/c-platform.scm index 91f1c0fe..a79e1b54 100644 --- a/c-platform.scm +++ b/c-platform.scm @@ -214,6 +214,27 @@ (rewrite '+ 19 0 "C_fixnum_plus" "C_u_fixnum_plus" #f) +(let () + ;; (add1 <x>) -> (##core#inline "C_fixnum_increase" <x>) [fixnum-mode] + ;; (add1 <x>) -> (##core#inline "C_u_fixnum_increase" <x>) [fixnum-mode + unsafe] + ;; (add1 <x>) -> (##core#inline_allocate ("C_s_a_i_plus" 36) <x> 1) + ;; (sub1 <x>) -> (##core#inline "C_fixnum_decrease" <x>) [fixnum-mode] + ;; (sub1 <x>) -> (##core#inline "C_u_fixnum_decrease" <x>) [fixnum-mode + unsafe] + ;; (sub1 <x>) -> (##core#inline_allocate ("C_s_a_i_minus" 36) <x> 1) + (define ((op1 fiop ufiop aiop) db classargs cont callargs) + (and (= (length callargs) 1) + (make-node + '##core#call (list #t) + (list + cont + (if (eq? 'fixnum number-type) + (make-node '##core#inline (list (if unsafe ufiop fiop)) callargs) + (make-node + '##core#inline_allocate (list aiop 36) + (list (car callargs) (qnode 1)))))))) + (rewrite 'add1 8 (op1 "C_fixnum_increase" "C_u_fixnum_increase" "C_s_a_i_plus")) + (rewrite 'sub1 8 (op1 "C_fixnum_decrease" "C_u_fixnum_decrease" "C_s_a_i_minus"))) + (let () (define (eqv?-id db classargs cont callargs) ;; (eqv? <var> <var>) -> (quote #t) [two identical objects] @@ -611,6 +632,9 @@ (rewrite 'lcm 18 1) (rewrite 'list 18 '()) +(rewrite '+ 16 2 "C_s_a_i_plus" #t 36) +(rewrite '- 16 2 "C_s_a_i_minus" #t 36) + (rewrite '= 17 2 "C_i_nequalp") (rewrite '> 17 2 "C_i_greaterp") (rewrite '< 17 2 "C_i_lessp") @@ -623,6 +647,9 @@ (rewrite '>= 13 #f "C_greater_or_equal_p" #t) (rewrite '<= 13 #f "C_less_or_equal_p" #t) +(rewrite '+ 13 #f "C_plus" #t) +(rewrite '- 13 '(1 . #f) "C_minus" #t) + (rewrite 'number->string 13 '(1 . 2) "C_number_to_string" #t) (rewrite '##sys#call-with-current-continuation 13 1 "C_call_cc" #t) (rewrite '##sys#allocate-vector 13 4 "C_allocate_vector" #t) diff --git a/chicken.h b/chicken.h index d8fcabbb..d89f88e4 100644 --- a/chicken.h +++ b/chicken.h @@ -1962,9 +1962,7 @@ C_fctexport void C_ccall C_u_call_with_values(C_word c, C_word closure, C_word k C_fctexport void C_ccall C_times(C_word c, C_word closure, C_word k, ...) C_noret; C_fctexport void C_ccall C_2_basic_times(C_word c, C_word self, C_word k, C_word x, C_word y) C_noret; C_fctexport void C_ccall C_u_2_integer_times(C_word c, C_word self, C_word k, C_word x, C_word y) C_noret; -/* XXX TODO OBSOLETE: This can be removed after recompiling c-platform.scm */ C_fctexport void C_ccall C_plus(C_word c, C_word closure, C_word k, ...) C_noret; -/* XXX TODO OBSOLETE: This can be removed after recompiling c-platform.scm */ C_fctexport void C_ccall C_minus(C_word c, C_word closure, C_word k, C_word n1, ...) C_noret; /* XXX TODO OBSOLETE: This can be removed after recompiling c-platform.scm */ C_fctexport void C_ccall C_divide(C_word c, C_word closure, C_word k, C_word n1, ...) C_noret; diff --git a/library.scm b/library.scm index e086b79b..3651ebdb 100644 --- a/library.scm +++ b/library.scm @@ -943,11 +943,8 @@ EOF (fp-check-flonum x 'fpinteger?) (##core#inline "C_u_i_fpintegerp" x)) -(define (##sys#=-2 a b) (##core#inline "C_i_nequalp" a b)) -(define (##sys#<-2 a b) (##core#inline "C_i_lessp" a b)) -(define (##sys#<=-2 a b) (##core#inline "C_i_less_or_equalp" a b)) -(define (##sys#>-2 a b) (##core#inline "C_i_greaterp" a b)) -(define (##sys#>=-2 a b) (##core#inline "C_i_greater_or_equalp" a b)) +(define-inline (integer-negate x) + (##core#inline_allocate ("C_s_a_u_i_integer_negate" 6) x)) (define = (##core#primitive "C_nequalp")) (define > (##core#primitive "C_greaterp")) @@ -955,8 +952,10 @@ EOF (define >= (##core#primitive "C_greater_or_equal_p")) (define <= (##core#primitive "C_less_or_equal_p")) -(define (add1 n) (##sys#+-2 n 1)) -(define (sub1 n) (##sys#--2 n 1)) +(define + (##core#primitive "C_plus")) +(define - (##core#primitive "C_minus")) +(define (add1 n) (+ n 1)) +(define (sub1 n) (- n 1)) (define (number? x) (##core#inline "C_i_numberp" x)) (define ##sys#number? number?) @@ -1020,7 +1019,7 @@ EOF (cond ((cplxnum? x) (let ((r (%cplxnum-real x)) (i (%cplxnum-imag x)) ) - (sqrt (##sys#+-2 (##sys#*-2 r r) (##sys#*-2 i i))) )) + (sqrt (+ (##sys#*-2 r r) (##sys#*-2 i i))) )) ((number? x) (abs x)) (else (##sys#error-bad-number x 'magnitude)))) @@ -1033,9 +1032,9 @@ EOF (define (ratnum m n) (cond ((eq? n 1) m) - ((eq? n -1) (##sys#integer-negate m)) + ((eq? n -1) (integer-negate m)) ((negative? n) - (%make-ratnum (##sys#integer-negate m) (##sys#integer-negate n))) + (%make-ratnum (integer-negate m) (integer-negate n))) (else (%make-ratnum m n)))) (define (numerator n) @@ -1112,56 +1111,8 @@ EOF ;;; Basic arithmetic: (define (abs x) (##core#inline_allocate ("C_s_a_i_abs" 10) x)) -;; OBSOLETE: Remove this (or change to define-inline) -(define (##sys#integer-abs x) - (##core#inline_allocate ("C_s_a_u_i_integer_abs" 6) x)) - -(define (+ . args) - (if (null? args) - 0 - (let ((x (##sys#slot args 0)) - (args (##sys#slot args 1))) - (if (null? args) - (if (number? x) x (##sys#error-bad-number x '+)) - (let loop ((args (##sys#slot args 1)) - (x (##sys#+-2 x (##sys#slot args 0)))) - (if (null? args) - x - (loop (##sys#slot args 1) - (##sys#+-2 x (##sys#slot args 0))) ) ) ) ) ) ) - -;; OBSOLETE: Remove this (or change to define-inline) -(define (##sys#+-2 x y) - (##core#inline_allocate ("C_s_a_i_plus" 36) x y)) -;; OBSOLETE: Remove this (or change to define-inline) -(define (##sys#integer-plus x y) - (##core#inline_allocate ("C_s_a_u_i_integer_plus" 6) x y)) - -;; OBSOLETE: Remove this (or change to define-inline) -(define (##sys#negate x) (##core#inline_allocate ("C_s_a_i_negate" 36) x)) -;; OBSOLETE: Remove this (or change to define-inline) -(define (##sys#integer-negate x) - (##core#inline_allocate ("C_s_a_u_i_integer_negate" 6) x)) - -(define (- arg1 . args) - (if (null? args) - (##sys#negate arg1) - (let loop ((args (##sys#slot args 1)) - (x (##sys#--2 arg1 (##sys#slot args 0)))) - (if (null? args) - x - (loop (##sys#slot args 1) - (##sys#--2 x (##sys#slot args 0))) ) ) ) ) - -;; OBSOLETE: Remove this (or change to define-inline) -(define (##sys#--2 x y) - (##core#inline_allocate ("C_s_a_i_minus" 36) x y)) -;; OBSOLETE: Remove this (or change to define-inline) -(define (##sys#integer-minus x y) - (##core#inline_allocate ("C_s_a_u_i_integer_minus" 6) x y)) (define ##sys#*-2 (##core#primitive "C_2_basic_times")) -(define ##sys#integer-times (##core#primitive "C_u_2_integer_times")) (define (* . args) (if (null? args) @@ -1190,7 +1141,7 @@ EOF (let* ((same? (eqv? x y)) ; Check before calling (abs) (rs (fx* (##core#inline "C_u_i_integer_signum" x) (##core#inline "C_u_i_integer_signum" y))) - (x (##sys#integer-abs x)) + (x (##core#inline_allocate ("C_s_a_u_i_integer_abs" 6) x)) (n (%bignum-digit-count y)) (n/2 (fxshr n 1)) (bits (fx* n/2 (foreign-value "C_BIGNUM_DIGIT_LENGTH" int))) @@ -1199,25 +1150,18 @@ EOF (if same? ; This looks pointless, but reduces garbage (let* ((a (##sys#*-2 x-hi x-hi)) (b (##sys#*-2 x-lo x-lo)) - (ab (##sys#--2 x-hi x-lo)) + (ab (- x-hi x-lo)) (c (##sys#*-2 ab ab))) - (##sys#+-2 (##sys#integer-shift a (fxshl bits 1)) - (##sys#+-2 (##sys#integer-shift - (##sys#+-2 b (##sys#--2 a c)) - bits) - b))) - (let* ((y (##sys#integer-abs y)) + (+ (##sys#integer-shift a (fxshl bits 1)) + (+ (##sys#integer-shift (+ b (- a c)) bits) b))) + (let* ((y (##core#inline_allocate ("C_s_a_u_i_integer_abs" 6) y)) (y-hi (##sys#bignum-extract-digits y n/2 #f)) (y-lo (##sys#bignum-extract-digits y 0 n/2)) (a (##sys#*-2 x-hi y-hi)) (b (##sys#*-2 x-lo y-lo)) - (c (##sys#*-2 (##sys#--2 x-hi x-lo) - (##sys#--2 y-hi y-lo)))) - (##sys#*-2 rs (##sys#+-2 (##sys#integer-shift a (fxshl bits 1)) - (##sys#+-2 (##sys#integer-shift - (##sys#+-2 b (##sys#--2 a c)) - bits) - b))))))) + (c (##sys#*-2 (- x-hi x-lo) (- y-hi y-lo)))) + (##sys#*-2 rs (+ (##sys#integer-shift a (fxshl bits 1)) + (+ (##sys#integer-shift (+ b (- a c)) bits) b))))))) (define (##sys#extended-times x y) (define (nonrat*rat x y) @@ -1232,8 +1176,8 @@ EOF (cond ((or (cplxnum? x) (cplxnum? y)) (let* ((a (real-part x)) (b (imag-part x)) (c (real-part y)) (d (imag-part y)) - (r (##sys#--2 (##sys#*-2 a c) (##sys#*-2 b d))) - (i (##sys#+-2 (##sys#*-2 a d) (##sys#*-2 b c))) ) + (r (- (##sys#*-2 a c) (##sys#*-2 b d))) + (i (+ (##sys#*-2 a d) (##sys#*-2 b c))) ) (make-complex r i) ) ) ((or (##core#inline "C_i_flonump" x) (##core#inline "C_i_flonump" y)) ;; This may be incorrect when one is a ratnum consisting of bignums @@ -1273,9 +1217,9 @@ EOF ((or (cplxnum? x) (cplxnum? y)) (let* ((a (real-part x)) (b (imag-part x)) (c (real-part y)) (d (imag-part y)) - (r (##sys#+-2 (##sys#*-2 c c) (##sys#*-2 d d))) - (x (##sys#/-2 (##sys#+-2 (##sys#*-2 a c) (##sys#*-2 b d)) r)) - (y (##sys#/-2 (##sys#--2 (##sys#*-2 b c) (##sys#*-2 a d)) r)) ) + (r (+ (##sys#*-2 c c) (##sys#*-2 d d))) + (x (##sys#/-2 (+ (##sys#*-2 a c) (##sys#*-2 b d)) r)) + (y (##sys#/-2 (- (##sys#*-2 b c) (##sys#*-2 a d)) r)) ) (make-complex x y) )) ((or (##core#inline "C_i_flonump" x) (##core#inline "C_i_flonump" y)) ;; This may be incorrect when one is a ratnum consisting of bignums @@ -1336,20 +1280,19 @@ EOF ;; up the number more than once. (define (burnikel-ziegler-3n/2n a12 a3 b b1 b2 n) (receive (q^ r1) - (if (##sys#<-2 (##sys#integer-shift a12 (fxneg (digit-bits n))) b1) + (if (< (##sys#integer-shift a12 (fxneg (digit-bits n))) b1) (let* ((n/2 (fxshr n 1)) (b11 (##sys#bignum-extract-digits b1 n/2 #f)) (b12 (##sys#bignum-extract-digits b1 0 n/2))) (burnikel-ziegler-2n/1n a12 b1 b11 b12 n)) (let ((base*n (digit-bits n))) - (values (##sys#--2 (##sys#integer-shift 1 base*n) 1) ; B^n-1 - (##sys#+-2 (##sys#--2 a12 (##sys#integer-shift b1 base*n)) - b1)))) - (let ((r1a3 (##sys#+-2 (##sys#integer-shift r1 (digit-bits n)) a3))) - (let lp ((r^ (##sys#--2 r1a3 (##sys#*-2 q^ b2))) + (values (- (##sys#integer-shift 1 base*n) 1) ; B^n-1 + (+ (- a12 (##sys#integer-shift b1 base*n)) b1)))) + (let ((r1a3 (+ (##sys#integer-shift r1 (digit-bits n)) a3))) + (let lp ((r^ (- r1a3 (##sys#*-2 q^ b2))) (q^ q^)) (if (negative? r^) - (lp (##sys#+-2 r^ b) (##sys#--2 q^ 1)) + (lp (+ r^ b) (- q^ 1)) (values q^ r^)))))) (define (burnikel-ziegler-2n/1n a b b1 b2 n) @@ -1362,8 +1305,7 @@ EOF (a4 (##sys#bignum-extract-digits a 0 n/2))) (receive (q1 r1) (burnikel-ziegler-3n/2n a12 a3 b b1 b2 n/2) (receive (q2 r) (burnikel-ziegler-3n/2n r1 a4 b b1 b2 n/2) - (values (##sys#+-2 (##sys#integer-shift q1 (digit-bits n/2)) - q2) + (values (+ (##sys#integer-shift q1 (digit-bits n/2)) q2) r)))))) ;; The caller will ensure that abs(x) > abs(y) @@ -1392,14 +1334,13 @@ EOF (i (fx- t 2)) (quot 0)) (receive (qi ri) (burnikel-ziegler-2n/1n zi y y-hi y-lo n) - (let ((quot (##sys#+-2 (##sys#integer-shift quot (digit-bits n)) qi))) + (let ((quot (+ (##sys#integer-shift quot (digit-bits n)) qi))) (if (fx> i 0) (let ((zi-1 (let* ((base*n*i-1 (fx* n (fx- i 1))) (base*n*i (fx* n i)) (xi-1 (##sys#bignum-extract-digits x base*n*i-1 base*n*i))) - (##sys#+-2 (##sys#integer-shift ri (digit-bits n)) - xi-1)))) + (+ (##sys#integer-shift ri (digit-bits n)) xi-1)))) (lp zi-1 (fx- i 1) quot)) (let ((rem (if (or (not return-rem?) (eq? 0 norm-shift)) ri @@ -1417,7 +1358,7 @@ EOF ;; (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)))) + (if (>= n 0) q (- q 1)))) (else (##sys#error-bad-real x 'floor)))) (define (ceiling x) @@ -1426,7 +1367,7 @@ EOF ;; (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))) + (if (>= n 0) (+ q 1) q))) (else (##sys#error-bad-real x 'ceiling))) ) (define (truncate x) @@ -1442,30 +1383,30 @@ EOF ((##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))) + (let* ((x+1/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))) + (if (and (= r x+1/2) (odd? r)) (- 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))) + (cond ((not (< fx x)) (list fx 1)) + ((= fx fy) + (let ((rat (sr (##sys#/-2 1 (- y fy)) + (##sys#/-2 1 (- x fx))))) + (list (+ (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)) + (else (list (+ 1 fx) 1))))) + (cond ((< y x) (find-ratio-between y x)) + ((not (< 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)))) + ((negative? y) (let ((rat (sr (- y) (- x)))) + (list (- (car rat)) (cadr rat)))) (else '(0 1)))) -(define (find-ratio x e) (find-ratio-between (##sys#--2 x e) (##sys#+-2 x e))) +(define (find-ratio x e) (find-ratio-between (- x e) (+ x e))) (define (rationalize x e) (let ((result (apply ##sys#/-2 (find-ratio x e)))) @@ -1485,18 +1426,18 @@ EOF (receive (div rem) (quotient&remainder x y) (if (positive? y) (if (negative? rem) - (values div (##sys#+-2 rem y)) + (values div (+ rem y)) (values div rem)) (if (positive? rem) - (values div (##sys#+-2 rem y)) + (values div (+ rem y)) (values div rem))))) ;; Modulo's sign follows y (whereas remainder's sign follows x) (define (modulo x y) (let ((r (remainder x y))) (if (positive? y) - (if (negative? r) (##sys#+-2 r y) r) - (if (positive? r) (##sys#+-2 r y) r)))) + (if (negative? r) (+ r y) r) + (if (positive? r) (+ r y) r)))) (define (even? n) (##core#inline "C_i_evenp" n)) (define (odd? n) (##core#inline "C_i_oddp" n)) @@ -1508,7 +1449,7 @@ EOF (if i (exact->inexact m) m) (let ((h (##sys#slot xs 0))) (loop (or i (##core#inline "C_i_flonump" h)) - (if (##sys#>-2 h m) h m) + (if (> h m) h m) (##sys#slot xs 1)) ) ) ) ) (define (min x1 . xs) @@ -1518,7 +1459,7 @@ EOF (if i (exact->inexact m) m) (let ((h (##sys#slot xs 0))) (loop (or i (##core#inline "C_i_flonump" h)) - (if (##sys#<-2 h m) h m) + (if (< h m) h m) (##sys#slot xs 1)) ) ) ) ) (define (exp n) @@ -1540,8 +1481,8 @@ EOF (##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)))) + (+ (##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))))) @@ -1558,14 +1499,14 @@ EOF (##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)) + (##sys#/-2 (- (exp in) (exp (- in))) %i2)) (##core#inline_allocate ("C_a_i_sin" 4) (exact->inexact n)))) (define (cos 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) ) + (##sys#/-2 (+ (exp in) (exp (- in))) 2) ) (##core#inline_allocate ("C_a_i_cos" 4) (exact->inexact n)))) (define (tan n) @@ -1586,9 +1527,8 @@ EOF ;; 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))))))))) + (+ (##sys#*-2 %i n) + (##sys#sqrt/loc 'asin (- 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) @@ -1603,7 +1543,7 @@ EOF (##core#inline_allocate ("C_a_i_fix_to_flo" 4) n))) ;; General definition can return compnums - (else (##sys#--2 asin1 (asin n))))))) + (else (- asin1 (asin n))))))) (define (atan n #!optional b) (##sys#check-number n 'atan) @@ -1611,8 +1551,8 @@ EOF (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)))) + (##sys#/-2 (- (##sys#log-1 (+ 1 in)) + (##sys#log-1 (- 1 in))) %i2)))) (b (##core#inline_allocate ("C_a_i_atan2" 4) (exact->inexact n) (exact->inexact b))) @@ -1637,19 +1577,19 @@ EOF ((len/2) (fxshl len/4 1)) ((s^ r^) (##sys#exact-integer-sqrt (##sys#integer-shift a (fxneg len/2)))) - ((mask) (##sys#--2 (##sys#integer-shift 1 len/4) 1)) + ((mask) (- (##sys#integer-shift 1 len/4) 1)) ((a0) (##sys#integer-bitwise-and a mask)) ((a1) (##sys#integer-bitwise-and (##sys#integer-shift a (fxneg len/4)) mask)) ((q u) (##sys#integer-quotient&remainder - (##sys#+-2 (arithmetic-shift r^ len/4) a1) + (+ (arithmetic-shift r^ len/4) a1) (##sys#integer-shift s^ 1))) - ((s) (##sys#+-2 (##sys#integer-shift s^ len/4) q)) - ((r) (##sys#+-2 (##sys#integer-shift u len/4) - (##sys#--2 a0 (##sys#*-2 q q))))) + ((s) (+ (##sys#integer-shift s^ len/4) q)) + ((r) (+ (##sys#integer-shift u len/4) + (- a0 (##sys#*-2 q q))))) (if (negative? r) - (values (##sys#--2 s 1) - (##sys#--2 (##sys#+-2 r (##sys#integer-shift s 1)) 1)) + (values (- s 1) + (- (+ r (##sys#integer-shift s 1)) 1)) (values s r))))) (define (exact-integer-sqrt x) @@ -1665,7 +1605,7 @@ EOF (make-complex (##sys#*-2 m (cos p)) (##sys#*-2 m (sin p)) ) )) ((negative? n) (make-complex .0 (##core#inline_allocate - ("C_a_i_sqrt" 4) (exact->inexact (##sys#negate n))))) + ("C_a_i_sqrt" 4) (exact->inexact (- n))))) ((exact-integer? n) (receive (s^2 r) (##sys#exact-integer-sqrt n) (if (eq? 0 r) @@ -1695,37 +1635,35 @@ EOF (if (or (eq? 0 k) (eq? 1 k) (eq? 1 n)) ; Maybe call exact-integer-sqrt on n=2? (values k 0) (let ((len (integer-length k))) - (if (##sys#<-2 len n) ; Idea from Gambit: 2^{len-1} <= k < 2^{len} - (values 1 (##sys#--2 k 1)) ; Since x >= 2, we know x^{n} can't exist + (if (< len n) ; Idea from Gambit: 2^{len-1} <= k < 2^{len} + (values 1 (- k 1)) ; Since x >= 2, we know x^{n} can't exist ;; Set initial guess to (at least) 2^ceil(ceil(log2(k))/n) (let* ((shift-amount (inexact->exact (ceiling (/ (fx+ len 1) n)))) (g0 (arithmetic-shift 1 shift-amount)) - (n-1 (##sys#--2 n 1))) + (n-1 (- n 1))) (let lp ((g0 g0) (g1 (quotient - (##sys#+-2 - (##sys#*-2 n-1 g0) - (quotient k (##sys#integer-power g0 n-1))) + (+ (##sys#*-2 n-1 g0) + (quotient k (##sys#integer-power g0 n-1))) n))) - (if (##sys#<-2 g1 g0) + (if (< g1 g0) (lp g1 (quotient - (##sys#+-2 - (##sys#*-2 n-1 g1) - (quotient k (##sys#integer-power g1 n-1))) + (+ (##sys#*-2 n-1 g1) + (quotient k (##sys#integer-power g1 n-1))) n)) - (values g0 (##sys#--2 k (##sys#integer-power g0 n)))))))))) + (values g0 (- k (##sys#integer-power g0 n)))))))))) (define (##sys#integer-power base e) (define (square x) (##sys#*-2 x x)) (if (negative? e) - (##sys#/-2 1 (##sys#integer-power base (##sys#integer-negate e))) + (##sys#/-2 1 (##sys#integer-power base (integer-negate e))) (let lp ((res 1) (e2 e)) (cond ((eq? e2 0) res) ((even? e2) ; recursion is faster than iteration here (##sys#*-2 res (square (lp 1 (##sys#integer-shift e2 -1))))) (else - (lp (##sys#*-2 res base) (##sys#--2 e2 1))))))) + (lp (##sys#*-2 res base) (- e2 1))))))) (define (expt a b) (define (log-expt a b) @@ -1742,7 +1680,7 @@ EOF ;; (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)))) + (expt (%ratnum-denominator a) (- b)))) ((ratnum? b) ;; x^{a/b} = (x^{1/b})^a (cond @@ -1890,7 +1828,7 @@ EOF (define (round-quotient n d) (let ((q (##sys#integer-quotient n d))) (if ((if (even? q) > >=) (##sys#*-2 (abs (remainder n d)) 2) (abs d)) - (##sys#+-2 q (if (eqv? (negative? n) (negative? d)) 1 -1)) + (+ q (if (eqv? (negative? n) (negative? d)) 1 -1)) q))) ;; Shorthand for readability. TODO: Replace other C_subchar calls with this @@ -1992,13 +1930,13 @@ EOF (let* ((te (and tail (fx- e (fx- (cdr tail) start)))) (num (and tail (car tail))) (t (safe-exponent num te))) - (cons (if t (##sys#+-2 h t) h) (cdr ee))))) + (cons (if t (+ h t) h) (cdr ee))))) (else (let* ((last (or next len)) (te (and tail (fx- start last))) (num (and tail (car tail))) (t (safe-exponent num te)) (h (or decimal-head 0))) - (cons (if t (##sys#+-2 h t) h) next))))))))) + (cons (if t (+ h t) h) next))))))))) (scan-ureal (lambda (start neg?) (if (and (fx> len (fx+ start 1)) (eq? radix 10) @@ -4487,7 +4425,7 @@ EOF (define (bitwise-not n) (##sys#check-exact-integer n 'bitwise-not) - (##sys#integer-minus -1 n)) + (##core#inline_allocate ("C_s_a_u_i_integer_minus" 6) -1 n)) (define (arithmetic-shift n m) (##sys#check-exact-integer n 'arithmetic-shift) diff --git a/runtime.c b/runtime.c index a476184f..614ded56 100644 --- a/runtime.c +++ b/runtime.c @@ -875,9 +875,7 @@ static C_PTABLE_ENTRY *create_initial_ptable() C_pte(C_apply_values); /* XXX TODO OBSOLETE: This can be removed after recompiling c-platform.scm */ C_pte(C_times); - /* XXX TODO OBSOLETE: This can be removed after recompiling c-platform.scm */ C_pte(C_minus); - /* XXX TODO OBSOLETE: This can be removed after recompiling c-platform.scm */ C_pte(C_plus); /* XXX TODO OBSOLETE: This can be removed after recompiling c-platform.scm */ C_pte(C_divide); @@ -7785,7 +7783,7 @@ static C_word rat_plusmin_rat(C_word **ptr, C_word x, C_word y, integer_plusmin_ /* The maximum size this needs is that required to store a complex * number result, where both real and imag parts consist of ratnums. * The maximum size of those ratnums is if they consist of two "fix - * bignums", so we're looking at C_SIZEOF_STRUCT(3) * 3 + + * bignums", so we're looking at C_SIZEOF_STRUCTURE(3) * 3 + * C_SIZEOF_FIX_BIGNUM * 4 = 36 words! */ C_regparm C_word C_fcall @@ -7939,56 +7937,28 @@ C_s_a_u_i_integer_plus(C_word **ptr, C_word n, C_word x, C_word y) } } -/* XXX TODO OBSOLETE: This can be removed after recompiling c-platform.scm */ void C_ccall C_plus(C_word c, C_word closure, C_word k, ...) { + C_word next_val, result = C_fix(0), prev_result = result; + C_word ab[2][C_SIZEOF_STRUCTURE(3) * 3 + C_SIZEOF_FIX_BIGNUM * 4], *a; va_list v; - C_word x, y; - C_word iresult = C_fix(0); - double fresult; - C_alloc_flonum; + c -= 2; va_start(v, k); - c -= 2; - - while(c--) { - x = va_arg(v, C_word); - - if(x & C_FIXNUM_BIT) { - y = C_i_o_fixnum_plus(iresult, x); - - if(y == C_SCHEME_FALSE) { - fresult = (double)C_unfix(iresult) + (double)C_unfix(x); - goto flonum_result; - } - else iresult = y; - } - else if(!C_immediatep(x) && C_block_header(x) == C_FLONUM_TAG) { - fresult = (double)C_unfix(iresult) + C_flonum_magnitude(x); - goto flonum_result; - } - else barf(C_BAD_ARGUMENT_TYPE_ERROR, "+", x); - } - - va_end(v); - C_kontinue(k, iresult); - - flonum_result: - while(c--) { - x = va_arg(v, C_word); - if(x & C_FIXNUM_BIT) - fresult += (double)C_unfix(x); - else if(!C_immediatep(x) && C_block_header(x) == C_FLONUM_TAG) - fresult += C_flonum_magnitude(x); - else barf(C_BAD_ARGUMENT_TYPE_ERROR, "+", x); + while (c--) { + next_val = va_arg(v, C_word); + a = ab[c&1]; /* One may hold last iteration result, the other is unused */ + result = C_s_a_i_plus(&a, 2, result, next_val); + result = move_buffer_object(&a, ab[(c+1)&1], result); + clear_buffer_object(ab[(c+1)&1], prev_result); + prev_result = result; } va_end(v); - C_kontinue_flonum(k, fresult); + C_kontinue(k, result); } - /* XXX TODO OBSOLETE: This can be removed after recompiling c-platform.scm */ C_regparm C_word C_fcall C_2_plus(C_word **ptr, C_word x, C_word y) { @@ -8226,73 +8196,35 @@ C_s_a_u_i_integer_minus(C_word **ptr, C_word n, C_word x, C_word y) } } -/* XXX TODO OBSOLETE: This can be removed after recompiling c-platform.scm */ void C_ccall C_minus(C_word c, C_word closure, C_word k, C_word n1, ...) { + C_word next_val, result = n1, prev_result = result; + C_word ab[2][C_SIZEOF_STRUCTURE(3) * 3 + C_SIZEOF_FIX_BIGNUM * 4], *a; va_list v; - C_word x, y; - C_word iresult; - double fresult; - int ff = 0; - C_alloc_flonum; - if(c < 3) C_bad_min_argc(c, 3); - - if(n1 & C_FIXNUM_BIT) iresult = n1; - else if(!C_immediatep(n1) && C_block_header(n1) == C_FLONUM_TAG) { - fresult = C_flonum_magnitude(n1); - ff = 1; - } - else barf(C_BAD_ARGUMENT_TYPE_ERROR, "-", n1); - - if(c == 3) { - if(!ff) C_kontinue(k, C_fix(-C_unfix(n1))); - else C_kontinue_flonum(k, -fresult); - } - - va_start(v, n1); - c -= 3; - - if(ff) goto flonum_result; - - while(c--) { - x = va_arg(v, C_word); - - if(x & C_FIXNUM_BIT) { - y = C_i_o_fixnum_difference(iresult, x); + if (c < 3) { + C_bad_min_argc(c, 3); + } else if (c == 3) { + a = ab[0]; + C_kontinue(k, C_s_a_i_negate(&a, 1, n1)); + } else { + c -= 2; + va_start(v, n1); - if(y == C_SCHEME_FALSE) { - fresult = (double)C_unfix(iresult) - (double)C_unfix(x); - goto flonum_result; - } - else iresult = y; - } - else if(!C_immediatep(x) && C_block_header(x) == C_FLONUM_TAG) { - fresult = (double)C_unfix(iresult) - C_flonum_magnitude(x); - goto flonum_result; + while (--c) { + next_val = va_arg(v, C_word); + a = ab[c&1]; /* One may hold last iteration result, the other is unused */ + result = C_s_a_i_minus(&a, 2, result, next_val); + result = move_buffer_object(&a, ab[(c+1)&1], result); + clear_buffer_object(ab[(c+1)&1], prev_result); + prev_result = result; } - else barf(C_BAD_ARGUMENT_TYPE_ERROR, "-", x); - } - - va_end(v); - C_kontinue(k, iresult); - - flonum_result: - while(c--) { - x = va_arg(v, C_word); - if(x & C_FIXNUM_BIT) - fresult -= (double)C_unfix(x); - else if(!C_immediatep(x) && C_block_header(x) == C_FLONUM_TAG) - fresult -= C_flonum_magnitude(x); - else barf(C_BAD_ARGUMENT_TYPE_ERROR, "-", x); + va_end(v); + C_kontinue(k, result); } - - va_end(v); - C_kontinue_flonum(k, fresult); } - /* XXX TODO OBSOLETE: This can be removed after recompiling c-platform.scm */ C_regparm C_word C_fcall C_2_minus(C_word **ptr, C_word x, C_word y) { diff --git a/types.db b/types.db index b8f410a5..cf53d1e6 100644 --- a/types.db +++ b/types.db @@ -898,7 +898,8 @@ ((integer integer) (##sys#integer-bitwise-xor #(1) #(2)))) (bitwise-not (#(procedure #:clean #:enforce #:foldable) bitwise-not (integer) integer) - ((integer) (##sys#integer-minus '-1 #(1)))) + ((integer) + (##core#inline_allocate ("C_s_a_u_i_integer_minus" 6) '1 #(1)))) (blob->string (#(procedure #:clean #:enforce) blob->string (blob) string))Trap