105241
|
1 ;;; data-debug.el --- Datastructure Debugger
|
|
2
|
105325
|
3 ;; Copyright (C) 2007, 2008, 2009 Free Software Foundation, Inc.
|
105241
|
4
|
|
5 ;; Author: Eric M. Ludlam <zappo@gnu.org>
|
|
6 ;; Version: 0.2
|
|
7 ;; Keywords: OO, lisp
|
|
8
|
|
9 ;; This file is part of GNU Emacs.
|
|
10
|
|
11 ;; GNU Emacs is free software: you can redistribute it and/or modify
|
|
12 ;; it under the terms of the GNU General Public License as published by
|
|
13 ;; the Free Software Foundation, either version 3 of the License, or
|
|
14 ;; (at your option) any later version.
|
|
15
|
|
16 ;; GNU Emacs is distributed in the hope that it will be useful,
|
|
17 ;; but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
18 ;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
19 ;; GNU General Public License for more details.
|
|
20
|
|
21 ;; You should have received a copy of the GNU General Public License
|
|
22 ;; along with GNU Emacs. If not, see <http://www.gnu.org/licenses/>.
|
|
23
|
|
24 ;;; Commentary:
|
|
25 ;;
|
|
26 ;; Provide a simple way to investigate particularly large and complex
|
|
27 ;; data structures.
|
|
28 ;;
|
|
29 ;; The best way to get started is to bind M-: to 'data-debug-eval-expression.
|
|
30 ;;
|
|
31 ;; (global-set-key "\M-:" 'data-debug-eval-expression)
|
|
32 ;;
|
|
33 ;; If you write functions with complex output that need debugging, you
|
|
34 ;; can make them interactive with data-debug-show-stuff. For example:
|
|
35 ;;
|
|
36 ;; (defun my-complex-output-fcn ()
|
|
37 ;; "Calculate something complicated at point, and return it."
|
|
38 ;; (interactive) ;; function not normally interactive
|
|
39 ;; (let ((stuff (do-stuff)))
|
|
40 ;; (when (interactive-p)
|
|
41 ;; (data-debug-show-stuff stuff "myStuff"))
|
|
42 ;; stuff))
|
|
43
|
|
44 (require 'font-lock)
|
|
45 (require 'ring)
|
|
46
|
|
47 ;;; Code:
|
|
48
|
|
49 ;;; Compatibility
|
|
50 ;;
|
|
51 (if (featurep 'xemacs)
|
|
52 (eval-and-compile
|
|
53 (defalias 'data-debug-overlay-properties 'extent-properties)
|
|
54 (defalias 'data-debug-overlay-p 'extentp)
|
|
55 (if (not (fboundp 'propertize))
|
|
56 (defun dd-propertize (string &rest properties)
|
|
57 "Mimic 'propertize' in from Emacs 23."
|
|
58 (add-text-properties 0 (length string) properties string)
|
|
59 string
|
|
60 )
|
|
61 (defalias 'dd-propertize 'propertize))
|
|
62 )
|
|
63 ;; Regular Emacs
|
|
64 (eval-and-compile
|
|
65 (defalias 'data-debug-overlay-properties 'overlay-properties)
|
|
66 (defalias 'data-debug-overlay-p 'overlayp)
|
|
67 (defalias 'dd-propertize 'propertize)
|
|
68 )
|
|
69 )
|
|
70
|
|
71 ;;; GENERIC STUFF
|
|
72 ;;
|
|
73 (defun data-debug-insert-property-list (proplist prefix &optional parent)
|
|
74 "Insert the property list PROPLIST.
|
|
75 Each line starts with PREFIX.
|
|
76 The attributes belong to the tag PARENT."
|
|
77 (while proplist
|
|
78 (let ((pretext (concat (symbol-name (car proplist)) " : ")))
|
|
79 (data-debug-insert-thing (car (cdr proplist))
|
|
80 prefix
|
|
81 pretext
|
|
82 parent))
|
|
83 (setq proplist (cdr (cdr proplist)))))
|
|
84
|
|
85 ;;; overlays
|
|
86 ;;
|
|
87 (defun data-debug-insert-overlay-props (overlay prefix)
|
|
88 "Insert all the parts of OVERLAY.
|
|
89 PREFIX specifies what to insert at the start of each line."
|
|
90 (let ((attrprefix (concat (make-string (length prefix) ? ) "# "))
|
|
91 (proplist (data-debug-overlay-properties overlay)))
|
|
92 (data-debug-insert-property-list
|
|
93 proplist attrprefix)
|
|
94 )
|
|
95 )
|
|
96
|
|
97 (defun data-debug-insert-overlay-from-point (point)
|
|
98 "Insert the overlay found at the overlay button at POINT."
|
|
99 (let ((overlay (get-text-property point 'ddebug))
|
|
100 (indent (get-text-property point 'ddebug-indent))
|
|
101 start
|
|
102 )
|
|
103 (end-of-line)
|
|
104 (setq start (point))
|
|
105 (forward-char 1)
|
|
106 (data-debug-insert-overlay-props overlay
|
|
107 (concat (make-string indent ? )
|
|
108 "| "))
|
|
109 (goto-char start)
|
|
110 ))
|
|
111
|
|
112 (defun data-debug-insert-overlay-button (overlay prefix prebuttontext)
|
|
113 "Insert a button representing OVERLAY.
|
105325
|
114 PREFIX is the text that precedes the button.
|
105241
|
115 PREBUTTONTEXT is some text between prefix and the overlay button."
|
|
116 (let ((start (point))
|
|
117 (end nil)
|
|
118 (str (format "%s" overlay))
|
|
119 (tip nil))
|
|
120 (insert prefix prebuttontext str)
|
|
121 (setq end (point))
|
|
122 (put-text-property (- end (length str)) end 'face 'font-lock-comment-face)
|
|
123 (put-text-property start end 'ddebug overlay)
|
|
124 (put-text-property start end 'ddebug-indent(length prefix))
|
|
125 (put-text-property start end 'ddebug-prefix prefix)
|
|
126 (put-text-property start end 'help-echo tip)
|
|
127 (put-text-property start end 'ddebug-function
|
|
128 'data-debug-insert-overlay-from-point)
|
|
129 (insert "\n")
|
|
130 )
|
|
131 )
|
|
132
|
|
133 ;;; overlay list
|
|
134 ;;
|
|
135 (defun data-debug-insert-overlay-list (overlaylist prefix)
|
|
136 "Insert all the parts of OVERLAYLIST.
|
|
137 PREFIX specifies what to insert at the start of each line."
|
|
138 (while overlaylist
|
|
139 (data-debug-insert-overlay-button (car overlaylist)
|
|
140 prefix
|
|
141 "")
|
|
142 (setq overlaylist (cdr overlaylist))))
|
|
143
|
|
144 (defun data-debug-insert-overlay-list-from-point (point)
|
|
145 "Insert the overlay found at the overlay list button at POINT."
|
|
146 (let ((overlaylist (get-text-property point 'ddebug))
|
|
147 (indent (get-text-property point 'ddebug-indent))
|
|
148 start
|
|
149 )
|
|
150 (end-of-line)
|
|
151 (setq start (point))
|
|
152 (forward-char 1)
|
|
153 (data-debug-insert-overlay-list overlaylist
|
|
154 (concat (make-string indent ? )
|
|
155 "* "))
|
|
156 (goto-char start)
|
|
157 ))
|
|
158
|
|
159 (defun data-debug-insert-overlay-list-button (overlaylist
|
|
160 prefix
|
|
161 prebuttontext)
|
|
162 "Insert a button representing OVERLAYLIST.
|
105325
|
163 PREFIX is the text that precedes the button.
|
105241
|
164 PREBUTTONTEXT is some text between prefix and the overlay list button."
|
|
165 (let ((start (point))
|
|
166 (end nil)
|
|
167 (str (format "#<overlay list: %d entries>" (length overlaylist)))
|
|
168 (tip nil))
|
|
169 (insert prefix prebuttontext str)
|
|
170 (setq end (point))
|
|
171 (put-text-property (- end (length str)) end 'face 'font-lock-comment-face)
|
|
172 (put-text-property start end 'ddebug overlaylist)
|
|
173 (put-text-property start end 'ddebug-indent(length prefix))
|
|
174 (put-text-property start end 'ddebug-prefix prefix)
|
|
175 (put-text-property start end 'help-echo tip)
|
|
176 (put-text-property start end 'ddebug-function
|
|
177 'data-debug-insert-overlay-list-from-point)
|
|
178 (insert "\n")
|
|
179 )
|
|
180 )
|
|
181
|
|
182 ;;; buffers
|
|
183 ;;
|
|
184 (defun data-debug-insert-buffer-props (buffer prefix)
|
|
185 "Insert all the parts of BUFFER.
|
|
186 PREFIX specifies what to insert at the start of each line."
|
|
187 (let ((attrprefix (concat (make-string (length prefix) ? ) "# "))
|
|
188 (proplist
|
|
189 (list :filename (buffer-file-name buffer)
|
|
190 :live (buffer-live-p buffer)
|
|
191 :modified (buffer-modified-p buffer)
|
|
192 :size (buffer-size buffer)
|
|
193 :process (get-buffer-process buffer)
|
|
194 :localvars (buffer-local-variables buffer)
|
|
195 )))
|
|
196 (data-debug-insert-property-list
|
|
197 proplist attrprefix)
|
|
198 )
|
|
199 )
|
|
200
|
|
201 (defun data-debug-insert-buffer-from-point (point)
|
|
202 "Insert the buffer found at the buffer button at POINT."
|
|
203 (let ((buffer (get-text-property point 'ddebug))
|
|
204 (indent (get-text-property point 'ddebug-indent))
|
|
205 start
|
|
206 )
|
|
207 (end-of-line)
|
|
208 (setq start (point))
|
|
209 (forward-char 1)
|
|
210 (data-debug-insert-buffer-props buffer
|
|
211 (concat (make-string indent ? )
|
|
212 "| "))
|
|
213 (goto-char start)
|
|
214 ))
|
|
215
|
|
216 (defun data-debug-insert-buffer-button (buffer prefix prebuttontext)
|
|
217 "Insert a button representing BUFFER.
|
105325
|
218 PREFIX is the text that precedes the button.
|
105241
|
219 PREBUTTONTEXT is some text between prefix and the buffer button."
|
|
220 (let ((start (point))
|
|
221 (end nil)
|
|
222 (str (format "%S" buffer))
|
|
223 (tip nil))
|
|
224 (insert prefix prebuttontext str)
|
|
225 (setq end (point))
|
|
226 (put-text-property (- end (length str)) end 'face 'font-lock-comment-face)
|
|
227 (put-text-property start end 'ddebug buffer)
|
|
228 (put-text-property start end 'ddebug-indent(length prefix))
|
|
229 (put-text-property start end 'ddebug-prefix prefix)
|
|
230 (put-text-property start end 'help-echo tip)
|
|
231 (put-text-property start end 'ddebug-function
|
|
232 'data-debug-insert-buffer-from-point)
|
|
233 (insert "\n")
|
|
234 )
|
|
235 )
|
|
236
|
|
237 ;;; buffer list
|
|
238 ;;
|
|
239 (defun data-debug-insert-buffer-list (bufferlist prefix)
|
|
240 "Insert all the parts of BUFFERLIST.
|
|
241 PREFIX specifies what to insert at the start of each line."
|
|
242 (while bufferlist
|
|
243 (data-debug-insert-buffer-button (car bufferlist)
|
|
244 prefix
|
|
245 "")
|
|
246 (setq bufferlist (cdr bufferlist))))
|
|
247
|
|
248 (defun data-debug-insert-buffer-list-from-point (point)
|
|
249 "Insert the buffer found at the buffer list button at POINT."
|
|
250 (let ((bufferlist (get-text-property point 'ddebug))
|
|
251 (indent (get-text-property point 'ddebug-indent))
|
|
252 start
|
|
253 )
|
|
254 (end-of-line)
|
|
255 (setq start (point))
|
|
256 (forward-char 1)
|
|
257 (data-debug-insert-buffer-list bufferlist
|
|
258 (concat (make-string indent ? )
|
|
259 "* "))
|
|
260 (goto-char start)
|
|
261 ))
|
|
262
|
|
263 (defun data-debug-insert-buffer-list-button (bufferlist
|
|
264 prefix
|
|
265 prebuttontext)
|
|
266 "Insert a button representing BUFFERLIST.
|
105325
|
267 PREFIX is the text that precedes the button.
|
105241
|
268 PREBUTTONTEXT is some text between prefix and the buffer list button."
|
|
269 (let ((start (point))
|
|
270 (end nil)
|
|
271 (str (format "#<buffer list: %d entries>" (length bufferlist)))
|
|
272 (tip nil))
|
|
273 (insert prefix prebuttontext str)
|
|
274 (setq end (point))
|
|
275 (put-text-property (- end (length str)) end 'face 'font-lock-comment-face)
|
|
276 (put-text-property start end 'ddebug bufferlist)
|
|
277 (put-text-property start end 'ddebug-indent(length prefix))
|
|
278 (put-text-property start end 'ddebug-prefix prefix)
|
|
279 (put-text-property start end 'help-echo tip)
|
|
280 (put-text-property start end 'ddebug-function
|
|
281 'data-debug-insert-buffer-list-from-point)
|
|
282 (insert "\n")
|
|
283 )
|
|
284 )
|
|
285
|
|
286 ;;; processes
|
|
287 ;;
|
|
288 (defun data-debug-insert-process-props (process prefix)
|
|
289 "Insert all the parts of PROCESS.
|
|
290 PREFIX specifies what to insert at the start of each line."
|
|
291 (let ((attrprefix (concat (make-string (length prefix) ? ) "# "))
|
|
292 (id (process-id process))
|
|
293 (tty (process-tty-name process))
|
|
294 (pcontact (process-contact process t))
|
|
295 (proplist (process-plist process)))
|
|
296 (data-debug-insert-property-list
|
|
297 (append
|
|
298 (if id (list 'id id))
|
|
299 (if tty (list 'tty tty))
|
|
300 (if pcontact pcontact)
|
|
301 proplist)
|
|
302 attrprefix)
|
|
303 )
|
|
304 )
|
|
305
|
|
306 (defun data-debug-insert-process-from-point (point)
|
|
307 "Insert the process found at the process button at POINT."
|
|
308 (let ((process (get-text-property point 'ddebug))
|
|
309 (indent (get-text-property point 'ddebug-indent))
|
|
310 start
|
|
311 )
|
|
312 (end-of-line)
|
|
313 (setq start (point))
|
|
314 (forward-char 1)
|
|
315 (data-debug-insert-process-props process
|
|
316 (concat (make-string indent ? )
|
|
317 "| "))
|
|
318 (goto-char start)
|
|
319 ))
|
|
320
|
|
321 (defun data-debug-insert-process-button (process prefix prebuttontext)
|
|
322 "Insert a button representing PROCESS.
|
105325
|
323 PREFIX is the text that precedes the button.
|
105241
|
324 PREBUTTONTEXT is some text between prefix and the process button."
|
|
325 (let ((start (point))
|
|
326 (end nil)
|
|
327 (str (format "%S : %s" process (process-status process)))
|
|
328 (tip nil))
|
|
329 (insert prefix prebuttontext str)
|
|
330 (setq end (point))
|
|
331 (put-text-property (- end (length str)) end 'face 'font-lock-comment-face)
|
|
332 (put-text-property start end 'ddebug process)
|
|
333 (put-text-property start end 'ddebug-indent(length prefix))
|
|
334 (put-text-property start end 'ddebug-prefix prefix)
|
|
335 (put-text-property start end 'help-echo tip)
|
|
336 (put-text-property start end 'ddebug-function
|
|
337 'data-debug-insert-process-from-point)
|
|
338 (insert "\n")
|
|
339 )
|
|
340 )
|
|
341
|
|
342 ;;; Rings
|
|
343 ;;
|
|
344 ;; A ring (like kill-ring, or whatever.)
|
|
345 (defun data-debug-insert-ring-contents (ring prefix)
|
|
346 "Insert all the parts of RING.
|
|
347 PREFIX specifies what to insert at the start of each line."
|
|
348 (let ((len (ring-length ring))
|
|
349 (idx 0)
|
|
350 )
|
|
351 (while (< idx len)
|
|
352 (data-debug-insert-thing (ring-ref ring idx) prefix "")
|
|
353 (setq idx (1+ idx))
|
|
354 )))
|
|
355
|
|
356 (defun data-debug-insert-ring-items-from-point (point)
|
|
357 "Insert the ring found at the ring button at POINT."
|
|
358 (let ((ring (get-text-property point 'ddebug))
|
|
359 (indent (get-text-property point 'ddebug-indent))
|
|
360 start
|
|
361 )
|
|
362 (end-of-line)
|
|
363 (setq start (point))
|
|
364 (forward-char 1)
|
|
365 (data-debug-insert-ring-contents ring
|
|
366 (concat (make-string indent ? )
|
|
367 "} "))
|
|
368 (goto-char start)
|
|
369 ))
|
|
370
|
|
371 (defun data-debug-insert-ring-button (ring
|
|
372 prefix
|
|
373 prebuttontext)
|
|
374 "Insert a button representing RING.
|
105325
|
375 PREFIX is the text that precedes the button.
|
105241
|
376 PREBUTTONTEXT is some text between prefix and the stuff list button."
|
|
377 (let* ((start (point))
|
|
378 (end nil)
|
|
379 (str (format "#<RING: %d, %d max>"
|
|
380 (ring-length ring)
|
|
381 (ring-size ring)))
|
|
382 (ringthing
|
|
383 (if (= (ring-length ring) 0) nil (ring-ref ring 0)))
|
|
384 (tip (format "Ring max-size %d, length %d."
|
|
385 (ring-size ring)
|
|
386 (ring-length ring)))
|
|
387 )
|
|
388 (insert prefix prebuttontext str)
|
|
389 (setq end (point))
|
|
390 (put-text-property (- end (length str)) end 'face 'font-lock-type-face)
|
|
391 (put-text-property start end 'ddebug ring)
|
|
392 (put-text-property start end 'ddebug-indent(length prefix))
|
|
393 (put-text-property start end 'ddebug-prefix prefix)
|
|
394 (put-text-property start end 'help-echo tip)
|
|
395 (put-text-property start end 'ddebug-function
|
|
396 'data-debug-insert-ring-items-from-point)
|
|
397 (insert "\n")
|
|
398 )
|
|
399 )
|
|
400
|
|
401
|
|
402 ;;; Hash-table
|
|
403 ;;
|
|
404
|
|
405 (defun data-debug-insert-hash-table (hash-table prefix)
|
|
406 "Insert the contents of HASH-TABLE inserting PREFIX before each element."
|
|
407 (maphash
|
|
408 (lambda (key value)
|
|
409 (data-debug-insert-thing
|
|
410 key prefix
|
|
411 (dd-propertize "key " 'face font-lock-comment-face))
|
|
412 (data-debug-insert-thing
|
|
413 value prefix
|
|
414 (dd-propertize "val " 'face font-lock-comment-face)))
|
|
415 hash-table))
|
|
416
|
|
417 (defun data-debug-insert-hash-table-from-point (point)
|
|
418 "Insert the contents of the hash-table button at POINT."
|
|
419 (let ((hash-table (get-text-property point 'ddebug))
|
|
420 (indent (get-text-property point 'ddebug-indent))
|
|
421 start)
|
|
422 (end-of-line)
|
|
423 (setq start (point))
|
|
424 (forward-char 1)
|
|
425 (data-debug-insert-hash-table
|
|
426 hash-table
|
|
427 (concat (make-string indent ? ) "> "))
|
|
428 (goto-char start))
|
|
429 )
|
|
430
|
|
431 (defun data-debug-insert-hash-table-button (hash-table prefix prebuttontext)
|
|
432 "Insert HASH-TABLE as expandable button with recursive prefix PREFIX and PREBUTTONTEXT in front of the button text."
|
|
433 (let ((string (dd-propertize (format "%s" hash-table)
|
|
434 'face 'font-lock-keyword-face)))
|
|
435 (insert (dd-propertize
|
|
436 (concat prefix prebuttontext string)
|
|
437 'ddebug hash-table
|
|
438 'ddebug-indent (length prefix)
|
|
439 'ddebug-prefix prefix
|
|
440 'help-echo
|
|
441 (format "Hash-table\nTest: %s\nWeakness: %s\nElements: %d (of %d)"
|
|
442 (hash-table-test hash-table)
|
|
443 (if (hash-table-weakness hash-table) "yes" "no")
|
|
444 (hash-table-count hash-table)
|
|
445 (hash-table-size hash-table))
|
|
446 'ddebug-function
|
|
447 'data-debug-insert-hash-table-from-point)
|
|
448 "\n"))
|
|
449 )
|
|
450
|
|
451 ;;; Widget
|
|
452 ;;
|
|
453 ;; Widgets have a long list of properties
|
|
454 (defun data-debug-insert-widget-properties (widget prefix)
|
|
455 "Insert the contents of WIDGET inserting PREFIX before each element."
|
|
456 (let ((type (car widget))
|
|
457 (rest (cdr widget)))
|
|
458 (while rest
|
|
459 (data-debug-insert-thing (car (cdr rest))
|
|
460 prefix
|
|
461 (concat
|
|
462 (dd-propertize (format "%s" (car rest))
|
|
463 'face font-lock-comment-face)
|
|
464 " : "))
|
|
465 (setq rest (cdr (cdr rest))))
|
|
466 ))
|
|
467
|
|
468 (defun data-debug-insert-widget-from-point (point)
|
|
469 "Insert the contents of the widget button at POINT."
|
|
470 (let ((widget (get-text-property point 'ddebug))
|
|
471 (indent (get-text-property point 'ddebug-indent))
|
|
472 start)
|
|
473 (end-of-line)
|
|
474 (setq start (point))
|
|
475 (forward-char 1)
|
|
476 (data-debug-insert-widget-properties
|
|
477 widget (concat (make-string indent ? ) "# "))
|
|
478 (goto-char start))
|
|
479 )
|
|
480
|
|
481 (defun data-debug-insert-widget (widget prefix prebuttontext)
|
|
482 "Insert one WIDGET.
|
|
483 A Symbol is a simple thing, but this provides some face and prefix rules.
|
105325
|
484 PREFIX is the text that precedes the button.
|
105241
|
485 PREBUTTONTEXT is some text between prefix and the thing."
|
|
486 (let ((string (dd-propertize (format "#<WIDGET %s>" (car widget))
|
|
487 'face 'font-lock-keyword-face)))
|
|
488 (insert (dd-propertize
|
|
489 (concat prefix prebuttontext string)
|
|
490 'ddebug widget
|
|
491 'ddebug-indent (length prefix)
|
|
492 'ddebug-prefix prefix
|
|
493 'help-echo
|
|
494 (format "Widget\nType: %s\n# Properties: %d"
|
|
495 (car widget)
|
|
496 (/ (1- (length widget)) 2))
|
|
497 'ddebug-function
|
|
498 'data-debug-insert-widget-from-point)
|
|
499 "\n")))
|
|
500
|
|
501 ;;; list of stuff
|
|
502 ;;
|
|
503 ;; just a list. random stuff inside.
|
|
504 (defun data-debug-insert-stuff-list (stufflist prefix)
|
|
505 "Insert all the parts of STUFFLIST.
|
|
506 PREFIX specifies what to insert at the start of each line."
|
|
507 (while stufflist
|
|
508 (data-debug-insert-thing
|
|
509 ;; Some lists may put a value in the CDR
|
|
510 (if (listp stufflist) (car stufflist) stufflist)
|
|
511 prefix
|
|
512 "")
|
|
513 (setq stufflist
|
|
514 (if (listp stufflist)
|
|
515 (cdr-safe stufflist)
|
|
516 nil))))
|
|
517
|
|
518 (defun data-debug-insert-stuff-list-from-point (point)
|
|
519 "Insert the stuff found at the stuff list button at POINT."
|
|
520 (let ((stufflist (get-text-property point 'ddebug))
|
|
521 (indent (get-text-property point 'ddebug-indent))
|
|
522 start
|
|
523 )
|
|
524 (end-of-line)
|
|
525 (setq start (point))
|
|
526 (forward-char 1)
|
|
527 (data-debug-insert-stuff-list stufflist
|
|
528 (concat (make-string indent ? )
|
|
529 "> "))
|
|
530 (goto-char start)
|
|
531 ))
|
|
532
|
|
533 (defun data-debug-insert-stuff-list-button (stufflist
|
|
534 prefix
|
|
535 prebuttontext)
|
|
536 "Insert a button representing STUFFLIST.
|
105325
|
537 PREFIX is the text that precedes the button.
|
105241
|
538 PREBUTTONTEXT is some text between prefix and the stuff list button."
|
|
539 (let ((start (point))
|
|
540 (end nil)
|
|
541 (str
|
|
542 (condition-case nil
|
|
543 (format "#<list o' stuff: %d entries>" (safe-length stufflist))
|
|
544 (error "#<list o' stuff>")))
|
|
545 (tip (if (or (listp (car stufflist))
|
|
546 (vectorp (car stufflist)))
|
|
547 ""
|
|
548 (format "%s" stufflist))))
|
|
549 (insert prefix prebuttontext str)
|
|
550 (setq end (point))
|
|
551 (put-text-property (- end (length str)) end 'face 'font-lock-variable-name-face)
|
|
552 (put-text-property start end 'ddebug stufflist)
|
|
553 (put-text-property start end 'ddebug-indent (length prefix))
|
|
554 (put-text-property start end 'ddebug-prefix prefix)
|
|
555 (put-text-property start end 'help-echo tip)
|
|
556 (put-text-property start end 'ddebug-function
|
|
557 'data-debug-insert-stuff-list-from-point)
|
|
558 (insert "\n")
|
|
559 )
|
|
560 )
|
|
561
|
|
562 ;;; vector of stuff
|
|
563 ;;
|
|
564 ;; just a vector. random stuff inside.
|
|
565 (defun data-debug-insert-stuff-vector (stuffvector prefix)
|
|
566 "Insert all the parts of STUFFVECTOR.
|
|
567 PREFIX specifies what to insert at the start of each line."
|
|
568 (let ((idx 0))
|
|
569 (while (< idx (length stuffvector))
|
|
570 (data-debug-insert-thing
|
|
571 ;; Some vectors may put a value in the CDR
|
|
572 (aref stuffvector idx)
|
|
573 prefix
|
|
574 "")
|
|
575 (setq idx (1+ idx)))))
|
|
576
|
|
577 (defun data-debug-insert-stuff-vector-from-point (point)
|
|
578 "Insert the stuff found at the stuff vector button at POINT."
|
|
579 (let ((stuffvector (get-text-property point 'ddebug))
|
|
580 (indent (get-text-property point 'ddebug-indent))
|
|
581 start
|
|
582 )
|
|
583 (end-of-line)
|
|
584 (setq start (point))
|
|
585 (forward-char 1)
|
|
586 (data-debug-insert-stuff-vector stuffvector
|
|
587 (concat (make-string indent ? )
|
|
588 "[ "))
|
|
589 (goto-char start)
|
|
590 ))
|
|
591
|
|
592 (defun data-debug-insert-stuff-vector-button (stuffvector
|
|
593 prefix
|
|
594 prebuttontext)
|
|
595 "Insert a button representing STUFFVECTOR.
|
105325
|
596 PREFIX is the text that precedes the button.
|
105241
|
597 PREBUTTONTEXT is some text between prefix and the stuff vector button."
|
|
598 (let* ((start (point))
|
|
599 (end nil)
|
|
600 (str (format "#<vector o' stuff: %d entries>" (length stuffvector)))
|
|
601 (tip str))
|
|
602 (insert prefix prebuttontext str)
|
|
603 (setq end (point))
|
|
604 (put-text-property (- end (length str)) end 'face 'font-lock-variable-name-face)
|
|
605 (put-text-property start end 'ddebug stuffvector)
|
|
606 (put-text-property start end 'ddebug-indent (length prefix))
|
|
607 (put-text-property start end 'ddebug-prefix prefix)
|
|
608 (put-text-property start end 'help-echo tip)
|
|
609 (put-text-property start end 'ddebug-function
|
|
610 'data-debug-insert-stuff-vector-from-point)
|
|
611 (insert "\n")
|
|
612 )
|
|
613 )
|
|
614
|
|
615 ;;; Symbol
|
|
616 ;;
|
|
617
|
|
618 (defun data-debug-insert-symbol-from-point (point)
|
|
619 "Insert attached properties and possibly the value of symbol at POINT."
|
|
620 (let ((symbol (get-text-property point 'ddebug))
|
|
621 (indent (get-text-property point 'ddebug-indent))
|
|
622 start)
|
|
623 (end-of-line)
|
|
624 (setq start (point))
|
|
625 (forward-char 1)
|
|
626 (when (and (not (fboundp symbol)) (boundp symbol))
|
|
627 (data-debug-insert-thing
|
|
628 (symbol-value symbol)
|
|
629 (concat (make-string indent ? ) "> ")
|
|
630 (concat
|
|
631 (dd-propertize "value"
|
|
632 'face 'font-lock-comment-face)
|
|
633 " ")))
|
|
634 (data-debug-insert-property-list
|
|
635 (symbol-plist symbol)
|
|
636 (concat (make-string indent ? ) "> "))
|
|
637 (goto-char start))
|
|
638 )
|
|
639
|
|
640 (defun data-debug-insert-symbol-button (symbol prefix prebuttontext)
|
|
641 "Insert a button representing SYMBOL.
|
105325
|
642 PREFIX is the text that precedes the button.
|
|
643 PREBUTTONTEXT is some text between prefix and the symbol button."
|
105241
|
644 (let ((string
|
|
645 (cond ((fboundp symbol)
|
|
646 (dd-propertize (concat "#'" (symbol-name symbol))
|
|
647 'face 'font-lock-function-name-face))
|
|
648 ((boundp symbol)
|
|
649 (dd-propertize (concat "'" (symbol-name symbol))
|
|
650 'face 'font-lock-variable-name-face))
|
|
651 (t (format "'%s" symbol)))))
|
|
652 (insert (dd-propertize
|
|
653 (concat prefix prebuttontext string)
|
|
654 'ddebug symbol
|
|
655 'ddebug-indent (length prefix)
|
|
656 'ddebug-prefix prefix
|
|
657 'help-echo ""
|
|
658 'ddebug-function
|
|
659 'data-debug-insert-symbol-from-point)
|
|
660 "\n"))
|
|
661 )
|
|
662
|
|
663 ;;; String
|
|
664 (defun data-debug-insert-string (thing prefix prebuttontext)
|
|
665 "Insert one symbol THING.
|
|
666 A Symbol is a simple thing, but this provides some face and prefix rules.
|
105325
|
667 PREFIX is the text that precedes the button.
|
105241
|
668 PREBUTTONTEXT is some text between prefix and the thing."
|
|
669 (let ((newstr thing))
|
|
670 (while (string-match "\n" newstr)
|
|
671 (setq newstr (replace-match "\\n" t t newstr)))
|
|
672 (while (string-match "\t" newstr)
|
|
673 (setq newstr (replace-match "\\t" t t newstr)))
|
|
674 (insert prefix prebuttontext
|
|
675 (dd-propertize (format "\"%s\"" newstr)
|
|
676 'face font-lock-string-face)
|
|
677 "\n" )))
|
|
678
|
|
679 ;;; Number
|
|
680 (defun data-debug-insert-number (thing prefix prebuttontext)
|
|
681 "Insert one symbol THING.
|
|
682 A Symbol is a simple thing, but this provides some face and prefix rules.
|
105325
|
683 PREFIX is the text that precedes the button.
|
105241
|
684 PREBUTTONTEXT is some text between prefix and the thing."
|
|
685 (insert prefix prebuttontext
|
|
686 (dd-propertize (format "%S" thing)
|
|
687 'face font-lock-string-face)
|
|
688 "\n"))
|
|
689
|
|
690 ;;; Lambda Expression
|
|
691 (defun data-debug-insert-lambda-expression (thing prefix prebuttontext)
|
|
692 "Insert one lambda expression THING.
|
|
693 A Symbol is a simple thing, but this provides some face and prefix rules.
|
105325
|
694 PREFIX is the text that precedes the button.
|
105241
|
695 PREBUTTONTEXT is some text between prefix and the thing."
|
|
696 (let ((txt (prin1-to-string thing)))
|
|
697 (data-debug-insert-simple-thing
|
|
698 txt prefix prebuttontext 'font-lock-keyword-face))
|
|
699 )
|
|
700
|
|
701 ;;; nil thing
|
|
702 (defun data-debug-insert-nil (thing prefix prebuttontext)
|
|
703 "Insert one simple THING with a face.
|
105325
|
704 PREFIX is the text that precedes the button.
|
105241
|
705 PREBUTTONTEXT is some text between prefix and the thing.
|
|
706 FACE is the face to use."
|
|
707 (insert prefix prebuttontext)
|
|
708 (insert ": ")
|
|
709 (let ((start (point))
|
|
710 (end nil))
|
|
711 (insert "nil")
|
|
712 (setq end (point))
|
|
713 (insert "\n" )
|
|
714 (put-text-property start end 'face 'font-lock-variable-name-face)
|
|
715 ))
|
|
716
|
|
717 ;;; simple thing
|
|
718 (defun data-debug-insert-simple-thing (thing prefix prebuttontext face)
|
|
719 "Insert one simple THING with a face.
|
105325
|
720 PREFIX is the text that precedes the button.
|
105241
|
721 PREBUTTONTEXT is some text between prefix and the thing.
|
|
722 FACE is the face to use."
|
|
723 (insert prefix prebuttontext)
|
|
724 (let ((start (point))
|
|
725 (end nil))
|
|
726 (insert (format "%s" thing))
|
|
727 (setq end (point))
|
|
728 (insert "\n" )
|
|
729 (put-text-property start end 'face face)
|
|
730 ))
|
|
731
|
|
732 ;;; custom thing
|
|
733 (defun data-debug-insert-custom (thingstring prefix prebuttontext face)
|
|
734 "Insert one simple THINGSTRING with a face.
|
|
735 Use for simple items that need a custom insert.
|
105325
|
736 PREFIX is the text that precedes the button.
|
105241
|
737 PREBUTTONTEXT is some text between prefix and the thing.
|
|
738 FACE is the face to use."
|
|
739 (insert prefix prebuttontext)
|
|
740 (let ((start (point))
|
|
741 (end nil))
|
|
742 (insert thingstring)
|
|
743 (setq end (point))
|
|
744 (insert "\n" )
|
|
745 (put-text-property start end 'face face)
|
|
746 ))
|
|
747
|
|
748
|
|
749 (defvar data-debug-thing-alist
|
|
750 '(
|
|
751 ;; nil
|
|
752 (null . data-debug-insert-nil)
|
|
753
|
|
754 ;; Overlay
|
|
755 (data-debug-overlay-p . data-debug-insert-overlay-button)
|
|
756
|
|
757 ;; Overlay list
|
|
758 ((lambda (thing) (and (consp thing) (data-debug-overlay-p (car thing)))) .
|
|
759 data-debug-insert-overlay-list-button)
|
|
760
|
|
761 ;; Buffer
|
|
762 (bufferp . data-debug-insert-buffer-button)
|
|
763
|
|
764 ;; Buffer list
|
|
765 ((lambda (thing) (and (consp thing) (bufferp (car thing)))) .
|
|
766 data-debug-insert-buffer-list-button)
|
|
767
|
|
768 ;; Process
|
|
769 (processp . data-debug-insert-process-button)
|
|
770
|
|
771 ;; String
|
|
772 (stringp . data-debug-insert-string)
|
|
773
|
|
774 ;; Number
|
|
775 (numberp . data-debug-insert-number)
|
|
776
|
|
777 ;; Symbol
|
|
778 (symbolp . data-debug-insert-symbol-button)
|
|
779
|
|
780 ;; Ring
|
|
781 (ring-p . data-debug-insert-ring-button)
|
|
782
|
|
783 ;; Lambda Expression
|
|
784 ((lambda (thing) (and (consp thing) (eq (car thing) 'lambda))) .
|
|
785 data-debug-insert-lambda-expression)
|
|
786
|
|
787 ;; Hash-table
|
|
788 (hash-table-p . data-debug-insert-hash-table-button)
|
|
789
|
|
790 ;; Widgets
|
|
791 (widgetp . data-debug-insert-widget)
|
|
792
|
|
793 ;; List of stuff
|
|
794 (listp . data-debug-insert-stuff-list-button)
|
|
795
|
|
796 ;; Vector of stuff
|
|
797 (vectorp . data-debug-insert-stuff-vector-button)
|
|
798 )
|
|
799 "Alist of methods used to insert things into an Ddebug buffer.")
|
|
800
|
|
801 ;; An augmentation function for the thing alist.
|
|
802 (defun data-debug-add-specialized-thing (predicate fcn)
|
|
803 "Add a new specialized thing to display with data-debug.
|
|
804 PREDICATE is a function that returns t if a thing is this new type.
|
|
805 FCN is a function that will display stuff in the data debug buffer."
|
|
806 (let ((entry (cons predicate fcn))
|
|
807 ;; Specialized entries show up AFTER nil,
|
|
808 ;; but before listp, vectorp, symbolp, and
|
|
809 ;; other general things. Splice it into
|
|
810 ;; the beginning.
|
|
811 (first (nthcdr 0 data-debug-thing-alist))
|
|
812 (second (nthcdr 1 data-debug-thing-alist))
|
|
813 )
|
|
814 (when (not (member entry data-debug-thing-alist))
|
|
815 (setcdr first (cons entry second)))))
|
|
816
|
|
817 ;; uber insert method
|
|
818 (defun data-debug-insert-thing (thing prefix prebuttontext &optional parent)
|
|
819 "Insert THING with PREFIX.
|
|
820 PREBUTTONTEXT is some text to insert between prefix and the thing
|
|
821 that is not included in the indentation calculation of any children.
|
|
822 If PARENT is non-nil, it is somehow related as a parent to thing."
|
|
823 (when (catch 'done
|
|
824 (dolist (test data-debug-thing-alist)
|
|
825 (when (funcall (car test) thing)
|
|
826 (condition-case nil
|
|
827 (funcall (cdr test) thing prefix prebuttontext parent)
|
|
828 (error
|
|
829 (funcall (cdr test) thing prefix prebuttontext)))
|
|
830 (throw 'done nil))
|
|
831 )
|
|
832 nil)
|
|
833 (data-debug-insert-simple-thing (format "%S" thing)
|
|
834 prefix
|
|
835 prebuttontext
|
|
836 'bold)))
|
|
837
|
|
838 ;;; MAJOR MODE
|
|
839 ;;
|
|
840 ;; The Ddebug major mode provides an interactive space to explore
|
|
841 ;; complicated data structures.
|
|
842 ;;
|
|
843 (defgroup data-debug nil
|
|
844 "data-debug group."
|
|
845 :group 'langauges)
|
|
846
|
|
847 (defvar data-debug-mode-syntax-table
|
|
848 (let ((table (make-syntax-table (standard-syntax-table))))
|
|
849 (modify-syntax-entry ?\; ". 12" table) ;; SEMI, Comment start ;;
|
|
850 (modify-syntax-entry ?\n ">" table) ;; Comment end
|
|
851 (modify-syntax-entry ?\" "\"" table) ;; String
|
|
852 (modify-syntax-entry ?\- "_" table) ;; Symbol
|
|
853 (modify-syntax-entry ?\\ "\\" table) ;; Quote
|
|
854 (modify-syntax-entry ?\` "'" table) ;; Prefix ` (backquote)
|
|
855 (modify-syntax-entry ?\' "'" table) ;; Prefix ' (quote)
|
|
856 (modify-syntax-entry ?\, "'" table) ;; Prefix , (comma)
|
|
857
|
|
858 table)
|
|
859 "Syntax table used in data-debug macro buffers.")
|
|
860
|
|
861 (defvar data-debug-map
|
|
862 (let ((km (make-sparse-keymap)))
|
|
863 (define-key km [mouse-2] 'data-debug-expand-or-contract-mouse)
|
|
864 (define-key km " " 'data-debug-expand-or-contract)
|
|
865 (define-key km "\C-m" 'data-debug-expand-or-contract)
|
|
866 (define-key km "n" 'data-debug-next)
|
|
867 (define-key km "p" 'data-debug-prev)
|
|
868 (define-key km "N" 'data-debug-next-expando)
|
|
869 (define-key km "P" 'data-debug-prev-expando)
|
|
870 km)
|
|
871 "Keymap used in data-debug.")
|
|
872
|
|
873 (defcustom data-debug-mode-hook nil
|
|
874 "*Hook run when data-debug starts."
|
|
875 :group 'data-debug
|
|
876 :type 'hook)
|
|
877
|
|
878 (defun data-debug-mode ()
|
|
879 "Major-mode for the Analyzer debugger.
|
|
880
|
|
881 \\{data-debug-map}"
|
|
882 (interactive)
|
|
883 (kill-all-local-variables)
|
|
884 (setq major-mode 'data-debug-mode
|
|
885 mode-name "DATA-DEBUG"
|
|
886 comment-start ";;"
|
|
887 comment-end "")
|
|
888 (set (make-local-variable 'comment-start-skip)
|
|
889 "\\(\\(^\\|[^\\\\\n]\\)\\(\\\\\\\\\\)*\\);+ *")
|
|
890 (set-syntax-table data-debug-mode-syntax-table)
|
|
891 (use-local-map data-debug-map)
|
|
892 (run-hooks 'data-debug-hook)
|
|
893 (buffer-disable-undo)
|
|
894 (set (make-local-variable 'font-lock-global-modes) nil)
|
|
895 (font-lock-mode -1)
|
|
896 )
|
|
897
|
|
898 ;;;###autoload
|
|
899 (defun data-debug-new-buffer (name)
|
|
900 "Create a new data-debug buffer with NAME."
|
|
901 (let ((b (get-buffer-create name)))
|
|
902 (pop-to-buffer b)
|
|
903 (set-buffer b)
|
|
904 (erase-buffer)
|
|
905 (data-debug-mode)
|
|
906 b))
|
|
907
|
|
908 ;;; Ddebug mode commands
|
|
909 ;;
|
|
910 (defun data-debug-next ()
|
|
911 "Go to the next line in the Ddebug buffer."
|
|
912 (interactive)
|
|
913 (forward-line 1)
|
|
914 (beginning-of-line)
|
|
915 (skip-chars-forward " *-><[]" (point-at-eol)))
|
|
916
|
|
917 (defun data-debug-prev ()
|
|
918 "Go to the next line in the Ddebug buffer."
|
|
919 (interactive)
|
|
920 (forward-line -1)
|
|
921 (beginning-of-line)
|
|
922 (skip-chars-forward " *-><[]" (point-at-eol)))
|
|
923
|
|
924 (defun data-debug-next-expando ()
|
|
925 "Go to the next line in the Ddebug buffer.
|
|
926 Contract the current line (if open) and expand the line
|
|
927 we move to."
|
|
928 (interactive)
|
|
929 (data-debug-contract-current-line)
|
|
930 (data-debug-next)
|
|
931 (data-debug-expand-current-line)
|
|
932 )
|
|
933
|
|
934 (defun data-debug-prev-expando ()
|
|
935 "Go to the previous line in the Ddebug buffer.
|
|
936 Contract the current line (if open) and expand the line
|
|
937 we move to."
|
|
938 (interactive)
|
|
939 (data-debug-contract-current-line)
|
|
940 (data-debug-prev)
|
|
941 (data-debug-expand-current-line)
|
|
942 )
|
|
943
|
|
944 (defun data-debug-current-line-expanded-p ()
|
|
945 "Return non-nil if the current line is expanded."
|
|
946 (let ((ti (current-indentation))
|
|
947 (ni (condition-case nil
|
|
948 (save-excursion
|
|
949 (end-of-line)
|
|
950 (forward-char 1)
|
|
951 (current-indentation))
|
|
952 (error 0))))
|
|
953 (> ni ti)))
|
|
954
|
|
955 (defun data-debug-line-expandable-p ()
|
|
956 "Return non-nil if the current line is expandable.
|
|
957 Lines that are not expandable are assumed to not be contractable."
|
|
958 (not (get-text-property (point) 'ddebug-noexpand)))
|
|
959
|
|
960 (defun data-debug-expand-current-line ()
|
|
961 "Expand the current line (if possible).
|
|
962 Do nothing if already expanded."
|
|
963 (when (or (not (data-debug-line-expandable-p))
|
|
964 (not (data-debug-current-line-expanded-p)))
|
|
965 ;; If the next line is the same or less indentation, expand.
|
|
966 (let ((fcn (get-text-property (point) 'ddebug-function)))
|
|
967 (when fcn
|
|
968 (funcall fcn (point))
|
|
969 (beginning-of-line)
|
|
970 ))))
|
|
971
|
|
972 (defun data-debug-contract-current-line ()
|
|
973 "Contract the current line (if possible).
|
|
974 Do nothing if already expanded."
|
|
975 (when (and (data-debug-current-line-expanded-p)
|
|
976 ;; Don't contract if the current line is not expandable.
|
|
977 (get-text-property (point) 'ddebug-function))
|
|
978 (let ((ti (current-indentation))
|
|
979 )
|
|
980 ;; If next indentation is larger, collapse.
|
|
981 (end-of-line)
|
|
982 (forward-char 1)
|
|
983 (let ((start (point))
|
|
984 (end nil))
|
|
985 (condition-case nil
|
|
986 (progn
|
|
987 ;; Keep checking indentation
|
|
988 (while (or (> (current-indentation) ti)
|
|
989 (looking-at "^\\s-*$"))
|
|
990 (end-of-line)
|
|
991 (forward-char 1))
|
|
992 (setq end (point))
|
|
993 )
|
|
994 (error (setq end (point-max))))
|
|
995 (delete-region start end)
|
|
996 (forward-char -1)
|
|
997 (beginning-of-line)))))
|
|
998
|
|
999 (defun data-debug-expand-or-contract ()
|
|
1000 "Expand or contract anything at the current point."
|
|
1001 (interactive)
|
|
1002 (if (and (data-debug-line-expandable-p)
|
|
1003 (data-debug-current-line-expanded-p))
|
|
1004 (data-debug-contract-current-line)
|
|
1005 (data-debug-expand-current-line))
|
|
1006 (skip-chars-forward " *-><[]" (point-at-eol)))
|
|
1007
|
|
1008 (defun data-debug-expand-or-contract-mouse (event)
|
|
1009 "Expand or contract anything at event EVENT."
|
|
1010 (interactive "e")
|
|
1011 (let* ((win (car (car (cdr event))))
|
|
1012 )
|
|
1013 (select-window win t)
|
|
1014 (save-excursion
|
|
1015 ;(goto-char (window-start win))
|
|
1016 (mouse-set-point event)
|
|
1017 (data-debug-expand-or-contract))
|
|
1018 ))
|
|
1019
|
|
1020 ;;; GENERIC STRUCTURE DUMP
|
|
1021 ;;
|
|
1022 (defun data-debug-show-stuff (stuff name)
|
|
1023 "Data debug STUFF in a buffer named *NAME DDebug*."
|
|
1024 (data-debug-new-buffer (concat "*" name " DDebug*"))
|
|
1025 (data-debug-insert-thing stuff "?" "")
|
|
1026 (goto-char (point-min))
|
|
1027 (when (data-debug-line-expandable-p)
|
|
1028 (data-debug-expand-current-line)))
|
|
1029
|
|
1030 ;;; DEBUG COMMANDS
|
|
1031 ;;
|
|
1032 ;; Various commands for displaying complex data structures.
|
|
1033
|
|
1034 (defun data-debug-edebug-expr (expr)
|
105325
|
1035 "Dump out the contents of some expression EXPR in edebug with ddebug."
|
105241
|
1036 (interactive
|
|
1037 (list (let ((minibuffer-completing-symbol t))
|
|
1038 (read-from-minibuffer "Eval: "
|
|
1039 nil read-expression-map t
|
|
1040 'read-expression-history))
|
|
1041 ))
|
|
1042 (let ((v (eval expr)))
|
|
1043 (if (not v)
|
|
1044 (message "Expression %s is nil." expr)
|
|
1045 (data-debug-show-stuff v "expression"))))
|
|
1046
|
|
1047 (defun data-debug-eval-expression (expr)
|
|
1048 "Evaluate EXPR and display the value.
|
|
1049 If the result is something simple, show it in the echo area.
|
|
1050 If the result is a list or vector, then use the data debugger to display it."
|
|
1051 (interactive
|
|
1052 (list (let ((minibuffer-completing-symbol t))
|
|
1053 (read-from-minibuffer "Eval: "
|
|
1054 nil read-expression-map t
|
|
1055 'read-expression-history))
|
|
1056 ))
|
|
1057
|
|
1058 (if (null eval-expression-debug-on-error)
|
|
1059 (setq values (cons (eval expr) values))
|
|
1060 (let ((old-value (make-symbol "t")) new-value)
|
|
1061 ;; Bind debug-on-error to something unique so that we can
|
|
1062 ;; detect when evaled code changes it.
|
|
1063 (let ((debug-on-error old-value))
|
|
1064 (setq values (cons (eval expr) values))
|
|
1065 (setq new-value debug-on-error))
|
|
1066 ;; If evaled code has changed the value of debug-on-error,
|
|
1067 ;; propagate that change to the global binding.
|
|
1068 (unless (eq old-value new-value)
|
|
1069 (setq debug-on-error new-value))))
|
|
1070
|
|
1071 (if (or (consp (car values)) (vectorp (car values)))
|
|
1072 (let ((v (car values)))
|
|
1073 (data-debug-show-stuff v "Expression"))
|
|
1074 ;; Old style
|
|
1075 (prog1
|
|
1076 (prin1 (car values) t)
|
|
1077 (let ((str (eval-expression-print-format (car values))))
|
|
1078 (if str (princ str t))))))
|
|
1079
|
|
1080 (provide 'data-debug)
|
|
1081
|
|
1082 (if (featurep 'eieio)
|
|
1083 (require 'eieio-datadebug))
|
|
1084
|
105377
|
1085 ;; arch-tag: 4807227d-08e7-45c4-8ea5-9e4595c3bfb1
|
105241
|
1086 ;;; data-debug.el ends here
|