xref: /netbsd-src/external/bsd/elftoolchain/dist/libelf/libelf_phdr.c (revision 5ac3bc719ce6e70593039505b491894133237d12)
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