xref: /freebsd-src/contrib/elftoolchain/libelf/_libelf.h (revision 573456a931b6dc23dfe3da1e42d88a0bc0fbba7d)
12de3b87aSKai Wang /*-
22de3b87aSKai Wang  * Copyright (c) 2006,2008-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  *
26d003e0d7SEd Maste  * $Id: _libelf.h 3738 2019-05-05 21:49:06Z jkoshy $
272de3b87aSKai Wang  */
282de3b87aSKai Wang 
292de3b87aSKai Wang #ifndef	__LIBELF_H_
302de3b87aSKai Wang #define	__LIBELF_H_
312de3b87aSKai Wang 
322de3b87aSKai Wang #include <sys/queue.h>
338c953901SMark Johnston #include <sys/tree.h>
342de3b87aSKai Wang 
352de3b87aSKai Wang #include "_libelf_config.h"
362de3b87aSKai Wang 
372de3b87aSKai Wang #include "_elftc.h"
382de3b87aSKai Wang 
392de3b87aSKai Wang /*
402de3b87aSKai Wang  * Library-private data structures.
412de3b87aSKai Wang  */
422de3b87aSKai Wang 
432de3b87aSKai Wang #define LIBELF_MSG_SIZE	256
442de3b87aSKai Wang 
452de3b87aSKai Wang struct _libelf_globals {
462de3b87aSKai Wang 	int		libelf_arch;
472de3b87aSKai Wang 	unsigned int	libelf_byteorder;
482de3b87aSKai Wang 	int		libelf_class;
492de3b87aSKai Wang 	int		libelf_error;
502de3b87aSKai Wang 	int		libelf_fillchar;
512de3b87aSKai Wang 	unsigned int	libelf_version;
52cf781b2eSEd Maste 	unsigned char	libelf_msg[LIBELF_MSG_SIZE];
532de3b87aSKai Wang };
542de3b87aSKai Wang 
552de3b87aSKai Wang extern struct _libelf_globals _libelf;
562de3b87aSKai Wang 
572de3b87aSKai Wang #define	LIBELF_PRIVATE(N)	(_libelf.libelf_##N)
582de3b87aSKai Wang 
592de3b87aSKai Wang #define	LIBELF_ELF_ERROR_MASK			0xFF
602de3b87aSKai Wang #define	LIBELF_OS_ERROR_SHIFT			8
612de3b87aSKai Wang 
622de3b87aSKai Wang #define	LIBELF_ERROR(E, O) (((E) & LIBELF_ELF_ERROR_MASK) |	\
632de3b87aSKai Wang 	((O) << LIBELF_OS_ERROR_SHIFT))
642de3b87aSKai Wang 
652de3b87aSKai Wang #define	LIBELF_SET_ERROR(E, O) do {					\
662de3b87aSKai Wang 		LIBELF_PRIVATE(error) = LIBELF_ERROR(ELF_E_##E, (O));	\
672de3b87aSKai Wang 	} while (0)
682de3b87aSKai Wang 
692de3b87aSKai Wang #define	LIBELF_ADJUST_AR_SIZE(S)	(((S) + 1U) & ~1U)
702de3b87aSKai Wang 
712de3b87aSKai Wang /*
722de3b87aSKai Wang  * Flags for library internal use.  These use the upper 16 bits of the
732de3b87aSKai Wang  * `e_flags' field.
742de3b87aSKai Wang  */
75cf781b2eSEd Maste #define	LIBELF_F_API_MASK	0x00FFFFU  /* Flags defined by the API. */
76cf781b2eSEd Maste #define	LIBELF_F_AR_HEADER	0x010000U  /* translated header available */
77cf781b2eSEd Maste #define	LIBELF_F_AR_VARIANT_SVR4 0x020000U /* BSD style ar(1) archive */
78cf781b2eSEd Maste #define	LIBELF_F_DATA_MALLOCED	0x040000U /* whether data was malloc'ed */
79cf781b2eSEd Maste #define	LIBELF_F_RAWFILE_MALLOC	0x080000U /* whether e_rawfile was malloc'ed */
80cf781b2eSEd Maste #define	LIBELF_F_RAWFILE_MMAP	0x100000U /* whether e_rawfile was mmap'ed */
81cf781b2eSEd Maste #define	LIBELF_F_SHDRS_LOADED	0x200000U /* whether all shdrs were read in */
82cf781b2eSEd Maste #define	LIBELF_F_SPECIAL_FILE	0x400000U /* non-regular file */
832de3b87aSKai Wang 
848c953901SMark Johnston RB_HEAD(scntree, _Elf_Scn);
858c953901SMark Johnston RB_PROTOTYPE(scntree, _Elf_Scn, e_scn, elfscn_cmp);
868c953901SMark Johnston 
872de3b87aSKai Wang struct _Elf {
882de3b87aSKai Wang 	int		e_activations;	/* activation count */
892de3b87aSKai Wang 	unsigned int	e_byteorder;	/* ELFDATA* */
902de3b87aSKai Wang 	int		e_class;	/* ELFCLASS*  */
912de3b87aSKai Wang 	Elf_Cmd		e_cmd;		/* ELF_C_* used at creation time */
922de3b87aSKai Wang 	int		e_fd;		/* associated file descriptor */
932de3b87aSKai Wang 	unsigned int	e_flags;	/* ELF_F_* & LIBELF_F_* flags */
942de3b87aSKai Wang 	Elf_Kind	e_kind;		/* ELF_K_* */
952de3b87aSKai Wang 	Elf		*e_parent; 	/* non-NULL for archive members */
96cf781b2eSEd Maste 	unsigned char	*e_rawfile;	/* uninterpreted bytes */
97d003e0d7SEd Maste 	off_t		e_rawsize;	/* size of uninterpreted bytes */
982de3b87aSKai Wang 	unsigned int	e_version;	/* file version */
992de3b87aSKai Wang 
1002de3b87aSKai Wang 	/*
1012de3b87aSKai Wang 	 * Header information for archive members.  See the
1022de3b87aSKai Wang 	 * LIBELF_F_AR_HEADER flag.
1032de3b87aSKai Wang 	 */
1042de3b87aSKai Wang 	union {
1052de3b87aSKai Wang 		Elf_Arhdr	*e_arhdr;	/* translated header */
106cf781b2eSEd Maste 		unsigned char	*e_rawhdr;	/* untranslated header */
1072de3b87aSKai Wang 	} e_hdr;
1082de3b87aSKai Wang 
1092de3b87aSKai Wang 	union {
1102de3b87aSKai Wang 		struct {		/* ar(1) archives */
1112de3b87aSKai Wang 			off_t	e_next;	/* set by elf_rand()/elf_next() */
1122de3b87aSKai Wang 			int	e_nchildren;
113cf781b2eSEd Maste 			unsigned char *e_rawstrtab; /* file name strings */
1142de3b87aSKai Wang 			size_t	e_rawstrtabsz;
115cf781b2eSEd Maste 			unsigned char *e_rawsymtab;	/* symbol table */
1162de3b87aSKai Wang 			size_t	e_rawsymtabsz;
1172de3b87aSKai Wang 			Elf_Arsym *e_symtab;
1182de3b87aSKai Wang 			size_t	e_symtabsz;
1192de3b87aSKai Wang 		} e_ar;
1202de3b87aSKai Wang 		struct {		/* regular ELF files */
1212de3b87aSKai Wang 			union {
1222de3b87aSKai Wang 				Elf32_Ehdr *e_ehdr32;
1232de3b87aSKai Wang 				Elf64_Ehdr *e_ehdr64;
1242de3b87aSKai Wang 			} e_ehdr;
1252de3b87aSKai Wang 			union {
1262de3b87aSKai Wang 				Elf32_Phdr *e_phdr32;
1272de3b87aSKai Wang 				Elf64_Phdr *e_phdr64;
1282de3b87aSKai Wang 			} e_phdr;
1298c953901SMark Johnston 			struct scntree	e_scn;	/* sections */
1302de3b87aSKai Wang 			size_t	e_nphdr;	/* number of Phdr entries */
1312de3b87aSKai Wang 			size_t	e_nscn;		/* number of sections */
1322de3b87aSKai Wang 			size_t	e_strndx;	/* string table section index */
1332de3b87aSKai Wang 		} e_elf;
1342de3b87aSKai Wang 	} e_u;
1352de3b87aSKai Wang };
1362de3b87aSKai Wang 
1372de3b87aSKai Wang /*
1382de3b87aSKai Wang  * The internal descriptor wrapping the "Elf_Data" type.
1392de3b87aSKai Wang  */
1402de3b87aSKai Wang struct _Libelf_Data {
1412de3b87aSKai Wang 	Elf_Data	d_data;		/* The exported descriptor. */
1422de3b87aSKai Wang 	Elf_Scn		*d_scn;		/* The containing section */
1432de3b87aSKai Wang 	unsigned int	d_flags;
1442de3b87aSKai Wang 	STAILQ_ENTRY(_Libelf_Data) d_next;
1452de3b87aSKai Wang };
1462de3b87aSKai Wang 
1472de3b87aSKai Wang struct _Elf_Scn {
1482de3b87aSKai Wang 	union {
1492de3b87aSKai Wang 		Elf32_Shdr	s_shdr32;
1502de3b87aSKai Wang 		Elf64_Shdr	s_shdr64;
1512de3b87aSKai Wang 	} s_shdr;
1522de3b87aSKai Wang 	STAILQ_HEAD(, _Libelf_Data) s_data;	/* translated data */
1532de3b87aSKai Wang 	STAILQ_HEAD(, _Libelf_Data) s_rawdata;	/* raw data */
1548c953901SMark Johnston 	RB_ENTRY(_Elf_Scn) s_tree;
1552de3b87aSKai Wang 	struct _Elf	*s_elf;		/* parent ELF descriptor */
1562de3b87aSKai Wang 	unsigned int	s_flags;	/* flags for the section as a whole */
1572de3b87aSKai Wang 	size_t		s_ndx;		/* index# for this section */
1582de3b87aSKai Wang 	uint64_t	s_offset;	/* managed by elf_update() */
1592de3b87aSKai Wang 	uint64_t	s_rawoff;	/* original offset in the file */
1602de3b87aSKai Wang 	uint64_t	s_size;		/* managed by elf_update() */
1612de3b87aSKai Wang };
1622de3b87aSKai Wang 
1632de3b87aSKai Wang 
1642de3b87aSKai Wang enum {
1652de3b87aSKai Wang 	ELF_TOFILE,
1662de3b87aSKai Wang 	ELF_TOMEMORY
1672de3b87aSKai Wang };
1682de3b87aSKai Wang 
169cf781b2eSEd Maste 
170cf781b2eSEd Maste /*
171cf781b2eSEd Maste  * The LIBELF_COPY macros are used to copy fields from a GElf_*
172cf781b2eSEd Maste  * structure to their 32-bit counterparts, while checking for out of
173cf781b2eSEd Maste  * range values.
174cf781b2eSEd Maste  *
175cf781b2eSEd Maste  * - LIBELF_COPY_U32 :: copy an unsigned 32 bit field.
176cf781b2eSEd Maste  * - LIBELF_COPY_S32 :: copy a signed 32 bit field.
177cf781b2eSEd Maste  */
178cf781b2eSEd Maste 
1792de3b87aSKai Wang #define	LIBELF_COPY_U32(DST, SRC, NAME)	do {			\
180cf781b2eSEd Maste 		if ((SRC)->NAME > UINT32_MAX) {			\
1812de3b87aSKai Wang 			LIBELF_SET_ERROR(RANGE, 0);		\
1822de3b87aSKai Wang 			return (0);				\
1832de3b87aSKai Wang 		}						\
184cf781b2eSEd Maste 		(DST)->NAME = (SRC)->NAME & 0xFFFFFFFFU;	\
1852de3b87aSKai Wang 	} while (0)
1862de3b87aSKai Wang 
1872de3b87aSKai Wang #define	LIBELF_COPY_S32(DST, SRC, NAME)	do {			\
188cf781b2eSEd Maste 		if ((SRC)->NAME > INT32_MAX ||			\
189cf781b2eSEd Maste 		    (SRC)->NAME < INT32_MIN) {			\
1902de3b87aSKai Wang 			LIBELF_SET_ERROR(RANGE, 0);		\
1912de3b87aSKai Wang 			return (0);				\
1922de3b87aSKai Wang 		}						\
193cf781b2eSEd Maste 		(DST)->NAME = (int32_t) (SRC)->NAME;		\
1942de3b87aSKai Wang 	} while (0)
1952de3b87aSKai Wang 
1962de3b87aSKai Wang 
1972de3b87aSKai Wang /*
1982de3b87aSKai Wang  * Function Prototypes.
1992de3b87aSKai Wang  */
2002de3b87aSKai Wang 
201ae500c1fSEd Maste typedef int _libelf_translator_function(unsigned char *_dst, size_t dsz,
202ae500c1fSEd Maste     unsigned char *_src, size_t _cnt, int _byteswap);
203ae500c1fSEd Maste 
20467d97fe7SEd Maste #ifdef __cplusplus
20567d97fe7SEd Maste extern "C" {
20667d97fe7SEd Maste #endif
2072de3b87aSKai Wang struct _Libelf_Data *_libelf_allocate_data(Elf_Scn *_s);
2082de3b87aSKai Wang Elf	*_libelf_allocate_elf(void);
2092de3b87aSKai Wang Elf_Scn	*_libelf_allocate_scn(Elf *_e, size_t _ndx);
2102de3b87aSKai Wang Elf_Arhdr *_libelf_ar_gethdr(Elf *_e);
2112de3b87aSKai Wang Elf	*_libelf_ar_open(Elf *_e, int _reporterror);
2122de3b87aSKai Wang Elf	*_libelf_ar_open_member(int _fd, Elf_Cmd _c, Elf *_ar);
2132de3b87aSKai Wang Elf_Arsym *_libelf_ar_process_bsd_symtab(Elf *_ar, size_t *_dst);
2142de3b87aSKai Wang Elf_Arsym *_libelf_ar_process_svr4_symtab(Elf *_ar, size_t *_dst);
215cf781b2eSEd Maste long	 _libelf_checksum(Elf *_e, int _elfclass);
2162de3b87aSKai Wang void	*_libelf_ehdr(Elf *_e, int _elfclass, int _allocate);
217ae500c1fSEd Maste int	_libelf_elfmachine(Elf *_e);
218cf781b2eSEd Maste unsigned int _libelf_falign(Elf_Type _t, int _elfclass);
2192de3b87aSKai Wang size_t	_libelf_fsize(Elf_Type _t, int _elfclass, unsigned int _version,
2202de3b87aSKai Wang     size_t count);
221ae500c1fSEd Maste _libelf_translator_function *_libelf_get_translator(Elf_Type _t,
222ae500c1fSEd Maste     int _direction, int _elfclass, int _elfmachine);
223*573456a9SEd Maste void	*_libelf_getchdr(Elf_Scn *_e, int _elfclass);
2242de3b87aSKai Wang void	*_libelf_getphdr(Elf *_e, int _elfclass);
2252de3b87aSKai Wang void	*_libelf_getshdr(Elf_Scn *_scn, int _elfclass);
2262de3b87aSKai Wang void	_libelf_init_elf(Elf *_e, Elf_Kind _kind);
227eb81f38aSJohn Baldwin int	_libelf_is_mips64el(Elf *e);
2282de3b87aSKai Wang int	_libelf_load_section_headers(Elf *e, void *ehdr);
229cf781b2eSEd Maste unsigned int _libelf_malign(Elf_Type _t, int _elfclass);
230cf781b2eSEd Maste Elf	*_libelf_memory(unsigned char *_image, size_t _sz, int _reporterror);
2312de3b87aSKai Wang size_t	_libelf_msize(Elf_Type _t, int _elfclass, unsigned int _version);
2322de3b87aSKai Wang void	*_libelf_newphdr(Elf *_e, int _elfclass, size_t _count);
2332de3b87aSKai Wang Elf	*_libelf_open_object(int _fd, Elf_Cmd _c, int _reporterror);
234eb81f38aSJohn Baldwin Elf64_Xword _libelf_mips64el_r_info_tof(Elf64_Xword r_info);
235eb81f38aSJohn Baldwin Elf64_Xword _libelf_mips64el_r_info_tom(Elf64_Xword r_info);
2362de3b87aSKai Wang struct _Libelf_Data *_libelf_release_data(struct _Libelf_Data *_d);
237d003e0d7SEd Maste void	_libelf_release_elf(Elf *_e);
2382de3b87aSKai Wang Elf_Scn	*_libelf_release_scn(Elf_Scn *_s);
2392de3b87aSKai Wang int	_libelf_setphnum(Elf *_e, void *_eh, int _elfclass, size_t _phnum);
2402de3b87aSKai Wang int	_libelf_setshnum(Elf *_e, void *_eh, int _elfclass, size_t _shnum);
2412de3b87aSKai Wang int	_libelf_setshstrndx(Elf *_e, void *_eh, int _elfclass,
2422de3b87aSKai Wang     size_t _shstrndx);
2432de3b87aSKai Wang Elf_Data *_libelf_xlate(Elf_Data *_d, const Elf_Data *_s,
244ae500c1fSEd Maste     unsigned int _encoding, int _elfclass, int _elfmachine, int _direction);
2452de3b87aSKai Wang int	_libelf_xlate_shtype(uint32_t _sht);
24667d97fe7SEd Maste #ifdef __cplusplus
24767d97fe7SEd Maste }
24867d97fe7SEd Maste #endif
2492de3b87aSKai Wang 
2502de3b87aSKai Wang #endif	/* __LIBELF_H_ */
251