mirror of
https://github.com/tobast/libunwind-eh_elf.git
synced 2025-01-24 09:10:29 +01:00
(ltoa): New inline function.
(skip_whitespace): Likewise. (scan_hex): Likewise. (scan_dec): Likewise. (scan_char): Likewise. (scan_string): Likewise. (maps_init): Avoid stdio operations since they're not async-signal safe. (maps_next): Likewise. (maps_close): Likewise. (Logical change 1.199)
This commit is contained in:
parent
fe9d715071
commit
64c702cea7
1 changed files with 176 additions and 15 deletions
191
src/os-linux.h
191
src/os-linux.h
|
@ -1,5 +1,5 @@
|
|||
/* libunwind - a platform-independent unwind library
|
||||
Copyright (C) 2003 Hewlett-Packard Co
|
||||
Copyright (C) 2003-2004 Hewlett-Packard Co
|
||||
Contributed by David Mosberger-Tang <davidm@hpl.hp.com>
|
||||
|
||||
This file is part of libunwind.
|
||||
|
@ -28,33 +28,194 @@ WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */
|
|||
|
||||
struct map_iterator
|
||||
{
|
||||
FILE *fp;
|
||||
off_t offset;
|
||||
int fd;
|
||||
};
|
||||
|
||||
static inline char *
|
||||
ltoa (char *buf, long val)
|
||||
{
|
||||
char *cp = buf, tmp;
|
||||
ssize_t i, len;
|
||||
|
||||
do
|
||||
{
|
||||
*cp++ = '0' + (val % 10);
|
||||
val /= 10;
|
||||
}
|
||||
while (val);
|
||||
|
||||
/* reverse the order of the digits: */
|
||||
len = cp - buf;
|
||||
--cp;
|
||||
for (i = 0; i < len / 2; ++i)
|
||||
{
|
||||
tmp = buf[i];
|
||||
buf[i] = cp[-i];
|
||||
cp[-i] = tmp;
|
||||
}
|
||||
return buf + len;
|
||||
}
|
||||
|
||||
static inline void
|
||||
maps_init (struct map_iterator *mi, pid_t pid)
|
||||
{
|
||||
char path[PATH_MAX];
|
||||
char path[PATH_MAX], *cp;
|
||||
|
||||
snprintf (path, sizeof (path), "/proc/%d/maps", pid);
|
||||
mi->fp = fopen (path, "r");
|
||||
memcpy (path, "/proc/", 6);
|
||||
cp = ltoa (path + 6, pid);
|
||||
memcpy (cp, "/maps", 6);
|
||||
|
||||
mi->fd = open (path, O_RDONLY);
|
||||
mi->offset = 0;
|
||||
}
|
||||
|
||||
static inline char *
|
||||
skip_whitespace (char *cp)
|
||||
{
|
||||
if (!cp)
|
||||
return NULL;
|
||||
|
||||
while (*cp == ' ' || *cp == '\t')
|
||||
++cp;
|
||||
return cp;
|
||||
}
|
||||
|
||||
static inline char *
|
||||
scan_hex (char *cp, unsigned long *valp)
|
||||
{
|
||||
unsigned long num_digits = 0, digit, val = 0;
|
||||
|
||||
cp = skip_whitespace (cp);
|
||||
if (!cp)
|
||||
return NULL;
|
||||
|
||||
while (1)
|
||||
{
|
||||
digit = *cp;
|
||||
if ((digit - '0') <= 9)
|
||||
digit -= '0';
|
||||
else if ((digit - 'a') < 6)
|
||||
digit -= 'a' - 10;
|
||||
else if ((digit - 'A') < 6)
|
||||
digit -= 'A' - 10;
|
||||
else
|
||||
break;
|
||||
val = (val << 4) | digit;
|
||||
++num_digits;
|
||||
++cp;
|
||||
}
|
||||
if (!num_digits)
|
||||
return NULL;
|
||||
*valp = val;
|
||||
return cp;
|
||||
}
|
||||
|
||||
static inline char *
|
||||
scan_dec (char *cp, unsigned long *valp)
|
||||
{
|
||||
unsigned long num_digits = 0, digit, val = 0;
|
||||
|
||||
if (!(cp = skip_whitespace (cp)))
|
||||
return NULL;
|
||||
|
||||
while (1)
|
||||
{
|
||||
digit = *cp++;
|
||||
if ((digit - '0') <= 9)
|
||||
digit -= '0';
|
||||
else
|
||||
break;
|
||||
val = (10 * val) + digit;
|
||||
++num_digits;
|
||||
}
|
||||
if (!num_digits)
|
||||
return NULL;
|
||||
*valp = val;
|
||||
return cp;
|
||||
}
|
||||
|
||||
static inline char *
|
||||
scan_char (char *cp, char *valp)
|
||||
{
|
||||
if (!cp)
|
||||
return NULL;
|
||||
|
||||
*valp = *cp++;
|
||||
return cp;
|
||||
}
|
||||
|
||||
/* Scan a string delimited by white-space. Fails on empty string or
|
||||
if string is doesn't fit in the specified buffer. */
|
||||
static inline char *
|
||||
scan_string (char *cp, char *valp, size_t buf_size)
|
||||
{
|
||||
size_t i = 0;
|
||||
|
||||
if (!(cp = skip_whitespace (cp)))
|
||||
return NULL;
|
||||
|
||||
while (*cp != ' ' && *cp != '\t' && *cp != '\0')
|
||||
{
|
||||
if (i < buf_size - 1)
|
||||
valp[i++] = *cp;
|
||||
++cp;
|
||||
}
|
||||
if (i == 0 || i >= buf_size)
|
||||
return NULL;
|
||||
valp[i] = '\0';
|
||||
return cp;
|
||||
}
|
||||
|
||||
static inline int
|
||||
maps_next (struct map_iterator *mi,
|
||||
unsigned long *low, unsigned long *high, unsigned long *offset,
|
||||
char *path)
|
||||
char *path, size_t path_size)
|
||||
{
|
||||
char line[256+PATH_MAX];
|
||||
char line[256 + PATH_MAX], perm[16], dash, colon, *cp;
|
||||
unsigned long major, minor, inum;
|
||||
size_t to_read = 256; /* most lines fit in 256 characters easy */
|
||||
ssize_t i, nread;
|
||||
|
||||
if (!mi->fp)
|
||||
if (mi->fd < 0)
|
||||
return 0;
|
||||
|
||||
while (fgets (line, sizeof (line), mi->fp))
|
||||
while (1)
|
||||
{
|
||||
if (sscanf (line, "%lx-%lx %*4c %lx %*x:%*x %*d %s\n",
|
||||
low, high, offset, path) == 4)
|
||||
return 1;
|
||||
lseek (mi->fd, mi->offset, SEEK_SET);
|
||||
|
||||
if ((nread = read (mi->fd, line, to_read)) <= 0)
|
||||
return 0;
|
||||
for (i = 0; i < nread && line[i] != '\n'; ++i)
|
||||
/* skip */;
|
||||
if (i < nread)
|
||||
{
|
||||
line[i] = '\0';
|
||||
mi->offset += i + 1;
|
||||
}
|
||||
else
|
||||
{
|
||||
if (to_read < sizeof (line))
|
||||
to_read = sizeof (line) - 1;
|
||||
else
|
||||
mi->offset += nread; /* not supposed to happen... */
|
||||
continue; /* duh, no newline found */
|
||||
}
|
||||
|
||||
/* scan: "LOW-HIGH PERM OFFSET MAJOR:MINOR INUM PATH" */
|
||||
cp = scan_hex (line, low);
|
||||
cp = scan_char (cp, &dash);
|
||||
cp = scan_hex (cp, high);
|
||||
cp = scan_string (cp, perm, sizeof (perm));
|
||||
cp = scan_hex (cp, offset);
|
||||
cp = scan_hex (cp, &major);
|
||||
cp = scan_char (cp, &colon);
|
||||
cp = scan_hex (cp, &minor);
|
||||
cp = scan_dec (cp, &inum);
|
||||
cp = scan_string (cp, path, path_size);
|
||||
if (!cp || dash != '-' || colon != ':')
|
||||
continue; /* skip line with unknown or bad format */
|
||||
return 1;
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
@ -62,10 +223,10 @@ maps_next (struct map_iterator *mi,
|
|||
static inline void
|
||||
maps_close (struct map_iterator *mi)
|
||||
{
|
||||
if (!mi->fp)
|
||||
if (mi->fd < 0)
|
||||
return;
|
||||
fclose (mi->fp);
|
||||
mi->fp = NULL;
|
||||
close (mi->fd);
|
||||
mi->fd = -1;
|
||||
}
|
||||
|
||||
#endif /* os_linux_h */
|
||||
|
|
Loading…
Reference in a new issue