blob: 3bed95484f7f9938c46ae4f1f14819c6fc47d72b [file] [log] [blame]
Ed Warnickecb9cada2015-12-08 15:45:58 -07001/*
2 * Copyright (c) 2015 Cisco and/or its affiliates.
3 * Licensed under the Apache License, Version 2.0 (the "License");
4 * you may not use this file except in compliance with the License.
5 * You may obtain a copy of the License at:
6 *
7 * http://www.apache.org/licenses/LICENSE-2.0
8 *
9 * Unless required by applicable law or agreed to in writing, software
10 * distributed under the License is distributed on an "AS IS" BASIS,
11 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12 * See the License for the specific language governing permissions and
13 * limitations under the License.
14 */
15/*
16 Copyright (c) 2008 Eliot Dresselhaus
17
18 Permission is hereby granted, free of charge, to any person obtaining
19 a copy of this software and associated documentation files (the
20 "Software"), to deal in the Software without restriction, including
21 without limitation the rights to use, copy, modify, merge, publish,
22 distribute, sublicense, and/or sell copies of the Software, and to
23 permit persons to whom the Software is furnished to do so, subject to
24 the following conditions:
25
26 The above copyright notice and this permission notice shall be
27 included in all copies or substantial portions of the Software.
28
29 THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
30 EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
31 MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
32 NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
33 LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
34 OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
35 WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
36*/
37
38#include <vppinfra/elf.h>
39
40#include <sys/types.h>
41#include <sys/stat.h>
42#include <fcntl.h>
43
44#ifndef CLIB_UNIX
45#error "unix only"
46#endif
47
Dave Barachc3799992016-08-15 11:12:27 -040048static clib_error_t *
49elf_set_interpreter (elf_main_t * em, char *interp)
Ed Warnickecb9cada2015-12-08 15:45:58 -070050{
Dave Barachc3799992016-08-15 11:12:27 -040051 elf_segment_t *g;
52 elf_section_t *s;
53 clib_error_t *error;
Ed Warnickecb9cada2015-12-08 15:45:58 -070054
55 vec_foreach (g, em->segments)
Dave Barachc3799992016-08-15 11:12:27 -040056 {
57 if (g->header.type == ELF_SEGMENT_INTERP)
58 break;
59 }
Ed Warnickecb9cada2015-12-08 15:45:58 -070060
61 if (g >= vec_end (em->segments))
62 return clib_error_return (0, "interpreter not found");
63
64 if (g->header.memory_size < 1 + strlen (interp))
Dave Barachc3799992016-08-15 11:12:27 -040065 return clib_error_return (0,
66 "given interpreter does not fit; must be less than %d bytes (`%s' given)",
Ed Warnickecb9cada2015-12-08 15:45:58 -070067 g->header.memory_size, interp);
68
Dave Barachc3799992016-08-15 11:12:27 -040069 error =
70 elf_get_section_by_start_address (em, g->header.virtual_address, &s);
Ed Warnickecb9cada2015-12-08 15:45:58 -070071 if (error)
72 return error;
73
74 /* Put in new null terminated string. */
Dave Barachb7b92992018-10-17 10:38:51 -040075 clib_memset (s->contents, 0, vec_len (s->contents));
Damjan Marionf1213b82016-03-13 02:22:06 +010076 clib_memcpy (s->contents, interp, strlen (interp));
Ed Warnickecb9cada2015-12-08 15:45:58 -070077
78 return 0;
79}
80
81static void
82delete_dynamic_rpath_entries_from_section (elf_main_t * em, elf_section_t * s)
83{
Dave Barachc3799992016-08-15 11:12:27 -040084 elf64_dynamic_entry_t *e;
85 elf64_dynamic_entry_t *new_es = 0;
Ed Warnickecb9cada2015-12-08 15:45:58 -070086
87 vec_foreach (e, em->dynamic_entries)
Dave Barachc3799992016-08-15 11:12:27 -040088 {
89 switch (e->type)
90 {
91 case ELF_DYNAMIC_ENTRY_RPATH:
92 case ELF_DYNAMIC_ENTRY_RUN_PATH:
93 break;
Ed Warnickecb9cada2015-12-08 15:45:58 -070094
Dave Barachc3799992016-08-15 11:12:27 -040095 default:
96 vec_add1 (new_es, e[0]);
97 break;
98 }
99 }
Ed Warnickecb9cada2015-12-08 15:45:58 -0700100
101 /* Pad so as to keep section size constant. */
102 {
103 elf64_dynamic_entry_t e_end;
104 e_end.type = ELF_DYNAMIC_ENTRY_END;
105 e_end.data = 0;
106 while (vec_len (new_es) < vec_len (em->dynamic_entries))
107 vec_add1 (new_es, e_end);
108 }
109
110 elf_set_dynamic_entries (em);
111}
112
Dave Barachc3799992016-08-15 11:12:27 -0400113static void
114elf_delete_dynamic_rpath_entries (elf_main_t * em)
Ed Warnickecb9cada2015-12-08 15:45:58 -0700115{
Dave Barachc3799992016-08-15 11:12:27 -0400116 elf_section_t *s;
Ed Warnickecb9cada2015-12-08 15:45:58 -0700117
118 vec_foreach (s, em->sections)
Dave Barachc3799992016-08-15 11:12:27 -0400119 {
120 switch (s->header.type)
121 {
122 case ELF_SECTION_DYNAMIC:
123 delete_dynamic_rpath_entries_from_section (em, s);
124 break;
Ed Warnickecb9cada2015-12-08 15:45:58 -0700125
Dave Barachc3799992016-08-15 11:12:27 -0400126 default:
127 break;
128 }
129 }
Ed Warnickecb9cada2015-12-08 15:45:58 -0700130}
131
Dave Barachc3799992016-08-15 11:12:27 -0400132typedef struct
133{
Ed Warnickecb9cada2015-12-08 15:45:58 -0700134 elf_main_t elf_main;
Dave Barachc3799992016-08-15 11:12:27 -0400135 char *input_file;
136 char *output_file;
137 char *set_interpreter;
Ed Warnickecb9cada2015-12-08 15:45:58 -0700138 int verbose;
139} elf_test_main_t;
140
Dave Barachc3799992016-08-15 11:12:27 -0400141int
142main (int argc, char *argv[])
Ed Warnickecb9cada2015-12-08 15:45:58 -0700143{
Dave Barachc3799992016-08-15 11:12:27 -0400144 elf_test_main_t _tm, *tm = &_tm;
145 elf_main_t *em = &tm->elf_main;
Ed Warnickecb9cada2015-12-08 15:45:58 -0700146 unformat_input_t i;
Dave Barachc3799992016-08-15 11:12:27 -0400147 clib_error_t *error = 0;
Ed Warnickecb9cada2015-12-08 15:45:58 -0700148
Dave Barachb7b92992018-10-17 10:38:51 -0400149 clib_memset (tm, 0, sizeof (tm[0]));
Ed Warnickecb9cada2015-12-08 15:45:58 -0700150
151 unformat_init_command_line (&i, argv);
152 while (unformat_check_input (&i) != UNFORMAT_END_OF_INPUT)
153 {
154 if (unformat (&i, "in %s", &tm->input_file))
155 ;
156 else if (unformat (&i, "out %s", &tm->output_file))
157 ;
158 else if (unformat (&i, "set-interpreter %s", &tm->set_interpreter))
159 ;
160 else if (unformat (&i, "verbose"))
161 tm->verbose = ~0;
162 else if (unformat (&i, "verbose-symbols"))
163 tm->verbose |= FORMAT_ELF_MAIN_SYMBOLS;
164 else if (unformat (&i, "verbose-relocations"))
165 tm->verbose |= FORMAT_ELF_MAIN_RELOCATIONS;
166 else if (unformat (&i, "verbose-dynamic"))
167 tm->verbose |= FORMAT_ELF_MAIN_DYNAMIC;
168 else
169 {
170 error = unformat_parse_error (&i);
171 goto done;
172 }
173 }
174
Florin Corase10e3722016-04-13 00:05:27 +0200175 if (!tm->input_file)
176 {
Dave Barachc3799992016-08-15 11:12:27 -0400177 clib_warning ("No input file! Using test_bihash_template");
Florin Corase10e3722016-04-13 00:05:27 +0200178 tm->input_file = "test_bihash_template";
179 }
Ed Warnickecb9cada2015-12-08 15:45:58 -0700180
181 error = elf_read_file (em, tm->input_file);
182 if (error)
183 goto done;
184
185 if (tm->set_interpreter)
186 {
Dave Barachc3799992016-08-15 11:12:27 -0400187 clib_error_t *error = elf_set_interpreter (em, tm->set_interpreter);
Ed Warnickecb9cada2015-12-08 15:45:58 -0700188 if (error)
189 goto done;
190 elf_delete_dynamic_rpath_entries (em);
191 }
192
193 if (tm->verbose)
194 fformat (stdout, "%U", format_elf_main, em, tm->verbose);
195
196 if (tm->output_file)
197 error = elf_write_file (em, tm->output_file);
198
199 elf_main_free (em);
200
Dave Barachc3799992016-08-15 11:12:27 -0400201done:
Ed Warnickecb9cada2015-12-08 15:45:58 -0700202 if (error)
203 {
204 clib_error_report (error);
205 return 1;
206 }
207 else
208 return 0;
209}
Dave Barachc3799992016-08-15 11:12:27 -0400210
211/*
212 * fd.io coding-style-patch-verification: ON
213 *
214 * Local Variables:
215 * eval: (c-set-style "gnu")
216 * End:
217 */