Blame libelf/elf_compress_gnu.c

Packit Service 97d2fb
/* Compress or decompress a section.
Packit Service 97d2fb
   Copyright (C) 2015 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 either
Packit Service 97d2fb
Packit Service 97d2fb
     * the GNU Lesser General Public License as published by the Free
Packit Service 97d2fb
       Software Foundation; either version 3 of the License, or (at
Packit Service 97d2fb
       your option) any later version
Packit Service 97d2fb
Packit Service 97d2fb
   or
Packit Service 97d2fb
Packit Service 97d2fb
     * the GNU General Public License as published by the Free
Packit Service 97d2fb
       Software Foundation; either version 2 of the License, or (at
Packit Service 97d2fb
       your option) any later version
Packit Service 97d2fb
Packit Service 97d2fb
   or both in parallel, as here.
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 GNU
Packit Service 97d2fb
   General Public License for more details.
Packit Service 97d2fb
Packit Service 97d2fb
   You should have received copies of the GNU General Public License and
Packit Service 97d2fb
   the GNU Lesser General Public License along with this program.  If
Packit Service 97d2fb
   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 <libelf.h>
Packit Service 97d2fb
#include "libelfP.h"
Packit Service 97d2fb
#include "common.h"
Packit Service 97d2fb
Packit Service 97d2fb
int
Packit Service 97d2fb
elf_compress_gnu (Elf_Scn *scn, int inflate, unsigned int flags)
Packit Service 97d2fb
{
Packit Service 97d2fb
  if (scn == NULL)
Packit Service 97d2fb
    return -1;
Packit Service 97d2fb
Packit Service 97d2fb
  if ((flags & ~ELF_CHF_FORCE) != 0)
Packit Service 97d2fb
    {
Packit Service 97d2fb
      __libelf_seterrno (ELF_E_INVALID_OPERAND);
Packit Service 97d2fb
      return -1;
Packit Service 97d2fb
    }
Packit Service 97d2fb
Packit Service 97d2fb
  bool force = (flags & ELF_CHF_FORCE) != 0;
Packit Service 97d2fb
Packit Service 97d2fb
  Elf *elf = scn->elf;
Packit Service 97d2fb
  GElf_Ehdr ehdr;
Packit Service 97d2fb
  if (gelf_getehdr (elf, &ehdr) == NULL)
Packit Service 97d2fb
    return -1;
Packit Service 97d2fb
Packit Service 97d2fb
  int elfclass = elf->class;
Packit Service 97d2fb
  int elfdata = ehdr.e_ident[EI_DATA];
Packit Service 97d2fb
Packit Service 97d2fb
  Elf64_Xword sh_flags;
Packit Service 97d2fb
  Elf64_Word sh_type;
Packit Service 97d2fb
  Elf64_Xword sh_addralign;
Packit Service 97d2fb
  if (elfclass == ELFCLASS32)
Packit Service 97d2fb
    {
Packit Service 97d2fb
      Elf32_Shdr *shdr = elf32_getshdr (scn);
Packit Service 97d2fb
      if (shdr == NULL)
Packit Service 97d2fb
	return -1;
Packit Service 97d2fb
Packit Service 97d2fb
      sh_flags = shdr->sh_flags;
Packit Service 97d2fb
      sh_type = shdr->sh_type;
Packit Service 97d2fb
      sh_addralign = shdr->sh_addralign;
Packit Service 97d2fb
    }
Packit Service 97d2fb
  else
Packit Service 97d2fb
    {
Packit Service 97d2fb
      Elf64_Shdr *shdr = elf64_getshdr (scn);
Packit Service 97d2fb
      if (shdr == NULL)
Packit Service 97d2fb
	return -1;
Packit Service 97d2fb
Packit Service 97d2fb
      sh_flags = shdr->sh_flags;
Packit Service 97d2fb
      sh_type = shdr->sh_type;
Packit Service 97d2fb
      sh_addralign = shdr->sh_addralign;
Packit Service 97d2fb
    }
Packit Service 97d2fb
Packit Service 97d2fb
  /* Allocated sections, or sections that are already are compressed
Packit Service 97d2fb
     cannot (also) be GNU compressed.  */
Packit Service 97d2fb
  if ((sh_flags & SHF_ALLOC) != 0 || (sh_flags & SHF_COMPRESSED))
Packit Service 97d2fb
    {
Packit Service 97d2fb
      __libelf_seterrno (ELF_E_INVALID_SECTION_FLAGS);
Packit Service 97d2fb
      return -1;
Packit Service 97d2fb
    }
Packit Service 97d2fb
Packit Service 97d2fb
  if (sh_type == SHT_NULL || sh_type == SHT_NOBITS)
Packit Service 97d2fb
    {
Packit Service 97d2fb
      __libelf_seterrno (ELF_E_INVALID_SECTION_TYPE);
Packit Service 97d2fb
      return -1;
Packit Service 97d2fb
    }
Packit Service 97d2fb
Packit Service 97d2fb
  /* For GNU compression we cannot really know whether the section is
Packit Service 97d2fb
     already compressed or not.  Just try and see what happens...  */
Packit Service 97d2fb
  // int compressed = (sh_flags & SHF_COMPRESSED);
Packit Service 97d2fb
  if (inflate == 1)
Packit Service 97d2fb
    {
Packit Service 97d2fb
      size_t hsize = 4 + 8; /* GNU "ZLIB" + 8 byte size.  */
Packit Service 97d2fb
      size_t orig_size, new_size, orig_addralign;
Packit Service 97d2fb
      void *out_buf = __libelf_compress (scn, hsize, elfdata,
Packit Service 97d2fb
					 &orig_size, &orig_addralign,
Packit Service 97d2fb
					 &new_size, force);
Packit Service 97d2fb
Packit Service 97d2fb
      /* Compression would make section larger, don't change anything.  */
Packit Service 97d2fb
      if (out_buf == (void *) -1)
Packit Service 97d2fb
	return 0;
Packit Service 97d2fb
Packit Service 97d2fb
      /* Compression failed, return error.  */
Packit Service 97d2fb
      if (out_buf == NULL)
Packit Service 97d2fb
	return -1;
Packit Service 97d2fb
Packit Service 97d2fb
      uint64_t be64_size = htobe64 (orig_size);
Packit Service 97d2fb
      memmove (out_buf, "ZLIB", 4);
Packit Service 97d2fb
      memmove (out_buf + 4, &be64_size, sizeof (be64_size));
Packit Service 97d2fb
Packit Service 97d2fb
      /* We don't know anything about sh_entsize, sh_addralign and
Packit Service 97d2fb
	 sh_flags won't have a SHF_COMPRESSED hint in the GNU format.
Packit Service 97d2fb
	 Just adjust the sh_size.  */
Packit Service 97d2fb
      if (elfclass == ELFCLASS32)
Packit Service 97d2fb
	{
Packit Service 97d2fb
	  Elf32_Shdr *shdr = elf32_getshdr (scn);
Packit Service 97d2fb
	  shdr->sh_size = new_size;
Packit Service 97d2fb
	}
Packit Service 97d2fb
      else
Packit Service 97d2fb
	{
Packit Service 97d2fb
	  Elf64_Shdr *shdr = elf64_getshdr (scn);
Packit Service 97d2fb
	  shdr->sh_size = new_size;
Packit Service 97d2fb
	}
Packit Service 97d2fb
Packit Service 97d2fb
      __libelf_reset_rawdata (scn, out_buf, new_size, 1, ELF_T_BYTE);
Packit Service 97d2fb
Packit Service 97d2fb
      /* The section is now compressed, we could keep the uncompressed
Packit Service 97d2fb
	 data around, but since that might have been multiple Elf_Data
Packit Service 97d2fb
	 buffers let the user uncompress it explicitly again if they
Packit Service 97d2fb
	 want it to simplify bookkeeping.  */
Packit Service 97d2fb
      scn->zdata_base = NULL;
Packit Service 97d2fb
Packit Service 97d2fb
      return 1;
Packit Service 97d2fb
    }
Packit Service 97d2fb
  else if (inflate == 0)
Packit Service 97d2fb
    {
Packit Service 97d2fb
      /* In theory the user could have constucted a compressed section
Packit Service 97d2fb
	 by hand.  And in practice they do. For example when copying
Packit Service 97d2fb
	 a section from one file to another using elf_newdata. So we
Packit Service 97d2fb
	 have to use elf_getdata (not elf_rawdata).  */
Packit Service 97d2fb
      Elf_Data *data = elf_getdata (scn, NULL);
Packit Service 97d2fb
      if (data == NULL)
Packit Service 97d2fb
	return -1;
Packit Service 97d2fb
Packit Service 97d2fb
      size_t hsize = 4 + 8; /* GNU "ZLIB" + 8 byte size.  */
Packit Service 97d2fb
      if (data->d_size < hsize || memcmp (data->d_buf, "ZLIB", 4) != 0)
Packit Service 97d2fb
	{
Packit Service 97d2fb
          __libelf_seterrno (ELF_E_NOT_COMPRESSED);
Packit Service 97d2fb
	  return -1;
Packit Service 97d2fb
	}
Packit Service 97d2fb
Packit Service 97d2fb
      /* There is a 12-byte header of "ZLIB" followed by
Packit Service 97d2fb
	 an 8-byte big-endian size.  There is only one type and
Packit Service 97d2fb
	 Alignment isn't preserved separately.  */
Packit Service 97d2fb
      uint64_t gsize;
Packit Service 97d2fb
      memcpy (&gsize, data->d_buf + 4, sizeof gsize);
Packit Service 97d2fb
      gsize = be64toh (gsize);
Packit Service 97d2fb
Packit Service 97d2fb
      /* One more sanity check, size should be bigger than original
Packit Service 97d2fb
	 data size plus some overhead (4 chars ZLIB + 8 bytes size + 6
Packit Service 97d2fb
	 bytes zlib stream overhead + 5 bytes overhead max for one 16K
Packit Service 97d2fb
	 block) and should fit into a size_t.  */
Packit Service 97d2fb
      if (gsize + 4 + 8 + 6 + 5 < data->d_size || gsize > SIZE_MAX)
Packit Service 97d2fb
	{
Packit Service 97d2fb
	  __libelf_seterrno (ELF_E_NOT_COMPRESSED);
Packit Service 97d2fb
	  return -1;
Packit Service 97d2fb
	}
Packit Service 97d2fb
Packit Service 97d2fb
      size_t size = gsize;
Packit Service 97d2fb
      size_t size_in = data->d_size - hsize;
Packit Service 97d2fb
      void *buf_in = data->d_buf + hsize;
Packit Service 97d2fb
      void *buf_out = __libelf_decompress (buf_in, size_in, size);
Packit Service 97d2fb
      if (buf_out == NULL)
Packit Service 97d2fb
	return -1;
Packit Service 97d2fb
Packit Service 97d2fb
      /* We don't know anything about sh_entsize, sh_addralign and
Packit Service 97d2fb
	 sh_flags won't have a SHF_COMPRESSED hint in the GNU format.
Packit Service 97d2fb
	 Just adjust the sh_size.  */
Packit Service 97d2fb
      if (elfclass == ELFCLASS32)
Packit Service 97d2fb
	{
Packit Service 97d2fb
	  Elf32_Shdr *shdr = elf32_getshdr (scn);
Packit Service 97d2fb
	  shdr->sh_size = size;
Packit Service 97d2fb
	}
Packit Service 97d2fb
      else
Packit Service 97d2fb
	{
Packit Service 97d2fb
	  Elf64_Shdr *shdr = elf64_getshdr (scn);
Packit Service 97d2fb
	  shdr->sh_size = size;
Packit Service 97d2fb
	}
Packit Service 97d2fb
Packit Service 97d2fb
      __libelf_reset_rawdata (scn, buf_out, size, sh_addralign,
Packit Service 97d2fb
			      __libelf_data_type (elf, sh_type, sh_addralign));
Packit Service 97d2fb
Packit Service 97d2fb
      scn->zdata_base = buf_out;
Packit Service 97d2fb
Packit Service 97d2fb
      return 1;
Packit Service 97d2fb
    }
Packit Service 97d2fb
  else
Packit Service 97d2fb
    {
Packit Service 97d2fb
      __libelf_seterrno (ELF_E_UNKNOWN_COMPRESSION_TYPE);
Packit Service 97d2fb
      return -1;
Packit Service 97d2fb
    }
Packit Service 97d2fb
}