97 lines
1.9 KiB
C
97 lines
1.9 KiB
C
|
// SPDX-License-Identifier: GPL-2.0
|
||
|
#include <byteswap.h>
|
||
|
#include <elf.h>
|
||
|
#include <endian.h>
|
||
|
#include <inttypes.h>
|
||
|
#include <stdint.h>
|
||
|
#include <stdio.h>
|
||
|
#include <stdlib.h>
|
||
|
#include <string.h>
|
||
|
|
||
|
#ifdef be32toh
|
||
|
/* If libc provides [bl]e{32,64}toh() then we'll use them */
|
||
|
#elif BYTE_ORDER == LITTLE_ENDIAN
|
||
|
# define be32toh(x) bswap_32(x)
|
||
|
# define le32toh(x) (x)
|
||
|
# define be64toh(x) bswap_64(x)
|
||
|
# define le64toh(x) (x)
|
||
|
#elif BYTE_ORDER == BIG_ENDIAN
|
||
|
# define be32toh(x) (x)
|
||
|
# define le32toh(x) bswap_32(x)
|
||
|
# define be64toh(x) (x)
|
||
|
# define le64toh(x) bswap_64(x)
|
||
|
#endif
|
||
|
|
||
|
__attribute__((noreturn))
|
||
|
static void die(const char *msg)
|
||
|
{
|
||
|
fputs(msg, stderr);
|
||
|
exit(EXIT_FAILURE);
|
||
|
}
|
||
|
|
||
|
int main(int argc, const char *argv[])
|
||
|
{
|
||
|
uint64_t entry;
|
||
|
size_t nread;
|
||
|
FILE *file;
|
||
|
union {
|
||
|
Elf32_Ehdr ehdr32;
|
||
|
Elf64_Ehdr ehdr64;
|
||
|
} hdr;
|
||
|
|
||
|
if (argc != 2)
|
||
|
die("Usage: elf-entry <elf-file>\n");
|
||
|
|
||
|
file = fopen(argv[1], "r");
|
||
|
if (!file) {
|
||
|
perror("Unable to open input file");
|
||
|
return EXIT_FAILURE;
|
||
|
}
|
||
|
|
||
|
nread = fread(&hdr, 1, sizeof(hdr), file);
|
||
|
if (nread != sizeof(hdr)) {
|
||
|
perror("Unable to read input file");
|
||
|
return EXIT_FAILURE;
|
||
|
}
|
||
|
|
||
|
if (memcmp(hdr.ehdr32.e_ident, ELFMAG, SELFMAG))
|
||
|
die("Input is not an ELF\n");
|
||
|
|
||
|
switch (hdr.ehdr32.e_ident[EI_CLASS]) {
|
||
|
case ELFCLASS32:
|
||
|
switch (hdr.ehdr32.e_ident[EI_DATA]) {
|
||
|
case ELFDATA2LSB:
|
||
|
entry = le32toh(hdr.ehdr32.e_entry);
|
||
|
break;
|
||
|
case ELFDATA2MSB:
|
||
|
entry = be32toh(hdr.ehdr32.e_entry);
|
||
|
break;
|
||
|
default:
|
||
|
die("Invalid ELF encoding\n");
|
||
|
}
|
||
|
|
||
|
/* Sign extend to form a canonical address */
|
||
|
entry = (int64_t)(int32_t)entry;
|
||
|
break;
|
||
|
|
||
|
case ELFCLASS64:
|
||
|
switch (hdr.ehdr32.e_ident[EI_DATA]) {
|
||
|
case ELFDATA2LSB:
|
||
|
entry = le64toh(hdr.ehdr64.e_entry);
|
||
|
break;
|
||
|
case ELFDATA2MSB:
|
||
|
entry = be64toh(hdr.ehdr64.e_entry);
|
||
|
break;
|
||
|
default:
|
||
|
die("Invalid ELF encoding\n");
|
||
|
}
|
||
|
break;
|
||
|
|
||
|
default:
|
||
|
die("Invalid ELF class\n");
|
||
|
}
|
||
|
|
||
|
printf("0x%016" PRIx64 "\n", entry);
|
||
|
return EXIT_SUCCESS;
|
||
|
}
|