1 ;;; Guile VM frame functions
3 ;;; Copyright (C) 2001 Free Software Foundation, Inc.
4 ;;; Copyright (C) 2005 Ludovic Courtès <ludovic.courtes@laas.fr>
6 ;;; This program is free software; you can redistribute it and/or modify
7 ;;; it under the terms of the GNU General Public License as published by
8 ;;; the Free Software Foundation; either version 2 of the License, or
9 ;;; (at your option) any later version.
11 ;;; This program is distributed in the hope that it will be useful,
12 ;;; but WITHOUT ANY WARRANTY; without even the implied warranty of
13 ;;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 ;;; GNU General Public License for more details.
16 ;;; You should have received a copy of the GNU General Public License
17 ;;; along with this program; if not, write to the Free Software
18 ;;; Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
22 (define-module (system vm frame)
23 :use-module (system vm program)
24 :use-module ((srfi srfi-1) :select (fold))
25 :export (frame-number frame-address
27 print-frame print-frame-chain-as-backtrace
28 frame-arguments frame-local-variables frame-external-variables
30 frame-variable-exists? frame-variable-ref frame-variable-set!
32 frame-local-ref frame-external-link frame-local-set!
33 frame-return-address frame-program
34 frame-dynamic-link frame?))
36 (dynamic-call "scm_init_frames" (dynamic-link "libguile-vm"))
42 (define frame-number (make-object-property))
43 (define frame-address (make-object-property))
45 (define (make-frame-chain frame addr)
46 (let* ((link (frame-dynamic-link frame))
51 link (frame-return-address frame))))))
52 (set! (frame-number frame) (1- (length chain)))
53 (set! (frame-address frame)
54 (- addr (program-base (frame-program frame))))
62 (define (frame-line-number frame)
63 (let ((addr (frame-address frame)))
64 (cond ((assv-ref (program-sources (frame-program frame)) addr)
66 (else (format #f "@~a" addr)))))
68 (define (frame-file frame prev)
69 (let ((sources (program-sources (frame-program frame))))
72 (or (source:file (car sources))
75 (define (print-frame frame)
76 (format #t "~4@a: ~a ~a\n" (frame-line-number frame) (frame-number frame)
77 (frame-call-representation frame)))
80 (define (frame-call-representation frame)
84 (list (abbrev (car x)) (abbrev (cadr x)) '...)
87 (cons (abbrev (car x)) (abbrev (cdr x))))
89 (case (vector-length x)
91 ((1) (vector (abbrev (vector-ref x 0))))
92 (else (vector (abbrev (vector-ref x 0)) '...))))
94 (abbrev (cons (program-name frame) (frame-arguments frame))))
96 (define (print-frame-chain-as-backtrace frames)
98 (format #t "No backtrace available.\n")
100 (format #t "Backtrace:\n")
101 (pk frames (map frame-program frames)
102 (map frame-address frames)
104 (fold (lambda (frame file)
105 (let ((new-file (frame-file frame file)))
106 (if (not (eqv? new-file file))
107 (format #t "In ~a:\n" new-file))
113 (define (program-name frame)
114 (let ((prog (frame-program frame))
115 (link (frame-dynamic-link frame)))
116 (or (object-property prog 'name)
118 (frame-object-name link (1- (frame-address link)) prog))
119 (hash-fold (lambda (s v d) (if (eq? prog (variable-ref v)) s d))
120 prog (module-obarray (current-module))))))
127 (define (frame-arguments frame)
128 (let* ((prog (frame-program frame))
129 (arity (program-arity prog)))
130 (do ((n (+ (arity:nargs arity) -1) (1- n))
131 (l '() (cons (frame-local-ref frame n) l)))
134 (define (frame-local-variables frame)
135 (let* ((prog (frame-program frame))
136 (arity (program-arity prog)))
137 (do ((n (+ (arity:nargs arity) (arity:nlocs arity) -1) (1- n))
138 (l '() (cons (frame-local-ref frame n) l)))
141 (define (frame-external-variables frame)
142 (frame-external-link frame))
144 (define (frame-external-ref frame index)
145 (list-ref (frame-external-link frame) index))
147 (define (frame-external-set! frame index val)
148 (list-set! (frame-external-link frame) index val))
150 (define (frame-binding-ref frame binding)
151 (if (binding:extp binding)
152 (frame-external-ref frame (binding:index binding))
153 (frame-local-ref frame (binding:index binding))))
155 (define (frame-binding-set! frame binding val)
156 (if (binding:extp binding)
157 (frame-external-set! frame (binding:index binding) val)
158 (frame-local-set! frame (binding:index binding) val)))
160 (define (frame-bindings frame addr)
161 (do ((bs (program-bindings (frame-program frame)) (cdr bs))
162 (ls '() (if (cdar bs) (cons (cdar bs) ls) (cdr ls))))
163 ((or (null? bs) (> (caar bs) addr))
166 (define (frame-lookup-binding frame addr sym)
167 (do ((bs (frame-bindings frame addr) (cdr bs)))
168 ((or (null? bs) (eq? sym (binding:name (car bs))))
169 (and (pair? bs) (car bs)))))
171 (define (frame-object-binding frame addr obj)
172 (do ((bs (frame-bindings frame addr) (cdr bs)))
173 ((or (null? bs) (eq? obj (frame-binding-ref frame (car bs))))
174 (and (pair? bs) (car bs)))))
176 (define (frame-environment frame addr)
177 (map (lambda (binding)
178 (cons (binding:name binding) (frame-binding-ref frame binding)))
179 (frame-bindings frame addr)))
181 (define (frame-variable-exists? frame addr sym)
182 (if (frame-lookup-binding frame addr sym) #t #f))
184 (define (frame-variable-ref frame addr sym)
185 (cond ((frame-lookup-binding frame addr sym) =>
186 (lambda (binding) (frame-binding-ref frame binding)))
187 (else (error "Unknown variable:" sym))))
189 (define (frame-variable-set! frame addr sym val)
190 (cond ((frame-lookup-binding frame addr sym) =>
191 (lambda (binding) (frame-binding-set! frame binding val)))
192 (else (error "Unknown variable:" sym))))
194 (define (frame-object-name frame addr obj)
195 (cond ((frame-object-binding frame addr obj) => binding:name)