Branch data Line data Source code
1 : : /* Get RELA relocation information at given index.
2 : : Copyright (C) 2000, 2001, 2002 Red Hat, Inc.
3 : : This file is part of elfutils.
4 : : Written by Ulrich Drepper <drepper@redhat.com>, 2000.
5 : :
6 : : This file is free software; you can redistribute it and/or modify
7 : : it under the terms of either
8 : :
9 : : * the GNU Lesser General Public License as published by the Free
10 : : Software Foundation; either version 3 of the License, or (at
11 : : your option) any later version
12 : :
13 : : or
14 : :
15 : : * the GNU General Public License as published by the Free
16 : : Software Foundation; either version 2 of the License, or (at
17 : : your option) any later version
18 : :
19 : : or both in parallel, as here.
20 : :
21 : : elfutils is distributed in the hope that it will be useful, but
22 : : WITHOUT ANY WARRANTY; without even the implied warranty of
23 : : MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
24 : : General Public License for more details.
25 : :
26 : : You should have received copies of the GNU General Public License and
27 : : the GNU Lesser General Public License along with this program. If
28 : : not, see <http://www.gnu.org/licenses/>. */
29 : :
30 : : #ifdef HAVE_CONFIG_H
31 : : # include <config.h>
32 : : #endif
33 : :
34 : : #include <gelf.h>
35 : : #include <string.h>
36 : :
37 : : #include "libelfP.h"
38 : :
39 : :
40 : : GElf_Rela *
41 : 65185 : gelf_getrela (data, ndx, dst)
42 : : Elf_Data *data;
43 : : int ndx;
44 : : GElf_Rela *dst;
45 : : {
46 : 65185 : Elf_Data_Scn *data_scn = (Elf_Data_Scn *) data;
47 : : Elf_Scn *scn;
48 : : GElf_Rela *result;
49 : :
50 [ + - ]: 65185 : if (data_scn == NULL)
51 : : return NULL;
52 : :
53 [ - + ]: 65185 : if (unlikely (ndx < 0))
54 : : {
55 : 0 : __libelf_seterrno (ELF_E_INVALID_INDEX);
56 : 0 : return NULL;
57 : : }
58 : :
59 [ - + ]: 65185 : if (unlikely (data_scn->d.d_type != ELF_T_RELA))
60 : : {
61 : 0 : __libelf_seterrno (ELF_E_INVALID_HANDLE);
62 : 0 : return NULL;
63 : : }
64 : :
65 : : /* This is the one place where we have to take advantage of the fact
66 : : that an `Elf_Data' pointer is also a pointer to `Elf_Data_Scn'.
67 : : The interface is broken so that it requires this hack. */
68 : 65185 : scn = data_scn->s;
69 : :
70 : : rwlock_rdlock (scn->elf->lock);
71 : :
72 [ + + ]: 65185 : if (scn->elf->class == ELFCLASS32)
73 : : {
74 : : /* We have to convert the data. */
75 [ - + ]: 109 : if (unlikely ((ndx + 1) * sizeof (Elf32_Rela) > data_scn->d.d_size))
76 : : {
77 : 0 : __libelf_seterrno (ELF_E_INVALID_INDEX);
78 : 0 : result = NULL;
79 : : }
80 : : else
81 : : {
82 : 109 : Elf32_Rela *src = &((Elf32_Rela *) data_scn->d.d_buf)[ndx];
83 : :
84 : 109 : dst->r_offset = src->r_offset;
85 : 109 : dst->r_info = GELF_R_INFO (ELF32_R_SYM (src->r_info),
86 : : ELF32_R_TYPE (src->r_info));
87 : 109 : dst->r_addend = src->r_addend;
88 : :
89 : 109 : result = dst;
90 : : }
91 : : }
92 : : else
93 : : {
94 : : /* Simply copy the data after we made sure we are actually getting
95 : : correct data. */
96 [ - + ]: 65076 : if (unlikely ((ndx + 1) * sizeof (Elf64_Rela) > data_scn->d.d_size))
97 : : {
98 : 0 : __libelf_seterrno (ELF_E_INVALID_INDEX);
99 : 0 : result = NULL;
100 : : }
101 : : else
102 : 65076 : result = memcpy (dst, &((Elf64_Rela *) data_scn->d.d_buf)[ndx],
103 : : sizeof (Elf64_Rela));
104 : : }
105 : :
106 : : rwlock_unlock (scn->elf->lock);
107 : :
108 : 65185 : return result;
109 : : }
|