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