mirror of
https://git.savannah.gnu.org/git/guile.git
synced 2025-05-09 15:10:29 +02:00
* module/system/xref.scm (ensure-callers-db): Store the callers db as a doubly-weak hash table.
125 lines
4.9 KiB
Scheme
125 lines
4.9 KiB
Scheme
;;;; Copyright (C) 2009 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 2.1 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 (system xref)
|
||
#:use-module (system base pmatch)
|
||
#:use-module (system base compile)
|
||
#:use-module (system vm program)
|
||
#:export (procedure-callees procedure-callers *xref-ignored-modules*))
|
||
|
||
(define (program-callees prog)
|
||
(cond
|
||
((program-objects prog)
|
||
=> (lambda (objects)
|
||
(let ((n (vector-length objects))
|
||
(progv (make-vector (vector-length objects) #f))
|
||
(asm (decompile (program-objcode prog) #:to 'assembly)))
|
||
(pmatch asm
|
||
((load-program ,nargs ,nrest ,nlocs ,next ,labels ,len . ,body)
|
||
(for-each
|
||
(lambda (x)
|
||
(pmatch x
|
||
((toplevel-ref ,n) (vector-set! progv n #t))
|
||
((toplevel-set ,n) (vector-set! progv n #t))))
|
||
body)))
|
||
(let lp ((i 0) (out '()))
|
||
(cond
|
||
((= i n) (reverse out))
|
||
((program? (vector-ref objects i))
|
||
(lp (1+ i) (append (reverse (program-callees
|
||
(vector-ref objects i)))
|
||
out)))
|
||
((vector-ref progv i)
|
||
(let ((obj (vector-ref objects i)))
|
||
(if (variable? obj)
|
||
(lp (1+ i) (if (variable-bound? obj)
|
||
(cons (variable-ref obj) out)
|
||
out))
|
||
;; otherwise it's an unmemoized binding
|
||
(pmatch obj
|
||
(,sym (guard (symbol? sym))
|
||
(let ((v (module-variable (or (program-module prog)
|
||
the-root-module)
|
||
sym)))
|
||
(lp (1+ i)
|
||
(if (and v (variable-bound? v))
|
||
(cons (variable-ref v) out)
|
||
out))))
|
||
((,mod ,sym ,public?)
|
||
;; hm, hacky.
|
||
(let ((m (nested-ref the-root-module
|
||
(append '(%app modules) mod))))
|
||
(let ((v (and m (module-variable
|
||
(if public? (module-public-interface m) m)
|
||
sym))))
|
||
(lp (1+ i)
|
||
(if (and v (variable-bound? v))
|
||
(cons (variable-ref v) out)
|
||
out)))))))))
|
||
(else (lp (1+ i) out)))))))
|
||
(else '())))
|
||
|
||
(define (hacky-procedure-callees proc)
|
||
;; we could analyze the memoized source or something
|
||
'())
|
||
|
||
(define (procedure-callees proc)
|
||
(cond
|
||
((program? proc) (program-callees proc))
|
||
((procedure-source proc) (hacky-procedure-callees proc))
|
||
(else '())))
|
||
|
||
(define *callers-db* #f)
|
||
|
||
(define *xref-ignored-modules* '((value-history)))
|
||
(define (on-module-modified m)
|
||
(if (not (member (module-name m) *xref-ignored-modules*))
|
||
(set! *callers-db* #f)))
|
||
|
||
(define (ensure-callers-db)
|
||
(let ((visited '())
|
||
(db #f))
|
||
(define (visit-procedure proc)
|
||
(for-each
|
||
(lambda (x)
|
||
(hashq-set! db x (cons proc (hashq-ref db x '()))))
|
||
(procedure-callees proc)))
|
||
|
||
(define (visit-module mod)
|
||
(set! visited (cons mod visited))
|
||
(if (not (memq on-module-modified (module-observers mod)))
|
||
(module-observe mod on-module-modified))
|
||
(module-for-each
|
||
(lambda (sym var)
|
||
(if (variable-bound? var)
|
||
(let ((x (variable-ref var)))
|
||
(cond
|
||
((procedure? x) (visit-procedure x))
|
||
((module? x) (if (not (memq x visited))
|
||
(visit-module x)))))))
|
||
mod))
|
||
|
||
(if (not *callers-db*)
|
||
(begin
|
||
(set! db (make-doubly-weak-hash-table))
|
||
(visit-module the-root-module)
|
||
(set! *callers-db* db)))))
|
||
|
||
(define (procedure-callers proc)
|
||
(ensure-callers-db)
|
||
(hashq-ref *callers-db* proc #f))
|