u-boot-brain/tools/prelink-riscv.c
Marcus Comstedt e604410d3e riscv: tools: Fix prelink-riscv to work on big endian hosts
All ELF fields whose values are inspected by the code are converted to
CPU byteorder first.  Values which are copied verbatim (relocation
fixups) are not swapped to CPU byteorder and back as it is not needed.

Signed-off-by: Marcus Comstedt <marcus@mc.pp.se>
Cc: Rick Chen <rick@andestech.com>
Reviewed-by: Rick Chen <rick@andestech.com>
2019-08-15 13:42:28 +08:00

99 lines
1.8 KiB
C

// SPDX-License-Identifier: GPL-2.0+
/*
* Copyright (C) 2017 Andes Technology
* Chih-Mao Chen <cmchen@andestech.com>
*
* Statically process runtime relocations on RISC-V ELF images
* so that it can be directly executed when loaded at LMA
* without fixup. Both RV32 and RV64 are supported.
*/
#include <errno.h>
#include <stdbool.h>
#include <stdint.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <elf.h>
#include <fcntl.h>
#include <sys/mman.h>
#include <sys/stat.h>
#include <sys/types.h>
#include <unistd.h>
#include <compiler.h>
#ifndef EM_RISCV
#define EM_RISCV 243
#endif
#ifndef R_RISCV_32
#define R_RISCV_32 1
#endif
#ifndef R_RISCV_64
#define R_RISCV_64 2
#endif
#ifndef R_RISCV_RELATIVE
#define R_RISCV_RELATIVE 3
#endif
const char *argv0;
#define die(fmt, ...) \
do { \
fprintf(stderr, "%s: " fmt "\n", argv0, ## __VA_ARGS__); \
exit(EXIT_FAILURE); \
} while (0)
#define PRELINK_INC_BITS 32
#include "prelink-riscv.inc"
#undef PRELINK_INC_BITS
#define PRELINK_INC_BITS 64
#include "prelink-riscv.inc"
#undef PRELINK_INC_BITS
int main(int argc, const char *const *argv)
{
argv0 = argv[0];
if (argc < 2) {
fprintf(stderr, "Usage: %s <u-boot>\n", argv0);
exit(EXIT_FAILURE);
}
int fd = open(argv[1], O_RDWR, 0);
if (fd < 0)
die("Cannot open %s: %s", argv[1], strerror(errno));
struct stat st;
if (fstat(fd, &st) < 0)
die("Cannot stat %s: %s", argv[1], strerror(errno));
void *data =
mmap(0, st.st_size, PROT_READ | PROT_WRITE, MAP_SHARED, fd, 0);
if (data == MAP_FAILED)
die("Cannot mmap %s: %s", argv[1], strerror(errno));
close(fd);
unsigned char *e_ident = (unsigned char *)data;
if (memcmp(e_ident, ELFMAG, SELFMAG) != 0)
die("Invalid ELF file %s", argv[1]);
bool is64 = e_ident[EI_CLASS] == ELFCLASS64;
if (is64)
prelink64(data);
else
prelink32(data);
return 0;
}