1da0c48c4Sopenharmony_ci/* Get section at specific index.
2da0c48c4Sopenharmony_ci   Copyright (C) 1998, 1999, 2000, 2001, 2002, 2004, 2015 Red Hat, Inc.
3da0c48c4Sopenharmony_ci   This file is part of elfutils.
4da0c48c4Sopenharmony_ci   Contributed by Ulrich Drepper <drepper@redhat.com>, 1998.
5da0c48c4Sopenharmony_ci
6da0c48c4Sopenharmony_ci   This file is free software; you can redistribute it and/or modify
7da0c48c4Sopenharmony_ci   it under the terms of either
8da0c48c4Sopenharmony_ci
9da0c48c4Sopenharmony_ci     * the GNU Lesser General Public License as published by the Free
10da0c48c4Sopenharmony_ci       Software Foundation; either version 3 of the License, or (at
11da0c48c4Sopenharmony_ci       your option) any later version
12da0c48c4Sopenharmony_ci
13da0c48c4Sopenharmony_ci   or
14da0c48c4Sopenharmony_ci
15da0c48c4Sopenharmony_ci     * the GNU General Public License as published by the Free
16da0c48c4Sopenharmony_ci       Software Foundation; either version 2 of the License, or (at
17da0c48c4Sopenharmony_ci       your option) any later version
18da0c48c4Sopenharmony_ci
19da0c48c4Sopenharmony_ci   or both in parallel, as here.
20da0c48c4Sopenharmony_ci
21da0c48c4Sopenharmony_ci   elfutils is distributed in the hope that it will be useful, but
22da0c48c4Sopenharmony_ci   WITHOUT ANY WARRANTY; without even the implied warranty of
23da0c48c4Sopenharmony_ci   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
24da0c48c4Sopenharmony_ci   General Public License for more details.
25da0c48c4Sopenharmony_ci
26da0c48c4Sopenharmony_ci   You should have received copies of the GNU General Public License and
27da0c48c4Sopenharmony_ci   the GNU Lesser General Public License along with this program.  If
28da0c48c4Sopenharmony_ci   not, see <http://www.gnu.org/licenses/>.  */
29da0c48c4Sopenharmony_ci
30da0c48c4Sopenharmony_ci#ifdef HAVE_CONFIG_H
31da0c48c4Sopenharmony_ci# include <config.h>
32da0c48c4Sopenharmony_ci#endif
33da0c48c4Sopenharmony_ci
34da0c48c4Sopenharmony_ci#include <assert.h>
35da0c48c4Sopenharmony_ci#include <stddef.h>
36da0c48c4Sopenharmony_ci#include <stdlib.h>
37da0c48c4Sopenharmony_ci
38da0c48c4Sopenharmony_ci#include "libelfP.h"
39da0c48c4Sopenharmony_ci
40da0c48c4Sopenharmony_ci
41da0c48c4Sopenharmony_ciElf_Scn *
42da0c48c4Sopenharmony_cielf_getscn (Elf *elf, size_t idx)
43da0c48c4Sopenharmony_ci{
44da0c48c4Sopenharmony_ci  if (elf == NULL)
45da0c48c4Sopenharmony_ci    return NULL;
46da0c48c4Sopenharmony_ci
47da0c48c4Sopenharmony_ci  if (unlikely (elf->kind != ELF_K_ELF))
48da0c48c4Sopenharmony_ci    {
49da0c48c4Sopenharmony_ci      __libelf_seterrno (ELF_E_INVALID_HANDLE);
50da0c48c4Sopenharmony_ci      return NULL;
51da0c48c4Sopenharmony_ci    }
52da0c48c4Sopenharmony_ci
53da0c48c4Sopenharmony_ci  rwlock_rdlock (elf->lock);
54da0c48c4Sopenharmony_ci
55da0c48c4Sopenharmony_ci  Elf_Scn *result = NULL;
56da0c48c4Sopenharmony_ci
57da0c48c4Sopenharmony_ci  /* Find the section in the list.  */
58da0c48c4Sopenharmony_ci  Elf_ScnList *runp = (elf->class == ELFCLASS32
59da0c48c4Sopenharmony_ci		       || (offsetof (struct Elf, state.elf32.scns)
60da0c48c4Sopenharmony_ci			   == offsetof (struct Elf, state.elf64.scns))
61da0c48c4Sopenharmony_ci		       ? &elf->state.elf32.scns : &elf->state.elf64.scns);
62da0c48c4Sopenharmony_ci
63da0c48c4Sopenharmony_ci  /* Section zero is special.  It always exists even if there is no
64da0c48c4Sopenharmony_ci     "first" section.  And it is needed to store "overflow" values
65da0c48c4Sopenharmony_ci     from the Elf header.  */
66da0c48c4Sopenharmony_ci  if (idx == 0 && runp->cnt == 0 && runp->max > 0)
67da0c48c4Sopenharmony_ci    {
68da0c48c4Sopenharmony_ci      Elf_Scn *scn0 = &runp->data[0];
69da0c48c4Sopenharmony_ci      if (elf->class == ELFCLASS32)
70da0c48c4Sopenharmony_ci	{
71da0c48c4Sopenharmony_ci	  scn0->shdr.e32 = calloc (1, sizeof (Elf32_Shdr));
72da0c48c4Sopenharmony_ci	  if (scn0->shdr.e32 == NULL)
73da0c48c4Sopenharmony_ci	    {
74da0c48c4Sopenharmony_ci	      __libelf_seterrno (ELF_E_NOMEM);
75da0c48c4Sopenharmony_ci	      goto out;
76da0c48c4Sopenharmony_ci	    }
77da0c48c4Sopenharmony_ci	}
78da0c48c4Sopenharmony_ci      else
79da0c48c4Sopenharmony_ci	{
80da0c48c4Sopenharmony_ci	  scn0->shdr.e64 = calloc (1, sizeof (Elf64_Shdr));
81da0c48c4Sopenharmony_ci	  if (scn0->shdr.e64 == NULL)
82da0c48c4Sopenharmony_ci	    {
83da0c48c4Sopenharmony_ci	      __libelf_seterrno (ELF_E_NOMEM);
84da0c48c4Sopenharmony_ci	      goto out;
85da0c48c4Sopenharmony_ci	    }
86da0c48c4Sopenharmony_ci	}
87da0c48c4Sopenharmony_ci      scn0->elf = elf;
88da0c48c4Sopenharmony_ci      scn0->shdr_flags = ELF_F_DIRTY | ELF_F_MALLOCED;
89da0c48c4Sopenharmony_ci      scn0->list = elf->state.elf.scns_last;
90da0c48c4Sopenharmony_ci      scn0->data_read = 1;
91da0c48c4Sopenharmony_ci      runp->cnt = 1;
92da0c48c4Sopenharmony_ci    }
93da0c48c4Sopenharmony_ci
94da0c48c4Sopenharmony_ci  while (1)
95da0c48c4Sopenharmony_ci    {
96da0c48c4Sopenharmony_ci      if (idx < runp->max)
97da0c48c4Sopenharmony_ci	{
98da0c48c4Sopenharmony_ci	  if (idx < runp->cnt)
99da0c48c4Sopenharmony_ci	    result = &runp->data[idx];
100da0c48c4Sopenharmony_ci	  else
101da0c48c4Sopenharmony_ci	    __libelf_seterrno (ELF_E_INVALID_INDEX);
102da0c48c4Sopenharmony_ci	  break;
103da0c48c4Sopenharmony_ci	}
104da0c48c4Sopenharmony_ci
105da0c48c4Sopenharmony_ci      idx -= runp->max;
106da0c48c4Sopenharmony_ci
107da0c48c4Sopenharmony_ci      runp = runp->next;
108da0c48c4Sopenharmony_ci      if (runp == NULL)
109da0c48c4Sopenharmony_ci	{
110da0c48c4Sopenharmony_ci	  __libelf_seterrno (ELF_E_INVALID_INDEX);
111da0c48c4Sopenharmony_ci	  break;
112da0c48c4Sopenharmony_ci	}
113da0c48c4Sopenharmony_ci    }
114da0c48c4Sopenharmony_ci
115da0c48c4Sopenharmony_ci out:
116da0c48c4Sopenharmony_ci  rwlock_unlock (elf->lock);
117da0c48c4Sopenharmony_ci
118da0c48c4Sopenharmony_ci  return result;
119da0c48c4Sopenharmony_ci}
120da0c48c4Sopenharmony_ciINTDEF(elf_getscn)
121