2005-05-30 10:30:51 +00:00
|
|
|
# emacs-buffer.gdb --- gdb macros for recovering buffers from emacs coredumps
|
|
|
|
|
2013-01-01 09:11:05 +00:00
|
|
|
# Copyright (C) 2005-2013 Free Software Foundation, Inc.
|
2005-05-30 10:30:51 +00:00
|
|
|
|
|
|
|
# Maintainer: Noah Friedman <friedman@splode.com>
|
|
|
|
# Created: 2005-04-28
|
|
|
|
|
|
|
|
# This file is part of GNU Emacs.
|
2008-05-15 07:32:14 +00:00
|
|
|
|
|
|
|
# GNU Emacs is free software: you can redistribute it and/or modify
|
2005-05-30 10:30:51 +00:00
|
|
|
# it under the terms of the GNU General Public License as published by
|
2008-05-15 07:32:14 +00:00
|
|
|
# the Free Software Foundation, either version 3 of the License, or
|
|
|
|
# (at your option) any later version.
|
|
|
|
|
2005-05-30 10:30:51 +00:00
|
|
|
# GNU Emacs 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 General Public License for more details.
|
2008-05-15 07:32:14 +00:00
|
|
|
|
2005-05-30 10:30:51 +00:00
|
|
|
# You should have received a copy of the GNU General Public License
|
2008-05-15 07:32:14 +00:00
|
|
|
# along with GNU Emacs. If not, see <http://www.gnu.org/licenses/>.
|
2005-05-30 10:30:51 +00:00
|
|
|
|
|
|
|
# Commentary:
|
|
|
|
|
|
|
|
# This is a set of gdb macros for recovering the contents of buffers from
|
|
|
|
# an Emacs coredump; they may not always be file-backed or have a recent
|
|
|
|
# autosave.
|
|
|
|
#
|
2006-01-25 03:58:43 +00:00
|
|
|
# The Emacs executable must have debugging symbols for this to work.
|
|
|
|
# But you never strip Emacs, right?
|
2005-05-30 10:30:51 +00:00
|
|
|
#
|
|
|
|
# The main commands of interest are `ybuffer-list', `yfile-buffers',
|
|
|
|
# `ysave-buffer', and `ybuffer-contents'. The `y' prefix avoids any
|
|
|
|
# namespace collisions with emacs/src/.gdbinit.
|
|
|
|
|
2011-04-22 23:29:02 +00:00
|
|
|
# Since the internal data structures in Emacs occasionally from time to
|
|
|
|
# time, you should use the version of this file that came with your
|
|
|
|
# particular Emacs version; older versions might not work anymore.
|
|
|
|
|
2005-05-30 10:30:51 +00:00
|
|
|
# Example usage:
|
|
|
|
#
|
|
|
|
# $ gdb /export/src/emacs/2005-05-02--03-17/src/emacs core.emacs.6.9845
|
|
|
|
# Current directory is /u/noah/
|
|
|
|
# GNU gdb (6.1post-1.20040607.43rh)
|
|
|
|
# ...
|
|
|
|
# #0 0x400007a2 in _dl_sysinfo_int80 () from /lib/ld-linux.so.2
|
|
|
|
# (gdb) source emacs-buffer.gdb
|
|
|
|
# (gdb) ybuffer-list
|
|
|
|
# B# M Size Name Mode File
|
|
|
|
# -- - ---- ---- ---- ----
|
|
|
|
# 0 * 556 mail to emacs-devel@gnu.org Mail
|
|
|
|
# 1 * 0 *Minibuf-1* Fundamental
|
|
|
|
# 2 145769 ChangeLog Change Log /u/noah/lib/elisp/noahf/ChangeLog
|
|
|
|
# 3 6619 ascii-table.el Elisp /u/noah/lib/elisp/noahf/ascii-table.el
|
|
|
|
# 4 * 48396 *Messages* Fundamental
|
|
|
|
# 5 3191 *Apropos* Apropos
|
|
|
|
# 6 17642 init-21.el Elisp /u/noah/etc/init/emacs/init-21.el
|
|
|
|
# 7 333 cpuid.c C /u/noah/cpuid.c
|
|
|
|
# 8 230 src Dired
|
|
|
|
# 9 218 noah Dired
|
|
|
|
# 10 * 21 *Echo Area 0* Fundamental
|
|
|
|
# 11 * 0 *Echo Area 1* Fundamental
|
|
|
|
# 12 319952 *bbdb data* Text /u/noah/.bbdb
|
|
|
|
# (gdb) ysave-buffer 0 mail.save
|
|
|
|
# [Wrote buffer "mail to emacs-devel@gnu.org" to file mail.save]
|
|
|
|
# (gdb) quit
|
|
|
|
# $ ls -l mail.save
|
|
|
|
# -rw-rw-rw- 1 noah user 556 May 2 04:05 mail.save
|
|
|
|
# $
|
|
|
|
|
|
|
|
# Code:
|
|
|
|
|
Simplify export of symbols to GDB; fix related .gdbinit bugs.
* etc/emacs-buffer.gdb ($tagmask, $valmask): Remove.
(ygetptr): Adjust to recent changes in lisp.h and emacs.c,
by using VALMASK instead of $valmask, CHECK_LISP_OBJECT_TYPE
instead of gdb_use_union, and DATA_SEG_BITS instead of
gdb_data_seg_bits. Also, use $ptr.i rather than $ptr.u.val.
* src/.gdbinit (xgetptr, xgetint, xgettype): Don't use "set $bugfix =
$bugfix.i", as this doesn't work (with GDB 7.4.1, anyway).
(xgetptr, xgetint, xgettype, xcoding, xcharset, xprintbytestr):
Adjust to changes in lisp.h and emacs.c, by using
CHECK_LISP_OBJECT_TYPE rather than gdb_use_struct, VALMASK instead
of $valmask, DATA_SEG_BITS instead of gdb_data_seg_bits,
INTTYPEBITS instead of gdb_gctypebits - 1, USE_LSB_TAG instead of
gdb_use_lsb, (1 << GCTYPEBITS) - 1 instead of $tagmask, VALBITS
instead of gdb_valbits.
(xvectype, xvector, xpr, xprintstr, xbacktrace): Similarly, use
PSEUDOVECTOR_FLAG instead of PVEC_FLAG, and ARRAY_MARK_FLAG
instead of gdb_array_mark_flag.
(xboolvector): Get size from $->size, not $->header.size.
Use BOOL_VECTOR_BITS_PER_CHAR rather than mystery constants.
(xreload, hook-run, hookpost-run): Remove.
* src/emacs.c: Include <verify.h>.
(gdb_use_lsb, gdb_use_struct, gdb_valbits, gdb_gctypebits)
(gdb_data_seg_bits, PVEC_FLAG, gdb_array_mark_flag, gdb_pvec_type):
Remove.
(gdb_CHECK_LISP_OBJECT_TYPE, gdb_DATA_SEG_BITS, gdb_GCTYPEBITS)
(gdb_USE_LSB_TAG): New enum constants.
(CHECK_LISP_OBJECT_TYPE, DATA_SEG_BITS, GCTYPEBITS, USE_LSB_TAG):
Also define these as enum constants, so they're visible to GDB.
(ARRAY_MARK_FLAG_VAL, PSEUDOVECTOR_FLAG_VAL, VALMASK_VAL): New macros.
(ARRAY_MARK_FLAG, PSEUDOVECTOR_FLAG, VALMASK): Also define these
as constants, so they're visible to GDB.
* src/lisp.h (VALBITS, INTTYPEBITS, FIXNUM_BITS, PSEUDOVECTOR_SIZE_BITS)
(PSEUDOVECTOR_SIZE_MASK, PVEC_TYPE_MASK, BOOL_VECTOR_BITS_PER_CHAR):
Now enum constants, not macros, so they're visible to GDB.
(CHECK_LISP_OBJECT_TYPE, DATA_SEG_BITS): Default to 0, as this is
more convenient now. All uses changed.
(VALMASK) [USE_LSB_TAG]: Also define in this case.
* src/mem-limits.h (EXCEEDS_LISP_PTR): Adjust to DATA_SEG_BITS change.
2012-07-26 08:12:03 +00:00
|
|
|
# Force loading of symbols, enough to give us VALMASK etc.
|
2005-05-30 10:30:51 +00:00
|
|
|
set main
|
|
|
|
|
|
|
|
# When nonzero, display some extra diagnostics in various commands
|
|
|
|
set $yverbose = 1
|
|
|
|
set $yfile_buffers_only = 0
|
|
|
|
|
|
|
|
define ygetptr
|
|
|
|
set $ptr = $arg0
|
Simplify export of symbols to GDB; fix related .gdbinit bugs.
* etc/emacs-buffer.gdb ($tagmask, $valmask): Remove.
(ygetptr): Adjust to recent changes in lisp.h and emacs.c,
by using VALMASK instead of $valmask, CHECK_LISP_OBJECT_TYPE
instead of gdb_use_union, and DATA_SEG_BITS instead of
gdb_data_seg_bits. Also, use $ptr.i rather than $ptr.u.val.
* src/.gdbinit (xgetptr, xgetint, xgettype): Don't use "set $bugfix =
$bugfix.i", as this doesn't work (with GDB 7.4.1, anyway).
(xgetptr, xgetint, xgettype, xcoding, xcharset, xprintbytestr):
Adjust to changes in lisp.h and emacs.c, by using
CHECK_LISP_OBJECT_TYPE rather than gdb_use_struct, VALMASK instead
of $valmask, DATA_SEG_BITS instead of gdb_data_seg_bits,
INTTYPEBITS instead of gdb_gctypebits - 1, USE_LSB_TAG instead of
gdb_use_lsb, (1 << GCTYPEBITS) - 1 instead of $tagmask, VALBITS
instead of gdb_valbits.
(xvectype, xvector, xpr, xprintstr, xbacktrace): Similarly, use
PSEUDOVECTOR_FLAG instead of PVEC_FLAG, and ARRAY_MARK_FLAG
instead of gdb_array_mark_flag.
(xboolvector): Get size from $->size, not $->header.size.
Use BOOL_VECTOR_BITS_PER_CHAR rather than mystery constants.
(xreload, hook-run, hookpost-run): Remove.
* src/emacs.c: Include <verify.h>.
(gdb_use_lsb, gdb_use_struct, gdb_valbits, gdb_gctypebits)
(gdb_data_seg_bits, PVEC_FLAG, gdb_array_mark_flag, gdb_pvec_type):
Remove.
(gdb_CHECK_LISP_OBJECT_TYPE, gdb_DATA_SEG_BITS, gdb_GCTYPEBITS)
(gdb_USE_LSB_TAG): New enum constants.
(CHECK_LISP_OBJECT_TYPE, DATA_SEG_BITS, GCTYPEBITS, USE_LSB_TAG):
Also define these as enum constants, so they're visible to GDB.
(ARRAY_MARK_FLAG_VAL, PSEUDOVECTOR_FLAG_VAL, VALMASK_VAL): New macros.
(ARRAY_MARK_FLAG, PSEUDOVECTOR_FLAG, VALMASK): Also define these
as constants, so they're visible to GDB.
* src/lisp.h (VALBITS, INTTYPEBITS, FIXNUM_BITS, PSEUDOVECTOR_SIZE_BITS)
(PSEUDOVECTOR_SIZE_MASK, PVEC_TYPE_MASK, BOOL_VECTOR_BITS_PER_CHAR):
Now enum constants, not macros, so they're visible to GDB.
(CHECK_LISP_OBJECT_TYPE, DATA_SEG_BITS): Default to 0, as this is
more convenient now. All uses changed.
(VALMASK) [USE_LSB_TAG]: Also define in this case.
* src/mem-limits.h (EXCEEDS_LISP_PTR): Adjust to DATA_SEG_BITS change.
2012-07-26 08:12:03 +00:00
|
|
|
set $ptr = ((CHECK_LISP_OBJECT_TYPE ? $ptr.i : $ptr) & VALMASK) | DATA_SEG_BITS
|
2005-05-30 10:30:51 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
define ybuffer-list
|
|
|
|
set $files_only = $yfile_buffers_only
|
|
|
|
set $yfile_buffers_only = 0
|
|
|
|
|
|
|
|
if $yverbose
|
|
|
|
printf "B# M Size Name Mode File\n"
|
|
|
|
printf "-- - ---- ---- ---- ----\n"
|
|
|
|
end
|
|
|
|
|
|
|
|
set $i = 0
|
|
|
|
set $alist = Vbuffer_alist
|
|
|
|
while $alist != Qnil
|
|
|
|
ygetptr $alist
|
|
|
|
set $this = ((struct Lisp_Cons *) $ptr)->car
|
2006-01-25 03:58:43 +00:00
|
|
|
set $alist = ((struct Lisp_Cons *) $ptr)->u.cdr
|
2005-05-30 10:30:51 +00:00
|
|
|
|
|
|
|
# Vbuffer_alist elts are pairs of the form (name . buffer)
|
|
|
|
ygetptr $this
|
2006-01-25 03:58:43 +00:00
|
|
|
set $buf = ((struct Lisp_Cons *) $ptr)->u.cdr
|
2005-05-30 10:30:51 +00:00
|
|
|
ygetptr $buf
|
|
|
|
set $buf = (struct buffer *) $ptr
|
|
|
|
|
2011-04-22 23:29:02 +00:00
|
|
|
if ! ($files_only && $buf->filename_ == Qnil)
|
|
|
|
ygetptr $buf->name_
|
2005-05-30 10:30:51 +00:00
|
|
|
set $name = ((struct Lisp_String *) $ptr)->data
|
|
|
|
set $modp = ($buf->text->modiff > $buf->text->save_modiff) ? '*' : ' '
|
|
|
|
|
2011-04-22 23:29:02 +00:00
|
|
|
ygetptr $buf->mode_name_
|
2005-05-30 10:30:51 +00:00
|
|
|
set $mode = ((struct Lisp_String *) $ptr)->data
|
|
|
|
|
2011-04-22 23:29:02 +00:00
|
|
|
if $buf->filename_ != Qnil
|
|
|
|
ygetptr $buf->filename_
|
2005-06-10 22:39:45 +00:00
|
|
|
printf "%2d %c %9d %-20s %-10s %s\n", \
|
|
|
|
$i, $modp, ($buf->text->z_byte - 1), $name, $mode, \
|
|
|
|
((struct Lisp_String *) $ptr)->data
|
2005-05-30 10:30:51 +00:00
|
|
|
else
|
2005-06-10 22:39:45 +00:00
|
|
|
printf "%2d %c %9d %-20s %-10s\n", \
|
|
|
|
$i, $modp, ($buf->text->z_byte - 1), $name, $mode
|
2005-05-30 10:30:51 +00:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
set $i++
|
|
|
|
end
|
|
|
|
end
|
|
|
|
document ybuffer-list
|
|
|
|
Display a list of buffer names, sizes, and other attributes.
|
|
|
|
The buffer number in the first column is used as an argument
|
|
|
|
to some other emacs-buffer recovery commands, e.g. `ysave-buffer'.
|
|
|
|
end
|
|
|
|
|
|
|
|
define yfile-buffers
|
|
|
|
set $yfile_buffers_only = 1
|
|
|
|
ybuffer-list
|
|
|
|
end
|
|
|
|
document yfile-buffers
|
|
|
|
Display a list of buffers which are associated with files.
|
|
|
|
This is like `ybuffer-list', but only buffers that were visiting files
|
|
|
|
are displayed.
|
|
|
|
end
|
|
|
|
|
|
|
|
define yset-buffer
|
|
|
|
set $i = $arg0
|
|
|
|
|
|
|
|
set $alist = Vbuffer_alist
|
|
|
|
while ($alist != Qnil && $i > 0)
|
|
|
|
ygetptr $alist
|
2006-01-25 03:58:43 +00:00
|
|
|
set $alist = ((struct Lisp_Cons *) $ptr)->u.cdr
|
2005-05-30 10:30:51 +00:00
|
|
|
set $i--
|
|
|
|
end
|
|
|
|
|
|
|
|
# Get car of alist; this is a pair (name . buffer)
|
|
|
|
ygetptr $alist
|
|
|
|
set $this = ((struct Lisp_Cons *) $ptr)->car
|
|
|
|
|
|
|
|
# Get the buffer object
|
|
|
|
ygetptr $this
|
2006-01-25 03:58:43 +00:00
|
|
|
set $this = ((struct Lisp_Cons *) $ptr)->u.cdr
|
2005-05-30 10:30:51 +00:00
|
|
|
|
|
|
|
ygetptr $this
|
|
|
|
set $ycurrent_buffer = (struct buffer *) $ptr
|
|
|
|
end
|
|
|
|
document yset-buffer
|
|
|
|
Set current buffer (for other emacs-buffer recovery commands) to the ARG'th
|
|
|
|
buffer as displayed by `ybuffer-list'.
|
|
|
|
end
|
|
|
|
|
|
|
|
define yget-buffer-pointers
|
|
|
|
yset-buffer $arg0
|
|
|
|
set $buf = $ycurrent_buffer->text
|
|
|
|
|
|
|
|
set $beg = $buf->beg
|
|
|
|
set $gap = $beg + $buf->gpt_byte
|
|
|
|
set $gap_end = $gap + $buf->gap_size - 1
|
|
|
|
set $end = $gap_end + ($buf->z_byte - $buf->gpt_byte)
|
|
|
|
|
|
|
|
set $modp = $buf->modiff > $buf->save_modiff
|
|
|
|
|
|
|
|
#print *$beg@($gap - $beg)
|
|
|
|
#print *$gap_end@($end - $gap_end)
|
|
|
|
end
|
|
|
|
document yget-buffer-pointers
|
|
|
|
Update convenience variables with address pointers for the ARG'th buffer
|
|
|
|
as displayed by `ybuffer-list'.
|
|
|
|
|
|
|
|
This also sets the current buffer using `yset-buffer' (which see).
|
|
|
|
end
|
|
|
|
|
|
|
|
define yget-current-buffer-name
|
2011-04-22 23:29:02 +00:00
|
|
|
set $this = $ycurrent_buffer->name_
|
2005-05-30 10:30:51 +00:00
|
|
|
ygetptr $this
|
|
|
|
set $ycurrent_buffer_name = ((struct Lisp_String *) $ptr)->data
|
|
|
|
end
|
|
|
|
document yget-current-buffer-name
|
|
|
|
Set $ycurrent_buffer_name to the name of the currently selected buffer.
|
|
|
|
end
|
|
|
|
|
|
|
|
define ycurrent-buffer
|
|
|
|
yget-current-buffer-name
|
|
|
|
printf "%s\n", $ycurrent_buffer_name
|
|
|
|
end
|
|
|
|
document ycurrent-buffer
|
|
|
|
Display the currently selected buffer.
|
|
|
|
end
|
|
|
|
|
|
|
|
define ydump-buffer
|
|
|
|
yget-buffer-pointers $arg0
|
|
|
|
if $buf->z_byte > 1
|
|
|
|
if $buf->z_byte <= $buf->gpt_byte
|
|
|
|
set $endptr = $beg + $buf->gpt_byte - 1
|
|
|
|
dump binary memory $arg1 $beg $endptr
|
|
|
|
else
|
|
|
|
dump binary memory $arg1 $beg $gap-1
|
|
|
|
append binary memory $arg1 $gap_end $end
|
|
|
|
set $endptr = $end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
document ydump-buffer
|
|
|
|
Write contents of buffer N (as numbered according to `ybuffer-list') to
|
|
|
|
file FILE.
|
|
|
|
|
|
|
|
This is mainly used as an internal subroutine for `ysave-buffer' and
|
|
|
|
`ybuffer-contents', which see.
|
|
|
|
end
|
|
|
|
|
|
|
|
define ysave-buffer
|
|
|
|
ydump-buffer $arg0 $arg1
|
|
|
|
if $yverbose
|
|
|
|
yget-current-buffer-name
|
|
|
|
if $buf->z_byte <= 1
|
|
|
|
printf "[Buffer \"%s\" is empty.]\n", $ycurrent_buffer_name
|
|
|
|
else
|
|
|
|
# Output string broken into separate calls as necessary to avoid
|
|
|
|
# requiring a running process for evaluation.
|
|
|
|
printf "[Wrote buffer \"%s\" to file ", $ycurrent_buffer_name
|
|
|
|
echo $arg1]\n
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
document ysave-buffer
|
|
|
|
Save contents of buffer N (as numbered according to `ybuffer-list') to
|
|
|
|
file FILE.
|
|
|
|
end
|
|
|
|
|
|
|
|
define ybuffer-contents
|
|
|
|
ydump-buffer $arg0 /dev/stdout
|
|
|
|
if $yverbose && $buf->z_byte <= 1
|
|
|
|
yget-current-buffer-name
|
|
|
|
printf "[Buffer \"%s\" is empty.]\n", $ycurrent_buffer_name
|
|
|
|
else
|
|
|
|
if *($endptr-1) != '\n'
|
|
|
|
echo \n
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
document ybuffer-contents
|
|
|
|
Write contents of buffer N (numbered according to `ybuffer-list') to stdout.
|
|
|
|
end
|
|
|
|
|
|
|
|
# local variables:
|
|
|
|
# mode: gdb-script
|
|
|
|
# end:
|