|
Packit |
032894 |
/* Return number of program headers in the ELF file.
|
|
Packit |
032894 |
Copyright (C) 2010, 2014, 2015, 2016 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 either
|
|
Packit |
032894 |
|
|
Packit |
032894 |
* the GNU Lesser General Public License as published by the Free
|
|
Packit |
032894 |
Software Foundation; either version 3 of the License, or (at
|
|
Packit |
032894 |
your option) any later version
|
|
Packit |
032894 |
|
|
Packit |
032894 |
or
|
|
Packit |
032894 |
|
|
Packit |
032894 |
* the GNU General Public License as published by the Free
|
|
Packit |
032894 |
Software Foundation; either version 2 of the License, or (at
|
|
Packit |
032894 |
your option) any later version
|
|
Packit |
032894 |
|
|
Packit |
032894 |
or both in parallel, as here.
|
|
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 GNU
|
|
Packit |
032894 |
General Public License for more details.
|
|
Packit |
032894 |
|
|
Packit |
032894 |
You should have received copies of the GNU General Public License and
|
|
Packit |
032894 |
the GNU Lesser General Public License along with this program. If
|
|
Packit |
032894 |
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 <assert.h>
|
|
Packit |
032894 |
#include <gelf.h>
|
|
Packit |
032894 |
#include <stddef.h>
|
|
Packit |
032894 |
|
|
Packit |
032894 |
#include "libelfP.h"
|
|
Packit |
032894 |
|
|
Packit |
032894 |
|
|
Packit |
032894 |
int
|
|
Packit |
032894 |
internal_function
|
|
Packit |
032894 |
__elf_getphdrnum_rdlock (Elf *elf, size_t *dst)
|
|
Packit |
032894 |
{
|
|
Packit |
032894 |
if (unlikely (elf->state.elf64.ehdr == NULL))
|
|
Packit |
032894 |
{
|
|
Packit |
032894 |
/* Maybe no ELF header was created yet. */
|
|
Packit |
032894 |
__libelf_seterrno (ELF_E_WRONG_ORDER_EHDR);
|
|
Packit |
032894 |
return -1;
|
|
Packit |
032894 |
}
|
|
Packit |
032894 |
|
|
Packit |
032894 |
*dst = (elf->class == ELFCLASS32
|
|
Packit |
032894 |
? elf->state.elf32.ehdr->e_phnum
|
|
Packit |
032894 |
: elf->state.elf64.ehdr->e_phnum);
|
|
Packit |
032894 |
|
|
Packit |
032894 |
if (*dst == PN_XNUM)
|
|
Packit |
032894 |
{
|
|
Packit |
032894 |
const Elf_ScnList *const scns = (elf->class == ELFCLASS32
|
|
Packit |
032894 |
? &elf->state.elf32.scns
|
|
Packit |
032894 |
: &elf->state.elf64.scns);
|
|
Packit |
032894 |
|
|
Packit |
032894 |
/* If there are no section headers, perhaps this is really just 65536
|
|
Packit |
032894 |
written without PN_XNUM support. Either that or it's bad data. */
|
|
Packit |
032894 |
|
|
Packit |
032894 |
if (elf->class == ELFCLASS32)
|
|
Packit |
032894 |
{
|
|
Packit |
032894 |
if (likely (scns->cnt > 0
|
|
Packit |
032894 |
&& elf->state.elf32.scns.data[0].shdr.e32 != NULL))
|
|
Packit |
032894 |
*dst = scns->data[0].shdr.e32->sh_info;
|
|
Packit |
032894 |
}
|
|
Packit |
032894 |
else
|
|
Packit |
032894 |
{
|
|
Packit |
032894 |
if (likely (scns->cnt > 0
|
|
Packit |
032894 |
&& elf->state.elf64.scns.data[0].shdr.e64 != NULL))
|
|
Packit |
032894 |
*dst = scns->data[0].shdr.e64->sh_info;
|
|
Packit |
032894 |
}
|
|
Packit |
032894 |
}
|
|
Packit |
032894 |
|
|
Packit |
032894 |
return 0;
|
|
Packit |
032894 |
}
|
|
Packit |
032894 |
|
|
Packit |
032894 |
int
|
|
Packit |
032894 |
internal_function
|
|
Packit |
032894 |
__elf_getphdrnum_chk_rdlock (Elf *elf, size_t *dst)
|
|
Packit |
032894 |
{
|
|
Packit |
032894 |
int result = __elf_getphdrnum_rdlock (elf, dst);
|
|
Packit |
032894 |
|
|
Packit |
032894 |
/* If the phdrs haven't been created or read in yet then do some
|
|
Packit |
032894 |
sanity checking to make sure phnum and phoff are consistent. */
|
|
Packit |
032894 |
if (elf->state.elf.phdr == NULL)
|
|
Packit |
032894 |
{
|
|
Packit |
032894 |
Elf64_Off off = (elf->class == ELFCLASS32
|
|
Packit |
032894 |
? elf->state.elf32.ehdr->e_phoff
|
|
Packit |
032894 |
: elf->state.elf64.ehdr->e_phoff);
|
|
Packit |
032894 |
if (unlikely (off == 0))
|
|
Packit |
032894 |
{
|
|
Packit |
032894 |
*dst = 0;
|
|
Packit |
032894 |
return result;
|
|
Packit |
032894 |
}
|
|
Packit |
032894 |
|
|
Packit |
032894 |
if (unlikely (off >= elf->maximum_size))
|
|
Packit |
032894 |
{
|
|
Packit |
032894 |
__libelf_seterrno (ELF_E_INVALID_DATA);
|
|
Packit |
032894 |
return -1;
|
|
Packit |
032894 |
}
|
|
Packit |
032894 |
|
|
Packit |
032894 |
/* Check for too many sections. */
|
|
Packit |
032894 |
size_t phdr_size = (elf->class == ELFCLASS32
|
|
Packit |
032894 |
? sizeof (Elf32_Phdr) : sizeof (Elf64_Phdr));
|
|
Packit |
032894 |
if (unlikely (*dst > SIZE_MAX / phdr_size))
|
|
Packit |
032894 |
{
|
|
Packit |
032894 |
__libelf_seterrno (ELF_E_INVALID_DATA);
|
|
Packit |
032894 |
return -1;
|
|
Packit |
032894 |
}
|
|
Packit |
032894 |
|
|
Packit |
032894 |
/* Truncated file? Don't return more than can be indexed. */
|
|
Packit |
032894 |
if (unlikely (elf->maximum_size - off < *dst * phdr_size))
|
|
Packit |
032894 |
*dst = (elf->maximum_size - off) / phdr_size;
|
|
Packit |
032894 |
}
|
|
Packit |
032894 |
|
|
Packit |
032894 |
return result;
|
|
Packit |
032894 |
}
|
|
Packit |
032894 |
|
|
Packit |
032894 |
int
|
|
Packit |
032894 |
elf_getphdrnum (Elf *elf, size_t *dst)
|
|
Packit |
032894 |
{
|
|
Packit |
032894 |
int result;
|
|
Packit |
032894 |
|
|
Packit |
032894 |
if (elf == NULL)
|
|
Packit |
032894 |
return -1;
|
|
Packit |
032894 |
|
|
Packit |
032894 |
if (unlikely (elf->kind != ELF_K_ELF))
|
|
Packit |
032894 |
{
|
|
Packit |
032894 |
__libelf_seterrno (ELF_E_INVALID_HANDLE);
|
|
Packit |
032894 |
return -1;
|
|
Packit |
032894 |
}
|
|
Packit |
032894 |
|
|
Packit |
032894 |
rwlock_rdlock (elf->lock);
|
|
Packit |
032894 |
result = __elf_getphdrnum_chk_rdlock (elf, dst);
|
|
Packit |
032894 |
rwlock_unlock (elf->lock);
|
|
Packit |
032894 |
|
|
Packit |
032894 |
return result;
|
|
Packit |
032894 |
}
|