Blame libelf/gelf_getnote.c

Packit 032894
/* Get note information at the supplied offset.
Packit 032894
   Copyright (C) 2007, 2014, 2015, 2018 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 <string.h>
Packit 032894
Packit 032894
#include "libelfP.h"
Packit 032894
Packit 032894
size_t
Packit 032894
gelf_getnote (Elf_Data *data, size_t offset, GElf_Nhdr *result,
Packit 032894
	      size_t *name_offset, size_t *desc_offset)
Packit 032894
{
Packit 032894
  if (data == NULL)
Packit 032894
    return 0;
Packit 032894
Packit 032894
  if (unlikely (data->d_type != ELF_T_NHDR && data->d_type != ELF_T_NHDR8))
Packit 032894
    {
Packit 032894
      __libelf_seterrno (ELF_E_INVALID_HANDLE);
Packit 032894
      return 0;
Packit 032894
    }
Packit 032894
Packit 032894
  /* It's easy to handle this type.  It has the same size for 32 and
Packit 032894
     64 bit objects.  */
Packit 032894
  assert (sizeof (GElf_Nhdr) == sizeof (Elf32_Nhdr));
Packit 032894
  assert (sizeof (GElf_Nhdr) == sizeof (Elf64_Nhdr));
Packit 032894
Packit 032894
  rwlock_rdlock (((Elf_Data_Scn *) data)->s->elf->lock);
Packit 032894
Packit 032894
  /* The data is already in the correct form.  Just make sure the
Packit 032894
     offset is OK.  */
Packit 032894
  if (unlikely (offset > data->d_size
Packit 032894
		|| data->d_size - offset < sizeof (GElf_Nhdr)))
Packit 032894
    {
Packit 032894
      __libelf_seterrno (ELF_E_OFFSET_RANGE);
Packit 032894
      offset = 0;
Packit 032894
    }
Packit 032894
  else
Packit 032894
    {
Packit 032894
      const GElf_Nhdr *n = data->d_buf + offset;
Packit 032894
      offset += sizeof *n;
Packit 032894
Packit 032894
      if (offset > data->d_size)
Packit 032894
	offset = 0;
Packit 032894
      else
Packit 032894
	{
Packit 032894
	  /* This is slightly tricky, offset is guaranteed to be 4
Packit 032894
	     byte aligned, which is what we need for the name_offset.
Packit 032894
	     And normally desc_offset is also 4 byte aligned, but not
Packit 032894
	     for GNU Property notes, then it should be 8.  So align
Packit 032894
	     the offset, after adding the namesz, and include padding
Packit 032894
	     in descsz to get to the end.  */
Packit 032894
	  *name_offset = offset;
Packit 032894
	  if (n->n_namesz > data->d_size
Packit 032894
	      || offset > data->d_size - n->n_namesz)
Packit 032894
	    offset = 0;
Packit 032894
	  else
Packit 032894
	    {
Packit 032894
	      offset += n->n_namesz;
Packit 032894
	      /* Include padding.  Check below for overflow.  */
Packit 032894
	      GElf_Word descsz = (data->d_type == ELF_T_NHDR8
Packit 032894
				  ? NOTE_ALIGN8 (n->n_descsz)
Packit 032894
				  : NOTE_ALIGN4 (n->n_descsz));
Packit 032894
Packit 032894
	      if (data->d_type == ELF_T_NHDR8)
Packit 032894
		offset = NOTE_ALIGN8 (offset);
Packit 032894
	      else
Packit 032894
		offset = NOTE_ALIGN4 (offset);
Packit 032894
Packit 032894
	      if (unlikely (offset > data->d_size
Packit 032894
			    || data->d_size - offset < descsz
Packit 032894
			    || (descsz == 0 && n->n_descsz != 0)))
Packit 032894
		offset = 0;
Packit 032894
	      else
Packit 032894
		{
Packit 032894
		  *desc_offset = offset;
Packit 032894
		  offset += descsz;
Packit 032894
		  *result = *n;
Packit 032894
		}
Packit 032894
	    }
Packit 032894
	}
Packit 032894
    }
Packit 032894
Packit 032894
  rwlock_unlock (((Elf_Data_Scn *) data)->s->elf->lock);
Packit 032894
Packit 032894
  return offset;
Packit 032894
}