From 2e8635c012cd66347e0f64fdae6e8e9295489cb0 Mon Sep 17 00:00:00 2001 From: cracauer Date: Wed, 19 Jan 2011 22:14:52 +0000 Subject: [PATCH] 1.0.45.5: life: fix slow compile. * Committing a patch I once got from Nikodemus. Without it my toy takes more than a week to compile. I've been using this since November in production, seems to work well. Should probably have made it into 1.0.44.28. ;;; FASTP is a KLUDGE: SBCL used to update the current-conflict only ;;; for the read-only case, but switched at one point to always ;;; updating it. This generally speeds up the compiler nicely, but ;;; sometimes it causes an infinite loop in the updating machinery, ;;; We cheat by switching of the fast path if it seems we're looping ;;; longer then expected. --- src/compiler/life.lisp | 29 +++++++++++++++++++++++------ version.lisp-expr | 2 +- 2 files changed, 24 insertions(+), 7 deletions(-) diff --git a/src/compiler/life.lisp b/src/compiler/life.lisp index 16449af..9d28fbd 100644 --- a/src/compiler/life.lisp +++ b/src/compiler/life.lisp @@ -527,7 +527,14 @@ ;;; We leave the CURRENT-CONFLICT pointing to the conflict for BLOCK1. ;;; The CURRENT-CONFLICT must be initialized to the head of the ;;; GLOBAL-CONFLICTS for the TN between each flow analysis iteration. -(defun propagate-live-tns (block1 block2) + +;;; FASTP is a KLUDGE: SBCL used to update the current-conflict only +;;; for the read-only case, but switched at one point to always +;;; updating it. This generally speeds up the compiler nicely, but +;;; sometimes it causes an infinite loop in the updating machinery, +;;; We cheat by switching of the fast path if it seems we're looping +;;; longer then expected. +(defun propagate-live-tns (block1 block2 fastp) (declare (type ir2-block block1 block2)) (let ((live-in (ir2-block-live-in block1)) (did-something nil)) @@ -556,10 +563,13 @@ (setf (svref (ir2-block-local-tns block1) (global-conflicts-number current)) nil) - (setf (global-conflicts-number current) nil)) + (setf (global-conflicts-number current) nil) + (unless fastp + (setf (tn-current-conflict tn) current))) (t (setf (sbit live-in (global-conflicts-number current)) 1))) - (setf (tn-current-conflict tn) current) + (when fastp + (setf (tn-current-conflict tn) current)) (return))))) (:write))) did-something)) @@ -567,7 +577,14 @@ ;;; Do backward global flow analysis to find all TNs live at each ;;; block boundary. (defun lifetime-flow-analysis (component) - (loop + ;; KLUDGE: This is the second part of the FASTP kludge in + ;; propagate-live-tns: we pass fastp for ten first attempts, + ;; and then switch to the works-for-sure version. + ;; + ;; The upstream uses the fast version always, but sometimes + ;; that gets stuck in a loop... + (loop for i = 0 then (1+ i) + do (reset-current-conflict component) (let ((did-something nil)) (do-blocks-backwards (block component) @@ -579,13 +596,13 @@ (dolist (b (block-succ block)) (when (and (block-start b) - (propagate-live-tns last (block-info b))) + (propagate-live-tns last (block-info b) (< i 10))) (setq did-something t))) (do ((b (ir2-block-prev last) (ir2-block-prev b)) (prev last b)) ((not (eq (ir2-block-block b) block))) - (when (propagate-live-tns b prev) + (when (propagate-live-tns b prev (< i 10)) (setq did-something t))))) (unless did-something (return)))) diff --git a/version.lisp-expr b/version.lisp-expr index 49f58c1..e21ac35 100644 --- a/version.lisp-expr +++ b/version.lisp-expr @@ -20,4 +20,4 @@ ;;; checkins which aren't released. (And occasionally for internal ;;; versions, especially for internal versions off the main CVS ;;; branch, it gets hairier, e.g. "0.pre7.14.flaky4.13".) -"1.0.45.5" +"1.0.45.6" -- 1.7.10.4