;;; a simple inliner ;; Copyright (C) 2009, 2010 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 (define-module (language tree-il inline) #:use-module (system base pmatch) #:use-module (system base syntax) #:use-module (language tree-il) #:export (inline!)) ;; Possible optimizations: ;; * constant folding, propagation ;; * procedure inlining ;; * always when single call site ;; * always for "trivial" procs ;; * otherwise who knows ;; * dead code elimination ;; * degenerate case optimizations ;; * "fixing letrec" ;; This is a completely brain-dead optimization pass whose sole claim to ;; fame is ((lambda () x)) => x. (define (inline! x) (define (inline1 x) (record-case x (( src proc args) (record-case proc ;; ((lambda (y ...) x) z ...) => (let ((y z) ...) x) (( body) (let lp ((lcase body)) (and lcase (record-case lcase (( req opt rest kw inits vars body alternate) (if (and (= (length vars) (length req) (length args))) (let ((x (make-let src req vars args body))) (or (inline1 x) x)) (lp alternate))))))) (( name) (case name ((@call-with-values) (pmatch args ;; (call-with-values (lambda () foo) (lambda (a b . c) bar)) ;; => (let-values (((a b . c) foo)) bar) ;; ;; Note that this is a singly-binding form of let-values. ;; Also note that Scheme's let-values expands into ;; call-with-values, then here we reduce it to tree-il's ;; let-values. ((,producer ,consumer) (guard (lambda? consumer) (lambda-case? (lambda-body consumer)) (not (lambda-case-opt (lambda-body consumer))) (not (lambda-case-kw (lambda-body consumer))) (not (lambda-case-alternate (lambda-body consumer)))) (make-let-values src (let ((x (make-application src producer '()))) (or (inline1 x) x)) (lambda-body consumer))) (else #f))) ((memq memv) (pmatch args ((,k ,l) (guard (const? l) (list? (const-exp l))) (if (null? (const-exp l)) (make-const #f #f) (let lp ((elts (const-exp l))) (let ((test (make-application #f (make-primitive-ref #f (case name ((memq) 'eq?) ((memv) 'eqv?) (else (error "what")))) (list k (make-const #f (car elts)))))) (if (null? (cdr elts)) test (make-conditional src test (make-const #f #t) (lp (cdr elts)))))))) (else #f))) (else #f))) (else #f))) (( vars body) (if (null? vars) body x)) (( vars body) (if (null? vars) body x)) (( vars body) (if (null? vars) body x)) (( req opt rest kw vars body alternate) (define (args-compatible? args vars) (let lp ((args args) (vars vars)) (cond ((null? args) (null? vars)) ((null? vars) #f) ((and (lexical-ref? (car args)) (eq? (lexical-ref-gensym (car args)) (car vars))) (lp (cdr args) (cdr vars))) (else #f)))) (and (not opt) (not kw) (not alternate) (record-case body (( proc args) ;; (lambda args (apply (lambda ...) args)) => (lambda ...) (and (primitive-ref? proc) (eq? (primitive-ref-name proc) '@apply) (pair? args) (lambda? (car args)) (args-compatible? (cdr args) vars) (lambda-body (car args)))) (else #f)))) ;; Actually the opposite of inlining -- if the prompt cannot be proven to ;; be escape-only, ensure that its body is the application of a thunk. (( src tag body handler) (define (escape-only? handler) (and (pair? (lambda-case-req handler)) (let ((cont (car (lambda-case-vars handler)))) (tree-il-fold (lambda (leaf escape-only?) (and escape-only? (not (and (lexical-ref? leaf) (eq? (lexical-ref-gensym leaf) cont))))) (lambda (down escape-only?) escape-only?) (lambda (up escape-only?) escape-only?) #t (lambda-case-body handler))))) (define (make-thunk body) (make-lambda #f '() (make-lambda-case #f '() #f #f #f '() '() body #f))) (if (or (and (application? body) (lambda? (application-proc body)) (null? (application-args body))) (escape-only? handler)) x (make-prompt src tag (make-application #f (make-thunk body) '()) handler))) (else #f))) (post-order! inline1 x))