2012-04-20 Tristan Gingold <gingold@adacore.com>
[platform/upstream/binutils.git] / gas / listing.c
1 /* listing.c - maintain assembly listings
2    Copyright 1991, 1992, 1993, 1994, 1995, 1996, 1997, 1998, 1999, 2000,
3    2001, 2002, 2003, 2005, 2006, 2007, 2008, 2009, 2010
4    Free Software Foundation, Inc.
5
6    This file is part of GAS, the GNU Assembler.
7
8    GAS is free software; you can redistribute it and/or modify
9    it under the terms of the GNU General Public License as published by
10    the Free Software Foundation; either version 3, or (at your option)
11    any later version.
12
13    GAS is distributed in the hope that it will be useful,
14    but WITHOUT ANY WARRANTY; without even the implied warranty of
15    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
16    GNU General Public License for more details.
17
18    You should have received a copy of the GNU General Public License
19    along with GAS; see the file COPYING.  If not, write to the Free
20    Software Foundation, 51 Franklin Street - Fifth Floor, Boston, MA
21    02110-1301, USA.  */
22
23 /* Contributed by Steve Chamberlain <sac@cygnus.com>
24
25  A listing page looks like:
26
27  LISTING_HEADER  sourcefilename pagenumber
28  TITLE LINE
29  SUBTITLE LINE
30  linenumber address data  source
31  linenumber address data  source
32  linenumber address data  source
33  linenumber address data  source
34
35  If not overridden, the listing commands are:
36
37  .title  "stuff"
38         Put "stuff" onto the title line
39  .sbttl  "stuff"
40         Put stuff onto the subtitle line
41
42   If these commands come within 10 lines of the top of the page, they
43   will affect the page they are on, as well as any subsequent page
44
45  .eject
46         Thow a page
47  .list
48         Increment the enable listing counter
49  .nolist
50         Decrement the enable listing counter
51
52  .psize Y[,X]
53         Set the paper size to X wide and Y high. Setting a psize Y of
54         zero will suppress form feeds except where demanded by .eject
55
56  If the counter goes below zero, listing is suppressed.
57
58  Listings are a maintained by read calling various listing_<foo>
59  functions.  What happens most is that the macro NO_LISTING is not
60  defined (from the Makefile), then the macro LISTING_NEWLINE expands
61  into a call to listing_newline.  The call is done from read.c, every
62  time it sees a newline, and -l is on the command line.
63
64  The function listing_newline remembers the frag associated with the
65  newline, and creates a new frag - note that this is wasteful, but not
66  a big deal, since listing slows things down a lot anyway.  The
67  function also remembers when the filename changes.
68
69  When all the input has finished, and gas has had a chance to settle
70  down, the listing is output. This is done by running down the list of
71  frag/source file records, and opening the files as needed and printing
72  out the bytes and chars associated with them.
73
74  The only things which the architecture can change about the listing
75  are defined in these macros:
76
77  LISTING_HEADER         The name of the architecture
78  LISTING_WORD_SIZE      The make of the number of bytes in a word, this determines
79                         the clumping of the output data. eg a value of
80                         2 makes words look like 1234 5678, whilst 1
81                         would make the same value look like 12 34 56
82                         78
83  LISTING_LHS_WIDTH      Number of words of above size for the lhs
84
85  LISTING_LHS_WIDTH_SECOND   Number of words for the data on the lhs
86                         for the second line
87
88  LISTING_LHS_CONT_LINES Max number of lines to use up for a continuation
89  LISTING_RHS_WIDTH      Number of chars from the input file to print
90                         on a line.  */
91
92 #include "as.h"
93 #include "filenames.h"
94 #include "obstack.h"
95 #include "safe-ctype.h"
96 #include "input-file.h"
97 #include "subsegs.h"
98 #include "bfdver.h"
99 #include <time.h>
100 #include <stdarg.h>
101
102 #ifndef NO_LISTING
103
104 #ifndef LISTING_HEADER
105 #define LISTING_HEADER "GAS LISTING"
106 #endif
107 #ifndef LISTING_WORD_SIZE
108 #define LISTING_WORD_SIZE 4
109 #endif
110 #ifndef LISTING_LHS_WIDTH
111 #define LISTING_LHS_WIDTH ((LISTING_WORD_SIZE) > 4 ? 1 : 4 / (LISTING_WORD_SIZE))
112 #endif
113 #ifndef LISTING_LHS_WIDTH_SECOND
114 #define LISTING_LHS_WIDTH_SECOND LISTING_LHS_WIDTH
115 #endif
116 #ifndef LISTING_RHS_WIDTH
117 #define LISTING_RHS_WIDTH 100
118 #endif
119 #ifndef LISTING_LHS_CONT_LINES
120 #define LISTING_LHS_CONT_LINES 4
121 #endif
122 #define MAX_DATELEN 30
123
124 /* This structure remembers which .s were used.  */
125 typedef struct file_info_struct
126 {
127   struct file_info_struct * next;
128   char *                    filename;
129   long                      pos;
130   unsigned int              linenum;
131   int                       at_end;
132 } file_info_type;
133
134 enum edict_enum
135 {
136   EDICT_NONE,
137   EDICT_SBTTL,
138   EDICT_TITLE,
139   EDICT_NOLIST,
140   EDICT_LIST,
141   EDICT_NOLIST_NEXT,
142   EDICT_EJECT
143 };
144
145
146 /* This structure remembers which line from which file goes into which
147    frag.  */
148 struct list_info_struct
149 {
150   /* Frag which this line of source is nearest to.  */
151   fragS *frag;
152
153   /* The actual line in the source file.  */
154   unsigned int line;
155
156   /* Pointer to the file info struct for the file which this line
157      belongs to.  */
158   file_info_type *file;
159
160   /* The expanded text of any macro that may have been executing.  */
161   char *line_contents;
162
163   /* Next in list.  */
164   struct list_info_struct *next;
165
166   /* Pointer to the file info struct for the high level language
167      source line that belongs here.  */
168   file_info_type *hll_file;
169
170   /* High level language source line.  */
171   unsigned int hll_line;
172
173   /* Pointer to any error message associated with this line.  */
174   char *message;
175
176   enum edict_enum edict;
177   char *edict_arg;
178
179   /* Nonzero if this line is to be omitted because it contains
180      debugging information.  This can become a flags field if we come
181      up with more information to store here.  */
182   int debugging;
183 };
184
185 typedef struct list_info_struct list_info_type;
186
187 int listing_lhs_width        = LISTING_LHS_WIDTH;
188 int listing_lhs_width_second = LISTING_LHS_WIDTH_SECOND;
189 int listing_lhs_cont_lines   = LISTING_LHS_CONT_LINES;
190 int listing_rhs_width        = LISTING_RHS_WIDTH;
191
192 struct list_info_struct *        listing_tail;
193
194 static file_info_type *          file_info_head;
195 static file_info_type *          last_open_file_info;
196 static FILE *                    last_open_file;
197 static struct list_info_struct * head;
198 static int                       paper_width = 200;
199 static int                       paper_height = 60;
200
201 extern int                       listing;
202
203 /* File to output listings to.  */
204 static FILE *list_file;
205
206 /* This static array is used to keep the text of data to be printed
207    before the start of the line.  */
208
209 #define MAX_BYTES                                                       \
210   (((LISTING_WORD_SIZE * 2) + 1) * listing_lhs_width                    \
211    + ((((LISTING_WORD_SIZE * 2) + 1) * listing_lhs_width_second)        \
212       * listing_lhs_cont_lines)                                         \
213    + 20)
214
215 static char *data_buffer;
216
217 /* Prototypes.  */
218 static void listing_message (const char *, const char *);
219 static file_info_type *file_info (const char *);
220 static void new_frag (void);
221 static void listing_page (list_info_type *);
222 static unsigned int calc_hex (list_info_type *);
223 static void print_lines (list_info_type *, unsigned int, char *, unsigned int);
224 static void list_symbol_table (void);
225 static int debugging_pseudo (list_info_type *, const char *);
226 static void listing_listing (char *);
227
228 static void
229 listing_message (const char *name, const char *message)
230 {
231   if (listing_tail != (list_info_type *) NULL)
232     {
233       unsigned int l = strlen (name) + strlen (message) + 1;
234       char *n = (char *) xmalloc (l);
235       strcpy (n, name);
236       strcat (n, message);
237       listing_tail->message = n;
238     }
239 }
240
241 void
242 listing_warning (const char *message)
243 {
244   listing_message (_("Warning:"), message);
245 }
246
247 void
248 listing_error (const char *message)
249 {
250   listing_message (_("Error:"), message);
251 }
252
253 static file_info_type *
254 file_info (const char *file_name)
255 {
256   /* Find an entry with this file name.  */
257   file_info_type *p = file_info_head;
258
259   while (p != (file_info_type *) NULL)
260     {
261       if (filename_cmp (p->filename, file_name) == 0)
262         return p;
263       p = p->next;
264     }
265
266   /* Make new entry.  */
267   p = (file_info_type *) xmalloc (sizeof (file_info_type));
268   p->next = file_info_head;
269   file_info_head = p;
270   p->filename = xstrdup (file_name);
271   p->pos = 0;
272   p->linenum = 0;
273   p->at_end = 0;
274
275   return p;
276 }
277
278 static void
279 new_frag (void)
280 {
281   frag_wane (frag_now);
282   frag_new (0);
283 }
284
285 void
286 listing_newline (char *ps)
287 {
288   char *file;
289   unsigned int line;
290   static unsigned int last_line = 0xffff;
291   static char *last_file = NULL;
292   list_info_type *new_i = NULL;
293
294   if (listing == 0)
295     return;
296
297   if (now_seg == absolute_section)
298     return;
299
300 #ifdef OBJ_ELF
301   /* In ELF, anything in a section beginning with .debug or .line is
302      considered to be debugging information.  This includes the
303      statement which switches us into the debugging section, which we
304      can only set after we are already in the debugging section.  */
305   if ((listing & LISTING_NODEBUG) != 0
306       && listing_tail != NULL
307       && ! listing_tail->debugging)
308     {
309       const char *segname;
310
311       segname = segment_name (now_seg);
312       if (strncmp (segname, ".debug", sizeof ".debug" - 1) == 0
313           || strncmp (segname, ".line", sizeof ".line" - 1) == 0)
314         listing_tail->debugging = 1;
315     }
316 #endif
317
318   as_where (&file, &line);
319   if (ps == NULL)
320     {
321       if (line == last_line
322           && !(last_file && file && filename_cmp (file, last_file)))
323         return;
324
325       new_i = (list_info_type *) xmalloc (sizeof (list_info_type));
326
327       /* Detect if we are reading from stdin by examining the file
328          name returned by as_where().
329
330          [FIXME: We rely upon the name in the strcmp below being the
331          same as the one used by input_scrub_new_file(), if that is
332          not true, then this code will fail].
333
334          If we are reading from stdin, then we need to save each input
335          line here (assuming of course that we actually have a line of
336          input to read), so that it can be displayed in the listing
337          that is produced at the end of the assembly.  */
338       if (strcmp (file, _("{standard input}")) == 0
339           && input_line_pointer != NULL)
340         {
341           char *copy;
342           int len;
343           int seen_quote = 0;
344           int seen_slash = 0;
345
346           for (copy = input_line_pointer;
347                *copy && (seen_quote
348                          || is_end_of_line [(unsigned char) *copy] != 1);
349                copy++)
350             {
351               if (seen_slash)
352                 seen_slash = 0;
353               else if (*copy == '\\')
354                 seen_slash = 1;
355               else if (*copy == '"')
356                 seen_quote = !seen_quote;
357             }
358
359           len = copy - input_line_pointer + 1;
360
361           copy = (char *) xmalloc (len);
362
363           if (copy != NULL)
364             {
365               char *src = input_line_pointer;
366               char *dest = copy;
367
368               while (--len)
369                 {
370                   unsigned char c = *src++;
371
372                   /* Omit control characters in the listing.  */
373                   if (!ISCNTRL (c))
374                     *dest++ = c;
375                 }
376
377               *dest = 0;
378             }
379
380           new_i->line_contents = copy;
381         }
382       else
383         new_i->line_contents = NULL;
384     }
385   else
386     {
387       new_i = (list_info_type *) xmalloc (sizeof (list_info_type));
388       new_i->line_contents = ps;
389     }
390
391   last_line = line;
392   last_file = file;
393
394   new_frag ();
395
396   if (listing_tail)
397     listing_tail->next = new_i;
398   else
399     head = new_i;
400
401   listing_tail = new_i;
402
403   new_i->frag = frag_now;
404   new_i->line = line;
405   new_i->file = file_info (file);
406   new_i->next = (list_info_type *) NULL;
407   new_i->message = (char *) NULL;
408   new_i->edict = EDICT_NONE;
409   new_i->hll_file = (file_info_type *) NULL;
410   new_i->hll_line = 0;
411   new_i->debugging = 0;
412
413   new_frag ();
414
415 #ifdef OBJ_ELF
416   /* In ELF, anything in a section beginning with .debug or .line is
417      considered to be debugging information.  */
418   if ((listing & LISTING_NODEBUG) != 0)
419     {
420       const char *segname;
421
422       segname = segment_name (now_seg);
423       if (strncmp (segname, ".debug", sizeof ".debug" - 1) == 0
424           || strncmp (segname, ".line", sizeof ".line" - 1) == 0)
425         new_i->debugging = 1;
426     }
427 #endif
428 }
429
430 /* Attach all current frags to the previous line instead of the
431    current line.  This is called by the MIPS backend when it discovers
432    that it needs to add some NOP instructions; the added NOP
433    instructions should go with the instruction that has the delay, not
434    with the new instruction.  */
435
436 void
437 listing_prev_line (void)
438 {
439   list_info_type *l;
440   fragS *f;
441
442   if (head == (list_info_type *) NULL
443       || head == listing_tail)
444     return;
445
446   new_frag ();
447
448   for (l = head; l->next != listing_tail; l = l->next)
449     ;
450
451   for (f = frchain_now->frch_root; f != (fragS *) NULL; f = f->fr_next)
452     if (f->line == listing_tail)
453       f->line = l;
454
455   listing_tail->frag = frag_now;
456   new_frag ();
457 }
458
459 /* This function returns the next source line from the file supplied,
460    truncated to size.  It appends a fake line to the end of each input
461    file to make using the returned buffer simpler.  */
462
463 static char *
464 buffer_line (file_info_type *file, char *line, unsigned int size)
465 {
466   unsigned int count = 0;
467   int c;
468   char *p = line;
469
470   /* If we couldn't open the file, return an empty line.  */
471   if (file->at_end)
472     return "";
473
474   /* Check the cache and see if we last used this file.  */
475   if (!last_open_file_info || file != last_open_file_info)
476     {
477       if (last_open_file)
478         {
479           last_open_file_info->pos = ftell (last_open_file);
480           fclose (last_open_file);
481         }
482
483       /* Open the file in the binary mode so that ftell above can
484          return a reliable value that we can feed to fseek below.  */
485       last_open_file_info = file;
486       last_open_file = fopen (file->filename, FOPEN_RB);
487       if (last_open_file == NULL)
488         {
489           file->at_end = 1;
490           return "";
491         }
492
493       /* Seek to where we were last time this file was open.  */
494       if (file->pos)
495         fseek (last_open_file, file->pos, SEEK_SET);
496     }
497
498   /* Leave room for null.  */
499   size -= 1;
500
501   c = fgetc (last_open_file);
502
503   while (c != EOF && c != '\n' && c != '\r')
504     {
505       if (count < size)
506         *p++ = c;
507       count++;
508
509       c = fgetc (last_open_file);
510     }
511
512   /* If '\r' is followed by '\n', swallow that.  Likewise, if '\n'
513      is followed by '\r', swallow that as well.  */
514   if (c == '\r' || c == '\n')
515     {
516       int next = fgetc (last_open_file);
517
518       if ((c == '\r' && next != '\n')
519           || (c == '\n' && next != '\r'))
520         ungetc (next, last_open_file);
521     }
522
523   if (c == EOF)
524     {
525       file->at_end = 1;
526       if (count + 2 < size)
527         {
528           *p++ = '.';
529           *p++ = '.';
530           *p++ = '.';
531         }
532     }
533   file->linenum++;
534   *p++ = 0;
535   return line;
536 }
537
538
539 /* This function rewinds the requested file back to the line requested,
540    reads it in again into the buffer provided and then restores the file
541    back to its original location.  */
542
543 static char *
544 rebuffer_line (file_info_type *  file,
545                unsigned int      linenum,
546                char *            buffer,
547                unsigned int      size)
548 {
549   unsigned int count = 0;
550   unsigned int current_line = 1;
551   char * p = buffer;
552   long pos;
553   int c;
554
555   /* Sanity checks.  */
556   if (file == NULL || buffer == NULL || size == 0 || file->linenum <= linenum)
557     return "";
558
559   /* Check the cache and see if we last used this file.  */
560   if (last_open_file_info == NULL || file != last_open_file_info)
561     {
562       if (last_open_file)
563         {
564           last_open_file_info->pos = ftell (last_open_file);
565           fclose (last_open_file);
566         }
567
568       /* Open the file in the binary mode so that ftell above can
569          return a reliable value that we can feed to fseek below.  */
570       last_open_file_info = file;
571       last_open_file = fopen (file->filename, FOPEN_RB);
572       if (last_open_file == NULL)
573         {
574           file->at_end = 1;
575           return "";
576         }
577
578       /* Seek to where we were last time this file was open.  */
579       if (file->pos)
580         fseek (last_open_file, file->pos, SEEK_SET);
581     }
582
583   /* Remember where we are in the current file.  */
584   pos = ftell (last_open_file);
585
586   /* Go back to the beginning.  */
587   fseek (last_open_file, 0, SEEK_SET);
588
589   /* Skip lines prior to the one we are interested in.  */
590   while (current_line < linenum)
591     {
592       /* fgets only stops on newlines and has a size limit,
593          so we read one character at a time instead.  */
594       do
595         {
596           c = fgetc (last_open_file);
597         }
598       while (c != EOF && c != '\n' && c != '\r');
599
600       ++ current_line;
601
602       if (c == '\r' || c == '\n')
603         {
604           int next = fgetc (last_open_file);
605
606           /* If '\r' is followed by '\n', swallow that.  Likewise, if '\n'
607              is followed by '\r', swallow that as well.  */
608           if ((c == '\r' && next != '\n')
609               || (c == '\n' && next != '\r'))
610             ungetc (next, last_open_file);
611         }
612     }
613
614   /* Leave room for the nul at the end of the buffer.  */
615   size -= 1;
616
617   /* Read in the line.  */
618   c = fgetc (last_open_file);
619
620   while (c != EOF && c != '\n' && c != '\r')
621     {
622       if (count < size)
623         *p++ = c;
624       count++;
625
626       c = fgetc (last_open_file);
627     }
628
629   /* If '\r' is followed by '\n', swallow that.  Likewise, if '\n'
630      is followed by '\r', swallow that as well.  */
631   if (c == '\r' || c == '\n')
632     {
633       int next = fgetc (last_open_file);
634
635       if ((c == '\r' && next != '\n')
636           || (c == '\n' && next != '\r'))
637         ungetc (next, last_open_file);
638     }
639
640   /* Terminate the line.  */
641   *p++ = 0;
642
643   /* Reset the file position.  */
644   fseek (last_open_file, pos, SEEK_SET);
645
646   return buffer;
647 }
648
649 static const char *fn;
650
651 static unsigned int eject;      /* Eject pending */
652 static unsigned int page;       /* Current page number */
653 static char *title;             /* Current title */
654 static char *subtitle;          /* Current subtitle */
655 static unsigned int on_page;    /* Number of lines printed on current page */
656
657 static void
658 listing_page (list_info_type *list)
659 {
660   /* Grope around, see if we can see a title or subtitle edict coming up
661      soon.  (we look down 10 lines of the page and see if it's there)  */
662   if ((eject || (on_page >= (unsigned int) paper_height))
663       && paper_height != 0)
664     {
665       unsigned int c = 10;
666       int had_title = 0;
667       int had_subtitle = 0;
668
669       page++;
670
671       while (c != 0 && list)
672         {
673           if (list->edict == EDICT_SBTTL && !had_subtitle)
674             {
675               had_subtitle = 1;
676               subtitle = list->edict_arg;
677             }
678           if (list->edict == EDICT_TITLE && !had_title)
679             {
680               had_title = 1;
681               title = list->edict_arg;
682             }
683           list = list->next;
684           c--;
685         }
686
687       if (page > 1)
688         {
689           fprintf (list_file, "\f");
690         }
691
692       fprintf (list_file, "%s %s \t\t\tpage %d\n", LISTING_HEADER, fn, page);
693       fprintf (list_file, "%s\n", title);
694       fprintf (list_file, "%s\n", subtitle);
695       on_page = 3;
696       eject = 0;
697     }
698 }
699
700 /* Print a line into the list_file.  Update the line count
701    and if necessary start a new page.  */
702
703 static void
704 emit_line (list_info_type * list, const char * format, ...)
705 {
706   va_list args;
707
708   va_start (args, format);
709
710   vfprintf (list_file, format, args);
711   on_page++;
712   listing_page (list);
713
714   va_end (args);
715 }
716
717 static unsigned int
718 calc_hex (list_info_type *list)
719 {
720   int data_buffer_size;
721   list_info_type *first = list;
722   unsigned int address = ~(unsigned int) 0;
723   fragS *frag;
724   fragS *frag_ptr;
725   unsigned int octet_in_frag;
726
727   /* Find first frag which says it belongs to this line.  */
728   frag = list->frag;
729   while (frag && frag->line != list)
730     frag = frag->fr_next;
731
732   frag_ptr = frag;
733
734   data_buffer_size = 0;
735
736   /* Dump all the frags which belong to this line.  */
737   while (frag_ptr != (fragS *) NULL && frag_ptr->line == first)
738     {
739       /* Print as many bytes from the fixed part as is sensible.  */
740       octet_in_frag = 0;
741       while ((offsetT) octet_in_frag < frag_ptr->fr_fix
742              && data_buffer_size < MAX_BYTES - 3)
743         {
744           if (address == ~(unsigned int) 0)
745             address = frag_ptr->fr_address / OCTETS_PER_BYTE;
746
747           sprintf (data_buffer + data_buffer_size,
748                    "%02X",
749                    (frag_ptr->fr_literal[octet_in_frag]) & 0xff);
750           data_buffer_size += 2;
751           octet_in_frag++;
752         }
753       if (frag_ptr->fr_type == rs_fill)
754         {
755           unsigned int var_rep_max = octet_in_frag;
756           unsigned int var_rep_idx = octet_in_frag;
757
758           /* Print as many bytes from the variable part as is sensible.  */
759           while (((offsetT) octet_in_frag
760                   < (frag_ptr->fr_fix + frag_ptr->fr_var * frag_ptr->fr_offset))
761                  && data_buffer_size < MAX_BYTES - 3)
762             {
763               if (address == ~(unsigned int) 0)
764                 address = frag_ptr->fr_address / OCTETS_PER_BYTE;
765
766               sprintf (data_buffer + data_buffer_size,
767                        "%02X",
768                        (frag_ptr->fr_literal[var_rep_idx]) & 0xff);
769               data_buffer_size += 2;
770
771               var_rep_idx++;
772               octet_in_frag++;
773
774               if ((offsetT) var_rep_idx >= frag_ptr->fr_fix + frag_ptr->fr_var)
775                 var_rep_idx = var_rep_max;
776             }
777         }
778
779       frag_ptr = frag_ptr->fr_next;
780     }
781   data_buffer[data_buffer_size] = '\0';
782   return address;
783 }
784
785 static void
786 print_lines (list_info_type *list, unsigned int lineno,
787              char *string, unsigned int address)
788 {
789   unsigned int idx;
790   unsigned int nchars;
791   unsigned int lines;
792   unsigned int octet_in_word = 0;
793   char *src = data_buffer;
794   int cur;
795
796   /* Print the stuff on the first line.  */
797   listing_page (list);
798   nchars = (LISTING_WORD_SIZE * 2 + 1) * listing_lhs_width;
799
800   /* Print the hex for the first line.  */
801   if (address == ~(unsigned int) 0)
802     {
803       fprintf (list_file, "% 4d     ", lineno);
804       for (idx = 0; idx < nchars; idx++)
805         fprintf (list_file, " ");
806
807       emit_line (NULL, "\t%s\n", string ? string : "");
808       return;
809     }
810
811   if (had_errors ())
812     fprintf (list_file, "% 4d ???? ", lineno);
813   else
814     fprintf (list_file, "% 4d %04x ", lineno, address);
815
816   /* And the data to go along with it.  */
817   idx = 0;
818   cur = 0;
819   while (src[cur] && idx < nchars)
820     {
821       int offset;
822       offset = cur;
823       fprintf (list_file, "%c%c", src[offset], src[offset + 1]);
824       cur += 2;
825       octet_in_word++;
826
827       if (octet_in_word == LISTING_WORD_SIZE)
828         {
829           fprintf (list_file, " ");
830           idx++;
831           octet_in_word = 0;
832         }
833
834       idx += 2;
835     }
836
837   for (; idx < nchars; idx++)
838     fprintf (list_file, " ");
839
840   emit_line (list, "\t%s\n", string ? string : "");
841
842   if (list->message)
843     emit_line (list, "****  %s\n", list->message);
844
845   for (lines = 0;
846        lines < (unsigned int) listing_lhs_cont_lines
847          && src[cur];
848        lines++)
849     {
850       nchars = ((LISTING_WORD_SIZE * 2) + 1) * listing_lhs_width_second - 1;
851       idx = 0;
852
853       /* Print any more lines of data, but more compactly.  */
854       fprintf (list_file, "% 4d      ", lineno);
855
856       while (src[cur] && idx < nchars)
857         {
858           int offset;
859           offset = cur;
860           fprintf (list_file, "%c%c", src[offset], src[offset + 1]);
861           cur += 2;
862           idx += 2;
863           octet_in_word++;
864
865           if (octet_in_word == LISTING_WORD_SIZE)
866             {
867               fprintf (list_file, " ");
868               idx++;
869               octet_in_word = 0;
870             }
871         }
872
873       emit_line (list, "\n");
874     }
875 }
876
877 static void
878 list_symbol_table (void)
879 {
880   extern symbolS *symbol_rootP;
881   int got_some = 0;
882
883   symbolS *ptr;
884   eject = 1;
885   listing_page (NULL);
886
887   for (ptr = symbol_rootP; ptr != (symbolS *) NULL; ptr = symbol_next (ptr))
888     {
889       if (SEG_NORMAL (S_GET_SEGMENT (ptr))
890           || S_GET_SEGMENT (ptr) == absolute_section)
891         {
892           /* Don't report section symbols.  They are not interesting.  */
893           if (symbol_section_p (ptr))
894             continue;
895
896           if (S_GET_NAME (ptr))
897             {
898               char buf[30], fmt[8];
899               valueT val = S_GET_VALUE (ptr);
900
901               /* @@ Note that this is dependent on the compilation options,
902                  not solely on the target characteristics.  */
903               if (sizeof (val) == 4 && sizeof (int) == 4)
904                 sprintf (buf, "%08lx", (unsigned long) val);
905               else if (sizeof (val) <= sizeof (unsigned long))
906                 {
907                   sprintf (fmt, "%%0%lulx",
908                            (unsigned long) (sizeof (val) * 2));
909                   sprintf (buf, fmt, (unsigned long) val);
910                 }
911 #if defined (BFD64)
912               else if (sizeof (val) > 4)
913                 sprintf_vma (buf, val);
914 #endif
915               else
916                 abort ();
917
918               if (!got_some)
919                 {
920                   fprintf (list_file, "DEFINED SYMBOLS\n");
921                   on_page++;
922                   got_some = 1;
923                 }
924
925               if (symbol_get_frag (ptr) && symbol_get_frag (ptr)->line)
926                 {
927                   fprintf (list_file, "%20s:%-5d  %s:%s %s\n",
928                            symbol_get_frag (ptr)->line->file->filename,
929                            symbol_get_frag (ptr)->line->line,
930                            segment_name (S_GET_SEGMENT (ptr)),
931                            buf, S_GET_NAME (ptr));
932                 }
933               else
934                 {
935                   fprintf (list_file, "%33s:%s %s\n",
936                            segment_name (S_GET_SEGMENT (ptr)),
937                            buf, S_GET_NAME (ptr));
938                 }
939
940               on_page++;
941               listing_page (NULL);
942             }
943         }
944
945     }
946   if (!got_some)
947     {
948       fprintf (list_file, "NO DEFINED SYMBOLS\n");
949       on_page++;
950     }
951   emit_line (NULL, "\n");
952
953   got_some = 0;
954
955   for (ptr = symbol_rootP; ptr != (symbolS *) NULL; ptr = symbol_next (ptr))
956     {
957       if (S_GET_NAME (ptr) && strlen (S_GET_NAME (ptr)) != 0)
958         {
959           if (S_GET_SEGMENT (ptr) == undefined_section)
960             {
961               if (!got_some)
962                 {
963                   got_some = 1;
964
965                   emit_line (NULL, "UNDEFINED SYMBOLS\n");
966                 }
967
968               emit_line (NULL, "%s\n", S_GET_NAME (ptr));
969             }
970         }
971     }
972
973   if (!got_some)
974     emit_line (NULL, "NO UNDEFINED SYMBOLS\n");
975 }
976
977 typedef struct cached_line
978 {
979   file_info_type * file;
980   unsigned int     line;
981   char             buffer [LISTING_RHS_WIDTH];
982 } cached_line;
983
984 static void
985 print_source (file_info_type *  current_file,
986               list_info_type *  list,
987               unsigned int      width)
988 {
989 #define NUM_CACHE_LINES  3
990   static cached_line cached_lines[NUM_CACHE_LINES];
991   static int next_free_line = 0;
992   cached_line * cache = NULL;
993
994   if (current_file->linenum > list->hll_line
995       && list->hll_line > 0)
996     {
997       /* This can happen with modern optimizing compilers.  The source
998          lines from the high level language input program are split up
999          and interleaved, meaning the line number we want to display
1000          (list->hll_line) can have already been displayed.  We have
1001          three choices:
1002
1003            a. Do nothing, since we have already displayed the source
1004               line.  This was the old behaviour.
1005
1006            b. Display the particular line requested again, but only
1007               that line.  This is the new behaviour.
1008
1009            c. Display the particular line requested again and reset
1010               the current_file->line_num value so that we redisplay
1011               all the following lines as well the next time we
1012               encounter a larger line number.  */
1013       int i;
1014
1015       /* Check the cache, maybe we already have the line saved.  */
1016       for (i = 0; i < NUM_CACHE_LINES; i++)
1017         if (cached_lines[i].file == current_file
1018             && cached_lines[i].line == list->hll_line)
1019           {
1020             cache = cached_lines + i;
1021             break;
1022           }
1023
1024       if (i == NUM_CACHE_LINES)
1025         {
1026           cache = cached_lines + next_free_line;
1027           next_free_line ++;
1028           if (next_free_line == NUM_CACHE_LINES)
1029             next_free_line = 0;
1030
1031           cache->file = current_file;
1032           cache->line = list->hll_line;
1033           cache->buffer[0] = 0;
1034           rebuffer_line (current_file, cache->line, cache->buffer, width);
1035         }
1036
1037       emit_line (list, "%4u:%-13s **** %s\n",
1038                  cache->line, cache->file->filename, cache->buffer);
1039       return;
1040     }
1041
1042   if (!current_file->at_end)
1043     {
1044       int num_lines_shown = 0;
1045
1046       while (current_file->linenum < list->hll_line
1047              && !current_file->at_end)
1048         {
1049           char *p;
1050
1051           cache = cached_lines + next_free_line;
1052           cache->file = current_file;
1053           cache->line = current_file->linenum + 1;
1054           cache->buffer[0] = 0;
1055           p = buffer_line (current_file, cache->buffer, width);
1056
1057           /* Cache optimization:  If printing a group of lines
1058              cache the first and last lines in the group.  */
1059           if (num_lines_shown == 0)
1060             {
1061               next_free_line ++;
1062               if (next_free_line == NUM_CACHE_LINES)
1063                 next_free_line = 0;
1064             }
1065
1066           emit_line (list, "%4u:%-13s **** %s\n",
1067                      cache->line, cache->file->filename, p);
1068           num_lines_shown ++;
1069         }
1070     }
1071 }
1072
1073 /* Sometimes the user doesn't want to be bothered by the debugging
1074    records inserted by the compiler, see if the line is suspicious.  */
1075
1076 static int
1077 debugging_pseudo (list_info_type *list, const char *line)
1078 {
1079 #ifdef OBJ_ELF
1080   static int in_debug;
1081   int was_debug;
1082 #endif
1083
1084   if (list->debugging)
1085     {
1086 #ifdef OBJ_ELF
1087       in_debug = 1;
1088 #endif
1089       return 1;
1090     }
1091 #ifdef OBJ_ELF
1092   was_debug = in_debug;
1093   in_debug = 0;
1094 #endif
1095
1096   while (ISSPACE (*line))
1097     line++;
1098
1099   if (*line != '.')
1100     {
1101 #ifdef OBJ_ELF
1102       /* The ELF compiler sometimes emits blank lines after switching
1103          out of a debugging section.  If the next line drops us back
1104          into debugging information, then don't print the blank line.
1105          This is a hack for a particular compiler behaviour, not a
1106          general case.  */
1107       if (was_debug
1108           && *line == '\0'
1109           && list->next != NULL
1110           && list->next->debugging)
1111         {
1112           in_debug = 1;
1113           return 1;
1114         }
1115 #endif
1116
1117       return 0;
1118     }
1119
1120   line++;
1121
1122   if (strncmp (line, "def", 3) == 0)
1123     return 1;
1124   if (strncmp (line, "val", 3) == 0)
1125     return 1;
1126   if (strncmp (line, "scl", 3) == 0)
1127     return 1;
1128   if (strncmp (line, "line", 4) == 0)
1129     return 1;
1130   if (strncmp (line, "endef", 5) == 0)
1131     return 1;
1132   if (strncmp (line, "ln", 2) == 0)
1133     return 1;
1134   if (strncmp (line, "type", 4) == 0)
1135     return 1;
1136   if (strncmp (line, "size", 4) == 0)
1137     return 1;
1138   if (strncmp (line, "dim", 3) == 0)
1139     return 1;
1140   if (strncmp (line, "tag", 3) == 0)
1141     return 1;
1142   if (strncmp (line, "stabs", 5) == 0)
1143     return 1;
1144   if (strncmp (line, "stabn", 5) == 0)
1145     return 1;
1146
1147   return 0;
1148 }
1149
1150 static void
1151 listing_listing (char *name ATTRIBUTE_UNUSED)
1152 {
1153   list_info_type *list = head;
1154   file_info_type *current_hll_file = (file_info_type *) NULL;
1155   char *buffer;
1156   char *p;
1157   int show_listing = 1;
1158   unsigned int width;
1159
1160   buffer = (char *) xmalloc (listing_rhs_width);
1161   data_buffer = (char *) xmalloc (MAX_BYTES);
1162   eject = 1;
1163   list = head->next;
1164
1165   while (list)
1166     {
1167       unsigned int list_line;
1168
1169       width = listing_rhs_width > paper_width ? paper_width :
1170         listing_rhs_width;
1171
1172       list_line = list->line;
1173       switch (list->edict)
1174         {
1175         case EDICT_LIST:
1176           /* Skip all lines up to the current.  */
1177           list_line--;
1178           break;
1179         case EDICT_NOLIST:
1180           show_listing--;
1181           break;
1182         case EDICT_NOLIST_NEXT:
1183           if (show_listing == 0)
1184             list_line--;
1185           break;
1186         case EDICT_EJECT:
1187           break;
1188         case EDICT_NONE:
1189           break;
1190         case EDICT_TITLE:
1191           title = list->edict_arg;
1192           break;
1193         case EDICT_SBTTL:
1194           subtitle = list->edict_arg;
1195           break;
1196         default:
1197           abort ();
1198         }
1199
1200       if (show_listing <= 0)
1201         {
1202           while (list->file->linenum < list_line
1203                  && !list->file->at_end)
1204             p = buffer_line (list->file, buffer, width);
1205         }
1206
1207       if (list->edict == EDICT_LIST
1208           || (list->edict == EDICT_NOLIST_NEXT && show_listing == 0))
1209         {
1210           /* Enable listing for the single line that caused the enable.  */
1211           list_line++;
1212           show_listing++;
1213         }
1214
1215       if (show_listing > 0)
1216         {
1217           /* Scan down the list and print all the stuff which can be done
1218              with this line (or lines).  */
1219           if (list->hll_file)
1220             current_hll_file = list->hll_file;
1221
1222           if (current_hll_file && list->hll_line && (listing & LISTING_HLL))
1223             print_source (current_hll_file, list, width);
1224
1225           if (list->line_contents)
1226             {
1227               if (!((listing & LISTING_NODEBUG)
1228                     && debugging_pseudo (list, list->line_contents)))
1229                 print_lines (list,
1230                              list->file->linenum == 0 ? list->line : list->file->linenum,
1231                              list->line_contents, calc_hex (list));
1232
1233               free (list->line_contents);
1234               list->line_contents = NULL;
1235             }
1236           else
1237             {
1238               while (list->file->linenum < list_line
1239                      && !list->file->at_end)
1240                 {
1241                   unsigned int address;
1242
1243                   p = buffer_line (list->file, buffer, width);
1244
1245                   if (list->file->linenum < list_line)
1246                     address = ~(unsigned int) 0;
1247                   else
1248                     address = calc_hex (list);
1249
1250                   if (!((listing & LISTING_NODEBUG)
1251                         && debugging_pseudo (list, p)))
1252                     print_lines (list, list->file->linenum, p, address);
1253                 }
1254             }
1255
1256           if (list->edict == EDICT_EJECT)
1257             eject = 1;
1258         }
1259
1260       if (list->edict == EDICT_NOLIST_NEXT && show_listing == 1)
1261         --show_listing;
1262
1263       list = list->next;
1264     }
1265
1266   free (buffer);
1267   free (data_buffer);
1268   data_buffer = NULL;
1269 }
1270
1271 /* Print time stamp in ISO format:  yyyy-mm-ddThh:mm:ss.ss+/-zzzz.  */
1272
1273 static void
1274 print_timestamp (void)
1275 {
1276   const time_t now = time (NULL);
1277   struct tm * timestamp;
1278   char stampstr[MAX_DATELEN];
1279
1280   /* Any portable way to obtain subsecond values???  */
1281   timestamp = localtime (&now);
1282   strftime (stampstr, MAX_DATELEN, "%Y-%m-%dT%H:%M:%S.000%z", timestamp);
1283   fprintf (list_file, _("\n time stamp    \t: %s\n\n"), stampstr);
1284 }
1285
1286 static void
1287 print_single_option (char * opt, int *pos)
1288 {
1289   int opt_len = strlen (opt);
1290
1291    if ((*pos + opt_len) < paper_width)
1292      {
1293         fprintf (list_file, _("%s "), opt);
1294         *pos = *pos + opt_len;
1295      }
1296    else
1297      {
1298         fprintf (list_file, _("\n\t%s "), opt);
1299         *pos = opt_len;
1300      }
1301 }
1302
1303 /* Print options passed to as.  */
1304
1305 static void
1306 print_options (char ** argv)
1307 {
1308   const char *field_name = _("\n options passed\t: ");
1309   int pos = strlen (field_name);
1310   char **p;
1311
1312   fputs (field_name, list_file);
1313   for (p = &argv[1]; *p != NULL; p++)
1314     if (**p == '-')
1315       {
1316         /* Ignore these.  */
1317         if (strcmp (*p, "-o") == 0)
1318           {
1319             if (p[1] != NULL)
1320               p++;
1321             continue;
1322           }
1323         if (strcmp (*p, "-v") == 0)
1324           continue;
1325
1326         print_single_option (*p, &pos);
1327       }
1328 }
1329
1330 /* Print a first section with basic info like file names, as version,
1331    options passed, target, and timestamp.
1332    The format of this section is as follows:
1333
1334    AS VERSION
1335
1336    fieldname TAB ':' fieldcontents
1337   { TAB fieldcontents-cont }  */
1338
1339 static void
1340 listing_general_info (char ** argv)
1341 {
1342   /* Print the stuff on the first line.  */
1343   eject = 1;
1344   listing_page (NULL);
1345
1346   fprintf (list_file,
1347            _(" GNU assembler version %s (%s)\n\t using BFD version %s."),
1348            VERSION, TARGET_ALIAS, BFD_VERSION_STRING);
1349   print_options (argv);
1350   fprintf (list_file, _("\n input file    \t: %s"), fn);
1351   fprintf (list_file, _("\n output file   \t: %s"), out_file_name);
1352   fprintf (list_file, _("\n target        \t: %s"), TARGET_CANONICAL);
1353   print_timestamp ();
1354 }
1355
1356 void
1357 listing_print (char *name, char **argv)
1358 {
1359   int using_stdout;
1360
1361   title = "";
1362   subtitle = "";
1363
1364   if (name == NULL)
1365     {
1366       list_file = stdout;
1367       using_stdout = 1;
1368     }
1369   else
1370     {
1371       list_file = fopen (name, FOPEN_WT);
1372       if (list_file != NULL)
1373         using_stdout = 0;
1374       else
1375         {
1376           as_warn (_("can't open %s: %s"), name, xstrerror (errno));
1377           list_file = stdout;
1378           using_stdout = 1;
1379         }
1380     }
1381
1382   if (listing & LISTING_NOFORM)
1383     paper_height = 0;
1384
1385   if (listing & LISTING_GENERAL)
1386     listing_general_info (argv);
1387
1388   if (listing & LISTING_LISTING)
1389     listing_listing (name);
1390
1391   if (listing & LISTING_SYMBOLS)
1392     list_symbol_table ();
1393
1394   if (! using_stdout)
1395     {
1396       if (fclose (list_file) == EOF)
1397         as_warn (_("can't close %s: %s"), name, xstrerror (errno));
1398     }
1399
1400   if (last_open_file)
1401     fclose (last_open_file);
1402 }
1403
1404 void
1405 listing_file (const char *name)
1406 {
1407   fn = name;
1408 }
1409
1410 void
1411 listing_eject (int ignore ATTRIBUTE_UNUSED)
1412 {
1413   if (listing)
1414     listing_tail->edict = EDICT_EJECT;
1415 }
1416
1417 /* Turn listing on or off.  An argument of 0 means to turn off
1418    listing.  An argument of 1 means to turn on listing.  An argument
1419    of 2 means to turn off listing, but as of the next line; that is,
1420    the current line should be listed, but the next line should not.  */
1421
1422 void
1423 listing_list (int on)
1424 {
1425   if (listing)
1426     {
1427       switch (on)
1428         {
1429         case 0:
1430           if (listing_tail->edict == EDICT_LIST)
1431             listing_tail->edict = EDICT_NONE;
1432           else
1433             listing_tail->edict = EDICT_NOLIST;
1434           break;
1435         case 1:
1436           if (listing_tail->edict == EDICT_NOLIST
1437               || listing_tail->edict == EDICT_NOLIST_NEXT)
1438             listing_tail->edict = EDICT_NONE;
1439           else
1440             listing_tail->edict = EDICT_LIST;
1441           break;
1442         case 2:
1443           listing_tail->edict = EDICT_NOLIST_NEXT;
1444           break;
1445         default:
1446           abort ();
1447         }
1448     }
1449 }
1450
1451 void
1452 listing_psize (int width_only)
1453 {
1454   if (! width_only)
1455     {
1456       paper_height = get_absolute_expression ();
1457
1458       if (paper_height < 0 || paper_height > 1000)
1459         {
1460           paper_height = 0;
1461           as_warn (_("strange paper height, set to no form"));
1462         }
1463
1464       if (*input_line_pointer != ',')
1465         {
1466           demand_empty_rest_of_line ();
1467           return;
1468         }
1469
1470       ++input_line_pointer;
1471     }
1472
1473   paper_width = get_absolute_expression ();
1474
1475   demand_empty_rest_of_line ();
1476 }
1477
1478 void
1479 listing_nopage (int ignore ATTRIBUTE_UNUSED)
1480 {
1481   paper_height = 0;
1482 }
1483
1484 void
1485 listing_title (int depth)
1486 {
1487   int quoted;
1488   char *start;
1489   char *ttl;
1490   unsigned int length;
1491
1492   SKIP_WHITESPACE ();
1493   if (*input_line_pointer != '\"')
1494     quoted = 0;
1495   else
1496     {
1497       quoted = 1;
1498       ++input_line_pointer;
1499     }
1500
1501   start = input_line_pointer;
1502
1503   while (*input_line_pointer)
1504     {
1505       if (quoted
1506           ? *input_line_pointer == '\"'
1507           : is_end_of_line[(unsigned char) *input_line_pointer])
1508         {
1509           if (listing)
1510             {
1511               length = input_line_pointer - start;
1512               ttl = (char *) xmalloc (length + 1);
1513               memcpy (ttl, start, length);
1514               ttl[length] = 0;
1515               listing_tail->edict = depth ? EDICT_SBTTL : EDICT_TITLE;
1516               listing_tail->edict_arg = ttl;
1517             }
1518           if (quoted)
1519             input_line_pointer++;
1520           demand_empty_rest_of_line ();
1521           return;
1522         }
1523       else if (*input_line_pointer == '\n')
1524         {
1525           as_bad (_("new line in title"));
1526           demand_empty_rest_of_line ();
1527           return;
1528         }
1529       else
1530         {
1531           input_line_pointer++;
1532         }
1533     }
1534 }
1535
1536 void
1537 listing_source_line (unsigned int line)
1538 {
1539   if (listing)
1540     {
1541       new_frag ();
1542       listing_tail->hll_line = line;
1543       new_frag ();
1544     }
1545 }
1546
1547 void
1548 listing_source_file (const char *file)
1549 {
1550   if (listing)
1551     listing_tail->hll_file = file_info (file);
1552 }
1553
1554 #else
1555
1556 /* Dummy functions for when compiled without listing enabled.  */
1557
1558 void
1559 listing_list (int on)
1560 {
1561   s_ignore (0);
1562 }
1563
1564 void
1565 listing_eject (int ignore)
1566 {
1567   s_ignore (0);
1568 }
1569
1570 void
1571 listing_psize (int ignore)
1572 {
1573   s_ignore (0);
1574 }
1575
1576 void
1577 listing_nopage (int ignore)
1578 {
1579   s_ignore (0);
1580 }
1581
1582 void
1583 listing_title (int depth)
1584 {
1585   s_ignore (0);
1586 }
1587
1588 void
1589 listing_file (const char *name)
1590 {
1591 }
1592
1593 void
1594 listing_newline (char *name)
1595 {
1596 }
1597
1598 void
1599 listing_source_line (unsigned int n)
1600 {
1601 }
1602
1603 void
1604 listing_source_file (const char *n)
1605 {
1606 }
1607
1608 #endif