/* $OpenBSD: archdep.h,v 1.5 2006/05/08 20:37:01 deraadt Exp $ */ /* * Copyright (c) 1998-2002 Opsycon AB, Sweden. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS * OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF * SUCH DAMAGE. * */ #ifndef _MIPS_ARCHDEP_H_ #define _MIPS_ARCHDEP_H_ #include #include "syscall.h" #include "resolve.h" #include "util.h" #define DL_MALLOC_ALIGN 8 /* Arch constraint or otherwise */ #define MACHID EM_MIPS /* ELF e_machine ID value checked */ #define RELOC_REL(relp, symp, adrp, val) \ do { \ _dl_mprotect(adrp, 8, PROT_EXEC|PROT_READ|PROT_WRITE); \ if (ELF64_R_TYPE(relp->r_info) == R_MIPS_REL32_64) { \ if (ELF64_R_SYM(rp->r_info) != 0) \ *adrp = symp->st_value + val; \ else \ *adrp += val; \ } else if (ELF64_R_TYPE(relp->r_info) != R_MIPS_NONE) { \ _dl_exit(ELF64_R_TYPE(relp->r_info)+100); \ } \ } while (0) #define RELOC_RELA(rela, sym, ptr, val) \ do { \ _dl_exit(20); /* We don't do RELA now */ \ } while (0) struct elf_object; #define RELOC_GOT(obj, off) \ do { \ struct elf_object *__dynld = obj; \ long __loff = off; \ Elf64_Addr *gotp; \ int i, n; \ const Elf_Sym *sp; \ \ /* Do all local gots */ \ gotp = __dynld->dyn.pltgot; \ n = __dynld->Dyn.info[DT_MIPS_LOCAL_GOTNO - DT_LOPROC + DT_NUM];\ \ for (i = ((gotp[1] & 0x0000000080000000) ? 2 : 1); i < n; i++) {\ gotp[i] += __loff; \ } \ gotp += n; \ \ /* Do symbol referencing gots. There should be no global... */ \ n = __dynld->Dyn.info[DT_MIPS_SYMTABNO - DT_LOPROC + DT_NUM] - \ __dynld->Dyn.info[DT_MIPS_GOTSYM - DT_LOPROC + DT_NUM]; \ sp = __dynld->dyn.symtab; \ sp += __dynld->Dyn.info[DT_MIPS_GOTSYM - DT_LOPROC + DT_NUM]; \ \ while (n--) { \ if (sp->st_shndx == SHN_UNDEF || \ sp->st_shndx == SHN_COMMON) { \ _dl_exit(6); \ } else if (ELF64_ST_TYPE(sp->st_info) == STT_FUNC) { \ *gotp += __loff; \ } else { \ *gotp = sp->st_value + __loff; \ } \ gotp++; \ sp++; \ } \ __dynld->status |= STAT_GOT_DONE; \ } while (0) #define GOT_PERMS PROT_READ #endif /* _MIPS_ARCHDEP_H_ */