104 lines
		
	
	
		
			2.0 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			104 lines
		
	
	
		
			2.0 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");
 | |
| 		fclose(file);
 | |
| 		return EXIT_FAILURE;
 | |
| 	}
 | |
| 
 | |
| 	if (memcmp(hdr.ehdr32.e_ident, ELFMAG, SELFMAG)) {
 | |
| 		fclose(file);
 | |
| 		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:
 | |
| 			fclose(file);
 | |
| 			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:
 | |
| 			fclose(file);
 | |
| 			die("Invalid ELF encoding\n");
 | |
| 		}
 | |
| 		break;
 | |
| 
 | |
| 	default:
 | |
| 		fclose(file);
 | |
| 		die("Invalid ELF class\n");
 | |
| 	}
 | |
| 
 | |
| 	printf("0x%016" PRIx64 "\n", entry);
 | |
| 	fclose(file);
 | |
| 	return EXIT_SUCCESS;
 | |
| }
 |