1
0
Fork 0
mirror of https://github.com/tobast/libunwind-eh_elf.git synced 2025-01-26 10:10:30 +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:
mostang.com!davidm 2004-03-31 07:38:06 +00:00
parent fe9d715071
commit 64c702cea7

View file

@ -1,5 +1,5 @@
/* libunwind - a platform-independent unwind library /* 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> Contributed by David Mosberger-Tang <davidm@hpl.hp.com>
This file is part of libunwind. 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 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 static inline void
maps_init (struct map_iterator *mi, pid_t pid) 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); memcpy (path, "/proc/", 6);
mi->fp = fopen (path, "r"); 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 static inline int
maps_next (struct map_iterator *mi, maps_next (struct map_iterator *mi,
unsigned long *low, unsigned long *high, unsigned long *offset, 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; return 0;
while (fgets (line, sizeof (line), mi->fp)) while (1)
{ {
if (sscanf (line, "%lx-%lx %*4c %lx %*x:%*x %*d %s\n", lseek (mi->fd, mi->offset, SEEK_SET);
low, high, offset, path) == 4)
return 1; 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; return 0;
} }
@ -62,10 +223,10 @@ maps_next (struct map_iterator *mi,
static inline void static inline void
maps_close (struct map_iterator *mi) maps_close (struct map_iterator *mi)
{ {
if (!mi->fp) if (mi->fd < 0)
return; return;
fclose (mi->fp); close (mi->fd);
mi->fp = NULL; mi->fd = -1;
} }
#endif /* os_linux_h */ #endif /* os_linux_h */