123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333 |
- (define-module (ps-compiler prescheme merge)
- #:use-module (prescheme scheme48)
- #:use-module (ps-compiler node arch)
- #:use-module (ps-compiler node let-nodes)
- #:use-module (ps-compiler node node)
- #:use-module (ps-compiler node node-util)
- #:use-module (ps-compiler node primop)
- #:use-module (ps-compiler node variable)
- #:use-module (ps-compiler prescheme c-util)
- #:use-module (ps-compiler prescheme form)
- #:use-module (ps-compiler prescheme type)
- #:use-module (ps-compiler prescheme type-var)
- #:use-module (ps-compiler util strong)
- #:use-module (ps-compiler util syntax)
- #:export (form-head
- form-merged
- form-return-count
- set-form-return-count!
- form-tail-called?
- merged-procedure-reference
- merge-forms))
- (define-subrecord form form-merge form-merge
- ((head)
- )
- (
- (status #f)
- tail-clients
-
- (tail-providers '())
- (merged '())
- (return-count 0)
- temp
- ))
- (define (show-merges form)
- (let ((merges (form-merged form)))
- (if (not (null? merges))
- (format #t " ~S: ~S~%" (form-name form) (map form-name merges)))))
- (define (show-providers form)
- (cond ((eq? (form-type form) 'lambda)
- (format #t "~S ~A~%"
- (form-name form)
- (if (form-exported? form) " (exported)" ""))
- (cond ((or (not (null? (form-providers form)))
- (not (null? (form-tail-providers form))))
- (format #t " ~S~% ~S~%"
- (map form-name (form-providers form))
- (map form-name (form-tail-providers form))))))))
- (define (do-merge form others)
- (let ((form (form-head form))
- (secondary (apply append (map form-merged others))))
- (set-form-merged! form (append others
- secondary
- (form-merged form)))
- (for-each (lambda (f)
- (set-form-head! f form))
- secondary)
- (for-each (lambda (f)
- (set-form-head! f form)
- (set-form-status! f 'merged)
- (set-form-type! f 'merged)
- (set-form-merged! f '()))
- others)))
- (define (merged-procedure-reference node)
- (cond ((and (reference-node? node)
- (maybe-variable->form (reference-variable node)))
- => (lambda (form)
- (if (eq? 'merged (form-type form))
- form
- #f)))
- (else
- #f)))
- (define (form-tail-called? form)
- (and (or (eq? 'lambda (form-type form))
- (eq? 'merged (form-type form)))
- (memq? 'tail-called (variable-flags (form-var form)))))
- (define (note-tail-called-procedure form)
- (if (and (eq? 'lambda (form-type form))
- (or (any (lambda (r)
- (used-as-label? r))
- (variable-refs (form-var form)))
- (eq? 'tail-called (lambda-protocol (form-value form)))))
- (set-variable-flags! (form-var form)
- (cons 'tail-called
- (variable-flags (form-var form))))))
- (define (used-as-label? node)
- (and (node? (node-parent node))
- (goto-call? (node-parent node))
- (= 1 (node-index node))))
- (define *merge-forms?* #t)
- (define (merge-forms forms)
- (for-each (lambda (f)
- (note-tail-called-procedure f)
- (set-form-merge! f (form-merge-maker f))
- (set-form-providers! f '()))
- forms)
- (if *merge-forms?*
- (let ((mergable-forms (filter determine-merger-graph forms)))
- (format #t "Call Graph:~%<procedure name>~%")
- (format #t " <called non-tail-recursively>~%")
- (format #t " <called tail-recursively>~%")
- (for-each show-providers forms)
- (format #t "Merging forms~%")
- (receive (tail other)
- (partition-list (lambda (f) (null? (form-clients f)))
- mergable-forms)
- (merge-tail-forms tail)
- (for-each merge-non-tail-forms forms)
- (for-each show-merges forms)
- (values)))))
- (define (determine-merger-graph form)
- (cond ((and (eq? 'lambda (form-type form))
- (really-determine-merger-graph form)
- (not (form-exported? form))
- (or (null? (form-clients form))
- (null? (form-tail-clients form))))
- #t)
- (else
- (set-form-status! form 'do-not-merge)
- #f)))
- (define (really-determine-merger-graph form)
- (let loop ((refs (variable-refs (form-var form)))
- (clients '()) (tail-clients '()) (okay? #t))
- (cond ((null? refs)
- (set-form-clients! form clients)
- (set-form-tail-clients! form tail-clients)
- okay?)
- (else
- (let* ((r (car refs))
- (f (node-form (car refs))))
- (if (and (called-node? r)
- (or (calls-this-primop? (node-parent r) 'tail-call)
- (calls-this-primop? (node-parent r) 'unknown-tail-call)))
- (loop (cdr refs)
- clients
- (add-to-client-list tail-clients r form f
- form-tail-providers
- set-form-tail-providers!)
- okay?)
- (loop (cdr refs)
- (add-to-client-list clients r form f
- form-providers
- set-form-providers!)
- tail-clients
- (and okay? (called-node? r)))))))))
- (define (add-to-client-list client-list ref form f getter setter)
- (cond ((assq f client-list)
- => (lambda (p)
- (set-cdr! p (cons ref (cdr p)))
- client-list))
- (else
- (setter f (cons form (getter f)))
- (cons (list f ref) client-list))))
- (define (merge-tail-forms forms)
- (for-each merge-tail-loop
- (reverse (strongly-connected-components
- forms
- (lambda (f)
- (filter (lambda (f) (memq? f forms))
- (map car (form-tail-clients f))))
- form-temp
- set-form-temp!))))
- (define (merge-tail-loop forms)
- (for-each (lambda (f) (set-form-temp! f forms)) forms)
- (receive (entries other)
- (partition-list (lambda (f)
- (any? (lambda (p)
- (not (eq? forms
- (form-temp (car p)))))
- (form-tail-clients f)))
- forms)
- (cond ((single-outside-client (if (null? entries)
- other
- entries)
- forms)
- => (lambda (f) (do-merge f forms)))
- ((and (not (null? entries))
- (null? (cdr entries))
- (not (null? other)))
- (do-merge (car entries) other)))
- (for-each (lambda (f) (set-form-temp! f #f)) forms)))
- (define (single-outside-client entries flag)
- (let loop ((entries entries) (form #f))
- (if (null? entries)
- form
- (let loop2 ((clients (form-tail-clients (car entries))) (form form))
- (cond ((null? clients)
- (loop (cdr entries) form))
- ((eq? (form-temp (caar clients)) flag)
- (loop2 (cdr clients) form))
- ((not form)
- (loop2 (cdr clients) (form-head (caar clients))))
- ((eq? (form-head (caar clients)) form)
- (loop2 (cdr clients) form))
- (else
- #f))))))
- (define (merge-non-tail-forms form)
- (for-each (lambda (f)
- (maybe-merge-non-tail-form f (form-head form)))
- (form-providers form)))
- (define (maybe-merge-non-tail-form form into)
- (cond ((and (not (eq? form into))
- (not (form-status form))
- (every? (lambda (p)
- (eq? (form-head (car p)) into))
- (form-clients form)))
- (do-merge into (list form))
- (for-each tail-call->call
- (variable-refs (form-var form)))
- (for-each tail-call->call
- (variable-refs (car (lambda-variables (form-node form)))))
- (for-each (lambda (f)
- (maybe-merge-non-tail-form f into))
- (form-providers form)))))
- (define (tail-call->call ref)
- (let ((this-call (node-parent ref)))
- (if (or (calls-this-primop? this-call 'tail-call)
- (calls-this-primop? this-call 'unknown-tail-call))
- (let ((type (arrow-type-result
- (maybe-follow-uvar (node-type (call-arg this-call 1))))))
- (move (call-arg this-call 0)
- (lambda (cont)
- (let-nodes ((new-cont ((v type)) (return 0 cont (* v))))
- new-cont)))
- (set-call-exits! this-call 1)
- (set-call-primop! this-call
- (get-primop (if (calls-this-primop? this-call 'tail-call)
- (enum primop-enum call)
- (enum primop-enum unknown-call))))))))
|