diff options
Diffstat (limited to 'trunk/gelf/gelf.c')
-rw-r--r-- | trunk/gelf/gelf.c | 573 |
1 files changed, 573 insertions, 0 deletions
diff --git a/trunk/gelf/gelf.c b/trunk/gelf/gelf.c new file mode 100644 index 0000000..915cf5b --- /dev/null +++ b/trunk/gelf/gelf.c @@ -0,0 +1,573 @@ +/* Generic ELF wrapper for libelf which does not support gelf_ API. + Copyright (C) 2001, 2002, 2004 Red Hat, Inc. + Written by Jakub Jelinek <jakub@redhat.com>, 2001. + + This program is free software; you can redistribute it and/or modify + it under the terms of the GNU General Public License as published by + the Free Software Foundation; either version 2, or (at your option) + any later version. + + This program 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 General Public License for more details. + + You should have received a copy of the GNU General Public License + along with this program; if not, write to the Free Software Foundation, + Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. */ + +#include <config.h> +#include <elf.h> +#include <libelf.h> +#include <stdlib.h> +#include <string.h> +#include <sys/types.h> +#include "gelf.h" + +inline int +gelf_getclass (Elf *elf) +{ + size_t size; + char *e_ident = elf_getident (elf, &size); + + if (e_ident == NULL) + return ELFCLASSNONE; + switch (e_ident [EI_CLASS]) + { + case ELFCLASS32: + case ELFCLASS64: + return e_ident [EI_CLASS]; + default: + return ELFCLASSNONE; + } +} + +size_t +gelf_fsize (Elf *elf, Elf_Type type, size_t count, unsigned int ver) +{ + switch (gelf_getclass (elf)) + { + case ELFCLASS32: + return elf32_fsize (type, count, ver); + case ELFCLASS64: + return elf64_fsize (type, count, ver); + default: + return 0; + } +} + +GElf_Ehdr * +gelf_getehdr (Elf *elf, GElf_Ehdr *dst) +{ + Elf32_Ehdr *ehdr32; + Elf64_Ehdr *ehdr64; + + switch (gelf_getclass (elf)) + { + case ELFCLASS32: + ehdr32 = elf32_getehdr (elf); + if (ehdr32 != NULL) + { + memcpy (dst->e_ident, ehdr32->e_ident, EI_NIDENT); + dst->e_type = ehdr32->e_type; + dst->e_machine = ehdr32->e_machine; + dst->e_version = ehdr32->e_version; + dst->e_entry = ehdr32->e_entry; + dst->e_phoff = ehdr32->e_phoff; + dst->e_shoff = ehdr32->e_shoff; + dst->e_flags = ehdr32->e_flags; + dst->e_ehsize = ehdr32->e_ehsize; + dst->e_phentsize = ehdr32->e_phentsize; + dst->e_phnum = ehdr32->e_phnum; + dst->e_shentsize = ehdr32->e_shentsize; + dst->e_shnum = ehdr32->e_shnum; + dst->e_shstrndx = ehdr32->e_shstrndx; + return dst; + } + break; + case ELFCLASS64: + ehdr64 = elf64_getehdr (elf); + if (ehdr64 != NULL) + { + memcpy (dst, ehdr64, sizeof (Elf64_Ehdr)); + return dst; + } + } + return NULL; +} + +int +gelf_update_ehdr (Elf *elf, GElf_Ehdr *src) +{ + Elf32_Ehdr *ehdr32; + Elf64_Ehdr *ehdr64; + + switch (gelf_getclass (elf)) + { + case ELFCLASS32: + ehdr32 = elf32_getehdr (elf); + if (ehdr32 == NULL) + return 0; + memcpy (ehdr32->e_ident, src->e_ident, EI_NIDENT); + ehdr32->e_type = src->e_type; + ehdr32->e_machine = src->e_machine; + ehdr32->e_version = src->e_version; + ehdr32->e_entry = src->e_entry; + ehdr32->e_phoff = src->e_phoff; + ehdr32->e_shoff = src->e_shoff; + ehdr32->e_flags = src->e_flags; + ehdr32->e_ehsize = src->e_ehsize; + ehdr32->e_phentsize = src->e_phentsize; + ehdr32->e_phnum = src->e_phnum; + ehdr32->e_shentsize = src->e_shentsize; + ehdr32->e_shnum = src->e_shnum; + ehdr32->e_shstrndx = src->e_shstrndx; + return 1; + case ELFCLASS64: + ehdr64 = elf64_getehdr (elf); + if (ehdr64 != NULL) + { + memcpy (ehdr64, src, sizeof (Elf64_Ehdr)); + return 1; + } + default: + break; + } + return 0; +} + +unsigned long +gelf_newehdr (Elf *elf, int class) +{ + switch (class) + { + case ELFCLASS32: + return (unsigned long) elf32_newehdr (elf); + case ELFCLASS64: + return (unsigned long) elf64_newehdr (elf); + default: + return 0; + } +} + +GElf_Phdr * +gelf_getphdr (Elf *elf, int ndx, GElf_Phdr *dst) +{ + Elf32_Ehdr *ehdr32; + Elf64_Ehdr *ehdr64; + Elf32_Phdr *phdr32; + Elf64_Phdr *phdr64; + + switch (gelf_getclass (elf)) + { + case ELFCLASS32: + phdr32 = elf32_getphdr (elf); + if (phdr32 == NULL) + return NULL; + ehdr32 = elf32_getehdr (elf); + if (ehdr32 == NULL) + return NULL; + if (ndx >= ehdr32->e_phnum) + return NULL; + phdr32 += ndx; + dst->p_type = phdr32->p_type; + dst->p_offset = phdr32->p_offset; + dst->p_vaddr = phdr32->p_vaddr; + dst->p_paddr = phdr32->p_paddr; + dst->p_filesz = phdr32->p_filesz; + dst->p_memsz = phdr32->p_memsz; + dst->p_flags = phdr32->p_flags; + dst->p_align = phdr32->p_align; + return dst; + case ELFCLASS64: + phdr64 = elf64_getphdr (elf); + if (phdr64 == NULL) + return NULL; + ehdr64 = elf64_getehdr (elf); + if (ehdr64 == NULL) + return NULL; + if (ndx >= ehdr64->e_phnum) + return NULL; + memcpy (dst, phdr64 + ndx, sizeof (Elf64_Phdr)); + return dst; + default: + return NULL; + } +} + +int +gelf_update_phdr (Elf *elf, int ndx, GElf_Phdr *src) +{ + Elf32_Ehdr *ehdr32; + Elf64_Ehdr *ehdr64; + Elf32_Phdr *phdr32; + Elf64_Phdr *phdr64; + + switch (gelf_getclass (elf)) + { + case ELFCLASS32: + phdr32 = elf32_getphdr (elf); + if (phdr32 == NULL) + return 0; + ehdr32 = elf32_getehdr (elf); + if (ehdr32 == NULL) + return 0; + if (ndx >= ehdr32->e_phnum) + return 0; + phdr32 += ndx; + phdr32->p_type = src->p_type; + phdr32->p_offset = src->p_offset; + phdr32->p_vaddr = src->p_vaddr; + phdr32->p_paddr = src->p_paddr; + phdr32->p_filesz = src->p_filesz; + phdr32->p_memsz = src->p_memsz; + phdr32->p_flags = src->p_flags; + phdr32->p_align = src->p_align; + return 1; + case ELFCLASS64: + phdr64 = elf64_getphdr (elf); + if (phdr64 == NULL) + return 0; + ehdr64 = elf64_getehdr (elf); + if (ehdr64 == NULL) + return 0; + if (ndx >= ehdr64->e_phnum) + return 0; + memcpy (phdr64 + ndx, src, sizeof (Elf64_Phdr)); + return 1; + default: + return 0; + } +} + +unsigned long +gelf_newphdr (Elf *elf, size_t phnum) +{ + switch (gelf_getclass (elf)) + { + case ELFCLASS32: + return (unsigned long) elf32_newphdr (elf, phnum); + case ELFCLASS64: + return (unsigned long) elf64_newphdr (elf, phnum); + default: + return 0; + } +} + +GElf_Shdr * +gelfx_getshdr (Elf *elf, Elf_Scn *scn, GElf_Shdr *dst) +{ + Elf32_Shdr *shdr32; + Elf64_Shdr *shdr64; + + switch (gelf_getclass (elf)) + { + case ELFCLASS32: + shdr32 = elf32_getshdr (scn); + if (shdr32 == NULL) + return NULL; + dst->sh_name = shdr32->sh_name; + dst->sh_type = shdr32->sh_type; + dst->sh_flags = shdr32->sh_flags; + dst->sh_addr = shdr32->sh_addr; + dst->sh_offset = shdr32->sh_offset; + dst->sh_size = shdr32->sh_size; + dst->sh_link = shdr32->sh_link; + dst->sh_info = shdr32->sh_info; + dst->sh_addralign = shdr32->sh_addralign; + dst->sh_entsize = shdr32->sh_entsize; + return dst; + case ELFCLASS64: + shdr64 = elf64_getshdr (scn); + if (shdr64 == NULL) + return NULL; + memcpy (dst, shdr64, sizeof (Elf64_Shdr)); + return dst; + default: + return NULL; + } +} + +int +gelfx_update_shdr (Elf *elf, Elf_Scn *scn, GElf_Shdr *src) +{ + Elf32_Shdr *shdr32; + Elf64_Shdr *shdr64; + + switch (gelf_getclass (elf)) + { + case ELFCLASS32: + shdr32 = elf32_getshdr (scn); + if (shdr32 == NULL) + return 0; + shdr32->sh_name = src->sh_name; + shdr32->sh_type = src->sh_type; + shdr32->sh_flags = src->sh_flags; + shdr32->sh_addr = src->sh_addr; + shdr32->sh_offset = src->sh_offset; + shdr32->sh_size = src->sh_size; + shdr32->sh_link = src->sh_link; + shdr32->sh_info = src->sh_info; + shdr32->sh_addralign = src->sh_addralign; + shdr32->sh_entsize = src->sh_entsize; + return 1; + case ELFCLASS64: + shdr64 = elf64_getshdr (scn); + if (shdr64 == NULL) + return 0; + memcpy (shdr64, src, sizeof (Elf64_Shdr)); + return 1; + default: + return 0; + } +} + +Elf_Data * +gelf_xlatetom (Elf *elf, Elf_Data *dst, const Elf_Data *src, unsigned encode) +{ + switch (gelf_getclass (elf)) + { + case ELFCLASS32: + return elf32_xlatetom (dst, src, encode); + case ELFCLASS64: + return elf64_xlatetom (dst, src, encode); + default: + return NULL; + } +} + +Elf_Data * +gelf_xlatetof (Elf *elf, Elf_Data *dst, const Elf_Data *src, unsigned encode) +{ + switch (gelf_getclass (elf)) + { + case ELFCLASS32: + return elf32_xlatetof (dst, src, encode); + case ELFCLASS64: + return elf64_xlatetof (dst, src, encode); + default: + return NULL; + } +} + +GElf_Sym *gelfx_getsym (Elf *elf, Elf_Data *data, int ndx, GElf_Sym *dst) +{ + Elf32_Sym *sym32; + + if (data->d_type != ELF_T_SYM) + return NULL; + + switch (gelf_getclass (elf)) + { + case ELFCLASS32: + if ((ndx + 1) * sizeof (Elf32_Sym) > data->d_size) + return NULL; + sym32 = &((Elf32_Sym *) data->d_buf)[ndx]; + dst->st_name = sym32->st_name; + dst->st_info = sym32->st_info; + dst->st_other = sym32->st_other; + dst->st_shndx = sym32->st_shndx; + dst->st_value = sym32->st_value; + dst->st_size = sym32->st_size; + return dst; + case ELFCLASS64: + if ((ndx + 1) * sizeof (Elf64_Sym) > data->d_size) + return NULL; + *dst = ((GElf_Sym *) data->d_buf)[ndx]; + return dst; + default: + return NULL; + } +} + +int gelfx_update_sym (Elf *elf, Elf_Data *data, int ndx, GElf_Sym *src) +{ + Elf32_Sym *sym32; + + if (data->d_type != ELF_T_SYM) + return 0; + + switch (gelf_getclass (elf)) + { + case ELFCLASS32: + if ((ndx + 1) * sizeof (Elf32_Sym) > data->d_size) + return 0; + sym32 = &((Elf32_Sym *) data->d_buf)[ndx]; + sym32->st_name = src->st_name; + sym32->st_info = src->st_info; + sym32->st_other = src->st_other; + sym32->st_shndx = src->st_shndx; + sym32->st_value = src->st_value; + sym32->st_size = src->st_size; + return 1; + case ELFCLASS64: + if ((ndx + 1) * sizeof (Elf64_Sym) > data->d_size) + return 0; + ((GElf_Sym *) data->d_buf)[ndx] = *src; + return 1; + default: + return 0; + } +} + +GElf_Dyn *gelfx_getdyn (Elf *elf, Elf_Data *data, int ndx, GElf_Dyn *dst) +{ + Elf32_Dyn *dyn32; + + if (data->d_type != ELF_T_DYN) + return NULL; + + switch (gelf_getclass (elf)) + { + case ELFCLASS32: + if ((ndx + 1) * sizeof (Elf32_Dyn) > data->d_size) + return NULL; + dyn32 = &((Elf32_Dyn *) data->d_buf)[ndx]; + dst->d_tag = dyn32->d_tag; + dst->d_un.d_val = dyn32->d_un.d_val; + return dst; + case ELFCLASS64: + if ((ndx + 1) * sizeof (Elf64_Dyn) > data->d_size) + return NULL; + *dst = ((GElf_Dyn *) data->d_buf)[ndx]; + return dst; + default: + return NULL; + } +} + +int gelfx_update_dyn (Elf *elf, Elf_Data *data, int ndx, GElf_Dyn *src) +{ + Elf32_Dyn *dyn32; + + if (data->d_type != ELF_T_DYN) + return 0; + + switch (gelf_getclass (elf)) + { + case ELFCLASS32: + if ((ndx + 1) * sizeof (Elf32_Dyn) > data->d_size) + return 0; + dyn32 = &((Elf32_Dyn *) data->d_buf)[ndx]; + dyn32->d_tag = src->d_tag; + dyn32->d_un.d_val = src->d_un.d_val; + return 1; + case ELFCLASS64: + if ((ndx + 1) * sizeof (Elf64_Dyn) > data->d_size) + return 0; + ((GElf_Dyn *) data->d_buf)[ndx] = *src; + return 1; + default: + return 0; + } +} + +GElf_Rel *gelfx_getrel (Elf *elf, Elf_Data *data, int ndx, GElf_Rel *dst) +{ + Elf32_Rel *rel32; + + if (data->d_type != ELF_T_REL) + return NULL; + + switch (gelf_getclass (elf)) + { + case ELFCLASS32: + if ((ndx + 1) * sizeof (Elf32_Rel) > data->d_size) + return NULL; + rel32 = &((Elf32_Rel *) data->d_buf)[ndx]; + dst->r_offset = rel32->r_offset; + dst->r_info = GELF_R_INFO (ELF32_R_SYM (rel32->r_info), + ELF32_R_TYPE (rel32->r_info)); + return dst; + case ELFCLASS64: + if ((ndx + 1) * sizeof (Elf64_Rel) > data->d_size) + return NULL; + *dst = ((GElf_Rel *) data->d_buf)[ndx]; + return dst; + default: + return NULL; + } +} + +int gelfx_update_rel (Elf *elf, Elf_Data *data, int ndx, GElf_Rel *src) +{ + Elf32_Rel *rel32; + + if (data->d_type != ELF_T_REL) + return 0; + + switch (gelf_getclass (elf)) + { + case ELFCLASS32: + if ((ndx + 1) * sizeof (Elf32_Rel) > data->d_size) + return 0; + rel32 = &((Elf32_Rel *) data->d_buf)[ndx]; + rel32->r_offset = src->r_offset; + rel32->r_info = ELF32_R_INFO (GELF_R_SYM (src->r_info), + GELF_R_TYPE (src->r_info)); + return 1; + case ELFCLASS64: + if ((ndx + 1) * sizeof (Elf64_Rel) > data->d_size) + return 0; + ((GElf_Rel *) data->d_buf)[ndx] = *src; + return 1; + default: + return 0; + } +} + +GElf_Rela *gelfx_getrela (Elf *elf, Elf_Data *data, int ndx, GElf_Rela *dst) +{ + Elf32_Rela *rela32; + + if (data->d_type != ELF_T_RELA) + return NULL; + + switch (gelf_getclass (elf)) + { + case ELFCLASS32: + if ((ndx + 1) * sizeof (Elf32_Rela) > data->d_size) + return NULL; + rela32 = &((Elf32_Rela *) data->d_buf)[ndx]; + dst->r_offset = rela32->r_offset; + dst->r_info = GELF_R_INFO (ELF32_R_SYM (rela32->r_info), + ELF32_R_TYPE (rela32->r_info)); + dst->r_addend = rela32->r_addend; + return dst; + case ELFCLASS64: + if ((ndx + 1) * sizeof (Elf64_Rela) > data->d_size) + return NULL; + *dst = ((GElf_Rela *) data->d_buf)[ndx]; + return dst; + default: + return NULL; + } +} + +int gelfx_update_rela (Elf *elf, Elf_Data *data, int ndx, GElf_Rela *src) +{ + Elf32_Rela *rela32; + + if (data->d_type != ELF_T_RELA) + return 0; + + switch (gelf_getclass (elf)) + { + case ELFCLASS32: + if ((ndx + 1) * sizeof (Elf32_Rela) > data->d_size) + return 0; + rela32 = &((Elf32_Rela *) data->d_buf)[ndx]; + rela32->r_offset = src->r_offset; + rela32->r_info = ELF32_R_INFO (GELF_R_SYM (src->r_info), + GELF_R_TYPE (src->r_info)); + rela32->r_addend = src->r_addend; + return 1; + case ELFCLASS64: + if ((ndx + 1) * sizeof (Elf64_Rela) > data->d_size) + return 0; + ((GElf_Rela *) data->d_buf)[ndx] = *src; + return 1; + default: + return 0; + } +} |