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 <assert.h> 282de3b87aSKai Wang #include <libelf.h> 292de3b87aSKai Wang 302de3b87aSKai Wang #include "_libelf.h" 312de3b87aSKai Wang 3267d97fe7SEd Maste ELFTC_VCSID("$Id: libelf_extended.c 3174 2015-03-27 17:13:41Z emaste $"); 332de3b87aSKai Wang 342de3b87aSKai Wang /* 352de3b87aSKai Wang * Retrieve section #0, allocating a new section if needed. 362de3b87aSKai Wang */ 372de3b87aSKai Wang static Elf_Scn * 382de3b87aSKai Wang _libelf_getscn0(Elf *e) 392de3b87aSKai Wang { 402de3b87aSKai Wang Elf_Scn *s; 412de3b87aSKai Wang 42*8c953901SMark Johnston if ((s = RB_MIN(scntree, &e->e_u.e_elf.e_scn)) != NULL) 432de3b87aSKai Wang return (s); 442de3b87aSKai Wang 452de3b87aSKai Wang return (_libelf_allocate_scn(e, (size_t) SHN_UNDEF)); 462de3b87aSKai Wang } 472de3b87aSKai Wang 482de3b87aSKai Wang int 492de3b87aSKai Wang _libelf_setshnum(Elf *e, void *eh, int ec, size_t shnum) 502de3b87aSKai Wang { 512de3b87aSKai Wang Elf_Scn *scn; 522de3b87aSKai Wang 532de3b87aSKai Wang if (shnum >= SHN_LORESERVE) { 542de3b87aSKai Wang if ((scn = _libelf_getscn0(e)) == NULL) 552de3b87aSKai Wang return (0); 562de3b87aSKai Wang 572de3b87aSKai Wang assert(scn->s_ndx == SHN_UNDEF); 582de3b87aSKai Wang 592de3b87aSKai Wang if (ec == ELFCLASS32) 602de3b87aSKai Wang scn->s_shdr.s_shdr32.sh_size = shnum; 612de3b87aSKai Wang else 622de3b87aSKai Wang scn->s_shdr.s_shdr64.sh_size = shnum; 632de3b87aSKai Wang 642de3b87aSKai Wang (void) elf_flagshdr(scn, ELF_C_SET, ELF_F_DIRTY); 652de3b87aSKai Wang 662de3b87aSKai Wang shnum = 0; 672de3b87aSKai Wang } 682de3b87aSKai Wang 692de3b87aSKai Wang if (ec == ELFCLASS32) 70cf781b2eSEd Maste ((Elf32_Ehdr *) eh)->e_shnum = shnum & 0xFFFFU; 712de3b87aSKai Wang else 72cf781b2eSEd Maste ((Elf64_Ehdr *) eh)->e_shnum = shnum & 0xFFFFU; 732de3b87aSKai Wang 742de3b87aSKai Wang 752de3b87aSKai Wang return (1); 762de3b87aSKai Wang } 772de3b87aSKai Wang 782de3b87aSKai Wang int 792de3b87aSKai Wang _libelf_setshstrndx(Elf *e, void *eh, int ec, size_t shstrndx) 802de3b87aSKai Wang { 812de3b87aSKai Wang Elf_Scn *scn; 822de3b87aSKai Wang 832de3b87aSKai Wang if (shstrndx >= SHN_LORESERVE) { 842de3b87aSKai Wang if ((scn = _libelf_getscn0(e)) == NULL) 852de3b87aSKai Wang return (0); 862de3b87aSKai Wang 872de3b87aSKai Wang assert(scn->s_ndx == SHN_UNDEF); 882de3b87aSKai Wang 892de3b87aSKai Wang if (ec == ELFCLASS32) 902de3b87aSKai Wang scn->s_shdr.s_shdr32.sh_link = shstrndx; 912de3b87aSKai Wang else 922de3b87aSKai Wang scn->s_shdr.s_shdr64.sh_link = shstrndx; 932de3b87aSKai Wang 942de3b87aSKai Wang (void) elf_flagshdr(scn, ELF_C_SET, ELF_F_DIRTY); 952de3b87aSKai Wang 962de3b87aSKai Wang shstrndx = SHN_XINDEX; 972de3b87aSKai Wang } 982de3b87aSKai Wang 992de3b87aSKai Wang if (ec == ELFCLASS32) 100cf781b2eSEd Maste ((Elf32_Ehdr *) eh)->e_shstrndx = shstrndx & 0xFFFFU; 1012de3b87aSKai Wang else 102cf781b2eSEd Maste ((Elf64_Ehdr *) eh)->e_shstrndx = shstrndx & 0xFFFFU; 1032de3b87aSKai Wang 1042de3b87aSKai Wang return (1); 1052de3b87aSKai Wang } 1062de3b87aSKai Wang 1072de3b87aSKai Wang int 1082de3b87aSKai Wang _libelf_setphnum(Elf *e, void *eh, int ec, size_t phnum) 1092de3b87aSKai Wang { 1102de3b87aSKai Wang Elf_Scn *scn; 1112de3b87aSKai Wang 1122de3b87aSKai Wang if (phnum >= PN_XNUM) { 1132de3b87aSKai Wang if ((scn = _libelf_getscn0(e)) == NULL) 1142de3b87aSKai Wang return (0); 1152de3b87aSKai Wang 1162de3b87aSKai Wang assert(scn->s_ndx == SHN_UNDEF); 1172de3b87aSKai Wang 1182de3b87aSKai Wang if (ec == ELFCLASS32) 1192de3b87aSKai Wang scn->s_shdr.s_shdr32.sh_info = phnum; 1202de3b87aSKai Wang else 1212de3b87aSKai Wang scn->s_shdr.s_shdr64.sh_info = phnum; 1222de3b87aSKai Wang 1232de3b87aSKai Wang (void) elf_flagshdr(scn, ELF_C_SET, ELF_F_DIRTY); 1242de3b87aSKai Wang 1252de3b87aSKai Wang phnum = PN_XNUM; 1262de3b87aSKai Wang } 1272de3b87aSKai Wang 1282de3b87aSKai Wang if (ec == ELFCLASS32) 129cf781b2eSEd Maste ((Elf32_Ehdr *) eh)->e_phnum = phnum & 0xFFFFU; 1302de3b87aSKai Wang else 131cf781b2eSEd Maste ((Elf64_Ehdr *) eh)->e_phnum = phnum & 0xFFFFU; 1322de3b87aSKai Wang 1332de3b87aSKai Wang return (1); 1342de3b87aSKai Wang } 135