Branch data Line data Source code
1 : : /* Generate an index to speed access to archives.
2 : : Copyright (C) 2005-2012 Red Hat, Inc.
3 : : This file is part of elfutils.
4 : : Written by Ulrich Drepper <drepper@redhat.com>, 2005.
5 : :
6 : : This file is free software; you can redistribute it and/or modify
7 : : it under the terms of the GNU General Public License as published by
8 : : the Free Software Foundation; either version 3 of the License, or
9 : : (at your option) any later version.
10 : :
11 : : elfutils is distributed in the hope that it will be useful, but
12 : : WITHOUT ANY WARRANTY; without even the implied warranty of
13 : : MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 : : GNU General Public License for more details.
15 : :
16 : : You should have received a copy of the GNU General Public License
17 : : along with this program. If not, see <http://www.gnu.org/licenses/>. */
18 : :
19 : : #ifdef HAVE_CONFIG_H
20 : : # include <config.h>
21 : : #endif
22 : :
23 : : #include <ar.h>
24 : : #include <argp.h>
25 : : #include <assert.h>
26 : : #include <errno.h>
27 : : #include <fcntl.h>
28 : : #include <gelf.h>
29 : : #include <locale.h>
30 : : #include <obstack.h>
31 : : #include <stdlib.h>
32 : : #include <stdio.h>
33 : : #include <stdio_ext.h>
34 : : #include <unistd.h>
35 : : #include <sys/mman.h>
36 : : #include <sys/stat.h>
37 : :
38 : : #include <system.h>
39 : : #include <printversion.h>
40 : :
41 : : #include "arlib.h"
42 : :
43 : :
44 : : /* Prototypes for local functions. */
45 : : static int handle_file (const char *fname);
46 : :
47 : :
48 : : /* Name and version of program. */
49 : : ARGP_PROGRAM_VERSION_HOOK_DEF = print_version;
50 : :
51 : : /* Bug report address. */
52 : : ARGP_PROGRAM_BUG_ADDRESS_DEF = PACKAGE_BUGREPORT;
53 : :
54 : :
55 : : /* Definitions of arguments for argp functions. */
56 : : static const struct argp_option options[] =
57 : : {
58 : : { NULL, 0, NULL, 0, NULL, 0 }
59 : : };
60 : :
61 : : /* Short description of program. */
62 : : static const char doc[] = N_("Generate an index to speed access to archives.");
63 : :
64 : : /* Strings for arguments in help texts. */
65 : : static const char args_doc[] = N_("ARCHIVE");
66 : :
67 : : /* Data structure to communicate with argp functions. */
68 : : static const struct argp argp =
69 : : {
70 : : options, NULL, args_doc, doc, arlib_argp_children, NULL, NULL
71 : : };
72 : :
73 : :
74 : : int
75 : 8 : main (int argc, char *argv[])
76 : : {
77 : : /* We use no threads here which can interfere with handling a stream. */
78 : 8 : (void) __fsetlocking (stdin, FSETLOCKING_BYCALLER);
79 : 8 : (void) __fsetlocking (stdout, FSETLOCKING_BYCALLER);
80 : 8 : (void) __fsetlocking (stderr, FSETLOCKING_BYCALLER);
81 : :
82 : : /* Set locale. */
83 : 8 : (void) setlocale (LC_ALL, "");
84 : :
85 : : /* Make sure the message catalog can be found. */
86 : 8 : (void) bindtextdomain (PACKAGE_TARNAME, LOCALEDIR);
87 : :
88 : : /* Initialize the message catalog. */
89 : 8 : (void) textdomain (PACKAGE_TARNAME);
90 : :
91 : : /* Parse and process arguments. */
92 : 8 : int remaining;
93 : 8 : (void) argp_parse (&argp, argc, argv, ARGP_IN_ORDER, &remaining, NULL);
94 : :
95 : : /* Tell the library which version we are expecting. */
96 : 8 : (void) elf_version (EV_CURRENT);
97 : :
98 : : /* There must at least be one more parameter specifying the archive. */
99 [ - + ]: 8 : if (remaining == argc)
100 : : {
101 : 0 : error (0, 0, _("Archive name required"));
102 : 0 : argp_help (&argp, stderr, ARGP_HELP_SEE, "ranlib");
103 : 0 : exit (EXIT_FAILURE);
104 : : }
105 : :
106 : : /* We accept the names of multiple archives. */
107 : : int status = 0;
108 : 8 : do
109 : 8 : status |= handle_file (argv[remaining]);
110 [ - + ]: 8 : while (++remaining < argc);
111 : :
112 : 8 : return status;
113 : : }
114 : :
115 : :
116 : : static int
117 : 6 : copy_content (Elf *elf, int newfd, off_t off, size_t n)
118 : : {
119 : 6 : size_t len;
120 : 6 : char *rawfile = elf_rawfile (elf, &len);
121 : :
122 [ - + ]: 6 : assert (off + n <= len);
123 : :
124 : : /* Tell the kernel we will read all the pages sequentially. */
125 : 6 : size_t ps = sysconf (_SC_PAGESIZE);
126 [ - + ]: 6 : if (n > 2 * ps)
127 : 0 : posix_madvise (rawfile + (off & ~(ps - 1)), n, POSIX_MADV_SEQUENTIAL);
128 : :
129 : 6 : return write_retry (newfd, rawfile + off, n) != (ssize_t) n;
130 : : }
131 : :
132 : :
133 : : /* Handle a file given on the command line. */
134 : : static int
135 : 8 : handle_file (const char *fname)
136 : : {
137 : 8 : int fd = open (fname, O_RDONLY);
138 [ - + ]: 8 : if (fd == -1)
139 : : {
140 : 0 : error (0, errno, _("cannot open '%s'"), fname);
141 : 0 : return 1;
142 : : }
143 : :
144 : 8 : struct stat st;
145 [ - + ]: 8 : if (fstat (fd, &st) != 0)
146 : : {
147 : 0 : error (0, errno, _("cannot stat '%s'"), fname);
148 : 0 : close (fd);
149 : 0 : return 1;
150 : : }
151 : :
152 : : /* First we walk through the file, looking for all ELF files to
153 : : collect symbols from. */
154 : 8 : Elf *arelf = elf_begin (fd, ELF_C_READ_MMAP, NULL);
155 [ - + ]: 8 : if (arelf == NULL)
156 : : {
157 : 0 : error (0, 0, _("cannot create ELF descriptor for '%s': %s"),
158 : : fname, elf_errmsg (-1));
159 : 0 : close (fd);
160 : 0 : return 1;
161 : : }
162 : :
163 [ - + ]: 8 : if (elf_kind (arelf) != ELF_K_AR)
164 : : {
165 : 0 : error (0, 0, _("'%s' is no archive"), fname);
166 : 0 : elf_end (arelf);
167 : 0 : close (fd);
168 : 0 : return 1;
169 : : }
170 : :
171 : 8 : arlib_init ();
172 : :
173 : : /* Iterate over the content of the archive. */
174 : 8 : off_t index_off = -1;
175 : 8 : size_t index_size = 0;
176 : 8 : off_t cur_off = SARMAG;
177 : 8 : Elf *elf;
178 : 8 : Elf_Cmd cmd = ELF_C_READ_MMAP;
179 [ + + ]: 42 : while ((elf = elf_begin (fd, cmd, arelf)) != NULL)
180 : : {
181 : 26 : Elf_Arhdr *arhdr = elf_getarhdr (elf);
182 [ - + ]: 26 : assert (arhdr != NULL);
183 : :
184 : : /* If this is the index, remember the location. */
185 [ + + ]: 26 : if (strcmp (arhdr->ar_name, "/") == 0)
186 : : {
187 : 4 : index_off = elf_getaroff (elf);
188 : 4 : index_size = arhdr->ar_size;
189 : : }
190 : : else
191 : : {
192 : 22 : arlib_add_symbols (elf, fname, arhdr->ar_name, cur_off);
193 : 22 : cur_off += (((arhdr->ar_size + 1) & ~((off_t) 1))
194 : : + sizeof (struct ar_hdr));
195 : : }
196 : :
197 : : /* Get next archive element. */
198 : 26 : cmd = elf_next (elf);
199 [ - + ]: 26 : if (elf_end (elf) != 0)
200 : 34 : error (0, 0, _("error while freeing sub-ELF descriptor: %s"),
201 : : elf_errmsg (-1));
202 : : }
203 : :
204 : 8 : arlib_finalize ();
205 : :
206 : : /* If the file contains no symbols we need not do anything. */
207 : 8 : int status = 0;
208 : 8 : if (symtab.symsnamelen != 0
209 : : /* We have to rewrite the file also if it initially had an index
210 : : but now does not need one anymore. */
211 [ + + ]: 8 : || (symtab.symsnamelen == 0 && index_size != 0))
212 : 6 : {
213 : : /* Create a new, temporary file in the same directory as the
214 : : original file. */
215 : 6 : char tmpfname[strlen (fname) + 7];
216 : 6 : strcpy (stpcpy (tmpfname, fname), "XXXXXX");
217 : 6 : int newfd = mkstemp (tmpfname);
218 [ - + ]: 6 : if (unlikely (newfd == -1))
219 : : {
220 : 0 : nonew:
221 : 0 : error (0, errno, _("cannot create new file"));
222 : 0 : status = 1;
223 : : }
224 : : else
225 : : {
226 : : /* Create the header. */
227 [ - + ]: 6 : if (unlikely (write_retry (newfd, ARMAG, SARMAG) != SARMAG))
228 : : {
229 : : // XXX Use /prof/self/fd/%d ???
230 : 0 : nonew_unlink:
231 : 0 : unlink (tmpfname);
232 [ # # ]: 0 : if (newfd != -1)
233 : 0 : close (newfd);
234 : 0 : goto nonew;
235 : : }
236 : :
237 : : /* Create the new file. There are three parts as far we are
238 : : concerned: 1. original context before the index, 2. the
239 : : new index, 3. everything after the new index. */
240 : 6 : off_t rest_off;
241 [ + + ]: 6 : if (index_off != -1)
242 : 4 : rest_off = (index_off + sizeof (struct ar_hdr)
243 : 4 : + ((index_size + 1) & ~1ul));
244 : : else
245 : : rest_off = SARMAG;
246 : :
247 [ + + ]: 6 : if (symtab.symsnamelen != 0
248 : 4 : && ((write_retry (newfd, symtab.symsoff,
249 : : symtab.symsofflen)
250 [ + - ]: 4 : != (ssize_t) symtab.symsofflen)
251 : 4 : || (write_retry (newfd, symtab.symsname,
252 : : symtab.symsnamelen)
253 [ - + ]: 4 : != (ssize_t) symtab.symsnamelen)))
254 : 0 : goto nonew_unlink;
255 : :
256 : : /* Even if the original file had content before the
257 : : symbol table, we write it in the correct order. */
258 [ - + ]: 6 : if ((index_off > SARMAG
259 [ # # ]: 0 : && copy_content (arelf, newfd, SARMAG, index_off - SARMAG))
260 [ - + ]: 6 : || copy_content (arelf, newfd, rest_off, st.st_size - rest_off))
261 : 0 : goto nonew_unlink;
262 : :
263 : : /* Never complain about fchown failing. */
264 : 6 : if (fchown (newfd, st.st_uid, st.st_gid) != 0) { ; }
265 : : /* Set the mode of the new file to the same values the
266 : : original file has. */
267 [ + - ]: 6 : if (fchmod (newfd, st.st_mode & ALLPERMS) != 0
268 [ - + ]: 6 : || close (newfd) != 0)
269 : 0 : goto nonew_unlink;
270 : 6 : newfd = -1;
271 [ - + ]: 6 : if (rename (tmpfname, fname) != 0)
272 : 0 : goto nonew_unlink;
273 : : }
274 : : }
275 : :
276 : 8 : elf_end (arelf);
277 : :
278 : 8 : arlib_fini ();
279 : :
280 : 8 : close (fd);
281 : :
282 : 8 : return status;
283 : : }
284 : :
285 : :
286 : : #include "debugpred.h"
|