Blame libelf/elf32_checksum.c

Packit Service 97d2fb
/* Compute simple checksum from permanent parts of the ELF file.
Packit Service 97d2fb
   Copyright (C) 2002, 2003, 2004, 2005, 2009, 2015 Red Hat, Inc.
Packit Service 97d2fb
   This file is part of elfutils.
Packit Service 97d2fb
   Written by Ulrich Drepper <drepper@redhat.com>, 2002.
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 <assert.h>
Packit Service 97d2fb
#include <endian.h>
Packit Service 97d2fb
#include <stdbool.h>
Packit Service 97d2fb
#include <stddef.h>
Packit Service 97d2fb
#include <string.h>
Packit Service 97d2fb
Packit Service 97d2fb
#include "gelf.h"
Packit Service 97d2fb
#include "libelfP.h"
Packit Service 97d2fb
#include "elf-knowledge.h"
Packit Service 97d2fb
Packit Service 97d2fb
#ifndef LIBELFBITS
Packit Service 97d2fb
# define LIBELFBITS 32
Packit Service 97d2fb
#endif
Packit Service 97d2fb
Packit Service 97d2fb
Packit Service 97d2fb
#define process_block(crc, data) \
Packit Service 97d2fb
  __libelf_crc32 (crc, data->d_buf, data->d_size)
Packit Service 97d2fb
Packit Service 97d2fb
Packit Service 97d2fb
long int
Packit Service 97d2fb
elfw2(LIBELFBITS,checksum) (Elf *elf)
Packit Service 97d2fb
{
Packit Service 97d2fb
  size_t shstrndx;
Packit Service 97d2fb
  Elf_Scn *scn;
Packit Service 97d2fb
  long int result = 0;
Packit Service 97d2fb
  unsigned char *ident;
Packit Service 97d2fb
  bool same_byte_order;
Packit Service 97d2fb
Packit Service 97d2fb
  if (elf == NULL)
Packit Service 97d2fb
    return -1l;
Packit Service 97d2fb
Packit Service 97d2fb
  /* Find the section header string table.  */
Packit Service 97d2fb
  if  (INTUSE(elf_getshdrstrndx) (elf, &shstrndx) < 0)
Packit Service 97d2fb
    {
Packit Service 97d2fb
      /* This can only happen if the ELF handle is not for real.  */
Packit Service 97d2fb
      __libelf_seterrno (ELF_E_INVALID_HANDLE);
Packit Service 97d2fb
      return -1l;
Packit Service 97d2fb
    }
Packit Service 97d2fb
Packit Service 97d2fb
  /* Determine whether the byte order of the file and that of the host
Packit Service 97d2fb
     is the same.  */
Packit Service 97d2fb
  ident = elf->state.ELFW(elf,LIBELFBITS).ehdr->e_ident;
Packit Service 97d2fb
  same_byte_order = ((ident[EI_DATA] == ELFDATA2LSB
Packit Service 97d2fb
		      && __BYTE_ORDER == __LITTLE_ENDIAN)
Packit Service 97d2fb
		     || (ident[EI_DATA] == ELFDATA2MSB
Packit Service 97d2fb
			 && __BYTE_ORDER == __BIG_ENDIAN));
Packit Service 97d2fb
Packit Service 97d2fb
  /* If we don't have native byte order, we will likely need to
Packit Service 97d2fb
     convert the data with xlate functions.  We do it upfront instead
Packit Service 97d2fb
     of relocking mid-iteration. */
Packit Service 97d2fb
  if (!likely (same_byte_order))
Packit Service 97d2fb
    rwlock_wrlock (elf->lock);
Packit Service 97d2fb
  else
Packit Service 97d2fb
    rwlock_rdlock (elf->lock);
Packit Service 97d2fb
Packit Service 97d2fb
  /* Iterate over all sections to find those which are not strippable.  */
Packit Service 97d2fb
  scn = NULL;
Packit Service 97d2fb
  while ((scn = INTUSE(elf_nextscn) (elf, scn)) != NULL)
Packit Service 97d2fb
    {
Packit Service 97d2fb
      GElf_Shdr shdr_mem;
Packit Service 97d2fb
      GElf_Shdr *shdr;
Packit Service 97d2fb
      Elf_Data *data;
Packit Service 97d2fb
Packit Service 97d2fb
      /* Get the section header.  */
Packit Service 97d2fb
      shdr = INTUSE(gelf_getshdr) (scn, &shdr_mem);
Packit Service 97d2fb
      if (shdr == NULL)
Packit Service 97d2fb
	{
Packit Service 97d2fb
	  __libelf_seterrno (ELF_E_INVALID_SECTION_HEADER);
Packit Service 97d2fb
	  result = -1l;
Packit Service 97d2fb
	  goto out;
Packit Service 97d2fb
	}
Packit Service 97d2fb
Packit Service 97d2fb
      if (SECTION_STRIP_P (shdr,
Packit Service 97d2fb
			   INTUSE(elf_strptr) (elf, shstrndx, shdr->sh_name),
Packit Service 97d2fb
			   true))
Packit Service 97d2fb
	/* The section can be stripped.  Don't use it.  */
Packit Service 97d2fb
	continue;
Packit Service 97d2fb
Packit Service 97d2fb
      /* Do not look at NOBITS sections.  */
Packit Service 97d2fb
      if (shdr->sh_type == SHT_NOBITS)
Packit Service 97d2fb
	continue;
Packit Service 97d2fb
Packit Service 97d2fb
      /* To compute the checksum we need to get to the data.  For
Packit Service 97d2fb
	 repeatable results we must use the external format.  The data
Packit Service 97d2fb
	 we get with 'elf'getdata' might be changed for endianess
Packit Service 97d2fb
	 reasons.  Therefore we use 'elf_rawdata' if possible.  But
Packit Service 97d2fb
	 this function can fail if the data was constructed by the
Packit Service 97d2fb
	 program.  In this case we have to use 'elf_getdata' and
Packit Service 97d2fb
	 eventually convert the data to the external format.  */
Packit Service 97d2fb
      data = INTUSE(elf_rawdata) (scn, NULL);
Packit Service 97d2fb
      if (data != NULL)
Packit Service 97d2fb
	{
Packit Service 97d2fb
	  /* The raw data is available.  */
Packit Service 97d2fb
	  result = process_block (result, data);
Packit Service 97d2fb
Packit Service 97d2fb
	  /* Maybe the user added more data.  These blocks cannot be
Packit Service 97d2fb
	     read using 'elf_rawdata'.  Simply proceed with looking
Packit Service 97d2fb
	     for more data block with 'elf_getdata'.  */
Packit Service 97d2fb
	}
Packit Service 97d2fb
Packit Service 97d2fb
      /* Iterate through the list of data blocks.  */
Packit Service 97d2fb
      while ((data = INTUSE(elf_getdata) (scn, data)) != NULL)
Packit Service 97d2fb
	/* If the file byte order is the same as the host byte order
Packit Service 97d2fb
	   process the buffer directly.  If the data is just a stream
Packit Service 97d2fb
	   of bytes which the library will not convert we can use it
Packit Service 97d2fb
	   as well.  */
Packit Service 97d2fb
	if (likely (same_byte_order) || data->d_type == ELF_T_BYTE)
Packit Service 97d2fb
	  result = process_block (result, data);
Packit Service 97d2fb
	else
Packit Service 97d2fb
	  {
Packit Service 97d2fb
	    /* Convert the data to file byte order.  */
Packit Service 97d2fb
	    if (INTUSE(elfw2(LIBELFBITS,xlatetof)) (data, data, ident[EI_DATA])
Packit Service 97d2fb
		== NULL)
Packit Service 97d2fb
	      {
Packit Service 97d2fb
		result = -1l;
Packit Service 97d2fb
		goto out;
Packit Service 97d2fb
	      }
Packit Service 97d2fb
Packit Service 97d2fb
	    result = process_block (result, data);
Packit Service 97d2fb
Packit Service 97d2fb
	    /* And convert it back.  */
Packit Service 97d2fb
	    if (INTUSE(elfw2(LIBELFBITS,xlatetom)) (data, data, ident[EI_DATA])
Packit Service 97d2fb
		== NULL)
Packit Service 97d2fb
	      {
Packit Service 97d2fb
		result = -1l;
Packit Service 97d2fb
		goto out;
Packit Service 97d2fb
	      }
Packit Service 97d2fb
	  }
Packit Service 97d2fb
    }
Packit Service 97d2fb
Packit Service 97d2fb
 out:
Packit Service 97d2fb
  rwlock_unlock (elf->lock);
Packit Service 97d2fb
  return result;
Packit Service 97d2fb
}
Packit Service 97d2fb
INTDEF(elfw2(LIBELFBITS,checksum))