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_move.c 2998 2014-03-18 17:19:00Z jkoshy $"); 372de3b87aSKai Wang 382de3b87aSKai Wang GElf_Move * 392de3b87aSKai Wang gelf_getmove(Elf_Data *ed, int ndx, GElf_Move *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_Move *move32; 472de3b87aSKai Wang Elf64_Move *move64; 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_MOVE) { 682de3b87aSKai Wang LIBELF_SET_ERROR(ARGUMENT, 0); 692de3b87aSKai Wang return (NULL); 702de3b87aSKai Wang } 712de3b87aSKai Wang 722de3b87aSKai Wang msz = _libelf_msize(ELF_T_MOVE, 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 842de3b87aSKai Wang move32 = (Elf32_Move *) d->d_data.d_buf + ndx; 852de3b87aSKai Wang 862de3b87aSKai Wang dst->m_value = move32->m_value; 872de3b87aSKai Wang dst->m_info = (Elf64_Xword) move32->m_info; 882de3b87aSKai Wang dst->m_poffset = (Elf64_Xword) move32->m_poffset; 892de3b87aSKai Wang dst->m_repeat = move32->m_repeat; 902de3b87aSKai Wang dst->m_stride = move32->m_stride; 912de3b87aSKai Wang } else { 922de3b87aSKai Wang 932de3b87aSKai Wang move64 = (Elf64_Move *) d->d_data.d_buf + ndx; 942de3b87aSKai Wang 952de3b87aSKai Wang *dst = *move64; 962de3b87aSKai Wang } 972de3b87aSKai Wang 982de3b87aSKai Wang return (dst); 992de3b87aSKai Wang } 1002de3b87aSKai Wang 1012de3b87aSKai Wang int 1022de3b87aSKai Wang gelf_update_move(Elf_Data *ed, int ndx, GElf_Move *gm) 1032de3b87aSKai Wang { 1042de3b87aSKai Wang int ec; 1052de3b87aSKai Wang Elf *e; 1062de3b87aSKai Wang size_t msz; 1072de3b87aSKai Wang Elf_Scn *scn; 1082de3b87aSKai Wang uint32_t sh_type; 1092de3b87aSKai Wang Elf32_Move *move32; 1102de3b87aSKai Wang Elf64_Move *move64; 1112de3b87aSKai Wang struct _Libelf_Data *d; 1122de3b87aSKai Wang 1132de3b87aSKai Wang d = (struct _Libelf_Data *) ed; 1142de3b87aSKai Wang 1152de3b87aSKai Wang if (d == NULL || ndx < 0 || gm == NULL || 1162de3b87aSKai Wang (scn = d->d_scn) == NULL || 1172de3b87aSKai Wang (e = scn->s_elf) == NULL) { 1182de3b87aSKai Wang LIBELF_SET_ERROR(ARGUMENT, 0); 1192de3b87aSKai Wang return (0); 1202de3b87aSKai Wang } 1212de3b87aSKai Wang 1222de3b87aSKai Wang ec = e->e_class; 1232de3b87aSKai Wang assert(ec == ELFCLASS32 || ec == ELFCLASS64); 1242de3b87aSKai Wang 1252de3b87aSKai Wang if (ec == ELFCLASS32) 1262de3b87aSKai Wang sh_type = scn->s_shdr.s_shdr32.sh_type; 1272de3b87aSKai Wang else 1282de3b87aSKai Wang sh_type = scn->s_shdr.s_shdr64.sh_type; 1292de3b87aSKai Wang 1302de3b87aSKai Wang if (_libelf_xlate_shtype(sh_type) != ELF_T_MOVE) { 1312de3b87aSKai Wang LIBELF_SET_ERROR(ARGUMENT, 0); 1322de3b87aSKai Wang return (0); 1332de3b87aSKai Wang } 1342de3b87aSKai Wang 1352de3b87aSKai Wang msz = _libelf_msize(ELF_T_MOVE, ec, e->e_version); 1362de3b87aSKai Wang 137cf781b2eSEd Maste assert(msz > 0); 138cf781b2eSEd Maste assert(ndx >= 0); 139cf781b2eSEd Maste 140cf781b2eSEd Maste if (msz * (size_t) ndx >= d->d_data.d_size) { 1412de3b87aSKai Wang LIBELF_SET_ERROR(ARGUMENT, 0); 1422de3b87aSKai Wang return (0); 1432de3b87aSKai Wang } 1442de3b87aSKai Wang 1452de3b87aSKai Wang if (ec == ELFCLASS32) { 1462de3b87aSKai Wang move32 = (Elf32_Move *) d->d_data.d_buf + ndx; 1472de3b87aSKai Wang 1482de3b87aSKai Wang move32->m_value = gm->m_value; 1492de3b87aSKai Wang LIBELF_COPY_U32(move32, gm, m_info); 1502de3b87aSKai Wang LIBELF_COPY_U32(move32, gm, m_poffset); 1512de3b87aSKai Wang move32->m_repeat = gm->m_repeat; 1522de3b87aSKai Wang move32->m_stride = gm->m_stride; 1532de3b87aSKai Wang 1542de3b87aSKai Wang } else { 1552de3b87aSKai Wang move64 = (Elf64_Move *) d->d_data.d_buf + ndx; 1562de3b87aSKai Wang 1572de3b87aSKai Wang *move64 = *gm; 1582de3b87aSKai Wang } 1592de3b87aSKai Wang 1602de3b87aSKai Wang return (1); 1612de3b87aSKai Wang } 162