/* Test dwarf_cu_info properties.
   Copyright (C) 2018 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 .  */
#ifdef HAVE_CONFIG_H
# include 
#endif
#include 
#include ELFUTILS_HEADER(dw)
#include 
#include 
#include 
#include 
#include 
#include 
/* Yeah, lazy, 16K CUs should be enough for everybody... */
#define MAX_UNITS 16384
struct info
{
  int dietag;
  int subtag;
  Dwarf_Half version;
  uint8_t unit_type;
  uint64_t id;
  uint8_t addr_size;
  uint8_t off_size;
};
static struct info unit_info[MAX_UNITS];
int
main (int argc, char *argv[])
{
  for (int i = 1; i < argc; i++)
    {
      printf ("file: %s\n", argv[i]);
      int fd = open (argv[i], O_RDONLY);
      Dwarf *dbg = dwarf_begin (fd, DWARF_C_READ);
      if (dbg == NULL)
	{
	  printf ("%s not usable: %s\n", argv[i], dwarf_errmsg (-1));
	  return -1;
	}
      Dwarf_CU *cu = NULL;
      Dwarf_Half version;
      Dwarf_Die cudie, subdie;
      uint8_t unit_type;
      size_t u, units;
      u = units = 0;
      printf ("Iterate getting all info, compare with dwarf_cu_info.\n");
      while (dwarf_get_units (dbg, cu, &cu, &version,
			      &unit_type, &cudie, &subdie) == 0)
	{
	  int dietag = dwarf_tag (&cudie);
	  int subtag = dwarf_tag (&subdie);
	  unit_info[u].dietag = dietag;
	  unit_info[u].subtag = subtag;
	  unit_info[u].version = version;
	  unit_info[u].unit_type = unit_type;
	  printf ("%zu cu dietag: %x, subtag: %x, version %" PRIx32
		  ", unit_type %" PRIx8 "\n",
		  u, dietag, subtag, version, unit_type);
	  uint64_t unit_id;
	  uint8_t addr_size, off_size;
	  if (dwarf_cu_info (cu,
			     &version, &unit_type, &cudie, &subdie,
			     &unit_id, &addr_size, &off_size) != 0)
	    {
	      printf ("Invalid dwarf_cu_info: %s\n", dwarf_errmsg (-1));
	      return -1;
	    }
	  dietag = dwarf_tag (&cudie);
	  subtag = dwarf_tag (&subdie);
	  if (unit_info[u].dietag != dietag)
	    {
	      printf("Unequal dietags\n");
	      return -1;
	    }
	  if (unit_info[u].subtag != subtag)
	    {
	      printf("Unequal subtags\n");
	      return -1;
	    }
	  if (unit_info[u].version != version)
	    {
	      printf("Unequal versions\n");
	      return -1;
	    }
	  if (unit_info[u].unit_type != unit_type)
	    {
	      printf("Unequal unit_types\n");
	      return -1;
	    }
	  unit_info[u].id = unit_id;
	  unit_info[u].addr_size = addr_size;
	  unit_info[u].off_size = off_size;
	  if (unit_type == DW_UT_skeleton)
	    {
	      if (dwarf_cu_info (subdie.cu,
				 &version, &unit_type, &cudie, &subdie,
				 &unit_id, &addr_size, &off_size) != 0)
		{
		  printf ("Invalid subdie dwarf_cu_info: %s\n",
			  dwarf_errmsg (-1));
		  return -1;
		}
	      dietag = dwarf_tag (&cudie);
	      subtag = dwarf_tag (&subdie);
	      printf ("%zu subdietag: %x, subtag: %x, version %" PRIx32
		      ", unit_type %" PRIx8 "\n",
		      u, dietag, subtag, version, unit_type);
	      /* subdie is now cudie.  */
	      if (unit_info[u].subtag != dietag)
	      {
		printf ("Inconsistent subdie tag\n");
		return -1;
	      }
	      if (unit_info[u].id != unit_id)
		{
		  printf ("Unequal subdie ids\n");
		  return -1;
		}
	      if (unit_info[u].addr_size != addr_size)
		{
		  printf ("Unequal subdie addr_size\n");
		  return -1;
		}
	      if (unit_info[u].off_size != off_size)
		{
		  printf ("Unequal subdie off_size\n");
		  return -1;
		}
	    }
	  if (u >= MAX_UNITS)
	    {
	      printf ("Oops, more than 16K units...\n");
	      return -1;
	    }
	  u = ++units;
	}
      dwarf_end (dbg);
      close (fd);
      /* And again... */
      printf ("rechecking: %s\n", argv[i]);
      fd = open (argv[i], O_RDONLY);
      dbg = dwarf_begin (fd, DWARF_C_READ);
      if (dbg == NULL)
	{
	  printf ("%s not usable: %s\n", argv[i], dwarf_errmsg (-1));
	  return -1;
	}
      cu = NULL;
      u = 0;
      printf ("Iterate no info, compare recorded info with dwarf_cu_info.\n");
      while (dwarf_get_units (dbg, cu, &cu, NULL, NULL, NULL, NULL) == 0)
	{
	  if (u > units)
	    {
	      printf ("Got too many units???\n");
	      return -1;
	    }
	  uint64_t unit_id;
	  uint8_t addr_size, off_size;
	  if (dwarf_cu_info (cu,
			     &version, &unit_type, &cudie, &subdie,
			     &unit_id, &addr_size, &off_size) != 0)
	    {
	      printf ("Invalid dwarf_cu_info: %s\n", dwarf_errmsg (-1));
	      return -1;
	    }
	  int dietag = dwarf_tag (&cudie);
	  int subtag = dwarf_tag (&subdie);
	  printf ("%zu re dietag: %x, subtag: %x, version %" PRIx32
		  ", unit_type %" PRIx8 "\n",
		  u, dietag, subtag, version, unit_type);
	  if (unit_info[u].dietag != dietag)
	    {
	      printf("Unequal dietags %x != %x\n", unit_info[u].dietag, dietag);
	      return -1;
	    }
	  if (unit_info[u].subtag != subtag)
	    {
	      printf("Unequal subtags\n");
	      return -1;
	    }
	  if (unit_info[u].version != version)
	    {
	      printf("Unequal versions\n");
	      return -1;
	    }
	  if (unit_info[u].unit_type != unit_type)
	    {
	      printf("Unequal unit_types\n");
	      return -1;
	    }
	  if (unit_info[u].id != unit_id)
	    {
	      printf ("Unequal subdie ids\n");
	      return -1;
	    }
	  if (unit_info[u].addr_size != addr_size)
	    {
	      printf ("Unequal subdie addr_size\n");
	      return -1;
	    }
	  if (unit_info[u].off_size != off_size)
	    {
	      printf ("Unequal subdie off_size\n");
	      return -1;
	    }
	  if (unit_type == DW_UT_skeleton)
	    {
	      if (dwarf_cu_info (subdie.cu,
				 &version, &unit_type, &cudie, &subdie,
				 &unit_id, &addr_size, &off_size) != 0)
		{
		  printf ("Invalid subdie dwarf_cu_info: %s\n",
			  dwarf_errmsg (-1));
		  return -1;
		}
	      dietag = dwarf_tag (&cudie);
	      subtag = dwarf_tag (&subdie);
	      printf ("%zu subdietag: %x, subtag: %x, version %" PRIx32
		      ", unit_type %" PRIx8 "\n",
		      u, dietag, subtag, version, unit_type);
	      /* subdie is now cudie.  */
	      subtag = dwarf_tag (&cudie);
	      if (unit_info[u].subtag != subtag)
	      {
		printf ("Inconsistent subdie tag\n");
		return -1;
	      }
	      if (unit_info[u].id != unit_id)
		{
		  printf ("Unequal subdie ids\n");
		  return -1;
		}
	      if (unit_info[u].addr_size != addr_size)
		{
		  printf ("Unequal subdie addr_size\n");
		  return -1;
		}
	      if (unit_info[u].off_size != off_size)
		{
		  printf ("Unequal subdie off_size\n");
		  return -1;
		}
	    }
	  if (u >= MAX_UNITS)
	    {
	      printf ("Oops, more than 16K units...\n");
	      return -1;
	    }
	  u++;
	}
      if (u != units)
	{
	  printf ("Got not enough units???\n");
	  return -1;
	}
      dwarf_end (dbg);
      close (fd);
      printf ("\n");
    }
  return 0;
}