#define ELFSIZE 64
#define MB (1024 * 1024)
#define ELF_ALIGN(x) (((x) + 7) & (~7))
#include <lib/libsa/stand.h>
#include <sys/param.h>
#include <sys/exec_elf.h>
#include <machine/boot_flag.h>
#ifdef SOFTRAID
#include <sys/queue.h>
#include <sys/disklabel.h>
#include <dev/biovar.h>
#include <dev/softraidvar.h>
#include "disk.h"
#include "softraid_sparc64.h"
#endif
#include <lib/libsa/arc4.h>
#include "openfirm.h"
extern int boothowto;
void syncicache(void *, int);
int
elf64_exec(int fd, Elf_Ehdr *elf, u_int64_t *entryp, void **ssymp, void **esymp){
Elf_Shdr *shp;
Elf_Off off;
void *addr;
size_t size;
u_int align;
int i, first = 1;
struct openbsd_bootdata *obd;
#ifdef SOFTRAID
struct sr_boot_volume *bv;
#endif
for (i = 0; i < elf->e_phnum; i++) {
Elf_Phdr phdr;
size = lseek(fd, (size_t)(elf->e_phoff + sizeof(phdr) * i),
SEEK_SET);
if (read(fd, (void *)&phdr, sizeof(phdr)) != sizeof(phdr)) {
printf("read phdr: %s\n", strerror(errno));
return (1);
}
if (phdr.p_type == PT_OPENBSD_BOOTDATA) {
memset((void *) (long)phdr.p_paddr, 0, phdr.p_filesz);
if (phdr.p_filesz < BOOTDATA_LEN_BOOTHOWTO)
continue;
obd = (struct openbsd_bootdata *)(long)phdr.p_paddr;
obd->version = BOOTDATA_VERSION;
obd->len = sizeof(struct openbsd_bootdata);
#ifdef SOFTRAID
if (bootdev_dip && bootdev_dip->sr_vol) {
bv = bootdev_dip->sr_vol;
memcpy(obd->sr_uuid, bv->sbv_uuid.sui_id,
sizeof(obd->sr_uuid));
if (bv->sbv_maskkey)
memcpy(obd->sr_maskkey, bv->sbv_maskkey,
sizeof(obd->sr_maskkey));
}
#endif
if (phdr.p_filesz < sizeof(struct openbsd_bootdata))
continue;
obd->boothowto = boothowto;
continue;
}
if (phdr.p_type == PT_OPENBSD_RANDOMIZE) {
extern struct rc4_ctx randomctx;
rc4_getbytes(&randomctx, (void *)(long)phdr.p_paddr,
phdr.p_filesz);
}
if (phdr.p_type != PT_LOAD ||
(phdr.p_flags & (PF_W|PF_X)) == 0)
continue;
printf("%s%lu@0x%lx", first ? "" : "+", (u_long)phdr.p_filesz,
(u_long)phdr.p_vaddr);
(void)lseek(fd, (size_t)phdr.p_offset, SEEK_SET);
align = phdr.p_align;
if ((phdr.p_vaddr & (4 * MB - 1)) == 0)
align = 4 * MB;
if (phdr.p_filesz < phdr.p_memsz)
phdr.p_memsz = roundup(phdr.p_memsz, 4 * MB);
phdr.p_memsz = roundup(phdr.p_memsz, PAGE_SIZE);
if (OF_claim((void *)(long)phdr.p_vaddr, phdr.p_memsz, align) ==
(void *)-1)
panic("cannot claim memory");
if (read(fd, (void *)(long)phdr.p_vaddr, phdr.p_filesz) !=
phdr.p_filesz) {
printf("read segment: %s\n", strerror(errno));
return (1);
}
if (phdr.p_flags & PF_X)
syncicache((void *)(long)phdr.p_vaddr, phdr.p_filesz);
if (phdr.p_filesz < phdr.p_memsz) {
printf("+%lu@0x%lx",
(u_long)phdr.p_memsz - phdr.p_filesz,
(u_long)(phdr.p_vaddr + phdr.p_filesz));
bzero((void *)(long)phdr.p_vaddr + phdr.p_filesz,
(size_t)phdr.p_memsz - phdr.p_filesz);
}
first = 0;
}
printf(" \n");
size = sizeof(Elf_Ehdr) + (elf->e_shnum * sizeof(Elf_Shdr));
shp = addr = alloc(elf->e_shnum * sizeof(Elf_Shdr));
(void)lseek(fd, (off_t)elf->e_shoff, SEEK_SET);
if (read(fd, addr, (size_t)(elf->e_shnum * sizeof(Elf_Shdr))) !=
elf->e_shnum * sizeof(Elf_Shdr)) {
printf("read section headers: %s\n", strerror(errno));
return (1);
}
size_t shstrsz = shp[elf->e_shstrndx].sh_size;
char *shstr = alloc(shstrsz);
if (lseek(fd, (off_t)shp[elf->e_shstrndx].sh_offset, SEEK_SET) == -1) {
printf("lseek section header string table: %s\n", strerror(errno));
return 1;
}
if (read(fd, shstr, shstrsz) != shstrsz) {
printf("read section header string table: %s\n", strerror(errno));
return 1;
}
for (i = 0; i < elf->e_shnum; i++, shp++) {
if (shp->sh_type == SHT_NULL)
continue;
if (shp->sh_type != SHT_SYMTAB
&& shp->sh_type != SHT_STRTAB
&& strcmp(shstr + shp->sh_name, ELF_CTF)) {
shp->sh_offset = 0;
continue;
}
size += shp->sh_size;
}
shp = addr;
if ((addr = OF_claim(0, roundup(size, PAGE_SIZE), PAGE_SIZE)) == (void *)-1)
panic("no space for symbol table");
elf->e_phoff = 0;
elf->e_shoff = sizeof(Elf_Ehdr);
elf->e_phentsize = 0;
elf->e_phnum = 0;
bcopy(elf, addr, sizeof(Elf_Ehdr));
bcopy(shp, addr + sizeof(Elf_Ehdr), elf->e_shnum * sizeof(Elf_Shdr));
free(shp, elf->e_shnum * sizeof(Elf_Shdr));
*ssymp = addr;
shp = addr + sizeof(Elf_Ehdr);
size = sizeof(Elf_Ehdr) + (elf->e_shnum * sizeof(Elf_Shdr));
size = ELF_ALIGN(size);
addr += size;
off = size;
for (first = 1, i = 0; i < elf->e_shnum; i++, shp++) {
if (shp->sh_type == SHT_SYMTAB
|| shp->sh_type == SHT_STRTAB
|| !strcmp(shstr + shp->sh_name, ELF_CTF)) {
if (first)
printf("symbols @ 0x%lx ", (u_long)addr);
printf("%s%d", first ? "" : "+", (int)shp->sh_size);
(void)lseek(fd, shp->sh_offset, SEEK_SET);
if (read(fd, addr, shp->sh_size) != shp->sh_size) {
printf("read symbols: %s\n", strerror(errno));
return (1);
}
addr += ELF_ALIGN(shp->sh_size);
shp->sh_offset = off;
shp->sh_flags |= SHF_ALLOC;
off += ELF_ALIGN(shp->sh_size);
first = 0;
}
}
*esymp = addr;
*entryp = elf->e_entry;
return (0);
}
#undef ELF_ALIGN