summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorAndy Wingo <wingo@pobox.com>2014-04-05 11:08:47 +0200
committerAndy Wingo <wingo@pobox.com>2014-04-05 11:40:21 +0200
commit634638801c72dec6bc09c88c53728f5a17e1a683 (patch)
tree3c2e7daef8633320e98c44a03c623ba172a2ed69
parent84d3ce20cd12c7f2bf84637bcc4843772d62191a (diff)
Add prune-bailouts pass
* module/language/cps/prune-bailouts.scm: New pass. * module/language/cps/compile-bytecode.scm: Wire it up. * module/Makefile.am: Add new file.
-rw-r--r--module/Makefile.am1
-rw-r--r--module/language/cps/compile-bytecode.scm4
-rw-r--r--module/language/cps/prune-bailouts.scm98
3 files changed, 102 insertions, 1 deletions
diff --git a/module/Makefile.am b/module/Makefile.am
index 783173e47..b3b96d9c0 100644
--- a/module/Makefile.am
+++ b/module/Makefile.am
@@ -130,6 +130,7 @@ CPS_LANG_SOURCES = \
language/cps/effects-analysis.scm \
language/cps/elide-values.scm \
language/cps/primitives.scm \
+ language/cps/prune-bailouts.scm \
language/cps/prune-top-level-scopes.scm \
language/cps/reify-primitives.scm \
language/cps/renumber.scm \
diff --git a/module/language/cps/compile-bytecode.scm b/module/language/cps/compile-bytecode.scm
index 992490272..c28d4d7e7 100644
--- a/module/language/cps/compile-bytecode.scm
+++ b/module/language/cps/compile-bytecode.scm
@@ -36,6 +36,7 @@
#:use-module (language cps dfg)
#:use-module (language cps elide-values)
#:use-module (language cps primitives)
+ #:use-module (language cps prune-bailouts)
#:use-module (language cps prune-top-level-scopes)
#:use-module (language cps reify-primitives)
#:use-module (language cps renumber)
@@ -62,7 +63,8 @@
;; called. The last is mainly to eliminate rest parameters that
;; aren't used, and thus shouldn't be consed.
- (let* ((exp (run-pass exp eliminate-dead-code #:eliminate-dead-code? #t))
+ (let* ((exp (run-pass exp prune-bailouts #:prune-bailouts? #t))
+ (exp (run-pass exp eliminate-dead-code #:eliminate-dead-code? #t))
(exp (run-pass exp prune-top-level-scopes #:prune-top-level-scopes? #t))
(exp (run-pass exp simplify #:simplify? #t))
(exp (run-pass exp contify #:contify? #t))
diff --git a/module/language/cps/prune-bailouts.scm b/module/language/cps/prune-bailouts.scm
new file mode 100644
index 000000000..91afc180b
--- /dev/null
+++ b/module/language/cps/prune-bailouts.scm
@@ -0,0 +1,98 @@
+;;; Continuation-passing style (CPS) intermediate language (IL)
+
+;; Copyright (C) 2013, 2014 Free Software Foundation, Inc.
+
+;;;; This library is free software; you can redistribute it and/or
+;;;; modify it under the terms of the GNU Lesser General Public
+;;;; License as published by the Free Software Foundation; either
+;;;; version 3 of the License, or (at your option) any later version.
+;;;;
+;;;; This library is distributed in the hope that it will be useful,
+;;;; but WITHOUT ANY WARRANTY; without even the implied warranty of
+;;;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+;;;; Lesser General Public License for more details.
+;;;;
+;;;; You should have received a copy of the GNU Lesser General Public
+;;;; License along with this library; if not, write to the Free Software
+;;;; Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
+
+;;; Commentary:
+;;;
+;;; A pass that prunes successors of expressions that bail out.
+;;;
+;;; Code:
+
+(define-module (language cps prune-bailouts)
+ #:use-module (ice-9 match)
+ #:use-module (language cps)
+ #:export (prune-bailouts))
+
+(define (module-box src module name public? bound? val-proc)
+ (let-fresh (kbox) (module-sym name-sym public?-sym bound?-sym box)
+ (build-cps-term
+ ($letconst (('module module-sym module)
+ ('name name-sym name)
+ ('public? public?-sym public?)
+ ('bound? bound?-sym bound?))
+ ($letk ((kbox ($kargs ('box) (box) ,(val-proc box))))
+ ($continue kbox src
+ ($primcall 'cached-module-box
+ (module-sym name-sym public?-sym bound?-sym))))))))
+
+(define (primitive-ref name k src)
+ (module-box #f '(guile) name #f #t
+ (lambda (box)
+ (build-cps-term
+ ($continue k src ($primcall 'box-ref (box)))))))
+
+(define (prune-bailouts* fun)
+ (define (visit-cont cont ktail)
+ (rewrite-cps-cont cont
+ (($ $cont label ($ $kargs names vars body))
+ (label ($kargs names vars ,(visit-term body ktail))))
+ (($ $cont label ($ $kentry self tail clause))
+ (label ($kentry self ,tail
+ ,(and clause (visit-cont clause ktail)))))
+ (($ $cont label ($ $kclause arity body alternate))
+ (label ($kclause ,arity ,(visit-cont body ktail)
+ ,(and alternate (visit-cont alternate ktail)))))
+ (_ ,cont)))
+
+ (define (visit-term term ktail)
+ (rewrite-cps-term term
+ (($ $letrec names vars funs body)
+ ($letrec names vars (map prune-bailouts* funs)
+ ,(visit-term body ktail)))
+ (($ $letk conts body)
+ ($letk ,(map (lambda (cont) (visit-cont cont ktail)) conts)
+ ,(visit-term body ktail)))
+ (($ $continue k src exp)
+ ,(visit-exp k src exp ktail))))
+
+ (define (visit-exp k src exp ktail)
+ (rewrite-cps-term exp
+ (($ $fun) ($continue k src ,(prune-bailouts* exp)))
+ (($ $primcall (and name (or 'error 'scm-error 'throw)) args)
+ ,(if (eq? k ktail)
+ (build-cps-term ($continue k src ,exp))
+ (let-fresh (kprim kresult kreceive) (prim rest)
+ (build-cps-term
+ ($letk ((kresult ($kargs ('rest) (rest)
+ ($continue ktail src ($values ()))))
+ (kreceive ($kreceive '() 'rest kresult))
+ (kprim ($kargs ('prim) (prim)
+ ($continue kreceive src
+ ($call prim args)))))
+ ,(primitive-ref name kprim src))))))
+ (_ ($continue k src ,exp))))
+
+ (rewrite-cps-exp fun
+ (($ $fun src meta free
+ ($ $cont kentry ($ $kentry self ($ $cont ktail ($ $ktail)) clause)))
+ ($fun src meta free
+ (kentry ($kentry self (ktail ($ktail))
+ ,(and clause (visit-cont clause ktail))))))))
+
+(define (prune-bailouts fun)
+ (with-fresh-name-state fun
+ (prune-bailouts* fun)))