Blame tests/asm-tst9.c.libasm-ebl

Packit 5f0dbf
/* Copyright (C) 2002-2010 Red Hat, Inc.
Packit 5f0dbf
   This file is part of elfutils.
Packit 5f0dbf
   Written by Ulrich Drepper <drepper@redhat.com>, 2002.
Packit 5f0dbf
Packit 5f0dbf
   This file is free software; you can redistribute it and/or modify
Packit 5f0dbf
   it under the terms of the GNU General Public License as published by
Packit 5f0dbf
   the Free Software Foundation; either version 3 of the License, or
Packit 5f0dbf
   (at your option) any later version.
Packit 5f0dbf
Packit 5f0dbf
   elfutils is distributed in the hope that it will be useful, but
Packit 5f0dbf
   WITHOUT ANY WARRANTY; without even the implied warranty of
Packit 5f0dbf
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
Packit 5f0dbf
   GNU General Public License for more details.
Packit 5f0dbf
Packit 5f0dbf
   You should have received a copy of the GNU General Public License
Packit 5f0dbf
   along with this program.  If not, see <http://www.gnu.org/licenses/>.  */
Packit 5f0dbf
Packit 5f0dbf
#ifdef HAVE_CONFIG_H
Packit 5f0dbf
# include <config.h>
Packit 5f0dbf
#endif
Packit 5f0dbf
Packit 5f0dbf
#include <fcntl.h>
Packit 5f0dbf
#include <inttypes.h>
Packit 5f0dbf
#include ELFUTILS_HEADER(asm)
Packit 5f0dbf
#include <libelf.h>
Packit 5f0dbf
#include <stdio.h>
Packit 5f0dbf
#include <string.h>
Packit 5f0dbf
#include <unistd.h>
Packit 5f0dbf
Packit 5f0dbf
Packit 5f0dbf
static const char fname[] = "asm-tst9-out.o";
Packit 5f0dbf
Packit 5f0dbf
Packit 5f0dbf
static int32_t input[] =
Packit 5f0dbf
  {
Packit 5f0dbf
    0, 1, 129, 510, 2000, 33000, 0x7ffffff, 0x7fffffff
Packit 5f0dbf
  };
Packit 5f0dbf
#define ninput (sizeof (input) / sizeof (input[0]))
Packit 5f0dbf
Packit 5f0dbf
Packit 5f0dbf
static const GElf_Ehdr expected_ehdr =
Packit 5f0dbf
  {
Packit 5f0dbf
    .e_ident = { [EI_MAG0] = ELFMAG0,
Packit 5f0dbf
		 [EI_MAG1] = ELFMAG1,
Packit 5f0dbf
		 [EI_MAG2] = ELFMAG2,
Packit 5f0dbf
		 [EI_MAG3] = ELFMAG3,
Packit 5f0dbf
		 [EI_CLASS] = ELFCLASS32,
Packit 5f0dbf
		 [EI_DATA] = ELFDATA2LSB,
Packit 5f0dbf
		 [EI_VERSION] = EV_CURRENT },
Packit 5f0dbf
    .e_type = ET_REL,
Packit 5f0dbf
    .e_machine = EM_386,
Packit 5f0dbf
    .e_version = EV_CURRENT,
Packit 5f0dbf
    .e_shoff = 180,
Packit 5f0dbf
    .e_ehsize = sizeof (Elf32_Ehdr),
Packit 5f0dbf
    .e_shentsize = sizeof (Elf32_Shdr),
Packit 5f0dbf
    .e_shnum = 3,
Packit 5f0dbf
    .e_shstrndx = 2
Packit 5f0dbf
  };
Packit 5f0dbf
Packit 5f0dbf
Packit 5f0dbf
static const char *scnnames[3] =
Packit 5f0dbf
  {
Packit 5f0dbf
    [0] = "",
Packit 5f0dbf
    [1] = ".data",
Packit 5f0dbf
    [2] = ".shstrtab"
Packit 5f0dbf
  };
Packit 5f0dbf
Packit 5f0dbf
Packit 5f0dbf
static const char expecteddata[] =
Packit 5f0dbf
  {
Packit 5f0dbf
    0x00, 0x00, 0x00, 0x00, 0x01, 0x01, 0xff, 0xff, 0xff, 0xff, 0x0f, 0x7f,
Packit 5f0dbf
    0x81, 0x01, 0x81, 0x01, 0xff, 0xfe, 0xff, 0xff, 0x0f, 0xff, 0x7e, 0xfe,
Packit 5f0dbf
    0x03, 0xfe, 0x03, 0x82, 0xfc, 0xff, 0xff, 0x0f, 0x82, 0x7c, 0xd0, 0x0f,
Packit 5f0dbf
    0xd0, 0x0f, 0xb0, 0xf0, 0xff, 0xff, 0x0f, 0xb0, 0x70, 0xe8, 0x81, 0x02,
Packit 5f0dbf
    0xe8, 0x81, 0x02, 0x98, 0xfe, 0xfd, 0xff, 0x0f, 0x98, 0xfe, 0x7d, 0xff,
Packit 5f0dbf
    0xff, 0xff, 0x3f, 0xff, 0xff, 0xff, 0x3f, 0x81, 0x80, 0x80, 0xc0, 0x0f,
Packit 5f0dbf
    0x81, 0x80, 0x80, 0x40, 0xff, 0xff, 0xff, 0xff, 0x07, 0xff, 0xff, 0xff,
Packit 5f0dbf
    0xff, 0x07, 0x81, 0x80, 0x80, 0x80, 0x08, 0x81, 0x80, 0x80, 0x80, 0x78
Packit 5f0dbf
  };
Packit 5f0dbf
Packit 5f0dbf
Packit 5f0dbf
int
Packit 5f0dbf
main (void)
Packit 5f0dbf
{
Packit 5f0dbf
  AsmCtx_t *ctx;
Packit 5f0dbf
  AsmScn_t *scn;
Packit 5f0dbf
  int result = 0;
Packit 5f0dbf
  int fd;
Packit 5f0dbf
  Elf *elf;
Packit 5f0dbf
  GElf_Ehdr ehdr_mem;
Packit 5f0dbf
  GElf_Ehdr *ehdr;
Packit 5f0dbf
  size_t cnt;
Packit 5f0dbf
Packit 5f0dbf
  elf_version (EV_CURRENT);
Packit 5f0dbf
Packit 5f0dbf
  Ebl *ebl = ebl_openbackend_machine (EM_386);
Packit 5f0dbf
  if (ebl == NULL)
Packit 5f0dbf
    {
Packit 5f0dbf
      puts ("cannot open backend library");
Packit 5f0dbf
      return 1;
Packit 5f0dbf
    }
Packit 5f0dbf
Packit 5f0dbf
  ctx = asm_begin (fname, ebl, false);
Packit 5f0dbf
  if (ctx == NULL)
Packit 5f0dbf
    {
Packit 5f0dbf
      printf ("cannot create assembler context: %s\n", asm_errmsg (-1));
Packit 5f0dbf
      return 1;
Packit 5f0dbf
    }
Packit 5f0dbf
Packit 5f0dbf
  /* Create two sections.  */
Packit 5f0dbf
  scn = asm_newscn (ctx, ".data", SHT_PROGBITS, SHF_ALLOC | SHF_WRITE);
Packit 5f0dbf
  if (scn == NULL)
Packit 5f0dbf
    {
Packit 5f0dbf
      printf ("cannot create section in output file: %s\n", asm_errmsg (-1));
Packit 5f0dbf
      asm_abort (ctx);
Packit 5f0dbf
      return 1;
Packit 5f0dbf
    }
Packit 5f0dbf
Packit 5f0dbf
  /* Special alignment for the .text section.  */
Packit 5f0dbf
  if (asm_align (scn, 16) != 0)
Packit 5f0dbf
    {
Packit 5f0dbf
      printf ("cannot align .text section: %s\n", asm_errmsg (-1));
Packit 5f0dbf
      result = 1;
Packit 5f0dbf
    }
Packit 5f0dbf
Packit 5f0dbf
  /* Add a few ULEB128 and SLEB128 numbers.  */
Packit 5f0dbf
  for (cnt = 0; cnt < ninput; ++cnt)
Packit 5f0dbf
    {
Packit 5f0dbf
      if (asm_adduleb128 (scn, input[cnt]) != 0)
Packit 5f0dbf
	{
Packit 5f0dbf
	  printf ("cannot insert uleb %" PRIu32 ": %s\n",
Packit 5f0dbf
		  (uint32_t) input[cnt], asm_errmsg (-1));
Packit 5f0dbf
	  result = 1;
Packit 5f0dbf
	}
Packit 5f0dbf
Packit 5f0dbf
      if (asm_addsleb128 (scn, input[cnt]) != 0)
Packit 5f0dbf
	{
Packit 5f0dbf
	  printf ("cannot insert sleb %" PRId32 ": %s\n",
Packit 5f0dbf
		  input[cnt], asm_errmsg (-1));
Packit 5f0dbf
	  result = 1;
Packit 5f0dbf
	}
Packit 5f0dbf
Packit 5f0dbf
      if (asm_adduleb128 (scn, -input[cnt]) != 0)
Packit 5f0dbf
	{
Packit 5f0dbf
	  printf ("cannot insert uleb %" PRIu32 ": %s\n",
Packit 5f0dbf
		  (uint32_t) -input[cnt], asm_errmsg (-1));
Packit 5f0dbf
	  result = 1;
Packit 5f0dbf
	}
Packit 5f0dbf
Packit 5f0dbf
      if (asm_addsleb128 (scn, -input[cnt]) != 0)
Packit 5f0dbf
	{
Packit 5f0dbf
	  printf ("cannot insert sleb %" PRId32 ": %s\n",
Packit 5f0dbf
		  -input[cnt], asm_errmsg (-1));
Packit 5f0dbf
	  result = 1;
Packit 5f0dbf
	}
Packit 5f0dbf
    }
Packit 5f0dbf
Packit 5f0dbf
  /* Create the output file.  */
Packit 5f0dbf
  if (asm_end (ctx) != 0)
Packit 5f0dbf
    {
Packit 5f0dbf
      printf ("cannot create output file: %s\n", asm_errmsg (-1));
Packit 5f0dbf
      asm_abort (ctx);
Packit 5f0dbf
      return 1;
Packit 5f0dbf
    }
Packit 5f0dbf
Packit 5f0dbf
  /* Check the file.  */
Packit 5f0dbf
  fd = open (fname, O_RDONLY);
Packit 5f0dbf
  if (fd == -1)
Packit 5f0dbf
    {
Packit 5f0dbf
      printf ("cannot open generated file: %m\n");
Packit 5f0dbf
      result = 1;
Packit 5f0dbf
      goto out;
Packit 5f0dbf
    }
Packit 5f0dbf
Packit 5f0dbf
  elf = elf_begin (fd, ELF_C_READ, NULL);
Packit 5f0dbf
  if (elf == NULL)
Packit 5f0dbf
    {
Packit 5f0dbf
      printf ("cannot create ELF descriptor: %s\n", elf_errmsg (-1));
Packit 5f0dbf
      result = 1;
Packit 5f0dbf
      goto out_close;
Packit 5f0dbf
    }
Packit 5f0dbf
  if (elf_kind (elf) != ELF_K_ELF)
Packit 5f0dbf
    {
Packit 5f0dbf
      puts ("not a valid ELF file");
Packit 5f0dbf
      result = 1;
Packit 5f0dbf
      goto out_close2;
Packit 5f0dbf
    }
Packit 5f0dbf
Packit 5f0dbf
  ehdr = gelf_getehdr (elf, &ehdr_mem);
Packit 5f0dbf
  if (ehdr == NULL)
Packit 5f0dbf
    {
Packit 5f0dbf
      printf ("cannot get ELF header: %s\n", elf_errmsg (-1));
Packit 5f0dbf
      result = 1;
Packit 5f0dbf
      goto out_close2;
Packit 5f0dbf
    }
Packit 5f0dbf
Packit 5f0dbf
  if (memcmp (ehdr, &expected_ehdr, sizeof (GElf_Ehdr)) != 0)
Packit 5f0dbf
    {
Packit 5f0dbf
      puts ("ELF header does not match");
Packit 5f0dbf
      result = 1;
Packit 5f0dbf
      goto out_close2;
Packit 5f0dbf
    }
Packit 5f0dbf
Packit 5f0dbf
  for (cnt = 1; cnt < 3; ++cnt)
Packit 5f0dbf
    {
Packit 5f0dbf
      Elf_Scn *escn;
Packit 5f0dbf
      GElf_Shdr shdr_mem;
Packit 5f0dbf
      GElf_Shdr *shdr;
Packit 5f0dbf
Packit 5f0dbf
      escn = elf_getscn (elf, cnt);
Packit 5f0dbf
      if (escn == NULL)
Packit 5f0dbf
	{
Packit 5f0dbf
	  printf ("cannot get section %zd: %s\n", cnt, elf_errmsg (-1));
Packit 5f0dbf
	  result = 1;
Packit 5f0dbf
	  continue;
Packit 5f0dbf
	}
Packit 5f0dbf
Packit 5f0dbf
      shdr = gelf_getshdr (escn, &shdr_mem);
Packit 5f0dbf
      if (shdr == NULL)
Packit 5f0dbf
	{
Packit 5f0dbf
	  printf ("cannot get section header for section %zd: %s\n",
Packit 5f0dbf
		  cnt, elf_errmsg (-1));
Packit 5f0dbf
	  result = 1;
Packit 5f0dbf
	  continue;
Packit 5f0dbf
	}
Packit 5f0dbf
Packit 5f0dbf
      if (strcmp (elf_strptr (elf, ehdr->e_shstrndx, shdr->sh_name),
Packit 5f0dbf
		  scnnames[cnt]) != 0)
Packit 5f0dbf
	{
Packit 5f0dbf
	  printf ("section %zd's name differs: %s vs %s\n", cnt,
Packit 5f0dbf
		  elf_strptr (elf, ehdr->e_shstrndx, shdr->sh_name),
Packit 5f0dbf
		  scnnames[cnt]);
Packit 5f0dbf
	  result = 1;
Packit 5f0dbf
	}
Packit 5f0dbf
Packit 5f0dbf
      if (shdr->sh_type != (cnt == 2 ? SHT_STRTAB : SHT_PROGBITS))
Packit 5f0dbf
	{
Packit 5f0dbf
	  printf ("section %zd's type differs\n", cnt);
Packit 5f0dbf
	  result = 1;
Packit 5f0dbf
	}
Packit 5f0dbf
Packit 5f0dbf
      if ((cnt == 1 && shdr->sh_flags != (SHF_ALLOC | SHF_WRITE))
Packit 5f0dbf
	  || (cnt == 2 && shdr->sh_flags != 0))
Packit 5f0dbf
	{
Packit 5f0dbf
	  printf ("section %zd's flags differs\n", cnt);
Packit 5f0dbf
	  result = 1;
Packit 5f0dbf
	}
Packit 5f0dbf
Packit 5f0dbf
      if (shdr->sh_addr != 0)
Packit 5f0dbf
	{
Packit 5f0dbf
	  printf ("section %zd's address differs\n", cnt);
Packit 5f0dbf
	  result = 1;
Packit 5f0dbf
	}
Packit 5f0dbf
Packit 5f0dbf
      if ((cnt == 1 && shdr->sh_offset != ((sizeof (Elf32_Ehdr) + 15) & ~15))
Packit 5f0dbf
	  || (cnt == 2
Packit 5f0dbf
	      && shdr->sh_offset != (((sizeof (Elf32_Ehdr) + 15) & ~15)
Packit 5f0dbf
				     + sizeof (expecteddata))))
Packit 5f0dbf
	{
Packit 5f0dbf
	  printf ("section %zd's offset differs\n", cnt);
Packit 5f0dbf
	  result = 1;
Packit 5f0dbf
	}
Packit 5f0dbf
Packit 5f0dbf
      if ((cnt == 1 && shdr->sh_size != sizeof (expecteddata))
Packit 5f0dbf
	  || (cnt == 2 && shdr->sh_size != 17))
Packit 5f0dbf
	{
Packit 5f0dbf
	  printf ("section %zd's size differs\n", cnt);
Packit 5f0dbf
	  result = 1;
Packit 5f0dbf
	}
Packit 5f0dbf
Packit 5f0dbf
      if (shdr->sh_link != 0)
Packit 5f0dbf
	{
Packit 5f0dbf
	  printf ("section %zd's link differs\n", cnt);
Packit 5f0dbf
	  result = 1;
Packit 5f0dbf
	}
Packit 5f0dbf
Packit 5f0dbf
      if (shdr->sh_info != 0)
Packit 5f0dbf
	{
Packit 5f0dbf
	  printf ("section %zd's info differs\n", cnt);
Packit 5f0dbf
	  result = 1;
Packit 5f0dbf
	}
Packit 5f0dbf
Packit 5f0dbf
      if ((cnt == 1 && shdr->sh_addralign != 16)
Packit 5f0dbf
	  || (cnt != 1 && shdr->sh_addralign != 1))
Packit 5f0dbf
	{
Packit 5f0dbf
	  printf ("section %zd's addralign differs\n", cnt);
Packit 5f0dbf
	  result = 1;
Packit 5f0dbf
	}
Packit 5f0dbf
Packit 5f0dbf
      if (shdr->sh_entsize != 0)
Packit 5f0dbf
	{
Packit 5f0dbf
	  printf ("section %zd's entsize differs\n", cnt);
Packit 5f0dbf
	  result = 1;
Packit 5f0dbf
	}
Packit 5f0dbf
Packit 5f0dbf
      if (cnt == 1)
Packit 5f0dbf
	{
Packit 5f0dbf
	  Elf_Data *data = elf_getdata (escn, NULL);
Packit 5f0dbf
Packit 5f0dbf
	  if (data == NULL)
Packit 5f0dbf
	    {
Packit 5f0dbf
	      printf ("cannot get data of section %zd\n", cnt);
Packit 5f0dbf
	      result = 1;
Packit 5f0dbf
	    }
Packit 5f0dbf
	  else
Packit 5f0dbf
	    {
Packit 5f0dbf
	      if (data->d_size != sizeof (expecteddata))
Packit 5f0dbf
		{
Packit 5f0dbf
		  printf ("data block size of section %zd wrong: got %zd, "
Packit 5f0dbf
			  "expected 96\n", cnt, data->d_size);
Packit 5f0dbf
		  result = 1;
Packit 5f0dbf
		}
Packit 5f0dbf
Packit 5f0dbf
	      if (memcmp (data->d_buf, expecteddata, sizeof (expecteddata))
Packit 5f0dbf
		  != 0)
Packit 5f0dbf
		{
Packit 5f0dbf
		  printf ("data block content of section %zd wrong\n", cnt);
Packit 5f0dbf
		  result = 1;
Packit 5f0dbf
		}
Packit 5f0dbf
	    }
Packit 5f0dbf
	}
Packit 5f0dbf
    }
Packit 5f0dbf
Packit 5f0dbf
 out_close2:
Packit 5f0dbf
  elf_end (elf);
Packit 5f0dbf
 out_close:
Packit 5f0dbf
  close (fd);
Packit 5f0dbf
 out:
Packit 5f0dbf
  /* We don't need the file anymore.  */
Packit 5f0dbf
  unlink (fname);
Packit 5f0dbf
Packit 5f0dbf
  ebl_closebackend (ebl);
Packit 5f0dbf
Packit 5f0dbf
  return result;
Packit 5f0dbf
}