183 lines
5.5 KiB
C
183 lines
5.5 KiB
C
/* Test program for adding section and program headers and ehdr updates.
|
|
Copyright (C) 2015 Red Hat, Inc.
|
|
This file is part of elfutils.
|
|
|
|
This file 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 3 of the License, or
|
|
(at your option) any later version.
|
|
|
|
elfutils 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, see <http://www.gnu.org/licenses/>. */
|
|
|
|
#include <config.h>
|
|
#include <assert.h>
|
|
#include ELFUTILS_HEADER(elf)
|
|
#include <gelf.h>
|
|
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
|
|
#include <sys/types.h>
|
|
#include <sys/stat.h>
|
|
#include <fcntl.h>
|
|
#include <unistd.h>
|
|
|
|
#include <stdbool.h>
|
|
|
|
void
|
|
check (const char *msg, bool statement)
|
|
{
|
|
if (! statement)
|
|
{
|
|
fprintf (stderr, "%s FAILED\n", msg);
|
|
exit (-1);
|
|
}
|
|
else
|
|
fprintf (stderr, "%s OK\n", msg);
|
|
}
|
|
|
|
void
|
|
check_elf (const char *msg, bool statement)
|
|
{
|
|
if (! statement)
|
|
{
|
|
fprintf (stderr, "%s: %s\n", msg, elf_errmsg (-1));
|
|
exit (-1);
|
|
}
|
|
else
|
|
fprintf (stderr, "%s OK\n", msg);
|
|
}
|
|
|
|
void
|
|
test (Elf *elf, int class, bool layout)
|
|
{
|
|
fprintf (stderr, "testing ELF class: %d, layout: %d\n", class, layout);
|
|
|
|
check_elf ("gelf_newehdr", gelf_newehdr (elf, class) != 0);
|
|
check_elf ("gelf_getclass", gelf_getclass (elf) == class);
|
|
|
|
check_elf ("elf_flagelf", elf_flagelf (elf, layout ? ELF_C_SET : ELF_C_CLR,
|
|
ELF_F_LAYOUT) != 0);
|
|
|
|
GElf_Ehdr ehdr;
|
|
check_elf ("gelf_getehdr", gelf_getehdr (elf, &ehdr) != NULL);
|
|
check ("e_shnum == 0", ehdr.e_shnum == 0);
|
|
check ("e_phnum == 0", ehdr.e_phnum == 0);
|
|
check ("e_shoff == 0", ehdr.e_shoff == 0);
|
|
check ("e_phoff == 0", ehdr.e_phoff == 0);
|
|
|
|
size_t shnum;
|
|
check_elf ("elf_getshdrnum", elf_getshdrnum (elf, &shnum) == 0);
|
|
check ("shnum == 0", shnum == 0);
|
|
|
|
size_t phnum;
|
|
check_elf ("elf_getphdrnum", elf_getphdrnum (elf, &phnum) == 0);
|
|
check ("phnum == 0", phnum == 0);
|
|
|
|
/* Lets fill in some info we are always responsible for. */
|
|
ehdr.e_ident[EI_DATA] = ELFDATANONE; /* Ask for native encoding. */
|
|
ehdr.e_type = ET_EXEC;
|
|
ehdr.e_machine = EM_386;
|
|
ehdr.e_version = EV_NONE; /* Ask for current version. */
|
|
check_elf ("gelf_update_ehdr", gelf_update_ehdr (elf, &ehdr) != 0);
|
|
|
|
check_elf ("elf_update", elf_update (elf, ELF_C_NULL) > 0);
|
|
|
|
check_elf ("gelf_getehdr", gelf_getehdr (elf, &ehdr) != NULL);
|
|
check ("EI_DATA", ehdr.e_ident[EI_DATA] != ELFDATANONE);
|
|
check ("e_version", ehdr.e_version == EV_CURRENT);
|
|
|
|
/* The sh/ph values shouldn't have changed. */
|
|
check ("e_shnum == 0", ehdr.e_shnum == 0);
|
|
check ("e_phnum == 0", ehdr.e_phnum == 0);
|
|
check ("e_shoff == 0", ehdr.e_shoff == 0);
|
|
check ("e_phoff == 0", ehdr.e_phoff == 0);
|
|
|
|
check_elf ("elf_getshdrnum", elf_getshdrnum (elf, &shnum) == 0);
|
|
check ("shnum == 0", shnum == 0);
|
|
|
|
check_elf ("elf_getphdrnum", elf_getphdrnum (elf, &phnum) == 0);
|
|
check ("phnum == 0", phnum == 0);
|
|
|
|
/* Lets add a header. */
|
|
check_elf ("elf_newscn", elf_newscn (elf) != NULL);
|
|
check_elf ("gelf_newphdr", gelf_newphdr (elf, 1) != 0);
|
|
|
|
/* If we are responsible for the layout ourselves we should also
|
|
tell where to put them. */
|
|
if (layout)
|
|
{
|
|
check_elf ("gelf_getehdr", gelf_getehdr (elf, &ehdr) != NULL);
|
|
/* phdrs go right after the ehdr. */
|
|
ehdr.e_phoff = ehdr.e_ehsize;
|
|
/* shdrs go right after the phdrs. */
|
|
ehdr.e_shoff = ehdr.e_phoff + ehdr.e_phnum * ehdr.e_phentsize;
|
|
check_elf ("gelf_update_ehdr", gelf_update_ehdr (elf, &ehdr) != 0);
|
|
}
|
|
|
|
check_elf ("elf_update", elf_update (elf, ELF_C_NULL) > 0);
|
|
|
|
check_elf ("elf_getshdrnum", elf_getshdrnum (elf, &shnum) == 0);
|
|
check ("shnum == 1", shnum == 2); /* section zero is also created. */
|
|
|
|
check_elf ("elf_getphdrnum", elf_getphdrnum (elf, &phnum) == 0);
|
|
check ("phnum == 1", phnum == 1);
|
|
|
|
check_elf ("gelf_getehdr", gelf_getehdr (elf, &ehdr) != NULL);
|
|
|
|
check ("EI_DATA", ehdr.e_ident[EI_DATA] != ELFDATANONE);
|
|
check ("e_version", ehdr.e_version == EV_CURRENT);
|
|
|
|
check ("e_shnum == 2", ehdr.e_shnum == 2);
|
|
check ("e_phnum == 1", ehdr.e_phnum == 1);
|
|
check ("e_shoff != 0", ehdr.e_shoff != 0);
|
|
check ("e_phoff != 0", ehdr.e_phoff != 0);
|
|
|
|
size_t shentsize = (class == ELFCLASS32
|
|
? sizeof (Elf32_Shdr) : sizeof (Elf64_Shdr));
|
|
check ("e_shentsize", ehdr.e_shentsize == shentsize);
|
|
size_t phentsize = (class == ELFCLASS32
|
|
? sizeof (Elf32_Phdr) : sizeof (Elf64_Phdr));
|
|
check ("e_phentsize", ehdr.e_phentsize == phentsize);
|
|
}
|
|
|
|
int
|
|
main (int argc __attribute__ ((unused)), char **argv __attribute ((unused)))
|
|
{
|
|
elf_version (EV_CURRENT);
|
|
|
|
int fd = open("/dev/null", O_WRONLY);
|
|
check ("open", fd >= 0);
|
|
|
|
Elf *elf;
|
|
|
|
elf = elf_begin (fd, ELF_C_WRITE, NULL);
|
|
check_elf ("elf_begin", elf != NULL);
|
|
test (elf, ELFCLASS32, false);
|
|
elf_end (elf);
|
|
|
|
elf = elf_begin (fd, ELF_C_WRITE, NULL);
|
|
check_elf ("elf_begin", elf != NULL);
|
|
test (elf, ELFCLASS32, true);
|
|
elf_end (elf);
|
|
|
|
elf = elf_begin (fd, ELF_C_WRITE, NULL);
|
|
check_elf ("elf_begin", elf != NULL);
|
|
test (elf, ELFCLASS64, false);
|
|
elf_end (elf);
|
|
|
|
elf = elf_begin (fd, ELF_C_WRITE, NULL);
|
|
check_elf ("elf_begin", elf != NULL);
|
|
test (elf, ELFCLASS64, true);
|
|
elf_end (elf);
|
|
|
|
close (fd);
|
|
return 0;
|
|
}
|