Blame libelf/elf_getphdrnum.c

Packit Service 97d2fb
/* Return number of program headers in the ELF file.
Packit Service 97d2fb
   Copyright (C) 2010, 2014, 2015, 2016 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 <assert.h>
Packit Service 97d2fb
#include <gelf.h>
Packit Service 97d2fb
#include <stddef.h>
Packit Service 97d2fb
Packit Service 97d2fb
#include "libelfP.h"
Packit Service 97d2fb
Packit Service 97d2fb
Packit Service 97d2fb
int
Packit Service 97d2fb
internal_function
Packit Service 97d2fb
__elf_getphdrnum_rdlock (Elf *elf, size_t *dst)
Packit Service 97d2fb
{
Packit Service 97d2fb
 if (unlikely (elf->state.elf64.ehdr == NULL))
Packit Service 97d2fb
   {
Packit Service 97d2fb
     /* Maybe no ELF header was created yet.  */
Packit Service 97d2fb
     __libelf_seterrno (ELF_E_WRONG_ORDER_EHDR);
Packit Service 97d2fb
     return -1;
Packit Service 97d2fb
   }
Packit Service 97d2fb
Packit Service 97d2fb
 *dst = (elf->class == ELFCLASS32
Packit Service 97d2fb
	 ? elf->state.elf32.ehdr->e_phnum
Packit Service 97d2fb
	 : elf->state.elf64.ehdr->e_phnum);
Packit Service 97d2fb
Packit Service 97d2fb
 if (*dst == PN_XNUM)
Packit Service 97d2fb
   {
Packit Service 97d2fb
     const Elf_ScnList *const scns = (elf->class == ELFCLASS32
Packit Service 97d2fb
				      ? &elf->state.elf32.scns
Packit Service 97d2fb
				      : &elf->state.elf64.scns);
Packit Service 97d2fb
Packit Service 97d2fb
     /* If there are no section headers, perhaps this is really just 65536
Packit Service 97d2fb
	written without PN_XNUM support.  Either that or it's bad data.  */
Packit Service 97d2fb
Packit Service 97d2fb
     if (elf->class == ELFCLASS32)
Packit Service 97d2fb
       {
Packit Service 97d2fb
	 if (likely (scns->cnt > 0))
Packit Service 97d2fb
	   {
Packit Service 97d2fb
	     Elf_Scn *scn = &elf->state.elf32.scns.data[0];
Packit Service 97d2fb
	     Elf32_Shdr *shdr = scn->shdr.e32 ?: __elf32_getshdr_rdlock (scn);
Packit Service 97d2fb
	     if (shdr)
Packit Service 97d2fb
	       *dst = shdr->sh_info;
Packit Service 97d2fb
	   }
Packit Service 97d2fb
       }
Packit Service 97d2fb
     else
Packit Service 97d2fb
       {
Packit Service 97d2fb
	 if (likely (scns->cnt > 0))
Packit Service 97d2fb
	   {
Packit Service 97d2fb
	     Elf_Scn *scn = &elf->state.elf64.scns.data[0];
Packit Service 97d2fb
	     Elf64_Shdr *shdr = scn->shdr.e64 ?: __elf64_getshdr_rdlock (scn);
Packit Service 97d2fb
	     if (shdr)
Packit Service 97d2fb
	       *dst = shdr->sh_info;
Packit Service 97d2fb
	   }
Packit Service 97d2fb
       }
Packit Service 97d2fb
   }
Packit Service 97d2fb
Packit Service 97d2fb
 return 0;
Packit Service 97d2fb
}
Packit Service 97d2fb
Packit Service 97d2fb
int
Packit Service 97d2fb
internal_function
Packit Service 97d2fb
__elf_getphdrnum_chk_rdlock (Elf *elf, size_t *dst)
Packit Service 97d2fb
{
Packit Service 97d2fb
  int result = __elf_getphdrnum_rdlock (elf, dst);
Packit Service 97d2fb
Packit Service 97d2fb
  /* If the phdrs haven't been created or read in yet then do some
Packit Service 97d2fb
     sanity checking to make sure phnum and phoff are consistent.  */
Packit Service 97d2fb
  if (elf->state.elf.phdr == NULL)
Packit Service 97d2fb
    {
Packit Service 97d2fb
      Elf64_Off off = (elf->class == ELFCLASS32
Packit Service 97d2fb
		       ? elf->state.elf32.ehdr->e_phoff
Packit Service 97d2fb
		       : elf->state.elf64.ehdr->e_phoff);
Packit Service 97d2fb
      if (unlikely (off == 0))
Packit Service 97d2fb
	{
Packit Service 97d2fb
	  *dst = 0;
Packit Service 97d2fb
	  return result;
Packit Service 97d2fb
	}
Packit Service 97d2fb
Packit Service 97d2fb
      if (unlikely (off >= elf->maximum_size))
Packit Service 97d2fb
	{
Packit Service 97d2fb
	  __libelf_seterrno (ELF_E_INVALID_DATA);
Packit Service 97d2fb
	  return -1;
Packit Service 97d2fb
	}
Packit Service 97d2fb
Packit Service 97d2fb
      /* Check for too many sections.  */
Packit Service 97d2fb
      size_t phdr_size = (elf->class == ELFCLASS32
Packit Service 97d2fb
			  ? sizeof (Elf32_Phdr) : sizeof (Elf64_Phdr));
Packit Service 97d2fb
      if (unlikely (*dst > SIZE_MAX / phdr_size))
Packit Service 97d2fb
	{
Packit Service 97d2fb
	  __libelf_seterrno (ELF_E_INVALID_DATA);
Packit Service 97d2fb
	  return -1;
Packit Service 97d2fb
	}
Packit Service 97d2fb
Packit Service 97d2fb
      /* Truncated file?  Don't return more than can be indexed.  */
Packit Service 97d2fb
      if (unlikely (elf->maximum_size - off < *dst * phdr_size))
Packit Service 97d2fb
	*dst = (elf->maximum_size - off) / phdr_size;
Packit Service 97d2fb
    }
Packit Service 97d2fb
Packit Service 97d2fb
  return result;
Packit Service 97d2fb
}
Packit Service 97d2fb
Packit Service 97d2fb
int
Packit Service 97d2fb
elf_getphdrnum (Elf *elf, size_t *dst)
Packit Service 97d2fb
{
Packit Service 97d2fb
  int result;
Packit Service 97d2fb
Packit Service 97d2fb
  if (elf == NULL)
Packit Service 97d2fb
    return -1;
Packit Service 97d2fb
Packit Service 97d2fb
  if (unlikely (elf->kind != ELF_K_ELF))
Packit Service 97d2fb
    {
Packit Service 97d2fb
      __libelf_seterrno (ELF_E_INVALID_HANDLE);
Packit Service 97d2fb
      return -1;
Packit Service 97d2fb
    }
Packit Service 97d2fb
Packit Service 97d2fb
  rwlock_rdlock (elf->lock);
Packit Service 97d2fb
  result = __elf_getphdrnum_chk_rdlock (elf, dst);
Packit Service 97d2fb
  rwlock_unlock (elf->lock);
Packit Service 97d2fb
Packit Service 97d2fb
  return result;
Packit Service 97d2fb
}