1*5ac3bc71Schristos /* $NetBSD: libelf_phdr.c,v 1.5 2024/03/03 17:37:34 christos Exp $ */
2e81373b4Schristos
39dd9d0cfSchristos /*-
49dd9d0cfSchristos * Copyright (c) 2006,2008 Joseph Koshy
59dd9d0cfSchristos * All rights reserved.
69dd9d0cfSchristos *
79dd9d0cfSchristos * Redistribution and use in source and binary forms, with or without
89dd9d0cfSchristos * modification, are permitted provided that the following conditions
99dd9d0cfSchristos * are met:
109dd9d0cfSchristos * 1. Redistributions of source code must retain the above copyright
119dd9d0cfSchristos * notice, this list of conditions and the following disclaimer.
129dd9d0cfSchristos * 2. Redistributions in binary form must reproduce the above copyright
139dd9d0cfSchristos * notice, this list of conditions and the following disclaimer in the
149dd9d0cfSchristos * documentation and/or other materials provided with the distribution.
159dd9d0cfSchristos *
169dd9d0cfSchristos * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
179dd9d0cfSchristos * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
189dd9d0cfSchristos * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
199dd9d0cfSchristos * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
209dd9d0cfSchristos * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
219dd9d0cfSchristos * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
229dd9d0cfSchristos * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
239dd9d0cfSchristos * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
249dd9d0cfSchristos * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
259dd9d0cfSchristos * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
269dd9d0cfSchristos * SUCH DAMAGE.
279dd9d0cfSchristos */
289dd9d0cfSchristos
29e81373b4Schristos #if HAVE_NBTOOL_CONFIG_H
30e81373b4Schristos # include "nbtool_config.h"
31e81373b4Schristos #endif
32e81373b4Schristos
339dd9d0cfSchristos #include <sys/cdefs.h>
349dd9d0cfSchristos
359dd9d0cfSchristos #include <assert.h>
369dd9d0cfSchristos #include <gelf.h>
379dd9d0cfSchristos #include <libelf.h>
389dd9d0cfSchristos #include <stdlib.h>
399dd9d0cfSchristos
409dd9d0cfSchristos #include "_libelf.h"
419dd9d0cfSchristos
42*5ac3bc71Schristos __RCSID("$NetBSD: libelf_phdr.c,v 1.5 2024/03/03 17:37:34 christos Exp $");
43*5ac3bc71Schristos ELFTC_VCSID("Id: libelf_phdr.c 3977 2022-05-01 06:45:34Z jkoshy");
449dd9d0cfSchristos
459dd9d0cfSchristos void *
_libelf_getphdr(Elf * e,int ec)469dd9d0cfSchristos _libelf_getphdr(Elf *e, int ec)
479dd9d0cfSchristos {
489dd9d0cfSchristos size_t phnum;
499dd9d0cfSchristos size_t fsz, msz;
509dd9d0cfSchristos uint64_t phoff;
519dd9d0cfSchristos Elf32_Ehdr *eh32;
529dd9d0cfSchristos Elf64_Ehdr *eh64;
539dd9d0cfSchristos void *ehdr, *phdr;
54*5ac3bc71Schristos _libelf_translator_function *xlator;
559dd9d0cfSchristos
569dd9d0cfSchristos assert(ec == ELFCLASS32 || ec == ELFCLASS64);
579dd9d0cfSchristos
589dd9d0cfSchristos if (e == NULL) {
599dd9d0cfSchristos LIBELF_SET_ERROR(ARGUMENT, 0);
609dd9d0cfSchristos return (NULL);
619dd9d0cfSchristos }
629dd9d0cfSchristos
639dd9d0cfSchristos if ((phdr = (ec == ELFCLASS32 ?
649dd9d0cfSchristos (void *) e->e_u.e_elf.e_phdr.e_phdr32 :
659dd9d0cfSchristos (void *) e->e_u.e_elf.e_phdr.e_phdr64)) != NULL)
669dd9d0cfSchristos return (phdr);
679dd9d0cfSchristos
689dd9d0cfSchristos /*
699dd9d0cfSchristos * Check the PHDR related fields in the EHDR for sanity.
709dd9d0cfSchristos */
719dd9d0cfSchristos
729dd9d0cfSchristos if ((ehdr = _libelf_ehdr(e, ec, 0)) == NULL)
739dd9d0cfSchristos return (NULL);
749dd9d0cfSchristos
759dd9d0cfSchristos phnum = e->e_u.e_elf.e_nphdr;
769dd9d0cfSchristos
779dd9d0cfSchristos if (ec == ELFCLASS32) {
789dd9d0cfSchristos eh32 = (Elf32_Ehdr *) ehdr;
799dd9d0cfSchristos phoff = (uint64_t) eh32->e_phoff;
809dd9d0cfSchristos } else {
819dd9d0cfSchristos eh64 = (Elf64_Ehdr *) ehdr;
829dd9d0cfSchristos phoff = (uint64_t) eh64->e_phoff;
839dd9d0cfSchristos }
849dd9d0cfSchristos
859dd9d0cfSchristos fsz = gelf_fsize(e, ELF_T_PHDR, phnum, e->e_version);
869dd9d0cfSchristos
879dd9d0cfSchristos assert(fsz > 0);
889dd9d0cfSchristos
89*5ac3bc71Schristos if (phoff + fsz < phoff) { /* Numeric overflow. */
90*5ac3bc71Schristos LIBELF_SET_ERROR(HEADER, 0);
91*5ac3bc71Schristos return (NULL);
92*5ac3bc71Schristos }
93*5ac3bc71Schristos
949dd9d0cfSchristos if ((uint64_t) e->e_rawsize < (phoff + fsz)) {
959dd9d0cfSchristos LIBELF_SET_ERROR(HEADER, 0);
969dd9d0cfSchristos return (NULL);
979dd9d0cfSchristos }
989dd9d0cfSchristos
99*5ac3bc71Schristos if ((msz = _libelf_msize(ELF_T_PHDR, ec, EV_CURRENT)) == 0)
100*5ac3bc71Schristos return (NULL);
1019dd9d0cfSchristos
1029dd9d0cfSchristos if ((phdr = calloc(phnum, msz)) == NULL) {
1039dd9d0cfSchristos LIBELF_SET_ERROR(RESOURCE, 0);
1049dd9d0cfSchristos return (NULL);
1059dd9d0cfSchristos }
1069dd9d0cfSchristos
1079dd9d0cfSchristos if (ec == ELFCLASS32)
1089dd9d0cfSchristos e->e_u.e_elf.e_phdr.e_phdr32 = phdr;
1099dd9d0cfSchristos else
1109dd9d0cfSchristos e->e_u.e_elf.e_phdr.e_phdr64 = phdr;
1119dd9d0cfSchristos
1129dd9d0cfSchristos
113*5ac3bc71Schristos xlator = _libelf_get_translator(ELF_T_PHDR, ELF_TOMEMORY, ec,
114*5ac3bc71Schristos _libelf_elfmachine(e));
1159dd9d0cfSchristos (*xlator)(phdr, phnum * msz, e->e_rawfile + phoff, phnum,
116*5ac3bc71Schristos e->e_byteorder != LIBELF_PRIVATE(byteorder));
1179dd9d0cfSchristos
1189dd9d0cfSchristos return (phdr);
1199dd9d0cfSchristos }
1209dd9d0cfSchristos
1219dd9d0cfSchristos void *
_libelf_newphdr(Elf * e,int ec,size_t count)1229dd9d0cfSchristos _libelf_newphdr(Elf *e, int ec, size_t count)
1239dd9d0cfSchristos {
1249dd9d0cfSchristos void *ehdr, *newphdr, *oldphdr;
1259dd9d0cfSchristos size_t msz;
1269dd9d0cfSchristos
1279dd9d0cfSchristos if (e == NULL) {
1289dd9d0cfSchristos LIBELF_SET_ERROR(ARGUMENT, 0);
1299dd9d0cfSchristos return (NULL);
1309dd9d0cfSchristos }
1319dd9d0cfSchristos
1329dd9d0cfSchristos if ((ehdr = _libelf_ehdr(e, ec, 0)) == NULL) {
1339dd9d0cfSchristos LIBELF_SET_ERROR(SEQUENCE, 0);
1349dd9d0cfSchristos return (NULL);
1359dd9d0cfSchristos }
1369dd9d0cfSchristos
1379dd9d0cfSchristos assert(e->e_class == ec);
1389dd9d0cfSchristos assert(ec == ELFCLASS32 || ec == ELFCLASS64);
1399dd9d0cfSchristos assert(e->e_version == EV_CURRENT);
1409dd9d0cfSchristos
141*5ac3bc71Schristos if ((msz = _libelf_msize(ELF_T_PHDR, ec, e->e_version)) == 0)
142*5ac3bc71Schristos return (NULL);
1439dd9d0cfSchristos
1449dd9d0cfSchristos newphdr = NULL;
1459dd9d0cfSchristos if (count > 0 && (newphdr = calloc(count, msz)) == NULL) {
1469dd9d0cfSchristos LIBELF_SET_ERROR(RESOURCE, 0);
1479dd9d0cfSchristos return (NULL);
1489dd9d0cfSchristos }
1499dd9d0cfSchristos
1509dd9d0cfSchristos if (ec == ELFCLASS32) {
1519dd9d0cfSchristos if ((oldphdr = (void *) e->e_u.e_elf.e_phdr.e_phdr32) != NULL)
1529dd9d0cfSchristos free(oldphdr);
1539dd9d0cfSchristos e->e_u.e_elf.e_phdr.e_phdr32 = (Elf32_Phdr *) newphdr;
1549dd9d0cfSchristos } else {
1559dd9d0cfSchristos if ((oldphdr = (void *) e->e_u.e_elf.e_phdr.e_phdr64) != NULL)
1569dd9d0cfSchristos free(oldphdr);
1579dd9d0cfSchristos e->e_u.e_elf.e_phdr.e_phdr64 = (Elf64_Phdr *) newphdr;
1589dd9d0cfSchristos }
1599dd9d0cfSchristos
1609dd9d0cfSchristos e->e_u.e_elf.e_nphdr = count;
1619dd9d0cfSchristos
1629dd9d0cfSchristos elf_flagphdr(e, ELF_C_SET, ELF_F_DIRTY);
1639dd9d0cfSchristos
1649dd9d0cfSchristos return (newphdr);
1659dd9d0cfSchristos }
166