Blame tests/elfputzdata.c

Packit 032894
/* Copyright (C) 2015 Red Hat, Inc.
Packit 032894
   This file is part of elfutils.
Packit 032894
Packit 032894
   This file is free software; you can redistribute it and/or modify
Packit 032894
   it under the terms of the GNU General Public License as published by
Packit 032894
   the Free Software Foundation; either version 3 of the License, or
Packit 032894
   (at your option) any later version.
Packit 032894
Packit 032894
   elfutils is distributed in the hope that it will be useful, but
Packit 032894
   WITHOUT ANY WARRANTY; without even the implied warranty of
Packit 032894
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
Packit 032894
   GNU General Public License for more details.
Packit 032894
Packit 032894
   You should have received a copy of the GNU General Public License
Packit 032894
   along with this program.  If not, see <http://www.gnu.org/licenses/>.  */
Packit 032894
Packit 032894
#ifdef HAVE_CONFIG_H
Packit 032894
# include <config.h>
Packit 032894
#endif
Packit 032894
Packit 032894
#include <sys/types.h>
Packit 032894
#include <sys/stat.h>
Packit 032894
#include <fcntl.h>
Packit 032894
#include <inttypes.h>
Packit 032894
#include <libelf.h>
Packit 032894
#include <gelf.h>
Packit 032894
#include <stdbool.h>
Packit 032894
#include <stdio.h>
Packit 032894
#include <stdlib.h>
Packit 032894
#include <string.h>
Packit 032894
#include <unistd.h>
Packit 032894
Packit 032894
Packit 032894
int
Packit 032894
main (int argc, char *argv[])
Packit 032894
{
Packit 032894
  int result = 0;
Packit 032894
  int cnt;
Packit 032894
Packit 032894
  if (argc < 3
Packit 032894
      || (strcmp (argv[1], "elf") != 0
Packit 032894
	  && strcmp (argv[1], "gnu") != 0))
Packit 032894
    {
Packit 032894
      printf ("Usage: (elf|gnu) files...\n");
Packit 032894
      return -1;
Packit 032894
    }
Packit 032894
Packit 032894
  int gnu;
Packit 032894
  if (strcmp (argv[1], "gnu") == 0)
Packit 032894
    gnu = 1;
Packit 032894
  else
Packit 032894
    gnu = 0;
Packit 032894
Packit 032894
  elf_version (EV_CURRENT);
Packit 032894
Packit 032894
  for (cnt = 2; cnt < argc; ++cnt)
Packit 032894
    {
Packit 032894
      int fd = open (argv[cnt], O_RDONLY);
Packit 032894
Packit 032894
      Elf *elf = elf_begin (fd, ELF_C_READ, NULL);
Packit 032894
      if (elf == NULL)
Packit 032894
	{
Packit 032894
	  printf ("%s not usable %s\n", argv[cnt], elf_errmsg (-1));
Packit 032894
	  result = 1;
Packit 032894
	  close (fd);
Packit 032894
	  continue;
Packit 032894
	}
Packit 032894
Packit 032894
      /* To get the section names.  */
Packit 032894
      size_t strndx;
Packit 032894
      elf_getshdrstrndx (elf, &strndx);
Packit 032894
Packit 032894
      Elf_Scn *scn = NULL;
Packit 032894
      while ((scn = elf_nextscn (elf, scn)) != NULL)
Packit 032894
	{
Packit 032894
	  size_t idx = elf_ndxscn (scn);
Packit 032894
	  GElf_Shdr mem;
Packit 032894
	  GElf_Shdr *shdr = gelf_getshdr (scn, &mem;;
Packit 032894
	  const char *name = elf_strptr (elf, strndx, shdr->sh_name);
Packit 032894
	  if (shdr->sh_type == SHT_NOBITS
Packit 032894
	      || (shdr->sh_flags & SHF_ALLOC) != 0)
Packit 032894
	    {
Packit 032894
	      printf ("Cannot compress %zd %s\n", idx, name);
Packit 032894
	    }
Packit 032894
	  else if ((shdr->sh_flags & SHF_COMPRESSED) != 0
Packit 032894
		   || strncmp (name, ".zdebug", strlen (".zdebug")) == 0)
Packit 032894
	    {
Packit 032894
	      printf ("Already compressed %zd %s\n", idx, name);
Packit 032894
	    }
Packit 032894
	  else
Packit 032894
	    {
Packit 032894
	      size_t orig_size = shdr->sh_size;
Packit 032894
	      printf ("Lets compress %zd %s, size: %" PRId64 "\n",
Packit 032894
		      idx, name, shdr->sh_size);
Packit 032894
	      Elf_Data *d = elf_getdata (scn, NULL);
Packit 032894
	      if (d == NULL)
Packit 032894
		{
Packit 032894
		  printf ("Couldn't get orig data for section %zd\n", idx);
Packit 032894
		  return -1;
Packit 032894
		}
Packit 032894
	      /* Make a copy so we can compare after
Packit 032894
		 compression/decompression.  */
Packit 032894
	      if (d->d_size != orig_size)
Packit 032894
		{
Packit 032894
		  printf ("Unexpected data size for orig section %zd\n", idx);
Packit 032894
		  return -1;
Packit 032894
		}
Packit Service 35cfd5
	      char *orig_buf = NULL;
Packit Service 35cfd5
	      if (orig_size > 0)
Packit 032894
		{
Packit Service 35cfd5
		  orig_buf = malloc (d->d_size);
Packit Service 35cfd5
		  if (orig_buf == NULL)
Packit Service 35cfd5
		    {
Packit Service 35cfd5
		      printf ("No memory to copy section %zd data\n", idx);
Packit Service 35cfd5
		      return -1;
Packit Service 35cfd5
		    }
Packit Service 35cfd5
		  memcpy (orig_buf, d->d_buf, orig_size);
Packit 032894
		}
Packit 032894
Packit 032894
	      bool forced = false;
Packit 032894
	      if (gnu)
Packit 032894
		{
Packit 032894
		  int res = elf_compress_gnu (scn, 1, 0);
Packit 032894
		  if (res == 0)
Packit 032894
		    {
Packit 032894
		      forced = true;
Packit 032894
		      res = elf_compress_gnu (scn, 1, ELF_CHF_FORCE);
Packit 032894
		    }
Packit 032894
		  if (res < 0)
Packit 032894
		    {
Packit 032894
		      printf ("elf_compress_gnu%sfailed for section %zd: %s\n",
Packit 032894
			      forced ? " (forced) " : " ",
Packit 032894
			      idx, elf_errmsg (-1));
Packit 032894
		      return -1;
Packit 032894
		    }
Packit 032894
		}
Packit 032894
	      else
Packit 032894
		{
Packit 032894
		  int res = elf_compress (scn, ELFCOMPRESS_ZLIB, 0);
Packit 032894
		  if (res == 0)
Packit 032894
		    {
Packit 032894
		      forced = true;
Packit 032894
		      res = elf_compress (scn, ELFCOMPRESS_ZLIB, ELF_CHF_FORCE);
Packit 032894
		    }
Packit 032894
		  if (res < 0)
Packit 032894
		    {
Packit 032894
		      printf ("elf_compress%sfailed for section %zd: %s\n",
Packit 032894
			      forced ? " (forced) " : " ",
Packit 032894
			      idx, elf_errmsg (-1));
Packit 032894
		      return -1;
Packit 032894
		    }
Packit 032894
		}
Packit 032894
	      GElf_Shdr newmem;
Packit 032894
	      GElf_Shdr *newshdr = gelf_getshdr (scn, &newmem);
Packit 032894
	      size_t new_size = newshdr->sh_size;
Packit 032894
	      d = elf_getdata (scn, NULL);
Packit 032894
	      // Don't check this, might depend on zlib implementation.
Packit 032894
	      // fprintf (stderr, "  new_size: %zd\n", new_size);
Packit 032894
	      if (d->d_size != new_size)
Packit 032894
		{
Packit 032894
		  printf ("Unexpected data size for compressed section %zd\n",
Packit 032894
			  idx);
Packit 032894
		  return -1;
Packit 032894
		}
Packit 032894
Packit 032894
	      if (forced && new_size < orig_size)
Packit 032894
		{
Packit 032894
		  printf ("section %zd forced to compress, but size smaller\n",
Packit 032894
			  idx);
Packit 032894
		  return -1;
Packit 032894
		}
Packit 032894
Packit 032894
	      if (! forced && new_size >= orig_size)
Packit 032894
		{
Packit 032894
		  printf ("section %zd compressed to bigger size\n",
Packit 032894
			  idx);
Packit 032894
		  return -1;
Packit 032894
		}
Packit 032894
Packit 032894
	      if (new_size == orig_size
Packit Service 35cfd5
		  && (orig_buf == NULL
Packit Service 35cfd5
		      || memcmp (orig_buf, d->d_buf, orig_size) == 0))
Packit 032894
		{
Packit 032894
		  printf ("section %zd didn't compress\n", idx);
Packit 032894
		  return -1;
Packit 032894
		}
Packit 032894
Packit 032894
	      if (gnu)
Packit 032894
		{
Packit 032894
		  if (elf_compress_gnu (scn, 0, 0) < 0)
Packit 032894
		    {
Packit 032894
		      printf ("elf_[un]compress_gnu failed for section %zd: %s\n",
Packit 032894
			      idx, elf_errmsg (-1));
Packit 032894
		      return -1;
Packit 032894
		    }
Packit 032894
		}
Packit 032894
	      else
Packit 032894
		{
Packit 032894
		  if (elf_compress (scn, 0, 0) < 0)
Packit 032894
		    {
Packit 032894
		      printf ("elf_[un]compress failed for section %zd: %s\n",
Packit 032894
			      idx, elf_errmsg (-1));
Packit 032894
		      return -1;
Packit 032894
		    }
Packit 032894
		}
Packit 032894
	      GElf_Shdr newermem;
Packit 032894
	      GElf_Shdr *newershdr = gelf_getshdr (scn, &newermem);
Packit 032894
	      size_t newer_size = newershdr->sh_size;
Packit 032894
	      d = elf_getdata (scn, NULL);
Packit 032894
	      // fprintf (stderr, "  newer_size: %zd\n", newer_size);
Packit 032894
	      if (d->d_size != newer_size)
Packit 032894
		{
Packit 032894
		  printf ("Unexpected data size for compressed section %zd\n",
Packit 032894
			  idx);
Packit 032894
		  return -1;
Packit 032894
		}
Packit 032894
	      if (newer_size != orig_size
Packit Service 35cfd5
		  && (orig_buf == NULL
Packit Service 35cfd5
		      || memcmp (orig_buf, d->d_buf, orig_size) != 0))
Packit 032894
		{
Packit 032894
		  printf ("section %zd didn't correctly uncompress\n", idx);
Packit 032894
		  return -1;
Packit 032894
		}
Packit 032894
	      free (orig_buf);
Packit 032894
	      // Recompress the string table, just to make sure
Packit 032894
	      // everything keeps working. See elf_strptr above.
Packit 032894
	      if (! gnu && idx == strndx
Packit 032894
		  && elf_compress (scn, ELFCOMPRESS_ZLIB, 0) < 0)
Packit 032894
		{
Packit 032894
		  printf ("couldn't recompress section header strings: %s\n",
Packit 032894
			  elf_errmsg (-1));
Packit 032894
		  return -1;
Packit 032894
		}
Packit 032894
	    }
Packit 032894
	}
Packit 032894
Packit 032894
      elf_end (elf);
Packit 032894
      close (fd);
Packit 032894
    }
Packit 032894
Packit 032894
  return result;
Packit 032894
}