1164190Sjkoshy/*- 2164190Sjkoshy * Copyright (c) 2006 Joseph Koshy 3164190Sjkoshy * All rights reserved. 4164190Sjkoshy * 5164190Sjkoshy * Redistribution and use in source and binary forms, with or without 6164190Sjkoshy * modification, are permitted provided that the following conditions 7164190Sjkoshy * are met: 8164190Sjkoshy * 1. Redistributions of source code must retain the above copyright 9164190Sjkoshy * notice, this list of conditions and the following disclaimer. 10164190Sjkoshy * 2. Redistributions in binary form must reproduce the above copyright 11164190Sjkoshy * notice, this list of conditions and the following disclaimer in the 12164190Sjkoshy * documentation and/or other materials provided with the distribution. 13164190Sjkoshy * 14164190Sjkoshy * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND 15164190Sjkoshy * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 16164190Sjkoshy * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 17164190Sjkoshy * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE 18164190Sjkoshy * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 19164190Sjkoshy * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 20164190Sjkoshy * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 21164190Sjkoshy * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 22164190Sjkoshy * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 23164190Sjkoshy * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 24164190Sjkoshy * SUCH DAMAGE. 25164190Sjkoshy */ 26164190Sjkoshy 27164190Sjkoshy#include <sys/cdefs.h> 28164190Sjkoshy__FBSDID("$FreeBSD$"); 29164190Sjkoshy 30164190Sjkoshy#include <sys/limits.h> 31164190Sjkoshy 32164190Sjkoshy#include <assert.h> 33164190Sjkoshy#include <gelf.h> 34164190Sjkoshy 35164190Sjkoshy#include "_libelf.h" 36164190Sjkoshy 37164190SjkoshyGElf_Sym * 38164190Sjkoshygelf_getsym(Elf_Data *d, int ndx, GElf_Sym *dst) 39164190Sjkoshy{ 40164190Sjkoshy int ec; 41164190Sjkoshy Elf *e; 42164190Sjkoshy Elf_Scn *scn; 43164190Sjkoshy Elf32_Sym *sym32; 44164190Sjkoshy Elf64_Sym *sym64; 45164190Sjkoshy size_t msz; 46164190Sjkoshy uint32_t sh_type; 47164190Sjkoshy 48164190Sjkoshy if (d == NULL || ndx < 0 || dst == NULL || 49164190Sjkoshy (scn = d->d_scn) == NULL || 50164190Sjkoshy (e = scn->s_elf) == NULL) { 51164190Sjkoshy LIBELF_SET_ERROR(ARGUMENT, 0); 52164190Sjkoshy return (NULL); 53164190Sjkoshy } 54164190Sjkoshy 55164190Sjkoshy ec = e->e_class; 56164190Sjkoshy assert(ec == ELFCLASS32 || ec == ELFCLASS64); 57164190Sjkoshy 58164190Sjkoshy if (ec == ELFCLASS32) 59164190Sjkoshy sh_type = scn->s_shdr.s_shdr32.sh_type; 60164190Sjkoshy else 61164190Sjkoshy sh_type = scn->s_shdr.s_shdr64.sh_type; 62164190Sjkoshy 63164190Sjkoshy if (_libelf_xlate_shtype(sh_type) != ELF_T_SYM) { 64164190Sjkoshy LIBELF_SET_ERROR(ARGUMENT, 0); 65164190Sjkoshy return (NULL); 66164190Sjkoshy } 67164190Sjkoshy 68164190Sjkoshy msz = _libelf_msize(ELF_T_SYM, ec, e->e_version); 69164190Sjkoshy 70164190Sjkoshy assert(msz > 0); 71164190Sjkoshy 72164190Sjkoshy if (msz * ndx >= d->d_size) { 73164190Sjkoshy LIBELF_SET_ERROR(ARGUMENT, 0); 74164190Sjkoshy return (NULL); 75164190Sjkoshy } 76164190Sjkoshy 77164190Sjkoshy if (ec == ELFCLASS32) { 78164190Sjkoshy 79164190Sjkoshy sym32 = (Elf32_Sym *) d->d_buf + ndx; 80164190Sjkoshy 81164190Sjkoshy dst->st_name = sym32->st_name; 82164190Sjkoshy dst->st_value = (Elf64_Addr) sym32->st_value; 83164190Sjkoshy dst->st_size = (Elf64_Xword) sym32->st_size; 84164190Sjkoshy dst->st_info = ELF64_ST_INFO(ELF32_ST_BIND(sym32->st_info), 85164190Sjkoshy ELF32_ST_TYPE(sym32->st_info)); 86164190Sjkoshy dst->st_other = sym32->st_other; 87164190Sjkoshy dst->st_shndx = sym32->st_shndx; 88164190Sjkoshy } else { 89164190Sjkoshy 90164190Sjkoshy sym64 = (Elf64_Sym *) d->d_buf + ndx; 91164190Sjkoshy 92164190Sjkoshy *dst = *sym64; 93164190Sjkoshy } 94164190Sjkoshy 95164190Sjkoshy return (dst); 96164190Sjkoshy} 97164190Sjkoshy 98164190Sjkoshyint 99164190Sjkoshygelf_update_sym(Elf_Data *d, int ndx, GElf_Sym *gs) 100164190Sjkoshy{ 101164190Sjkoshy int ec; 102164190Sjkoshy Elf *e; 103164190Sjkoshy Elf_Scn *scn; 104164190Sjkoshy Elf32_Sym *sym32; 105164190Sjkoshy Elf64_Sym *sym64; 106164190Sjkoshy size_t msz; 107164190Sjkoshy uint32_t sh_type; 108164190Sjkoshy 109164190Sjkoshy if (d == NULL || ndx < 0 || gs == NULL || 110164190Sjkoshy (scn = d->d_scn) == NULL || 111164190Sjkoshy (e = scn->s_elf) == NULL) { 112164190Sjkoshy LIBELF_SET_ERROR(ARGUMENT, 0); 113164190Sjkoshy return (0); 114164190Sjkoshy } 115164190Sjkoshy 116164190Sjkoshy ec = e->e_class; 117164190Sjkoshy assert(ec == ELFCLASS32 || ec == ELFCLASS64); 118164190Sjkoshy 119164190Sjkoshy if (ec == ELFCLASS32) 120164190Sjkoshy sh_type = scn->s_shdr.s_shdr32.sh_type; 121164190Sjkoshy else 122164190Sjkoshy sh_type = scn->s_shdr.s_shdr64.sh_type; 123164190Sjkoshy 124164190Sjkoshy if (_libelf_xlate_shtype(sh_type) != ELF_T_SYM) { 125164190Sjkoshy LIBELF_SET_ERROR(ARGUMENT, 0); 126164190Sjkoshy return (0); 127164190Sjkoshy } 128164190Sjkoshy 129164190Sjkoshy msz = _libelf_msize(ELF_T_SYM, ec, e->e_version); 130164190Sjkoshy assert(msz > 0); 131164190Sjkoshy 132164190Sjkoshy if (msz * ndx >= d->d_size) { 133164190Sjkoshy LIBELF_SET_ERROR(ARGUMENT, 0); 134164190Sjkoshy return (0); 135164190Sjkoshy } 136164190Sjkoshy 137164190Sjkoshy if (ec == ELFCLASS32) { 138164190Sjkoshy sym32 = (Elf32_Sym *) d->d_buf + ndx; 139164190Sjkoshy 140164190Sjkoshy sym32->st_name = gs->st_name; 141164190Sjkoshy sym32->st_info = gs->st_info; 142164190Sjkoshy sym32->st_other = gs->st_other; 143164190Sjkoshy sym32->st_shndx = gs->st_shndx; 144164190Sjkoshy 145164190Sjkoshy LIBELF_COPY_U32(sym32, gs, st_value); 146164190Sjkoshy LIBELF_COPY_U32(sym32, gs, st_size); 147164190Sjkoshy } else { 148164190Sjkoshy sym64 = (Elf64_Sym *) d->d_buf + ndx; 149164190Sjkoshy 150164190Sjkoshy *sym64 = *gs; 151164190Sjkoshy } 152164190Sjkoshy 153164190Sjkoshy return (1); 154164190Sjkoshy} 155