From ba4555ae4fdbbdac69c798f4210472f10aea583c Mon Sep 17 00:00:00 2001 From: William Harold Newman Date: Tue, 4 Dec 2001 21:51:42 +0000 Subject: [PATCH] 0.pre7.86.flaky7.20: (This version bootstrapped successfully from 0.6.13, passed regression tests, served as its own bootstrap host, and passed regression tests again. But it still has debugger/signal/whatever problems, so that A doesn't work from the debugger prompt, and after it fails, (SB-EXT:QUIT) doesn't work either.) The failure in clocc-ansi-test was in the find-the-home-lambda part of the new code which updates LAMBDA-REFERS-TO-VAR in IR1 translation of SETQ. We try to find the home lambda by looking backwards through continuations, but in the (SETQ X 3) form in (SETQ X 2) (RETURN-FROM STOP) (SETQ X 3)) it doesn't work to look backwards. (To make a point which eluded me for a while, this isn't a bug in the way that continuations are set up, it's because the (SETQ X 3) form is orphaned by the way that control always jumps out through RETURN-FROM, so that the (SETQ X 3) form effectively has no home lambda.) So... ...redid CONTINUATION-HOME-LAMBDA in terms of CONTINUATION-HOME-LAMBDA-OR-NULL, and BLOCK-HOME-LAMBDA in terms of BLOCK-HOME-LAMBDA-OR-NULL ...made the update-LAMBDA-REFERS-TO-VAR code accept a NIL answer from CONTINUATION-HOME-LAMBDA-OR-NULL --- NEWS | 6 ++- src/compiler/ir1-translators.lisp | 22 ++++++++--- src/compiler/ir1tran.lisp | 12 +++--- src/compiler/ir1util.lisp | 73 ++++++++++++++++++++++++++----------- version.lisp-expr | 2 +- 5 files changed, 81 insertions(+), 34 deletions(-) diff --git a/NEWS b/NEWS index 8c3bbd5..b699c6e 100644 --- a/NEWS +++ b/NEWS @@ -844,7 +844,11 @@ changes in sbcl-0.7.0 relative to sbcl-0.6.13: introduced some new bugs, but since the new logic is simpler and more orthogonal, hopefully it will be easier to clean up bugs in the new code than it was in the old code. -* The rewritten compiler is still a little unsteady on its feet. +* Some obscure compiler bugs (notably a messy one involving closures + over variables when the closure relationship was stretched over + multiple compilation components) were exposed by the new way that + the compiler is used, and were fixed. +* The revised compiler is still a little unsteady on its feet. In particular, the debugging information it produces is sometimes broken, and the support for inlining FOO when you (DECLAIM (INLINE FOO)) then do (DEFUN FOO ..) in a non-null diff --git a/src/compiler/ir1-translators.lisp b/src/compiler/ir1-translators.lisp index e7c4b01..4d71fe5 100644 --- a/src/compiler/ir1-translators.lisp +++ b/src/compiler/ir1-translators.lisp @@ -85,9 +85,6 @@ (ir1-convert-progn-body dummy cont forms)))) -;;; We make CONT start a block just so that it will have a block -;;; assigned. People assume that when they pass a continuation into -;;; IR1-CONVERT as CONT, it will have a block when it is done. (def-ir1-translator return-from ((name &optional value) start cont) #!+sb-doc @@ -95,6 +92,20 @@ Evaluate the Value-Form, returning its values from the lexically enclosing BLOCK Block-Name. This is constrained to be used only within the dynamic extent of the BLOCK." + ;; CMU CL comment: + ;; We make CONT start a block just so that it will have a block + ;; assigned. People assume that when they pass a continuation into + ;; IR1-CONVERT as CONT, it will have a block when it is done. + ;; KLUDGE: Note that this block is basically fictitious. In the code + ;; (BLOCK B (RETURN-FROM B) (SETQ X 3)) + ;; it's the block which answers the question "which block is + ;; the (SETQ X 3) in?" when the right answer is that (SETQ X 3) is + ;; dead code and so doesn't really have a block at all. The existence + ;; of this block, and that way that it doesn't explicitly say + ;; "I'm actually nowhere at all" makes some logic (e.g. + ;; BLOCK-HOME-LAMBDA-OR-NULL) more obscure, and it might be better + ;; to get rid of it, perhaps using a special placeholder value + ;; to indicate the orphanedness of the code. (continuation-starts-block cont) (let* ((found (or (lexenv-find name blocks) (compiler-error "return for unknown block: ~S" name))) @@ -742,8 +753,9 @@ (when (constant-p leaf) (compiler-error "~S is a constant and thus can't be set." name)) (when (lambda-var-p leaf) - (let ((home-lambda (continuation-home-lambda start))) - (pushnew leaf (lambda-refers-to-vars home-lambda))) + (let ((home-lambda (continuation-home-lambda-or-null start))) + (when home-lambda + (pushnew leaf (lambda-refers-to-vars home-lambda)))) (when (lambda-var-ignorep leaf) ;; ANSI's definition of "Declaration IGNORE, IGNORABLE" ;; requires that this be a STYLE-WARNING, not a full warning. diff --git a/src/compiler/ir1tran.lisp b/src/compiler/ir1tran.lisp index 61b9a74..c530557 100644 --- a/src/compiler/ir1tran.lisp +++ b/src/compiler/ir1tran.lisp @@ -524,8 +524,9 @@ (etypecase var (leaf (when (lambda-var-p var) - (pushnew var - (lambda-refers-to-vars (continuation-home-lambda start))) + (let ((home (continuation-home-lambda-or-null start))) + (when home + (pushnew var (lambda-refers-to-vars home)))) (when (lambda-var-ignorep var) ;; (ANSI's specification for the IGNORE declaration requires ;; that this be a STYLE-WARNING, not a full WARNING.) @@ -556,8 +557,8 @@ (t (ir1-convert-global-functoid-no-cmacro start cont form fun))))) -;;; Handle the case of where the call was not a compiler macro, or was a -;;; compiler macro and passed. +;;; Handle the case of where the call was not a compiler macro, or was +;;; a compiler macro and passed. (defun ir1-convert-global-functoid-no-cmacro (start cont form fun) (declare (type continuation start cont) (list form)) ;; FIXME: Couldn't all the INFO calls here be converted into @@ -675,7 +676,8 @@ (return)) (let ((this-cont (make-continuation))) (ir1-convert this-start this-cont form) - (setq this-start this-cont forms (cdr forms))))))) + (setq this-start this-cont + forms (cdr forms))))))) (values)) ;;;; converting combinations diff --git a/src/compiler/ir1util.lisp b/src/compiler/ir1util.lisp index aa4668e..2bd0896 100644 --- a/src/compiler/ir1util.lisp +++ b/src/compiler/ir1util.lisp @@ -266,25 +266,48 @@ (declare (type cblock block)) (node-enclosing-cleanup (block-last block))) -;;; Return the non-LET LAMBDA that holds BLOCK's code. -(defun block-home-lambda (block) +;;; Return the non-LET LAMBDA that holds BLOCK's code, or NIL +;;; if there is none. +;;; +;;; There can legitimately be no home lambda in dead code early in the +;;; IR1 conversion process, e.g. when IR1-converting the SETQ form in +;;; (BLOCK B (RETURN-FROM B) (SETQ X 3)) +;;; where the block is just a placeholder during parsing and doesn't +;;; actually correspond to code which will be written anywhere. +(defun block-home-lambda-or-null (block) (declare (type cblock block)) (if (node-p (block-last block)) ;; This is the old CMU CL way of doing it. (node-home-lambda (block-last block)) - ;; The CMU CL approach sometimes fails, e.g. in IR1-CONVERT of - ;; one of the legs of an IF, now that SBCL uses this operation - ;; more aggressively than CMU CL did. - ;; - ;; In this case we reason that previous-in-target-execution-order - ;; blocks should be in the same lambda, and that they seem in - ;; practice to be previous-in-compilation-order blocks too, - ;; so we look back to find one which is sufficiently - ;; initialized to tell us what the home lambda is. We could - ;; get fancy about this, flooding the graph of all the - ;; previous blocks, but in practice it seems to work just - ;; to grab the first previous block and use it. - (node-home-lambda (block-last (first (block-pred block)))))) + ;; Now that SBCL uses this operation more aggressively than CMU + ;; CL did, the old CMU CL way of doing it can fail in two ways. + ;; 1. It can fail in a few cases even when a meaningful home + ;; lambda exists, e.g. in IR1-CONVERT of one of the legs of + ;; an IF. + ;; 2. It can fail when converting a form which is born orphaned + ;; so that it never had a meaningful home lambda, e.g. a form + ;; which follows a RETURN-FROM or GO form. + (let ((pred-list (block-pred block))) + ;; To deal with case 1, we reason that + ;; previous-in-target-execution-order blocks should be in the + ;; same lambda, and that they seem in practice to be + ;; previous-in-compilation-order blocks too, so we look back + ;; to find one which is sufficiently initialized to tell us + ;; what the home lambda is. + (if pred-list + ;; We could get fancy about this, flooding through the + ;; graph of all the previous blocks, but in practice it + ;; seems to work just to grab the first previous block and + ;; use it. + (node-home-lambda (block-last (first pred-list))) + ;; In case 2, we end up with an empty PRED-LIST and + ;; have to punt: There's no home lambda. + nil)))) + +;;; Return the non-LET LAMBDA that holds BLOCK's code. +(defun block-home-lambda (block) + (the clambda + (block-home-lambda-or-null block))) ;;; Return the IR1 physical environment for BLOCK. (defun block-physenv (block) @@ -333,22 +356,28 @@ (values (node-source-form use) t) (values nil nil)))) -;;; Return the LAMBDA that is CONT's home. -(defun continuation-home-lambda (cont) +;;; Return the LAMBDA that is CONT's home, or NIL if there is none. +(defun continuation-home-lambda-or-null (cont) ;; KLUDGE: This function is a post-CMU-CL hack by WHN, and this ;; implementation might not be quite right, or might be uglier than ;; necessary. It appears that the original Python never found a need ;; to do this operation. The obvious things based on - ;; NODE-HOME-LAMBDA of CONTINUATION-USE usually works; then if that + ;; NODE-HOME-LAMBDA of CONTINUATION-USE usually work; then if that ;; fails, BLOCK-HOME-LAMBDA of CONTINUATION-BLOCK works, given that - ;; generalize it enough to grovel harder when the simple CMU CL - ;; approach fails. -- WHN 2001-12-02 + ;; we generalize it enough to grovel harder when the simple CMU CL + ;; approach fails, and furthermore realize that in some exceptional + ;; cases it might return NIL. -- WHN 2001-12-04 (cond ((continuation-use cont) (node-home-lambda (continuation-use cont))) ((continuation-block cont) - (block-home-lambda (continuation-block cont))) + (block-home-lambda-or-null (continuation-block cont))) (t - (error "internal error: can't find home lambda for ~S")))) + (error "internal error: confused about home lambda for ~S")))) + +;;; Return the LAMBDA that is CONT's home. +(defun continuation-home-lambda (cont) + (the clambda + (continuation-home-lambda-or-null cont))) ;;; Return a new LEXENV just like DEFAULT except for the specified ;;; slot values. Values for the alist slots are NCONCed to the diff --git a/version.lisp-expr b/version.lisp-expr index e45fbb1..6ff8fba 100644 --- a/version.lisp-expr +++ b/version.lisp-expr @@ -18,4 +18,4 @@ ;;; for internal versions, especially for internal versions off the ;;; main CVS branch, it gets hairier, e.g. "0.pre7.14.flaky4.13".) -"0.pre7.86.flaky7.19" +"0.pre7.86.flaky7.20" -- 1.7.10.4