• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /* bucomm.c -- Bin Utils COMmon code.
2    Copyright (C) 1991-2014 Free Software Foundation, Inc.
3 
4    This file is part of GNU Binutils.
5 
6    This program 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    This program is distributed in the hope that it will be useful,
12    but 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, write to the Free Software
18    Foundation, Inc., 51 Franklin Street - Fifth Floor, Boston, MA
19    02110-1301, USA.  */
20 
21 /* We might put this in a library someday so it could be dynamically
22    loaded, but for now it's not necessary.  */
23 
24 #include "sysdep.h"
25 #include "bfd.h"
26 #include "libiberty.h"
27 #include "filenames.h"
28 #include "libbfd.h"
29 
30 #include <time.h>		/* ctime, maybe time_t */
31 #include <assert.h>
32 #include "bucomm.h"
33 
34 #ifndef HAVE_TIME_T_IN_TIME_H
35 #ifndef HAVE_TIME_T_IN_TYPES_H
36 typedef long time_t;
37 #endif
38 #endif
39 
40 static const char * endian_string (enum bfd_endian);
41 static int display_target_list (void);
42 static int display_info_table (int, int);
43 static int display_target_tables (void);
44 
45 /* Error reporting.  */
46 
47 char *program_name;
48 
49 void
bfd_nonfatal(const char * string)50 bfd_nonfatal (const char *string)
51 {
52   const char *errmsg;
53 
54   errmsg = bfd_errmsg (bfd_get_error ());
55   fflush (stdout);
56   if (string)
57     fprintf (stderr, "%s: %s: %s\n", program_name, string, errmsg);
58   else
59     fprintf (stderr, "%s: %s\n", program_name, errmsg);
60 }
61 
62 /* Issue a non fatal error message.  FILENAME, or if NULL then BFD,
63    are used to indicate the problematic file.  SECTION, if non NULL,
64    is used to provide a section name.  If FORMAT is non-null, then it
65    is used to print additional information via vfprintf.  Finally the
66    bfd error message is printed.  In summary, error messages are of
67    one of the following forms:
68 
69    PROGRAM:file: bfd-error-message
70    PROGRAM:file[section]: bfd-error-message
71    PROGRAM:file: printf-message: bfd-error-message
72    PROGRAM:file[section]: printf-message: bfd-error-message.  */
73 
74 void
bfd_nonfatal_message(const char * filename,const bfd * abfd,const asection * section,const char * format,...)75 bfd_nonfatal_message (const char *filename,
76 		      const bfd *abfd,
77 		      const asection *section,
78 		      const char *format, ...)
79 {
80   const char *errmsg;
81   const char *section_name;
82   va_list args;
83 
84   errmsg = bfd_errmsg (bfd_get_error ());
85   fflush (stdout);
86   section_name = NULL;
87   va_start (args, format);
88   fprintf (stderr, "%s", program_name);
89 
90   if (abfd)
91     {
92       if (!filename)
93 	filename = bfd_get_archive_filename (abfd);
94       if (section)
95 	section_name = bfd_get_section_name (abfd, section);
96     }
97   if (section_name)
98     fprintf (stderr, ":%s[%s]", filename, section_name);
99   else
100     fprintf (stderr, ":%s", filename);
101 
102   if (format)
103     {
104       fprintf (stderr, ": ");
105       vfprintf (stderr, format, args);
106     }
107   fprintf (stderr, ": %s\n", errmsg);
108   va_end (args);
109 }
110 
111 void
bfd_fatal(const char * string)112 bfd_fatal (const char *string)
113 {
114   bfd_nonfatal (string);
115   xexit (1);
116 }
117 
118 void
report(const char * format,va_list args)119 report (const char * format, va_list args)
120 {
121   fflush (stdout);
122   fprintf (stderr, "%s: ", program_name);
123   vfprintf (stderr, format, args);
124   putc ('\n', stderr);
125 }
126 
127 void
fatal(const char * format,...)128 fatal (const char *format, ...)
129 {
130   va_list args;
131 
132   va_start (args, format);
133 
134   report (format, args);
135   va_end (args);
136   xexit (1);
137 }
138 
139 void
non_fatal(const char * format,...)140 non_fatal (const char *format, ...)
141 {
142   va_list args;
143 
144   va_start (args, format);
145 
146   report (format, args);
147   va_end (args);
148 }
149 
150 /* Set the default BFD target based on the configured target.  Doing
151    this permits the binutils to be configured for a particular target,
152    and linked against a shared BFD library which was configured for a
153    different target.  */
154 
155 void
set_default_bfd_target(void)156 set_default_bfd_target (void)
157 {
158   /* The macro TARGET is defined by Makefile.  */
159   const char *target = TARGET;
160 
161   if (! bfd_set_default_target (target))
162     fatal (_("can't set BFD default target to `%s': %s"),
163 	   target, bfd_errmsg (bfd_get_error ()));
164 }
165 
166 /* After a FALSE return from bfd_check_format_matches with
167    bfd_get_error () == bfd_error_file_ambiguously_recognized, print
168    the possible matching targets.  */
169 
170 void
list_matching_formats(char ** p)171 list_matching_formats (char **p)
172 {
173   fflush (stdout);
174   fprintf (stderr, _("%s: Matching formats:"), program_name);
175   while (*p)
176     fprintf (stderr, " %s", *p++);
177   fputc ('\n', stderr);
178 }
179 
180 /* List the supported targets.  */
181 
182 void
list_supported_targets(const char * name,FILE * f)183 list_supported_targets (const char *name, FILE *f)
184 {
185   int t;
186   const char **targ_names;
187 
188   if (name == NULL)
189     fprintf (f, _("Supported targets:"));
190   else
191     fprintf (f, _("%s: supported targets:"), name);
192 
193   targ_names = bfd_target_list ();
194   for (t = 0; targ_names[t] != NULL; t++)
195     fprintf (f, " %s", targ_names[t]);
196   fprintf (f, "\n");
197   free (targ_names);
198 }
199 
200 /* List the supported architectures.  */
201 
202 void
list_supported_architectures(const char * name,FILE * f)203 list_supported_architectures (const char *name, FILE *f)
204 {
205   const char ** arch;
206   const char ** arches;
207 
208   if (name == NULL)
209     fprintf (f, _("Supported architectures:"));
210   else
211     fprintf (f, _("%s: supported architectures:"), name);
212 
213   for (arch = arches = bfd_arch_list (); *arch; arch++)
214     fprintf (f, " %s", *arch);
215   fprintf (f, "\n");
216   free (arches);
217 }
218 
219 /* The length of the longest architecture name + 1.  */
220 #define LONGEST_ARCH sizeof ("powerpc:common")
221 
222 static const char *
endian_string(enum bfd_endian endian)223 endian_string (enum bfd_endian endian)
224 {
225   switch (endian)
226     {
227     case BFD_ENDIAN_BIG: return _("big endian");
228     case BFD_ENDIAN_LITTLE: return _("little endian");
229     default: return _("endianness unknown");
230     }
231 }
232 
233 /* List the targets that BFD is configured to support, each followed
234    by its endianness and the architectures it supports.  */
235 
236 static int
display_target_list(void)237 display_target_list (void)
238 {
239   char *dummy_name;
240   int t;
241   int ret = 1;
242 
243   dummy_name = make_temp_file (NULL);
244   for (t = 0; bfd_target_vector[t]; t++)
245     {
246       const bfd_target *p = bfd_target_vector[t];
247       bfd *abfd = bfd_openw (dummy_name, p->name);
248       int a;
249 
250       printf (_("%s\n (header %s, data %s)\n"), p->name,
251 	      endian_string (p->header_byteorder),
252 	      endian_string (p->byteorder));
253 
254       if (abfd == NULL)
255 	{
256           bfd_nonfatal (dummy_name);
257           ret = 0;
258 	  continue;
259 	}
260 
261       if (! bfd_set_format (abfd, bfd_object))
262 	{
263 	  if (bfd_get_error () != bfd_error_invalid_operation)
264             {
265 	      bfd_nonfatal (p->name);
266               ret = 0;
267             }
268 	  bfd_close_all_done (abfd);
269 	  continue;
270 	}
271 
272       for (a = bfd_arch_obscure + 1; a < bfd_arch_last; a++)
273 	if (bfd_set_arch_mach (abfd, (enum bfd_architecture) a, 0))
274 	  printf ("  %s\n",
275 		  bfd_printable_arch_mach ((enum bfd_architecture) a, 0));
276       bfd_close_all_done (abfd);
277     }
278   unlink (dummy_name);
279   free (dummy_name);
280 
281   return ret;
282 }
283 
284 /* Print a table showing which architectures are supported for entries
285    FIRST through LAST-1 of bfd_target_vector (targets across,
286    architectures down).  */
287 
288 static int
display_info_table(int first,int last)289 display_info_table (int first, int last)
290 {
291   int t;
292   int ret = 1;
293   char *dummy_name;
294   int a;
295 
296   /* Print heading of target names.  */
297   printf ("\n%*s", (int) LONGEST_ARCH, " ");
298   for (t = first; t < last && bfd_target_vector[t]; t++)
299     printf ("%s ", bfd_target_vector[t]->name);
300   putchar ('\n');
301 
302   dummy_name = make_temp_file (NULL);
303   for (a = bfd_arch_obscure + 1; a < bfd_arch_last; a++)
304     if (strcmp (bfd_printable_arch_mach ((enum bfd_architecture) a, 0),
305                 "UNKNOWN!") != 0)
306       {
307 	printf ("%*s ", (int) LONGEST_ARCH - 1,
308 		bfd_printable_arch_mach ((enum bfd_architecture) a, 0));
309 	for (t = first; t < last && bfd_target_vector[t]; t++)
310 	  {
311 	    const bfd_target *p = bfd_target_vector[t];
312 	    bfd_boolean ok = TRUE;
313 	    bfd *abfd = bfd_openw (dummy_name, p->name);
314 
315 	    if (abfd == NULL)
316 	      {
317 		bfd_nonfatal (p->name);
318                 ret = 0;
319 		ok = FALSE;
320 	      }
321 
322 	    if (ok)
323 	      {
324 		if (! bfd_set_format (abfd, bfd_object))
325 		  {
326 		    if (bfd_get_error () != bfd_error_invalid_operation)
327                       {
328 		        bfd_nonfatal (p->name);
329                         ret = 0;
330                       }
331 		    ok = FALSE;
332 		  }
333 	      }
334 
335 	    if (ok)
336 	      {
337 		if (! bfd_set_arch_mach (abfd, (enum bfd_architecture) a, 0))
338 		  ok = FALSE;
339 	      }
340 
341 	    if (ok)
342 	      printf ("%s ", p->name);
343 	    else
344 	      {
345 		int l = strlen (p->name);
346 		while (l--)
347 		  putchar ('-');
348 		putchar (' ');
349 	      }
350 	    if (abfd != NULL)
351 	      bfd_close_all_done (abfd);
352 	  }
353 	putchar ('\n');
354       }
355   unlink (dummy_name);
356   free (dummy_name);
357 
358   return ret;
359 }
360 
361 /* Print tables of all the target-architecture combinations that
362    BFD has been configured to support.  */
363 
364 static int
display_target_tables(void)365 display_target_tables (void)
366 {
367   int t;
368   int columns;
369   int ret = 1;
370   char *colum;
371 
372   columns = 0;
373   colum = getenv ("COLUMNS");
374   if (colum != NULL)
375     columns = atoi (colum);
376   if (columns == 0)
377     columns = 80;
378 
379   t = 0;
380   while (bfd_target_vector[t] != NULL)
381     {
382       int oldt = t, wid;
383 
384       wid = LONGEST_ARCH + strlen (bfd_target_vector[t]->name) + 1;
385       ++t;
386       while (wid < columns && bfd_target_vector[t] != NULL)
387 	{
388 	  int newwid;
389 
390 	  newwid = wid + strlen (bfd_target_vector[t]->name) + 1;
391 	  if (newwid >= columns)
392 	    break;
393 	  wid = newwid;
394 	  ++t;
395 	}
396       if (! display_info_table (oldt, t))
397         ret = 0;
398     }
399 
400   return ret;
401 }
402 
403 int
display_info(void)404 display_info (void)
405 {
406   printf (_("BFD header file version %s\n"), BFD_VERSION_STRING);
407   if (! display_target_list () || ! display_target_tables ())
408     return 1;
409   else
410     return 0;
411 }
412 
413 /* Display the archive header for an element as if it were an ls -l listing:
414 
415    Mode       User\tGroup\tSize\tDate               Name */
416 
417 void
print_arelt_descr(FILE * file,bfd * abfd,bfd_boolean verbose)418 print_arelt_descr (FILE *file, bfd *abfd, bfd_boolean verbose)
419 {
420   struct stat buf;
421 
422   if (verbose)
423     {
424       if (bfd_stat_arch_elt (abfd, &buf) == 0)
425 	{
426 	  char modebuf[11];
427 	  char timebuf[40];
428 	  time_t when = buf.st_mtime;
429 	  const char *ctime_result = (const char *) ctime (&when);
430 	  bfd_size_type size;
431 
432 	  /* POSIX format:  skip weekday and seconds from ctime output.  */
433 	  sprintf (timebuf, "%.12s %.4s", ctime_result + 4, ctime_result + 20);
434 
435 	  mode_string (buf.st_mode, modebuf);
436 	  modebuf[10] = '\0';
437 	  size = buf.st_size;
438 	  /* POSIX 1003.2/D11 says to skip first character (entry type).  */
439 	  fprintf (file, "%s %ld/%ld %6" BFD_VMA_FMT "u %s ", modebuf + 1,
440 		   (long) buf.st_uid, (long) buf.st_gid,
441 		   size, timebuf);
442 	}
443     }
444 
445   fprintf (file, "%s\n", bfd_get_filename (abfd));
446 }
447 
448 /* Return a path for a new temporary file in the same directory
449    as file PATH.  */
450 
451 static char *
template_in_dir(const char * path)452 template_in_dir (const char *path)
453 {
454 #define template "stXXXXXX"
455   const char *slash = strrchr (path, '/');
456   char *tmpname;
457   size_t len;
458 
459 #ifdef HAVE_DOS_BASED_FILE_SYSTEM
460   {
461     /* We could have foo/bar\\baz, or foo\\bar, or d:bar.  */
462     char *bslash = strrchr (path, '\\');
463 
464     if (slash == NULL || (bslash != NULL && bslash > slash))
465       slash = bslash;
466     if (slash == NULL && path[0] != '\0' && path[1] == ':')
467       slash = path + 1;
468   }
469 #endif
470 
471   if (slash != (char *) NULL)
472     {
473       len = slash - path;
474       tmpname = (char *) xmalloc (len + sizeof (template) + 2);
475       memcpy (tmpname, path, len);
476 
477 #ifdef HAVE_DOS_BASED_FILE_SYSTEM
478       /* If tmpname is "X:", appending a slash will make it a root
479 	 directory on drive X, which is NOT the same as the current
480 	 directory on drive X.  */
481       if (len == 2 && tmpname[1] == ':')
482 	tmpname[len++] = '.';
483 #endif
484       tmpname[len++] = '/';
485     }
486   else
487     {
488       tmpname = (char *) xmalloc (sizeof (template));
489       len = 0;
490     }
491 
492   memcpy (tmpname + len, template, sizeof (template));
493   return tmpname;
494 #undef template
495 }
496 
497 /* Return the name of a created temporary file in the same directory
498    as FILENAME.  */
499 
500 char *
make_tempname(char * filename)501 make_tempname (char *filename)
502 {
503   char *tmpname = template_in_dir (filename);
504   int fd;
505 
506 #ifdef HAVE_MKSTEMP
507   fd = mkstemp (tmpname);
508 #else
509   tmpname = mktemp (tmpname);
510   if (tmpname == NULL)
511     return NULL;
512   fd = open (tmpname, O_RDWR | O_CREAT | O_EXCL, 0600);
513 #endif
514   if (fd == -1)
515     {
516       free (tmpname);
517       return NULL;
518     }
519   close (fd);
520   return tmpname;
521 }
522 
523 /* Return the name of a created temporary directory inside the
524    directory containing FILENAME.  */
525 
526 char *
make_tempdir(char * filename)527 make_tempdir (char *filename)
528 {
529   char *tmpname = template_in_dir (filename);
530 
531 #ifdef HAVE_MKDTEMP
532   return mkdtemp (tmpname);
533 #else
534   tmpname = mktemp (tmpname);
535   if (tmpname == NULL)
536     return NULL;
537 #if defined (_WIN32) && !defined (__CYGWIN32__)
538   if (mkdir (tmpname) != 0)
539     return NULL;
540 #else
541   if (mkdir (tmpname, 0700) != 0)
542     return NULL;
543 #endif
544   return tmpname;
545 #endif
546 }
547 
548 /* Parse a string into a VMA, with a fatal error if it can't be
549    parsed.  */
550 
551 bfd_vma
parse_vma(const char * s,const char * arg)552 parse_vma (const char *s, const char *arg)
553 {
554   bfd_vma ret;
555   const char *end;
556 
557   ret = bfd_scan_vma (s, &end, 0);
558 
559   if (*end != '\0')
560     fatal (_("%s: bad number: %s"), arg, s);
561 
562   return ret;
563 }
564 
565 /* Returns the size of the named file.  If the file does not
566    exist, or if it is not a real file, then a suitable non-fatal
567    error message is printed and (off_t) -1 is returned.  */
568 
569 off_t
get_file_size(const char * file_name)570 get_file_size (const char * file_name)
571 {
572   struct stat statbuf;
573 
574   if (stat (file_name, &statbuf) < 0)
575     {
576       if (errno == ENOENT)
577 	non_fatal (_("'%s': No such file"), file_name);
578       else
579 	non_fatal (_("Warning: could not locate '%s'.  reason: %s"),
580 		   file_name, strerror (errno));
581     }
582   else if (! S_ISREG (statbuf.st_mode))
583     non_fatal (_("Warning: '%s' is not an ordinary file"), file_name);
584   else if (statbuf.st_size < 0)
585     non_fatal (_("Warning: '%s' has negative size, probably it is too large"),
586                file_name);
587   else
588     return statbuf.st_size;
589 
590   return (off_t) -1;
591 }
592 
593 /* Return the filename in a static buffer.  */
594 
595 const char *
bfd_get_archive_filename(const bfd * abfd)596 bfd_get_archive_filename (const bfd *abfd)
597 {
598   static size_t curr = 0;
599   static char *buf;
600   size_t needed;
601 
602   assert (abfd != NULL);
603 
604   if (!abfd->my_archive)
605     return bfd_get_filename (abfd);
606 
607   needed = (strlen (bfd_get_filename (abfd->my_archive))
608 	    + strlen (bfd_get_filename (abfd)) + 3);
609   if (needed > curr)
610     {
611       if (curr)
612 	free (buf);
613       curr = needed + (needed >> 1);
614       buf = (char *) bfd_malloc (curr);
615       /* If we can't malloc, fail safe by returning just the file name.
616 	 This function is only used when building error messages.  */
617       if (!buf)
618 	{
619 	  curr = 0;
620 	  return bfd_get_filename (abfd);
621 	}
622     }
623   sprintf (buf, "%s(%s)", bfd_get_filename (abfd->my_archive),
624 	   bfd_get_filename (abfd));
625   return buf;
626 }
627 
628 /* Returns TRUE iff PATHNAME, a filename of an archive member,
629    is valid for writing.  For security reasons absolute paths
630    and paths containing /../ are not allowed.  See PR 17533.  */
631 
632 bfd_boolean
is_valid_archive_path(char const * pathname)633 is_valid_archive_path (char const * pathname)
634 {
635   const char * n = pathname;
636 
637   if (IS_ABSOLUTE_PATH (n))
638     return FALSE;
639 
640   while (*n)
641     {
642       if (*n == '.' && *++n == '.' && ( ! *++n || IS_DIR_SEPARATOR (*n)))
643 	return FALSE;
644 
645       while (*n && ! IS_DIR_SEPARATOR (*n))
646 	n++;
647       while (IS_DIR_SEPARATOR (*n))
648 	n++;
649     }
650 
651   return TRUE;
652 }
653