12de3b87aSKai Wang /*-
22de3b87aSKai Wang  * Copyright (c) 2006,2008-2009,2011 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 <libelf.h>
282de3b87aSKai Wang 
292de3b87aSKai Wang #include "_libelf.h"
302de3b87aSKai Wang 
31*67d97fe7SEd Maste ELFTC_VCSID("$Id: elf_flag.c 3174 2015-03-27 17:13:41Z emaste $");
322de3b87aSKai Wang 
332de3b87aSKai Wang unsigned int
elf_flagarhdr(Elf_Arhdr * a,Elf_Cmd c,unsigned int flags)342de3b87aSKai Wang elf_flagarhdr(Elf_Arhdr *a, Elf_Cmd c, unsigned int flags)
352de3b87aSKai Wang {
362de3b87aSKai Wang 	unsigned int r;
372de3b87aSKai Wang 
382de3b87aSKai Wang 	if (a == NULL)
392de3b87aSKai Wang 		return (0);
402de3b87aSKai Wang 
412de3b87aSKai Wang 	if ((c != ELF_C_SET && c != ELF_C_CLR) ||
422de3b87aSKai Wang 	    (flags & ~ELF_F_DIRTY) != 0) {
432de3b87aSKai Wang 		LIBELF_SET_ERROR(ARGUMENT, 0);
442de3b87aSKai Wang 		return (0);
452de3b87aSKai Wang 	}
462de3b87aSKai Wang 
472de3b87aSKai Wang 	if (c == ELF_C_SET)
482de3b87aSKai Wang 		r = a->ar_flags |= flags;
492de3b87aSKai Wang 	else
502de3b87aSKai Wang 		r = a->ar_flags &= ~flags;
512de3b87aSKai Wang 
522de3b87aSKai Wang 	return (r & LIBELF_F_API_MASK);
532de3b87aSKai Wang }
542de3b87aSKai Wang 
552de3b87aSKai Wang unsigned int
elf_flagdata(Elf_Data * d,Elf_Cmd c,unsigned int flags)562de3b87aSKai Wang elf_flagdata(Elf_Data *d, Elf_Cmd c, unsigned int flags)
572de3b87aSKai Wang {
582de3b87aSKai Wang 	unsigned int r;
592de3b87aSKai Wang 	struct _Libelf_Data *ld;
602de3b87aSKai Wang 
612de3b87aSKai Wang 	if (d == NULL)
622de3b87aSKai Wang 		return (0);
632de3b87aSKai Wang 
642de3b87aSKai Wang 	if ((c != ELF_C_SET && c != ELF_C_CLR) ||
652de3b87aSKai Wang 	    (flags & ~ELF_F_DIRTY) != 0) {
662de3b87aSKai Wang 		LIBELF_SET_ERROR(ARGUMENT, 0);
672de3b87aSKai Wang 		return (0);
682de3b87aSKai Wang 	}
692de3b87aSKai Wang 
702de3b87aSKai Wang 	ld = (struct _Libelf_Data *) d;
712de3b87aSKai Wang 
722de3b87aSKai Wang 	if (c == ELF_C_SET)
732de3b87aSKai Wang 		r = ld->d_flags |= flags;
742de3b87aSKai Wang 	else
752de3b87aSKai Wang 		r = ld->d_flags &= ~flags;
762de3b87aSKai Wang 
772de3b87aSKai Wang 	return (r & LIBELF_F_API_MASK);
782de3b87aSKai Wang }
792de3b87aSKai Wang 
802de3b87aSKai Wang unsigned int
elf_flagehdr(Elf * e,Elf_Cmd c,unsigned int flags)812de3b87aSKai Wang elf_flagehdr(Elf *e, Elf_Cmd c, unsigned int flags)
822de3b87aSKai Wang {
832de3b87aSKai Wang 	int ec;
842de3b87aSKai Wang 	void *ehdr;
852de3b87aSKai Wang 
862de3b87aSKai Wang 	if (e == NULL)
872de3b87aSKai Wang 		return (0);
882de3b87aSKai Wang 
892de3b87aSKai Wang 	if ((c != ELF_C_SET && c != ELF_C_CLR) ||
902de3b87aSKai Wang 	    (e->e_kind != ELF_K_ELF) || (flags & ~ELF_F_DIRTY) != 0 ||
912de3b87aSKai Wang 	    ((ec = e->e_class) != ELFCLASS32 && ec != ELFCLASS64)) {
922de3b87aSKai Wang 		LIBELF_SET_ERROR(ARGUMENT, 0);
932de3b87aSKai Wang 		return (0);
942de3b87aSKai Wang 	}
952de3b87aSKai Wang 
962de3b87aSKai Wang 	if (ec == ELFCLASS32)
972de3b87aSKai Wang 		ehdr = e->e_u.e_elf.e_ehdr.e_ehdr32;
982de3b87aSKai Wang 	else
992de3b87aSKai Wang 		ehdr = e->e_u.e_elf.e_ehdr.e_ehdr64;
1002de3b87aSKai Wang 
1012de3b87aSKai Wang 	if (ehdr == NULL) {
1022de3b87aSKai Wang 		LIBELF_SET_ERROR(SEQUENCE, 0);
1032de3b87aSKai Wang 		return (0);
1042de3b87aSKai Wang 	}
1052de3b87aSKai Wang 
1062de3b87aSKai Wang 	return (elf_flagelf(e, c, flags));
1072de3b87aSKai Wang }
1082de3b87aSKai Wang 
1092de3b87aSKai Wang unsigned int
elf_flagelf(Elf * e,Elf_Cmd c,unsigned int flags)1102de3b87aSKai Wang elf_flagelf(Elf *e, Elf_Cmd c, unsigned int flags)
1112de3b87aSKai Wang {
112cf781b2eSEd Maste 	unsigned int r;
1132de3b87aSKai Wang 
1142de3b87aSKai Wang 	if (e == NULL)
1152de3b87aSKai Wang 		return (0);
1162de3b87aSKai Wang 
1172de3b87aSKai Wang 	if ((c != ELF_C_SET && c != ELF_C_CLR) ||
1182de3b87aSKai Wang 	    (e->e_kind != ELF_K_ELF) ||
1192de3b87aSKai Wang 	    (flags & ~(ELF_F_ARCHIVE | ELF_F_ARCHIVE_SYSV |
1202de3b87aSKai Wang 	    ELF_F_DIRTY | ELF_F_LAYOUT)) != 0) {
1212de3b87aSKai Wang 		LIBELF_SET_ERROR(ARGUMENT, 0);
1222de3b87aSKai Wang 		return (0);
1232de3b87aSKai Wang 	}
1242de3b87aSKai Wang 
1252de3b87aSKai Wang 	if ((flags & ELF_F_ARCHIVE_SYSV) && (flags & ELF_F_ARCHIVE) == 0) {
1262de3b87aSKai Wang 		LIBELF_SET_ERROR(ARGUMENT, 0);
1272de3b87aSKai Wang 		return (0);
1282de3b87aSKai Wang 	}
1292de3b87aSKai Wang 
1302de3b87aSKai Wang 	if ((flags & ELF_F_ARCHIVE) && e->e_cmd != ELF_C_WRITE) {
1312de3b87aSKai Wang 		LIBELF_SET_ERROR(MODE, 0);
1322de3b87aSKai Wang 		return (0);
1332de3b87aSKai Wang 	}
1342de3b87aSKai Wang 
1352de3b87aSKai Wang 	if (c == ELF_C_SET)
1362de3b87aSKai Wang 		r = e->e_flags |= flags;
1372de3b87aSKai Wang 	else
1382de3b87aSKai Wang 		r = e->e_flags &= ~flags;
1392de3b87aSKai Wang 	return (r & LIBELF_F_API_MASK);
1402de3b87aSKai Wang }
1412de3b87aSKai Wang 
1422de3b87aSKai Wang unsigned int
elf_flagphdr(Elf * e,Elf_Cmd c,unsigned int flags)1432de3b87aSKai Wang elf_flagphdr(Elf *e, Elf_Cmd c, unsigned int flags)
1442de3b87aSKai Wang {
1452de3b87aSKai Wang 	int ec;
1462de3b87aSKai Wang 	void *phdr;
1472de3b87aSKai Wang 
1482de3b87aSKai Wang 	if (e == NULL)
1492de3b87aSKai Wang 		return (0);
1502de3b87aSKai Wang 
1512de3b87aSKai Wang 	if ((c != ELF_C_SET && c != ELF_C_CLR) ||
1522de3b87aSKai Wang 	    (e->e_kind != ELF_K_ELF) || (flags & ~ELF_F_DIRTY) != 0 ||
1532de3b87aSKai Wang 	    ((ec = e->e_class) != ELFCLASS32 && ec != ELFCLASS64)) {
1542de3b87aSKai Wang 		LIBELF_SET_ERROR(ARGUMENT, 0);
1552de3b87aSKai Wang 		return (0);
1562de3b87aSKai Wang 	}
1572de3b87aSKai Wang 
1582de3b87aSKai Wang 	if (ec == ELFCLASS32)
1592de3b87aSKai Wang 		phdr = e->e_u.e_elf.e_phdr.e_phdr32;
1602de3b87aSKai Wang 	else
1612de3b87aSKai Wang 		phdr = e->e_u.e_elf.e_phdr.e_phdr64;
1622de3b87aSKai Wang 
1632de3b87aSKai Wang 	if (phdr == NULL) {
1642de3b87aSKai Wang 		LIBELF_SET_ERROR(SEQUENCE, 0);
1652de3b87aSKai Wang 		return (0);
1662de3b87aSKai Wang 	}
1672de3b87aSKai Wang 
1682de3b87aSKai Wang 	return (elf_flagelf(e, c, flags));
1692de3b87aSKai Wang }
1702de3b87aSKai Wang 
1712de3b87aSKai Wang unsigned int
elf_flagscn(Elf_Scn * s,Elf_Cmd c,unsigned int flags)1722de3b87aSKai Wang elf_flagscn(Elf_Scn *s, Elf_Cmd c, unsigned int flags)
1732de3b87aSKai Wang {
174cf781b2eSEd Maste 	unsigned int r;
1752de3b87aSKai Wang 
1762de3b87aSKai Wang 	if (s == NULL)
1772de3b87aSKai Wang 		return (0);
1782de3b87aSKai Wang 
1792de3b87aSKai Wang 	if ((c != ELF_C_SET && c != ELF_C_CLR) ||
1802de3b87aSKai Wang 	    (flags & ~ELF_F_DIRTY) != 0) {
1812de3b87aSKai Wang 		LIBELF_SET_ERROR(ARGUMENT, 0);
1822de3b87aSKai Wang 		return (0);
1832de3b87aSKai Wang 	}
1842de3b87aSKai Wang 
1852de3b87aSKai Wang 	if (c == ELF_C_SET)
1862de3b87aSKai Wang 		r = s->s_flags |= flags;
1872de3b87aSKai Wang 	else
1882de3b87aSKai Wang 		r = s->s_flags &= ~flags;
1892de3b87aSKai Wang 	return (r & LIBELF_F_API_MASK);
1902de3b87aSKai Wang }
1912de3b87aSKai Wang 
1922de3b87aSKai Wang unsigned int
elf_flagshdr(Elf_Scn * s,Elf_Cmd c,unsigned int flags)1932de3b87aSKai Wang elf_flagshdr(Elf_Scn *s, Elf_Cmd c, unsigned int flags)
1942de3b87aSKai Wang {
1952de3b87aSKai Wang 	return (elf_flagscn(s, c, flags));
1962de3b87aSKai Wang }
197