|
|
435ea7 |
diff --git a/src/readelf.c b/src/readelf.c
|
|
|
435ea7 |
index 9651239..807affc 100644
|
|
|
435ea7 |
--- a/src/readelf.c
|
|
|
435ea7 |
+++ b/src/readelf.c
|
|
|
435ea7 |
@@ -41,6 +41,8 @@ FILE_RCSID("@(#)$File: readelf.c,v 1.90 2011/08/23 08:01:12 christos Exp $")
|
|
|
435ea7 |
#include "readelf.h"
|
|
|
435ea7 |
#include "magic.h"
|
|
|
435ea7 |
|
|
|
435ea7 |
+#define __arraycount(a) (sizeof(a) / sizeof(a[0]))
|
|
|
435ea7 |
+
|
|
|
435ea7 |
#ifdef ELFCORE
|
|
|
435ea7 |
private int dophn_core(struct magic_set *, int, int, int, off_t, int, size_t,
|
|
|
435ea7 |
off_t, int *);
|
|
|
435ea7 |
@@ -50,7 +52,7 @@ private int dophn_exec(struct magic_set *, int, int, int, off_t, int, size_t,
|
|
|
435ea7 |
private int doshn(struct magic_set *, int, int, int, off_t, int, size_t,
|
|
|
435ea7 |
off_t, int *, int, int);
|
|
|
435ea7 |
private size_t donote(struct magic_set *, void *, size_t, size_t, int,
|
|
|
435ea7 |
- int, size_t, int *);
|
|
|
435ea7 |
+ int, size_t, int *, int, off_t, int, off_t);
|
|
|
435ea7 |
|
|
|
435ea7 |
#define ELF_ALIGN(a) ((((a) + align - 1) / align) * align)
|
|
|
435ea7 |
|
|
|
435ea7 |
@@ -184,6 +186,11 @@ getu64(int swap, uint64_t value)
|
|
|
435ea7 |
elf_getu32(swap, ph32.p_align) : 4) \
|
|
|
435ea7 |
: (off_t) (ph64.p_align ? \
|
|
|
435ea7 |
elf_getu64(swap, ph64.p_align) : 4)))
|
|
|
435ea7 |
+#define xph_vaddr (size_t)((clazz == ELFCLASS32 \
|
|
|
435ea7 |
+ ? (off_t) (ph32.p_vaddr ? \
|
|
|
435ea7 |
+ elf_getu32(swap, ph32.p_vaddr) : 4) \
|
|
|
435ea7 |
+ : (off_t) (ph64.p_vaddr ? \
|
|
|
435ea7 |
+ elf_getu64(swap, ph64.p_vaddr) : 4)))
|
|
|
435ea7 |
#define xph_filesz (size_t)((clazz == ELFCLASS32 \
|
|
|
435ea7 |
? elf_getu32(swap, ph32.p_filesz) \
|
|
|
435ea7 |
: elf_getu64(swap, ph64.p_filesz)))
|
|
|
435ea7 |
@@ -194,8 +201,8 @@ getu64(int swap, uint64_t value)
|
|
|
435ea7 |
? elf_getu32(swap, ph32.p_memsz) \
|
|
|
435ea7 |
: elf_getu64(swap, ph64.p_memsz)))
|
|
|
435ea7 |
#define xnh_sizeof (clazz == ELFCLASS32 \
|
|
|
435ea7 |
- ? sizeof nh32 \
|
|
|
435ea7 |
- : sizeof nh64)
|
|
|
435ea7 |
+ ? sizeof(nh32) \
|
|
|
435ea7 |
+ : sizeof(nh64))
|
|
|
435ea7 |
#define xnh_type (clazz == ELFCLASS32 \
|
|
|
435ea7 |
? elf_getu32(swap, nh32.n_type) \
|
|
|
435ea7 |
: elf_getu32(swap, nh64.n_type))
|
|
|
435ea7 |
@@ -220,6 +227,18 @@ getu64(int swap, uint64_t value)
|
|
|
435ea7 |
#define xcap_val (clazz == ELFCLASS32 \
|
|
|
435ea7 |
? elf_getu32(swap, cap32.c_un.c_val) \
|
|
|
435ea7 |
: elf_getu64(swap, cap64.c_un.c_val))
|
|
|
435ea7 |
+#define xauxv_addr (clazz == ELFCLASS32 \
|
|
|
435ea7 |
+ ? (void *)&auxv32 \
|
|
|
435ea7 |
+ : (void *)&auxv64)
|
|
|
435ea7 |
+#define xauxv_sizeof (clazz == ELFCLASS32 \
|
|
|
435ea7 |
+ ? sizeof(auxv32) \
|
|
|
435ea7 |
+ : sizeof(auxv64))
|
|
|
435ea7 |
+#define xauxv_type (clazz == ELFCLASS32 \
|
|
|
435ea7 |
+ ? elf_getu32(swap, auxv32.a_type) \
|
|
|
435ea7 |
+ : elf_getu64(swap, auxv64.a_type))
|
|
|
435ea7 |
+#define xauxv_val (clazz == ELFCLASS32 \
|
|
|
435ea7 |
+ ? elf_getu32(swap, auxv32.a_v) \
|
|
|
435ea7 |
+ : elf_getu64(swap, auxv64.a_v))
|
|
|
435ea7 |
|
|
|
435ea7 |
#ifdef ELFCORE
|
|
|
435ea7 |
/*
|
|
|
435ea7 |
@@ -306,6 +325,7 @@ private const char os_style_names[][8] = {
|
|
|
435ea7 |
#define FLAGS_DID_BUILD_ID 0x04
|
|
|
435ea7 |
#define FLAGS_DID_CORE_STYLE 0x08
|
|
|
435ea7 |
#define FLAGS_IS_CORE 0x10
|
|
|
435ea7 |
+#define FLAGS_DID_AUXV 0x200
|
|
|
435ea7 |
|
|
|
435ea7 |
private int
|
|
|
435ea7 |
dophn_core(struct magic_set *ms, int clazz, int swap, int fd, off_t off,
|
|
|
435ea7 |
@@ -316,6 +336,8 @@ dophn_core(struct magic_set *ms, int clazz, int swap, int fd, off_t off,
|
|
|
435ea7 |
size_t offset, len;
|
|
|
435ea7 |
unsigned char nbuf[BUFSIZ];
|
|
|
435ea7 |
ssize_t bufsize;
|
|
|
435ea7 |
+ off_t ph_off = off;
|
|
|
435ea7 |
+ int ph_num = num;
|
|
|
435ea7 |
|
|
|
435ea7 |
if (size != xph_sizeof) {
|
|
|
435ea7 |
if (file_printf(ms, ", corrupted program header size") == -1)
|
|
|
435ea7 |
@@ -355,7 +377,8 @@ dophn_core(struct magic_set *ms, int clazz, int swap, int fd, off_t off,
|
|
|
435ea7 |
if (offset >= (size_t)bufsize)
|
|
|
435ea7 |
break;
|
|
|
435ea7 |
offset = donote(ms, nbuf, offset, (size_t)bufsize,
|
|
|
435ea7 |
- clazz, swap, 4, flags);
|
|
|
435ea7 |
+ clazz, swap, 4, flags, fd, ph_off,
|
|
|
435ea7 |
+ ph_num, fsize);
|
|
|
435ea7 |
if (offset == 0)
|
|
|
435ea7 |
break;
|
|
|
435ea7 |
|
|
|
435ea7 |
@@ -365,9 +388,160 @@ dophn_core(struct magic_set *ms, int clazz, int swap, int fd, off_t off,
|
|
|
435ea7 |
}
|
|
|
435ea7 |
#endif
|
|
|
435ea7 |
|
|
|
435ea7 |
+private off_t
|
|
|
435ea7 |
+get_offset_from_virtaddr(struct magic_set *ms, int swap, int clazz, int fd,
|
|
|
435ea7 |
+ off_t off, int num, off_t fsize, uint64_t virtaddr)
|
|
|
435ea7 |
+{
|
|
|
435ea7 |
+ Elf32_Phdr ph32;
|
|
|
435ea7 |
+ Elf64_Phdr ph64;
|
|
|
435ea7 |
+
|
|
|
435ea7 |
+ /*
|
|
|
435ea7 |
+ * Loop through all the program headers and find the header with
|
|
|
435ea7 |
+ * virtual address in which the "virtaddr" belongs to.
|
|
|
435ea7 |
+ */
|
|
|
435ea7 |
+ for ( ; num; num--) {
|
|
|
435ea7 |
+ if (pread(fd, xph_addr, xph_sizeof, off) < (ssize_t)xph_sizeof) {
|
|
|
435ea7 |
+ file_badread(ms);
|
|
|
435ea7 |
+ return -1;
|
|
|
435ea7 |
+ }
|
|
|
435ea7 |
+ off += xph_sizeof;
|
|
|
435ea7 |
+
|
|
|
435ea7 |
+ if (xph_offset > fsize) {
|
|
|
435ea7 |
+ /* Perhaps warn here */
|
|
|
435ea7 |
+ continue;
|
|
|
435ea7 |
+ }
|
|
|
435ea7 |
+
|
|
|
435ea7 |
+ if (virtaddr >= xph_vaddr && virtaddr < xph_vaddr + xph_filesz)
|
|
|
435ea7 |
+ return xph_offset + (virtaddr - xph_vaddr);
|
|
|
435ea7 |
+ }
|
|
|
435ea7 |
+ return 0;
|
|
|
435ea7 |
+}
|
|
|
435ea7 |
+
|
|
|
435ea7 |
+private size_t
|
|
|
435ea7 |
+get_string_on_virtaddr(struct magic_set *ms,
|
|
|
435ea7 |
+ int swap, int clazz, int fd, off_t ph_off, int ph_num,
|
|
|
435ea7 |
+ off_t fsize, uint64_t virtaddr, char *buf, ssize_t buflen)
|
|
|
435ea7 |
+{
|
|
|
435ea7 |
+ char *bptr;
|
|
|
435ea7 |
+ off_t offset;
|
|
|
435ea7 |
+
|
|
|
435ea7 |
+ if (buflen == 0)
|
|
|
435ea7 |
+ return 0;
|
|
|
435ea7 |
+
|
|
|
435ea7 |
+ offset = get_offset_from_virtaddr(ms, swap, clazz, fd, ph_off, ph_num,
|
|
|
435ea7 |
+ fsize, virtaddr);
|
|
|
435ea7 |
+ if (offset < 0 || (buflen = pread(fd, buf, buflen, offset)) <= 0) {
|
|
|
435ea7 |
+ file_badread(ms);
|
|
|
435ea7 |
+ return 0;
|
|
|
435ea7 |
+ }
|
|
|
435ea7 |
+
|
|
|
435ea7 |
+ buf[buflen - 1] = '\0';
|
|
|
435ea7 |
+
|
|
|
435ea7 |
+ /* We expect only printable characters, so return if buffer contains
|
|
|
435ea7 |
+ * non-printable character before the '\0' or just '\0'. */
|
|
|
435ea7 |
+ for (bptr = buf; *bptr && isprint((unsigned char)*bptr); bptr++)
|
|
|
435ea7 |
+ continue;
|
|
|
435ea7 |
+ if (*bptr != '\0')
|
|
|
435ea7 |
+ return 0;
|
|
|
435ea7 |
+
|
|
|
435ea7 |
+ return bptr - buf;
|
|
|
435ea7 |
+}
|
|
|
435ea7 |
+
|
|
|
435ea7 |
+
|
|
|
435ea7 |
+private int
|
|
|
435ea7 |
+do_auxv_note(struct magic_set *ms, unsigned char *nbuf, uint32_t type,
|
|
|
435ea7 |
+ int swap, uint32_t namesz __attribute__((__unused__)),
|
|
|
435ea7 |
+ uint32_t descsz __attribute__((__unused__)),
|
|
|
435ea7 |
+ size_t noff __attribute__((__unused__)), size_t doff,
|
|
|
435ea7 |
+ int *flags, size_t size __attribute__((__unused__)), int clazz,
|
|
|
435ea7 |
+ int fd, off_t ph_off, int ph_num, off_t fsize)
|
|
|
435ea7 |
+{
|
|
|
435ea7 |
+#ifdef ELFCORE
|
|
|
435ea7 |
+ Aux32Info auxv32;
|
|
|
435ea7 |
+ Aux64Info auxv64;
|
|
|
435ea7 |
+ size_t elsize = xauxv_sizeof;
|
|
|
435ea7 |
+ const char *tag;
|
|
|
435ea7 |
+ int is_string;
|
|
|
435ea7 |
+ size_t nval;
|
|
|
435ea7 |
+ size_t off;
|
|
|
435ea7 |
+
|
|
|
435ea7 |
+
|
|
|
435ea7 |
+ if (type != NT_AUXV || (*flags & FLAGS_IS_CORE) == 0)
|
|
|
435ea7 |
+ return 0;
|
|
|
435ea7 |
+
|
|
|
435ea7 |
+ *flags |= FLAGS_DID_AUXV;
|
|
|
435ea7 |
+
|
|
|
435ea7 |
+ nval = 0;
|
|
|
435ea7 |
+ for (off = 0; off + elsize <= descsz; off += elsize) {
|
|
|
435ea7 |
+ (void)memcpy(xauxv_addr, &nbuf[doff + off], xauxv_sizeof);
|
|
|
435ea7 |
+ /* Limit processing to 50 vector entries to prevent DoS */
|
|
|
435ea7 |
+ if (nval++ >= 50) {
|
|
|
435ea7 |
+ file_error(ms, 0, "Too many ELF Auxv elements");
|
|
|
435ea7 |
+ return 1;
|
|
|
435ea7 |
+ }
|
|
|
435ea7 |
+
|
|
|
435ea7 |
+ switch(xauxv_type) {
|
|
|
435ea7 |
+ case AT_LINUX_EXECFN:
|
|
|
435ea7 |
+ is_string = 1;
|
|
|
435ea7 |
+ tag = "execfn";
|
|
|
435ea7 |
+ break;
|
|
|
435ea7 |
+ case AT_LINUX_PLATFORM:
|
|
|
435ea7 |
+ is_string = 1;
|
|
|
435ea7 |
+ tag = "platform";
|
|
|
435ea7 |
+ break;
|
|
|
435ea7 |
+ case AT_LINUX_UID:
|
|
|
435ea7 |
+ is_string = 0;
|
|
|
435ea7 |
+ tag = "real uid";
|
|
|
435ea7 |
+ break;
|
|
|
435ea7 |
+ case AT_LINUX_GID:
|
|
|
435ea7 |
+ is_string = 0;
|
|
|
435ea7 |
+ tag = "real gid";
|
|
|
435ea7 |
+ break;
|
|
|
435ea7 |
+ case AT_LINUX_EUID:
|
|
|
435ea7 |
+ is_string = 0;
|
|
|
435ea7 |
+ tag = "effective uid";
|
|
|
435ea7 |
+ break;
|
|
|
435ea7 |
+ case AT_LINUX_EGID:
|
|
|
435ea7 |
+ is_string = 0;
|
|
|
435ea7 |
+ tag = "effective gid";
|
|
|
435ea7 |
+ break;
|
|
|
435ea7 |
+ default:
|
|
|
435ea7 |
+ is_string = 0;
|
|
|
435ea7 |
+ tag = NULL;
|
|
|
435ea7 |
+ break;
|
|
|
435ea7 |
+ }
|
|
|
435ea7 |
+
|
|
|
435ea7 |
+ if (tag == NULL)
|
|
|
435ea7 |
+ continue;
|
|
|
435ea7 |
+
|
|
|
435ea7 |
+ if (is_string) {
|
|
|
435ea7 |
+ char buf[256];
|
|
|
435ea7 |
+ ssize_t buflen;
|
|
|
435ea7 |
+ buflen = get_string_on_virtaddr(ms, swap, clazz, fd,
|
|
|
435ea7 |
+ ph_off, ph_num, fsize, xauxv_val, buf, sizeof(buf));
|
|
|
435ea7 |
+
|
|
|
435ea7 |
+ if (buflen == 0)
|
|
|
435ea7 |
+ continue;
|
|
|
435ea7 |
+
|
|
|
435ea7 |
+ if (file_printf(ms, ", %s: '%s'", tag, buf) == -1)
|
|
|
435ea7 |
+ return 0;
|
|
|
435ea7 |
+ } else {
|
|
|
435ea7 |
+ if (file_printf(ms, ", %s: %d", tag, (int) xauxv_val)
|
|
|
435ea7 |
+ == -1)
|
|
|
435ea7 |
+ return 0;
|
|
|
435ea7 |
+ }
|
|
|
435ea7 |
+ }
|
|
|
435ea7 |
+ return 1;
|
|
|
435ea7 |
+#else
|
|
|
435ea7 |
+ return 0;
|
|
|
435ea7 |
+#endif
|
|
|
435ea7 |
+}
|
|
|
435ea7 |
+
|
|
|
435ea7 |
+
|
|
|
435ea7 |
private size_t
|
|
|
435ea7 |
donote(struct magic_set *ms, void *vbuf, size_t offset, size_t size,
|
|
|
435ea7 |
- int clazz, int swap, size_t align, int *flags)
|
|
|
435ea7 |
+ int clazz, int swap, size_t align, int *flags,
|
|
|
435ea7 |
+ int fd, off_t ph_off, int ph_num, off_t fsize)
|
|
|
435ea7 |
{
|
|
|
435ea7 |
Elf32_Nhdr nh32;
|
|
|
435ea7 |
Elf64_Nhdr nh64;
|
|
|
435ea7 |
@@ -390,6 +564,7 @@ donote(struct magic_set *ms, void *vbuf, size_t offset, size_t size,
|
|
|
435ea7 |
|
|
|
435ea7 |
namesz = xnh_namesz;
|
|
|
435ea7 |
descsz = xnh_descsz;
|
|
|
435ea7 |
+
|
|
|
435ea7 |
if ((namesz == 0) && (descsz == 0)) {
|
|
|
435ea7 |
/*
|
|
|
435ea7 |
* We're out of note headers.
|
|
|
435ea7 |
@@ -438,37 +613,37 @@ donote(struct magic_set *ms, void *vbuf, size_t offset, size_t size,
|
|
|
435ea7 |
(void)memcpy(desc, &nbuf[doff], sizeof(desc));
|
|
|
435ea7 |
|
|
|
435ea7 |
if (file_printf(ms, ", for GNU/") == -1)
|
|
|
435ea7 |
- return size;
|
|
|
435ea7 |
+ return offset;
|
|
|
435ea7 |
switch (elf_getu32(swap, desc[0])) {
|
|
|
435ea7 |
case GNU_OS_LINUX:
|
|
|
435ea7 |
if (file_printf(ms, "Linux") == -1)
|
|
|
435ea7 |
- return size;
|
|
|
435ea7 |
+ return offset;
|
|
|
435ea7 |
break;
|
|
|
435ea7 |
case GNU_OS_HURD:
|
|
|
435ea7 |
if (file_printf(ms, "Hurd") == -1)
|
|
|
435ea7 |
- return size;
|
|
|
435ea7 |
+ return offset;
|
|
|
435ea7 |
break;
|
|
|
435ea7 |
case GNU_OS_SOLARIS:
|
|
|
435ea7 |
if (file_printf(ms, "Solaris") == -1)
|
|
|
435ea7 |
- return size;
|
|
|
435ea7 |
+ return offset;
|
|
|
435ea7 |
break;
|
|
|
435ea7 |
case GNU_OS_KFREEBSD:
|
|
|
435ea7 |
if (file_printf(ms, "kFreeBSD") == -1)
|
|
|
435ea7 |
- return size;
|
|
|
435ea7 |
+ return offset;
|
|
|
435ea7 |
break;
|
|
|
435ea7 |
case GNU_OS_KNETBSD:
|
|
|
435ea7 |
if (file_printf(ms, "kNetBSD") == -1)
|
|
|
435ea7 |
- return size;
|
|
|
435ea7 |
+ return offset;
|
|
|
435ea7 |
break;
|
|
|
435ea7 |
default:
|
|
|
435ea7 |
if (file_printf(ms, "<unknown>") == -1)
|
|
|
435ea7 |
- return size;
|
|
|
435ea7 |
+ return offset;
|
|
|
435ea7 |
}
|
|
|
435ea7 |
if (file_printf(ms, " %d.%d.%d", elf_getu32(swap, desc[1]),
|
|
|
435ea7 |
elf_getu32(swap, desc[2]), elf_getu32(swap, desc[3])) == -1)
|
|
|
435ea7 |
- return size;
|
|
|
435ea7 |
+ return offset;
|
|
|
435ea7 |
*flags |= FLAGS_DID_NOTE;
|
|
|
435ea7 |
- return size;
|
|
|
435ea7 |
+ return offset;
|
|
|
435ea7 |
}
|
|
|
435ea7 |
|
|
|
435ea7 |
if (namesz == 4 && strcmp((char *)&nbuf[noff], "GNU") == 0 &&
|
|
|
435ea7 |
@@ -492,7 +667,7 @@ donote(struct magic_set *ms, void *vbuf, size_t offset, size_t size,
|
|
|
435ea7 |
desc = elf_getu32(swap, desc);
|
|
|
435ea7 |
|
|
|
435ea7 |
if (file_printf(ms, ", for NetBSD") == -1)
|
|
|
435ea7 |
- return size;
|
|
|
435ea7 |
+ return offset;
|
|
|
435ea7 |
/*
|
|
|
435ea7 |
* The version number used to be stuck as 199905, and was thus
|
|
|
435ea7 |
* basically content-free. Newer versions of NetBSD have fixed
|
|
|
435ea7 |
@@ -512,23 +687,23 @@ donote(struct magic_set *ms, void *vbuf, size_t offset, size_t size,
|
|
|
435ea7 |
uint32_t ver_maj = desc / 100000000;
|
|
|
435ea7 |
|
|
|
435ea7 |
if (file_printf(ms, " %u.%u", ver_maj, ver_min) == -1)
|
|
|
435ea7 |
- return size;
|
|
|
435ea7 |
+ return offset;
|
|
|
435ea7 |
if (ver_rel == 0 && ver_patch != 0) {
|
|
|
435ea7 |
if (file_printf(ms, ".%u", ver_patch) == -1)
|
|
|
435ea7 |
- return size;
|
|
|
435ea7 |
+ return offset;
|
|
|
435ea7 |
} else if (ver_rel != 0) {
|
|
|
435ea7 |
while (ver_rel > 26) {
|
|
|
435ea7 |
if (file_printf(ms, "Z") == -1)
|
|
|
435ea7 |
- return size;
|
|
|
435ea7 |
+ return offset;
|
|
|
435ea7 |
ver_rel -= 26;
|
|
|
435ea7 |
}
|
|
|
435ea7 |
if (file_printf(ms, "%c", 'A' + ver_rel - 1)
|
|
|
435ea7 |
== -1)
|
|
|
435ea7 |
- return size;
|
|
|
435ea7 |
+ return offset;
|
|
|
435ea7 |
}
|
|
|
435ea7 |
}
|
|
|
435ea7 |
*flags |= FLAGS_DID_NOTE;
|
|
|
435ea7 |
- return size;
|
|
|
435ea7 |
+ return offset;
|
|
|
435ea7 |
}
|
|
|
435ea7 |
|
|
|
435ea7 |
if (namesz == 8 && strcmp((char *)&nbuf[noff], "FreeBSD") == 0 &&
|
|
|
435ea7 |
@@ -537,7 +712,7 @@ donote(struct magic_set *ms, void *vbuf, size_t offset, size_t size,
|
|
|
435ea7 |
(void)memcpy(&desc, &nbuf[doff], sizeof(desc));
|
|
|
435ea7 |
desc = elf_getu32(swap, desc);
|
|
|
435ea7 |
if (file_printf(ms, ", for FreeBSD") == -1)
|
|
|
435ea7 |
- return size;
|
|
|
435ea7 |
+ return offset;
|
|
|
435ea7 |
|
|
|
435ea7 |
/*
|
|
|
435ea7 |
* Contents is __FreeBSD_version, whose relation to OS
|
|
|
435ea7 |
@@ -567,69 +742,69 @@ donote(struct magic_set *ms, void *vbuf, size_t offset, size_t size,
|
|
|
435ea7 |
*/
|
|
|
435ea7 |
if (desc == 460002) {
|
|
|
435ea7 |
if (file_printf(ms, " 4.6.2") == -1)
|
|
|
435ea7 |
- return size;
|
|
|
435ea7 |
+ return offset;
|
|
|
435ea7 |
} else if (desc < 460100) {
|
|
|
435ea7 |
if (file_printf(ms, " %d.%d", desc / 100000,
|
|
|
435ea7 |
desc / 10000 % 10) == -1)
|
|
|
435ea7 |
- return size;
|
|
|
435ea7 |
+ return offset;
|
|
|
435ea7 |
if (desc / 1000 % 10 > 0)
|
|
|
435ea7 |
if (file_printf(ms, ".%d", desc / 1000 % 10)
|
|
|
435ea7 |
== -1)
|
|
|
435ea7 |
- return size;
|
|
|
435ea7 |
+ return offset;
|
|
|
435ea7 |
if ((desc % 1000 > 0) || (desc % 100000 == 0))
|
|
|
435ea7 |
if (file_printf(ms, " (%d)", desc) == -1)
|
|
|
435ea7 |
- return size;
|
|
|
435ea7 |
+ return offset;
|
|
|
435ea7 |
} else if (desc < 500000) {
|
|
|
435ea7 |
if (file_printf(ms, " %d.%d", desc / 100000,
|
|
|
435ea7 |
desc / 10000 % 10 + desc / 1000 % 10) == -1)
|
|
|
435ea7 |
- return size;
|
|
|
435ea7 |
+ return offset;
|
|
|
435ea7 |
if (desc / 100 % 10 > 0) {
|
|
|
435ea7 |
if (file_printf(ms, " (%d)", desc) == -1)
|
|
|
435ea7 |
- return size;
|
|
|
435ea7 |
+ return offset;
|
|
|
435ea7 |
} else if (desc / 10 % 10 > 0) {
|
|
|
435ea7 |
if (file_printf(ms, ".%d", desc / 10 % 10)
|
|
|
435ea7 |
== -1)
|
|
|
435ea7 |
- return size;
|
|
|
435ea7 |
+ return offset;
|
|
|
435ea7 |
}
|
|
|
435ea7 |
} else {
|
|
|
435ea7 |
if (file_printf(ms, " %d.%d", desc / 100000,
|
|
|
435ea7 |
desc / 1000 % 100) == -1)
|
|
|
435ea7 |
- return size;
|
|
|
435ea7 |
+ return offset;
|
|
|
435ea7 |
if ((desc / 100 % 10 > 0) ||
|
|
|
435ea7 |
(desc % 100000 / 100 == 0)) {
|
|
|
435ea7 |
if (file_printf(ms, " (%d)", desc) == -1)
|
|
|
435ea7 |
- return size;
|
|
|
435ea7 |
+ return offset;
|
|
|
435ea7 |
} else if (desc / 10 % 10 > 0) {
|
|
|
435ea7 |
if (file_printf(ms, ".%d", desc / 10 % 10)
|
|
|
435ea7 |
== -1)
|
|
|
435ea7 |
- return size;
|
|
|
435ea7 |
+ return offset;
|
|
|
435ea7 |
}
|
|
|
435ea7 |
}
|
|
|
435ea7 |
*flags |= FLAGS_DID_NOTE;
|
|
|
435ea7 |
- return size;
|
|
|
435ea7 |
+ return offset;
|
|
|
435ea7 |
}
|
|
|
435ea7 |
|
|
|
435ea7 |
if (namesz == 8 && strcmp((char *)&nbuf[noff], "OpenBSD") == 0 &&
|
|
|
435ea7 |
xnh_type == NT_OPENBSD_VERSION && descsz == 4) {
|
|
|
435ea7 |
if (file_printf(ms, ", for OpenBSD") == -1)
|
|
|
435ea7 |
- return size;
|
|
|
435ea7 |
+ return offset;
|
|
|
435ea7 |
/* Content of note is always 0 */
|
|
|
435ea7 |
*flags |= FLAGS_DID_NOTE;
|
|
|
435ea7 |
- return size;
|
|
|
435ea7 |
+ return offset;
|
|
|
435ea7 |
}
|
|
|
435ea7 |
|
|
|
435ea7 |
if (namesz == 10 && strcmp((char *)&nbuf[noff], "DragonFly") == 0 &&
|
|
|
435ea7 |
xnh_type == NT_DRAGONFLY_VERSION && descsz == 4) {
|
|
|
435ea7 |
uint32_t desc;
|
|
|
435ea7 |
if (file_printf(ms, ", for DragonFly") == -1)
|
|
|
435ea7 |
- return size;
|
|
|
435ea7 |
+ return offset;
|
|
|
435ea7 |
(void)memcpy(&desc, &nbuf[doff], sizeof(desc));
|
|
|
435ea7 |
desc = elf_getu32(swap, desc);
|
|
|
435ea7 |
if (file_printf(ms, " %d.%d.%d", desc / 100000,
|
|
|
435ea7 |
desc / 10000 % 10, desc % 10000) == -1)
|
|
|
435ea7 |
- return size;
|
|
|
435ea7 |
+ return offset;
|
|
|
435ea7 |
*flags |= FLAGS_DID_NOTE;
|
|
|
435ea7 |
- return size;
|
|
|
435ea7 |
+ return offset;
|
|
|
435ea7 |
}
|
|
|
435ea7 |
|
|
|
435ea7 |
core:
|
|
|
435ea7 |
@@ -661,14 +836,22 @@ core:
|
|
|
435ea7 |
os_style = OS_STYLE_NETBSD;
|
|
|
435ea7 |
}
|
|
|
435ea7 |
|
|
|
435ea7 |
+ if ((*flags & FLAGS_DID_AUXV) == 0) {
|
|
|
435ea7 |
+ if (do_auxv_note(ms, nbuf, xnh_type, swap,
|
|
|
435ea7 |
+ namesz, descsz, noff, doff, flags, size, clazz,
|
|
|
435ea7 |
+ fd, ph_off, ph_num, fsize))
|
|
|
435ea7 |
+ return offset;
|
|
|
435ea7 |
+ }
|
|
|
435ea7 |
+
|
|
|
435ea7 |
+
|
|
|
435ea7 |
#ifdef ELFCORE
|
|
|
435ea7 |
if ((*flags & FLAGS_DID_CORE) != 0)
|
|
|
435ea7 |
- return size;
|
|
|
435ea7 |
+ return offset;
|
|
|
435ea7 |
|
|
|
435ea7 |
if (os_style != -1 && (*flags & FLAGS_DID_CORE_STYLE) == 0) {
|
|
|
435ea7 |
if (file_printf(ms, ", %s-style", os_style_names[os_style])
|
|
|
435ea7 |
== -1)
|
|
|
435ea7 |
- return size;
|
|
|
435ea7 |
+ return offset;
|
|
|
435ea7 |
*flags |= FLAGS_DID_CORE_STYLE;
|
|
|
435ea7 |
}
|
|
|
435ea7 |
|
|
|
435ea7 |
@@ -683,7 +866,7 @@ core:
|
|
|
435ea7 |
*/
|
|
|
435ea7 |
if (file_printf(ms, ", from '%.31s'",
|
|
|
435ea7 |
&nbuf[doff + 0x7c]) == -1)
|
|
|
435ea7 |
- return size;
|
|
|
435ea7 |
+ return offset;
|
|
|
435ea7 |
|
|
|
435ea7 |
/*
|
|
|
435ea7 |
* Extract the signal number. It is at
|
|
|
435ea7 |
@@ -693,9 +876,9 @@ core:
|
|
|
435ea7 |
sizeof(signo));
|
|
|
435ea7 |
if (file_printf(ms, " (signal %u)",
|
|
|
435ea7 |
elf_getu32(swap, signo)) == -1)
|
|
|
435ea7 |
- return size;
|
|
|
435ea7 |
+ return offset;
|
|
|
435ea7 |
*flags |= FLAGS_DID_CORE;
|
|
|
435ea7 |
- return size;
|
|
|
435ea7 |
+ return offset;
|
|
|
435ea7 |
}
|
|
|
435ea7 |
break;
|
|
|
435ea7 |
|
|
|
435ea7 |
@@ -793,9 +976,9 @@ core:
|
|
|
435ea7 |
cp--;
|
|
|
435ea7 |
if (file_printf(ms, ", from '%.*s'",
|
|
|
435ea7 |
(int)(cp - cname), cname) == -1)
|
|
|
435ea7 |
- return size;
|
|
|
435ea7 |
+ return offset;
|
|
|
435ea7 |
*flags |= FLAGS_DID_CORE;
|
|
|
435ea7 |
- return size;
|
|
|
435ea7 |
+ return offset;
|
|
|
435ea7 |
|
|
|
435ea7 |
tryanother:
|
|
|
435ea7 |
;
|
|
|
435ea7 |
@@ -936,7 +1119,7 @@ doshn(struct magic_set *ms, int clazz, int swap, int fd, off_t off, int num,
|
|
|
435ea7 |
if (noff >= (off_t)xsh_size)
|
|
|
435ea7 |
break;
|
|
|
435ea7 |
noff = donote(ms, nbuf, (size_t)noff,
|
|
|
435ea7 |
- xsh_size, clazz, swap, 4, flags);
|
|
|
435ea7 |
+ xsh_size, clazz, swap, 4, flags, 0, 0, 0, 0);
|
|
|
435ea7 |
if (noff == 0)
|
|
|
435ea7 |
break;
|
|
|
435ea7 |
}
|
|
|
435ea7 |
@@ -1130,7 +1313,7 @@ dophn_exec(struct magic_set *ms, int clazz, int swap, int fd, off_t off,
|
|
|
435ea7 |
break;
|
|
|
435ea7 |
offset = donote(ms, nbuf, offset,
|
|
|
435ea7 |
(size_t)bufsize, clazz, swap, align,
|
|
|
435ea7 |
- flags);
|
|
|
435ea7 |
+ flags, 0, 0, 0, 0);
|
|
|
435ea7 |
if (offset == 0)
|
|
|
435ea7 |
break;
|
|
|
435ea7 |
}
|
|
|
435ea7 |
diff --git a/src/readelf.h b/src/readelf.h
|
|
|
435ea7 |
index ab4b5d1..fb34585 100644
|
|
|
435ea7 |
--- a/src/readelf.h
|
|
|
435ea7 |
+++ b/src/readelf.h
|
|
|
435ea7 |
@@ -62,6 +62,42 @@ typedef uint8_t Elf64_Char;
|
|
|
435ea7 |
#define EI_NIDENT 16
|
|
|
435ea7 |
|
|
|
435ea7 |
typedef struct {
|
|
|
435ea7 |
+ Elf32_Word a_type; /* 32-bit id */
|
|
|
435ea7 |
+ Elf32_Word a_v; /* 32-bit id */
|
|
|
435ea7 |
+} Aux32Info;
|
|
|
435ea7 |
+
|
|
|
435ea7 |
+typedef struct {
|
|
|
435ea7 |
+ Elf64_Xword a_type; /* 64-bit id */
|
|
|
435ea7 |
+ Elf64_Xword a_v; /* 64-bit id */
|
|
|
435ea7 |
+} Aux64Info;
|
|
|
435ea7 |
+
|
|
|
435ea7 |
+#define AT_NULL 0 /* end of vector */
|
|
|
435ea7 |
+#define AT_IGNORE 1 /* entry should be ignored */
|
|
|
435ea7 |
+#define AT_EXECFD 2 /* file descriptor of program */
|
|
|
435ea7 |
+#define AT_PHDR 3 /* program headers for program */
|
|
|
435ea7 |
+#define AT_PHENT 4 /* size of program header entry */
|
|
|
435ea7 |
+#define AT_PHNUM 5 /* number of program headers */
|
|
|
435ea7 |
+#define AT_PAGESZ 6 /* system page size */
|
|
|
435ea7 |
+#define AT_BASE 7 /* base address of interpreter */
|
|
|
435ea7 |
+#define AT_FLAGS 8 /* flags */
|
|
|
435ea7 |
+#define AT_ENTRY 9 /* entry point of program */
|
|
|
435ea7 |
+#define AT_LINUX_NOTELF 10 /* program is not ELF */
|
|
|
435ea7 |
+#define AT_LINUX_UID 11 /* real uid */
|
|
|
435ea7 |
+#define AT_LINUX_EUID 12 /* effective uid */
|
|
|
435ea7 |
+#define AT_LINUX_GID 13 /* real gid */
|
|
|
435ea7 |
+#define AT_LINUX_EGID 14 /* effective gid */
|
|
|
435ea7 |
+#define AT_LINUX_PLATFORM 15 /* string identifying CPU for optimizations */
|
|
|
435ea7 |
+#define AT_LINUX_HWCAP 16 /* arch dependent hints at CPU capabilities */
|
|
|
435ea7 |
+#define AT_LINUX_CLKTCK 17 /* frequency at which times() increments */
|
|
|
435ea7 |
+/* AT_* values 18 through 22 are reserved */
|
|
|
435ea7 |
+#define AT_LINUX_SECURE 23 /* secure mode boolean */
|
|
|
435ea7 |
+#define AT_LINUX_BASE_PLATFORM 24 /* string identifying real platform, may
|
|
|
435ea7 |
+ * differ from AT_PLATFORM. */
|
|
|
435ea7 |
+#define AT_LINUX_RANDOM 25 /* address of 16 random bytes */
|
|
|
435ea7 |
+#define AT_LINUX_HWCAP2 26 /* extension of AT_HWCAP */
|
|
|
435ea7 |
+#define AT_LINUX_EXECFN 31 /* filename of program */
|
|
|
435ea7 |
+
|
|
|
435ea7 |
+typedef struct {
|
|
|
435ea7 |
Elf32_Char e_ident[EI_NIDENT];
|
|
|
435ea7 |
Elf32_Half e_type;
|
|
|
435ea7 |
Elf32_Half e_machine;
|