|
|
/*
|
|
|
* private.h - private definitions for libelf.
|
|
|
* Copyright (C) 1995 - 2007 Michael Riepe
|
|
|
*
|
|
|
* This library is free software; you can redistribute it and/or
|
|
|
* modify it under the terms of the GNU Library General Public
|
|
|
* License as published by the Free Software Foundation; either
|
|
|
* version 2 of the License, or (at your option) any later version.
|
|
|
*
|
|
|
* This library is distributed in the hope that it will be useful,
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
* Library General Public License for more details.
|
|
|
*
|
|
|
* You should have received a copy of the GNU Library General Public
|
|
|
* License along with this library; if not, write to the Free Software
|
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
|
|
|
*/
|
|
|
|
|
|
/* @(#) $Id: private.h,v 1.40 2009/11/01 13:04:19 michael Exp $ */
|
|
|
|
|
|
#ifndef _PRIVATE_H
|
|
|
#define _PRIVATE_H
|
|
|
|
|
|
#define __LIBELF_INTERNAL__ 1
|
|
|
|
|
|
#if HAVE_CONFIG_H
|
|
|
# include <config.h>
|
|
|
#endif /* HAVE_CONFIG_H */
|
|
|
|
|
|
/*
|
|
|
* Workaround for GLIBC bug:
|
|
|
* include <stdint.h> before <sys/types.h>
|
|
|
*/
|
|
|
#if HAVE_STDINT_H
|
|
|
#include <stdint.h>
|
|
|
#endif
|
|
|
#include <sys/types.h>
|
|
|
|
|
|
#if STDC_HEADERS
|
|
|
# include <stdlib.h>
|
|
|
# include <string.h>
|
|
|
#else /* STDC_HEADERS */
|
|
|
extern void *malloc(), *realloc();
|
|
|
extern void free(), bcopy(), abort();
|
|
|
extern int strcmp(), strncmp(), memcmp();
|
|
|
extern void *memcpy(), *memmove(), *memset();
|
|
|
#endif /* STDC_HEADERS */
|
|
|
|
|
|
#if defined(_WIN32)
|
|
|
#include <io.h>
|
|
|
#else
|
|
|
#if HAVE_UNISTD_H
|
|
|
# include <unistd.h>
|
|
|
#else /* HAVE_UNISTD_H */
|
|
|
extern int read(), write(), close();
|
|
|
extern off_t lseek();
|
|
|
#if HAVE_FTRUNCATE
|
|
|
extern int ftruncate();
|
|
|
#endif /* HAVE_FTRUNCATE */
|
|
|
#endif /* HAVE_UNISTD_H */
|
|
|
#endif /* defined(_WIN32) */
|
|
|
|
|
|
#ifndef SEEK_SET
|
|
|
#define SEEK_SET 0
|
|
|
#endif /* SEEK_SET */
|
|
|
#ifndef SEEK_CUR
|
|
|
#define SEEK_CUR 1
|
|
|
#endif /* SEEK_CUR */
|
|
|
#ifndef SEEK_END
|
|
|
#define SEEK_END 2
|
|
|
#endif /* SEEK_END */
|
|
|
|
|
|
#if !HAVE_MEMCMP
|
|
|
# define memcmp strncmp
|
|
|
#endif /* !HAVE_MEMCMP */
|
|
|
#if !HAVE_MEMCPY
|
|
|
# define memcpy(d,s,n) bcopy(s,d,n)
|
|
|
#endif /* !HAVE_MEMCPY */
|
|
|
#if !HAVE_MEMMOVE
|
|
|
# define memmove(d,s,n) bcopy(s,d,n)
|
|
|
#endif /* !HAVE_MEMMOVE */
|
|
|
|
|
|
#if !HAVE_MEMSET
|
|
|
# define memset _elf_memset
|
|
|
extern void *_elf_memset();
|
|
|
#endif /* !HAVE_MEMSET */
|
|
|
|
|
|
#if HAVE_STRUCT_NLIST_DECLARATION
|
|
|
# define nlist __override_nlist_declaration
|
|
|
#endif /* HAVE_STRUCT_NLIST_DECLARATION */
|
|
|
|
|
|
#if __LIBELF_NEED_LINK_H
|
|
|
# include <link.h>
|
|
|
#elif __LIBELF_NEED_SYS_LINK_H
|
|
|
# include <sys/link.h>
|
|
|
#endif /* __LIBELF_NEED_LINK_H */
|
|
|
|
|
|
#if HAVE_AR_H
|
|
|
#include <ar.h>
|
|
|
#else /* HAVE_AR_H */
|
|
|
|
|
|
#define ARMAG "!<arch>\n"
|
|
|
#define SARMAG 8
|
|
|
|
|
|
struct ar_hdr {
|
|
|
char ar_name[16];
|
|
|
char ar_date[12];
|
|
|
char ar_uid[6];
|
|
|
char ar_gid[6];
|
|
|
char ar_mode[8];
|
|
|
char ar_size[10];
|
|
|
char ar_fmag[2];
|
|
|
};
|
|
|
|
|
|
#define ARFMAG "`\n"
|
|
|
|
|
|
#endif /* HAVE_AR_H */
|
|
|
|
|
|
#include <libelf.h>
|
|
|
|
|
|
#if HAVE_STRUCT_NLIST_DECLARATION
|
|
|
# undef nlist
|
|
|
#endif /* HAVE_STRUCT_NLIST_DECLARATION */
|
|
|
|
|
|
#if __LIBELF64
|
|
|
#include <gelf.h>
|
|
|
#endif /* __LIBELF64 */
|
|
|
|
|
|
typedef struct Scn_Data Scn_Data;
|
|
|
|
|
|
/*
|
|
|
* ELF descriptor
|
|
|
*/
|
|
|
struct Elf {
|
|
|
/* common */
|
|
|
size_t e_size; /* file/member size */
|
|
|
size_t e_dsize; /* size of memory image */
|
|
|
Elf_Kind e_kind; /* kind of file */
|
|
|
char* e_data; /* file/member data */
|
|
|
char* e_rawdata; /* file/member raw data */
|
|
|
size_t e_idlen; /* identifier size */
|
|
|
int e_fd; /* file descriptor */
|
|
|
unsigned e_count; /* activation count */
|
|
|
/* archive members (still common) */
|
|
|
Elf* e_parent; /* NULL if not an archive member */
|
|
|
size_t e_next; /* 0 if not an archive member */
|
|
|
size_t e_base; /* 0 if not an archive member */
|
|
|
Elf* e_link; /* next archive member or NULL */
|
|
|
Elf_Arhdr* e_arhdr; /* archive member header or NULL */
|
|
|
/* archives */
|
|
|
size_t e_off; /* current member offset (for elf_begin) */
|
|
|
Elf* e_members; /* linked list of active archive members */
|
|
|
char* e_symtab; /* archive symbol table */
|
|
|
size_t e_symlen; /* length of archive symbol table */
|
|
|
char* e_strtab; /* archive string table */
|
|
|
size_t e_strlen; /* length of archive string table */
|
|
|
/* ELF files */
|
|
|
unsigned e_class; /* ELF class */
|
|
|
unsigned e_encoding; /* ELF data encoding */
|
|
|
unsigned e_version; /* ELF version */
|
|
|
char* e_ehdr; /* ELF header */
|
|
|
char* e_phdr; /* ELF program header table */
|
|
|
size_t e_phnum; /* size of program header table */
|
|
|
Elf_Scn* e_scn_1; /* first section */
|
|
|
Elf_Scn* e_scn_n; /* last section */
|
|
|
unsigned e_elf_flags; /* elf flags (ELF_F_*) */
|
|
|
unsigned e_ehdr_flags; /* ehdr flags (ELF_F_*) */
|
|
|
unsigned e_phdr_flags; /* phdr flags (ELF_F_*) */
|
|
|
/* misc flags */
|
|
|
unsigned e_readable : 1; /* file is readable */
|
|
|
unsigned e_writable : 1; /* file is writable */
|
|
|
unsigned e_disabled : 1; /* e_fd has been disabled */
|
|
|
unsigned e_cooked : 1; /* e_data was modified */
|
|
|
unsigned e_free_syms : 1; /* e_symtab is malloc'ed */
|
|
|
unsigned e_unmap_data : 1; /* e_data is mmap'ed */
|
|
|
unsigned e_memory : 1; /* created by elf_memory() */
|
|
|
/* magic number for debugging */
|
|
|
long e_magic;
|
|
|
};
|
|
|
|
|
|
#define ELF_MAGIC 0x012b649e
|
|
|
|
|
|
#define INIT_ELF {\
|
|
|
/* e_size */ 0,\
|
|
|
/* e_dsize */ 0,\
|
|
|
/* e_kind */ ELF_K_NONE,\
|
|
|
/* e_data */ NULL,\
|
|
|
/* e_rawdata */ NULL,\
|
|
|
/* e_idlen */ 0,\
|
|
|
/* e_fd */ -1,\
|
|
|
/* e_count */ 1,\
|
|
|
/* e_parent */ NULL,\
|
|
|
/* e_next */ 0,\
|
|
|
/* e_base */ 0,\
|
|
|
/* e_link */ NULL,\
|
|
|
/* e_arhdr */ NULL,\
|
|
|
/* e_off */ 0,\
|
|
|
/* e_members */ NULL,\
|
|
|
/* e_symtab */ NULL,\
|
|
|
/* e_symlen */ 0,\
|
|
|
/* e_strtab */ NULL,\
|
|
|
/* e_strlen */ 0,\
|
|
|
/* e_class */ ELFCLASSNONE,\
|
|
|
/* e_encoding */ ELFDATANONE,\
|
|
|
/* e_version */ EV_NONE,\
|
|
|
/* e_ehdr */ NULL,\
|
|
|
/* e_phdr */ NULL,\
|
|
|
/* e_phnum */ 0,\
|
|
|
/* e_scn_1 */ NULL,\
|
|
|
/* e_scn_n */ NULL,\
|
|
|
/* e_elf_flags */ 0,\
|
|
|
/* e_ehdr_flags */ 0,\
|
|
|
/* e_phdr_flags */ 0,\
|
|
|
/* e_readable */ 0,\
|
|
|
/* e_writable */ 0,\
|
|
|
/* e_disabled */ 0,\
|
|
|
/* e_cooked */ 0,\
|
|
|
/* e_free_syms */ 0,\
|
|
|
/* e_unmap_data */ 0,\
|
|
|
/* e_memory */ 0,\
|
|
|
/* e_magic */ ELF_MAGIC\
|
|
|
}
|
|
|
|
|
|
/*
|
|
|
* Section descriptor
|
|
|
*/
|
|
|
struct Elf_Scn {
|
|
|
Elf_Scn* s_link; /* pointer to next Elf_Scn */
|
|
|
Elf* s_elf; /* pointer to elf descriptor */
|
|
|
size_t s_index; /* number of this section */
|
|
|
unsigned s_scn_flags; /* section flags (ELF_F_*) */
|
|
|
unsigned s_shdr_flags; /* shdr flags (ELF_F_*) */
|
|
|
Scn_Data* s_data_1; /* first data buffer */
|
|
|
Scn_Data* s_data_n; /* last data buffer */
|
|
|
Scn_Data* s_rawdata; /* raw data buffer */
|
|
|
/* data copied from shdr */
|
|
|
unsigned s_type; /* section type */
|
|
|
size_t s_offset; /* section offset */
|
|
|
size_t s_size; /* section size */
|
|
|
/* misc flags */
|
|
|
unsigned s_freeme : 1; /* this Elf_Scn was malloc'ed */
|
|
|
/* section header */
|
|
|
union {
|
|
|
#if __LIBELF64
|
|
|
Elf64_Shdr u_shdr64;
|
|
|
#endif /* __LIBELF64 */
|
|
|
Elf32_Shdr u_shdr32;
|
|
|
} s_uhdr;
|
|
|
/* magic number for debugging */
|
|
|
long s_magic;
|
|
|
};
|
|
|
#define s_shdr32 s_uhdr.u_shdr32
|
|
|
#define s_shdr64 s_uhdr.u_shdr64
|
|
|
|
|
|
#define SCN_MAGIC 0x012c747d
|
|
|
|
|
|
#define INIT_SCN {\
|
|
|
/* s_link */ NULL,\
|
|
|
/* s_elf */ NULL,\
|
|
|
/* s_index */ 0,\
|
|
|
/* s_scn_flags */ 0,\
|
|
|
/* s_shdr_flags */ 0,\
|
|
|
/* s_data_1 */ NULL,\
|
|
|
/* s_data_n */ NULL,\
|
|
|
/* s_rawdata */ NULL,\
|
|
|
/* s_type */ SHT_NULL,\
|
|
|
/* s_offset */ 0,\
|
|
|
/* s_size */ 0,\
|
|
|
/* s_freeme */ 0,\
|
|
|
/* s_uhdr */ {{0,}},\
|
|
|
/* s_magic */ SCN_MAGIC\
|
|
|
}
|
|
|
|
|
|
/*
|
|
|
* Data descriptor
|
|
|
*/
|
|
|
struct Scn_Data {
|
|
|
Elf_Data sd_data; /* must be first! */
|
|
|
Scn_Data* sd_link; /* pointer to next Scn_Data */
|
|
|
Elf_Scn* sd_scn; /* pointer to section */
|
|
|
char* sd_memdata; /* memory image of section */
|
|
|
unsigned sd_data_flags; /* data flags (ELF_F_*) */
|
|
|
/* misc flags */
|
|
|
unsigned sd_freeme : 1; /* this Scn_Data was malloc'ed */
|
|
|
unsigned sd_free_data : 1; /* sd_memdata is malloc'ed */
|
|
|
/* magic number for debugging */
|
|
|
long sd_magic;
|
|
|
};
|
|
|
|
|
|
#define DATA_MAGIC 0x01072639
|
|
|
|
|
|
#define INIT_DATA {\
|
|
|
{\
|
|
|
/* d_buf */ NULL,\
|
|
|
/* d_type */ ELF_T_BYTE,\
|
|
|
/* d_size */ 0,\
|
|
|
/* d_off */ 0,\
|
|
|
/* d_align */ 0,\
|
|
|
/* d_version */ EV_NONE\
|
|
|
},\
|
|
|
/* sd_link */ NULL,\
|
|
|
/* sd_scn */ NULL,\
|
|
|
/* sd_memdata */ NULL,\
|
|
|
/* sd_data_flags */ 0,\
|
|
|
/* sd_freeme */ 0,\
|
|
|
/* sd_free_data */ 0,\
|
|
|
/* sd_magic */ DATA_MAGIC\
|
|
|
}
|
|
|
|
|
|
/*
|
|
|
* Private status variables
|
|
|
*/
|
|
|
extern unsigned _elf_version;
|
|
|
extern int _elf_errno;
|
|
|
extern int _elf_fill;
|
|
|
extern int _elf_sanity_checks;
|
|
|
#define SANITY_CHECK_STRPTR (1u << 0)
|
|
|
|
|
|
/*
|
|
|
* Private functions
|
|
|
*/
|
|
|
extern void *_elf_read __P((Elf*, void*, size_t, size_t));
|
|
|
extern void *_elf_mmap __P((Elf*));
|
|
|
extern int _elf_cook __P((Elf*));
|
|
|
extern char *_elf_getehdr __P((Elf*, unsigned));
|
|
|
extern char *_elf_getphdr __P((Elf*, unsigned));
|
|
|
extern Elf_Data *_elf_xlatetom __P((const Elf*, Elf_Data*, const Elf_Data*));
|
|
|
extern Elf_Type _elf_scn_type __P((unsigned));
|
|
|
extern size_t _elf32_xltsize __P((const Elf_Data *__src, unsigned __dv, unsigned __encode, int __tof));
|
|
|
extern size_t _elf64_xltsize __P((const Elf_Data *__src, unsigned __dv, unsigned __encode, int __tof));
|
|
|
extern int _elf_update_shnum(Elf *__elf, size_t __shnum);
|
|
|
extern Elf_Scn *_elf_first_scn(Elf *__elf);
|
|
|
|
|
|
/*
|
|
|
* Special translators
|
|
|
*/
|
|
|
extern size_t _elf_verdef_32L11_tof __P((unsigned char *dst, const unsigned char *src, size_t n));
|
|
|
extern size_t _elf_verdef_32L11_tom __P((unsigned char *dst, const unsigned char *src, size_t n));
|
|
|
extern size_t _elf_verdef_32M11_tof __P((unsigned char *dst, const unsigned char *src, size_t n));
|
|
|
extern size_t _elf_verdef_32M11_tom __P((unsigned char *dst, const unsigned char *src, size_t n));
|
|
|
extern size_t _elf_verdef_64L11_tof __P((unsigned char *dst, const unsigned char *src, size_t n));
|
|
|
extern size_t _elf_verdef_64L11_tom __P((unsigned char *dst, const unsigned char *src, size_t n));
|
|
|
extern size_t _elf_verdef_64M11_tof __P((unsigned char *dst, const unsigned char *src, size_t n));
|
|
|
extern size_t _elf_verdef_64M11_tom __P((unsigned char *dst, const unsigned char *src, size_t n));
|
|
|
extern size_t _elf_verneed_32L11_tof __P((unsigned char *dst, const unsigned char *src, size_t n));
|
|
|
extern size_t _elf_verneed_32L11_tom __P((unsigned char *dst, const unsigned char *src, size_t n));
|
|
|
extern size_t _elf_verneed_32M11_tof __P((unsigned char *dst, const unsigned char *src, size_t n));
|
|
|
extern size_t _elf_verneed_32M11_tom __P((unsigned char *dst, const unsigned char *src, size_t n));
|
|
|
extern size_t _elf_verneed_64L11_tof __P((unsigned char *dst, const unsigned char *src, size_t n));
|
|
|
extern size_t _elf_verneed_64L11_tom __P((unsigned char *dst, const unsigned char *src, size_t n));
|
|
|
extern size_t _elf_verneed_64M11_tof __P((unsigned char *dst, const unsigned char *src, size_t n));
|
|
|
extern size_t _elf_verneed_64M11_tom __P((unsigned char *dst, const unsigned char *src, size_t n));
|
|
|
|
|
|
/*
|
|
|
* Private data
|
|
|
*/
|
|
|
extern const Elf_Scn _elf_scn_init;
|
|
|
extern const Scn_Data _elf_data_init;
|
|
|
extern const size_t _elf_fmsize[2][EV_CURRENT - EV_NONE][ELF_T_NUM][2];
|
|
|
|
|
|
/*
|
|
|
* Access macros for _elf_fmsize[]
|
|
|
*/
|
|
|
#define _fmsize(c,v,t,w) \
|
|
|
(_elf_fmsize[(c)-ELFCLASS32][(v)-EV_NONE-1][(t)-ELF_T_BYTE][(w)])
|
|
|
#define _fsize(c,v,t) _fmsize((c),(v),(t),1)
|
|
|
#define _msize(c,v,t) _fmsize((c),(v),(t),0)
|
|
|
|
|
|
/*
|
|
|
* Various checks
|
|
|
*/
|
|
|
#define valid_class(c) ((c) >= ELFCLASS32 && (c) <= ELFCLASS64)
|
|
|
#define valid_encoding(e) ((e) >= ELFDATA2LSB && (e) <= ELFDATA2MSB)
|
|
|
#define valid_version(v) ((v) > EV_NONE && (v) <= EV_CURRENT)
|
|
|
#define valid_type(t) ((unsigned)(t) < ELF_T_NUM)
|
|
|
|
|
|
/*
|
|
|
* Error codes
|
|
|
*/
|
|
|
enum {
|
|
|
#define __err__(a,b) a,
|
|
|
#include <errors.h> /* include constants from errors.h */
|
|
|
#undef __err__
|
|
|
ERROR_NUM
|
|
|
};
|
|
|
|
|
|
#define seterr(err) (_elf_errno = (err))
|
|
|
|
|
|
/*
|
|
|
* Sizes of data types (external representation)
|
|
|
* These definitions should be in <elf.h>, but...
|
|
|
*/
|
|
|
#ifndef ELF32_FSZ_ADDR
|
|
|
# define ELF32_FSZ_ADDR 4
|
|
|
# define ELF32_FSZ_HALF 2
|
|
|
# define ELF32_FSZ_OFF 4
|
|
|
# define ELF32_FSZ_SWORD 4
|
|
|
# define ELF32_FSZ_WORD 4
|
|
|
#endif /* ELF32_FSZ_ADDR */
|
|
|
#ifndef ELF64_FSZ_ADDR
|
|
|
# define ELF64_FSZ_ADDR 8
|
|
|
# define ELF64_FSZ_HALF 2
|
|
|
# define ELF64_FSZ_OFF 8
|
|
|
# define ELF64_FSZ_SWORD 4
|
|
|
# define ELF64_FSZ_SXWORD 8
|
|
|
# define ELF64_FSZ_WORD 4
|
|
|
# define ELF64_FSZ_XWORD 8
|
|
|
#endif /* ELF64_FSZ_ADDR */
|
|
|
|
|
|
/*
|
|
|
* More missing pieces, in no particular order
|
|
|
*/
|
|
|
#ifndef SHT_SYMTAB_SHNDX
|
|
|
#define SHT_SYMTAB_SHNDX 18
|
|
|
#endif /* SHT_SYMTAB_SHNDX */
|
|
|
|
|
|
#ifndef SHN_XINDEX
|
|
|
#define SHN_XINDEX 0xffff
|
|
|
#endif /* SHN_XINDEX */
|
|
|
|
|
|
#ifndef PN_XNUM
|
|
|
#define PN_XNUM 0xffff
|
|
|
#endif /* PN_XNUM */
|
|
|
|
|
|
/*
|
|
|
* Debugging
|
|
|
*/
|
|
|
#if ENABLE_DEBUG
|
|
|
extern void __elf_assert __P((const char*, unsigned, const char*));
|
|
|
# if (__STDC__ + 0)
|
|
|
# define elf_assert(x) do{if(!(x))__elf_assert(__FILE__,__LINE__,#x);}while(0)
|
|
|
# else /* __STDC__ */
|
|
|
# define elf_assert(x) do{if(!(x))__elf_assert(__FILE__,__LINE__,"x");}while(0)
|
|
|
# endif /* __STDC__ */
|
|
|
#else /* ENABLE_DEBUG */
|
|
|
# define elf_assert(x) do{}while(0)
|
|
|
#endif /* ENABLE_DEBUG */
|
|
|
|
|
|
/*
|
|
|
* Return values for certain functions
|
|
|
*/
|
|
|
#define LIBELF_SUCCESS 1
|
|
|
#define LIBELF_FAILURE 0
|
|
|
|
|
|
#endif /* _PRIVATE_H */
|
|
|
|