1 /* Routines to link ECOFF debugging information.
2 Copyright (C) 1993-2014 Free Software Foundation, Inc.
3 Written by Ian Lance Taylor, Cygnus Support, <ian@cygnus.com>.
4
5 This file is part of BFD, the Binary File Descriptor library.
6
7 This program is free software; you can redistribute it and/or modify
8 it under the terms of the GNU General Public License as published by
9 the Free Software Foundation; either version 3 of the License, or
10 (at your option) any later version.
11
12 This program is distributed in the hope that it will be useful,
13 but WITHOUT ANY WARRANTY; without even the implied warranty of
14 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15 GNU General Public License for more details.
16
17 You should have received a copy of the GNU General Public License
18 along with this program; if not, write to the Free Software
19 Foundation, Inc., 51 Franklin Street - Fifth Floor, Boston,
20 MA 02110-1301, USA. */
21
22 #include "sysdep.h"
23 #include "bfd.h"
24 #include "bfdlink.h"
25 #include "libbfd.h"
26 #include "objalloc.h"
27 #include "aout/stab_gnu.h"
28 #include "coff/internal.h"
29 #include "coff/sym.h"
30 #include "coff/symconst.h"
31 #include "coff/ecoff.h"
32 #include "libcoff.h"
33 #include "libecoff.h"
34
35 /* Routines to swap auxiliary information in and out. I am assuming
36 that the auxiliary information format is always going to be target
37 independent. */
38
39 /* Swap in a type information record.
40 BIGEND says whether AUX symbols are big-endian or little-endian; this
41 info comes from the file header record (fh-fBigendian). */
42
43 void
_bfd_ecoff_swap_tir_in(int bigend,const struct tir_ext * ext_copy,TIR * intern)44 _bfd_ecoff_swap_tir_in (int bigend, const struct tir_ext *ext_copy,
45 TIR *intern)
46 {
47 struct tir_ext ext[1];
48
49 *ext = *ext_copy; /* Make it reasonable to do in-place. */
50
51 /* now the fun stuff... */
52 if (bigend)
53 {
54 intern->fBitfield = 0 != (ext->t_bits1[0] & TIR_BITS1_FBITFIELD_BIG);
55 intern->continued = 0 != (ext->t_bits1[0] & TIR_BITS1_CONTINUED_BIG);
56 intern->bt = (ext->t_bits1[0] & TIR_BITS1_BT_BIG)
57 >> TIR_BITS1_BT_SH_BIG;
58 intern->tq4 = (ext->t_tq45[0] & TIR_BITS_TQ4_BIG)
59 >> TIR_BITS_TQ4_SH_BIG;
60 intern->tq5 = (ext->t_tq45[0] & TIR_BITS_TQ5_BIG)
61 >> TIR_BITS_TQ5_SH_BIG;
62 intern->tq0 = (ext->t_tq01[0] & TIR_BITS_TQ0_BIG)
63 >> TIR_BITS_TQ0_SH_BIG;
64 intern->tq1 = (ext->t_tq01[0] & TIR_BITS_TQ1_BIG)
65 >> TIR_BITS_TQ1_SH_BIG;
66 intern->tq2 = (ext->t_tq23[0] & TIR_BITS_TQ2_BIG)
67 >> TIR_BITS_TQ2_SH_BIG;
68 intern->tq3 = (ext->t_tq23[0] & TIR_BITS_TQ3_BIG)
69 >> TIR_BITS_TQ3_SH_BIG;
70 }
71 else
72 {
73 intern->fBitfield = 0 != (ext->t_bits1[0] & TIR_BITS1_FBITFIELD_LITTLE);
74 intern->continued = 0 != (ext->t_bits1[0] & TIR_BITS1_CONTINUED_LITTLE);
75 intern->bt = (ext->t_bits1[0] & TIR_BITS1_BT_LITTLE)
76 >> TIR_BITS1_BT_SH_LITTLE;
77 intern->tq4 = (ext->t_tq45[0] & TIR_BITS_TQ4_LITTLE)
78 >> TIR_BITS_TQ4_SH_LITTLE;
79 intern->tq5 = (ext->t_tq45[0] & TIR_BITS_TQ5_LITTLE)
80 >> TIR_BITS_TQ5_SH_LITTLE;
81 intern->tq0 = (ext->t_tq01[0] & TIR_BITS_TQ0_LITTLE)
82 >> TIR_BITS_TQ0_SH_LITTLE;
83 intern->tq1 = (ext->t_tq01[0] & TIR_BITS_TQ1_LITTLE)
84 >> TIR_BITS_TQ1_SH_LITTLE;
85 intern->tq2 = (ext->t_tq23[0] & TIR_BITS_TQ2_LITTLE)
86 >> TIR_BITS_TQ2_SH_LITTLE;
87 intern->tq3 = (ext->t_tq23[0] & TIR_BITS_TQ3_LITTLE)
88 >> TIR_BITS_TQ3_SH_LITTLE;
89 }
90
91 #ifdef TEST
92 if (memcmp ((char *)ext, (char *)intern, sizeof (*intern)) != 0)
93 abort ();
94 #endif
95 }
96
97 /* Swap out a type information record.
98 BIGEND says whether AUX symbols are big-endian or little-endian; this
99 info comes from the file header record (fh-fBigendian). */
100
101 void
_bfd_ecoff_swap_tir_out(int bigend,const TIR * intern_copy,struct tir_ext * ext)102 _bfd_ecoff_swap_tir_out (int bigend,
103 const TIR *intern_copy,
104 struct tir_ext *ext)
105 {
106 TIR intern[1];
107
108 *intern = *intern_copy; /* Make it reasonable to do in-place. */
109
110 /* now the fun stuff... */
111 if (bigend)
112 {
113 ext->t_bits1[0] = ((intern->fBitfield ? TIR_BITS1_FBITFIELD_BIG : 0)
114 | (intern->continued ? TIR_BITS1_CONTINUED_BIG : 0)
115 | ((intern->bt << TIR_BITS1_BT_SH_BIG)
116 & TIR_BITS1_BT_BIG));
117 ext->t_tq45[0] = (((intern->tq4 << TIR_BITS_TQ4_SH_BIG)
118 & TIR_BITS_TQ4_BIG)
119 | ((intern->tq5 << TIR_BITS_TQ5_SH_BIG)
120 & TIR_BITS_TQ5_BIG));
121 ext->t_tq01[0] = (((intern->tq0 << TIR_BITS_TQ0_SH_BIG)
122 & TIR_BITS_TQ0_BIG)
123 | ((intern->tq1 << TIR_BITS_TQ1_SH_BIG)
124 & TIR_BITS_TQ1_BIG));
125 ext->t_tq23[0] = (((intern->tq2 << TIR_BITS_TQ2_SH_BIG)
126 & TIR_BITS_TQ2_BIG)
127 | ((intern->tq3 << TIR_BITS_TQ3_SH_BIG)
128 & TIR_BITS_TQ3_BIG));
129 }
130 else
131 {
132 ext->t_bits1[0] = ((intern->fBitfield ? TIR_BITS1_FBITFIELD_LITTLE : 0)
133 | (intern->continued ? TIR_BITS1_CONTINUED_LITTLE : 0)
134 | ((intern->bt << TIR_BITS1_BT_SH_LITTLE)
135 & TIR_BITS1_BT_LITTLE));
136 ext->t_tq45[0] = (((intern->tq4 << TIR_BITS_TQ4_SH_LITTLE)
137 & TIR_BITS_TQ4_LITTLE)
138 | ((intern->tq5 << TIR_BITS_TQ5_SH_LITTLE)
139 & TIR_BITS_TQ5_LITTLE));
140 ext->t_tq01[0] = (((intern->tq0 << TIR_BITS_TQ0_SH_LITTLE)
141 & TIR_BITS_TQ0_LITTLE)
142 | ((intern->tq1 << TIR_BITS_TQ1_SH_LITTLE)
143 & TIR_BITS_TQ1_LITTLE));
144 ext->t_tq23[0] = (((intern->tq2 << TIR_BITS_TQ2_SH_LITTLE)
145 & TIR_BITS_TQ2_LITTLE)
146 | ((intern->tq3 << TIR_BITS_TQ3_SH_LITTLE)
147 & TIR_BITS_TQ3_LITTLE));
148 }
149
150 #ifdef TEST
151 if (memcmp ((char *)ext, (char *)intern, sizeof (*intern)) != 0)
152 abort ();
153 #endif
154 }
155
156 /* Swap in a relative symbol record. BIGEND says whether it is in
157 big-endian or little-endian format.*/
158
159 void
_bfd_ecoff_swap_rndx_in(int bigend,const struct rndx_ext * ext_copy,RNDXR * intern)160 _bfd_ecoff_swap_rndx_in (int bigend,
161 const struct rndx_ext *ext_copy,
162 RNDXR *intern)
163 {
164 struct rndx_ext ext[1];
165
166 *ext = *ext_copy; /* Make it reasonable to do in-place. */
167
168 /* now the fun stuff... */
169 if (bigend)
170 {
171 intern->rfd = (ext->r_bits[0] << RNDX_BITS0_RFD_SH_LEFT_BIG)
172 | ((ext->r_bits[1] & RNDX_BITS1_RFD_BIG)
173 >> RNDX_BITS1_RFD_SH_BIG);
174 intern->index = ((ext->r_bits[1] & RNDX_BITS1_INDEX_BIG)
175 << RNDX_BITS1_INDEX_SH_LEFT_BIG)
176 | (ext->r_bits[2] << RNDX_BITS2_INDEX_SH_LEFT_BIG)
177 | (ext->r_bits[3] << RNDX_BITS3_INDEX_SH_LEFT_BIG);
178 }
179 else
180 {
181 intern->rfd = (ext->r_bits[0] << RNDX_BITS0_RFD_SH_LEFT_LITTLE)
182 | ((ext->r_bits[1] & RNDX_BITS1_RFD_LITTLE)
183 << RNDX_BITS1_RFD_SH_LEFT_LITTLE);
184 intern->index = ((ext->r_bits[1] & RNDX_BITS1_INDEX_LITTLE)
185 >> RNDX_BITS1_INDEX_SH_LITTLE)
186 | (ext->r_bits[2] << RNDX_BITS2_INDEX_SH_LEFT_LITTLE)
187 | ((unsigned int) ext->r_bits[3]
188 << RNDX_BITS3_INDEX_SH_LEFT_LITTLE);
189 }
190
191 #ifdef TEST
192 if (memcmp ((char *)ext, (char *)intern, sizeof (*intern)) != 0)
193 abort ();
194 #endif
195 }
196
197 /* Swap out a relative symbol record. BIGEND says whether it is in
198 big-endian or little-endian format.*/
199
200 void
_bfd_ecoff_swap_rndx_out(int bigend,const RNDXR * intern_copy,struct rndx_ext * ext)201 _bfd_ecoff_swap_rndx_out (int bigend,
202 const RNDXR *intern_copy,
203 struct rndx_ext *ext)
204 {
205 RNDXR intern[1];
206
207 *intern = *intern_copy; /* Make it reasonable to do in-place. */
208
209 /* now the fun stuff... */
210 if (bigend)
211 {
212 ext->r_bits[0] = intern->rfd >> RNDX_BITS0_RFD_SH_LEFT_BIG;
213 ext->r_bits[1] = (((intern->rfd << RNDX_BITS1_RFD_SH_BIG)
214 & RNDX_BITS1_RFD_BIG)
215 | ((intern->index >> RNDX_BITS1_INDEX_SH_LEFT_BIG)
216 & RNDX_BITS1_INDEX_BIG));
217 ext->r_bits[2] = intern->index >> RNDX_BITS2_INDEX_SH_LEFT_BIG;
218 ext->r_bits[3] = intern->index >> RNDX_BITS3_INDEX_SH_LEFT_BIG;
219 }
220 else
221 {
222 ext->r_bits[0] = intern->rfd >> RNDX_BITS0_RFD_SH_LEFT_LITTLE;
223 ext->r_bits[1] = (((intern->rfd >> RNDX_BITS1_RFD_SH_LEFT_LITTLE)
224 & RNDX_BITS1_RFD_LITTLE)
225 | ((intern->index << RNDX_BITS1_INDEX_SH_LITTLE)
226 & RNDX_BITS1_INDEX_LITTLE));
227 ext->r_bits[2] = intern->index >> RNDX_BITS2_INDEX_SH_LEFT_LITTLE;
228 ext->r_bits[3] = intern->index >> RNDX_BITS3_INDEX_SH_LEFT_LITTLE;
229 }
230
231 #ifdef TEST
232 if (memcmp ((char *)ext, (char *)intern, sizeof (*intern)) != 0)
233 abort ();
234 #endif
235 }
236
237 /* The minimum amount of data to allocate. */
238 #define ALLOC_SIZE (4064)
239
240 /* Add bytes to a buffer. Return success. */
241
242 static bfd_boolean
ecoff_add_bytes(char ** buf,char ** bufend,size_t need)243 ecoff_add_bytes (char **buf, char **bufend, size_t need)
244 {
245 size_t have;
246 size_t want;
247 char *newbuf;
248
249 have = *bufend - *buf;
250 if (have > need)
251 want = ALLOC_SIZE;
252 else
253 {
254 want = need - have;
255 if (want < ALLOC_SIZE)
256 want = ALLOC_SIZE;
257 }
258 newbuf = (char *) bfd_realloc (*buf, (bfd_size_type) have + want);
259 if (newbuf == NULL)
260 return FALSE;
261 *buf = newbuf;
262 *bufend = *buf + have + want;
263 return TRUE;
264 }
265
266 /* We keep a hash table which maps strings to numbers. We use it to
267 map FDR names to indices in the output file, and to map local
268 strings when combining stabs debugging information. */
269
270 struct string_hash_entry
271 {
272 struct bfd_hash_entry root;
273 /* FDR index or string table offset. */
274 long val;
275 /* Next entry in string table. */
276 struct string_hash_entry *next;
277 };
278
279 struct string_hash_table
280 {
281 struct bfd_hash_table table;
282 };
283
284 /* Routine to create an entry in a string hash table. */
285
286 static struct bfd_hash_entry *
string_hash_newfunc(struct bfd_hash_entry * entry,struct bfd_hash_table * table,const char * string)287 string_hash_newfunc (struct bfd_hash_entry *entry,
288 struct bfd_hash_table *table,
289 const char *string)
290 {
291 struct string_hash_entry *ret = (struct string_hash_entry *) entry;
292
293 /* Allocate the structure if it has not already been allocated by a
294 subclass. */
295 if (ret == (struct string_hash_entry *) NULL)
296 ret = ((struct string_hash_entry *)
297 bfd_hash_allocate (table, sizeof (struct string_hash_entry)));
298 if (ret == (struct string_hash_entry *) NULL)
299 return NULL;
300
301 /* Call the allocation method of the superclass. */
302 ret = ((struct string_hash_entry *)
303 bfd_hash_newfunc ((struct bfd_hash_entry *) ret, table, string));
304
305 if (ret)
306 {
307 /* Initialize the local fields. */
308 ret->val = -1;
309 ret->next = NULL;
310 }
311
312 return (struct bfd_hash_entry *) ret;
313 }
314
315 /* Look up an entry in an string hash table. */
316
317 #define string_hash_lookup(t, string, create, copy) \
318 ((struct string_hash_entry *) \
319 bfd_hash_lookup (&(t)->table, (string), (create), (copy)))
320
321 /* We can't afford to read in all the debugging information when we do
322 a link. Instead, we build a list of these structures to show how
323 different parts of the input file map to the output file. */
324
325 struct shuffle
326 {
327 /* The next entry in this linked list. */
328 struct shuffle *next;
329 /* The length of the information. */
330 unsigned long size;
331 /* Whether this information comes from a file or not. */
332 bfd_boolean filep;
333 union
334 {
335 struct
336 {
337 /* The BFD the data comes from. */
338 bfd *input_bfd;
339 /* The offset within input_bfd. */
340 file_ptr offset;
341 } file;
342 /* The data to be written out. */
343 void * memory;
344 } u;
345 };
346
347 /* This structure holds information across calls to
348 bfd_ecoff_debug_accumulate. */
349
350 struct accumulate
351 {
352 /* The FDR hash table. */
353 struct string_hash_table fdr_hash;
354 /* The strings hash table. */
355 struct string_hash_table str_hash;
356 /* Linked lists describing how to shuffle the input debug
357 information into the output file. We keep a pointer to both the
358 head and the tail. */
359 struct shuffle *line;
360 struct shuffle *line_end;
361 struct shuffle *pdr;
362 struct shuffle *pdr_end;
363 struct shuffle *sym;
364 struct shuffle *sym_end;
365 struct shuffle *opt;
366 struct shuffle *opt_end;
367 struct shuffle *aux;
368 struct shuffle *aux_end;
369 struct shuffle *ss;
370 struct shuffle *ss_end;
371 struct string_hash_entry *ss_hash;
372 struct string_hash_entry *ss_hash_end;
373 struct shuffle *fdr;
374 struct shuffle *fdr_end;
375 struct shuffle *rfd;
376 struct shuffle *rfd_end;
377 /* The size of the largest file shuffle. */
378 unsigned long largest_file_shuffle;
379 /* An objalloc for debugging information. */
380 struct objalloc *memory;
381 };
382
383 /* Add a file entry to a shuffle list. */
384
385 static bfd_boolean
add_file_shuffle(struct accumulate * ainfo,struct shuffle ** head,struct shuffle ** tail,bfd * input_bfd,file_ptr offset,unsigned long size)386 add_file_shuffle (struct accumulate *ainfo,
387 struct shuffle **head,
388 struct shuffle **tail,
389 bfd *input_bfd,
390 file_ptr offset,
391 unsigned long size)
392 {
393 struct shuffle *n;
394
395 if (*tail != (struct shuffle *) NULL
396 && (*tail)->filep
397 && (*tail)->u.file.input_bfd == input_bfd
398 && (*tail)->u.file.offset + (*tail)->size == (unsigned long) offset)
399 {
400 /* Just merge this entry onto the existing one. */
401 (*tail)->size += size;
402 if ((*tail)->size > ainfo->largest_file_shuffle)
403 ainfo->largest_file_shuffle = (*tail)->size;
404 return TRUE;
405 }
406
407 n = (struct shuffle *) objalloc_alloc (ainfo->memory,
408 sizeof (struct shuffle));
409 if (!n)
410 {
411 bfd_set_error (bfd_error_no_memory);
412 return FALSE;
413 }
414 n->next = NULL;
415 n->size = size;
416 n->filep = TRUE;
417 n->u.file.input_bfd = input_bfd;
418 n->u.file.offset = offset;
419 if (*head == (struct shuffle *) NULL)
420 *head = n;
421 if (*tail != (struct shuffle *) NULL)
422 (*tail)->next = n;
423 *tail = n;
424 if (size > ainfo->largest_file_shuffle)
425 ainfo->largest_file_shuffle = size;
426 return TRUE;
427 }
428
429 /* Add a memory entry to a shuffle list. */
430
431 static bfd_boolean
add_memory_shuffle(struct accumulate * ainfo,struct shuffle ** head,struct shuffle ** tail,bfd_byte * data,unsigned long size)432 add_memory_shuffle (struct accumulate *ainfo,
433 struct shuffle **head,
434 struct shuffle **tail,
435 bfd_byte *data,
436 unsigned long size)
437 {
438 struct shuffle *n;
439
440 n = (struct shuffle *) objalloc_alloc (ainfo->memory,
441 sizeof (struct shuffle));
442 if (!n)
443 {
444 bfd_set_error (bfd_error_no_memory);
445 return FALSE;
446 }
447 n->next = NULL;
448 n->size = size;
449 n->filep = FALSE;
450 n->u.memory = data;
451 if (*head == (struct shuffle *) NULL)
452 *head = n;
453 if (*tail != (struct shuffle *) NULL)
454 (*tail)->next = n;
455 *tail = n;
456 return TRUE;
457 }
458
459 /* Initialize the FDR hash table. This returns a handle which is then
460 passed in to bfd_ecoff_debug_accumulate, et. al. */
461
462 void *
bfd_ecoff_debug_init(bfd * output_bfd ATTRIBUTE_UNUSED,struct ecoff_debug_info * output_debug,const struct ecoff_debug_swap * output_swap ATTRIBUTE_UNUSED,struct bfd_link_info * info)463 bfd_ecoff_debug_init (bfd *output_bfd ATTRIBUTE_UNUSED,
464 struct ecoff_debug_info *output_debug,
465 const struct ecoff_debug_swap *output_swap ATTRIBUTE_UNUSED,
466 struct bfd_link_info *info)
467 {
468 struct accumulate *ainfo;
469 bfd_size_type amt = sizeof (struct accumulate);
470
471 ainfo = (struct accumulate *) bfd_malloc (amt);
472 if (!ainfo)
473 return NULL;
474 if (!bfd_hash_table_init_n (&ainfo->fdr_hash.table, string_hash_newfunc,
475 sizeof (struct string_hash_entry), 1021))
476 return NULL;
477
478 ainfo->line = NULL;
479 ainfo->line_end = NULL;
480 ainfo->pdr = NULL;
481 ainfo->pdr_end = NULL;
482 ainfo->sym = NULL;
483 ainfo->sym_end = NULL;
484 ainfo->opt = NULL;
485 ainfo->opt_end = NULL;
486 ainfo->aux = NULL;
487 ainfo->aux_end = NULL;
488 ainfo->ss = NULL;
489 ainfo->ss_end = NULL;
490 ainfo->ss_hash = NULL;
491 ainfo->ss_hash_end = NULL;
492 ainfo->fdr = NULL;
493 ainfo->fdr_end = NULL;
494 ainfo->rfd = NULL;
495 ainfo->rfd_end = NULL;
496
497 ainfo->largest_file_shuffle = 0;
498
499 if (! info->relocatable)
500 {
501 if (!bfd_hash_table_init (&ainfo->str_hash.table, string_hash_newfunc,
502 sizeof (struct string_hash_entry)))
503 return NULL;
504
505 /* The first entry in the string table is the empty string. */
506 output_debug->symbolic_header.issMax = 1;
507 }
508
509 ainfo->memory = objalloc_create ();
510 if (ainfo->memory == NULL)
511 {
512 bfd_set_error (bfd_error_no_memory);
513 return NULL;
514 }
515
516 return ainfo;
517 }
518
519 /* Free the accumulated debugging information. */
520
521 void
bfd_ecoff_debug_free(void * handle,bfd * output_bfd ATTRIBUTE_UNUSED,struct ecoff_debug_info * output_debug ATTRIBUTE_UNUSED,const struct ecoff_debug_swap * output_swap ATTRIBUTE_UNUSED,struct bfd_link_info * info)522 bfd_ecoff_debug_free (void * handle,
523 bfd *output_bfd ATTRIBUTE_UNUSED,
524 struct ecoff_debug_info *output_debug ATTRIBUTE_UNUSED,
525 const struct ecoff_debug_swap *output_swap ATTRIBUTE_UNUSED,
526 struct bfd_link_info *info)
527 {
528 struct accumulate *ainfo = (struct accumulate *) handle;
529
530 bfd_hash_table_free (&ainfo->fdr_hash.table);
531
532 if (! info->relocatable)
533 bfd_hash_table_free (&ainfo->str_hash.table);
534
535 objalloc_free (ainfo->memory);
536
537 free (ainfo);
538 }
539
540 /* Accumulate the debugging information from INPUT_BFD into
541 OUTPUT_BFD. The INPUT_DEBUG argument points to some ECOFF
542 debugging information which we want to link into the information
543 pointed to by the OUTPUT_DEBUG argument. OUTPUT_SWAP and
544 INPUT_SWAP point to the swapping information needed. INFO is the
545 linker information structure. HANDLE is returned by
546 bfd_ecoff_debug_init. */
547
548 bfd_boolean
bfd_ecoff_debug_accumulate(void * handle,bfd * output_bfd,struct ecoff_debug_info * output_debug,const struct ecoff_debug_swap * output_swap,bfd * input_bfd,struct ecoff_debug_info * input_debug,const struct ecoff_debug_swap * input_swap,struct bfd_link_info * info)549 bfd_ecoff_debug_accumulate (void * handle,
550 bfd *output_bfd,
551 struct ecoff_debug_info *output_debug,
552 const struct ecoff_debug_swap *output_swap,
553 bfd *input_bfd,
554 struct ecoff_debug_info *input_debug,
555 const struct ecoff_debug_swap *input_swap,
556 struct bfd_link_info *info)
557 {
558 struct accumulate *ainfo = (struct accumulate *) handle;
559 void (* const swap_sym_in) (bfd *, void *, SYMR *)
560 = input_swap->swap_sym_in;
561 void (* const swap_rfd_in) (bfd *, void *, RFDT *)
562 = input_swap->swap_rfd_in;
563 void (* const swap_sym_out) (bfd *, const SYMR *, void *)
564 = output_swap->swap_sym_out;
565 void (* const swap_fdr_out) (bfd *, const FDR *, void *)
566 = output_swap->swap_fdr_out;
567 void (* const swap_rfd_out) (bfd *, const RFDT *, void *)
568 = output_swap->swap_rfd_out;
569 bfd_size_type external_pdr_size = output_swap->external_pdr_size;
570 bfd_size_type external_sym_size = output_swap->external_sym_size;
571 bfd_size_type external_opt_size = output_swap->external_opt_size;
572 bfd_size_type external_fdr_size = output_swap->external_fdr_size;
573 bfd_size_type external_rfd_size = output_swap->external_rfd_size;
574 HDRR * const output_symhdr = &output_debug->symbolic_header;
575 HDRR * const input_symhdr = &input_debug->symbolic_header;
576 bfd_vma section_adjust[scMax];
577 asection *sec;
578 bfd_byte *fdr_start;
579 bfd_byte *fdr_ptr;
580 bfd_byte *fdr_end;
581 bfd_size_type fdr_add;
582 unsigned int copied;
583 RFDT i;
584 unsigned long sz;
585 bfd_byte *rfd_out;
586 bfd_byte *rfd_in;
587 bfd_byte *rfd_end;
588 long newrfdbase = 0;
589 long oldrfdbase = 0;
590 bfd_byte *fdr_out;
591 bfd_size_type amt;
592
593 /* Use section_adjust to hold the value to add to a symbol in a
594 particular section. */
595 memset (section_adjust, 0, sizeof section_adjust);
596
597 #define SET(name, indx) \
598 sec = bfd_get_section_by_name (input_bfd, name); \
599 if (sec != NULL) \
600 section_adjust[indx] = (sec->output_section->vma \
601 + sec->output_offset \
602 - sec->vma);
603
604 SET (".text", scText);
605 SET (".data", scData);
606 SET (".bss", scBss);
607 SET (".sdata", scSData);
608 SET (".sbss", scSBss);
609 /* scRdata section may be either .rdata or .rodata. */
610 SET (".rdata", scRData);
611 SET (".rodata", scRData);
612 SET (".init", scInit);
613 SET (".fini", scFini);
614 SET (".rconst", scRConst);
615
616 #undef SET
617
618 /* Find all the debugging information based on the FDR's. We need
619 to handle them whether they are swapped or not. */
620 if (input_debug->fdr != (FDR *) NULL)
621 {
622 fdr_start = (bfd_byte *) input_debug->fdr;
623 fdr_add = sizeof (FDR);
624 }
625 else
626 {
627 fdr_start = (bfd_byte *) input_debug->external_fdr;
628 fdr_add = input_swap->external_fdr_size;
629 }
630 fdr_end = fdr_start + input_symhdr->ifdMax * fdr_add;
631
632 amt = input_symhdr->ifdMax;
633 amt *= sizeof (RFDT);
634 input_debug->ifdmap = (RFDT *) bfd_alloc (input_bfd, amt);
635
636 sz = (input_symhdr->crfd + input_symhdr->ifdMax) * external_rfd_size;
637 rfd_out = (bfd_byte *) objalloc_alloc (ainfo->memory, sz);
638 if (!input_debug->ifdmap || !rfd_out)
639 {
640 bfd_set_error (bfd_error_no_memory);
641 return FALSE;
642 }
643 if (!add_memory_shuffle (ainfo, &ainfo->rfd, &ainfo->rfd_end, rfd_out, sz))
644 return FALSE;
645
646 copied = 0;
647
648 /* Look through the FDR's to see which ones we are going to include
649 in the final output. We do not want duplicate FDR information
650 for header files, because ECOFF debugging is often very large.
651 When we find an FDR with no line information which can be merged,
652 we look it up in a hash table to ensure that we only include it
653 once. We keep a table mapping FDR numbers to the final number
654 they get with the BFD, so that we can refer to it when we write
655 out the external symbols. */
656 for (fdr_ptr = fdr_start, i = 0;
657 fdr_ptr < fdr_end;
658 fdr_ptr += fdr_add, i++, rfd_out += external_rfd_size)
659 {
660 FDR fdr;
661
662 if (input_debug->fdr != (FDR *) NULL)
663 fdr = *(FDR *) fdr_ptr;
664 else
665 (*input_swap->swap_fdr_in) (input_bfd, fdr_ptr, &fdr);
666
667 /* See if this FDR can be merged with an existing one. */
668 if (fdr.cbLine == 0 && fdr.rss != -1 && fdr.fMerge)
669 {
670 const char *name;
671 char *lookup;
672 struct string_hash_entry *fh;
673
674 /* We look up a string formed from the file name and the
675 number of symbols and aux entries. Sometimes an include
676 file will conditionally define a typedef or something
677 based on the order of include files. Using the number of
678 symbols and aux entries as a hash reduces the chance that
679 we will merge symbol information that should not be
680 merged. */
681 name = input_debug->ss + fdr.issBase + fdr.rss;
682
683 lookup = (char *) bfd_malloc ((bfd_size_type) strlen (name) + 20);
684 if (lookup == NULL)
685 return FALSE;
686 sprintf (lookup, "%s %lx %lx", name, (unsigned long) fdr.csym,
687 (unsigned long) fdr.caux);
688
689 fh = string_hash_lookup (&ainfo->fdr_hash, lookup, TRUE, TRUE);
690 free (lookup);
691 if (fh == (struct string_hash_entry *) NULL)
692 return FALSE;
693
694 if (fh->val != -1)
695 {
696 input_debug->ifdmap[i] = fh->val;
697 (*swap_rfd_out) (output_bfd, input_debug->ifdmap + i, rfd_out);
698
699 /* Don't copy this FDR. */
700 continue;
701 }
702
703 fh->val = output_symhdr->ifdMax + copied;
704 }
705
706 input_debug->ifdmap[i] = output_symhdr->ifdMax + copied;
707 (*swap_rfd_out) (output_bfd, input_debug->ifdmap + i, rfd_out);
708 ++copied;
709 }
710
711 newrfdbase = output_symhdr->crfd;
712 output_symhdr->crfd += input_symhdr->ifdMax;
713
714 /* Copy over any existing RFD's. RFD's are only created by the
715 linker, so this will only happen for input files which are the
716 result of a partial link. */
717 rfd_in = (bfd_byte *) input_debug->external_rfd;
718 rfd_end = rfd_in + input_symhdr->crfd * input_swap->external_rfd_size;
719 for (;
720 rfd_in < rfd_end;
721 rfd_in += input_swap->external_rfd_size)
722 {
723 RFDT rfd;
724
725 (*swap_rfd_in) (input_bfd, rfd_in, &rfd);
726 BFD_ASSERT (rfd >= 0 && rfd < input_symhdr->ifdMax);
727 rfd = input_debug->ifdmap[rfd];
728 (*swap_rfd_out) (output_bfd, &rfd, rfd_out);
729 rfd_out += external_rfd_size;
730 }
731
732 oldrfdbase = output_symhdr->crfd;
733 output_symhdr->crfd += input_symhdr->crfd;
734
735 /* Look through the FDR's and copy over all associated debugging
736 information. */
737 sz = copied * external_fdr_size;
738 fdr_out = (bfd_byte *) objalloc_alloc (ainfo->memory, sz);
739 if (!fdr_out)
740 {
741 bfd_set_error (bfd_error_no_memory);
742 return FALSE;
743 }
744 if (!add_memory_shuffle (ainfo, &ainfo->fdr, &ainfo->fdr_end, fdr_out, sz))
745 return FALSE;
746 for (fdr_ptr = fdr_start, i = 0;
747 fdr_ptr < fdr_end;
748 fdr_ptr += fdr_add, i++)
749 {
750 FDR fdr;
751 bfd_byte *sym_out;
752 bfd_byte *lraw_src;
753 bfd_byte *lraw_end;
754 bfd_boolean fgotfilename;
755
756 if (input_debug->ifdmap[i] < output_symhdr->ifdMax)
757 {
758 /* We are not copying this FDR. */
759 continue;
760 }
761
762 if (input_debug->fdr != (FDR *) NULL)
763 fdr = *(FDR *) fdr_ptr;
764 else
765 (*input_swap->swap_fdr_in) (input_bfd, fdr_ptr, &fdr);
766
767 /* FIXME: It is conceivable that this FDR points to the .init or
768 .fini section, in which case this will not do the right
769 thing. */
770 fdr.adr += section_adjust[scText];
771
772 /* Swap in the local symbols, adjust their values, and swap them
773 out again. */
774 fgotfilename = FALSE;
775 sz = fdr.csym * external_sym_size;
776 sym_out = (bfd_byte *) objalloc_alloc (ainfo->memory, sz);
777 if (!sym_out)
778 {
779 bfd_set_error (bfd_error_no_memory);
780 return FALSE;
781 }
782 if (!add_memory_shuffle (ainfo, &ainfo->sym, &ainfo->sym_end, sym_out,
783 sz))
784 return FALSE;
785 lraw_src = ((bfd_byte *) input_debug->external_sym
786 + fdr.isymBase * input_swap->external_sym_size);
787 lraw_end = lraw_src + fdr.csym * input_swap->external_sym_size;
788 for (; lraw_src < lraw_end; lraw_src += input_swap->external_sym_size)
789 {
790 SYMR internal_sym;
791
792 (*swap_sym_in) (input_bfd, lraw_src, &internal_sym);
793
794 BFD_ASSERT (internal_sym.sc != scCommon
795 && internal_sym.sc != scSCommon);
796
797 /* Adjust the symbol value if appropriate. */
798 switch (internal_sym.st)
799 {
800 case stNil:
801 if (ECOFF_IS_STAB (&internal_sym))
802 break;
803 /* Fall through. */
804 case stGlobal:
805 case stStatic:
806 case stLabel:
807 case stProc:
808 case stStaticProc:
809 internal_sym.value += section_adjust[internal_sym.sc];
810 break;
811
812 default:
813 break;
814 }
815
816 /* If we are doing a final link, we hash all the strings in
817 the local symbol table together. This reduces the amount
818 of space required by debugging information. We don't do
819 this when performing a relocatable link because it would
820 prevent us from easily merging different FDR's. */
821 if (! info->relocatable)
822 {
823 bfd_boolean ffilename;
824 const char *name;
825
826 if (! fgotfilename && internal_sym.iss == fdr.rss)
827 ffilename = TRUE;
828 else
829 ffilename = FALSE;
830
831 /* Hash the name into the string table. */
832 name = input_debug->ss + fdr.issBase + internal_sym.iss;
833 if (*name == '\0')
834 internal_sym.iss = 0;
835 else
836 {
837 struct string_hash_entry *sh;
838
839 sh = string_hash_lookup (&ainfo->str_hash, name, TRUE, TRUE);
840 if (sh == (struct string_hash_entry *) NULL)
841 return FALSE;
842 if (sh->val == -1)
843 {
844 sh->val = output_symhdr->issMax;
845 output_symhdr->issMax += strlen (name) + 1;
846 if (ainfo->ss_hash == (struct string_hash_entry *) NULL)
847 ainfo->ss_hash = sh;
848 if (ainfo->ss_hash_end
849 != (struct string_hash_entry *) NULL)
850 ainfo->ss_hash_end->next = sh;
851 ainfo->ss_hash_end = sh;
852 }
853 internal_sym.iss = sh->val;
854 }
855
856 if (ffilename)
857 {
858 fdr.rss = internal_sym.iss;
859 fgotfilename = TRUE;
860 }
861 }
862
863 (*swap_sym_out) (output_bfd, &internal_sym, sym_out);
864 sym_out += external_sym_size;
865 }
866
867 fdr.isymBase = output_symhdr->isymMax;
868 output_symhdr->isymMax += fdr.csym;
869
870 /* Copy the information that does not need swapping. */
871
872 /* FIXME: If we are relaxing, we need to adjust the line
873 numbers. Frankly, forget it. Anybody using stabs debugging
874 information will not use this line number information, and
875 stabs are adjusted correctly. */
876 if (fdr.cbLine > 0)
877 {
878 file_ptr pos = input_symhdr->cbLineOffset + fdr.cbLineOffset;
879 if (!add_file_shuffle (ainfo, &ainfo->line, &ainfo->line_end,
880 input_bfd, pos, (unsigned long) fdr.cbLine))
881 return FALSE;
882 fdr.ilineBase = output_symhdr->ilineMax;
883 fdr.cbLineOffset = output_symhdr->cbLine;
884 output_symhdr->ilineMax += fdr.cline;
885 output_symhdr->cbLine += fdr.cbLine;
886 }
887 if (fdr.caux > 0)
888 {
889 file_ptr pos = (input_symhdr->cbAuxOffset
890 + fdr.iauxBase * sizeof (union aux_ext));
891 if (!add_file_shuffle (ainfo, &ainfo->aux, &ainfo->aux_end,
892 input_bfd, pos,
893 fdr.caux * sizeof (union aux_ext)))
894 return FALSE;
895 fdr.iauxBase = output_symhdr->iauxMax;
896 output_symhdr->iauxMax += fdr.caux;
897 }
898 if (! info->relocatable)
899 {
900
901 /* When are are hashing strings, we lie about the number of
902 strings attached to each FDR. We need to set cbSs
903 because some versions of dbx apparently use it to decide
904 how much of the string table to read in. */
905 fdr.issBase = 0;
906 fdr.cbSs = output_symhdr->issMax;
907 }
908 else if (fdr.cbSs > 0)
909 {
910 file_ptr pos = input_symhdr->cbSsOffset + fdr.issBase;
911 if (!add_file_shuffle (ainfo, &ainfo->ss, &ainfo->ss_end,
912 input_bfd, pos, (unsigned long) fdr.cbSs))
913 return FALSE;
914 fdr.issBase = output_symhdr->issMax;
915 output_symhdr->issMax += fdr.cbSs;
916 }
917
918 if (output_bfd->xvec->header_byteorder
919 == input_bfd->xvec->header_byteorder)
920 {
921 /* The two BFD's have the same endianness, and we don't have
922 to adjust the PDR addresses, so simply copying the
923 information will suffice. */
924 BFD_ASSERT (external_pdr_size == input_swap->external_pdr_size);
925 if (fdr.cpd > 0)
926 {
927 file_ptr pos = (input_symhdr->cbPdOffset
928 + fdr.ipdFirst * external_pdr_size);
929 unsigned long size = fdr.cpd * external_pdr_size;
930 if (!add_file_shuffle (ainfo, &ainfo->pdr, &ainfo->pdr_end,
931 input_bfd, pos, size))
932 return FALSE;
933 }
934 BFD_ASSERT (external_opt_size == input_swap->external_opt_size);
935 if (fdr.copt > 0)
936 {
937 file_ptr pos = (input_symhdr->cbOptOffset
938 + fdr.ioptBase * external_opt_size);
939 unsigned long size = fdr.copt * external_opt_size;
940 if (!add_file_shuffle (ainfo, &ainfo->opt, &ainfo->opt_end,
941 input_bfd, pos, size))
942 return FALSE;
943 }
944 }
945 else
946 {
947 bfd_size_type outsz, insz;
948 bfd_byte *in;
949 bfd_byte *end;
950 bfd_byte *out;
951
952 /* The two BFD's have different endianness, so we must swap
953 everything in and out. This code would always work, but
954 it would be unnecessarily slow in the normal case. */
955 outsz = external_pdr_size;
956 insz = input_swap->external_pdr_size;
957 in = ((bfd_byte *) input_debug->external_pdr
958 + fdr.ipdFirst * insz);
959 end = in + fdr.cpd * insz;
960 sz = fdr.cpd * outsz;
961 out = (bfd_byte *) objalloc_alloc (ainfo->memory, sz);
962 if (!out)
963 {
964 bfd_set_error (bfd_error_no_memory);
965 return FALSE;
966 }
967 if (!add_memory_shuffle (ainfo, &ainfo->pdr, &ainfo->pdr_end, out,
968 sz))
969 return FALSE;
970 for (; in < end; in += insz, out += outsz)
971 {
972 PDR pdr;
973
974 (*input_swap->swap_pdr_in) (input_bfd, in, &pdr);
975 (*output_swap->swap_pdr_out) (output_bfd, &pdr, out);
976 }
977
978 /* Swap over the optimization information. */
979 outsz = external_opt_size;
980 insz = input_swap->external_opt_size;
981 in = ((bfd_byte *) input_debug->external_opt
982 + fdr.ioptBase * insz);
983 end = in + fdr.copt * insz;
984 sz = fdr.copt * outsz;
985 out = (bfd_byte *) objalloc_alloc (ainfo->memory, sz);
986 if (!out)
987 {
988 bfd_set_error (bfd_error_no_memory);
989 return FALSE;
990 }
991 if (!add_memory_shuffle (ainfo, &ainfo->opt, &ainfo->opt_end, out,
992 sz))
993 return FALSE;
994 for (; in < end; in += insz, out += outsz)
995 {
996 OPTR opt;
997
998 (*input_swap->swap_opt_in) (input_bfd, in, &opt);
999 (*output_swap->swap_opt_out) (output_bfd, &opt, out);
1000 }
1001 }
1002
1003 fdr.ipdFirst = output_symhdr->ipdMax;
1004 output_symhdr->ipdMax += fdr.cpd;
1005 fdr.ioptBase = output_symhdr->ioptMax;
1006 output_symhdr->ioptMax += fdr.copt;
1007
1008 if (fdr.crfd <= 0)
1009 {
1010 /* Point this FDR at the table of RFD's we created. */
1011 fdr.rfdBase = newrfdbase;
1012 fdr.crfd = input_symhdr->ifdMax;
1013 }
1014 else
1015 {
1016 /* Point this FDR at the remapped RFD's. */
1017 fdr.rfdBase += oldrfdbase;
1018 }
1019
1020 (*swap_fdr_out) (output_bfd, &fdr, fdr_out);
1021 fdr_out += external_fdr_size;
1022 ++output_symhdr->ifdMax;
1023 }
1024
1025 return TRUE;
1026 }
1027
1028 /* Add a string to the debugging information we are accumulating.
1029 Return the offset from the fdr string base. */
1030
1031 static long
ecoff_add_string(struct accumulate * ainfo,struct bfd_link_info * info,struct ecoff_debug_info * debug,FDR * fdr,const char * string)1032 ecoff_add_string (struct accumulate *ainfo,
1033 struct bfd_link_info *info,
1034 struct ecoff_debug_info *debug,
1035 FDR *fdr,
1036 const char *string)
1037 {
1038 HDRR *symhdr;
1039 size_t len;
1040 bfd_size_type ret;
1041
1042 symhdr = &debug->symbolic_header;
1043 len = strlen (string);
1044 if (info->relocatable)
1045 {
1046 if (!add_memory_shuffle (ainfo, &ainfo->ss, &ainfo->ss_end,
1047 (bfd_byte *) string, len + 1))
1048 return -1;
1049 ret = symhdr->issMax;
1050 symhdr->issMax += len + 1;
1051 fdr->cbSs += len + 1;
1052 }
1053 else
1054 {
1055 struct string_hash_entry *sh;
1056
1057 sh = string_hash_lookup (&ainfo->str_hash, string, TRUE, TRUE);
1058 if (sh == (struct string_hash_entry *) NULL)
1059 return -1;
1060 if (sh->val == -1)
1061 {
1062 sh->val = symhdr->issMax;
1063 symhdr->issMax += len + 1;
1064 if (ainfo->ss_hash == (struct string_hash_entry *) NULL)
1065 ainfo->ss_hash = sh;
1066 if (ainfo->ss_hash_end
1067 != (struct string_hash_entry *) NULL)
1068 ainfo->ss_hash_end->next = sh;
1069 ainfo->ss_hash_end = sh;
1070 }
1071 ret = sh->val;
1072 }
1073
1074 return ret;
1075 }
1076
1077 /* Add debugging information from a non-ECOFF file. */
1078
1079 bfd_boolean
bfd_ecoff_debug_accumulate_other(void * handle,bfd * output_bfd,struct ecoff_debug_info * output_debug,const struct ecoff_debug_swap * output_swap,bfd * input_bfd,struct bfd_link_info * info)1080 bfd_ecoff_debug_accumulate_other (void * handle,
1081 bfd *output_bfd,
1082 struct ecoff_debug_info *output_debug,
1083 const struct ecoff_debug_swap *output_swap,
1084 bfd *input_bfd,
1085 struct bfd_link_info *info)
1086 {
1087 struct accumulate *ainfo = (struct accumulate *) handle;
1088 void (* const swap_sym_out) (bfd *, const SYMR *, void *)
1089 = output_swap->swap_sym_out;
1090 HDRR *output_symhdr = &output_debug->symbolic_header;
1091 FDR fdr;
1092 asection *sec;
1093 asymbol **symbols;
1094 asymbol **sym_ptr;
1095 asymbol **sym_end;
1096 long symsize;
1097 long symcount;
1098 void * external_fdr;
1099
1100 memset (&fdr, 0, sizeof fdr);
1101
1102 sec = bfd_get_section_by_name (input_bfd, ".text");
1103 if (sec != NULL)
1104 fdr.adr = sec->output_section->vma + sec->output_offset;
1105 else
1106 {
1107 /* FIXME: What about .init or .fini? */
1108 fdr.adr = 0;
1109 }
1110
1111 fdr.issBase = output_symhdr->issMax;
1112 fdr.cbSs = 0;
1113 fdr.rss = ecoff_add_string (ainfo, info, output_debug, &fdr,
1114 input_bfd->filename);
1115 if (fdr.rss == -1)
1116 return FALSE;
1117 fdr.isymBase = output_symhdr->isymMax;
1118
1119 /* Get the local symbols from the input BFD. */
1120 symsize = bfd_get_symtab_upper_bound (input_bfd);
1121 if (symsize < 0)
1122 return FALSE;
1123 symbols = (asymbol **) bfd_alloc (output_bfd, (bfd_size_type) symsize);
1124 if (symbols == (asymbol **) NULL)
1125 return FALSE;
1126 symcount = bfd_canonicalize_symtab (input_bfd, symbols);
1127 if (symcount < 0)
1128 return FALSE;
1129 sym_end = symbols + symcount;
1130
1131 /* Handle the local symbols. Any external symbols are handled
1132 separately. */
1133 fdr.csym = 0;
1134 for (sym_ptr = symbols; sym_ptr != sym_end; sym_ptr++)
1135 {
1136 SYMR internal_sym;
1137 void * external_sym;
1138
1139 if (((*sym_ptr)->flags & BSF_EXPORT) != 0)
1140 continue;
1141 memset (&internal_sym, 0, sizeof internal_sym);
1142 internal_sym.iss = ecoff_add_string (ainfo, info, output_debug, &fdr,
1143 (*sym_ptr)->name);
1144
1145 if (internal_sym.iss == -1)
1146 return FALSE;
1147 if (bfd_is_com_section ((*sym_ptr)->section)
1148 || bfd_is_und_section ((*sym_ptr)->section))
1149 internal_sym.value = (*sym_ptr)->value;
1150 else
1151 internal_sym.value = ((*sym_ptr)->value
1152 + (*sym_ptr)->section->output_offset
1153 + (*sym_ptr)->section->output_section->vma);
1154 internal_sym.st = stNil;
1155 internal_sym.sc = scUndefined;
1156 internal_sym.index = indexNil;
1157
1158 external_sym = objalloc_alloc (ainfo->memory,
1159 output_swap->external_sym_size);
1160 if (!external_sym)
1161 {
1162 bfd_set_error (bfd_error_no_memory);
1163 return FALSE;
1164 }
1165 (*swap_sym_out) (output_bfd, &internal_sym, external_sym);
1166 add_memory_shuffle (ainfo, &ainfo->sym, &ainfo->sym_end,
1167 (bfd_byte *) external_sym,
1168 (unsigned long) output_swap->external_sym_size);
1169 ++fdr.csym;
1170 ++output_symhdr->isymMax;
1171 }
1172
1173 bfd_release (output_bfd, symbols);
1174
1175 /* Leave everything else in the FDR zeroed out. This will cause
1176 the lang field to be langC. The fBigendian field will
1177 indicate little endian format, but it doesn't matter because
1178 it only applies to aux fields and there are none. */
1179 external_fdr = objalloc_alloc (ainfo->memory,
1180 output_swap->external_fdr_size);
1181 if (!external_fdr)
1182 {
1183 bfd_set_error (bfd_error_no_memory);
1184 return FALSE;
1185 }
1186 (*output_swap->swap_fdr_out) (output_bfd, &fdr, external_fdr);
1187 add_memory_shuffle (ainfo, &ainfo->fdr, &ainfo->fdr_end,
1188 (bfd_byte *) external_fdr,
1189 (unsigned long) output_swap->external_fdr_size);
1190
1191 ++output_symhdr->ifdMax;
1192
1193 return TRUE;
1194 }
1195
1196 /* Set up ECOFF debugging information for the external symbols.
1197 FIXME: This is done using a memory buffer, but it should be
1198 probably be changed to use a shuffle structure. The assembler uses
1199 this interface, so that must be changed to do something else. */
1200
1201 bfd_boolean
bfd_ecoff_debug_externals(bfd * abfd,struct ecoff_debug_info * debug,const struct ecoff_debug_swap * swap,bfd_boolean relocatable,bfd_boolean (* get_extr)(asymbol *,EXTR *),void (* set_index)(asymbol *,bfd_size_type))1202 bfd_ecoff_debug_externals (bfd *abfd,
1203 struct ecoff_debug_info *debug,
1204 const struct ecoff_debug_swap *swap,
1205 bfd_boolean relocatable,
1206 bfd_boolean (*get_extr) (asymbol *, EXTR *),
1207 void (*set_index) (asymbol *, bfd_size_type))
1208 {
1209 HDRR * const symhdr = &debug->symbolic_header;
1210 asymbol **sym_ptr_ptr;
1211 size_t c;
1212
1213 sym_ptr_ptr = bfd_get_outsymbols (abfd);
1214 if (sym_ptr_ptr == NULL)
1215 return TRUE;
1216
1217 for (c = bfd_get_symcount (abfd); c > 0; c--, sym_ptr_ptr++)
1218 {
1219 asymbol *sym_ptr;
1220 EXTR esym;
1221
1222 sym_ptr = *sym_ptr_ptr;
1223
1224 /* Get the external symbol information. */
1225 if (! (*get_extr) (sym_ptr, &esym))
1226 continue;
1227
1228 /* If we're producing an executable, move common symbols into
1229 bss. */
1230 if (! relocatable)
1231 {
1232 if (esym.asym.sc == scCommon)
1233 esym.asym.sc = scBss;
1234 else if (esym.asym.sc == scSCommon)
1235 esym.asym.sc = scSBss;
1236 }
1237
1238 if (bfd_is_com_section (sym_ptr->section)
1239 || bfd_is_und_section (sym_ptr->section)
1240 || sym_ptr->section->output_section == (asection *) NULL)
1241 {
1242 /* FIXME: gas does not keep the value of a small undefined
1243 symbol in the symbol itself, because of relocation
1244 problems. */
1245 if (esym.asym.sc != scSUndefined
1246 || esym.asym.value == 0
1247 || sym_ptr->value != 0)
1248 esym.asym.value = sym_ptr->value;
1249 }
1250 else
1251 esym.asym.value = (sym_ptr->value
1252 + sym_ptr->section->output_offset
1253 + sym_ptr->section->output_section->vma);
1254
1255 if (set_index)
1256 (*set_index) (sym_ptr, (bfd_size_type) symhdr->iextMax);
1257
1258 if (! bfd_ecoff_debug_one_external (abfd, debug, swap,
1259 sym_ptr->name, &esym))
1260 return FALSE;
1261 }
1262
1263 return TRUE;
1264 }
1265
1266 /* Add a single external symbol to the debugging information. */
1267
1268 bfd_boolean
bfd_ecoff_debug_one_external(bfd * abfd,struct ecoff_debug_info * debug,const struct ecoff_debug_swap * swap,const char * name,EXTR * esym)1269 bfd_ecoff_debug_one_external (bfd *abfd,
1270 struct ecoff_debug_info *debug,
1271 const struct ecoff_debug_swap *swap,
1272 const char *name,
1273 EXTR *esym)
1274 {
1275 const bfd_size_type external_ext_size = swap->external_ext_size;
1276 void (* const swap_ext_out) (bfd *, const EXTR *, void *)
1277 = swap->swap_ext_out;
1278 HDRR * const symhdr = &debug->symbolic_header;
1279 size_t namelen;
1280
1281 namelen = strlen (name);
1282
1283 if ((size_t) (debug->ssext_end - debug->ssext)
1284 < symhdr->issExtMax + namelen + 1)
1285 {
1286 if (! ecoff_add_bytes ((char **) &debug->ssext,
1287 (char **) &debug->ssext_end,
1288 symhdr->issExtMax + namelen + 1))
1289 return FALSE;
1290 }
1291 if ((size_t) ((char *) debug->external_ext_end
1292 - (char *) debug->external_ext)
1293 < (symhdr->iextMax + 1) * external_ext_size)
1294 {
1295 char *external_ext = (char *) debug->external_ext;
1296 char *external_ext_end = (char *) debug->external_ext_end;
1297 if (! ecoff_add_bytes ((char **) &external_ext,
1298 (char **) &external_ext_end,
1299 (symhdr->iextMax + 1) * (size_t) external_ext_size))
1300 return FALSE;
1301 debug->external_ext = external_ext;
1302 debug->external_ext_end = external_ext_end;
1303 }
1304
1305 esym->asym.iss = symhdr->issExtMax;
1306
1307 (*swap_ext_out) (abfd, esym,
1308 ((char *) debug->external_ext
1309 + symhdr->iextMax * swap->external_ext_size));
1310
1311 ++symhdr->iextMax;
1312
1313 strcpy (debug->ssext + symhdr->issExtMax, name);
1314 symhdr->issExtMax += namelen + 1;
1315
1316 return TRUE;
1317 }
1318
1319 /* Align the ECOFF debugging information. */
1320
1321 static void
ecoff_align_debug(bfd * abfd ATTRIBUTE_UNUSED,struct ecoff_debug_info * debug,const struct ecoff_debug_swap * swap)1322 ecoff_align_debug (bfd *abfd ATTRIBUTE_UNUSED,
1323 struct ecoff_debug_info *debug,
1324 const struct ecoff_debug_swap *swap)
1325 {
1326 HDRR * const symhdr = &debug->symbolic_header;
1327 bfd_size_type debug_align, aux_align, rfd_align;
1328 size_t add;
1329
1330 /* Adjust the counts so that structures are aligned. */
1331 debug_align = swap->debug_align;
1332 aux_align = debug_align / sizeof (union aux_ext);
1333 rfd_align = debug_align / swap->external_rfd_size;
1334
1335 add = debug_align - (symhdr->cbLine & (debug_align - 1));
1336 if (add != debug_align)
1337 {
1338 if (debug->line != (unsigned char *) NULL)
1339 memset ((debug->line + symhdr->cbLine), 0, add);
1340 symhdr->cbLine += add;
1341 }
1342
1343 add = debug_align - (symhdr->issMax & (debug_align - 1));
1344 if (add != debug_align)
1345 {
1346 if (debug->ss != (char *) NULL)
1347 memset ((debug->ss + symhdr->issMax), 0, add);
1348 symhdr->issMax += add;
1349 }
1350
1351 add = debug_align - (symhdr->issExtMax & (debug_align - 1));
1352 if (add != debug_align)
1353 {
1354 if (debug->ssext != (char *) NULL)
1355 memset ((debug->ssext + symhdr->issExtMax), 0, add);
1356 symhdr->issExtMax += add;
1357 }
1358
1359 add = aux_align - (symhdr->iauxMax & (aux_align - 1));
1360 if (add != aux_align)
1361 {
1362 if (debug->external_aux != (union aux_ext *) NULL)
1363 memset ((debug->external_aux + symhdr->iauxMax), 0,
1364 add * sizeof (union aux_ext));
1365 symhdr->iauxMax += add;
1366 }
1367
1368 add = rfd_align - (symhdr->crfd & (rfd_align - 1));
1369 if (add != rfd_align)
1370 {
1371 if (debug->external_rfd != NULL)
1372 memset (((char *) debug->external_rfd
1373 + symhdr->crfd * swap->external_rfd_size),
1374 0, (size_t) (add * swap->external_rfd_size));
1375 symhdr->crfd += add;
1376 }
1377 }
1378
1379 /* Return the size required by the ECOFF debugging information. */
1380
1381 bfd_size_type
bfd_ecoff_debug_size(bfd * abfd,struct ecoff_debug_info * debug,const struct ecoff_debug_swap * swap)1382 bfd_ecoff_debug_size (bfd *abfd,
1383 struct ecoff_debug_info *debug,
1384 const struct ecoff_debug_swap *swap)
1385 {
1386 bfd_size_type tot;
1387
1388 ecoff_align_debug (abfd, debug, swap);
1389 tot = swap->external_hdr_size;
1390
1391 #define ADD(count, size) \
1392 tot += debug->symbolic_header.count * size
1393
1394 ADD (cbLine, sizeof (unsigned char));
1395 ADD (idnMax, swap->external_dnr_size);
1396 ADD (ipdMax, swap->external_pdr_size);
1397 ADD (isymMax, swap->external_sym_size);
1398 ADD (ioptMax, swap->external_opt_size);
1399 ADD (iauxMax, sizeof (union aux_ext));
1400 ADD (issMax, sizeof (char));
1401 ADD (issExtMax, sizeof (char));
1402 ADD (ifdMax, swap->external_fdr_size);
1403 ADD (crfd, swap->external_rfd_size);
1404 ADD (iextMax, swap->external_ext_size);
1405
1406 #undef ADD
1407
1408 return tot;
1409 }
1410
1411 /* Write out the ECOFF symbolic header, given the file position it is
1412 going to be placed at. This assumes that the counts are set
1413 correctly. */
1414
1415 static bfd_boolean
ecoff_write_symhdr(bfd * abfd,struct ecoff_debug_info * debug,const struct ecoff_debug_swap * swap,file_ptr where)1416 ecoff_write_symhdr (bfd *abfd,
1417 struct ecoff_debug_info *debug,
1418 const struct ecoff_debug_swap *swap,
1419 file_ptr where)
1420 {
1421 HDRR * const symhdr = &debug->symbolic_header;
1422 char *buff = NULL;
1423
1424 ecoff_align_debug (abfd, debug, swap);
1425
1426 /* Go to the right location in the file. */
1427 if (bfd_seek (abfd, where, SEEK_SET) != 0)
1428 return FALSE;
1429
1430 where += swap->external_hdr_size;
1431
1432 symhdr->magic = swap->sym_magic;
1433
1434 /* Fill in the file offsets. */
1435 #define SET(offset, count, size) \
1436 if (symhdr->count == 0) \
1437 symhdr->offset = 0; \
1438 else \
1439 { \
1440 symhdr->offset = where; \
1441 where += symhdr->count * size; \
1442 }
1443
1444 SET (cbLineOffset, cbLine, sizeof (unsigned char));
1445 SET (cbDnOffset, idnMax, swap->external_dnr_size);
1446 SET (cbPdOffset, ipdMax, swap->external_pdr_size);
1447 SET (cbSymOffset, isymMax, swap->external_sym_size);
1448 SET (cbOptOffset, ioptMax, swap->external_opt_size);
1449 SET (cbAuxOffset, iauxMax, sizeof (union aux_ext));
1450 SET (cbSsOffset, issMax, sizeof (char));
1451 SET (cbSsExtOffset, issExtMax, sizeof (char));
1452 SET (cbFdOffset, ifdMax, swap->external_fdr_size);
1453 SET (cbRfdOffset, crfd, swap->external_rfd_size);
1454 SET (cbExtOffset, iextMax, swap->external_ext_size);
1455 #undef SET
1456
1457 buff = (char *) bfd_malloc (swap->external_hdr_size);
1458 if (buff == NULL && swap->external_hdr_size != 0)
1459 goto error_return;
1460
1461 (*swap->swap_hdr_out) (abfd, symhdr, buff);
1462 if (bfd_bwrite (buff, swap->external_hdr_size, abfd)
1463 != swap->external_hdr_size)
1464 goto error_return;
1465
1466 if (buff != NULL)
1467 free (buff);
1468 return TRUE;
1469 error_return:
1470 if (buff != NULL)
1471 free (buff);
1472 return FALSE;
1473 }
1474
1475 /* Write out the ECOFF debugging information. This function assumes
1476 that the information (the pointers and counts) in *DEBUG have been
1477 set correctly. WHERE is the position in the file to write the
1478 information to. This function fills in the file offsets in the
1479 symbolic header. */
1480
1481 bfd_boolean
bfd_ecoff_write_debug(bfd * abfd,struct ecoff_debug_info * debug,const struct ecoff_debug_swap * swap,file_ptr where)1482 bfd_ecoff_write_debug (bfd *abfd,
1483 struct ecoff_debug_info *debug,
1484 const struct ecoff_debug_swap *swap,
1485 file_ptr where)
1486 {
1487 HDRR * const symhdr = &debug->symbolic_header;
1488
1489 if (! ecoff_write_symhdr (abfd, debug, swap, where))
1490 return FALSE;
1491
1492 #define WRITE(ptr, count, size, offset) \
1493 BFD_ASSERT (symhdr->offset == 0 \
1494 || (bfd_vma) bfd_tell (abfd) == symhdr->offset); \
1495 if (bfd_bwrite (debug->ptr, (bfd_size_type) size * symhdr->count, abfd)\
1496 != size * symhdr->count) \
1497 return FALSE;
1498
1499 WRITE (line, cbLine, sizeof (unsigned char), cbLineOffset);
1500 WRITE (external_dnr, idnMax, swap->external_dnr_size, cbDnOffset);
1501 WRITE (external_pdr, ipdMax, swap->external_pdr_size, cbPdOffset);
1502 WRITE (external_sym, isymMax, swap->external_sym_size, cbSymOffset);
1503 WRITE (external_opt, ioptMax, swap->external_opt_size, cbOptOffset);
1504 WRITE (external_aux, iauxMax, (bfd_size_type) sizeof (union aux_ext),
1505 cbAuxOffset);
1506 WRITE (ss, issMax, sizeof (char), cbSsOffset);
1507 WRITE (ssext, issExtMax, sizeof (char), cbSsExtOffset);
1508 WRITE (external_fdr, ifdMax, swap->external_fdr_size, cbFdOffset);
1509 WRITE (external_rfd, crfd, swap->external_rfd_size, cbRfdOffset);
1510 WRITE (external_ext, iextMax, swap->external_ext_size, cbExtOffset);
1511 #undef WRITE
1512
1513 return TRUE;
1514 }
1515
1516 /* Write out a shuffle list. */
1517
1518
1519 static bfd_boolean
ecoff_write_shuffle(bfd * abfd,const struct ecoff_debug_swap * swap,struct shuffle * shuffle,void * space)1520 ecoff_write_shuffle (bfd *abfd,
1521 const struct ecoff_debug_swap *swap,
1522 struct shuffle *shuffle,
1523 void * space)
1524 {
1525 struct shuffle *l;
1526 unsigned long total;
1527
1528 total = 0;
1529 for (l = shuffle; l != (struct shuffle *) NULL; l = l->next)
1530 {
1531 if (! l->filep)
1532 {
1533 if (bfd_bwrite (l->u.memory, (bfd_size_type) l->size, abfd)
1534 != l->size)
1535 return FALSE;
1536 }
1537 else
1538 {
1539 if (bfd_seek (l->u.file.input_bfd, l->u.file.offset, SEEK_SET) != 0
1540 || bfd_bread (space, (bfd_size_type) l->size,
1541 l->u.file.input_bfd) != l->size
1542 || bfd_bwrite (space, (bfd_size_type) l->size, abfd) != l->size)
1543 return FALSE;
1544 }
1545 total += l->size;
1546 }
1547
1548 if ((total & (swap->debug_align - 1)) != 0)
1549 {
1550 unsigned int i;
1551 bfd_byte *s;
1552
1553 i = swap->debug_align - (total & (swap->debug_align - 1));
1554 s = (bfd_byte *) bfd_zmalloc ((bfd_size_type) i);
1555 if (s == NULL && i != 0)
1556 return FALSE;
1557
1558 if (bfd_bwrite (s, (bfd_size_type) i, abfd) != i)
1559 {
1560 free (s);
1561 return FALSE;
1562 }
1563 free (s);
1564 }
1565
1566 return TRUE;
1567 }
1568
1569 /* Write out debugging information using accumulated linker
1570 information. */
1571
1572 bfd_boolean
bfd_ecoff_write_accumulated_debug(void * handle,bfd * abfd,struct ecoff_debug_info * debug,const struct ecoff_debug_swap * swap,struct bfd_link_info * info,file_ptr where)1573 bfd_ecoff_write_accumulated_debug (void * handle,
1574 bfd *abfd,
1575 struct ecoff_debug_info *debug,
1576 const struct ecoff_debug_swap *swap,
1577 struct bfd_link_info *info,
1578 file_ptr where)
1579 {
1580 struct accumulate *ainfo = (struct accumulate *) handle;
1581 void * space = NULL;
1582 bfd_size_type amt;
1583
1584 if (! ecoff_write_symhdr (abfd, debug, swap, where))
1585 goto error_return;
1586
1587 amt = ainfo->largest_file_shuffle;
1588 space = bfd_malloc (amt);
1589 if (space == NULL && ainfo->largest_file_shuffle != 0)
1590 goto error_return;
1591
1592 if (! ecoff_write_shuffle (abfd, swap, ainfo->line, space)
1593 || ! ecoff_write_shuffle (abfd, swap, ainfo->pdr, space)
1594 || ! ecoff_write_shuffle (abfd, swap, ainfo->sym, space)
1595 || ! ecoff_write_shuffle (abfd, swap, ainfo->opt, space)
1596 || ! ecoff_write_shuffle (abfd, swap, ainfo->aux, space))
1597 goto error_return;
1598
1599 /* The string table is written out from the hash table if this is a
1600 final link. */
1601 if (info->relocatable)
1602 {
1603 BFD_ASSERT (ainfo->ss_hash == (struct string_hash_entry *) NULL);
1604 if (! ecoff_write_shuffle (abfd, swap, ainfo->ss, space))
1605 goto error_return;
1606 }
1607 else
1608 {
1609 unsigned long total;
1610 bfd_byte null;
1611 struct string_hash_entry *sh;
1612
1613 BFD_ASSERT (ainfo->ss == (struct shuffle *) NULL);
1614 null = 0;
1615 if (bfd_bwrite (&null, (bfd_size_type) 1, abfd) != 1)
1616 goto error_return;
1617 total = 1;
1618 BFD_ASSERT (ainfo->ss_hash == NULL || ainfo->ss_hash->val == 1);
1619 for (sh = ainfo->ss_hash;
1620 sh != (struct string_hash_entry *) NULL;
1621 sh = sh->next)
1622 {
1623 size_t len;
1624
1625 len = strlen (sh->root.string);
1626 amt = len + 1;
1627 if (bfd_bwrite (sh->root.string, amt, abfd) != amt)
1628 goto error_return;
1629 total += len + 1;
1630 }
1631
1632 if ((total & (swap->debug_align - 1)) != 0)
1633 {
1634 unsigned int i;
1635 bfd_byte *s;
1636
1637 i = swap->debug_align - (total & (swap->debug_align - 1));
1638 s = (bfd_byte *) bfd_zmalloc ((bfd_size_type) i);
1639 if (s == NULL && i != 0)
1640 goto error_return;
1641
1642 if (bfd_bwrite (s, (bfd_size_type) i, abfd) != i)
1643 {
1644 free (s);
1645 goto error_return;
1646 }
1647 free (s);
1648 }
1649 }
1650
1651 /* The external strings and symbol are not converted over to using
1652 shuffles. FIXME: They probably should be. */
1653 amt = debug->symbolic_header.issExtMax;
1654 if (bfd_bwrite (debug->ssext, amt, abfd) != amt)
1655 goto error_return;
1656 if ((debug->symbolic_header.issExtMax & (swap->debug_align - 1)) != 0)
1657 {
1658 unsigned int i;
1659 bfd_byte *s;
1660
1661 i = (swap->debug_align
1662 - (debug->symbolic_header.issExtMax & (swap->debug_align - 1)));
1663 s = (bfd_byte *) bfd_zmalloc ((bfd_size_type) i);
1664 if (s == NULL && i != 0)
1665 goto error_return;
1666
1667 if (bfd_bwrite (s, (bfd_size_type) i, abfd) != i)
1668 {
1669 free (s);
1670 goto error_return;
1671 }
1672 free (s);
1673 }
1674
1675 if (! ecoff_write_shuffle (abfd, swap, ainfo->fdr, space)
1676 || ! ecoff_write_shuffle (abfd, swap, ainfo->rfd, space))
1677 goto error_return;
1678
1679 BFD_ASSERT (debug->symbolic_header.cbExtOffset == 0
1680 || (debug->symbolic_header.cbExtOffset
1681 == (bfd_vma) bfd_tell (abfd)));
1682
1683 amt = debug->symbolic_header.iextMax * swap->external_ext_size;
1684 if (bfd_bwrite (debug->external_ext, amt, abfd) != amt)
1685 goto error_return;
1686
1687 if (space != NULL)
1688 free (space);
1689 return TRUE;
1690
1691 error_return:
1692 if (space != NULL)
1693 free (space);
1694 return FALSE;
1695 }
1696
1697 /* Handle the find_nearest_line function for both ECOFF and MIPS ELF
1698 files. */
1699
1700 /* Compare FDR entries. This is called via qsort. */
1701
1702 static int
cmp_fdrtab_entry(const void * leftp,const void * rightp)1703 cmp_fdrtab_entry (const void * leftp, const void * rightp)
1704 {
1705 const struct ecoff_fdrtab_entry *lp =
1706 (const struct ecoff_fdrtab_entry *) leftp;
1707 const struct ecoff_fdrtab_entry *rp =
1708 (const struct ecoff_fdrtab_entry *) rightp;
1709
1710 if (lp->base_addr < rp->base_addr)
1711 return -1;
1712 if (lp->base_addr > rp->base_addr)
1713 return 1;
1714 return 0;
1715 }
1716
1717 /* Each file descriptor (FDR) has a memory address, to simplify
1718 looking up an FDR by address, we build a table covering all FDRs
1719 that have a least one procedure descriptor in them. The final
1720 table will be sorted by address so we can look it up via binary
1721 search. */
1722
1723 static bfd_boolean
mk_fdrtab(bfd * abfd,struct ecoff_debug_info * const debug_info,const struct ecoff_debug_swap * const debug_swap,struct ecoff_find_line * line_info)1724 mk_fdrtab (bfd *abfd,
1725 struct ecoff_debug_info * const debug_info,
1726 const struct ecoff_debug_swap * const debug_swap,
1727 struct ecoff_find_line *line_info)
1728 {
1729 struct ecoff_fdrtab_entry *tab;
1730 FDR *fdr_ptr;
1731 FDR *fdr_start;
1732 FDR *fdr_end;
1733 bfd_boolean stabs;
1734 long len;
1735 bfd_size_type amt;
1736
1737 fdr_start = debug_info->fdr;
1738 fdr_end = fdr_start + debug_info->symbolic_header.ifdMax;
1739
1740 /* First, let's see how long the table needs to be. */
1741 for (len = 0, fdr_ptr = fdr_start; fdr_ptr < fdr_end; fdr_ptr++)
1742 {
1743 if (fdr_ptr->cpd == 0) /* Skip FDRs that have no PDRs. */
1744 continue;
1745 ++len;
1746 }
1747
1748 /* Now, create and fill in the table. */
1749 amt = (bfd_size_type) len * sizeof (struct ecoff_fdrtab_entry);
1750 line_info->fdrtab = (struct ecoff_fdrtab_entry*) bfd_zalloc (abfd, amt);
1751 if (line_info->fdrtab == NULL)
1752 return FALSE;
1753 line_info->fdrtab_len = len;
1754
1755 tab = line_info->fdrtab;
1756 for (fdr_ptr = fdr_start; fdr_ptr < fdr_end; fdr_ptr++)
1757 {
1758 if (fdr_ptr->cpd == 0)
1759 continue;
1760
1761 /* Check whether this file has stabs debugging information. In
1762 a file with stabs debugging information, the second local
1763 symbol is named @stabs. */
1764 stabs = FALSE;
1765 if (fdr_ptr->csym >= 2)
1766 {
1767 char *sym_ptr;
1768 SYMR sym;
1769
1770 sym_ptr = ((char *) debug_info->external_sym
1771 + (fdr_ptr->isymBase + 1) * debug_swap->external_sym_size);
1772 (*debug_swap->swap_sym_in) (abfd, sym_ptr, &sym);
1773 if (strcmp (debug_info->ss + fdr_ptr->issBase + sym.iss,
1774 STABS_SYMBOL) == 0)
1775 stabs = TRUE;
1776 }
1777
1778 if (!stabs)
1779 {
1780 /* eraxxon: There are at least two problems with this computation:
1781 1) PDRs do *not* contain offsets but full vma's; and typically the
1782 address of the first PDR is the address of the FDR, which will
1783 make (most) of the results of the original computation 0!
1784 2) Once in a wacky while, the Compaq compiler generated PDR
1785 addresses do not equal the FDR vma, but they (the PDR address)
1786 are still vma's and not offsets. Cf. comments in
1787 'lookup_line'. */
1788 /* The address of the first PDR is the offset of that
1789 procedure relative to the beginning of file FDR. */
1790 tab->base_addr = fdr_ptr->adr;
1791 }
1792 else
1793 {
1794 /* XXX I don't know about stabs, so this is a guess
1795 (davidm@cs.arizona.edu). */
1796 tab->base_addr = fdr_ptr->adr;
1797 }
1798 tab->fdr = fdr_ptr;
1799 ++tab;
1800 }
1801
1802 /* Finally, the table is sorted in increasing memory-address order.
1803 The table is mostly sorted already, but there are cases (e.g.,
1804 static functions in include files), where this does not hold.
1805 Use "odump -PFv" to verify... */
1806 qsort (line_info->fdrtab, (size_t) len,
1807 sizeof (struct ecoff_fdrtab_entry), cmp_fdrtab_entry);
1808
1809 return TRUE;
1810 }
1811
1812 /* Return index of first FDR that covers to OFFSET. */
1813
1814 static long
fdrtab_lookup(struct ecoff_find_line * line_info,bfd_vma offset)1815 fdrtab_lookup (struct ecoff_find_line *line_info, bfd_vma offset)
1816 {
1817 long low, high, len;
1818 long mid = -1;
1819 struct ecoff_fdrtab_entry *tab;
1820
1821 len = line_info->fdrtab_len;
1822 if (len == 0)
1823 return -1;
1824
1825 tab = line_info->fdrtab;
1826 for (low = 0, high = len - 1 ; low != high ;)
1827 {
1828 mid = (high + low) / 2;
1829 if (offset >= tab[mid].base_addr && offset < tab[mid + 1].base_addr)
1830 goto find_min;
1831
1832 if (tab[mid].base_addr > offset)
1833 high = mid;
1834 else
1835 low = mid + 1;
1836 }
1837
1838 /* eraxxon: at this point 'offset' is either lower than the lowest entry or
1839 higher than the highest entry. In the former case high = low = mid = 0;
1840 we want to return -1. In the latter case, low = high and mid = low - 1;
1841 we want to return the index of the highest entry. Only in former case
1842 will the following 'catch-all' test be true. */
1843 ++mid;
1844
1845 /* Last entry is catch-all for all higher addresses. */
1846 if (offset < tab[mid].base_addr)
1847 return -1;
1848
1849 find_min:
1850
1851 /* eraxxon: There may be multiple FDRs in the table with the
1852 same base_addr; make sure that we are at the first one. */
1853 while (mid > 0 && tab[mid - 1].base_addr == tab[mid].base_addr)
1854 --mid;
1855
1856 return mid;
1857 }
1858
1859 /* Look up a line given an address, storing the information in
1860 LINE_INFO->cache. */
1861
1862 static bfd_boolean
lookup_line(bfd * abfd,struct ecoff_debug_info * const debug_info,const struct ecoff_debug_swap * const debug_swap,struct ecoff_find_line * line_info)1863 lookup_line (bfd *abfd,
1864 struct ecoff_debug_info * const debug_info,
1865 const struct ecoff_debug_swap * const debug_swap,
1866 struct ecoff_find_line *line_info)
1867 {
1868 struct ecoff_fdrtab_entry *tab;
1869 bfd_vma offset;
1870 bfd_boolean stabs;
1871 FDR *fdr_ptr;
1872 int i;
1873
1874 /* eraxxon: note that 'offset' is the full vma, not a section offset. */
1875 offset = line_info->cache.start;
1876
1877 /* Build FDR table (sorted by object file's base-address) if we
1878 don't have it already. */
1879 if (line_info->fdrtab == NULL
1880 && !mk_fdrtab (abfd, debug_info, debug_swap, line_info))
1881 return FALSE;
1882
1883 tab = line_info->fdrtab;
1884
1885 /* Find first FDR for address OFFSET. */
1886 i = fdrtab_lookup (line_info, offset);
1887 if (i < 0)
1888 return FALSE; /* no FDR, no fun... */
1889
1890 /* eraxxon: 'fdrtab_lookup' doesn't give what we want, at least for Compaq's
1891 C++ compiler 6.2. Consider three FDRs with starting addresses of x, y,
1892 and z, respectively, such that x < y < z. Assume further that
1893 y < 'offset' < z. It is possible at times that the PDR for 'offset' is
1894 associated with FDR x and *not* with FDR y. Erg!!
1895
1896 From a binary dump of my C++ test case 'moo' using Compaq's coffobjanl
1897 (output format has been edited for our purposes):
1898
1899 FDR [2]: (main.C): First instruction: 0x12000207c <x>
1900 PDR [5] for File [2]: LoopTest__Xv <0x1200020a0> (a)
1901 PDR [7] for File [2]: foo__Xv <0x120002168>
1902 FDR [1]: (-1): First instruction: 0x1200020e8 <y>
1903 PDR [3] for File [1]: <0x120001ad0> (b)
1904 FDR [6]: (-1): First instruction: 0x1200026f0 <z>
1905
1906 (a) In the case of PDR5, the vma is such that the first few instructions
1907 of the procedure can be found. But since the size of this procedure is
1908 160b, the vma will soon cross into the 'address space' of FDR1 and no
1909 debugging info will be found. How repugnant!
1910
1911 (b) It is also possible for a PDR to have a *lower* vma than its associated
1912 FDR; see FDR1 and PDR3. Gross!
1913
1914 Since the FDRs that are causing so much havok (in this case) 1) do not
1915 describe actual files (fdr.rss == -1), and 2) contain only compiler
1916 generated routines, I thought a simple fix would be to exclude them from
1917 the FDR table in 'mk_fdrtab'. But, besides not knowing for certain
1918 whether this would be correct, it creates an additional problem. If we
1919 happen to ask for source file info on a compiler generated (procedure)
1920 symbol -- which is still in the symbol table -- the result can be
1921 information from a real procedure! This is because compiler generated
1922 procedures with vma's higher than the last FDR in the fdr table will be
1923 associated with a PDR from this FDR, specifically the PDR with the
1924 highest vma. This wasn't a problem before, because each procedure had a
1925 PDR. (Yes, this problem could be eliminated if we kept the size of the
1926 last PDR around, but things are already getting ugly).
1927
1928 Probably, a better solution would be to have a sorted PDR table. Each
1929 PDR would have a pointer to its FDR so file information could still be
1930 obtained. A FDR table could still be constructed if necessary -- since
1931 it only contains pointers, not much extra memory would be used -- but
1932 the PDR table would be searched to locate debugging info.
1933
1934 There is still at least one remaining issue. Sometimes a FDR can have a
1935 bogus name, but contain PDRs that should belong to another FDR with a
1936 real name. E.g:
1937
1938 FDR [3]: 0000000120001b50 (/home/.../Array.H~alt~deccxx_5E5A62AD)
1939 PDR [a] for File [3]: 0000000120001b50
1940 PDR [b] for File [3]: 0000000120001cf0
1941 PDR [c] for File [3]: 0000000120001dc8
1942 PDR [d] for File [3]: 0000000120001e40
1943 PDR [e] for File [3]: 0000000120001eb8
1944 PDR [f] for File [3]: 0000000120001f4c
1945 FDR [4]: 0000000120001b50 (/home/.../Array.H)
1946
1947 Here, FDR4 has the correct name, but should (seemingly) contain PDRa-f.
1948 The symbol table for PDR4 does contain symbols for PDRa-f, but so does
1949 the symbol table for FDR3. However the former is different; perhaps this
1950 can be detected easily. (I'm not sure at this point.) This problem only
1951 seems to be associated with files with templates. I am assuming the idea
1952 is that there is a 'fake' FDR (with PDRs) for each differently typed set
1953 of templates that must be generated. Currently, FDR4 is completely
1954 excluded from the FDR table in 'mk_fdrtab' because it contains no PDRs.
1955
1956 Since I don't have time to prepare a real fix for this right now, be
1957 prepared for 'A Horrible Hack' to force the inspection of all non-stabs
1958 FDRs. It's coming... */
1959 fdr_ptr = tab[i].fdr;
1960
1961 /* Check whether this file has stabs debugging information. In a
1962 file with stabs debugging information, the second local symbol is
1963 named @stabs. */
1964 stabs = FALSE;
1965 if (fdr_ptr->csym >= 2)
1966 {
1967 char *sym_ptr;
1968 SYMR sym;
1969
1970 sym_ptr = ((char *) debug_info->external_sym
1971 + (fdr_ptr->isymBase + 1) * debug_swap->external_sym_size);
1972 (*debug_swap->swap_sym_in) (abfd, sym_ptr, &sym);
1973 if (strcmp (debug_info->ss + fdr_ptr->issBase + sym.iss,
1974 STABS_SYMBOL) == 0)
1975 stabs = TRUE;
1976 }
1977
1978 if (!stabs)
1979 {
1980 bfd_size_type external_pdr_size;
1981 char *pdr_ptr;
1982 char *best_pdr = NULL;
1983 FDR *best_fdr;
1984 bfd_signed_vma best_dist = -1;
1985 PDR pdr;
1986 unsigned char *line_ptr;
1987 unsigned char *line_end;
1988 int lineno;
1989 /* This file uses ECOFF debugging information. Each FDR has a
1990 list of procedure descriptors (PDR). The address in the FDR
1991 is the absolute address of the first procedure. The address
1992 in the first PDR gives the offset of that procedure relative
1993 to the object file's base-address. The addresses in
1994 subsequent PDRs specify each procedure's address relative to
1995 the object file's base-address. To make things more juicy,
1996 whenever the PROF bit in the PDR is set, the real entry point
1997 of the procedure may be 16 bytes below what would normally be
1998 the procedure's entry point. Instead, DEC came up with a
1999 wicked scheme to create profiled libraries "on the fly":
2000 instead of shipping a regular and a profiled version of each
2001 library, they insert 16 bytes of unused space in front of
2002 each procedure and set the "prof" bit in the PDR to indicate
2003 that there is a gap there (this is done automagically by "as"
2004 when option "-pg" is specified). Thus, normally, you link
2005 against such a library and, except for lots of 16 byte gaps
2006 between functions, things will behave as usual. However,
2007 when invoking "ld" with option "-pg", it will fill those gaps
2008 with code that calls mcount(). It then moves the function's
2009 entry point down by 16 bytes, and out pops a binary that has
2010 all functions profiled.
2011
2012 NOTE: Neither FDRs nor PDRs are strictly sorted in memory
2013 order. For example, when including header-files that
2014 define functions, the FDRs follow behind the including
2015 file, even though their code may have been generated at
2016 a lower address. File coff-alpha.c from libbfd
2017 illustrates this (use "odump -PFv" to look at a file's
2018 FDR/PDR). Similarly, PDRs are sometimes out of order
2019 as well. An example of this is OSF/1 v3.0 libc's
2020 malloc.c. I'm not sure why this happens, but it could
2021 be due to optimizations that reorder a function's
2022 position within an object-file.
2023
2024 Strategy:
2025
2026 On the first call to this function, we build a table of FDRs
2027 that is sorted by the base-address of the object-file the FDR
2028 is referring to. Notice that each object-file may contain
2029 code from multiple source files (e.g., due to code defined in
2030 include files). Thus, for any given base-address, there may
2031 be multiple FDRs (but this case is, fortunately, uncommon).
2032 lookup(addr) guarantees to return the first FDR that applies
2033 to address ADDR. Thus, after invoking lookup(), we have a
2034 list of FDRs that may contain the PDR for ADDR. Next, we
2035 walk through the PDRs of these FDRs and locate the one that
2036 is closest to ADDR (i.e., for which the difference between
2037 ADDR and the PDR's entry point is positive and minimal).
2038 Once, the right FDR and PDR are located, we simply walk
2039 through the line-number table to lookup the line-number that
2040 best matches ADDR. Obviously, things could be sped up by
2041 keeping a sorted list of PDRs instead of a sorted list of
2042 FDRs. However, this would increase space requirements
2043 considerably, which is undesirable. */
2044 external_pdr_size = debug_swap->external_pdr_size;
2045
2046 /* eraxxon: The Horrible Hack: Because of the problems above, set 'i'
2047 to 0 so we look through all FDRs.
2048
2049 Because FDR's without any symbols are assumed to be non-stabs,
2050 searching through all FDRs may cause the following code to try to
2051 read stabs FDRs as ECOFF ones. However, I don't think this will
2052 harm anything. */
2053 i = 0;
2054
2055 /* Search FDR list starting at tab[i] for the PDR that best matches
2056 OFFSET. Normally, the FDR list is only one entry long. */
2057 best_fdr = NULL;
2058 do
2059 {
2060 /* eraxxon: 'dist' and 'min_dist' can be negative now
2061 because we iterate over every FDR rather than just ones
2062 with a base address less than or equal to 'offset'. */
2063 bfd_signed_vma dist = -1, min_dist = -1;
2064 char *pdr_hold;
2065 char *pdr_end;
2066
2067 fdr_ptr = tab[i].fdr;
2068
2069 pdr_ptr = ((char *) debug_info->external_pdr
2070 + fdr_ptr->ipdFirst * external_pdr_size);
2071 pdr_end = pdr_ptr + fdr_ptr->cpd * external_pdr_size;
2072 (*debug_swap->swap_pdr_in) (abfd, pdr_ptr, &pdr);
2073 /* Find PDR that is closest to OFFSET. If pdr.prof is set,
2074 the procedure entry-point *may* be 0x10 below pdr.adr. We
2075 simply pretend that pdr.prof *implies* a lower entry-point.
2076 This is safe because it just means that may identify 4 NOPs
2077 in front of the function as belonging to the function. */
2078 for (pdr_hold = NULL;
2079 pdr_ptr < pdr_end;
2080 (pdr_ptr += external_pdr_size,
2081 (*debug_swap->swap_pdr_in) (abfd, pdr_ptr, &pdr)))
2082 {
2083 if (offset >= (pdr.adr - 0x10 * pdr.prof))
2084 {
2085 dist = offset - (pdr.adr - 0x10 * pdr.prof);
2086
2087 /* eraxxon: 'dist' can be negative now. Note that
2088 'min_dist' can be negative if 'pdr_hold' below is NULL. */
2089 if (!pdr_hold || (dist >= 0 && dist < min_dist))
2090 {
2091 min_dist = dist;
2092 pdr_hold = pdr_ptr;
2093 }
2094 }
2095 }
2096
2097 if (!best_pdr || (min_dist >= 0 && min_dist < best_dist))
2098 {
2099 best_dist = (bfd_vma) min_dist;
2100 best_fdr = fdr_ptr;
2101 best_pdr = pdr_hold;
2102 }
2103 /* Continue looping until base_addr of next entry is different. */
2104 }
2105 /* eraxxon: We want to iterate over all FDRs.
2106 See previous comment about 'fdrtab_lookup'. */
2107 while (++i < line_info->fdrtab_len);
2108
2109 if (!best_fdr || !best_pdr)
2110 return FALSE; /* Shouldn't happen... */
2111
2112 /* Phew, finally we got something that we can hold onto. */
2113 fdr_ptr = best_fdr;
2114 pdr_ptr = best_pdr;
2115 (*debug_swap->swap_pdr_in) (abfd, pdr_ptr, &pdr);
2116 /* Now we can look for the actual line number. The line numbers
2117 are stored in a very funky format, which I won't try to
2118 describe. The search is bounded by the end of the FDRs line
2119 number entries. */
2120 line_end = debug_info->line + fdr_ptr->cbLineOffset + fdr_ptr->cbLine;
2121
2122 /* Make offset relative to procedure entry. */
2123 offset -= pdr.adr - 0x10 * pdr.prof;
2124 lineno = pdr.lnLow;
2125 line_ptr = debug_info->line + fdr_ptr->cbLineOffset + pdr.cbLineOffset;
2126 while (line_ptr < line_end)
2127 {
2128 int delta;
2129 unsigned int count;
2130
2131 delta = *line_ptr >> 4;
2132 if (delta >= 0x8)
2133 delta -= 0x10;
2134 count = (*line_ptr & 0xf) + 1;
2135 ++line_ptr;
2136 if (delta == -8)
2137 {
2138 delta = (((line_ptr[0]) & 0xff) << 8) + ((line_ptr[1]) & 0xff);
2139 if (delta >= 0x8000)
2140 delta -= 0x10000;
2141 line_ptr += 2;
2142 }
2143 lineno += delta;
2144 if (offset < count * 4)
2145 {
2146 line_info->cache.stop += count * 4 - offset;
2147 break;
2148 }
2149 offset -= count * 4;
2150 }
2151
2152 /* If fdr_ptr->rss is -1, then this file does not have full
2153 symbols, at least according to gdb/mipsread.c. */
2154 if (fdr_ptr->rss == -1)
2155 {
2156 line_info->cache.filename = NULL;
2157 if (pdr.isym == -1)
2158 line_info->cache.functionname = NULL;
2159 else
2160 {
2161 EXTR proc_ext;
2162
2163 (*debug_swap->swap_ext_in)
2164 (abfd,
2165 ((char *) debug_info->external_ext
2166 + pdr.isym * debug_swap->external_ext_size),
2167 &proc_ext);
2168 line_info->cache.functionname = (debug_info->ssext
2169 + proc_ext.asym.iss);
2170 }
2171 }
2172 else
2173 {
2174 SYMR proc_sym;
2175
2176 line_info->cache.filename = (debug_info->ss
2177 + fdr_ptr->issBase
2178 + fdr_ptr->rss);
2179 (*debug_swap->swap_sym_in)
2180 (abfd,
2181 ((char *) debug_info->external_sym
2182 + ((fdr_ptr->isymBase + pdr.isym)
2183 * debug_swap->external_sym_size)),
2184 &proc_sym);
2185 line_info->cache.functionname = (debug_info->ss
2186 + fdr_ptr->issBase
2187 + proc_sym.iss);
2188 }
2189 if (lineno == ilineNil)
2190 lineno = 0;
2191 line_info->cache.line_num = lineno;
2192 }
2193 else
2194 {
2195 bfd_size_type external_sym_size;
2196 const char *directory_name;
2197 const char *main_file_name;
2198 const char *current_file_name;
2199 const char *function_name;
2200 const char *line_file_name;
2201 bfd_vma low_func_vma;
2202 bfd_vma low_line_vma;
2203 bfd_boolean past_line;
2204 bfd_boolean past_fn;
2205 char *sym_ptr, *sym_ptr_end;
2206 size_t len, funclen;
2207 char *buffer = NULL;
2208
2209 /* This file uses stabs debugging information. When gcc is not
2210 optimizing, it will put the line number information before
2211 the function name stabs entry. When gcc is optimizing, it
2212 will put the stabs entry for all the function first, followed
2213 by the line number information. (This appears to happen
2214 because of the two output files used by the -mgpopt switch,
2215 which is implied by -O). This means that we must keep
2216 looking through the symbols until we find both a line number
2217 and a function name which are beyond the address we want. */
2218
2219 line_info->cache.filename = NULL;
2220 line_info->cache.functionname = NULL;
2221 line_info->cache.line_num = 0;
2222
2223 directory_name = NULL;
2224 main_file_name = NULL;
2225 current_file_name = NULL;
2226 function_name = NULL;
2227 line_file_name = NULL;
2228 low_func_vma = 0;
2229 low_line_vma = 0;
2230 past_line = FALSE;
2231 past_fn = FALSE;
2232
2233 external_sym_size = debug_swap->external_sym_size;
2234
2235 sym_ptr = ((char *) debug_info->external_sym
2236 + (fdr_ptr->isymBase + 2) * external_sym_size);
2237 sym_ptr_end = sym_ptr + (fdr_ptr->csym - 2) * external_sym_size;
2238 for (;
2239 sym_ptr < sym_ptr_end && (! past_line || ! past_fn);
2240 sym_ptr += external_sym_size)
2241 {
2242 SYMR sym;
2243
2244 (*debug_swap->swap_sym_in) (abfd, sym_ptr, &sym);
2245
2246 if (ECOFF_IS_STAB (&sym))
2247 {
2248 switch (ECOFF_UNMARK_STAB (sym.index))
2249 {
2250 case N_SO:
2251 main_file_name = current_file_name =
2252 debug_info->ss + fdr_ptr->issBase + sym.iss;
2253
2254 /* Check the next symbol to see if it is also an
2255 N_SO symbol. */
2256 if (sym_ptr + external_sym_size < sym_ptr_end)
2257 {
2258 SYMR nextsym;
2259
2260 (*debug_swap->swap_sym_in) (abfd,
2261 sym_ptr + external_sym_size,
2262 &nextsym);
2263 if (ECOFF_IS_STAB (&nextsym)
2264 && ECOFF_UNMARK_STAB (nextsym.index) == N_SO)
2265 {
2266 directory_name = current_file_name;
2267 main_file_name = current_file_name =
2268 debug_info->ss + fdr_ptr->issBase + nextsym.iss;
2269 sym_ptr += external_sym_size;
2270 }
2271 }
2272 break;
2273
2274 case N_SOL:
2275 current_file_name =
2276 debug_info->ss + fdr_ptr->issBase + sym.iss;
2277 break;
2278
2279 case N_FUN:
2280 if (sym.value > offset)
2281 past_fn = TRUE;
2282 else if (sym.value >= low_func_vma)
2283 {
2284 low_func_vma = sym.value;
2285 function_name =
2286 debug_info->ss + fdr_ptr->issBase + sym.iss;
2287 }
2288 break;
2289 }
2290 }
2291 else if (sym.st == stLabel && sym.index != indexNil)
2292 {
2293 if (sym.value > offset)
2294 past_line = TRUE;
2295 else if (sym.value >= low_line_vma)
2296 {
2297 low_line_vma = sym.value;
2298 line_file_name = current_file_name;
2299 line_info->cache.line_num = sym.index;
2300 }
2301 }
2302 }
2303
2304 if (line_info->cache.line_num != 0)
2305 main_file_name = line_file_name;
2306
2307 /* We need to remove the stuff after the colon in the function
2308 name. We also need to put the directory name and the file
2309 name together. */
2310 if (function_name == NULL)
2311 len = funclen = 0;
2312 else
2313 len = funclen = strlen (function_name) + 1;
2314
2315 if (main_file_name != NULL
2316 && directory_name != NULL
2317 && main_file_name[0] != '/')
2318 len += strlen (directory_name) + strlen (main_file_name) + 1;
2319
2320 if (len != 0)
2321 {
2322 if (line_info->find_buffer != NULL)
2323 free (line_info->find_buffer);
2324 buffer = (char *) bfd_malloc ((bfd_size_type) len);
2325 if (buffer == NULL)
2326 return FALSE;
2327 line_info->find_buffer = buffer;
2328 }
2329
2330 if (function_name != NULL)
2331 {
2332 char *colon;
2333
2334 strcpy (buffer, function_name);
2335 colon = strchr (buffer, ':');
2336 if (colon != NULL)
2337 *colon = '\0';
2338 line_info->cache.functionname = buffer;
2339 }
2340
2341 if (main_file_name != NULL)
2342 {
2343 if (directory_name == NULL || main_file_name[0] == '/')
2344 line_info->cache.filename = main_file_name;
2345 else
2346 {
2347 sprintf (buffer + funclen, "%s%s", directory_name,
2348 main_file_name);
2349 line_info->cache.filename = buffer + funclen;
2350 }
2351 }
2352 }
2353
2354 return TRUE;
2355 }
2356
2357 /* Do the work of find_nearest_line. */
2358
2359 bfd_boolean
_bfd_ecoff_locate_line(bfd * abfd,asection * section,bfd_vma offset,struct ecoff_debug_info * const debug_info,const struct ecoff_debug_swap * const debug_swap,struct ecoff_find_line * line_info,const char ** filename_ptr,const char ** functionname_ptr,unsigned int * retline_ptr)2360 _bfd_ecoff_locate_line (bfd *abfd,
2361 asection *section,
2362 bfd_vma offset,
2363 struct ecoff_debug_info * const debug_info,
2364 const struct ecoff_debug_swap * const debug_swap,
2365 struct ecoff_find_line *line_info,
2366 const char **filename_ptr,
2367 const char **functionname_ptr,
2368 unsigned int *retline_ptr)
2369 {
2370 offset += section->vma;
2371
2372 if (line_info->cache.sect == NULL
2373 || line_info->cache.sect != section
2374 || offset < line_info->cache.start
2375 || offset >= line_info->cache.stop)
2376 {
2377 line_info->cache.sect = section;
2378 line_info->cache.start = offset;
2379 line_info->cache.stop = offset;
2380 if (! lookup_line (abfd, debug_info, debug_swap, line_info))
2381 {
2382 line_info->cache.sect = NULL;
2383 return FALSE;
2384 }
2385 }
2386
2387 *filename_ptr = line_info->cache.filename;
2388 *functionname_ptr = line_info->cache.functionname;
2389 *retline_ptr = line_info->cache.line_num;
2390
2391 return TRUE;
2392 }
2393
2394 /* These routines copy symbolic information into a memory buffer.
2395
2396 FIXME: The whole point of the shuffle code is to avoid storing
2397 everything in memory, since the linker is such a memory hog. This
2398 code makes that effort useless. It is only called by the MIPS ELF
2399 code when generating a shared library, so it is not that big a
2400 deal, but it should be fixed eventually. */
2401
2402 /* Collect a shuffle into a memory buffer. */
2403
2404 static bfd_boolean
ecoff_collect_shuffle(struct shuffle * l,bfd_byte * buff)2405 ecoff_collect_shuffle (struct shuffle *l, bfd_byte *buff)
2406 {
2407 unsigned long total;
2408
2409 total = 0;
2410 for (; l != (struct shuffle *) NULL; l = l->next)
2411 {
2412 if (! l->filep)
2413 memcpy (buff, l->u.memory, l->size);
2414 else
2415 {
2416 if (bfd_seek (l->u.file.input_bfd, l->u.file.offset, SEEK_SET) != 0
2417 || (bfd_bread (buff, (bfd_size_type) l->size, l->u.file.input_bfd)
2418 != l->size))
2419 return FALSE;
2420 }
2421 total += l->size;
2422 buff += l->size;
2423 }
2424
2425 return TRUE;
2426 }
2427
2428 /* Copy PDR information into a memory buffer. */
2429
2430 bfd_boolean
_bfd_ecoff_get_accumulated_pdr(void * handle,bfd_byte * buff)2431 _bfd_ecoff_get_accumulated_pdr (void * handle,
2432 bfd_byte *buff)
2433 {
2434 struct accumulate *ainfo = (struct accumulate *) handle;
2435
2436 return ecoff_collect_shuffle (ainfo->pdr, buff);
2437 }
2438
2439 /* Copy symbol information into a memory buffer. */
2440
2441 bfd_boolean
_bfd_ecoff_get_accumulated_sym(void * handle,bfd_byte * buff)2442 _bfd_ecoff_get_accumulated_sym (void * handle, bfd_byte *buff)
2443 {
2444 struct accumulate *ainfo = (struct accumulate *) handle;
2445
2446 return ecoff_collect_shuffle (ainfo->sym, buff);
2447 }
2448
2449 /* Copy the string table into a memory buffer. */
2450
2451 bfd_boolean
_bfd_ecoff_get_accumulated_ss(void * handle,bfd_byte * buff)2452 _bfd_ecoff_get_accumulated_ss (void * handle, bfd_byte *buff)
2453 {
2454 struct accumulate *ainfo = (struct accumulate *) handle;
2455 struct string_hash_entry *sh;
2456 unsigned long total;
2457
2458 /* The string table is written out from the hash table if this is a
2459 final link. */
2460 BFD_ASSERT (ainfo->ss == (struct shuffle *) NULL);
2461 *buff++ = '\0';
2462 total = 1;
2463 BFD_ASSERT (ainfo->ss_hash == NULL || ainfo->ss_hash->val == 1);
2464 for (sh = ainfo->ss_hash;
2465 sh != (struct string_hash_entry *) NULL;
2466 sh = sh->next)
2467 {
2468 size_t len;
2469
2470 len = strlen (sh->root.string);
2471 memcpy (buff, sh->root.string, len + 1);
2472 total += len + 1;
2473 buff += len + 1;
2474 }
2475
2476 return TRUE;
2477 }
2478