declare smobs in alloc.c
[bpt/emacs.git] / etc / emacs-buffer.gdb
CommitLineData
a7868bfa
NF
1# emacs-buffer.gdb --- gdb macros for recovering buffers from emacs coredumps
2
ba318903 3# Copyright (C) 2005-2014 Free Software Foundation, Inc.
a7868bfa
NF
4
5# Maintainer: Noah Friedman <friedman@splode.com>
a7868bfa
NF
6# Created: 2005-04-28
7
a7868bfa 8# This file is part of GNU Emacs.
ab73e885
GM
9
10# GNU Emacs is free software: you can redistribute it and/or modify
a7868bfa 11# it under the terms of the GNU General Public License as published by
ab73e885
GM
12# the Free Software Foundation, either version 3 of the License, or
13# (at your option) any later version.
14
a7868bfa
NF
15# GNU Emacs is distributed in the hope that it will be useful,
16# but WITHOUT ANY WARRANTY; without even the implied warranty of
17# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
18# GNU General Public License for more details.
ab73e885 19
a7868bfa 20# You should have received a copy of the GNU General Public License
ab73e885 21# along with GNU Emacs. If not, see <http://www.gnu.org/licenses/>.
a7868bfa
NF
22
23# Commentary:
24
25# This is a set of gdb macros for recovering the contents of buffers from
26# an Emacs coredump; they may not always be file-backed or have a recent
27# autosave.
28#
69f0b7f5
NF
29# The Emacs executable must have debugging symbols for this to work.
30# But you never strip Emacs, right?
a7868bfa
NF
31#
32# The main commands of interest are `ybuffer-list', `yfile-buffers',
33# `ysave-buffer', and `ybuffer-contents'. The `y' prefix avoids any
34# namespace collisions with emacs/src/.gdbinit.
35
c243bc79
NF
36# Since the internal data structures in Emacs occasionally from time to
37# time, you should use the version of this file that came with your
38# particular Emacs version; older versions might not work anymore.
39
a7868bfa
NF
40# Example usage:
41#
42# $ gdb /export/src/emacs/2005-05-02--03-17/src/emacs core.emacs.6.9845
43# Current directory is /u/noah/
44# GNU gdb (6.1post-1.20040607.43rh)
45# ...
46# #0 0x400007a2 in _dl_sysinfo_int80 () from /lib/ld-linux.so.2
47# (gdb) source emacs-buffer.gdb
48# (gdb) ybuffer-list
49# B# M Size Name Mode File
50# -- - ---- ---- ---- ----
51# 0 * 556 mail to emacs-devel@gnu.org Mail
52# 1 * 0 *Minibuf-1* Fundamental
53# 2 145769 ChangeLog Change Log /u/noah/lib/elisp/noahf/ChangeLog
54# 3 6619 ascii-table.el Elisp /u/noah/lib/elisp/noahf/ascii-table.el
55# 4 * 48396 *Messages* Fundamental
56# 5 3191 *Apropos* Apropos
57# 6 17642 init-21.el Elisp /u/noah/etc/init/emacs/init-21.el
58# 7 333 cpuid.c C /u/noah/cpuid.c
59# 8 230 src Dired
60# 9 218 noah Dired
61# 10 * 21 *Echo Area 0* Fundamental
62# 11 * 0 *Echo Area 1* Fundamental
63# 12 319952 *bbdb data* Text /u/noah/.bbdb
64# (gdb) ysave-buffer 0 mail.save
65# [Wrote buffer "mail to emacs-devel@gnu.org" to file mail.save]
66# (gdb) quit
67# $ ls -l mail.save
68# -rw-rw-rw- 1 noah user 556 May 2 04:05 mail.save
69# $
70
71# Code:
72
1781b9e9 73# Force loading of symbols, enough to give us VALMASK etc.
a7868bfa
NF
74set main
75
76# When nonzero, display some extra diagnostics in various commands
77set $yverbose = 1
78set $yfile_buffers_only = 0
79
a7868bfa
NF
80define ygetptr
81 set $ptr = $arg0
5403b2d3 82 set $ptr = (CHECK_LISP_OBJECT_TYPE ? $ptr.i : $ptr) & VALMASK
a7868bfa
NF
83end
84
85define ybuffer-list
86 set $files_only = $yfile_buffers_only
87 set $yfile_buffers_only = 0
88
89 if $yverbose
90 printf "B# M Size Name Mode File\n"
91 printf "-- - ---- ---- ---- ----\n"
92 end
93
94 set $i = 0
95 set $alist = Vbuffer_alist
96 while $alist != Qnil
97 ygetptr $alist
98 set $this = ((struct Lisp_Cons *) $ptr)->car
69f0b7f5 99 set $alist = ((struct Lisp_Cons *) $ptr)->u.cdr
a7868bfa
NF
100
101 # Vbuffer_alist elts are pairs of the form (name . buffer)
102 ygetptr $this
69f0b7f5 103 set $buf = ((struct Lisp_Cons *) $ptr)->u.cdr
a7868bfa
NF
104 ygetptr $buf
105 set $buf = (struct buffer *) $ptr
106
c243bc79
NF
107 if ! ($files_only && $buf->filename_ == Qnil)
108 ygetptr $buf->name_
a7868bfa
NF
109 set $name = ((struct Lisp_String *) $ptr)->data
110 set $modp = ($buf->text->modiff > $buf->text->save_modiff) ? '*' : ' '
111
c243bc79 112 ygetptr $buf->mode_name_
a7868bfa
NF
113 set $mode = ((struct Lisp_String *) $ptr)->data
114
c243bc79
NF
115 if $buf->filename_ != Qnil
116 ygetptr $buf->filename_
58f498fe
NF
117 printf "%2d %c %9d %-20s %-10s %s\n", \
118 $i, $modp, ($buf->text->z_byte - 1), $name, $mode, \
119 ((struct Lisp_String *) $ptr)->data
a7868bfa 120 else
58f498fe
NF
121 printf "%2d %c %9d %-20s %-10s\n", \
122 $i, $modp, ($buf->text->z_byte - 1), $name, $mode
a7868bfa 123 end
a7868bfa
NF
124 end
125
126 set $i++
127 end
128end
129document ybuffer-list
130 Display a list of buffer names, sizes, and other attributes.
131 The buffer number in the first column is used as an argument
132 to some other emacs-buffer recovery commands, e.g. `ysave-buffer'.
133end
134
135define yfile-buffers
136 set $yfile_buffers_only = 1
137 ybuffer-list
138end
139document yfile-buffers
140 Display a list of buffers which are associated with files.
141 This is like `ybuffer-list', but only buffers that were visiting files
142 are displayed.
143end
144
145define yset-buffer
146 set $i = $arg0
147
148 set $alist = Vbuffer_alist
149 while ($alist != Qnil && $i > 0)
150 ygetptr $alist
69f0b7f5 151 set $alist = ((struct Lisp_Cons *) $ptr)->u.cdr
a7868bfa
NF
152 set $i--
153 end
154
155 # Get car of alist; this is a pair (name . buffer)
156 ygetptr $alist
157 set $this = ((struct Lisp_Cons *) $ptr)->car
158
159 # Get the buffer object
160 ygetptr $this
69f0b7f5 161 set $this = ((struct Lisp_Cons *) $ptr)->u.cdr
a7868bfa
NF
162
163 ygetptr $this
164 set $ycurrent_buffer = (struct buffer *) $ptr
165end
166document yset-buffer
167 Set current buffer (for other emacs-buffer recovery commands) to the ARG'th
168 buffer as displayed by `ybuffer-list'.
169end
170
171define yget-buffer-pointers
172 yset-buffer $arg0
173 set $buf = $ycurrent_buffer->text
174
175 set $beg = $buf->beg
176 set $gap = $beg + $buf->gpt_byte
177 set $gap_end = $gap + $buf->gap_size - 1
178 set $end = $gap_end + ($buf->z_byte - $buf->gpt_byte)
179
180 set $modp = $buf->modiff > $buf->save_modiff
181
182 #print *$beg@($gap - $beg)
183 #print *$gap_end@($end - $gap_end)
184end
185document yget-buffer-pointers
186 Update convenience variables with address pointers for the ARG'th buffer
187 as displayed by `ybuffer-list'.
188
189 This also sets the current buffer using `yset-buffer' (which see).
190end
191
192define yget-current-buffer-name
c243bc79 193 set $this = $ycurrent_buffer->name_
a7868bfa
NF
194 ygetptr $this
195 set $ycurrent_buffer_name = ((struct Lisp_String *) $ptr)->data
196end
197document yget-current-buffer-name
198 Set $ycurrent_buffer_name to the name of the currently selected buffer.
199end
200
201define ycurrent-buffer
202 yget-current-buffer-name
203 printf "%s\n", $ycurrent_buffer_name
204end
205document ycurrent-buffer
206 Display the currently selected buffer.
207end
208
209define ydump-buffer
210 yget-buffer-pointers $arg0
211 if $buf->z_byte > 1
212 if $buf->z_byte <= $buf->gpt_byte
213 set $endptr = $beg + $buf->gpt_byte - 1
214 dump binary memory $arg1 $beg $endptr
215 else
216 dump binary memory $arg1 $beg $gap-1
217 append binary memory $arg1 $gap_end $end
218 set $endptr = $end
219 end
220 end
221end
222document ydump-buffer
223 Write contents of buffer N (as numbered according to `ybuffer-list') to
224 file FILE.
225
226 This is mainly used as an internal subroutine for `ysave-buffer' and
227 `ybuffer-contents', which see.
228end
229
230define ysave-buffer
231 ydump-buffer $arg0 $arg1
232 if $yverbose
233 yget-current-buffer-name
234 if $buf->z_byte <= 1
235 printf "[Buffer \"%s\" is empty.]\n", $ycurrent_buffer_name
236 else
237 # Output string broken into separate calls as necessary to avoid
238 # requiring a running process for evaluation.
239 printf "[Wrote buffer \"%s\" to file ", $ycurrent_buffer_name
240 echo $arg1]\n
241 end
242 end
243end
244document ysave-buffer
245 Save contents of buffer N (as numbered according to `ybuffer-list') to
246 file FILE.
247end
248
249define ybuffer-contents
250 ydump-buffer $arg0 /dev/stdout
251 if $yverbose && $buf->z_byte <= 1
252 yget-current-buffer-name
253 printf "[Buffer \"%s\" is empty.]\n", $ycurrent_buffer_name
254 else
255 if *($endptr-1) != '\n'
256 echo \n
257 end
258 end
259end
260document ybuffer-contents
261 Write contents of buffer N (numbered according to `ybuffer-list') to stdout.
262end
263
264# local variables:
265# mode: gdb-script
266# end: