xref: /netbsd-src/external/bsd/elftoolchain/dist/libelf/gelf_cap.c (revision 5ac3bc719ce6e70593039505b491894133237d12)
1*5ac3bc71Schristos /*	$NetBSD: gelf_cap.c,v 1.5 2024/03/03 17:37:33 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 <limits.h>
3842bd3019Schristos #include <stdint.h>
399dd9d0cfSchristos 
409dd9d0cfSchristos #include "_libelf.h"
419dd9d0cfSchristos 
42*5ac3bc71Schristos __RCSID("$NetBSD: gelf_cap.c,v 1.5 2024/03/03 17:37:33 christos Exp $");
439dd9d0cfSchristos 
449dd9d0cfSchristos GElf_Cap *
gelf_getcap(Elf_Data * ed,int ndx,GElf_Cap * dst)459dd9d0cfSchristos gelf_getcap(Elf_Data *ed, int ndx, GElf_Cap *dst)
469dd9d0cfSchristos {
479dd9d0cfSchristos 	int ec;
489dd9d0cfSchristos 	Elf *e;
499dd9d0cfSchristos 	size_t msz;
509dd9d0cfSchristos 	Elf_Scn *scn;
519dd9d0cfSchristos 	Elf32_Cap *cap32;
529dd9d0cfSchristos 	Elf64_Cap *cap64;
539dd9d0cfSchristos 	uint32_t sh_type;
549dd9d0cfSchristos 	struct _Libelf_Data *d;
559dd9d0cfSchristos 
569dd9d0cfSchristos 	d = (struct _Libelf_Data *) ed;
579dd9d0cfSchristos 
589dd9d0cfSchristos 	if (d == NULL || ndx < 0 || dst == NULL ||
599dd9d0cfSchristos 	    (scn = d->d_scn) == NULL ||
609dd9d0cfSchristos 	    (e = scn->s_elf) == NULL) {
619dd9d0cfSchristos 		LIBELF_SET_ERROR(ARGUMENT, 0);
629dd9d0cfSchristos 		return (NULL);
639dd9d0cfSchristos 	}
649dd9d0cfSchristos 
659dd9d0cfSchristos 	ec = e->e_class;
669dd9d0cfSchristos 	assert(ec == ELFCLASS32 || ec == ELFCLASS64);
679dd9d0cfSchristos 
689dd9d0cfSchristos 	if (ec == ELFCLASS32)
699dd9d0cfSchristos 		sh_type = scn->s_shdr.s_shdr32.sh_type;
709dd9d0cfSchristos 	else
719dd9d0cfSchristos 		sh_type = scn->s_shdr.s_shdr64.sh_type;
729dd9d0cfSchristos 
739dd9d0cfSchristos 	if (_libelf_xlate_shtype(sh_type) != ELF_T_CAP) {
749dd9d0cfSchristos 		LIBELF_SET_ERROR(ARGUMENT, 0);
759dd9d0cfSchristos 		return (NULL);
769dd9d0cfSchristos 	}
779dd9d0cfSchristos 
78*5ac3bc71Schristos 	if ((msz = _libelf_msize(ELF_T_CAP, ec, e->e_version)) == 0)
79*5ac3bc71Schristos 		return (NULL);
809dd9d0cfSchristos 
8142bd3019Schristos 	if (msz * (size_t) ndx >= d->d_data.d_size) {
829dd9d0cfSchristos 		LIBELF_SET_ERROR(ARGUMENT, 0);
839dd9d0cfSchristos 		return (NULL);
849dd9d0cfSchristos 	}
859dd9d0cfSchristos 
869dd9d0cfSchristos 	if (ec == ELFCLASS32) {
879dd9d0cfSchristos 
889dd9d0cfSchristos 		cap32 = (Elf32_Cap *) d->d_data.d_buf + ndx;
899dd9d0cfSchristos 
909dd9d0cfSchristos 		dst->c_tag  = cap32->c_tag;
919dd9d0cfSchristos 		dst->c_un.c_val = (Elf64_Xword) cap32->c_un.c_val;
929dd9d0cfSchristos 
939dd9d0cfSchristos 	} else {
949dd9d0cfSchristos 
959dd9d0cfSchristos 		cap64 = (Elf64_Cap *) d->d_data.d_buf + ndx;
969dd9d0cfSchristos 
979dd9d0cfSchristos 		*dst = *cap64;
989dd9d0cfSchristos 	}
999dd9d0cfSchristos 
1009dd9d0cfSchristos 	return (dst);
1019dd9d0cfSchristos }
1029dd9d0cfSchristos 
1039dd9d0cfSchristos int
gelf_update_cap(Elf_Data * ed,int ndx,GElf_Cap * gc)1049dd9d0cfSchristos gelf_update_cap(Elf_Data *ed, int ndx, GElf_Cap *gc)
1059dd9d0cfSchristos {
1069dd9d0cfSchristos 	int ec;
1079dd9d0cfSchristos 	Elf *e;
1089dd9d0cfSchristos 	size_t msz;
1099dd9d0cfSchristos 	Elf_Scn *scn;
1109dd9d0cfSchristos 	Elf32_Cap *cap32;
1119dd9d0cfSchristos 	Elf64_Cap *cap64;
1129dd9d0cfSchristos 	uint32_t sh_type;
1139dd9d0cfSchristos 	struct _Libelf_Data *d;
1149dd9d0cfSchristos 
1159dd9d0cfSchristos 	d = (struct _Libelf_Data *) ed;
1169dd9d0cfSchristos 
1179dd9d0cfSchristos 	if (d == NULL || ndx < 0 || gc == NULL ||
1189dd9d0cfSchristos 	    (scn = d->d_scn) == NULL ||
1199dd9d0cfSchristos 	    (e = scn->s_elf) == NULL) {
1209dd9d0cfSchristos 		LIBELF_SET_ERROR(ARGUMENT, 0);
1219dd9d0cfSchristos 		return (0);
1229dd9d0cfSchristos 	}
1239dd9d0cfSchristos 
1249dd9d0cfSchristos 	ec = e->e_class;
1259dd9d0cfSchristos 	assert(ec == ELFCLASS32 || ec == ELFCLASS64);
1269dd9d0cfSchristos 
1279dd9d0cfSchristos 	if (ec == ELFCLASS32)
1289dd9d0cfSchristos 		sh_type = scn->s_shdr.s_shdr32.sh_type;
1299dd9d0cfSchristos 	else
1309dd9d0cfSchristos 		sh_type = scn->s_shdr.s_shdr64.sh_type;
1319dd9d0cfSchristos 
1329dd9d0cfSchristos 	if (_libelf_xlate_shtype(sh_type) != ELF_T_CAP) {
1339dd9d0cfSchristos 		LIBELF_SET_ERROR(ARGUMENT, 0);
1349dd9d0cfSchristos 		return (0);
1359dd9d0cfSchristos 	}
1369dd9d0cfSchristos 
137*5ac3bc71Schristos 	if ((msz = _libelf_msize(ELF_T_CAP, ec, e->e_version)) == 0)
138*5ac3bc71Schristos 		return (0);
1399dd9d0cfSchristos 
14042bd3019Schristos 	if (msz * (size_t) ndx >= d->d_data.d_size) {
1419dd9d0cfSchristos 		LIBELF_SET_ERROR(ARGUMENT, 0);
1429dd9d0cfSchristos 		return (0);
1439dd9d0cfSchristos 	}
1449dd9d0cfSchristos 
1459dd9d0cfSchristos 	if (ec == ELFCLASS32) {
1469dd9d0cfSchristos 		cap32 = (Elf32_Cap *) d->d_data.d_buf + ndx;
1479dd9d0cfSchristos 
1489dd9d0cfSchristos 		LIBELF_COPY_U32(cap32, gc, c_tag);
1499dd9d0cfSchristos 		LIBELF_COPY_U32(cap32, gc, c_un.c_val);
1509dd9d0cfSchristos 	} else {
1519dd9d0cfSchristos 		cap64 = (Elf64_Cap *) d->d_data.d_buf + ndx;
1529dd9d0cfSchristos 
1539dd9d0cfSchristos 		*cap64 = *gc;
1549dd9d0cfSchristos 	}
1559dd9d0cfSchristos 
1569dd9d0cfSchristos 	return (1);
1579dd9d0cfSchristos }
158