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