12de3b87aSKai Wang /*- 22de3b87aSKai Wang * Copyright (c) 2006,2008 Joseph Koshy 32de3b87aSKai Wang * All rights reserved. 42de3b87aSKai Wang * 52de3b87aSKai Wang * Redistribution and use in source and binary forms, with or without 62de3b87aSKai Wang * modification, are permitted provided that the following conditions 72de3b87aSKai Wang * are met: 82de3b87aSKai Wang * 1. Redistributions of source code must retain the above copyright 92de3b87aSKai Wang * notice, this list of conditions and the following disclaimer. 102de3b87aSKai Wang * 2. Redistributions in binary form must reproduce the above copyright 112de3b87aSKai Wang * notice, this list of conditions and the following disclaimer in the 122de3b87aSKai Wang * documentation and/or other materials provided with the distribution. 132de3b87aSKai Wang * 142de3b87aSKai Wang * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND 152de3b87aSKai Wang * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 162de3b87aSKai Wang * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 172de3b87aSKai Wang * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE 182de3b87aSKai Wang * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 192de3b87aSKai Wang * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 202de3b87aSKai Wang * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 212de3b87aSKai Wang * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 222de3b87aSKai Wang * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 232de3b87aSKai Wang * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 242de3b87aSKai Wang * SUCH DAMAGE. 252de3b87aSKai Wang */ 262de3b87aSKai Wang 272de3b87aSKai Wang #include <sys/cdefs.h> 282de3b87aSKai Wang 292de3b87aSKai Wang #include <assert.h> 302de3b87aSKai Wang #include <gelf.h> 312de3b87aSKai Wang #include <limits.h> 32*b4e9f239SEd Maste #include <stdint.h> 332de3b87aSKai Wang 342de3b87aSKai Wang #include "_libelf.h" 352de3b87aSKai Wang 36cf781b2eSEd Maste ELFTC_VCSID("$Id: gelf_sym.c 2999 2014-03-18 17:19:06Z jkoshy $"); 372de3b87aSKai Wang 382de3b87aSKai Wang GElf_Sym * 392de3b87aSKai Wang gelf_getsym(Elf_Data *ed, int ndx, GElf_Sym *dst) 402de3b87aSKai Wang { 412de3b87aSKai Wang int ec; 422de3b87aSKai Wang Elf *e; 432de3b87aSKai Wang size_t msz; 442de3b87aSKai Wang Elf_Scn *scn; 452de3b87aSKai Wang uint32_t sh_type; 462de3b87aSKai Wang Elf32_Sym *sym32; 472de3b87aSKai Wang Elf64_Sym *sym64; 482de3b87aSKai Wang struct _Libelf_Data *d; 492de3b87aSKai Wang 502de3b87aSKai Wang d = (struct _Libelf_Data *) ed; 512de3b87aSKai Wang 522de3b87aSKai Wang if (d == NULL || ndx < 0 || dst == NULL || 532de3b87aSKai Wang (scn = d->d_scn) == NULL || 542de3b87aSKai Wang (e = scn->s_elf) == NULL) { 552de3b87aSKai Wang LIBELF_SET_ERROR(ARGUMENT, 0); 562de3b87aSKai Wang return (NULL); 572de3b87aSKai Wang } 582de3b87aSKai Wang 592de3b87aSKai Wang ec = e->e_class; 602de3b87aSKai Wang assert(ec == ELFCLASS32 || ec == ELFCLASS64); 612de3b87aSKai Wang 622de3b87aSKai Wang if (ec == ELFCLASS32) 632de3b87aSKai Wang sh_type = scn->s_shdr.s_shdr32.sh_type; 642de3b87aSKai Wang else 652de3b87aSKai Wang sh_type = scn->s_shdr.s_shdr64.sh_type; 662de3b87aSKai Wang 672de3b87aSKai Wang if (_libelf_xlate_shtype(sh_type) != ELF_T_SYM) { 682de3b87aSKai Wang LIBELF_SET_ERROR(ARGUMENT, 0); 692de3b87aSKai Wang return (NULL); 702de3b87aSKai Wang } 712de3b87aSKai Wang 722de3b87aSKai Wang msz = _libelf_msize(ELF_T_SYM, ec, e->e_version); 732de3b87aSKai Wang 742de3b87aSKai Wang assert(msz > 0); 75cf781b2eSEd Maste assert(ndx >= 0); 762de3b87aSKai Wang 77cf781b2eSEd Maste if (msz * (size_t) ndx >= d->d_data.d_size) { 782de3b87aSKai Wang LIBELF_SET_ERROR(ARGUMENT, 0); 792de3b87aSKai Wang return (NULL); 802de3b87aSKai Wang } 812de3b87aSKai Wang 822de3b87aSKai Wang if (ec == ELFCLASS32) { 832de3b87aSKai Wang sym32 = (Elf32_Sym *) d->d_data.d_buf + ndx; 842de3b87aSKai Wang 852de3b87aSKai Wang dst->st_name = sym32->st_name; 862de3b87aSKai Wang dst->st_value = (Elf64_Addr) sym32->st_value; 872de3b87aSKai Wang dst->st_size = (Elf64_Xword) sym32->st_size; 88cf781b2eSEd Maste dst->st_info = sym32->st_info; 892de3b87aSKai Wang dst->st_other = sym32->st_other; 902de3b87aSKai Wang dst->st_shndx = sym32->st_shndx; 912de3b87aSKai Wang } else { 922de3b87aSKai Wang sym64 = (Elf64_Sym *) d->d_data.d_buf + ndx; 932de3b87aSKai Wang 942de3b87aSKai Wang *dst = *sym64; 952de3b87aSKai Wang } 962de3b87aSKai Wang 972de3b87aSKai Wang return (dst); 982de3b87aSKai Wang } 992de3b87aSKai Wang 1002de3b87aSKai Wang int 1012de3b87aSKai Wang gelf_update_sym(Elf_Data *ed, int ndx, GElf_Sym *gs) 1022de3b87aSKai Wang { 1032de3b87aSKai Wang int ec; 1042de3b87aSKai Wang Elf *e; 1052de3b87aSKai Wang size_t msz; 1062de3b87aSKai Wang Elf_Scn *scn; 1072de3b87aSKai Wang uint32_t sh_type; 1082de3b87aSKai Wang Elf32_Sym *sym32; 1092de3b87aSKai Wang Elf64_Sym *sym64; 1102de3b87aSKai Wang struct _Libelf_Data *d; 1112de3b87aSKai Wang 1122de3b87aSKai Wang d = (struct _Libelf_Data *) ed; 1132de3b87aSKai Wang 1142de3b87aSKai Wang if (d == NULL || ndx < 0 || gs == NULL || 1152de3b87aSKai Wang (scn = d->d_scn) == NULL || 1162de3b87aSKai Wang (e = scn->s_elf) == NULL) { 1172de3b87aSKai Wang LIBELF_SET_ERROR(ARGUMENT, 0); 1182de3b87aSKai Wang return (0); 1192de3b87aSKai Wang } 1202de3b87aSKai Wang 1212de3b87aSKai Wang ec = e->e_class; 1222de3b87aSKai Wang assert(ec == ELFCLASS32 || ec == ELFCLASS64); 1232de3b87aSKai Wang 1242de3b87aSKai Wang if (ec == ELFCLASS32) 1252de3b87aSKai Wang sh_type = scn->s_shdr.s_shdr32.sh_type; 1262de3b87aSKai Wang else 1272de3b87aSKai Wang sh_type = scn->s_shdr.s_shdr64.sh_type; 1282de3b87aSKai Wang 1292de3b87aSKai Wang if (_libelf_xlate_shtype(sh_type) != ELF_T_SYM) { 1302de3b87aSKai Wang LIBELF_SET_ERROR(ARGUMENT, 0); 1312de3b87aSKai Wang return (0); 1322de3b87aSKai Wang } 1332de3b87aSKai Wang 1342de3b87aSKai Wang msz = _libelf_msize(ELF_T_SYM, ec, e->e_version); 1352de3b87aSKai Wang 136cf781b2eSEd Maste assert(msz > 0); 137cf781b2eSEd Maste assert(ndx >= 0); 138cf781b2eSEd Maste 139cf781b2eSEd Maste if (msz * (size_t) ndx >= d->d_data.d_size) { 1402de3b87aSKai Wang LIBELF_SET_ERROR(ARGUMENT, 0); 1412de3b87aSKai Wang return (0); 1422de3b87aSKai Wang } 1432de3b87aSKai Wang 1442de3b87aSKai Wang if (ec == ELFCLASS32) { 1452de3b87aSKai Wang sym32 = (Elf32_Sym *) d->d_data.d_buf + ndx; 1462de3b87aSKai Wang 1472de3b87aSKai Wang sym32->st_name = gs->st_name; 1482de3b87aSKai Wang sym32->st_info = gs->st_info; 1492de3b87aSKai Wang sym32->st_other = gs->st_other; 1502de3b87aSKai Wang sym32->st_shndx = gs->st_shndx; 1512de3b87aSKai Wang 1522de3b87aSKai Wang LIBELF_COPY_U32(sym32, gs, st_value); 1532de3b87aSKai Wang LIBELF_COPY_U32(sym32, gs, st_size); 1542de3b87aSKai Wang } else { 1552de3b87aSKai Wang sym64 = (Elf64_Sym *) d->d_data.d_buf + ndx; 1562de3b87aSKai Wang 1572de3b87aSKai Wang *sym64 = *gs; 1582de3b87aSKai Wang } 1592de3b87aSKai Wang 1602de3b87aSKai Wang return (1); 1612de3b87aSKai Wang } 162