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