2009-09-02 Tristan Gingold <gingold@adacore.com>
[binutils.git] / binutils / sysdump.c
blob9f50d65684acfc0f268fc7fde97d5b8b034932ef
1 /* Sysroff object format dumper.
2 Copyright 1994, 1995, 1998, 1999, 2000, 2001, 2002, 2003, 2005, 2007, 2009
3 Free Software Foundation, Inc.
5 This file is part of GNU Binutils.
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 3 of the License, or
10 (at your option) any later version.
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.
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. */
23 /* Written by Steve Chamberlain <sac@cygnus.com>.
25 This program reads a SYSROFF object file and prints it in an
26 almost human readable form to stdout. */
28 #include "sysdep.h"
29 #include "bfd.h"
30 #include "safe-ctype.h"
31 #include "libiberty.h"
32 #include "getopt.h"
33 #include "bucomm.h"
34 #include "sysroff.h"
36 static int dump = 1;
37 static int segmented_p;
38 static int code;
39 static int addrsize = 4;
40 static FILE *file;
42 static void dh (unsigned char *, int);
43 static void itheader (char *, int);
44 static void p (void);
45 static void tabout (void);
46 static void pbarray (barray *);
47 static int getone (int);
48 static int opt (int);
49 static void must (int);
50 static void tab (int, char *);
51 static void dump_symbol_info (void);
52 static void derived_type (void);
53 static void module (void);
54 static void show_usage (FILE *, int);
56 extern int main (int, char **);
58 static char *
59 getCHARS (unsigned char *ptr, int *idx, int size, int max)
61 int oc = *idx / 8;
62 char *r;
63 int b = size;
65 if (b >= max)
66 return "*undefined*";
68 if (b == 0)
70 /* Got to work out the length of the string from self. */
71 b = ptr[oc++];
72 (*idx) += 8;
75 *idx += b * 8;
76 r = xcalloc (b + 1, 1);
77 memcpy (r, ptr + oc, b);
78 r[b] = 0;
80 return r;
83 static void
84 dh (unsigned char *ptr, int size)
86 int i;
87 int j;
88 int span = 16;
90 printf ("\n************************************************************\n");
92 for (i = 0; i < size; i += span)
94 for (j = 0; j < span; j++)
96 if (j + i < size)
97 printf ("%02x ", ptr[i + j]);
98 else
99 printf (" ");
102 for (j = 0; j < span && j + i < size; j++)
104 int c = ptr[i + j];
106 if (c < 32 || c > 127)
107 c = '.';
108 printf ("%c", c);
111 printf ("\n");
115 static int
116 fillup (unsigned char *ptr)
118 int size;
119 int sum;
120 int i;
122 size = getc (file);
123 if (size == EOF
124 || size <= 2)
125 return 0;
127 size -= 2;
128 if (fread (ptr, size, 1, file) != 1)
129 return 0;
131 sum = code + size + 2;
133 for (i = 0; i < size; i++)
134 sum += ptr[i];
136 if ((sum & 0xff) != 0xff)
137 printf ("SUM IS %x\n", sum);
139 if (dump)
140 dh (ptr, size);
142 return size;
145 static barray
146 getBARRAY (unsigned char *ptr, int *idx, int dsize ATTRIBUTE_UNUSED,
147 int max ATTRIBUTE_UNUSED)
149 barray res;
150 int i;
151 int byte = *idx / 8;
152 int size = ptr[byte++];
154 res.len = size;
155 res.data = (unsigned char *) xmalloc (size);
157 for (i = 0; i < size; i++)
158 res.data[i] = ptr[byte++];
160 return res;
163 static int
164 getINT (unsigned char *ptr, int *idx, int size, int max)
166 int n = 0;
167 int byte = *idx / 8;
169 if (byte >= max)
170 return 0;
172 if (size == -2)
173 size = addrsize;
175 if (size == -1)
176 size = 0;
178 switch (size)
180 case 0:
181 return 0;
182 case 1:
183 n = (ptr[byte]);
184 break;
185 case 2:
186 n = (ptr[byte + 0] << 8) + ptr[byte + 1];
187 break;
188 case 4:
189 n = (ptr[byte + 0] << 24) + (ptr[byte + 1] << 16) + (ptr[byte + 2] << 8) + (ptr[byte + 3]);
190 break;
191 default:
192 abort ();
195 *idx += size * 8;
196 return n;
199 static int
200 getBITS (unsigned char *ptr, int *idx, int size, int max)
202 int byte = *idx / 8;
203 int bit = *idx % 8;
205 if (byte >= max)
206 return 0;
208 *idx += size;
210 return (ptr[byte] >> (8 - bit - size)) & ((1 << size) - 1);
213 static void
214 itheader (char *name, int code)
216 printf ("\n%s 0x%02x\n", name, code);
219 static int indent;
221 static void
222 p (void)
224 int i;
226 for (i = 0; i < indent; i++)
227 printf ("| ");
229 printf ("> ");
232 static void
233 tabout (void)
235 p ();
238 static void
239 pbarray (barray *y)
241 int x;
243 printf ("%d (", y->len);
245 for (x = 0; x < y->len; x++)
246 printf ("(%02x %c)", y->data[x],
247 ISPRINT (y->data[x]) ? y->data[x] : '.');
249 printf (")\n");
252 #define SYSROFF_PRINT
253 #define SYSROFF_SWAP_IN
255 #include "sysroff.c"
257 /* FIXME: sysinfo, which generates sysroff.[ch] from sysroff.info, can't
258 hack the special case of the tr block, which has no contents. So we
259 implement our own functions for reading in and printing out the tr
260 block. */
262 #define IT_tr_CODE 0x7f
264 static void
265 sysroff_swap_tr_in (void)
267 unsigned char raw[255];
269 memset (raw, 0, 255);
270 fillup (raw);
273 static void
274 sysroff_print_tr_out (void)
276 itheader ("tr", IT_tr_CODE);
279 static int
280 getone (int type)
282 int c = getc (file);
284 code = c;
286 if ((c & 0x7f) != type)
288 ungetc (c, file);
289 return 0;
292 switch (c & 0x7f)
294 case IT_cs_CODE:
296 struct IT_cs dummy;
297 sysroff_swap_cs_in (&dummy);
298 sysroff_print_cs_out (&dummy);
300 break;
302 case IT_dln_CODE:
304 struct IT_dln dummy;
305 sysroff_swap_dln_in (&dummy);
306 sysroff_print_dln_out (&dummy);
308 break;
310 case IT_hd_CODE:
312 struct IT_hd dummy;
313 sysroff_swap_hd_in (&dummy);
314 addrsize = dummy.afl;
315 sysroff_print_hd_out (&dummy);
317 break;
319 case IT_dar_CODE:
321 struct IT_dar dummy;
322 sysroff_swap_dar_in (&dummy);
323 sysroff_print_dar_out (&dummy);
325 break;
327 case IT_dsy_CODE:
329 struct IT_dsy dummy;
330 sysroff_swap_dsy_in (&dummy);
331 sysroff_print_dsy_out (&dummy);
333 break;
335 case IT_dfp_CODE:
337 struct IT_dfp dummy;
338 sysroff_swap_dfp_in (&dummy);
339 sysroff_print_dfp_out (&dummy);
341 break;
343 case IT_dso_CODE:
345 struct IT_dso dummy;
346 sysroff_swap_dso_in (&dummy);
347 sysroff_print_dso_out (&dummy);
349 break;
351 case IT_dpt_CODE:
353 struct IT_dpt dummy;
354 sysroff_swap_dpt_in (&dummy);
355 sysroff_print_dpt_out (&dummy);
357 break;
359 case IT_den_CODE:
361 struct IT_den dummy;
362 sysroff_swap_den_in (&dummy);
363 sysroff_print_den_out (&dummy);
365 break;
367 case IT_dbt_CODE:
369 struct IT_dbt dummy;
370 sysroff_swap_dbt_in (&dummy);
371 sysroff_print_dbt_out (&dummy);
373 break;
375 case IT_dty_CODE:
377 struct IT_dty dummy;
378 sysroff_swap_dty_in (&dummy);
379 sysroff_print_dty_out (&dummy);
381 break;
383 case IT_un_CODE:
385 struct IT_un dummy;
386 sysroff_swap_un_in (&dummy);
387 sysroff_print_un_out (&dummy);
389 break;
391 case IT_sc_CODE:
393 struct IT_sc dummy;
394 sysroff_swap_sc_in (&dummy);
395 sysroff_print_sc_out (&dummy);
397 break;
399 case IT_er_CODE:
401 struct IT_er dummy;
402 sysroff_swap_er_in (&dummy);
403 sysroff_print_er_out (&dummy);
405 break;
407 case IT_ed_CODE:
409 struct IT_ed dummy;
410 sysroff_swap_ed_in (&dummy);
411 sysroff_print_ed_out (&dummy);
413 break;
415 case IT_sh_CODE:
417 struct IT_sh dummy;
418 sysroff_swap_sh_in (&dummy);
419 sysroff_print_sh_out (&dummy);
421 break;
423 case IT_ob_CODE:
425 struct IT_ob dummy;
426 sysroff_swap_ob_in (&dummy);
427 sysroff_print_ob_out (&dummy);
429 break;
431 case IT_rl_CODE:
433 struct IT_rl dummy;
434 sysroff_swap_rl_in (&dummy);
435 sysroff_print_rl_out (&dummy);
437 break;
439 case IT_du_CODE:
441 struct IT_du dummy;
442 sysroff_swap_du_in (&dummy);
444 sysroff_print_du_out (&dummy);
446 break;
448 case IT_dus_CODE:
450 struct IT_dus dummy;
451 sysroff_swap_dus_in (&dummy);
452 sysroff_print_dus_out (&dummy);
454 break;
456 case IT_dul_CODE:
458 struct IT_dul dummy;
459 sysroff_swap_dul_in (&dummy);
460 sysroff_print_dul_out (&dummy);
462 break;
464 case IT_dss_CODE:
466 struct IT_dss dummy;
467 sysroff_swap_dss_in (&dummy);
468 sysroff_print_dss_out (&dummy);
470 break;
472 case IT_hs_CODE:
474 struct IT_hs dummy;
475 sysroff_swap_hs_in (&dummy);
476 sysroff_print_hs_out (&dummy);
478 break;
480 case IT_dps_CODE:
482 struct IT_dps dummy;
483 sysroff_swap_dps_in (&dummy);
484 sysroff_print_dps_out (&dummy);
486 break;
488 case IT_tr_CODE:
489 sysroff_swap_tr_in ();
490 sysroff_print_tr_out ();
491 break;
493 case IT_dds_CODE:
495 struct IT_dds dummy;
497 sysroff_swap_dds_in (&dummy);
498 sysroff_print_dds_out (&dummy);
500 break;
502 default:
503 printf ("GOT A %x\n", c);
504 return 0;
505 break;
508 return 1;
511 static int
512 opt (int x)
514 return getone (x);
517 static void
518 must (int x)
520 if (!getone (x))
521 printf ("WANTED %x!!\n", x);
524 static void
525 tab (int i, char *s)
527 indent += i;
529 if (s)
531 p ();
532 puts (s);
536 static void
537 dump_symbol_info (void)
539 tab (1, "SYMBOL INFO");
541 while (opt (IT_dsy_CODE))
543 if (opt (IT_dty_CODE))
545 must (IT_dbt_CODE);
546 derived_type ();
547 must (IT_dty_CODE);
551 tab (-1, "");
554 static void
555 derived_type (void)
557 tab (1, "DERIVED TYPE");
559 while (1)
561 if (opt (IT_dpp_CODE))
563 dump_symbol_info ();
564 must (IT_dpp_CODE);
566 else if (opt (IT_dfp_CODE))
568 dump_symbol_info ();
569 must (IT_dfp_CODE);
571 else if (opt (IT_den_CODE))
573 dump_symbol_info ();
574 must (IT_den_CODE);
576 else if (opt (IT_den_CODE))
578 dump_symbol_info ();
579 must (IT_den_CODE);
581 else if (opt (IT_dds_CODE))
583 dump_symbol_info ();
584 must (IT_dds_CODE);
586 else if (opt (IT_dar_CODE))
589 else if (opt (IT_dpt_CODE))
592 else if (opt (IT_dul_CODE))
595 else if (opt (IT_dse_CODE))
598 else if (opt (IT_dot_CODE))
601 else
602 break;
605 tab (-1, "");
608 static void
609 module (void)
611 int c = 0;
612 int l = 0;
614 tab (1, "MODULE***\n");
618 c = getc (file);
619 ungetc (c, file);
621 c &= 0x7f;
623 while (getone (c) && c != IT_tr_CODE);
625 tab (-1, "");
627 c = getc (file);
628 while (c != EOF)
630 printf ("%02x ", c);
631 l++;
632 if (l == 32)
634 printf ("\n");
635 l = 0;
637 c = getc (file);
641 char *program_name;
643 static void
644 show_usage (FILE *file, int status)
646 fprintf (file, _("Usage: %s [option(s)] in-file\n"), program_name);
647 fprintf (file, _("Print a human readable interpretation of a SYSROFF object file\n"));
648 fprintf (file, _(" The options are:\n\
649 -h --help Display this information\n\
650 -v --version Print the program's version number\n"));
652 if (REPORT_BUGS_TO[0] && status == 0)
653 fprintf (file, _("Report bugs to %s\n"), REPORT_BUGS_TO);
654 exit (status);
658 main (int ac, char **av)
660 char *input_file = NULL;
661 int opt;
662 static struct option long_options[] =
664 {"help", no_argument, 0, 'h'},
665 {"version", no_argument, 0, 'V'},
666 {NULL, no_argument, 0, 0}
669 #if defined (HAVE_SETLOCALE) && defined (HAVE_LC_MESSAGES)
670 setlocale (LC_MESSAGES, "");
671 #endif
672 #if defined (HAVE_SETLOCALE)
673 setlocale (LC_CTYPE, "");
674 #endif
675 bindtextdomain (PACKAGE, LOCALEDIR);
676 textdomain (PACKAGE);
678 program_name = av[0];
679 xmalloc_set_program_name (program_name);
681 expandargv (&ac, &av);
683 while ((opt = getopt_long (ac, av, "HhVv", long_options, (int *) NULL)) != EOF)
685 switch (opt)
687 case 'H':
688 case 'h':
689 show_usage (stdout, 0);
690 /*NOTREACHED*/
691 case 'v':
692 case 'V':
693 print_version ("sysdump");
694 exit (0);
695 /*NOTREACHED*/
696 case 0:
697 break;
698 default:
699 show_usage (stderr, 1);
700 /*NOTREACHED*/
704 /* The input and output files may be named on the command line. */
706 if (optind < ac)
707 input_file = av[optind];
709 if (!input_file)
710 fatal (_("no input file specified"));
712 file = fopen (input_file, FOPEN_RB);
714 if (!file)
715 fatal (_("cannot open input file %s"), input_file);
717 module ();
718 return 0;