mirror of
https://git.savannah.gnu.org/git/guile.git
synced 2025-06-17 17:20:29 +02:00
New instructions load-f64, load-u64
* libguile/instructions.c (FOR_EACH_INSTRUCTION_WORD_TYPE): Add word types for immediate f64 and u64 values. (TYPE_WIDTH): Bump up by a bit, now that we have 32 word types. (NOP, parse_instruction): Use 64-bit meta type. * libguile/vm-engine.c (load-f64, load-u64): New instructions. * module/language/bytecode.scm (compute-instruction-arity): Add parser for new instruction word types. * module/language/cps/compile-bytecode.scm (compile-function): Add special-cased assemblers for new instructions, and also for scm->u64 and u64->scm which I missed before. * module/language/cps/effects-analysis.scm (load-f64, load-u64): New instructions. * module/language/cps/slot-allocation.scm (compute-needs-slot): load-f64 and load-u64 don't need slots. (compute-var-representations): Update for new instructions. * module/language/cps/specialize-primcalls.scm (specialize-primcalls): Specialize scm->f64 and scm->u64 to make-f64 and make-u64. * module/language/cps/types.scm (load-f64, load-u64): Wire up to type inference, though currently type inference only runs before specialization. * module/language/cps/utils.scm (compute-defining-expressions): For some reason I don't understand, it's possible to see two definitions that are equal but not equal? here. Allow for now. (compute-constant-values): Punch through type conversions to get constant u64/f64 values. * module/system/vm/assembler.scm (assembler): Support for new word types. Export the new assemblers.
This commit is contained in:
parent
bdfa1c1b42
commit
f34688ad25
11 changed files with 119 additions and 23 deletions
|
@ -51,8 +51,8 @@
|
|||
(case word
|
||||
((C32) 1)
|
||||
((I32) 1)
|
||||
((A32) 1)
|
||||
((B32) 0)
|
||||
((A32 AU32 AF32) 1)
|
||||
((B32 BF32 BU32) 0)
|
||||
((N32) 1)
|
||||
((R32) 1)
|
||||
((L32) 1)
|
||||
|
|
|
@ -196,8 +196,16 @@
|
|||
(emit-builtin-ref asm (from-sp dst) (constant name)))
|
||||
(($ $primcall 'scm->f64 (src))
|
||||
(emit-scm->f64 asm (from-sp dst) (from-sp (slot src))))
|
||||
(($ $primcall 'load-f64 (src))
|
||||
(emit-load-f64 asm (from-sp dst) (constant src)))
|
||||
(($ $primcall 'f64->scm (src))
|
||||
(emit-f64->scm asm (from-sp dst) (from-sp (slot src))))
|
||||
(($ $primcall 'scm->u64 (src))
|
||||
(emit-scm->u64 asm (from-sp dst) (from-sp (slot src))))
|
||||
(($ $primcall 'load-u64 (src))
|
||||
(emit-load-u64 asm (from-sp dst) (constant src)))
|
||||
(($ $primcall 'u64->scm (src))
|
||||
(emit-u64->scm asm (from-sp dst) (from-sp (slot src))))
|
||||
(($ $primcall 'bv-length (bv))
|
||||
(emit-bv-length asm (from-sp dst) (from-sp (slot bv))))
|
||||
(($ $primcall 'bv-u8-ref (bv idx))
|
||||
|
|
|
@ -354,8 +354,10 @@ is or might be a read or a write to the same location as A."
|
|||
;; Unboxed floats and integers.
|
||||
(define-primitive-effects
|
||||
((scm->f64 _) &type-check)
|
||||
((load-f64 _))
|
||||
((f64->scm _))
|
||||
((scm->u64 _) &type-check)
|
||||
((load-u64 _))
|
||||
((u64->scm _)))
|
||||
|
||||
;; Bytevectors.
|
||||
|
|
|
@ -323,6 +323,8 @@ the definitions that are live before and after LABEL, as intsets."
|
|||
(match exp
|
||||
(($ $const)
|
||||
empty-intset)
|
||||
(($ $primcall (or 'load-f64 'load-u64) (val))
|
||||
empty-intset)
|
||||
(($ $primcall 'free-ref (closure slot))
|
||||
(defs+ closure))
|
||||
(($ $primcall 'free-set! (closure slot value))
|
||||
|
@ -794,10 +796,11 @@ are comparable with eqv?. A tmp slot may be used."
|
|||
(($ $values (arg))
|
||||
(intmap-add representations var
|
||||
(intmap-ref representations arg)))
|
||||
(($ $primcall (or 'scm->f64 'bv-f32-ref 'bv-f64-ref
|
||||
(($ $primcall (or 'scm->f64 'load-f64
|
||||
'bv-f32-ref 'bv-f64-ref
|
||||
'fadd 'fsub 'fmul 'fdiv))
|
||||
(intmap-add representations var 'f64))
|
||||
(($ $primcall (or 'scm->u64 'bv-length
|
||||
(($ $primcall (or 'scm->u64 'load-u64 'bv-length
|
||||
'uadd 'usub 'umul
|
||||
'uadd/immediate 'usub/immediate 'umul/immediate))
|
||||
(intmap-add representations var 'u64))
|
||||
|
|
|
@ -36,6 +36,12 @@
|
|||
(define (u8? var)
|
||||
(let ((val (intmap-ref constants var (lambda (_) #f))))
|
||||
(and (exact-integer? val) (<= 0 val 255))))
|
||||
(define (u64? var)
|
||||
(let ((val (intmap-ref constants var (lambda (_) #f))))
|
||||
(and (exact-integer? val) (<= 0 val #xffffFFFFffffFFFF))))
|
||||
(define (f64? var)
|
||||
(let ((val (intmap-ref constants var (lambda (_) #f))))
|
||||
(and (number? val) (inexact? val) (real? val))))
|
||||
(define (specialize-primcall name args)
|
||||
(define (rename name)
|
||||
(build-exp ($primcall name args)))
|
||||
|
@ -54,6 +60,8 @@
|
|||
(('usub x (? u8? y)) (build-exp ($primcall 'usub/immediate (x y))))
|
||||
(('umul x (? u8? y)) (build-exp ($primcall 'umul/immediate (x y))))
|
||||
(('umul (? u8? x) y) (build-exp ($primcall 'umul/immediate (y x))))
|
||||
(('scm->f64 (? f64?)) (rename 'load-f64))
|
||||
(('scm->u64 (? u64?)) (rename 'load-u64))
|
||||
(_ #f)))
|
||||
(intmap-map
|
||||
(lambda (label cont)
|
||||
|
|
|
@ -706,6 +706,7 @@ minimum, and maximum."
|
|||
(define-type-inferrer (scm->f64 scm result)
|
||||
(restrict! scm &real -inf.0 +inf.0)
|
||||
(define! result &f64 (&min scm) (&max scm)))
|
||||
(define-type-aliases scm->f64 load-f64)
|
||||
|
||||
(define-type-checker (f64->scm f64)
|
||||
#t)
|
||||
|
@ -717,6 +718,7 @@ minimum, and maximum."
|
|||
(define-type-inferrer (scm->u64 scm result)
|
||||
(restrict! scm &exact-integer 0 #xffffffffffffffff)
|
||||
(define! result &u64 (max (&min scm) 0) (min (&max scm) #xffffffffffffffff)))
|
||||
(define-type-aliases scm->u64 load-u64)
|
||||
|
||||
(define-type-checker (u64->scm u64)
|
||||
#t)
|
||||
|
|
|
@ -182,9 +182,11 @@ disjoint, an error will be signalled."
|
|||
|
||||
(define (compute-defining-expressions conts)
|
||||
(define (meet-defining-expressions old new)
|
||||
;; If there are multiple definitions, punt and
|
||||
;; record #f.
|
||||
#f)
|
||||
;; If there are multiple definitions and they are different, punt
|
||||
;; and record #f.
|
||||
(if (equal? old new)
|
||||
old
|
||||
#f))
|
||||
(persistent-intmap
|
||||
(intmap-fold (lambda (label cont defs)
|
||||
(match cont
|
||||
|
@ -198,14 +200,35 @@ disjoint, an error will be signalled."
|
|||
empty-intmap)))
|
||||
|
||||
(define (compute-constant-values conts)
|
||||
(persistent-intmap
|
||||
(intmap-fold (lambda (var exp out)
|
||||
(match exp
|
||||
(($ $const val)
|
||||
(intmap-add! out var val))
|
||||
(_ out)))
|
||||
(compute-defining-expressions conts)
|
||||
empty-intmap)))
|
||||
(let ((defs (compute-defining-expressions conts)))
|
||||
(persistent-intmap
|
||||
(intmap-fold
|
||||
(lambda (var exp out)
|
||||
(match exp
|
||||
(($ $primcall (or 'load-f64 'load-u64) (val))
|
||||
(intmap-add! out var (intmap-ref out val)))
|
||||
;; Punch through type conversions to allow uadd to specialize
|
||||
;; to uadd/immediate.
|
||||
(($ $primcall 'scm->f64 (val))
|
||||
(let ((f64 (intmap-ref out val (lambda (_) #f))))
|
||||
(if (and f64 (number? f64) (inexact? f64) (real? f64))
|
||||
(intmap-add! out var f64)
|
||||
out)))
|
||||
(($ $primcall 'scm->u64 (val))
|
||||
(let ((u64 (intmap-ref out val (lambda (_) #f))))
|
||||
(if (and u64 (number? u64) (exact-integer? u64)
|
||||
(<= 0 u64 #xffffFFFFffffFFFF))
|
||||
(intmap-add! out var u64)
|
||||
out)))
|
||||
(_ out)))
|
||||
defs
|
||||
(intmap-fold (lambda (var exp out)
|
||||
(match exp
|
||||
(($ $const val)
|
||||
(intmap-add! out var val))
|
||||
(_ out)))
|
||||
defs
|
||||
empty-intmap)))))
|
||||
|
||||
(define (compute-function-body conts kfun)
|
||||
(persistent-intset
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue