weakening the optimization: it's not sound if there's mutation, and I don't currently analyze operands for mutation yet.

This commit is contained in:
Danny Yoo 2011-04-01 13:37:07 -04:00
parent dbd8d5bebc
commit f581b027a2
3 changed files with 60 additions and 4 deletions

View File

@ -533,7 +533,7 @@
(kernel-primitive-expected-operand-types kernel-op n)]
[(constant-operands rest-operands)
(split-operands-by-constant-or-stack-references
(split-operands-by-constants
(App-operands exp))]
;; here, we rewrite the stack references so they assume no scratch space
@ -668,13 +668,15 @@
#f])]))
(: split-operands-by-constant-or-stack-references
(: split-operands-by-constants
((Listof Expression) ->
(values (Listof (U Constant LocalRef ToplevelRef))
(Listof Expression))))
;; Splits off the list of operations into two: a prefix of constant
;; or simple expressions, and the remainder.
(define (split-operands-by-constant-or-stack-references rands)
;; TODO: if we can statically determine what arguments are immutable, regardless of
;; side effects, we can do a much better job here...
(define (split-operands-by-constants rands)
(let: loop : (values (Listof (U Constant LocalRef ToplevelRef)) (Listof Expression))
([rands : (Listof Expression) rands]
[constants : (Listof (U Constant LocalRef ToplevelRef))
@ -683,7 +685,9 @@
(values (reverse constants) empty)]
[else (let ([e (first rands)])
(if (or (Constant? e)
(and (LocalRef? e) (not (LocalRef-unbox? e)))
;; These two are commented out because it's not sound otherwise.
#;(and (LocalRef? e) (not (LocalRef-unbox? e)))
#;(and (ToplevelRef? e)
(let ([prefix (ensure-prefix
(list-ref cenv (ToplevelRef-depth e)))])

View File

@ -811,6 +811,34 @@
(test '(begin (define (f)
(define cont #f)
(define n 0)
(call/cc (lambda (x) (set! cont x)))
(set! n (add1 n))
(if (< n 10)
(cont 'dontcare))
n)
(f))
10)
;; This should produce 0 because there needs to be a continuation prompt around each evaluation.
#;(test '(begin
(define cont #f)
(define n 0)
(call/cc (lambda (x) (set! cont x)))
(set! n (add1 n))
(if (< n 10)
(cont 'dontcare))
n)
0)
;; FIXME: this test is failing. I think we need prompts to delimit
;; the continuation capture.
#;(test '(begin

View File

@ -325,6 +325,30 @@
#t)))
;; This hsould only show 0, because there should be a prompt that controls continuation capture
#;(test '(begin (define cont #f)
(define n 0)
(call/cc (lambda (x) (set! cont x)))
(displayln n)
(set! n (add1 n))
(when (< n 10)
(cont))))
;; This should show the numbers 0 through 10
#;(test '(begin (define (f)
(define cont #f)
(define n 0)
(call/cc (lambda (x) (set! cont x)))
(displayln n)
(set! n (add1 n))
(when (< n 10)
(cont)))
(f)))
#;(test (parse '(letrec ([x (lambda (x) (y x))]
[y (lambda (x) (x y))])
(x y)))