mirror of
https://github.com/tobast/libunwind-eh_elf.git
synced 2024-11-26 09:07:38 +01:00
Rename: tests/test-dyn1.c -> tests/Gtest-dyn1.c
}(Logical change 1.53)
This commit is contained in:
parent
a1aed8c295
commit
1a6bf76301
1 changed files with 0 additions and 197 deletions
|
@ -1,197 +0,0 @@
|
|||
/* libunwind - a platform-independent unwind library
|
||||
Copyright (C) 2002-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. */
|
||||
|
||||
/* This file tests dynamic code-generation via function-cloning. */
|
||||
|
||||
#include <libunwind.h>
|
||||
#include <malloc.h>
|
||||
#include <stdio.h>
|
||||
#include <stdlib.h>
|
||||
#include <string.h>
|
||||
#include <unistd.h>
|
||||
|
||||
#include <sys/mman.h>
|
||||
|
||||
#define panic(args...) \
|
||||
{ fprintf (stderr, args); exit (-1); }
|
||||
|
||||
typedef void (*template_t) (int, void (*)(),
|
||||
int (*)(const char *, ...), const char *,
|
||||
const char **);
|
||||
|
||||
int verbose;
|
||||
|
||||
static const char *strarr[] =
|
||||
{
|
||||
"i", "ii", "iii", "iv", "v", "vi", "vii", "viii", "ix", "x", NULL
|
||||
};
|
||||
|
||||
#ifdef __ia64__
|
||||
struct fdesc
|
||||
{
|
||||
long code;
|
||||
long gp;
|
||||
};
|
||||
# define get_fdesc(fdesc,func) (fdesc = *(struct fdesc *) &(func))
|
||||
# define get_funcp(fdesc) ((template_t) &(fdesc))
|
||||
# define get_gp(fdesc) ((fdesc).gp)
|
||||
#else
|
||||
struct fdesc
|
||||
{
|
||||
long code;
|
||||
};
|
||||
# define get_fdesc(fdesc,func) (fdesc.code = (long) &(func))
|
||||
# define get_funcp(fdesc) ((template_t) (fdesc).code)
|
||||
# define get_gp(fdesc) (0)
|
||||
#endif
|
||||
|
||||
static void
|
||||
flush_cache (void *addr, unsigned long len)
|
||||
{
|
||||
#ifdef __ia64__
|
||||
void *end = (char *) addr + len;
|
||||
|
||||
while (addr < end)
|
||||
{
|
||||
asm volatile ("fc %0" :: "r"(addr));
|
||||
addr = (char *) addr + 32;
|
||||
}
|
||||
asm volatile (";;sync.i;;srlz.i;;");
|
||||
#endif
|
||||
}
|
||||
|
||||
void
|
||||
template (int i, template_t self,
|
||||
int (*printer)(const char *, ...), const char *fmt, const char **arr)
|
||||
{
|
||||
(*printer) (fmt, arr[11 - i][0], arr[11 - i] + 1);
|
||||
if (i > 0)
|
||||
(*self) (i - 1, self, printer, fmt, arr);
|
||||
}
|
||||
|
||||
static void
|
||||
sighandler (int signal)
|
||||
{
|
||||
unw_cursor_t cursor;
|
||||
char name[128], off[32];
|
||||
unw_word_t ip, offset;
|
||||
unw_context_t uc;
|
||||
int count = 0;
|
||||
|
||||
if (verbose)
|
||||
printf ("caught signal %d\n", signal);
|
||||
|
||||
unw_getcontext (&uc);
|
||||
unw_init_local (&cursor, &uc);
|
||||
|
||||
while (!unw_is_signal_frame (&cursor))
|
||||
if (unw_step (&cursor) < 0)
|
||||
panic ("failed to find signal frame!\n");
|
||||
unw_step (&cursor);
|
||||
|
||||
do
|
||||
{
|
||||
unw_get_reg (&cursor, UNW_REG_IP, &ip);
|
||||
name[0] = '\0';
|
||||
off[0] = '\0';
|
||||
if (unw_get_proc_name (&cursor, name, sizeof (name), &offset) == 0
|
||||
&& off > 0)
|
||||
snprintf (off, sizeof (off), "+0x%lx", (long) offset);
|
||||
if (verbose)
|
||||
printf ("ip = %lx <%s%s>\n", (long) ip, name, off);
|
||||
++count;
|
||||
}
|
||||
while (unw_step (&cursor) > 0);
|
||||
|
||||
if (count != 13)
|
||||
panic ("FAILURE: expected 13, not %d frames below signal frame\n", count);
|
||||
|
||||
if (verbose)
|
||||
printf ("SUCCESS\n");
|
||||
exit (0);
|
||||
}
|
||||
|
||||
int
|
||||
dev_null (const char *format, ...)
|
||||
{
|
||||
return 0;
|
||||
}
|
||||
|
||||
int
|
||||
main (int argc, char *argv[])
|
||||
{
|
||||
unw_dyn_region_info_t *region;
|
||||
unw_dyn_info_t di;
|
||||
struct fdesc fdesc;
|
||||
template_t funcp;
|
||||
void *mem;
|
||||
|
||||
if (argc > 1)
|
||||
++verbose;
|
||||
|
||||
mem = malloc (getpagesize ());
|
||||
|
||||
get_fdesc (fdesc, template);
|
||||
|
||||
if (verbose)
|
||||
printf ("old code @ %p, new code @ %p\n", (void *) fdesc.code, mem);
|
||||
|
||||
memcpy (mem, (void *) fdesc.code, 256);
|
||||
mprotect ((void *) ((long) mem & ~(getpagesize () - 1)),
|
||||
2*getpagesize(), PROT_READ | PROT_WRITE | PROT_EXEC);
|
||||
flush_cache (mem, 256);
|
||||
|
||||
signal (SIGSEGV, sighandler);
|
||||
|
||||
/* register the new function: */
|
||||
region = alloca (_U_dyn_region_info_size (2));
|
||||
region->next = NULL;
|
||||
region->insn_count = 256;
|
||||
region->op_count = 2;
|
||||
_U_dyn_op_alias (®ion->op[0], 0, -1, fdesc.code);
|
||||
_U_dyn_op_stop (®ion->op[1]);
|
||||
|
||||
memset (&di, 0, sizeof (di));
|
||||
di.start_ip = (long) mem;
|
||||
di.end_ip = (long) mem + 256;
|
||||
di.gp = get_gp (fdesc);
|
||||
di.format = UNW_INFO_FORMAT_DYNAMIC;
|
||||
di.u.pi.name_ptr = (unw_word_t) "copy_of_template";
|
||||
di.u.pi.regions = region;
|
||||
|
||||
_U_dyn_register (&di);
|
||||
|
||||
/* call new function: */
|
||||
fdesc.code = (long) mem;
|
||||
funcp = get_funcp (fdesc);
|
||||
|
||||
if (verbose)
|
||||
(*funcp) (10, funcp, printf, "iteration %c%s\n", strarr);
|
||||
else
|
||||
(*funcp) (10, funcp, dev_null, "iteration %c%s\n", strarr);
|
||||
|
||||
_U_dyn_cancel (&di);
|
||||
return -1;
|
||||
}
|
Loading…
Reference in a new issue