mirror of
https://github.com/tobast/libunwind-eh_elf.git
synced 2025-01-25 01:30:30 +01:00
(Logical change 1.45)
This commit is contained in:
parent
5f94e2d488
commit
824d6619b5
21 changed files with 1667 additions and 0 deletions
|
@ -0,0 +1,136 @@
|
||||||
|
'\" t
|
||||||
|
.\" Manual page created with latex2man on Sat Feb 8 01:31:56 PST 2003
|
||||||
|
.\" NOTE: This file is generated, DO NOT EDIT.
|
||||||
|
.de Vb
|
||||||
|
.ft CW
|
||||||
|
.nf
|
||||||
|
..
|
||||||
|
.de Ve
|
||||||
|
.ft R
|
||||||
|
|
||||||
|
.fi
|
||||||
|
..
|
||||||
|
.TH "LIBUNWIND\-SETJMP" "3" "08 February 2003" "Programming Library " "Programming Library "
|
||||||
|
.SH NAME
|
||||||
|
|
||||||
|
libunwind\-setjmp \-\- libunwind\-based non\-local gotos
|
||||||
|
.PP
|
||||||
|
.SH SYNOPSIS
|
||||||
|
|
||||||
|
.PP
|
||||||
|
#include <setjmp.h>
|
||||||
|
.br
|
||||||
|
.PP
|
||||||
|
int
|
||||||
|
setjmp(jmp_buf env);
|
||||||
|
.br
|
||||||
|
void
|
||||||
|
longjmp(jmp_buf env,
|
||||||
|
int val);
|
||||||
|
.br
|
||||||
|
int
|
||||||
|
_setjmp(jmp_buf env);
|
||||||
|
.br
|
||||||
|
void
|
||||||
|
_longjmp(jmp_buf env,
|
||||||
|
int val);
|
||||||
|
.br
|
||||||
|
int
|
||||||
|
setjmp(sigjmp_buf env,
|
||||||
|
int savemask);
|
||||||
|
.br
|
||||||
|
void
|
||||||
|
siglongjmp(sigjmp_buf env,
|
||||||
|
int val);
|
||||||
|
.br
|
||||||
|
.PP
|
||||||
|
.SH DESCRIPTION
|
||||||
|
|
||||||
|
.PP
|
||||||
|
The unwind\-setjmp
|
||||||
|
library offers a libunwind\-based
|
||||||
|
implementation of non\-local gotos. This implementation is intended to
|
||||||
|
be a drop\-in replacement for the normal, system\-provided routines of
|
||||||
|
the same name. The main advantage of using the unwind\-setjmp
|
||||||
|
library is that setting up a non\-local goto via one of the
|
||||||
|
setjmp()
|
||||||
|
routines is very fast. Typically, just 2 or 3 words
|
||||||
|
need to be saved in the jump\-buffer (plus one call to
|
||||||
|
sigprocmask(2),
|
||||||
|
in the case of sigsetjmp).
|
||||||
|
On the
|
||||||
|
other hand, executing a non\-local goto by calling one of the
|
||||||
|
longjmp()
|
||||||
|
routines tends to be much slower than with the
|
||||||
|
system\-provided routines. In fact, the time spent on a
|
||||||
|
longjmp()
|
||||||
|
will be proportional to the number of call frames
|
||||||
|
that exist between the points where setjmp()
|
||||||
|
and
|
||||||
|
longjmp()
|
||||||
|
were called. For this reason, the
|
||||||
|
unwind\-setjmp
|
||||||
|
library is beneficial primarily in applications
|
||||||
|
that frequently call setjmp()
|
||||||
|
but only rarely call
|
||||||
|
longjmp().
|
||||||
|
.PP
|
||||||
|
.SH CAVEATS
|
||||||
|
|
||||||
|
.PP
|
||||||
|
.TP
|
||||||
|
.B *
|
||||||
|
The correct operation of this library depends on the presence of
|
||||||
|
correct unwind information. On newer platforms, this is rarely an
|
||||||
|
issue. On older platforms, care needs to be taken to
|
||||||
|
ensure that each of the functions whose stack frames may have to be
|
||||||
|
unwound during a longjmp()
|
||||||
|
have correct unwind information
|
||||||
|
(on those platforms, there is usually a compiler\-switch, such as
|
||||||
|
\fB\-funwind\-tables\fP,
|
||||||
|
to request the generation of unwind
|
||||||
|
information).
|
||||||
|
.TP
|
||||||
|
.B *
|
||||||
|
The contents of jmp_buf and sigjmp_buf as setup
|
||||||
|
and used by these routines is completely different from the ones
|
||||||
|
used by the system\-provided routines. Thus, a jump\-buffer created
|
||||||
|
by the libunwind\-based setjmp()/_setjmp
|
||||||
|
may only be
|
||||||
|
used in a call to the libunwind\-based
|
||||||
|
longjmp()/_longjmp().
|
||||||
|
The analogous applies for
|
||||||
|
sigjmp_buf
|
||||||
|
with sigsetjmp()
|
||||||
|
and siglongjmp().
|
||||||
|
.PP
|
||||||
|
.SH FILES
|
||||||
|
|
||||||
|
.PP
|
||||||
|
.TP
|
||||||
|
\fB\-l\fPunwind\-setjmp
|
||||||
|
The library an application should
|
||||||
|
be linked against to ensure it uses the libunwind\-based non\-local
|
||||||
|
goto routines.
|
||||||
|
.PP
|
||||||
|
.SH SEE ALSO
|
||||||
|
|
||||||
|
.PP
|
||||||
|
libunwind(3),
|
||||||
|
setjmp(3), longjmp(3),
|
||||||
|
_setjmp(3), _longjmp(3),
|
||||||
|
sigsetjmp(3), siglongjmp(3)
|
||||||
|
.PP
|
||||||
|
.SH AUTHOR
|
||||||
|
|
||||||
|
.PP
|
||||||
|
David Mosberger\-Tang
|
||||||
|
.br
|
||||||
|
Hewlett\-Packard Labs
|
||||||
|
.br
|
||||||
|
Palo\-Alto, CA 94304
|
||||||
|
.br
|
||||||
|
Email: \fBdavidm@hpl.hp.com\fP
|
||||||
|
.br
|
||||||
|
WWW: \fBhttp://www.hpl.hp.com/research/linux/libunwind/\fP\&.
|
||||||
|
.\" NOTE: This file is generated, DO NOT EDIT.
|
|
@ -0,0 +1,90 @@
|
||||||
|
\documentclass{article}
|
||||||
|
\usepackage[fancyhdr,pdf]{latex2man}
|
||||||
|
|
||||||
|
\input{common.tex}
|
||||||
|
|
||||||
|
\begin{document}
|
||||||
|
|
||||||
|
\begin{Name}{3}{libunwind-setjmp}{David Mosberger-Tang}{Programming Library}{libunwind-based non-local gotos}
|
||||||
|
libunwind-setjmp -- libunwind-based non-local gotos
|
||||||
|
\end{Name}
|
||||||
|
|
||||||
|
\section{Synopsis}
|
||||||
|
|
||||||
|
\File{\#include $<$setjmp.h$>$}\\
|
||||||
|
|
||||||
|
\noindent
|
||||||
|
\Type{int} \Func{setjmp}(\Type{jmp\_buf}~\Var{env});\\
|
||||||
|
\Type{void} \Func{longjmp}(\Type{jmp\_buf}~\Var{env}, \Type{int}~\Var{val});\\
|
||||||
|
\Type{int} \Func{\_setjmp}(\Type{jmp\_buf}~\Var{env});\\
|
||||||
|
\Type{void} \Func{\_longjmp}(\Type{jmp\_buf}~\Var{env}, \Type{int}~\Var{val});\\
|
||||||
|
\Type{int} \Func{setjmp}(\Type{sigjmp\_buf}~\Var{env}, \Type{int}~\Var{savemask});\\
|
||||||
|
\Type{void} \Func{siglongjmp}(\Type{sigjmp\_buf}~\Var{env}, \Type{int}~\Var{val});\\
|
||||||
|
|
||||||
|
\section{Description}
|
||||||
|
|
||||||
|
The \Prog{unwind-setjmp} library offers a \Prog{libunwind}-based
|
||||||
|
implementation of non-local gotos. This implementation is intended to
|
||||||
|
be a drop-in replacement for the normal, system-provided routines of
|
||||||
|
the same name. The main advantage of using the \Prog{unwind-setjmp}
|
||||||
|
library is that setting up a non-local goto via one of the
|
||||||
|
\Func{setjmp}() routines is very fast. Typically, just 2 or 3 words
|
||||||
|
need to be saved in the jump-buffer (plus one call to
|
||||||
|
\Func{sigprocmask}(2), in the case of \Func{sigsetjmp}). On the
|
||||||
|
other hand, executing a non-local goto by calling one of the
|
||||||
|
\Func{longjmp}() routines tends to be much slower than with the
|
||||||
|
system-provided routines. In fact, the time spent on a
|
||||||
|
\Func{longjmp}() will be proportional to the number of call frames
|
||||||
|
that exist between the points where \Func{setjmp}() and
|
||||||
|
\Func{longjmp}() were called. For this reason, the
|
||||||
|
\Prog{unwind-setjmp} library is beneficial primarily in applications
|
||||||
|
that frequently call \Func{setjmp}() but only rarely call
|
||||||
|
\Func{longjmp}().
|
||||||
|
|
||||||
|
\section{Caveats}
|
||||||
|
|
||||||
|
\begin{itemize}
|
||||||
|
\item The correct operation of this library depends on the presence of
|
||||||
|
correct unwind information. On newer platforms, this is rarely an
|
||||||
|
issue. On older platforms, care needs to be taken to
|
||||||
|
ensure that each of the functions whose stack frames may have to be
|
||||||
|
unwound during a \Func{longjmp}() have correct unwind information
|
||||||
|
(on those platforms, there is usually a compiler-switch, such as
|
||||||
|
\Opt{-funwind-tables}, to request the generation of unwind
|
||||||
|
information).
|
||||||
|
\item The contents of \Type{jmp\_buf} and \Type{sigjmp\_buf} as setup
|
||||||
|
and used by these routines is completely different from the ones
|
||||||
|
used by the system-provided routines. Thus, a jump-buffer created
|
||||||
|
by the libunwind-based \Func{setjmp}()/\Func{\_setjmp} may only be
|
||||||
|
used in a call to the libunwind-based
|
||||||
|
\Func{longjmp}()/\Func{\_longjmp}(). The analogous applies for
|
||||||
|
\Type{sigjmp\_buf} with \Func{sigsetjmp}() and \Func{siglongjmp}().
|
||||||
|
\end{itemize}
|
||||||
|
|
||||||
|
\section{Files}
|
||||||
|
|
||||||
|
\begin{Description}
|
||||||
|
\item[\Opt{-l}\File{unwind-setjmp}] The library an application should
|
||||||
|
be linked against to ensure it uses the libunwind-based non-local
|
||||||
|
goto routines.
|
||||||
|
\end{Description}
|
||||||
|
|
||||||
|
|
||||||
|
\section{See Also}
|
||||||
|
|
||||||
|
\SeeAlso{libunwind(3)},
|
||||||
|
setjmp(3), longjmp(3),
|
||||||
|
\_setjmp(3), \_longjmp(3),
|
||||||
|
sigsetjmp(3), siglongjmp(3)
|
||||||
|
|
||||||
|
\section{Author}
|
||||||
|
|
||||||
|
\noindent
|
||||||
|
David Mosberger-Tang\\
|
||||||
|
Hewlett-Packard Labs\\
|
||||||
|
Palo-Alto, CA 94304\\
|
||||||
|
Email: \Email{davidm@hpl.hp.com}\\
|
||||||
|
WWW: \URL{http://www.hpl.hp.com/research/linux/libunwind/}.
|
||||||
|
\LatexManEnd
|
||||||
|
|
||||||
|
\end{document}
|
|
@ -0,0 +1,107 @@
|
||||||
|
'\" t
|
||||||
|
.\" Manual page created with latex2man on Sat Feb 8 01:21:05 PST 2003
|
||||||
|
.\" NOTE: This file is generated, DO NOT EDIT.
|
||||||
|
.de Vb
|
||||||
|
.ft CW
|
||||||
|
.nf
|
||||||
|
..
|
||||||
|
.de Ve
|
||||||
|
.ft R
|
||||||
|
|
||||||
|
.fi
|
||||||
|
..
|
||||||
|
.TH "UNW\\_RESUME" "3" "08 February 2003" "Programming Library " "Programming Library "
|
||||||
|
.SH NAME
|
||||||
|
|
||||||
|
.PP
|
||||||
|
unw_resume \-\- resume execution in a particular stack frame
|
||||||
|
.PP
|
||||||
|
.SH SYNOPSIS
|
||||||
|
|
||||||
|
.PP
|
||||||
|
#include <libunwind.h>
|
||||||
|
.br
|
||||||
|
.PP
|
||||||
|
int
|
||||||
|
unw_resume(unw_cursor_t *cursor);
|
||||||
|
.br
|
||||||
|
.PP
|
||||||
|
.SH DESCRIPTION
|
||||||
|
|
||||||
|
.PP
|
||||||
|
The unw_resume()
|
||||||
|
routine resumes execution at the stack frame
|
||||||
|
identified by cursor\&.
|
||||||
|
Normally, this is accomplished by
|
||||||
|
restoring the ``preserved\&'' (callee\-saved) machine state. However, if
|
||||||
|
execution in any of the stack frames younger (more deeply nested) than
|
||||||
|
the one identified by cursor
|
||||||
|
was interrupted by a signal, then
|
||||||
|
unw_resume()
|
||||||
|
will restore the entire machine state, including
|
||||||
|
the ``preserved\&'' and ``scratch\&'' (caller\-saved) registers, as well as
|
||||||
|
the signal mask.
|
||||||
|
.PP
|
||||||
|
Most platforms reserve some registers to pass arguments to exception
|
||||||
|
handlers (e.g., IA\-64 uses r15\-r18
|
||||||
|
for this
|
||||||
|
purpose). These registers are normally treated like ``scratch\&''
|
||||||
|
registers. However, if libunwind
|
||||||
|
is used to define an
|
||||||
|
exception argument register, e.g., by calling unw_set_reg(),
|
||||||
|
then unw_resume()
|
||||||
|
will always install the new value as the
|
||||||
|
contents of that register. In other words, the exception handling
|
||||||
|
arguments are installed even in cases where normally only the
|
||||||
|
``preserved\&'' registers are restored.
|
||||||
|
.PP
|
||||||
|
.SH RETURN VALUE
|
||||||
|
|
||||||
|
.PP
|
||||||
|
For local unwinding, unw_resume()
|
||||||
|
does not return on success.
|
||||||
|
For remote unwinding, it returns 0 on success. On failure, the
|
||||||
|
negative value of one of the errors below is returned.
|
||||||
|
.PP
|
||||||
|
.SH ERRORS
|
||||||
|
|
||||||
|
.PP
|
||||||
|
.TP
|
||||||
|
UNW_EUNSPEC
|
||||||
|
An unspecified error occurred.
|
||||||
|
.TP
|
||||||
|
UNW_EBADREG
|
||||||
|
A register needed by unw_resume()
|
||||||
|
wasn\&'t
|
||||||
|
accessible.
|
||||||
|
.TP
|
||||||
|
UNW_EINVALIDIP
|
||||||
|
The instruction pointer identified by
|
||||||
|
cursor
|
||||||
|
is not valid.
|
||||||
|
.TP
|
||||||
|
UNW_BADFRAME
|
||||||
|
The stack frame identified by
|
||||||
|
cursor
|
||||||
|
is not valid.
|
||||||
|
.PP
|
||||||
|
.SH SEE ALSO
|
||||||
|
|
||||||
|
.PP
|
||||||
|
libunwind(3),
|
||||||
|
unw_set_reg(3),
|
||||||
|
sigprocmask(2)
|
||||||
|
.PP
|
||||||
|
.SH AUTHOR
|
||||||
|
|
||||||
|
.PP
|
||||||
|
David Mosberger\-Tang
|
||||||
|
.br
|
||||||
|
Hewlett\-Packard Labs
|
||||||
|
.br
|
||||||
|
Palo\-Alto, CA 94304
|
||||||
|
.br
|
||||||
|
Email: \fBdavidm@hpl.hp.com\fP
|
||||||
|
.br
|
||||||
|
WWW: \fBhttp://www.hpl.hp.com/research/linux/libunwind/\fP\&.
|
||||||
|
.\" NOTE: This file is generated, DO NOT EDIT.
|
|
@ -0,0 +1,74 @@
|
||||||
|
\documentclass{article}
|
||||||
|
\usepackage[fancyhdr,pdf]{latex2man}
|
||||||
|
|
||||||
|
\input{common.tex}
|
||||||
|
|
||||||
|
\begin{document}
|
||||||
|
|
||||||
|
\begin{Name}{3}{unw\_resume}{David Mosberger-Tang}{Programming Library}{unw\_resume}
|
||||||
|
|
||||||
|
unw\_resume -- resume execution in a particular stack frame
|
||||||
|
\end{Name}
|
||||||
|
|
||||||
|
\section{Synopsis}
|
||||||
|
|
||||||
|
\File{\#include $<$libunwind.h$>$}\\
|
||||||
|
|
||||||
|
\Type{int} \Func{unw\_resume}(\Type{unw\_cursor\_t~*}\Var{cursor});\\
|
||||||
|
|
||||||
|
\section{Description}
|
||||||
|
|
||||||
|
The \Func{unw\_resume}() routine resumes execution at the stack frame
|
||||||
|
identified by \Var{cursor}. Normally, this is accomplished by
|
||||||
|
restoring the ``preserved'' (callee-saved) machine state. However, if
|
||||||
|
execution in any of the stack frames younger (more deeply nested) than
|
||||||
|
the one identified by \Var{cursor} was interrupted by a signal, then
|
||||||
|
\Func{unw\_resume}() will restore the entire machine state, including
|
||||||
|
the ``preserved'' and ``scratch'' (caller-saved) registers, as well as
|
||||||
|
the signal mask.
|
||||||
|
|
||||||
|
Most platforms reserve some registers to pass arguments to exception
|
||||||
|
handlers (e.g., IA-64 uses \texttt{r15}-\texttt{r18} for this
|
||||||
|
purpose). These registers are normally treated like ``scratch''
|
||||||
|
registers. However, if \Prog{libunwind} is used to define an
|
||||||
|
exception argument register, e.g., by calling \Func{unw\_set\_reg}(),
|
||||||
|
then \Func{unw\_resume}() will always install the new value as the
|
||||||
|
contents of that register. In other words, the exception handling
|
||||||
|
arguments are installed even in cases where normally only the
|
||||||
|
``preserved'' registers are restored.
|
||||||
|
|
||||||
|
\section{Return Value}
|
||||||
|
|
||||||
|
For local unwinding, \Func{unw\_resume}() does not return on success.
|
||||||
|
For remote unwinding, it returns 0 on success. On failure, the
|
||||||
|
negative value of one of the errors below is returned.
|
||||||
|
|
||||||
|
\section{Errors}
|
||||||
|
|
||||||
|
\begin{Description}
|
||||||
|
\item[\Const{UNW\_EUNSPEC}] An unspecified error occurred.
|
||||||
|
\item[\Const{UNW\_EBADREG}] A register needed by \Func{unw\_resume}() wasn't
|
||||||
|
accessible.
|
||||||
|
\item[\Const{UNW\_EINVALIDIP}] The instruction pointer identified by
|
||||||
|
\Var{cursor} is not valid.
|
||||||
|
\item[\Const{UNW\_BADFRAME}] The stack frame identified by
|
||||||
|
\Var{cursor} is not valid.
|
||||||
|
\end{Description}
|
||||||
|
|
||||||
|
\section{See Also}
|
||||||
|
|
||||||
|
\SeeAlso{libunwind(3)},
|
||||||
|
\SeeAlso{unw\_set\_reg(3)},
|
||||||
|
sigprocmask(2)
|
||||||
|
|
||||||
|
\section{Author}
|
||||||
|
|
||||||
|
\noindent
|
||||||
|
David Mosberger-Tang\\
|
||||||
|
Hewlett-Packard Labs\\
|
||||||
|
Palo-Alto, CA 94304\\
|
||||||
|
Email: \Email{davidm@hpl.hp.com}\\
|
||||||
|
WWW: \URL{http://www.hpl.hp.com/research/linux/libunwind/}.
|
||||||
|
\LatexManEnd
|
||||||
|
|
||||||
|
\end{document}
|
|
@ -0,0 +1,2 @@
|
||||||
|
#include "elf32.h"
|
||||||
|
#include "elfxx.c"
|
|
@ -0,0 +1,7 @@
|
||||||
|
#ifndef elf32_h
|
||||||
|
#define elf32_h
|
||||||
|
|
||||||
|
#define ELF_CLASS ELFCLASS32
|
||||||
|
#include "elfxx.h"
|
||||||
|
|
||||||
|
#endif /* elf32_h */
|
|
@ -0,0 +1,2 @@
|
||||||
|
#include "elf64.h"
|
||||||
|
#include "elfxx.c"
|
|
@ -0,0 +1,7 @@
|
||||||
|
#ifndef elf64_h
|
||||||
|
#define elf64_h
|
||||||
|
|
||||||
|
#define ELF_CLASS ELFCLASS64
|
||||||
|
#include "elfxx.h"
|
||||||
|
|
||||||
|
#endif /* elf64_h */
|
168
src/elfxx.c
168
src/elfxx.c
|
@ -0,0 +1,168 @@
|
||||||
|
/* libunwind - a platform-independent unwind library
|
||||||
|
Copyright (C) 2003 Hewlett-Packard Co
|
||||||
|
Contributed by David Mosberger-Tang <davidm@hpl.hp.com>
|
||||||
|
|
||||||
|
This file is part of libunwind.
|
||||||
|
|
||||||
|
Permission is hereby granted, free of charge, to any person obtaining
|
||||||
|
a copy of this software and associated documentation files (the
|
||||||
|
"Software"), to deal in the Software without restriction, including
|
||||||
|
without limitation the rights to use, copy, modify, merge, publish,
|
||||||
|
distribute, sublicense, and/or sell copies of the Software, and to
|
||||||
|
permit persons to whom the Software is furnished to do so, subject to
|
||||||
|
the following conditions:
|
||||||
|
|
||||||
|
The above copyright notice and this permission notice shall be
|
||||||
|
included in all copies or substantial portions of the Software.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
||||||
|
EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
||||||
|
MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
||||||
|
NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
|
||||||
|
LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
|
||||||
|
OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
||||||
|
WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */
|
||||||
|
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <string.h>
|
||||||
|
#include <unistd.h>
|
||||||
|
|
||||||
|
#include <sys/types.h>
|
||||||
|
|
||||||
|
|
||||||
|
extern HIDDEN int
|
||||||
|
elfW (valid_object) (struct elf_image *ei)
|
||||||
|
{
|
||||||
|
if (ei->size <= EI_CLASS)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
return (memcmp (ei->image, ELFMAG, SELFMAG) == 0
|
||||||
|
&& ((uint8_t *) ei->image)[EI_CLASS] == ELF_CLASS);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static int
|
||||||
|
elfW (lookup_symbol) (unw_word_t ip, struct elf_image *ei,
|
||||||
|
ElfW (Addr) load_offset,
|
||||||
|
char *buf, size_t buf_len, unw_word_t *offp)
|
||||||
|
{
|
||||||
|
size_t syment_size, str_size;
|
||||||
|
ElfW (Ehdr) *ehdr = ei->image;
|
||||||
|
ElfW (Sym) *sym, *symtab, *symtab_end;
|
||||||
|
ElfW (Off) soff, str_soff;
|
||||||
|
ElfW (Shdr) *shdr, *str_shdr;
|
||||||
|
ElfW (Addr) val, min_dist = ~(ElfW (Addr))0;
|
||||||
|
char *strtab;
|
||||||
|
int i;
|
||||||
|
|
||||||
|
if (!elfW (valid_object) (ei))
|
||||||
|
return -1;
|
||||||
|
|
||||||
|
soff = ehdr->e_shoff;
|
||||||
|
if (soff + ehdr->e_shnum * ehdr->e_shentsize > ei->size)
|
||||||
|
{
|
||||||
|
debug (1, "%s: section table outside of image? (%lu > %lu)\n",
|
||||||
|
__FUNCTION__, soff + ehdr->e_shnum * ehdr->e_shentsize,
|
||||||
|
ei->size);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
shdr = (ElfW (Shdr) *) ((char *) ei->image + soff);
|
||||||
|
|
||||||
|
for (i = 0; i < ehdr->e_shnum; ++i)
|
||||||
|
{
|
||||||
|
switch (shdr->sh_type)
|
||||||
|
{
|
||||||
|
case SHT_SYMTAB:
|
||||||
|
case SHT_DYNSYM:
|
||||||
|
symtab = (ElfW (Sym) *) ((char *) ei->image + shdr->sh_offset);
|
||||||
|
symtab_end = (ElfW (Sym) *) ((char *) symtab + shdr->sh_size);
|
||||||
|
syment_size = shdr->sh_entsize;
|
||||||
|
|
||||||
|
str_soff = soff + (shdr->sh_link * ehdr->e_shentsize);
|
||||||
|
if (str_soff + ehdr->e_shentsize >= ei->size)
|
||||||
|
{
|
||||||
|
debug (1, "%s: string table outside of image? (%lu >= %lu)\n",
|
||||||
|
__FUNCTION__, str_soff + ehdr->e_shentsize, ei->size);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
str_shdr = (ElfW (Shdr) *) ((char *) ei->image + str_soff);
|
||||||
|
str_size = str_shdr->sh_size;
|
||||||
|
strtab = (char *) ei->image + str_shdr->sh_offset;
|
||||||
|
|
||||||
|
debug (10, "symtab=0x%lx[%d], strtab=0x%lx\n", shdr->sh_offset,
|
||||||
|
shdr->sh_type, str_shdr->sh_offset);
|
||||||
|
|
||||||
|
for (sym = symtab;
|
||||||
|
sym < symtab_end;
|
||||||
|
sym = (ElfW (Sym) *) ((char *) sym + syment_size))
|
||||||
|
{
|
||||||
|
if (ELFW (ST_TYPE) (sym->st_info) == STT_FUNC
|
||||||
|
&& sym->st_shndx != SHN_UNDEF)
|
||||||
|
{
|
||||||
|
val = sym->st_value;
|
||||||
|
if (sym->st_shndx != SHN_ABS)
|
||||||
|
val += load_offset;
|
||||||
|
debug (100, "0x%016lx info=0x%02x %s\n",
|
||||||
|
val, sym->st_info, strtab + sym->st_name);
|
||||||
|
|
||||||
|
if ((ElfW (Addr)) (ip - val) < min_dist)
|
||||||
|
{
|
||||||
|
min_dist = (ElfW (Addr)) (ip - val);
|
||||||
|
buf[buf_len - 1] = 'x';
|
||||||
|
strncpy (buf, strtab + sym->st_name, buf_len);
|
||||||
|
buf[buf_len - 1] = '\0';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
shdr = (Elf64_Shdr *) (((char *) shdr) + ehdr->e_shentsize);
|
||||||
|
}
|
||||||
|
if (min_dist >= ei->size)
|
||||||
|
return -1; /* not found */
|
||||||
|
if (offp)
|
||||||
|
*offp = min_dist;
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Find the ELF image that contains IP and return the "closest"
|
||||||
|
procedure name, if there is one. With some caching, this could be
|
||||||
|
sped up greatly, but until an application materializes that's
|
||||||
|
sensitive to the performance of this routine, why bother... */
|
||||||
|
|
||||||
|
HIDDEN int
|
||||||
|
elfW (get_proc_name) (unw_word_t ip, char *buf, size_t buf_len,
|
||||||
|
unw_word_t *offp)
|
||||||
|
{
|
||||||
|
unsigned long segbase, mapoff;
|
||||||
|
ElfW (Addr) load_offset = 0;
|
||||||
|
struct elf_image ei;
|
||||||
|
ElfW (Ehdr) *ehdr;
|
||||||
|
ElfW (Phdr) *phdr;
|
||||||
|
int i, ret;
|
||||||
|
|
||||||
|
ret = tdep_get_elf_image (&ei, getpid (), ip, &segbase, &mapoff);
|
||||||
|
if (ret < 0)
|
||||||
|
return ret;
|
||||||
|
|
||||||
|
ehdr = ei.image;
|
||||||
|
phdr = (Elf64_Phdr *) ((char *) ei.image + ehdr->e_phoff);
|
||||||
|
|
||||||
|
for (i = 0; i < ehdr->e_phnum; ++i)
|
||||||
|
if (phdr[i].p_type == PT_LOAD && phdr[i].p_offset == mapoff)
|
||||||
|
{
|
||||||
|
load_offset = segbase - phdr[i].p_vaddr;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
ret = elfW (lookup_symbol) (ip, &ei, load_offset, buf, buf_len, offp);
|
||||||
|
|
||||||
|
munmap (ei.image, ei.size);
|
||||||
|
ei.image = NULL;
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
71
src/elfxx.h
71
src/elfxx.h
|
@ -0,0 +1,71 @@
|
||||||
|
/* libunwind - a platform-independent unwind library
|
||||||
|
Copyright (C) 2003 Hewlett-Packard Co
|
||||||
|
Contributed by David Mosberger-Tang <davidm@hpl.hp.com>
|
||||||
|
|
||||||
|
This file is part of libunwind.
|
||||||
|
|
||||||
|
Permission is hereby granted, free of charge, to any person obtaining
|
||||||
|
a copy of this software and associated documentation files (the
|
||||||
|
"Software"), to deal in the Software without restriction, including
|
||||||
|
without limitation the rights to use, copy, modify, merge, publish,
|
||||||
|
distribute, sublicense, and/or sell copies of the Software, and to
|
||||||
|
permit persons to whom the Software is furnished to do so, subject to
|
||||||
|
the following conditions:
|
||||||
|
|
||||||
|
The above copyright notice and this permission notice shall be
|
||||||
|
included in all copies or substantial portions of the Software.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
||||||
|
EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
||||||
|
MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
||||||
|
NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
|
||||||
|
LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
|
||||||
|
OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
||||||
|
WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */
|
||||||
|
|
||||||
|
#include <elf.h>
|
||||||
|
#include <fcntl.h>
|
||||||
|
#include <unistd.h>
|
||||||
|
|
||||||
|
#include <sys/mman.h>
|
||||||
|
|
||||||
|
#include "internal.h"
|
||||||
|
|
||||||
|
#if ELF_CLASS == ELFCLASS32
|
||||||
|
# define ELFW(x) ELF32_##x
|
||||||
|
# define ElfW(x) Elf32_##x
|
||||||
|
# define elfW(x) _Uelf32_##x
|
||||||
|
#else
|
||||||
|
# define ELFW(x) ELF64_##x
|
||||||
|
# define ElfW(x) Elf64_##x
|
||||||
|
# define elfW(x) _Uelf64_##x
|
||||||
|
#endif
|
||||||
|
|
||||||
|
static inline int
|
||||||
|
elf_map_image (struct elf_image *ei, char *path)
|
||||||
|
{
|
||||||
|
struct stat stat;
|
||||||
|
int fd;
|
||||||
|
|
||||||
|
fd = open (path, O_RDONLY);
|
||||||
|
if (fd < 0)
|
||||||
|
return -1;
|
||||||
|
|
||||||
|
if (fstat (fd, &stat) < 0)
|
||||||
|
{
|
||||||
|
close (fd);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
ei->size = stat.st_size;
|
||||||
|
ei->image = mmap (NULL, ei->size, PROT_READ, MAP_PRIVATE, fd, 0);
|
||||||
|
close (fd);
|
||||||
|
if (ei->image == MAP_FAILED)
|
||||||
|
return -1;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
extern HIDDEN int elfW (valid_object) (struct elf_image *ei);
|
||||||
|
extern HIDDEN int elfW (get_proc_name) (unw_word_t ip, char *buf, size_t len,
|
||||||
|
unw_word_t *offp);
|
|
@ -0,0 +1,104 @@
|
||||||
|
/* libunwind - a platform-independent unwind library
|
||||||
|
Copyright (C) 2003 Hewlett-Packard Co
|
||||||
|
Contributed by David Mosberger-Tang <davidm@hpl.hp.com>
|
||||||
|
|
||||||
|
This file is part of libunwind.
|
||||||
|
|
||||||
|
Permission is hereby granted, free of charge, to any person obtaining
|
||||||
|
a copy of this software and associated documentation files (the
|
||||||
|
"Software"), to deal in the Software without restriction, including
|
||||||
|
without limitation the rights to use, copy, modify, merge, publish,
|
||||||
|
distribute, sublicense, and/or sell copies of the Software, and to
|
||||||
|
permit persons to whom the Software is furnished to do so, subject to
|
||||||
|
the following conditions:
|
||||||
|
|
||||||
|
The above copyright notice and this permission notice shall be
|
||||||
|
included in all copies or substantial portions of the Software.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
||||||
|
EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
||||||
|
MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
||||||
|
NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
|
||||||
|
LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
|
||||||
|
OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
||||||
|
WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */
|
||||||
|
|
||||||
|
/* Utility to generate cursor_i.h. */
|
||||||
|
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <string.h>
|
||||||
|
|
||||||
|
#include "internal.h"
|
||||||
|
|
||||||
|
#ifdef offsetof
|
||||||
|
# undef offsetof
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#define offsetof(type,field) ((char *) &((type *) 0)->field - (char *) 0)
|
||||||
|
|
||||||
|
struct
|
||||||
|
{
|
||||||
|
const char name[256];
|
||||||
|
unsigned long value;
|
||||||
|
}
|
||||||
|
tab[] =
|
||||||
|
{
|
||||||
|
{ "IP_OFF", offsetof (struct cursor, ip) },
|
||||||
|
{ "PR_OFF", offsetof (struct cursor, pr) },
|
||||||
|
{ "BSP_OFF", offsetof (struct cursor, bsp) },
|
||||||
|
{ "PSP_OFF", offsetof (struct cursor, psp) },
|
||||||
|
{ "PFS_LOC_OFF", offsetof (struct cursor, pfs_loc) },
|
||||||
|
{ "RNAT_LOC_OFF", offsetof (struct cursor, rnat_loc) },
|
||||||
|
{ "UNAT_LOC_OFF", offsetof (struct cursor, unat_loc) },
|
||||||
|
{ "LC_LOC_OFF", offsetof (struct cursor, lc_loc) },
|
||||||
|
{ "FPSR_LOC_OFF", offsetof (struct cursor, fpsr_loc) },
|
||||||
|
{ "B1_LOC_OFF", offsetof (struct cursor, b1_loc) },
|
||||||
|
{ "B2_LOC_OFF", offsetof (struct cursor, b2_loc) },
|
||||||
|
{ "B3_LOC_OFF", offsetof (struct cursor, b3_loc) },
|
||||||
|
{ "B4_LOC_OFF", offsetof (struct cursor, b4_loc) },
|
||||||
|
{ "B5_LOC_OFF", offsetof (struct cursor, b5_loc) },
|
||||||
|
{ "F2_LOC_OFF", offsetof (struct cursor, f2_loc) },
|
||||||
|
{ "F3_LOC_OFF", offsetof (struct cursor, f3_loc) },
|
||||||
|
{ "F4_LOC_OFF", offsetof (struct cursor, f4_loc) },
|
||||||
|
{ "F5_LOC_OFF", offsetof (struct cursor, f5_loc) },
|
||||||
|
{ "FR_LOC_OFF", offsetof (struct cursor, fr_loc) },
|
||||||
|
{ "SIGCONTEXT_LOC_OFF", offsetof (struct cursor, sigcontext_loc) },
|
||||||
|
};
|
||||||
|
|
||||||
|
static const char *tabs = "\t\t\t\t\t\t\t\t\t\t";
|
||||||
|
|
||||||
|
int
|
||||||
|
main (int argc, char **argv)
|
||||||
|
{
|
||||||
|
const char *space;
|
||||||
|
int i, num_tabs;
|
||||||
|
size_t len;
|
||||||
|
|
||||||
|
printf ("#ifndef cursor_i_h\n");
|
||||||
|
printf ("#define cursor_i_h\n\n");
|
||||||
|
|
||||||
|
printf ("/*\n * DO NOT MODIFY\n *\n * This file was generated by "
|
||||||
|
"print_offsets.\n *\n */\n\n");
|
||||||
|
|
||||||
|
for (i = 0; i < (int) (sizeof (tab) / sizeof (tab[0])); ++i)
|
||||||
|
{
|
||||||
|
if (tab[i].name[0] == '\0')
|
||||||
|
printf ("\n");
|
||||||
|
else
|
||||||
|
{
|
||||||
|
len = strlen (tab[i].name);
|
||||||
|
|
||||||
|
num_tabs = (40 - len) / 8;
|
||||||
|
if (num_tabs <= 0)
|
||||||
|
space = " ";
|
||||||
|
else
|
||||||
|
space = strchr(tabs, '\0') - (40 - len) / 8;
|
||||||
|
|
||||||
|
printf ("#define %s%s%lu\t/* 0x%lx */\n",
|
||||||
|
tab[i].name, space, tab[i].value, tab[i].value);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
printf ("\n#endif /* cursor_i_h */\n");
|
||||||
|
return 0;
|
||||||
|
}
|
|
@ -0,0 +1,45 @@
|
||||||
|
/* libunwind - a platform-independent unwind library
|
||||||
|
Copyright (C) 2003 Hewlett-Packard Co
|
||||||
|
Contributed by David Mosberger-Tang <davidm@hpl.hp.com>
|
||||||
|
|
||||||
|
This file is part of libunwind.
|
||||||
|
|
||||||
|
Permission is hereby granted, free of charge, to any person obtaining
|
||||||
|
a copy of this software and associated documentation files (the
|
||||||
|
"Software"), to deal in the Software without restriction, including
|
||||||
|
without limitation the rights to use, copy, modify, merge, publish,
|
||||||
|
distribute, sublicense, and/or sell copies of the Software, and to
|
||||||
|
permit persons to whom the Software is furnished to do so, subject to
|
||||||
|
the following conditions:
|
||||||
|
|
||||||
|
The above copyright notice and this permission notice shall be
|
||||||
|
included in all copies or substantial portions of the Software.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
||||||
|
EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
||||||
|
MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
||||||
|
NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
|
||||||
|
LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
|
||||||
|
OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
||||||
|
WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */
|
||||||
|
|
||||||
|
.align 32
|
||||||
|
|
||||||
|
.global _setjmp
|
||||||
|
|
||||||
|
.proc _setjmp
|
||||||
|
|
||||||
|
_setjmp:
|
||||||
|
mov r2 = ar.bsp
|
||||||
|
st8 [r32] = r12 // jmp_buf[0] = sp
|
||||||
|
mov r3 = rp
|
||||||
|
|
||||||
|
adds r16 = 8, r32
|
||||||
|
adds r17 = 16, r32
|
||||||
|
mov r8 = 0
|
||||||
|
;;
|
||||||
|
st8 [r16] = r3 // jmp_buf[1] = rp
|
||||||
|
st8 [r17] = r2 // jmp_buf[2] = bsp
|
||||||
|
br.ret.sptk.many rp
|
||||||
|
|
||||||
|
.endp _setjmp
|
|
@ -0,0 +1,64 @@
|
||||||
|
/* libunwind - a platform-independent unwind library
|
||||||
|
Copyright (C) 2003 Hewlett-Packard Co
|
||||||
|
Contributed by David Mosberger-Tang <davidm@hpl.hp.com>
|
||||||
|
|
||||||
|
This file is part of libunwind.
|
||||||
|
|
||||||
|
Permission is hereby granted, free of charge, to any person obtaining
|
||||||
|
a copy of this software and associated documentation files (the
|
||||||
|
"Software"), to deal in the Software without restriction, including
|
||||||
|
without limitation the rights to use, copy, modify, merge, publish,
|
||||||
|
distribute, sublicense, and/or sell copies of the Software, and to
|
||||||
|
permit persons to whom the Software is furnished to do so, subject to
|
||||||
|
the following conditions:
|
||||||
|
|
||||||
|
The above copyright notice and this permission notice shall be
|
||||||
|
included in all copies or substantial portions of the Software.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
||||||
|
EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
||||||
|
MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
||||||
|
NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
|
||||||
|
LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
|
||||||
|
OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
||||||
|
WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */
|
||||||
|
|
||||||
|
#define SIG_SETMASK 2
|
||||||
|
|
||||||
|
.globl _UI_siglongjmp_cont
|
||||||
|
|
||||||
|
.align 32
|
||||||
|
.proc siglongjmp_continuation
|
||||||
|
siglongjmp_continuation:
|
||||||
|
_UI_siglongjmp_cont: // non-function label for {sig,}longjmp.c
|
||||||
|
.prologue
|
||||||
|
.save rp, r15
|
||||||
|
.body
|
||||||
|
nop 0
|
||||||
|
nop 0
|
||||||
|
br.call.sptk.many b6 = 1f
|
||||||
|
;;
|
||||||
|
.prologue
|
||||||
|
.save ar.pfs, r33
|
||||||
|
1: alloc loc1 = ar.pfs, 0, 3, 3, 0
|
||||||
|
/*
|
||||||
|
* Note: we can use the scratch stack are because the caller
|
||||||
|
* of sigsetjmp() by definition is not a leaf-procedure.
|
||||||
|
*/
|
||||||
|
st8 [sp] = r17 // store signal mask
|
||||||
|
.save rp, loc0
|
||||||
|
mov loc0 = r15 // final continuation point
|
||||||
|
;;
|
||||||
|
.body
|
||||||
|
mov loc2 = r16 // value to return in r8
|
||||||
|
|
||||||
|
mov out0 = SIG_SETMASK
|
||||||
|
mov out1 = sp
|
||||||
|
mov out2 = r0
|
||||||
|
br.call.sptk.many rp = sigprocmask
|
||||||
|
;;
|
||||||
|
mov rp = loc0
|
||||||
|
mov ar.pfs = loc1
|
||||||
|
mov r8 = loc2
|
||||||
|
br.ret.sptk.many rp
|
||||||
|
.endp siglongjmp_continuation
|
|
@ -0,0 +1,64 @@
|
||||||
|
/* libunwind - a platform-independent unwind library
|
||||||
|
Copyright (C) 2003 Hewlett-Packard Co
|
||||||
|
Contributed by David Mosberger-Tang <davidm@hpl.hp.com>
|
||||||
|
|
||||||
|
This file is part of libunwind.
|
||||||
|
|
||||||
|
Permission is hereby granted, free of charge, to any person obtaining
|
||||||
|
a copy of this software and associated documentation files (the
|
||||||
|
"Software"), to deal in the Software without restriction, including
|
||||||
|
without limitation the rights to use, copy, modify, merge, publish,
|
||||||
|
distribute, sublicense, and/or sell copies of the Software, and to
|
||||||
|
permit persons to whom the Software is furnished to do so, subject to
|
||||||
|
the following conditions:
|
||||||
|
|
||||||
|
The above copyright notice and this permission notice shall be
|
||||||
|
included in all copies or substantial portions of the Software.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
||||||
|
EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
||||||
|
MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
||||||
|
NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
|
||||||
|
LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
|
||||||
|
OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
||||||
|
WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */
|
||||||
|
|
||||||
|
#define SIG_BLOCK 0
|
||||||
|
|
||||||
|
.align 32
|
||||||
|
|
||||||
|
.global __sigsetjmp
|
||||||
|
|
||||||
|
.proc __sigsetjmp
|
||||||
|
|
||||||
|
__sigsetjmp:
|
||||||
|
.prologue
|
||||||
|
.save ar.pfs, r35
|
||||||
|
alloc loc1 = ar.pfs, 2, 3, 3, 0
|
||||||
|
add r16 = 16, in0
|
||||||
|
add out2 = 24, in0
|
||||||
|
|
||||||
|
st8 [in0] = sp, 8 // sigjmp_buf[0] = sp
|
||||||
|
mov out0 = SIG_BLOCK
|
||||||
|
.save rp, loc0
|
||||||
|
mov loc0 = rp
|
||||||
|
.body
|
||||||
|
;;
|
||||||
|
st8 [in0] = loc0, 24 // sigjmp_buf[1] = rp
|
||||||
|
st8 [r16] = in1 // sigjmp_buf[2] = savemask
|
||||||
|
cmp.ne p6, p0 = in1, r0
|
||||||
|
|
||||||
|
mov out1 = r0
|
||||||
|
mov loc2 = ar.bsp
|
||||||
|
(p6) br.call.sptk.many rp = sigprocmask // sigjmp_buf[3] = sigmask
|
||||||
|
;;
|
||||||
|
|
||||||
|
st8 [in0] = loc2 // sigjmp_buf[4] = bsp
|
||||||
|
mov rp = loc0
|
||||||
|
nop 0
|
||||||
|
|
||||||
|
mov r8 = 0
|
||||||
|
mov.i ar.pfs = loc1
|
||||||
|
br.ret.sptk.many rp
|
||||||
|
|
||||||
|
.endp __sigsetjmp
|
109
src/longjmp.c
109
src/longjmp.c
|
@ -0,0 +1,109 @@
|
||||||
|
/* libunwind - a platform-independent unwind library
|
||||||
|
Copyright (C) 2003 Hewlett-Packard Co
|
||||||
|
Contributed by David Mosberger-Tang <davidm@hpl.hp.com>
|
||||||
|
|
||||||
|
This file is part of libunwind.
|
||||||
|
|
||||||
|
Permission is hereby granted, free of charge, to any person obtaining
|
||||||
|
a copy of this software and associated documentation files (the
|
||||||
|
"Software"), to deal in the Software without restriction, including
|
||||||
|
without limitation the rights to use, copy, modify, merge, publish,
|
||||||
|
distribute, sublicense, and/or sell copies of the Software, and to
|
||||||
|
permit persons to whom the Software is furnished to do so, subject to
|
||||||
|
the following conditions:
|
||||||
|
|
||||||
|
The above copyright notice and this permission notice shall be
|
||||||
|
included in all copies or substantial portions of the Software.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
||||||
|
EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
||||||
|
MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
||||||
|
NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
|
||||||
|
LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
|
||||||
|
OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
||||||
|
WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */
|
||||||
|
|
||||||
|
#define UNW_LOCAL_ONLY
|
||||||
|
|
||||||
|
#include <libunwind.h>
|
||||||
|
#include <setjmp.h>
|
||||||
|
#include <signal.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
|
||||||
|
#if UNW_TARGET_IA64
|
||||||
|
# include "ia64/rse.h"
|
||||||
|
#endif
|
||||||
|
|
||||||
|
void
|
||||||
|
_longjmp (jmp_buf env, int val)
|
||||||
|
{
|
||||||
|
extern int _UI_siglongjmp_cont;
|
||||||
|
sigset_t current_mask;
|
||||||
|
unw_context_t uc;
|
||||||
|
unw_cursor_t c;
|
||||||
|
unw_word_t sp;
|
||||||
|
unw_word_t *wp = (unw_word_t *) env;
|
||||||
|
|
||||||
|
if (unw_getcontext (&uc) < 0 || unw_init_local (&c, &uc) < 0)
|
||||||
|
abort ();
|
||||||
|
|
||||||
|
do
|
||||||
|
{
|
||||||
|
if (unw_get_reg (&c, UNW_REG_SP, &sp) < 0)
|
||||||
|
abort ();
|
||||||
|
if (sp != wp[0])
|
||||||
|
continue;
|
||||||
|
|
||||||
|
#if UNW_TARGET_IA64
|
||||||
|
{
|
||||||
|
unw_word_t bsp, pfs, sol;
|
||||||
|
|
||||||
|
if (unw_get_reg (&c, UNW_IA64_BSP, &bsp) < 0
|
||||||
|
|| unw_get_reg (&c, UNW_IA64_AR_PFS, &pfs) < 0)
|
||||||
|
abort ();
|
||||||
|
|
||||||
|
/* simulate the effect of "br.call setjmp" on ar.bsp: */
|
||||||
|
sol = (pfs >> 7) & 0x7f;
|
||||||
|
bsp = ia64_rse_skip_regs (bsp, sol);
|
||||||
|
|
||||||
|
if (bsp != wp[2])
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/* found the right frame: */
|
||||||
|
|
||||||
|
if (sigprocmask (SIG_BLOCK, NULL, ¤t_mask) < 0)
|
||||||
|
abort ();
|
||||||
|
|
||||||
|
if (unw_set_reg (&c, UNW_REG_EH_ARG0, wp[1]) < 0
|
||||||
|
|| unw_set_reg (&c, UNW_REG_EH_ARG1, val) < 0
|
||||||
|
|| unw_set_reg (&c, UNW_REG_EH_ARG2,
|
||||||
|
((unw_word_t *) ¤t_mask)[0]) < 0
|
||||||
|
|| unw_set_reg (&c, UNW_REG_IP,
|
||||||
|
(unw_word_t) &_UI_siglongjmp_cont))
|
||||||
|
abort ();
|
||||||
|
|
||||||
|
if (_NSIG > 8 * sizeof (unw_word_t))
|
||||||
|
{
|
||||||
|
if (_NSIG > 16 * sizeof (unw_word_t))
|
||||||
|
abort ();
|
||||||
|
if (unw_set_reg (&c, UNW_REG_EH_ARG3,
|
||||||
|
((unw_word_t *) ¤t_mask)[1]) < 0)
|
||||||
|
abort ();
|
||||||
|
}
|
||||||
|
|
||||||
|
unw_resume (&c);
|
||||||
|
|
||||||
|
abort ();
|
||||||
|
}
|
||||||
|
while (unw_step (&c) >= 0);
|
||||||
|
|
||||||
|
abort ();
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
longjmp (jmp_buf env, int val)
|
||||||
|
{
|
||||||
|
_longjmp (env, val);
|
||||||
|
}
|
|
@ -0,0 +1,63 @@
|
||||||
|
/* libunwind - a platform-independent unwind library
|
||||||
|
Copyright (C) 2003 Hewlett-Packard Co
|
||||||
|
Contributed by David Mosberger-Tang <davidm@hpl.hp.com>
|
||||||
|
|
||||||
|
This file is part of libunwind.
|
||||||
|
|
||||||
|
Permission is hereby granted, free of charge, to any person obtaining
|
||||||
|
a copy of this software and associated documentation files (the
|
||||||
|
"Software"), to deal in the Software without restriction, including
|
||||||
|
without limitation the rights to use, copy, modify, merge, publish,
|
||||||
|
distribute, sublicense, and/or sell copies of the Software, and to
|
||||||
|
permit persons to whom the Software is furnished to do so, subject to
|
||||||
|
the following conditions:
|
||||||
|
|
||||||
|
The above copyright notice and this permission notice shall be
|
||||||
|
included in all copies or substantial portions of the Software.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
||||||
|
EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
||||||
|
MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
||||||
|
NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
|
||||||
|
LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
|
||||||
|
OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
||||||
|
WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */
|
||||||
|
|
||||||
|
#ifndef UNW_REMOTE_ONLY
|
||||||
|
|
||||||
|
#include <limits.h>
|
||||||
|
#include <stdio.h>
|
||||||
|
|
||||||
|
#include "internal.h"
|
||||||
|
#include "os-linux.h"
|
||||||
|
|
||||||
|
/* Here, it doesn't matter whether we include elf64.h or elf32.h.
|
||||||
|
Both define "struct elf_image" and elf_map_image() in an identical
|
||||||
|
fashion. */
|
||||||
|
#include "elf64.h"
|
||||||
|
|
||||||
|
HIDDEN int
|
||||||
|
tdep_get_elf_image (struct elf_image *ei, pid_t pid, unw_word_t ip,
|
||||||
|
unsigned long *segbase, unsigned long *mapoff)
|
||||||
|
{
|
||||||
|
struct map_iterator mi;
|
||||||
|
char path[PATH_MAX];
|
||||||
|
int found = 0;
|
||||||
|
unsigned long hi;
|
||||||
|
|
||||||
|
maps_init (&mi, pid);
|
||||||
|
while (maps_next (&mi, segbase, &hi, mapoff, path))
|
||||||
|
if (ip >= *segbase && ip < hi)
|
||||||
|
{
|
||||||
|
found = 1;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
maps_close (&mi);
|
||||||
|
|
||||||
|
if (!found)
|
||||||
|
return -1;
|
||||||
|
|
||||||
|
return elf_map_image (ei, path);
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif /* UNW_REMOTE_ONLY */
|
|
@ -0,0 +1,69 @@
|
||||||
|
/* libunwind - a platform-independent unwind library
|
||||||
|
Copyright (C) 2003 Hewlett-Packard Co
|
||||||
|
Contributed by David Mosberger-Tang <davidm@hpl.hp.com>
|
||||||
|
|
||||||
|
This file is part of libunwind.
|
||||||
|
|
||||||
|
Permission is hereby granted, free of charge, to any person obtaining
|
||||||
|
a copy of this software and associated documentation files (the
|
||||||
|
"Software"), to deal in the Software without restriction, including
|
||||||
|
without limitation the rights to use, copy, modify, merge, publish,
|
||||||
|
distribute, sublicense, and/or sell copies of the Software, and to
|
||||||
|
permit persons to whom the Software is furnished to do so, subject to
|
||||||
|
the following conditions:
|
||||||
|
|
||||||
|
The above copyright notice and this permission notice shall be
|
||||||
|
included in all copies or substantial portions of the Software.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
||||||
|
EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
||||||
|
MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
||||||
|
NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
|
||||||
|
LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
|
||||||
|
OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
||||||
|
WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */
|
||||||
|
|
||||||
|
#ifndef os_linux_h
|
||||||
|
#define os_linux_h
|
||||||
|
|
||||||
|
struct map_iterator
|
||||||
|
{
|
||||||
|
FILE *fp;
|
||||||
|
};
|
||||||
|
|
||||||
|
static inline void
|
||||||
|
maps_init (struct map_iterator *mi, pid_t pid)
|
||||||
|
{
|
||||||
|
char path[PATH_MAX];
|
||||||
|
|
||||||
|
snprintf (path, sizeof (path), "/proc/%d/maps", pid);
|
||||||
|
mi->fp = fopen (path, "r");
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline int
|
||||||
|
maps_next (struct map_iterator *mi,
|
||||||
|
unsigned long *low, unsigned long *high, unsigned long *offset,
|
||||||
|
char *path)
|
||||||
|
{
|
||||||
|
char line[256+PATH_MAX];
|
||||||
|
|
||||||
|
if (!mi->fp)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
while (fgets (line, sizeof (line), mi->fp))
|
||||||
|
{
|
||||||
|
if (sscanf (line, "%lx-%lx %*4c %lx %*x:%*x %*d %s\n",
|
||||||
|
low, high, offset, path) == 4)
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline void
|
||||||
|
maps_close (struct map_iterator *mi)
|
||||||
|
{
|
||||||
|
fclose (mi->fp);
|
||||||
|
mi->fp = NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif /* os_linux_h */
|
52
src/setjmp.c
52
src/setjmp.c
|
@ -0,0 +1,52 @@
|
||||||
|
/* libunwind - a platform-independent unwind library
|
||||||
|
Copyright (C) 2003 Hewlett-Packard Co
|
||||||
|
Contributed by David Mosberger-Tang <davidm@hpl.hp.com>
|
||||||
|
|
||||||
|
This file is part of libunwind.
|
||||||
|
|
||||||
|
Permission is hereby granted, free of charge, to any person obtaining
|
||||||
|
a copy of this software and associated documentation files (the
|
||||||
|
"Software"), to deal in the Software without restriction, including
|
||||||
|
without limitation the rights to use, copy, modify, merge, publish,
|
||||||
|
distribute, sublicense, and/or sell copies of the Software, and to
|
||||||
|
permit persons to whom the Software is furnished to do so, subject to
|
||||||
|
the following conditions:
|
||||||
|
|
||||||
|
The above copyright notice and this permission notice shall be
|
||||||
|
included in all copies or substantial portions of the Software.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
||||||
|
EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
||||||
|
MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
||||||
|
NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
|
||||||
|
LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
|
||||||
|
OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
||||||
|
WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */
|
||||||
|
|
||||||
|
#include <libunwind.h>
|
||||||
|
#include <setjmp.h>
|
||||||
|
|
||||||
|
/* Why use K&R syntax here? setjmp() is often a macro and that
|
||||||
|
expands into a call to, say, __setjmp() and we need to define the
|
||||||
|
libunwind-version of setjmp() with the name of the actual function.
|
||||||
|
Using K&R syntax lets us keep the setjmp() macro while keeping the
|
||||||
|
syntax valid... This trick works provided setjmp() doesn't do
|
||||||
|
anything other than a function call. */
|
||||||
|
|
||||||
|
int
|
||||||
|
setjmp (env)
|
||||||
|
jmp_buf env;
|
||||||
|
{
|
||||||
|
void **wp = (void **) env;
|
||||||
|
|
||||||
|
#if UNW_TARGET_IA64
|
||||||
|
wp[0] = __builtin_dwarf_cfa () - 16;
|
||||||
|
wp[1] = __builtin_ia64_bsp ();
|
||||||
|
#else
|
||||||
|
/* this should work on most platforms, but may not be
|
||||||
|
performance-optimal; check the code! */
|
||||||
|
wp[0] = __builtin_frame_address (0);
|
||||||
|
wp[1] = (void *) (uintptr_t) 0;
|
||||||
|
#endif
|
||||||
|
return 0;
|
||||||
|
}
|
108
src/siglongjmp.c
108
src/siglongjmp.c
|
@ -0,0 +1,108 @@
|
||||||
|
/* libunwind - a platform-independent unwind library
|
||||||
|
Copyright (C) 2003 Hewlett-Packard Co
|
||||||
|
Contributed by David Mosberger-Tang <davidm@hpl.hp.com>
|
||||||
|
|
||||||
|
This file is part of libunwind.
|
||||||
|
|
||||||
|
Permission is hereby granted, free of charge, to any person obtaining
|
||||||
|
a copy of this software and associated documentation files (the
|
||||||
|
"Software"), to deal in the Software without restriction, including
|
||||||
|
without limitation the rights to use, copy, modify, merge, publish,
|
||||||
|
distribute, sublicense, and/or sell copies of the Software, and to
|
||||||
|
permit persons to whom the Software is furnished to do so, subject to
|
||||||
|
the following conditions:
|
||||||
|
|
||||||
|
The above copyright notice and this permission notice shall be
|
||||||
|
included in all copies or substantial portions of the Software.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
||||||
|
EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
||||||
|
MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
||||||
|
NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
|
||||||
|
LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
|
||||||
|
OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
||||||
|
WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */
|
||||||
|
|
||||||
|
#define UNW_LOCAL_ONLY
|
||||||
|
|
||||||
|
#include <libunwind.h>
|
||||||
|
#include <setjmp.h>
|
||||||
|
#include <signal.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
|
||||||
|
#if UNW_TARGET_IA64
|
||||||
|
# include "ia64/rse.h"
|
||||||
|
#endif
|
||||||
|
|
||||||
|
void
|
||||||
|
siglongjmp (sigjmp_buf env, int val)
|
||||||
|
{
|
||||||
|
unw_word_t *mp, *wp = (unw_word_t *) env;
|
||||||
|
extern int _UI_siglongjmp_cont;
|
||||||
|
sigset_t current_mask;
|
||||||
|
unw_context_t uc;
|
||||||
|
unw_cursor_t c;
|
||||||
|
unw_word_t sp;
|
||||||
|
|
||||||
|
if (unw_getcontext (&uc) < 0 || unw_init_local (&c, &uc) < 0)
|
||||||
|
abort ();
|
||||||
|
|
||||||
|
do
|
||||||
|
{
|
||||||
|
if (unw_get_reg (&c, UNW_REG_SP, &sp) < 0)
|
||||||
|
abort ();
|
||||||
|
if (sp != wp[0])
|
||||||
|
continue;
|
||||||
|
|
||||||
|
#if UNW_TARGET_IA64
|
||||||
|
{
|
||||||
|
unw_word_t bsp, pfs, sol;
|
||||||
|
|
||||||
|
if (unw_get_reg (&c, UNW_IA64_BSP, &bsp) < 0
|
||||||
|
|| unw_get_reg (&c, UNW_IA64_AR_PFS, &pfs) < 0)
|
||||||
|
abort ();
|
||||||
|
|
||||||
|
/* simulate the effect of "br.call sigsetjmp" on ar.bsp: */
|
||||||
|
sol = (pfs >> 7) & 0x7f;
|
||||||
|
bsp = ia64_rse_skip_regs (bsp, sol);
|
||||||
|
|
||||||
|
if (bsp != wp[4])
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/* found the right frame: */
|
||||||
|
|
||||||
|
if (wp[2])
|
||||||
|
mp = wp + 3;
|
||||||
|
else
|
||||||
|
{
|
||||||
|
/* sigmask wasn't saved; get it now so we can leave it unchanged. */
|
||||||
|
if (sigprocmask (SIG_BLOCK, NULL, ¤t_mask) < 0)
|
||||||
|
abort ();
|
||||||
|
mp = (unw_word_t *) ¤t_mask;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (unw_set_reg (&c, UNW_REG_EH_ARG0, wp[1]) < 0
|
||||||
|
|| unw_set_reg (&c, UNW_REG_EH_ARG1, val) < 0
|
||||||
|
|| unw_set_reg (&c, UNW_REG_EH_ARG2, mp[0]) < 0
|
||||||
|
|| unw_set_reg (&c, UNW_REG_IP,
|
||||||
|
(unw_word_t) &_UI_siglongjmp_cont))
|
||||||
|
abort ();
|
||||||
|
|
||||||
|
if (_NSIG > 8 * sizeof (unw_word_t))
|
||||||
|
{
|
||||||
|
if (_NSIG > 16 * sizeof (unw_word_t))
|
||||||
|
abort ();
|
||||||
|
if (unw_set_reg (&c, UNW_REG_EH_ARG3, mp[1]) < 0)
|
||||||
|
abort ();
|
||||||
|
}
|
||||||
|
|
||||||
|
unw_resume (&c);
|
||||||
|
|
||||||
|
abort ();
|
||||||
|
}
|
||||||
|
while (unw_step (&c) >= 0);
|
||||||
|
|
||||||
|
abort ();
|
||||||
|
}
|
|
@ -0,0 +1,47 @@
|
||||||
|
/* libunwind - a platform-independent unwind library
|
||||||
|
Copyright (C) 2003 Hewlett-Packard Co
|
||||||
|
Contributed by David Mosberger-Tang <davidm@hpl.hp.com>
|
||||||
|
|
||||||
|
This file is part of libunwind.
|
||||||
|
|
||||||
|
Permission is hereby granted, free of charge, to any person obtaining
|
||||||
|
a copy of this software and associated documentation files (the
|
||||||
|
"Software"), to deal in the Software without restriction, including
|
||||||
|
without limitation the rights to use, copy, modify, merge, publish,
|
||||||
|
distribute, sublicense, and/or sell copies of the Software, and to
|
||||||
|
permit persons to whom the Software is furnished to do so, subject to
|
||||||
|
the following conditions:
|
||||||
|
|
||||||
|
The above copyright notice and this permission notice shall be
|
||||||
|
included in all copies or substantial portions of the Software.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
||||||
|
EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
||||||
|
MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
||||||
|
NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
|
||||||
|
LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
|
||||||
|
OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
||||||
|
WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */
|
||||||
|
|
||||||
|
#include <libunwind.h>
|
||||||
|
#include <setjmp.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
|
||||||
|
int
|
||||||
|
sigsetjmp (sigjmp_buf env, int savemask)
|
||||||
|
{
|
||||||
|
unw_word_t *wp = (unw_word_t *) env;
|
||||||
|
|
||||||
|
/* This should work on most platforms, but may not be
|
||||||
|
performance-optimal; check the code! */
|
||||||
|
|
||||||
|
wp[0] = (unw_word_t) __builtin_frame_address (0);
|
||||||
|
wp[1] = (unw_word_t) __builtin_return_address (0);
|
||||||
|
wp[2] = savemask;
|
||||||
|
|
||||||
|
/* Note: we assume here that "wp" has same or better alignment as
|
||||||
|
sigset_t. */
|
||||||
|
if (savemask && sigprocmask (SIG_BLOCK, NULL, (sigset_t *) (wp + 3)) < 0)
|
||||||
|
abort ();
|
||||||
|
return 0;
|
||||||
|
}
|
|
@ -0,0 +1,278 @@
|
||||||
|
/* libunwind - a platform-independent unwind library
|
||||||
|
Copyright (C) 2003 Hewlett-Packard Co
|
||||||
|
Contributed by David Mosberger-Tang <davidm@hpl.hp.com>
|
||||||
|
|
||||||
|
Permission is hereby granted, free of charge, to any person obtaining
|
||||||
|
a copy of this software and associated documentation files (the
|
||||||
|
"Software"), to deal in the Software without restriction, including
|
||||||
|
without limitation the rights to use, copy, modify, merge, publish,
|
||||||
|
distribute, sublicense, and/or sell copies of the Software, and to
|
||||||
|
permit persons to whom the Software is furnished to do so, subject to
|
||||||
|
the following conditions:
|
||||||
|
|
||||||
|
The above copyright notice and this permission notice shall be
|
||||||
|
included in all copies or substantial portions of the Software.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
||||||
|
EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
||||||
|
MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
||||||
|
NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
|
||||||
|
LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
|
||||||
|
OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
||||||
|
WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */
|
||||||
|
|
||||||
|
/* The setjmp()/longjmp(), sigsetjmp()/siglongjmp(). */
|
||||||
|
|
||||||
|
#include <setjmp.h>
|
||||||
|
#include <signal.h>
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <string.h>
|
||||||
|
#include <unistd.h>
|
||||||
|
|
||||||
|
int nerrors;
|
||||||
|
int verbose;
|
||||||
|
|
||||||
|
static jmp_buf jbuf;
|
||||||
|
static sigjmp_buf sigjbuf;
|
||||||
|
static sigset_t sigset4;
|
||||||
|
|
||||||
|
void
|
||||||
|
raise_longjmp (jmp_buf jbuf, int i, int n)
|
||||||
|
{
|
||||||
|
while (i < n)
|
||||||
|
raise_longjmp (jbuf, i + 1, n);
|
||||||
|
|
||||||
|
longjmp (jbuf, n);
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
test_setjmp (void)
|
||||||
|
{
|
||||||
|
volatile int i;
|
||||||
|
jmp_buf jbuf;
|
||||||
|
int ret;
|
||||||
|
|
||||||
|
for (i = 0; i < 10; ++i)
|
||||||
|
{
|
||||||
|
if ((ret = setjmp (jbuf)))
|
||||||
|
{
|
||||||
|
if (verbose)
|
||||||
|
printf ("%s: secondary setjmp () return, ret=%d\n",
|
||||||
|
__FUNCTION__, ret);
|
||||||
|
if (ret != i + 1)
|
||||||
|
{
|
||||||
|
fprintf (stderr, "%s: setjmp() returned %d, expected %d\n",
|
||||||
|
__FUNCTION__, ret, i + 1);
|
||||||
|
++nerrors;
|
||||||
|
}
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if (verbose)
|
||||||
|
printf ("%s.%d: done with setjmp(); calling children\n",
|
||||||
|
__FUNCTION__, i + 1);
|
||||||
|
|
||||||
|
raise_longjmp (jbuf, 0, i + 1);
|
||||||
|
|
||||||
|
fprintf (stderr, "%s: raise_longjmp() returned unexpected\n",
|
||||||
|
__FUNCTION__);
|
||||||
|
++nerrors;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void
|
||||||
|
raise_siglongjmp (sigjmp_buf jbuf, int i, int n)
|
||||||
|
{
|
||||||
|
while (i < n)
|
||||||
|
raise_siglongjmp (jbuf, i + 1, n);
|
||||||
|
|
||||||
|
siglongjmp (jbuf, n);
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
test_sigsetjmp (void)
|
||||||
|
{
|
||||||
|
sigjmp_buf jbuf;
|
||||||
|
int i, ret;
|
||||||
|
|
||||||
|
for (i = 0; i < 10; ++i)
|
||||||
|
{
|
||||||
|
if ((ret = sigsetjmp (jbuf, 1)))
|
||||||
|
{
|
||||||
|
if (verbose)
|
||||||
|
printf ("%s: secondary sigsetjmp () return, ret=%d\n",
|
||||||
|
__FUNCTION__, ret);
|
||||||
|
if (ret != i + 1)
|
||||||
|
{
|
||||||
|
fprintf (stderr, "%s: sigsetjmp() returned %d, expected %d\n",
|
||||||
|
__FUNCTION__, ret, i + 1);
|
||||||
|
++nerrors;
|
||||||
|
}
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if (verbose)
|
||||||
|
printf ("%s.%d: done with sigsetjmp(); calling children\n",
|
||||||
|
__FUNCTION__, i + 1);
|
||||||
|
|
||||||
|
raise_siglongjmp (jbuf, 0, i + 1);
|
||||||
|
|
||||||
|
fprintf (stderr, "%s: raise_siglongjmp() returned unexpected\n",
|
||||||
|
__FUNCTION__);
|
||||||
|
++nerrors;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
sighandler (int signal)
|
||||||
|
{
|
||||||
|
if (verbose)
|
||||||
|
printf ("%s: got signal %d\n", __FUNCTION__, signal);
|
||||||
|
|
||||||
|
sigprocmask (SIG_BLOCK, NULL, (sigset_t *) &sigset4);
|
||||||
|
siglongjmp (sigjbuf, 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
int
|
||||||
|
main (int argc, char **argv)
|
||||||
|
{
|
||||||
|
volatile sigset_t sigset1, sigset2, sigset3;
|
||||||
|
volatile struct sigaction act;
|
||||||
|
|
||||||
|
if (argc > 1)
|
||||||
|
verbose = 1;
|
||||||
|
|
||||||
|
sigemptyset ((sigset_t *) &sigset1);
|
||||||
|
sigaddset ((sigset_t *) &sigset1, SIGUSR1);
|
||||||
|
sigemptyset ((sigset_t *) &sigset2);
|
||||||
|
sigaddset ((sigset_t *) &sigset2, SIGUSR2);
|
||||||
|
|
||||||
|
memset ((void *) &act, 0, sizeof (act));
|
||||||
|
act.sa_handler = sighandler;
|
||||||
|
sigaction (SIGTERM, (struct sigaction *) &act, NULL);
|
||||||
|
|
||||||
|
test_setjmp ();
|
||||||
|
test_sigsetjmp ();
|
||||||
|
|
||||||
|
/* _setjmp() MUST NOT change signal mask: */
|
||||||
|
sigprocmask (SIG_SETMASK, (sigset_t *) &sigset1, NULL);
|
||||||
|
if (_setjmp (jbuf))
|
||||||
|
{
|
||||||
|
sigemptyset ((sigset_t *) &sigset3);
|
||||||
|
sigprocmask (SIG_BLOCK, NULL, (sigset_t *) &sigset3);
|
||||||
|
if (memcmp ((sigset_t *) &sigset3, (sigset_t *) &sigset2,
|
||||||
|
sizeof (sigset_t)) != 0)
|
||||||
|
{
|
||||||
|
fprintf (stderr, "FAILURE: _longjmp() manipulated signal mask!\n");
|
||||||
|
++nerrors;
|
||||||
|
}
|
||||||
|
else if (verbose)
|
||||||
|
printf ("OK: _longjmp() seems not to change signal mask\n");
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
sigprocmask (SIG_SETMASK, (sigset_t *) &sigset2, NULL);
|
||||||
|
_longjmp (jbuf, 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* sigsetjmp(jbuf, 1) MUST preserve signal mask: */
|
||||||
|
sigprocmask (SIG_SETMASK, (sigset_t *) &sigset1, NULL);
|
||||||
|
if (sigsetjmp (sigjbuf, 1))
|
||||||
|
{
|
||||||
|
sigemptyset ((sigset_t *) &sigset3);
|
||||||
|
sigprocmask (SIG_BLOCK, NULL, (sigset_t *) &sigset3);
|
||||||
|
if (memcmp ((sigset_t *) &sigset3, (sigset_t *) &sigset1,
|
||||||
|
sizeof (sigset_t)) != 0)
|
||||||
|
{
|
||||||
|
fprintf (stderr,
|
||||||
|
"FAILURE: siglongjmp() didn't restore signal mask!\n");
|
||||||
|
++nerrors;
|
||||||
|
}
|
||||||
|
else if (verbose)
|
||||||
|
printf ("OK: siglongjmp() restores signal mask when asked to\n");
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
sigprocmask (SIG_SETMASK, (sigset_t *) &sigset2, NULL);
|
||||||
|
siglongjmp (sigjbuf, 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* sigsetjmp(jbuf, 0) MUST NOT preserve signal mask: */
|
||||||
|
sigprocmask (SIG_SETMASK, (sigset_t *) &sigset1, NULL);
|
||||||
|
if (sigsetjmp (sigjbuf, 0))
|
||||||
|
{
|
||||||
|
sigemptyset ((sigset_t *) &sigset3);
|
||||||
|
sigprocmask (SIG_BLOCK, NULL, (sigset_t *) &sigset3);
|
||||||
|
if (memcmp ((sigset_t *) &sigset3, (sigset_t *) &sigset2,
|
||||||
|
sizeof (sigset_t)) != 0)
|
||||||
|
{
|
||||||
|
fprintf (stderr,
|
||||||
|
"FAILURE: siglongjmp() changed signal mask!\n");
|
||||||
|
++nerrors;
|
||||||
|
}
|
||||||
|
else if (verbose)
|
||||||
|
printf ("OK: siglongjmp() leaves signal mask along when asked to\n");
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
sigprocmask (SIG_SETMASK, (sigset_t *) &sigset2, NULL);
|
||||||
|
siglongjmp (sigjbuf, 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* sigsetjmp(jbuf, 1) MUST preserve signal mask: */
|
||||||
|
sigprocmask (SIG_SETMASK, (sigset_t *) &sigset1, NULL);
|
||||||
|
if (sigsetjmp (sigjbuf, 1))
|
||||||
|
{
|
||||||
|
sigemptyset ((sigset_t *) &sigset3);
|
||||||
|
sigprocmask (SIG_BLOCK, NULL, (sigset_t *) &sigset3);
|
||||||
|
if (memcmp ((sigset_t *) &sigset3, (sigset_t *) &sigset1,
|
||||||
|
sizeof (sigset_t)) != 0)
|
||||||
|
{
|
||||||
|
fprintf (stderr,
|
||||||
|
"FAILURE: siglongjmp() didn't restore signal mask!\n");
|
||||||
|
++nerrors;
|
||||||
|
}
|
||||||
|
else if (verbose)
|
||||||
|
printf ("OK: siglongjmp() restores signal mask when asked to\n");
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
sigprocmask (SIG_SETMASK, (sigset_t *) &sigset2, NULL);
|
||||||
|
kill (getpid (), SIGTERM);
|
||||||
|
fprintf (stderr, "FAILURE: unexpected return from kill()\n");
|
||||||
|
++nerrors;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* sigsetjmp(jbuf, 0) MUST NOT preserve signal mask: */
|
||||||
|
sigprocmask (SIG_SETMASK, (sigset_t *) &sigset1, NULL);
|
||||||
|
if (sigsetjmp (sigjbuf, 0))
|
||||||
|
{
|
||||||
|
sigemptyset ((sigset_t *) &sigset3);
|
||||||
|
sigprocmask (SIG_BLOCK, NULL, (sigset_t *) &sigset3);
|
||||||
|
if (memcmp ((sigset_t *) &sigset3, (sigset_t *) &sigset4,
|
||||||
|
sizeof (sigset_t)) != 0)
|
||||||
|
{
|
||||||
|
fprintf (stderr,
|
||||||
|
"FAILURE: siglongjmp() changed signal mask!\n");
|
||||||
|
++nerrors;
|
||||||
|
}
|
||||||
|
else if (verbose)
|
||||||
|
printf ("OK: siglongjmp() leaves signal mask along when asked to\n");
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
sigprocmask (SIG_SETMASK, (sigset_t *) &sigset2, NULL);
|
||||||
|
kill (getpid (), SIGTERM);
|
||||||
|
fprintf (stderr, "FAILURE: unexpected return from kill()\n");
|
||||||
|
++nerrors;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (nerrors > 0)
|
||||||
|
{
|
||||||
|
fprintf (stderr, "FAILURE: detected %d failures\n", nerrors);
|
||||||
|
exit (-1);
|
||||||
|
}
|
||||||
|
if (verbose)
|
||||||
|
printf ("SUCCESS\n");
|
||||||
|
return 0;
|
||||||
|
}
|
Loading…
Reference in a new issue