Branch data Line data Source code
1 : : /* Update 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 <stdlib.h>
36 : :
37 : : #include "libelfP.h"
38 : :
39 : :
40 : : int
41 : 26051 : gelf_update_rela (Elf_Data *dst, int ndx, GElf_Rela *src)
42 : : {
43 : 26051 : Elf_Data_Scn *data_scn = (Elf_Data_Scn *) dst;
44 : : Elf_Scn *scn;
45 : 26051 : int result = 0;
46 : :
47 [ + - ]: 26051 : if (dst == NULL)
48 : : return 0;
49 : :
50 [ - + ]: 26051 : if (unlikely (ndx < 0))
51 : : {
52 : 0 : __libelf_seterrno (ELF_E_INVALID_INDEX);
53 : 0 : return 0;
54 : : }
55 : :
56 [ - + ]: 26051 : if (unlikely (data_scn->d.d_type != ELF_T_RELA))
57 : : {
58 : : /* The type of the data better should match. */
59 : 0 : __libelf_seterrno (ELF_E_DATA_MISMATCH);
60 : 0 : return 0;
61 : : }
62 : :
63 : 26051 : scn = data_scn->s;
64 : : rwlock_wrlock (scn->elf->lock);
65 : :
66 [ + + ]: 26051 : if (scn->elf->class == ELFCLASS32)
67 : : {
68 : : Elf32_Rela *rel;
69 : :
70 : : /* There is the possibility that the values in the input are
71 : : too large. */
72 [ + - ]: 31 : if (unlikely (src->r_offset > 0xffffffffull)
73 [ + - ]: 31 : || unlikely (GELF_R_SYM (src->r_info) > 0xffffff)
74 [ + - ]: 31 : || unlikely (GELF_R_TYPE (src->r_info) > 0xff)
75 [ + - ]: 31 : || unlikely (src->r_addend < -0x80000000ll)
76 [ - + ]: 31 : || unlikely (src->r_addend > 0x7fffffffll))
77 : : {
78 : 0 : __libelf_seterrno (ELF_E_INVALID_DATA);
79 : 0 : goto out;
80 : : }
81 : :
82 : : /* Check whether we have to resize the data buffer. */
83 [ - + ]: 31 : if (unlikely ((ndx + 1) * sizeof (Elf32_Rela) > data_scn->d.d_size))
84 : : {
85 : 0 : __libelf_seterrno (ELF_E_INVALID_INDEX);
86 : 0 : goto out;
87 : : }
88 : :
89 : 31 : rel = &((Elf32_Rela *) data_scn->d.d_buf)[ndx];
90 : :
91 : 31 : rel->r_offset = src->r_offset;
92 : 31 : rel->r_info = ELF32_R_INFO (GELF_R_SYM (src->r_info),
93 : : GELF_R_TYPE (src->r_info));
94 : 31 : rel->r_addend = src->r_addend;
95 : : }
96 : : else
97 : : {
98 : : /* Check whether we have to resize the data buffer. */
99 [ - + ]: 26020 : if (unlikely ((ndx + 1) * sizeof (Elf64_Rela) > data_scn->d.d_size))
100 : : {
101 : 0 : __libelf_seterrno (ELF_E_INVALID_INDEX);
102 : 0 : goto out;
103 : : }
104 : :
105 : 26020 : ((Elf64_Rela *) data_scn->d.d_buf)[ndx] = *src;
106 : : }
107 : :
108 : 26051 : result = 1;
109 : :
110 : : /* Mark the section as modified. */
111 : 26051 : scn->flags |= ELF_F_DIRTY;
112 : :
113 : : out:
114 : : rwlock_unlock (scn->elf->lock);
115 : :
116 : 26051 : return result;
117 : : }
|