Wed Mar 30 16:25:41 1994 Ian Lance Taylor (ian@tweedledumb.cygnus.com)
[external/binutils.git] / bfd / tekhex.c
1 /* BFD backend for Extended Tektronix Hex Format  objects.
2    Copyright (C) 1992, 1993 Free Software Foundation, Inc.
3
4    Written by Steve Chamberlain of Cygnus Support <sac@cygnus.com>.
5
6 This file is part of BFD, the Binary File Descriptor library.
7
8 This program 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 2 of the License, or
11 (at your option) any later version.
12
13 This program 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 this program; if not, write to the Free Software
20 Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.  */
21
22 /*
23 SUBSECTION
24         Tektronix Hex Format handling
25
26 DESCRIPTION
27         
28         Tek Hex records can hold symbols and data, but not
29         relocations. Their main application is communication with
30         devices like PROM programmers and ICE equipment.
31         
32         It seems that the sections are descibed as being really big,
33         the example I have says that the text section is 0..ffffffff.
34         BFD would barf with this, many apps would try to alloc 4GB to
35         read in the file.
36
37         Tex Hex may contain many sections, but the data which comes in
38         has no tag saying which section it belongs to, so we create
39         one section for each block of data, called "blknnnn" which we
40         stick all the data into.
41
42         TekHex may come out of  order and there is no header, so an
43         initial scan is required  to discover the minimum and maximum
44         addresses used to create the vma and size of the sections we
45         create.
46         We read in the data into pages of CHUNK_MASK+1 size and read
47         them out from that whenever we need to.
48
49         Any number of sections may be created for output, we save them
50         up and output them when it's time to close the bfd.
51
52
53         A TekHex record looks like:
54 EXAMPLE
55         %<block length><type><checksum><stuff><cr>
56         
57 DESCRIPTION
58         Where
59         o length
60         is the number of bytes in the record not including the % sign.
61         o type
62         is one of:
63         3) symbol record
64         6) data record
65         8) termination record
66         
67
68 The data can come out of order, and may be discontigous. This is a
69 serial protocol, so big files are unlikely, so we keep a list of 8k chunks
70 */
71
72 #include "bfd.h"
73 #include "sysdep.h"
74 #include "libbfd.h"
75
76 typedef struct
77   {
78     bfd_vma low;
79     bfd_vma high;
80   } addr_range_type;
81
82 typedef struct tekhex_symbol_struct
83   {
84
85     asymbol symbol;
86     struct tekhex_symbol_struct *prev;
87
88   } tekhex_symbol_type;
89
90 static char digs[] = "0123456789ABCDEF";
91
92 static char sum_block[256];
93
94 /* Horrible ascii dependent macros for converting between hex and
95    binary */
96
97 #define CHARS_IN_SET 256
98 static char hex_value[CHARS_IN_SET];
99
100 #define NOT_HEX 20
101 #define NIBBLE(x) hex_value[x]
102 #define HEX(buffer) ((NIBBLE((buffer)[0])<<4) + NIBBLE((buffer)[1]))
103 #define TOHEX(d,x) \
104 (d)[1] = digs[(x) & 0xf]; \
105 (d)[0] = digs[((x)>>4)&0xf];
106 #define ISHEX(x)  (hex_value[x] != NOT_HEX)
107
108 /*
109 Here's an example
110 %3A6C6480004E56FFFC4E717063B0AEFFFC6D0652AEFFFC60F24E5E4E75
111 %1B3709T_SEGMENT1108FFFFFFFF
112 %2B3AB9T_SEGMENT7Dgcc_compiled$1087hello$c10
113 %373829T_SEGMENT80int$t1$r1$$214741080char$t2$r2$0$12710
114 %373769T_SEGMENT80long$int$t3$r1$$1080unsigned$int$t4$10
115 %373CA9T_SEGMENT80long$unsigned$in1080short$int$t6$r1$10
116 %373049T_SEGMENT80long$long$int$t71080short$unsigned$i10
117 %373A29T_SEGMENT80long$long$unsign1080signed$char$t10$10
118 %373D69T_SEGMENT80unsigned$char$t11080float$t12$r1$4$010
119 %373D19T_SEGMENT80double$t13$r1$8$1080long$double$t14$10
120 %2734D9T_SEGMENT8Bvoid$t15$151035_main10
121 %2F3CA9T_SEGMENT81$1081$1681$1E81$21487main$F110
122 %2832F9T_SEGMENT83i$18FFFFFFFC81$1481$214
123 %07 8 10 10
124
125 explanation:
126 %3A6C6480004E56FFFC4E717063B0AEFFFC6D0652AEFFFC60F24E5E4E75
127  ^ ^^ ^     ^-data
128  | || +------ 4 char integer 0x8000
129  | |+-------- checksum
130  | +--------- type 6 (data record)
131  +----------- length 3a chars
132  <---------------------- 3a (58 chars) ------------------->
133
134 %1B3709T_SEGMENT1108FFFFFFFF
135       ^         ^^ ^- 8 character integer 0xffffffff
136       |         |+-   1 character integer 0
137       |         +--   type 1 symbol (section definition)
138       +------------   9 char symbol T_SEGMENT
139
140 %2B3AB9T_SEGMENT7Dgcc_compiled$1087hello$c10
141 %373829T_SEGMENT80int$t1$r1$$214741080char$t2$r2$0$12710
142 %373769T_SEGMENT80long$int$t3$r1$$1080unsigned$int$t4$10
143 %373CA9T_SEGMENT80long$unsigned$in1080short$int$t6$r1$10
144 %373049T_SEGMENT80long$long$int$t71080short$unsigned$i10
145 %373A29T_SEGMENT80long$long$unsign1080signed$char$t10$10
146 %373D69T_SEGMENT80unsigned$char$t11080float$t12$r1$4$010
147 %373D19T_SEGMENT80double$t13$r1$8$1080long$double$t14$10
148 %2734D9T_SEGMENT8Bvoid$t15$151035_main10
149 %2F3CA9T_SEGMENT81$1081$1681$1E81$21487main$F110
150 %2832F9T_SEGMENT83i$18FFFFFFFC81$1481$214
151 %0781010
152
153 Turns into
154 sac@thepub$ ./objdump -dx -m m68k f
155
156 f:     file format tekhex
157 -----x--- 9/55728 -134219416 Sep 29 15:13 1995 f
158 architecture: UNKNOWN!, flags 0x00000010:
159 HAS_SYMS
160 start address 0x00000000
161 SECTION 0 [D00000000]   : size 00020000 vma 00000000 align 2**0
162  ALLOC, LOAD
163 SECTION 1 [D00008000]   : size 00002001 vma 00008000 align 2**0
164
165 SECTION 2 [T_SEGMENT]   : size ffffffff vma 00000000 align 2**0
166
167 SYMBOL TABLE:
168 00000000  g       T_SEGMENT gcc_compiled$
169 00000000  g       T_SEGMENT hello$c
170 00000000  g       T_SEGMENT int$t1$r1$$21474
171 00000000  g       T_SEGMENT char$t2$r2$0$127
172 00000000  g       T_SEGMENT long$int$t3$r1$$
173 00000000  g       T_SEGMENT unsigned$int$t4$
174 00000000  g       T_SEGMENT long$unsigned$in
175 00000000  g       T_SEGMENT short$int$t6$r1$
176 00000000  g       T_SEGMENT long$long$int$t7
177 00000000  g       T_SEGMENT short$unsigned$i
178 00000000  g       T_SEGMENT long$long$unsign
179 00000000  g       T_SEGMENT signed$char$t10$
180 00000000  g       T_SEGMENT unsigned$char$t1
181 00000000  g       T_SEGMENT float$t12$r1$4$0
182 00000000  g       T_SEGMENT double$t13$r1$8$
183 00000000  g       T_SEGMENT long$double$t14$
184 00000000  g       T_SEGMENT void$t15$15
185 00000000  g       T_SEGMENT _main
186 00000000  g       T_SEGMENT $
187 00000000  g       T_SEGMENT $
188 00000000  g       T_SEGMENT $
189 00000010  g       T_SEGMENT $
190 00000000  g       T_SEGMENT main$F1
191 fcffffff  g       T_SEGMENT i$1
192 00000000  g       T_SEGMENT $
193 00000010  g       T_SEGMENT $
194
195
196 RELOCATION RECORDS FOR [D00000000]: (none)
197
198 RELOCATION RECORDS FOR [D00008000]: (none)
199
200 RELOCATION RECORDS FOR [T_SEGMENT]: (none)
201
202 Disassembly of section D00000000:
203 ...
204 00008000 ($+)7ff0 linkw fp,#-4
205 00008004 ($+)7ff4 nop
206 00008006 ($+)7ff6 movel #99,d0
207 00008008 ($+)7ff8 cmpl fp@(-4),d0
208 0000800c ($+)7ffc blts 00008014 ($+)8004
209 0000800e ($+)7ffe addql #1,fp@(-4)
210 00008012 ($+)8002 bras 00008006 ($+)7ff6
211 00008014 ($+)8004 unlk fp
212 00008016 ($+)8006 rts
213 ...
214
215 */
216
217 static void
218 tekhex_init ()
219 {
220   unsigned int i;
221   static boolean inited = false;
222   int val;
223
224   if (inited == false)
225     {
226
227       inited = true;
228
229       for (i = 0; i < CHARS_IN_SET; i++)
230         {
231           hex_value[i] = NOT_HEX;
232         }
233
234       for (i = 0; i < 10; i++)
235         {
236           hex_value[i + '0'] = i;
237
238         }
239       for (i = 0; i < 6; i++)
240         {
241           hex_value[i + 'a'] = i + 10;
242           hex_value[i + 'A'] = i + 10;
243         }
244       val = 0;
245       for (i = 0; i < 10; i++)
246         {
247           sum_block[i + '0'] = val++;
248         }
249       for (i = 'A'; i <= 'Z'; i++)
250         {
251           sum_block[i] = val++;
252         }
253       sum_block['$'] = val++;
254       sum_block['%'] = val++;
255       sum_block['.'] = val++;
256       sum_block['_'] = val++;
257       for (i = 'a'; i <= 'z'; i++)
258         {
259           sum_block[i] = val++;
260         }
261     }
262 }
263
264 /* The maximum number of bytes on a line is FF */
265 #define MAXCHUNK 0xff
266 /* The number of bytes we fit onto a line on output */
267 #define CHUNK 21
268
269 /* We cannot output our tekhexords as we see them, we have to glue them
270    together, this is done in this structure : */
271
272 struct tekhex_data_list_struct
273 {
274   unsigned char *data;
275   bfd_vma where;
276   bfd_size_type size;
277   struct tekhex_data_list_struct *next;
278
279 };
280 typedef struct tekhex_data_list_struct tekhex_data_list_type;
281
282 #define CHUNK_MASK 0x1fff
283
284 struct data_struct
285   {
286     char chunk_data[CHUNK_MASK + 1];
287     char chunk_init[CHUNK_MASK + 1];
288     bfd_vma vma;
289     struct data_struct *next;
290   };
291
292 typedef struct tekhex_data_struct
293 {
294   tekhex_data_list_type *head;
295   unsigned int type;
296   struct tekhex_symbol_struct *symbols;
297   struct data_struct *data;
298 } tdata_type;
299
300 #define enda(x) (x->vma + x->size)
301
302 static bfd_vma
303 getvalue (srcp)
304      char **srcp;
305 {
306   char *src = *srcp;
307   bfd_vma value = 0;
308   unsigned int len = hex_value[*src++];
309
310   if (len == 0)
311     len = 16;
312   while (len--)
313     {
314       value = value << 4 | hex_value[*src++];
315     }
316   *srcp = src;
317   return value;
318 }
319
320 static unsigned int
321 getsym (dstp, srcp)
322      char *dstp;
323      char **srcp;
324 {
325   char *src = *srcp;
326   unsigned int i;
327   unsigned int len = hex_value[*src++];
328
329   if (len == 0)
330     len = 16;
331   for (i = 0; i < len; i++)
332     dstp[i] = src[i];
333   dstp[i] = 0;
334   *srcp = src + i;
335   return len;
336 }
337
338 struct data_struct *
339 find_chunk (abfd, vma)
340      bfd *abfd;
341      bfd_vma vma;
342 {
343   struct data_struct *d = abfd->tdata.tekhex_data->data;
344
345   vma &= ~CHUNK_MASK;
346   while (d && (d->vma) != vma)
347     {
348       d = d->next;
349     }
350   if (!d)
351     {
352       char *sname = bfd_alloc (abfd, 12);
353       asection *s;
354
355       /* No chunk for this address, so make one up */
356       d = (struct data_struct *)
357         bfd_alloc (abfd, sizeof (struct data_struct));
358
359       if (!sname || !d)
360         {
361           bfd_set_error (bfd_error_no_memory);
362           return NULL;
363         }
364
365       memset (d->chunk_init, 0, CHUNK_MASK + 1);
366       memset (d->chunk_data, 0, CHUNK_MASK + 1);
367       d->next = abfd->tdata.tekhex_data->data;
368       d->vma = vma;
369       abfd->tdata.tekhex_data->data = d;
370     }
371   return d;
372 }
373
374 static void
375 insert_byte (abfd, value, addr)
376      bfd *abfd;
377      int value;
378      bfd_vma addr;
379 {
380   /* Find the chunk that this byte needs and put it in */
381   struct data_struct *d = find_chunk (abfd, addr);
382
383   d->chunk_data[addr & CHUNK_MASK] = value;
384   d->chunk_init[addr & CHUNK_MASK] = 1;
385 }
386
387 /* The first pass is to find the names of all the sections, and see
388   how big the data is */
389 static void
390 first_phase (abfd, type, src)
391      bfd *abfd;
392      char type;
393      char *src;
394 {
395   asection *section = &bfd_abs_section;
396   int len;
397   char sym[17];                 /* A symbol can only be 16chars long */
398
399   switch (type)
400     {
401     case '6':
402       /* Data record - read it and store it */
403       {
404         bfd_vma addr = getvalue (&src);
405
406         while (*src)
407           {
408             insert_byte (abfd, HEX (src), addr);
409             src += 2;
410             addr++;
411           }
412       }
413
414       return;
415     case '3':
416       /* Symbol record, read the segment */
417       len = getsym (sym, &src);
418       section = bfd_get_section_by_name (abfd, sym);
419       if (section == (asection *) NULL)
420         {
421           char *n = bfd_alloc (abfd, len + 1);
422
423           if (!n)
424             {
425               bfd_set_error (bfd_error_no_memory);
426               abort();          /* FIXME */
427             }
428           memcpy (n, sym, len + 1);
429           section = bfd_make_section (abfd, n);
430         }
431       while (*src)
432         {
433           switch (*src)
434             {
435               asection *s;
436
437             case '1':           /* section range */
438               src++;
439               section->vma = getvalue (&src);
440               section->_raw_size = getvalue (&src) - section->vma;
441               section->flags = SEC_HAS_CONTENTS | SEC_LOAD | SEC_ALLOC;
442               break;
443
444             case '2':
445             case '3':
446             case '4':
447             case '6':
448             case '7':
449             case '8':
450               /* Symbols, add to section */
451               {
452                 tekhex_symbol_type *new =
453                 (tekhex_symbol_type *) bfd_alloc (abfd,
454                                                sizeof (tekhex_symbol_type));
455                 char type = (*src);
456
457                 if (!new)
458                   {
459                     bfd_set_error (bfd_error_no_memory);
460                     abort();    /* FIXME */
461                   }
462                 new->symbol.the_bfd = abfd;
463                 src++;
464                 abfd->symcount++;
465                 abfd->flags |= HAS_SYMS;
466                 new->prev = abfd->tdata.tekhex_data->symbols;
467                 abfd->tdata.tekhex_data->symbols = new;
468                 len = getsym (sym, &src);
469                 new->symbol.name = bfd_alloc (abfd, len + 1);
470                 if (!new->symbol.name)
471                   {
472                     bfd_set_error (bfd_error_no_memory);
473                     abort();    /* FIXME */
474                   }
475                 memcpy ((char *) (new->symbol.name), sym, len + 1);
476                 new->symbol.section = section;
477                 if (type <= '4')
478                   new->symbol.flags = (BSF_GLOBAL | BSF_EXPORT);
479                 else
480                   new->symbol.flags = BSF_LOCAL;
481                 new->symbol.value = getvalue (&src) - section->vma;
482               }
483             }
484         }
485     }
486 }
487
488 /* Pass over an tekhex, calling one of the above functions on each
489    record.  */
490
491 static void
492  pass_over (abfd, func)
493      bfd *abfd;
494      void (*func) ();
495 {
496   unsigned int chars_on_line;
497   boolean eof = false;
498
499   /* To the front of the file */
500   bfd_seek (abfd, (file_ptr) 0, SEEK_SET);
501   while (eof == false)
502     {
503       char buffer[MAXCHUNK];
504       char *src = buffer;
505       char type;
506       bfd_vma address = 0;
507
508       /* Find first '%' */
509       eof = (boolean) (bfd_read (src, 1, 1, abfd) != 1);
510       while (*src != '%' && !eof)
511         {
512           eof = (boolean) (bfd_read (src, 1, 1, abfd) != 1);
513         }
514       if (eof)
515         break;
516       src++;
517
518       /* Fetch the type and the length and the checksum */
519       bfd_read (src, 1, 5, abfd);
520
521       type = src[2];
522
523       if (!ISHEX (src[0]) || !ISHEX (src[1]))
524         break;
525
526       chars_on_line = HEX (src) - 5;    /* Already read five char */
527
528       bfd_read (src, 1, chars_on_line, abfd);
529       src[chars_on_line] = 0;   /* put a null at the end */
530
531       func (abfd, type, src);
532     }
533
534 }
535
536 long
537 tekhex_get_symtab (abfd, table)
538      bfd *abfd;
539      asymbol **table;
540
541 {
542   tekhex_symbol_type *p = abfd->tdata.tekhex_data->symbols;
543   unsigned int c = bfd_get_symcount (abfd);
544
545   table[c] = 0;
546   while (p)
547     {
548       table[--c] = &(p->symbol);
549       p = p->prev;
550     }
551
552   return bfd_get_symcount (abfd);
553 }
554
555 long
556 tekhex_get_symtab_upper_bound (abfd)
557      bfd *abfd;
558 {
559   return (abfd->symcount + 1) * (sizeof (struct tekhex_asymbol_struct *));
560
561 }
562
563 static boolean
564 tekhex_mkobject (abfd)
565      bfd *abfd;
566 {
567   tdata_type *tdata = (tdata_type *) bfd_alloc (abfd, sizeof (tdata_type));
568
569   if (!tdata)
570     {
571       bfd_set_error (bfd_error_no_memory);
572       return false;
573     }
574   abfd->tdata.tekhex_data = tdata;
575   tdata->type = 1;
576   tdata->head = (tekhex_data_list_type *) NULL;
577   tdata->symbols = (struct tekhex_symbol_struct *) NULL;
578   tdata->data = (struct data_struct *) NULL;
579   return true;
580 }
581
582 /*
583   Return true if the file looks like it's in TekHex format. Just look
584   for a percent sign and some hex digits */
585
586 static bfd_target *
587 tekhex_object_p (abfd)
588      bfd *abfd;
589 {
590   char b[4];
591   asection *section;
592
593   tekhex_init ();
594
595   bfd_seek (abfd, (file_ptr) 0, SEEK_SET);
596   bfd_read (b, 1, 4, abfd);
597
598   if (b[0] != '%' || !ISHEX (b[1]) || !ISHEX (b[2]) || !ISHEX (b[3]))
599     return (bfd_target *) NULL;
600
601   tekhex_mkobject (abfd);
602
603   pass_over (abfd, first_phase);
604   return abfd->xvec;
605 }
606
607 static boolean
608 move_section_contents (abfd, section, locationp, offset, count, get)
609      bfd *abfd;
610      asection *section;
611      PTR locationp;
612      file_ptr offset;
613      bfd_size_type count;
614      boolean get;
615 {
616   bfd_vma addr;
617   char *location = (char *) locationp;
618   bfd_vma prev_number = 1;      /* Nothing can have this as a high bit*/
619   struct data_struct *d = (struct data_struct *) NULL;
620
621   for (addr = section->vma; count != 0; count--, addr++)
622     {
623
624       bfd_vma chunk_number = addr & ~CHUNK_MASK;        /* Get high bits of address */
625       bfd_vma low_bits = addr & CHUNK_MASK;
626
627       if (chunk_number != prev_number)
628         {
629           /* Different chunk, so move pointer */
630           d = find_chunk (abfd, chunk_number);
631         }
632
633       if (get)
634         {
635           if (d->chunk_init[low_bits])
636             {
637               *location = d->chunk_data[low_bits];
638             }
639           else
640             {
641               *location = 0;
642             }
643         }
644       else
645         {
646           d->chunk_data[low_bits] = *location;
647           d->chunk_init[low_bits] = (*location != 0);
648         }
649
650       location++;
651
652     }
653
654 }
655 static boolean
656 tekhex_get_section_contents (abfd, section, locationp, offset, count)
657      bfd *abfd;
658      asection *section;
659      PTR locationp;
660      file_ptr offset;
661      bfd_size_type count;
662 {
663   if (section->flags & (SEC_LOAD | SEC_ALLOC))
664     {
665       move_section_contents (abfd, section, locationp, offset, count, true);
666       return true;
667     }
668   else
669     return false;
670 }
671
672 boolean
673 tekhex_set_arch_mach (abfd, arch, machine)
674      bfd *abfd;
675      enum bfd_architecture arch;
676      unsigned long machine;
677 {
678   return bfd_default_set_arch_mach (abfd, arch, machine);
679 }
680
681 /* we have to save up all the Tekhexords for a splurge before output,
682     */
683
684 static boolean
685 tekhex_set_section_contents (abfd, section, locationp, offset, bytes_to_do)
686      bfd *abfd;
687      sec_ptr section;
688      PTR locationp;
689      file_ptr offset;
690      bfd_size_type bytes_to_do;
691 {
692
693   if (abfd->output_has_begun == false)
694     {
695       /* The first time around, allocate enough sections to hold all the chunks */
696       asection *s = abfd->sections;
697       bfd_vma vma;
698
699       for (s = abfd->sections; s; s = s->next)
700         {
701           if (s->flags & SEC_LOAD)
702             {
703               for (vma = s->vma & ~CHUNK_MASK;
704                    vma < s->vma + s->_raw_size;
705                    vma += CHUNK_MASK)
706                 find_chunk (abfd, vma);
707             }
708         }
709
710     }
711   if (section->flags & (SEC_LOAD | SEC_ALLOC))
712     {
713       move_section_contents (abfd, section, locationp, offset, bytes_to_do, false);
714       return true;
715     }
716   else
717     return false;
718
719 }
720
721 static void
722 writevalue (dst, value)
723      char **dst;
724      bfd_vma value;
725 {
726   char *p = *dst;
727   int len;
728   int shift;
729
730   for (len = 8, shift = 28; shift; shift -= 4, len--)
731     {
732       if ((value >> shift) & 0xf)
733         {
734           *p++ = len + '0';
735           while (len)
736             {
737               *p++ = digs[(value >> shift) & 0xf];
738               shift -= 4;
739               len--;
740             }
741           *dst = p;
742           return;
743
744         }
745     }
746   *p++ = '1';
747   *p++ = '0';
748   *dst = p;
749 }
750
751 static void
752 writesym (dst, sym)
753      char **dst;
754      CONST char *sym;
755 {
756   char *p = *dst;
757   int len = (sym ? strlen (sym) : 0);
758
759   if (len >= 16)
760     {
761       *p++ = '0';
762       len = 16;
763     }
764
765   else
766     {
767       if (len == 0)
768         {
769           *p++ = '1';
770           sym = "$";
771           len = 1;
772         }
773       else
774         {
775           *p++ = digs[len];
776         }
777     }
778
779   while (len--)
780     {
781       *p++ = *sym++;
782     }
783   *dst = p;
784 }
785
786 static void
787 out (abfd, type, start, end)
788      bfd *abfd;
789      char type;
790      char *start;
791      char *end;
792 {
793   int sum = 0;
794   char *s;
795   char front[6];
796
797   front[0] = '%';
798   TOHEX (front + 1, end - start + 5);
799   front[3] = type;
800
801   for (s = start; s < end; s++)
802     {
803       sum += sum_block[*s];
804     }
805
806   sum += sum_block[front[1]];   /*  length */
807   sum += sum_block[front[2]];
808   sum += sum_block[front[3]];   /* type */
809   TOHEX (front + 4, sum);
810   bfd_write (front, 1, 6, abfd);
811   end[0] = '\n';
812   bfd_write (start, 1, end - start + 1, abfd);
813
814 }
815
816 static boolean
817 tekhex_write_object_contents (abfd)
818      bfd *abfd;
819 {
820   int bytes_written;
821   char buffer[100];
822   asymbol **p;
823   tdata_type *tdata = abfd->tdata.tekhex_data;
824   tekhex_data_list_type *list;
825   asection *s;
826   struct data_struct *d;
827
828   bytes_written = 0;
829
830   /* And the raw data */
831   for (d = abfd->tdata.tekhex_data->data;
832        d != (struct data_struct *) NULL;
833        d = d->next)
834     {
835       int low;
836
837       CONST int span = 32;
838       int addr;
839
840       /* Write it in blocks of 32 bytes */
841
842       for (addr = 0; addr < CHUNK_MASK + 1; addr += span)
843         {
844           int need = 0;
845
846           /* Check to see if necessary */
847           for (low = 0; !need && low < span; low++)
848             {
849               if (d->chunk_init[addr + low])
850                 need = 1;
851             }
852           if (need)
853             {
854               char *dst = buffer;
855
856               writevalue (&dst, addr + d->vma);
857               for (low = 0; low < span; low++)
858                 {
859                   TOHEX (dst, d->chunk_data[addr + low]);
860                   dst += 2;
861                 }
862               out (abfd, '6', buffer, dst);
863             }
864         }
865     }
866   /* write all the section headers for the sections */
867   for (s = abfd->sections; s != (asection *) NULL; s = s->next)
868     {
869       char *dst = buffer;
870
871       writesym (&dst, s->name);
872       *dst++ = '1';
873       writevalue (&dst, s->vma);
874       writevalue (&dst, s->vma + s->_raw_size);
875       out (abfd, '3', buffer, dst);
876     }
877
878   /* And the symbols */
879   for (p = abfd->outsymbols; *p; p++)
880     {
881       int section_code = bfd_decode_symclass (*p);
882
883       if (section_code != '?')
884         {                       /* do not include debug symbols */
885           asymbol *s = *p;
886           char *dst = buffer;
887
888           writesym (&dst, s->section->name);
889
890           switch (section_code)
891             {
892             case 'A':
893               *dst++ = '2';
894               break;
895             case 'a':
896               *dst++ = '6';
897               break;
898             case 'D':
899             case 'B':
900             case 'O':
901               *dst++ = '4';
902               break;
903             case 'd':
904             case 'b':
905             case 'o':
906               *dst++ = '8';
907               break;
908             case 'T':
909               *dst++ = '3';
910               break;
911             case 't':
912               *dst++ = '7';
913               break;
914             case 'C':
915             case 'U':
916               bfd_set_error (bfd_error_wrong_format);
917               return false;
918             }
919
920           writesym (&dst, s->name);
921           writevalue (&dst, s->value + s->section->vma);
922           out (abfd, '3', buffer, dst);
923         }
924     }
925
926   /* And the terminator */
927   bfd_write ("%7081010\n", 1, 9, abfd);
928   return true;
929 }
930
931 static int
932   tekhex_sizeof_headers (abfd, exec)
933      bfd *abfd;
934      boolean exec;
935
936 {
937   return 0;
938 }
939
940 static asymbol *
941 tekhex_make_empty_symbol (abfd)
942      bfd *abfd;
943 {
944   tekhex_symbol_type *new =
945   (tekhex_symbol_type *) bfd_zalloc (abfd, sizeof (struct tekhex_symbol_struct));
946
947   if (!new)
948     {
949       bfd_set_error (bfd_error_no_memory);
950       return NULL;
951     }
952   new->symbol.the_bfd = abfd;
953   new->prev = (struct tekhex_symbol_struct *) NULL;
954   return &(new->symbol);
955 }
956
957 static void
958 tekhex_get_symbol_info (ignore_abfd, symbol, ret)
959      bfd *ignore_abfd;
960      asymbol *symbol;
961      symbol_info *ret;
962 {
963   bfd_symbol_info (symbol, ret);
964 }
965
966 static void
967 tekhex_print_symbol (ignore_abfd, filep, symbol, how)
968      bfd *ignore_abfd;
969      PTR filep;
970      asymbol *symbol;
971      bfd_print_symbol_type how;
972 {
973   FILE *file = (FILE *) filep;
974
975   switch (how)
976     {
977     case bfd_print_symbol_name:
978       fprintf (file, "%s", symbol->name);
979       break;
980     case bfd_print_symbol_more:
981       break;
982
983     case bfd_print_symbol_all:
984       {
985         CONST char *section_name = symbol->section->name;
986
987         bfd_print_symbol_vandf ((PTR) file, symbol);
988
989         fprintf (file, " %-5s %s",
990                  section_name,
991                  symbol->name);
992       }
993     }
994 }
995
996 #define FOO PROTO
997 #define tekhex_new_section_hook (FOO(boolean, (*), (bfd *, asection *)))bfd_true
998 #define tekhex_get_reloc_upper_bound (FOO(long, (*),(bfd*, asection *)))bfd_0l
999 #define tekhex_canonicalize_reloc (FOO(long, (*),(bfd*,asection *, arelent **, asymbol **))) bfd_0l
1000
1001 #define tekhex_openr_next_archived_file (FOO(bfd *, (*), (bfd*,bfd*))) bfd_nullvoidptr
1002 #define tekhex_find_nearest_line (FOO(boolean, (*),(bfd*,asection*,asymbol**,bfd_vma, CONST char**, CONST char**, unsigned int *))) bfd_false
1003 #define tekhex_generic_stat_arch_elt  (FOO(int, (*), (bfd *,struct stat *))) bfd_0
1004
1005 #define tekhex_core_file_failing_command (char *(*)())(bfd_nullvoidptr)
1006 #define tekhex_core_file_failing_signal (int (*)())bfd_0
1007 #define tekhex_core_file_matches_executable_p (FOO(boolean, (*),(bfd*, bfd*)))bfd_false
1008 #define tekhex_slurp_armap bfd_true
1009 #define tekhex_slurp_extended_name_table bfd_true
1010 #define tekhex_truncate_arname (void (*)())bfd_nullvoidptr
1011 #define tekhex_write_armap  (FOO( boolean, (*),(bfd *, unsigned int, struct orl *, unsigned int, int))) bfd_nullvoidptr
1012 #define tekhex_get_lineno (struct lineno_cache_entry *(*)())bfd_nullvoidptr
1013 #define tekhex_close_and_cleanup        bfd_generic_close_and_cleanup
1014 #define tekhex_bfd_debug_info_start bfd_void
1015 #define tekhex_bfd_debug_info_end bfd_void
1016 #define tekhex_bfd_debug_info_accumulate  (FOO(void, (*), (bfd *,        asection *))) bfd_void
1017 #define tekhex_bfd_get_relocated_section_contents bfd_generic_get_relocated_section_contents
1018 #define tekhex_bfd_relax_section bfd_generic_relax_section
1019 #define tekhex_bfd_reloc_type_lookup \
1020   ((CONST struct reloc_howto_struct *(*) PARAMS ((bfd *, bfd_reloc_code_real_type))) bfd_nullvoidptr)
1021 #define tekhex_bfd_make_debug_symbol \
1022   ((asymbol *(*) PARAMS ((bfd *, void *, unsigned long))) bfd_nullvoidptr)
1023 #define tekhex_bfd_link_hash_table_create _bfd_generic_link_hash_table_create
1024 #define tekhex_bfd_link_add_symbols _bfd_generic_link_add_symbols
1025 #define tekhex_bfd_final_link _bfd_generic_final_link
1026 #define tekhex_bfd_copy_private_section_data \
1027   ((boolean (*) PARAMS ((bfd *, asection *, bfd *, asection *))) bfd_true)
1028 #define tekhex_bfd_copy_private_bfd_data \
1029   ((boolean (*) PARAMS ((bfd *, bfd *))) bfd_true)
1030 #define tekhex_bfd_is_local_label bfd_generic_is_local_label
1031
1032 bfd_target tekhex_vec =
1033 {
1034   "tekhex",                     /* name */
1035   bfd_target_tekhex_flavour,
1036   true,                         /* target byte order */
1037   true,                         /* target headers byte order */
1038   (EXEC_P |                     /* object flags */
1039    HAS_SYMS | HAS_LINENO | HAS_DEBUG | HAS_RELOC | HAS_LOCALS |
1040    WP_TEXT | D_PAGED),
1041   (SEC_CODE | SEC_DATA | SEC_ROM | SEC_HAS_CONTENTS
1042    | SEC_ALLOC | SEC_LOAD | SEC_RELOC), /* section flags */
1043   0,                            /* leading underscore */
1044   ' ',                          /* ar_pad_char */
1045   16,                           /* ar_max_namelen */
1046   1,                            /* minimum alignment */
1047   bfd_getb64, bfd_getb_signed_64, bfd_putb64,
1048   bfd_getb32, bfd_getb_signed_32, bfd_putb32,
1049   bfd_getb16, bfd_getb_signed_16, bfd_putb16,   /* data */
1050   bfd_getb64, bfd_getb_signed_64, bfd_putb64,
1051   bfd_getb32, bfd_getb_signed_32, bfd_putb32,
1052   bfd_getb16, bfd_getb_signed_16, bfd_putb16,   /* hdrs */
1053
1054   {
1055     _bfd_dummy_target,
1056     tekhex_object_p,            /* bfd_check_format */
1057     (struct bfd_target * (*)()) bfd_nullvoidptr,
1058     (struct bfd_target * (*)()) bfd_nullvoidptr,
1059   },
1060   {
1061     bfd_false,
1062     tekhex_mkobject,
1063     _bfd_generic_mkarchive,
1064     bfd_false,
1065   },
1066   {                             /* bfd_write_contents */
1067     bfd_false,
1068     tekhex_write_object_contents,
1069     _bfd_write_archive_contents,
1070     bfd_false,
1071   },
1072   JUMP_TABLE (tekhex),
1073   (PTR) 0
1074 };