Blame tests/asm-tst3.c

Packit Service 97d2fb
/* Copyright (C) 2002, 2005 Red Hat, Inc.
Packit Service 97d2fb
   This file is part of elfutils.
Packit Service 97d2fb
   Written by Ulrich Drepper <drepper@redhat.com>, 2002.
Packit Service 97d2fb
Packit Service 97d2fb
   This file is free software; you can redistribute it and/or modify
Packit Service 97d2fb
   it under the terms of the GNU General Public License as published by
Packit Service 97d2fb
   the Free Software Foundation; either version 3 of the License, or
Packit Service 97d2fb
   (at your option) any later version.
Packit Service 97d2fb
Packit Service 97d2fb
   elfutils is distributed in the hope that it will be useful, but
Packit Service 97d2fb
   WITHOUT ANY WARRANTY; without even the implied warranty of
Packit Service 97d2fb
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
Packit Service 97d2fb
   GNU General Public License for more details.
Packit Service 97d2fb
Packit Service 97d2fb
   You should have received a copy of the GNU General Public License
Packit Service 97d2fb
   along with this program.  If not, see <http://www.gnu.org/licenses/>.  */
Packit Service 97d2fb
Packit Service 97d2fb
#ifdef HAVE_CONFIG_H
Packit Service 97d2fb
# include <config.h>
Packit Service 97d2fb
#endif
Packit Service 97d2fb
Packit Service 97d2fb
#include <fcntl.h>
Packit Service 97d2fb
#include ELFUTILS_HEADER(asm)
Packit Service 97d2fb
#include ELFUTILS_HEADER(ebl)
Packit Service 97d2fb
#include <libelf.h>
Packit Service 97d2fb
#include <stdio.h>
Packit Service 97d2fb
#include <string.h>
Packit Service 97d2fb
#include <unistd.h>
Packit Service 97d2fb
Packit Service 97d2fb
Packit Service 97d2fb
static const char fname[] = "asm-tst3-out.o";
Packit Service 97d2fb
Packit Service 97d2fb
Packit Service 97d2fb
static const char *scnnames[5] =
Packit Service 97d2fb
  {
Packit Service 97d2fb
    [0] = "",
Packit Service 97d2fb
    [1] = ".data",
Packit Service 97d2fb
    [2] = ".strtab",
Packit Service 97d2fb
    [3] = ".symtab",
Packit Service 97d2fb
    [4] = ".shstrtab"
Packit Service 97d2fb
  };
Packit Service 97d2fb
Packit Service 97d2fb
Packit Service 97d2fb
static unsigned int scntypes[5] =
Packit Service 97d2fb
  {
Packit Service 97d2fb
    [0] = SHT_NULL,
Packit Service 97d2fb
    [1] = SHT_PROGBITS,
Packit Service 97d2fb
    [2] = SHT_STRTAB,
Packit Service 97d2fb
    [3] = SHT_SYMTAB,
Packit Service 97d2fb
    [4] = SHT_STRTAB
Packit Service 97d2fb
  };
Packit Service 97d2fb
Packit Service 97d2fb
Packit Service 97d2fb
int
Packit Service 97d2fb
main (void)
Packit Service 97d2fb
{
Packit Service 97d2fb
  AsmCtx_t *ctx;
Packit Service 97d2fb
  AsmScn_t *scn1;
Packit Service 97d2fb
  AsmScn_t *scn2;
Packit Service 97d2fb
  int result = 0;
Packit Service 97d2fb
  int fd;
Packit Service 97d2fb
  Elf *elf;
Packit Service 97d2fb
  GElf_Ehdr ehdr_mem;
Packit Service 97d2fb
  GElf_Ehdr *ehdr;
Packit Service 97d2fb
  size_t cnt;
Packit Service 97d2fb
Packit Service 97d2fb
  elf_version (EV_CURRENT);
Packit Service 97d2fb
Packit Service 97d2fb
  Ebl *ebl = ebl_openbackend_machine (EM_386);
Packit Service 97d2fb
  if (ebl == NULL)
Packit Service 97d2fb
    {
Packit Service 97d2fb
      puts ("cannot open backend library");
Packit Service 97d2fb
      return 1;
Packit Service 97d2fb
    }
Packit Service 97d2fb
Packit Service 97d2fb
  ctx = asm_begin (fname, ebl, false);
Packit Service 97d2fb
  if (ctx == NULL)
Packit Service 97d2fb
    {
Packit Service 97d2fb
      printf ("cannot create assembler context: %s\n", asm_errmsg (-1));
Packit Service 97d2fb
      return 1;
Packit Service 97d2fb
    }
Packit Service 97d2fb
Packit Service 97d2fb
  /* Create two sections.  */
Packit Service 97d2fb
  scn1 = asm_newscn (ctx, ".data", SHT_PROGBITS, SHF_ALLOC | SHF_WRITE);
Packit Service 97d2fb
  scn2 = asm_newsubscn (scn1, 1);
Packit Service 97d2fb
  if (scn1 == NULL || scn2 == NULL)
Packit Service 97d2fb
    {
Packit Service 97d2fb
      printf ("cannot create section in output file: %s\n", asm_errmsg (-1));
Packit Service 97d2fb
      asm_abort (ctx);
Packit Service 97d2fb
      return 1;
Packit Service 97d2fb
    }
Packit Service 97d2fb
Packit Service 97d2fb
  /* Special alignment for the .text section.  */
Packit Service 97d2fb
  if (asm_align (scn1, 16) != 0)
Packit Service 97d2fb
    {
Packit Service 97d2fb
      printf ("cannot align .text section: %s\n", asm_errmsg (-1));
Packit Service 97d2fb
      result = 1;
Packit Service 97d2fb
    }
Packit Service 97d2fb
Packit Service 97d2fb
  /* Add a few strings with names.  */
Packit Service 97d2fb
  if (asm_newsym (scn1, "one", 4, STT_OBJECT, STB_GLOBAL) == NULL)
Packit Service 97d2fb
    {
Packit Service 97d2fb
      printf ("cannot create first name: %s\n", asm_errmsg (-1));
Packit Service 97d2fb
      result = 1;
Packit Service 97d2fb
    }
Packit Service 97d2fb
  if (asm_addstrz (scn1, "one", 4) != 0)
Packit Service 97d2fb
    {
Packit Service 97d2fb
      printf ("cannot insert first string: %s\n", asm_errmsg (-1));
Packit Service 97d2fb
      result = 1;
Packit Service 97d2fb
    }
Packit Service 97d2fb
  if (asm_newsym (scn2, "three", 6, STT_OBJECT, STB_WEAK) == NULL)
Packit Service 97d2fb
    {
Packit Service 97d2fb
      printf ("cannot create second name: %s\n", asm_errmsg (-1));
Packit Service 97d2fb
      result = 1;
Packit Service 97d2fb
    }
Packit Service 97d2fb
  if (asm_addstrz (scn2, "three", 0) != 0)
Packit Service 97d2fb
    {
Packit Service 97d2fb
      printf ("cannot insert second string: %s\n", asm_errmsg (-1));
Packit Service 97d2fb
      result = 1;
Packit Service 97d2fb
    }
Packit Service 97d2fb
  if (asm_newsym (scn1, "two", 4, STT_OBJECT, STB_LOCAL) == NULL)
Packit Service 97d2fb
    {
Packit Service 97d2fb
      printf ("cannot create third name: %s\n", asm_errmsg (-1));
Packit Service 97d2fb
      result = 1;
Packit Service 97d2fb
    }
Packit Service 97d2fb
  if (asm_addstrz (scn1, "two", 4) != 0)
Packit Service 97d2fb
    {
Packit Service 97d2fb
      printf ("cannot insert third string: %s\n", asm_errmsg (-1));
Packit Service 97d2fb
      result = 1;
Packit Service 97d2fb
    }
Packit Service 97d2fb
Packit Service 97d2fb
  /* Create the output file.  */
Packit Service 97d2fb
  if (asm_end (ctx) != 0)
Packit Service 97d2fb
    {
Packit Service 97d2fb
      printf ("cannot create output file: %s\n", asm_errmsg (-1));
Packit Service 97d2fb
      asm_abort (ctx);
Packit Service 97d2fb
      return 1;
Packit Service 97d2fb
    }
Packit Service 97d2fb
Packit Service 97d2fb
  /* Check the file.  */
Packit Service 97d2fb
  fd = open (fname, O_RDONLY);
Packit Service 97d2fb
  if (fd == -1)
Packit Service 97d2fb
    {
Packit Service 97d2fb
      printf ("cannot open generated file: %m\n");
Packit Service 97d2fb
      result = 1;
Packit Service 97d2fb
      goto out;
Packit Service 97d2fb
    }
Packit Service 97d2fb
Packit Service 97d2fb
  elf = elf_begin (fd, ELF_C_READ, NULL);
Packit Service 97d2fb
  if (elf == NULL)
Packit Service 97d2fb
    {
Packit Service 97d2fb
      printf ("cannot create ELF descriptor: %s\n", elf_errmsg (-1));
Packit Service 97d2fb
      result = 1;
Packit Service 97d2fb
      goto out_close;
Packit Service 97d2fb
    }
Packit Service 97d2fb
  if (elf_kind (elf) != ELF_K_ELF)
Packit Service 97d2fb
    {
Packit Service 97d2fb
      puts ("not a valid ELF file");
Packit Service 97d2fb
      result = 1;
Packit Service 97d2fb
      goto out_close2;
Packit Service 97d2fb
    }
Packit Service 97d2fb
Packit Service 97d2fb
  ehdr = gelf_getehdr (elf, &ehdr_mem);
Packit Service 97d2fb
  if (ehdr == NULL)
Packit Service 97d2fb
    {
Packit Service 97d2fb
      printf ("cannot get ELF header: %s\n", elf_errmsg (-1));
Packit Service 97d2fb
      result = 1;
Packit Service 97d2fb
      goto out_close2;
Packit Service 97d2fb
    }
Packit Service 97d2fb
Packit Service 97d2fb
  for (cnt = 1; cnt < 5; ++cnt)
Packit Service 97d2fb
    {
Packit Service 97d2fb
      Elf_Scn *scn;
Packit Service 97d2fb
      GElf_Shdr shdr_mem;
Packit Service 97d2fb
      GElf_Shdr *shdr;
Packit Service 97d2fb
Packit Service 97d2fb
      scn = elf_getscn (elf, cnt);
Packit Service 97d2fb
      if (scn == NULL)
Packit Service 97d2fb
	{
Packit Service 97d2fb
	  printf ("cannot get section %zd: %s\n", cnt, elf_errmsg (-1));
Packit Service 97d2fb
	  result = 1;
Packit Service 97d2fb
	  continue;
Packit Service 97d2fb
	}
Packit Service 97d2fb
Packit Service 97d2fb
      shdr = gelf_getshdr (scn, &shdr_mem);
Packit Service 97d2fb
      if (shdr == NULL)
Packit Service 97d2fb
	{
Packit Service 97d2fb
	  printf ("cannot get section header for section %zd: %s\n",
Packit Service 97d2fb
		  cnt, elf_errmsg (-1));
Packit Service 97d2fb
	  result = 1;
Packit Service 97d2fb
	  continue;
Packit Service 97d2fb
	}
Packit Service 97d2fb
Packit Service 97d2fb
      if (strcmp (elf_strptr (elf, ehdr->e_shstrndx, shdr->sh_name),
Packit Service 97d2fb
		  scnnames[cnt]) != 0)
Packit Service 97d2fb
	{
Packit Service 97d2fb
	  printf ("section %zd's name differs: %s vs %s\n", cnt,
Packit Service 97d2fb
		  elf_strptr (elf, ehdr->e_shstrndx, shdr->sh_name),
Packit Service 97d2fb
		  scnnames[cnt]);
Packit Service 97d2fb
	  result = 1;
Packit Service 97d2fb
	}
Packit Service 97d2fb
Packit Service 97d2fb
      if (shdr->sh_type != scntypes[cnt])
Packit Service 97d2fb
	{
Packit Service 97d2fb
	  printf ("section %zd's type differs\n", cnt);
Packit Service 97d2fb
	  result = 1;
Packit Service 97d2fb
	}
Packit Service 97d2fb
Packit Service 97d2fb
      if ((cnt == 1 && shdr->sh_flags != (SHF_ALLOC | SHF_WRITE))
Packit Service 97d2fb
	  || (cnt != 1 && shdr->sh_flags != 0))
Packit Service 97d2fb
	{
Packit Service 97d2fb
	  printf ("section %zd's flags differs\n", cnt);
Packit Service 97d2fb
	  result = 1;
Packit Service 97d2fb
	}
Packit Service 97d2fb
Packit Service 97d2fb
      if (shdr->sh_addr != 0)
Packit Service 97d2fb
	{
Packit Service 97d2fb
	  printf ("section %zd's address differs\n", cnt);
Packit Service 97d2fb
	  result = 1;
Packit Service 97d2fb
	}
Packit Service 97d2fb
Packit Service 97d2fb
      if (cnt == 3)
Packit Service 97d2fb
	{
Packit Service 97d2fb
	  Elf_Data *data;
Packit Service 97d2fb
Packit Service 97d2fb
	  if (shdr->sh_link != 2)
Packit Service 97d2fb
	    {
Packit Service 97d2fb
	      puts ("symbol table has incorrect link");
Packit Service 97d2fb
	      result = 1;
Packit Service 97d2fb
	    }
Packit Service 97d2fb
Packit Service 97d2fb
	  data = elf_getdata (scn, NULL);
Packit Service 97d2fb
	  if (data == NULL)
Packit Service 97d2fb
	    {
Packit Service 97d2fb
	      puts ("cannot get data of symbol table");
Packit Service 97d2fb
	      result = 1;
Packit Service 97d2fb
	    }
Packit Service 97d2fb
	  else
Packit Service 97d2fb
	    {
Packit Service 97d2fb
	      size_t inner;
Packit Service 97d2fb
Packit Service 97d2fb
	      for (inner = 1;
Packit Service 97d2fb
		   inner < (shdr->sh_size
Packit Service 97d2fb
			    / gelf_fsize (elf, ELF_T_SYM, 1, EV_CURRENT));
Packit Service 97d2fb
		   ++inner)
Packit Service 97d2fb
		{
Packit Service 97d2fb
		  GElf_Sym sym_mem;
Packit Service 97d2fb
		  GElf_Sym *sym;
Packit Service 97d2fb
Packit Service 97d2fb
		  sym = gelf_getsym (data, inner, &sym_mem);
Packit Service 97d2fb
		  if (sym == NULL)
Packit Service 97d2fb
		    {
Packit Service 97d2fb
		      printf ("cannot get symbol %zu: %s\n",
Packit Service 97d2fb
			      inner, elf_errmsg (-1));
Packit Service 97d2fb
		      result = 1;
Packit Service 97d2fb
		    }
Packit Service 97d2fb
		  else
Packit Service 97d2fb
		    {
Packit Service 97d2fb
		      /* The order of the third and fourth entry depends
Packit Service 97d2fb
			 on how the hash table is organized.  */
Packit Service 97d2fb
		      static const char *names[4] =
Packit Service 97d2fb
			{
Packit Service 97d2fb
			  [0] = "",
Packit Service 97d2fb
			  [1] = "two",
Packit Service 97d2fb
			  [2] = "one",
Packit Service 97d2fb
			  [3] = "three"
Packit Service 97d2fb
			};
Packit Service 97d2fb
		      static const int info[4] =
Packit Service 97d2fb
			{
Packit Service 97d2fb
			  [0] = GELF_ST_INFO (STB_LOCAL, STT_NOTYPE),
Packit Service 97d2fb
			  [1] = GELF_ST_INFO (STB_LOCAL, STT_OBJECT),
Packit Service 97d2fb
			  [2] = GELF_ST_INFO (STB_GLOBAL, STT_OBJECT),
Packit Service 97d2fb
			  [3] = GELF_ST_INFO (STB_WEAK, STT_OBJECT)
Packit Service 97d2fb
			};
Packit Service 97d2fb
		      static const unsigned value[4] =
Packit Service 97d2fb
			{
Packit Service 97d2fb
			  [0] = 0,
Packit Service 97d2fb
			  [1] = 4,
Packit Service 97d2fb
			  [2] = 0,
Packit Service 97d2fb
			  [3] = 8
Packit Service 97d2fb
			};
Packit Service 97d2fb
Packit Service 97d2fb
		      if (strcmp (names[inner],
Packit Service 97d2fb
				  elf_strptr (elf, shdr->sh_link,
Packit Service 97d2fb
					      sym->st_name)) != 0)
Packit Service 97d2fb
			{
Packit Service 97d2fb
			  printf ("symbol %zu has different name\n", inner);
Packit Service 97d2fb
			  result = 1;
Packit Service 97d2fb
			}
Packit Service 97d2fb
Packit Service 97d2fb
		      if (sym->st_value != value[inner])
Packit Service 97d2fb
			{
Packit Service 97d2fb
			  printf ("symbol %zu has wrong value\n", inner);
Packit Service 97d2fb
			  result = 1;
Packit Service 97d2fb
			}
Packit Service 97d2fb
Packit Service 97d2fb
		      if (sym->st_other != 0)
Packit Service 97d2fb
			{
Packit Service 97d2fb
			  printf ("symbol %zu has wrong other info\n", inner);
Packit Service 97d2fb
			  result = 1;
Packit Service 97d2fb
			}
Packit Service 97d2fb
Packit Service 97d2fb
		      if (sym->st_shndx != 1)
Packit Service 97d2fb
			{
Packit Service 97d2fb
			  printf ("symbol %zu has wrong section reference\n",
Packit Service 97d2fb
				  inner);
Packit Service 97d2fb
			  result = 1;
Packit Service 97d2fb
			}
Packit Service 97d2fb
Packit Service 97d2fb
		      if (sym->st_info != info[inner])
Packit Service 97d2fb
			{
Packit Service 97d2fb
			  printf ("symbol %zu has wrong type or binding\n",
Packit Service 97d2fb
				  inner);
Packit Service 97d2fb
			  result = 1;
Packit Service 97d2fb
			}
Packit Service 97d2fb
Packit Service 97d2fb
		      if ((inner != 3 && sym->st_size != 4)
Packit Service 97d2fb
			  || (inner == 3 && sym->st_size != 6))
Packit Service 97d2fb
			{
Packit Service 97d2fb
			  printf ("symbol %zu has wrong size\n", inner);
Packit Service 97d2fb
			  result = 1;
Packit Service 97d2fb
			}
Packit Service 97d2fb
		    }
Packit Service 97d2fb
		}
Packit Service 97d2fb
	    }
Packit Service 97d2fb
	}
Packit Service 97d2fb
    }
Packit Service 97d2fb
Packit Service 97d2fb
 out_close2:
Packit Service 97d2fb
  elf_end (elf);
Packit Service 97d2fb
 out_close:
Packit Service 97d2fb
  close (fd);
Packit Service 97d2fb
 out:
Packit Service 97d2fb
  /* We don't need the file anymore.  */
Packit Service 97d2fb
  unlink (fname);
Packit Service 97d2fb
Packit Service 97d2fb
  ebl_closebackend (ebl);
Packit Service 97d2fb
Packit Service 97d2fb
  return result;
Packit Service 97d2fb
}