Blame tests/next_cfi.c

Packit Service 97d2fb
/* Test program for dwarf_next_cfi
Packit Service 97d2fb
   Copyright (C) 2018 Red Hat, Inc.
Packit Service 97d2fb
   This file is part of elfutils.
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
#include <config.h>
Packit Service 97d2fb
#include <assert.h>
Packit Service 97d2fb
#include <inttypes.h>
Packit Service 97d2fb
#include ELFUTILS_HEADER(dw)
Packit Service 97d2fb
#include <dwarf.h>
Packit Service 97d2fb
#include <argp.h>
Packit Service 97d2fb
#include <stdbool.h>
Packit Service 97d2fb
#include <stdio.h>
Packit Service 97d2fb
#include <sys/types.h>
Packit Service 97d2fb
#include <sys/stat.h>
Packit Service 97d2fb
#include <fcntl.h>
Packit Service 97d2fb
#include <locale.h>
Packit Service 97d2fb
#include <stdlib.h>
Packit Service 97d2fb
#include <string.h>
Packit Service 97d2fb
#include <unistd.h>
Packit Service 97d2fb
#include "system.h"
Packit Service 97d2fb
Packit Service 97d2fb
void
Packit Service 97d2fb
handle_section (char *name, const unsigned char e_ident[],
Packit Service 97d2fb
		Elf_Scn *scn, const bool is_eh)
Packit Service 97d2fb
{
Packit Service 97d2fb
  if (is_eh)
Packit Service 97d2fb
    printf (".eh_frame\n");
Packit Service 97d2fb
  else
Packit Service 97d2fb
    printf (".debug_frame\n");
Packit Service 97d2fb
Packit Service 97d2fb
  GElf_Shdr mem;
Packit Service 97d2fb
  GElf_Shdr *shdr = gelf_getshdr (scn, &mem;;
Packit Service 97d2fb
  if (shdr == NULL)
Packit Service 97d2fb
    error (EXIT_FAILURE, 0, "Couldn't get section header: %s",
Packit Service 97d2fb
	   elf_errmsg (-1));
Packit Service 97d2fb
  if ((shdr->sh_flags & SHF_COMPRESSED) != 0)
Packit Service 97d2fb
    {
Packit Service 97d2fb
      if (elf_compress (scn, 0, 0) < 0)
Packit Service 97d2fb
	error (EXIT_FAILURE, 0, "Couldn't decompress section: %s",
Packit Service 97d2fb
	       elf_errmsg (-1));
Packit Service 97d2fb
    }
Packit Service 97d2fb
  else if (name[0] == '.' && name[1] == 'z')
Packit Service 97d2fb
    {
Packit Service 97d2fb
      if (elf_compress_gnu (scn, 0, 0) < 0)
Packit Service 97d2fb
	error (EXIT_FAILURE, 0, "Couldn't decompress section: %s",
Packit Service 97d2fb
	       elf_errmsg (-1));
Packit Service 97d2fb
    }
Packit Service 97d2fb
Packit Service 97d2fb
  Elf_Data *data = elf_getdata (scn, NULL);
Packit Service 97d2fb
  if (data == NULL || data->d_buf == NULL)
Packit Service 97d2fb
    error (EXIT_FAILURE, 0, "no section data");
Packit Service 97d2fb
Packit Service 97d2fb
  int res;
Packit Service 97d2fb
  Dwarf_Off off;
Packit Service 97d2fb
  Dwarf_Off next_off = 0;
Packit Service 97d2fb
  Dwarf_CFI_Entry entry;
Packit Service 97d2fb
  while ((res = dwarf_next_cfi (e_ident, data, is_eh, off = next_off,
Packit Service 97d2fb
				&next_off, &entry)) == 0)
Packit Service 97d2fb
    {
Packit Service 97d2fb
      printf ("[%" PRId64 "] ", off);
Packit Service 97d2fb
      if (dwarf_cfi_cie_p (&entry))
Packit Service 97d2fb
	printf ("CIE augmentation=\"%s\"\n", entry.cie.augmentation);
Packit Service 97d2fb
      else
Packit Service 97d2fb
	{
Packit Service 97d2fb
	  printf ("FDE cie=[%" PRId64 "]\n", entry.fde.CIE_pointer);
Packit Service 97d2fb
Packit Service 97d2fb
	  Dwarf_Off cie_off = entry.fde.CIE_pointer;
Packit Service 97d2fb
	  Dwarf_Off cie_off_next;
Packit Service 97d2fb
	  Dwarf_CFI_Entry cie_entry;
Packit Service 97d2fb
	  if (dwarf_next_cfi (e_ident, data, is_eh, cie_off, &cie_off_next,
Packit Service 97d2fb
			      &cie_entry) != 0
Packit Service 97d2fb
	      || !dwarf_cfi_cie_p (&cie_entry))
Packit Service 97d2fb
	    error (EXIT_FAILURE, 0, "FDE doesn't point to CIE");
Packit Service 97d2fb
	}
Packit Service 97d2fb
    }
Packit Service 97d2fb
Packit Service 97d2fb
  if (res < 0)
Packit Service 97d2fb
    error (EXIT_FAILURE, 0, "dwarf_next_cfi failed: %s\n",
Packit Service 97d2fb
	   dwarf_errmsg (-1));
Packit Service 97d2fb
}
Packit Service 97d2fb
Packit Service 97d2fb
int
Packit Service 97d2fb
main (int argc, char *argv[])
Packit Service 97d2fb
{
Packit Service 97d2fb
  if (argc != 2)
Packit Service 97d2fb
    error (EXIT_FAILURE, 0, "need file name argument");
Packit Service 97d2fb
Packit Service 97d2fb
  const char *file = argv[1];
Packit Service 97d2fb
  printf ("%s\n", file);
Packit Service 97d2fb
Packit Service 97d2fb
  int fd = open (file, O_RDONLY);
Packit Service 97d2fb
  if (fd == -1)
Packit Service 97d2fb
    error (EXIT_FAILURE, errno, "cannot open input file `%s'", file);
Packit Service 97d2fb
Packit Service 97d2fb
  elf_version (EV_CURRENT);
Packit Service 97d2fb
Packit Service 97d2fb
  Elf *elf = elf_begin (fd, ELF_C_READ, NULL);
Packit Service 97d2fb
  if (elf == NULL)
Packit Service 97d2fb
    error (EXIT_FAILURE, 0, "cannot create ELF descriptor: %s",
Packit Service 97d2fb
	   elf_errmsg (-1));
Packit Service 97d2fb
Packit Service 97d2fb
  size_t esize;
Packit Service 97d2fb
  const unsigned char *ident = (const unsigned char *) elf_getident (elf,
Packit Service 97d2fb
								     &esize);
Packit Service 97d2fb
  if (ident == NULL || esize < EI_NIDENT)
Packit Service 97d2fb
    error (EXIT_FAILURE, 0, "no, or too small, ELF ident");
Packit Service 97d2fb
Packit Service 97d2fb
  GElf_Ehdr ehdr;
Packit Service 97d2fb
  if (gelf_getehdr (elf, &ehdr) == NULL)
Packit Service 97d2fb
    error (EXIT_FAILURE, 0, "cannot get the ELF header: %s\n",
Packit Service 97d2fb
	   elf_errmsg (-1));
Packit Service 97d2fb
Packit Service 97d2fb
  size_t strndx = ehdr.e_shstrndx;
Packit Service 97d2fb
Packit Service 97d2fb
  Elf_Scn *scn = NULL;
Packit Service 97d2fb
  while ((scn = elf_nextscn (elf, scn)) != NULL)
Packit Service 97d2fb
    {
Packit Service 97d2fb
      GElf_Shdr shdr;
Packit Service 97d2fb
      if (gelf_getshdr (scn, &shdr) != NULL)
Packit Service 97d2fb
	{
Packit Service 97d2fb
	  char *name = elf_strptr (elf, strndx, (size_t) shdr.sh_name);
Packit Service 97d2fb
	  if (name != NULL && shdr.sh_type == SHT_PROGBITS)
Packit Service 97d2fb
	    {
Packit Service 97d2fb
	      if (strcmp (name, ".eh_frame") == 0)
Packit Service 97d2fb
		handle_section (name, ident, scn, true);
Packit Service 97d2fb
	      if (strcmp (name, ".debug_frame") == 0
Packit Service 97d2fb
		  || strcmp (name, ".zdebug_frame") == 0)
Packit Service 97d2fb
		handle_section (name, ident, scn, false);
Packit Service 97d2fb
	    }
Packit Service 97d2fb
	}
Packit Service 97d2fb
    }
Packit Service 97d2fb
Packit Service 97d2fb
  elf_end (elf);
Packit Service 97d2fb
  close (fd);
Packit Service 97d2fb
Packit Service 97d2fb
  return 0;
Packit Service 97d2fb
}