#ifdef SUPPORT_ANDROID_PRELINK_TAGS #include #include #include #include #include #include #include #include #include typedef struct { uint32_t mmap_addr; char tag[4]; /* 'P', 'R', 'E', ' ' */ } __attribute__((packed)) prelink_info_t; static inline void set_prelink(long *prelink_addr, int elf_little, prelink_info_t *info) { FAILIF(sizeof(prelink_info_t) != 8, "Unexpected sizeof(prelink_info_t) == %zd!\n", sizeof(prelink_info_t)); if (prelink_addr) { if (!(elf_little ^ is_host_little())) { /* Same endianness */ *prelink_addr = info->mmap_addr; } else { /* Different endianness */ *prelink_addr = switch_endianness(info->mmap_addr); } } } int check_prelinked(const char *fname, int elf_little, long *prelink_addr) { FAILIF(sizeof(prelink_info_t) != 8, "Unexpected sizeof(prelink_info_t) == %zd!\n", sizeof(prelink_info_t)); int fd = open(fname, O_RDONLY); FAILIF(fd < 0, "open(%s, O_RDONLY): %s (%d)!\n", fname, strerror(errno), errno); off_t end = lseek(fd, 0, SEEK_END); #ifndef DEBUG (void)end; #endif int nr = sizeof(prelink_info_t); off_t sz = lseek(fd, -nr, SEEK_CUR); ASSERT((long)(end - sz) == (long)nr); FAILIF(sz == (off_t)-1, "lseek(%d, 0, SEEK_END): %s (%d)!\n", fd, strerror(errno), errno); prelink_info_t info; ssize_t num_read = read(fd, &info, nr); FAILIF(num_read < 0, "read(%d, &info, sizeof(prelink_info_t)): %s (%d)!\n", fd, strerror(errno), errno); FAILIF((size_t)num_read != sizeof(info), "read(%d, &info, sizeof(prelink_info_t)): did not read %zd bytes as " "expected (read %zd)!\n", fd, sizeof(info), (size_t)num_read); int prelinked = 0; if (!strncmp(info.tag, "PRE ", 4)) { set_prelink(prelink_addr, elf_little, &info); prelinked = 1; } FAILIF(close(fd) < 0, "close(%d): %s (%d)!\n", fd, strerror(errno), errno); return prelinked; } void setup_prelink_info(const char *fname, int elf_little, long base) { FAILIF(sizeof(prelink_info_t) != 8, "Unexpected sizeof(prelink_info_t) == %zd!\n", sizeof(prelink_info_t)); int fd = open(fname, O_WRONLY); FAILIF(fd < 0, "open(%s, O_WRONLY): %s (%d)\n" , fname, strerror(errno), errno); prelink_info_t info; off_t sz = lseek(fd, 0, SEEK_END); FAILIF(sz == (off_t)-1, "lseek(%d, 0, SEEK_END): %s (%d)!\n", fd, strerror(errno), errno); if (!(elf_little ^ is_host_little())) { /* Same endianness */ INFO("Host and ELF file [%s] have same endianness.\n", fname); info.mmap_addr = base; } else { /* Different endianness */ INFO("Host and ELF file [%s] have different endianness.\n", fname); info.mmap_addr = switch_endianness(base); } strncpy(info.tag, "PRE ", 4); ssize_t num_written = write(fd, &info, sizeof(info)); FAILIF(num_written < 0, "write(%d, &info, sizeof(info)): %s (%d)\n", fd, strerror(errno), errno); FAILIF(sizeof(info) != (size_t)num_written, "Could not write %zd bytes (wrote only %zd bytes) as expected!\n", sizeof(info), (size_t)num_written); FAILIF(close(fd) < 0, "close(%d): %s (%d)!\n", fd, strerror(errno), errno); } #endif /*SUPPORT_ANDROID_PRELINK_TAGS*/