1 /* Generic COFF swapping routines, for BFD.
2    Copyright (C) 1990-2014 Free Software Foundation, Inc.
3    Written by Cygnus Support.
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 /* This file contains routines used to swap COFF data.  It is a header
23    file because the details of swapping depend on the details of the
24    structures used by each COFF implementation.  This is included by
25    coffcode.h, as well as by the ECOFF backend.
26 
27    Any file which uses this must first include "coff/internal.h" and
28    "coff/CPU.h".  The functions will then be correct for that CPU.  */
29 
30 #ifndef GET_FCN_LNNOPTR
31 #define GET_FCN_LNNOPTR(abfd, ext) \
32   H_GET_32 (abfd, ext->x_sym.x_fcnary.x_fcn.x_lnnoptr)
33 #endif
34 
35 #ifndef GET_FCN_ENDNDX
36 #define GET_FCN_ENDNDX(abfd, ext) \
37   H_GET_32 (abfd, ext->x_sym.x_fcnary.x_fcn.x_endndx)
38 #endif
39 
40 #ifndef PUT_FCN_LNNOPTR
41 #define PUT_FCN_LNNOPTR(abfd, in, ext) \
42   H_PUT_32 (abfd,  in, ext->x_sym.x_fcnary.x_fcn.x_lnnoptr)
43 #endif
44 #ifndef PUT_FCN_ENDNDX
45 #define PUT_FCN_ENDNDX(abfd, in, ext) \
46   H_PUT_32 (abfd, in, ext->x_sym.x_fcnary.x_fcn.x_endndx)
47 #endif
48 #ifndef GET_LNSZ_LNNO
49 #define GET_LNSZ_LNNO(abfd, ext) \
50   H_GET_16 (abfd, ext->x_sym.x_misc.x_lnsz.x_lnno)
51 #endif
52 #ifndef GET_LNSZ_SIZE
53 #define GET_LNSZ_SIZE(abfd, ext) \
54   H_GET_16 (abfd, ext->x_sym.x_misc.x_lnsz.x_size)
55 #endif
56 #ifndef PUT_LNSZ_LNNO
57 #define PUT_LNSZ_LNNO(abfd, in, ext) \
58   H_PUT_16 (abfd, in, ext->x_sym.x_misc.x_lnsz.x_lnno)
59 #endif
60 #ifndef PUT_LNSZ_SIZE
61 #define PUT_LNSZ_SIZE(abfd, in, ext) \
62   H_PUT_16 (abfd, in, ext->x_sym.x_misc.x_lnsz.x_size)
63 #endif
64 #ifndef GET_SCN_SCNLEN
65 #define GET_SCN_SCNLEN(abfd, ext) \
66   H_GET_32 (abfd, ext->x_scn.x_scnlen)
67 #endif
68 #ifndef GET_SCN_NRELOC
69 #define GET_SCN_NRELOC(abfd, ext) \
70   H_GET_16 (abfd, ext->x_scn.x_nreloc)
71 #endif
72 #ifndef GET_SCN_NLINNO
73 #define GET_SCN_NLINNO(abfd, ext) \
74   H_GET_16 (abfd, ext->x_scn.x_nlinno)
75 #endif
76 #ifndef PUT_SCN_SCNLEN
77 #define PUT_SCN_SCNLEN(abfd, in, ext) \
78   H_PUT_32 (abfd, in, ext->x_scn.x_scnlen)
79 #endif
80 #ifndef PUT_SCN_NRELOC
81 #define PUT_SCN_NRELOC(abfd, in, ext) \
82   H_PUT_16 (abfd, in, ext->x_scn.x_nreloc)
83 #endif
84 #ifndef PUT_SCN_NLINNO
85 #define PUT_SCN_NLINNO(abfd, in, ext) \
86   H_PUT_16 (abfd, in, ext->x_scn.x_nlinno)
87 #endif
88 #ifndef GET_LINENO_LNNO
89 #define GET_LINENO_LNNO(abfd, ext) \
90   H_GET_16 (abfd, ext->l_lnno);
91 #endif
92 #ifndef PUT_LINENO_LNNO
93 #define PUT_LINENO_LNNO(abfd, val, ext) \
94   H_PUT_16 (abfd, val, ext->l_lnno);
95 #endif
96 
97 /* The f_symptr field in the filehdr is sometimes 64 bits.  */
98 #ifndef GET_FILEHDR_SYMPTR
99 #define GET_FILEHDR_SYMPTR H_GET_32
100 #endif
101 #ifndef PUT_FILEHDR_SYMPTR
102 #define PUT_FILEHDR_SYMPTR H_PUT_32
103 #endif
104 
105 /* Some fields in the aouthdr are sometimes 64 bits.  */
106 #ifndef GET_AOUTHDR_TSIZE
107 #define GET_AOUTHDR_TSIZE H_GET_32
108 #endif
109 #ifndef PUT_AOUTHDR_TSIZE
110 #define PUT_AOUTHDR_TSIZE H_PUT_32
111 #endif
112 #ifndef GET_AOUTHDR_DSIZE
113 #define GET_AOUTHDR_DSIZE H_GET_32
114 #endif
115 #ifndef PUT_AOUTHDR_DSIZE
116 #define PUT_AOUTHDR_DSIZE H_PUT_32
117 #endif
118 #ifndef GET_AOUTHDR_BSIZE
119 #define GET_AOUTHDR_BSIZE H_GET_32
120 #endif
121 #ifndef PUT_AOUTHDR_BSIZE
122 #define PUT_AOUTHDR_BSIZE H_PUT_32
123 #endif
124 #ifndef GET_AOUTHDR_ENTRY
125 #define GET_AOUTHDR_ENTRY H_GET_32
126 #endif
127 #ifndef PUT_AOUTHDR_ENTRY
128 #define PUT_AOUTHDR_ENTRY H_PUT_32
129 #endif
130 #ifndef GET_AOUTHDR_TEXT_START
131 #define GET_AOUTHDR_TEXT_START H_GET_32
132 #endif
133 #ifndef PUT_AOUTHDR_TEXT_START
134 #define PUT_AOUTHDR_TEXT_START H_PUT_32
135 #endif
136 #ifndef GET_AOUTHDR_DATA_START
137 #define GET_AOUTHDR_DATA_START H_GET_32
138 #endif
139 #ifndef PUT_AOUTHDR_DATA_START
140 #define PUT_AOUTHDR_DATA_START H_PUT_32
141 #endif
142 
143 /* Some fields in the scnhdr are sometimes 64 bits.  */
144 #ifndef GET_SCNHDR_PADDR
145 #define GET_SCNHDR_PADDR H_GET_32
146 #endif
147 #ifndef PUT_SCNHDR_PADDR
148 #define PUT_SCNHDR_PADDR H_PUT_32
149 #endif
150 #ifndef GET_SCNHDR_VADDR
151 #define GET_SCNHDR_VADDR H_GET_32
152 #endif
153 #ifndef PUT_SCNHDR_VADDR
154 #define PUT_SCNHDR_VADDR H_PUT_32
155 #endif
156 #ifndef GET_SCNHDR_SIZE
157 #define GET_SCNHDR_SIZE H_GET_32
158 #endif
159 #ifndef PUT_SCNHDR_SIZE
160 #define PUT_SCNHDR_SIZE H_PUT_32
161 #endif
162 #ifndef GET_SCNHDR_SCNPTR
163 #define GET_SCNHDR_SCNPTR H_GET_32
164 #endif
165 #ifndef PUT_SCNHDR_SCNPTR
166 #define PUT_SCNHDR_SCNPTR H_PUT_32
167 #endif
168 #ifndef GET_SCNHDR_RELPTR
169 #define GET_SCNHDR_RELPTR H_GET_32
170 #endif
171 #ifndef PUT_SCNHDR_RELPTR
172 #define PUT_SCNHDR_RELPTR H_PUT_32
173 #endif
174 #ifndef GET_SCNHDR_LNNOPTR
175 #define GET_SCNHDR_LNNOPTR H_GET_32
176 #endif
177 #ifndef PUT_SCNHDR_LNNOPTR
178 #define PUT_SCNHDR_LNNOPTR H_PUT_32
179 #endif
180 #ifndef GET_SCNHDR_NRELOC
181 #define GET_SCNHDR_NRELOC H_GET_16
182 #endif
183 #ifndef MAX_SCNHDR_NRELOC
184 #define MAX_SCNHDR_NRELOC 0xffff
185 #endif
186 #ifndef PUT_SCNHDR_NRELOC
187 #define PUT_SCNHDR_NRELOC H_PUT_16
188 #endif
189 #ifndef GET_SCNHDR_NLNNO
190 #define GET_SCNHDR_NLNNO H_GET_16
191 #endif
192 #ifndef MAX_SCNHDR_NLNNO
193 #define MAX_SCNHDR_NLNNO 0xffff
194 #endif
195 #ifndef PUT_SCNHDR_NLNNO
196 #define PUT_SCNHDR_NLNNO H_PUT_16
197 #endif
198 #ifndef GET_SCNHDR_FLAGS
199 #define GET_SCNHDR_FLAGS H_GET_32
200 #endif
201 #ifndef PUT_SCNHDR_FLAGS
202 #define PUT_SCNHDR_FLAGS H_PUT_32
203 #endif
204 
205 #ifndef GET_RELOC_VADDR
206 #define GET_RELOC_VADDR H_GET_32
207 #endif
208 #ifndef PUT_RELOC_VADDR
209 #define PUT_RELOC_VADDR H_PUT_32
210 #endif
211 
212 #ifndef NO_COFF_RELOCS
213 
214 static void
coff_swap_reloc_in(bfd * abfd,void * src,void * dst)215 coff_swap_reloc_in (bfd * abfd, void * src, void * dst)
216 {
217   RELOC *reloc_src = (RELOC *) src;
218   struct internal_reloc *reloc_dst = (struct internal_reloc *) dst;
219 
220   reloc_dst->r_vaddr  = GET_RELOC_VADDR (abfd, reloc_src->r_vaddr);
221   reloc_dst->r_symndx = H_GET_S32 (abfd, reloc_src->r_symndx);
222   reloc_dst->r_type   = H_GET_16 (abfd, reloc_src->r_type);
223 
224 #ifdef SWAP_IN_RELOC_OFFSET
225   reloc_dst->r_offset = SWAP_IN_RELOC_OFFSET (abfd, reloc_src->r_offset);
226 #endif
227 }
228 
229 static unsigned int
coff_swap_reloc_out(bfd * abfd,void * src,void * dst)230 coff_swap_reloc_out (bfd * abfd, void * src, void * dst)
231 {
232   struct internal_reloc *reloc_src = (struct internal_reloc *) src;
233   struct external_reloc *reloc_dst = (struct external_reloc *) dst;
234 
235   PUT_RELOC_VADDR (abfd, reloc_src->r_vaddr, reloc_dst->r_vaddr);
236   H_PUT_32 (abfd, reloc_src->r_symndx, reloc_dst->r_symndx);
237   H_PUT_16 (abfd, reloc_src->r_type, reloc_dst->r_type);
238 
239 #ifdef SWAP_OUT_RELOC_OFFSET
240   SWAP_OUT_RELOC_OFFSET (abfd, reloc_src->r_offset, reloc_dst->r_offset);
241 #endif
242 #ifdef SWAP_OUT_RELOC_EXTRA
243   SWAP_OUT_RELOC_EXTRA (abfd, reloc_src, reloc_dst);
244 #endif
245 
246   return bfd_coff_relsz (abfd);
247 }
248 
249 #endif /* NO_COFF_RELOCS */
250 
251 static void
coff_swap_filehdr_in(bfd * abfd,void * src,void * dst)252 coff_swap_filehdr_in (bfd * abfd, void * src, void * dst)
253 {
254   FILHDR *filehdr_src = (FILHDR *) src;
255   struct internal_filehdr *filehdr_dst = (struct internal_filehdr *) dst;
256 
257 #ifdef COFF_ADJUST_FILEHDR_IN_PRE
258   COFF_ADJUST_FILEHDR_IN_PRE (abfd, src, dst);
259 #endif
260   filehdr_dst->f_magic  = H_GET_16 (abfd, filehdr_src->f_magic);
261   filehdr_dst->f_nscns  = H_GET_16 (abfd, filehdr_src->f_nscns);
262   filehdr_dst->f_timdat = H_GET_32 (abfd, filehdr_src->f_timdat);
263   filehdr_dst->f_symptr = GET_FILEHDR_SYMPTR (abfd, filehdr_src->f_symptr);
264   filehdr_dst->f_nsyms  = H_GET_32 (abfd, filehdr_src->f_nsyms);
265   filehdr_dst->f_opthdr = H_GET_16 (abfd, filehdr_src->f_opthdr);
266   filehdr_dst->f_flags  = H_GET_16 (abfd, filehdr_src->f_flags);
267 #ifdef TIC80_TARGET_ID
268   filehdr_dst->f_target_id = H_GET_16 (abfd, filehdr_src->f_target_id);
269 #endif
270 
271 #ifdef COFF_ADJUST_FILEHDR_IN_POST
272   COFF_ADJUST_FILEHDR_IN_POST (abfd, src, dst);
273 #endif
274 }
275 
276 static  unsigned int
coff_swap_filehdr_out(bfd * abfd,void * in,void * out)277 coff_swap_filehdr_out (bfd *abfd, void * in, void * out)
278 {
279   struct internal_filehdr *filehdr_in = (struct internal_filehdr *) in;
280   FILHDR *filehdr_out = (FILHDR *) out;
281 
282 #ifdef COFF_ADJUST_FILEHDR_OUT_PRE
283   COFF_ADJUST_FILEHDR_OUT_PRE (abfd, in, out);
284 #endif
285   H_PUT_16 (abfd, filehdr_in->f_magic, filehdr_out->f_magic);
286   H_PUT_16 (abfd, filehdr_in->f_nscns, filehdr_out->f_nscns);
287   H_PUT_32 (abfd, filehdr_in->f_timdat, filehdr_out->f_timdat);
288   PUT_FILEHDR_SYMPTR (abfd, filehdr_in->f_symptr, filehdr_out->f_symptr);
289   H_PUT_32 (abfd, filehdr_in->f_nsyms, filehdr_out->f_nsyms);
290   H_PUT_16 (abfd, filehdr_in->f_opthdr, filehdr_out->f_opthdr);
291   H_PUT_16 (abfd, filehdr_in->f_flags, filehdr_out->f_flags);
292 #ifdef TIC80_TARGET_ID
293   H_PUT_16 (abfd, filehdr_in->f_target_id, filehdr_out->f_target_id);
294 #endif
295 
296 #ifdef COFF_ADJUST_FILEHDR_OUT_POST
297   COFF_ADJUST_FILEHDR_OUT_POST (abfd, in, out);
298 #endif
299   return bfd_coff_filhsz (abfd);
300 }
301 
302 #ifndef NO_COFF_SYMBOLS
303 
304 static void
coff_swap_sym_in(bfd * abfd,void * ext1,void * in1)305 coff_swap_sym_in (bfd * abfd, void * ext1, void * in1)
306 {
307   SYMENT *ext = (SYMENT *) ext1;
308   struct internal_syment *in = (struct internal_syment *) in1;
309 
310   if (ext->e.e_name[0] == 0)
311     {
312       in->_n._n_n._n_zeroes = 0;
313       in->_n._n_n._n_offset = H_GET_32 (abfd, ext->e.e.e_offset);
314     }
315   else
316     {
317 #if SYMNMLEN != E_SYMNMLEN
318 #error we need to cope with truncating or extending SYMNMLEN
319 #else
320       memcpy (in->_n._n_name, ext->e.e_name, SYMNMLEN);
321 #endif
322     }
323 
324   in->n_value = H_GET_32 (abfd, ext->e_value);
325   in->n_scnum = H_GET_16 (abfd, ext->e_scnum);
326   if (sizeof (ext->e_type) == 2)
327     in->n_type = H_GET_16 (abfd, ext->e_type);
328   else
329     in->n_type = H_GET_32 (abfd, ext->e_type);
330   in->n_sclass = H_GET_8 (abfd, ext->e_sclass);
331   in->n_numaux = H_GET_8 (abfd, ext->e_numaux);
332 #ifdef COFF_ADJUST_SYM_IN_POST
333   COFF_ADJUST_SYM_IN_POST (abfd, ext1, in1);
334 #endif
335 }
336 
337 static unsigned int
coff_swap_sym_out(bfd * abfd,void * inp,void * extp)338 coff_swap_sym_out (bfd * abfd, void * inp, void * extp)
339 {
340   struct internal_syment *in = (struct internal_syment *) inp;
341   SYMENT *ext =(SYMENT *) extp;
342 
343 #ifdef COFF_ADJUST_SYM_OUT_PRE
344   COFF_ADJUST_SYM_OUT_PRE (abfd, inp, extp);
345 #endif
346 
347   if (in->_n._n_name[0] == 0)
348     {
349       H_PUT_32 (abfd, 0, ext->e.e.e_zeroes);
350       H_PUT_32 (abfd, in->_n._n_n._n_offset, ext->e.e.e_offset);
351     }
352   else
353     {
354 #if SYMNMLEN != E_SYMNMLEN
355 #error we need to cope with truncating or extending SYMNMLEN
356 #else
357       memcpy (ext->e.e_name, in->_n._n_name, SYMNMLEN);
358 #endif
359     }
360 
361   H_PUT_32 (abfd, in->n_value, ext->e_value);
362   H_PUT_16 (abfd, in->n_scnum, ext->e_scnum);
363 
364   if (sizeof (ext->e_type) == 2)
365     H_PUT_16 (abfd, in->n_type, ext->e_type);
366   else
367     H_PUT_32 (abfd, in->n_type, ext->e_type);
368 
369   H_PUT_8 (abfd, in->n_sclass, ext->e_sclass);
370   H_PUT_8 (abfd, in->n_numaux, ext->e_numaux);
371 
372 #ifdef COFF_ADJUST_SYM_OUT_POST
373   COFF_ADJUST_SYM_OUT_POST (abfd, inp, extp);
374 #endif
375 
376   return SYMESZ;
377 }
378 
379 static void
coff_swap_aux_in(bfd * abfd,void * ext1,int type,int in_class,int indx,int numaux,void * in1)380 coff_swap_aux_in (bfd *abfd,
381 		  void * ext1,
382 		  int type,
383 		  int in_class,
384 		  int indx,
385 		  int numaux,
386 		  void * in1)
387 {
388   AUXENT *ext = (AUXENT *) ext1;
389   union internal_auxent *in = (union internal_auxent *) in1;
390 
391 #ifdef COFF_ADJUST_AUX_IN_PRE
392   COFF_ADJUST_AUX_IN_PRE (abfd, ext1, type, in_class, indx, numaux, in1);
393 #endif
394 
395   switch (in_class)
396     {
397     case C_FILE:
398       if (ext->x_file.x_fname[0] == 0)
399 	{
400 	  in->x_file.x_n.x_zeroes = 0;
401 	  in->x_file.x_n.x_offset = H_GET_32 (abfd, ext->x_file.x_n.x_offset);
402 	}
403       else
404 	{
405 #if FILNMLEN != E_FILNMLEN
406 #error we need to cope with truncating or extending FILNMLEN
407 #else
408 	  if (numaux > 1)
409 	    {
410 	      if (indx == 0)
411 		memcpy (in->x_file.x_fname, ext->x_file.x_fname,
412 			numaux * sizeof (AUXENT));
413 	    }
414 	  else
415 	    memcpy (in->x_file.x_fname, ext->x_file.x_fname, FILNMLEN);
416 #endif
417 	}
418       goto end;
419 
420     case C_STAT:
421 #ifdef C_LEAFSTAT
422     case C_LEAFSTAT:
423 #endif
424     case C_HIDDEN:
425       if (type == T_NULL)
426 	{
427 	  in->x_scn.x_scnlen = GET_SCN_SCNLEN (abfd, ext);
428 	  in->x_scn.x_nreloc = GET_SCN_NRELOC (abfd, ext);
429 	  in->x_scn.x_nlinno = GET_SCN_NLINNO (abfd, ext);
430 
431 	  /* PE defines some extra fields; we zero them out for
432              safety.  */
433 	  in->x_scn.x_checksum = 0;
434 	  in->x_scn.x_associated = 0;
435 	  in->x_scn.x_comdat = 0;
436 
437 	  goto end;
438 	}
439       break;
440     }
441 
442   in->x_sym.x_tagndx.l = H_GET_32 (abfd, ext->x_sym.x_tagndx);
443 #ifndef NO_TVNDX
444   in->x_sym.x_tvndx = H_GET_16 (abfd, ext->x_sym.x_tvndx);
445 #endif
446 
447   if (in_class == C_BLOCK || in_class == C_FCN || ISFCN (type)
448       || ISTAG (in_class))
449     {
450       in->x_sym.x_fcnary.x_fcn.x_lnnoptr = GET_FCN_LNNOPTR (abfd, ext);
451       in->x_sym.x_fcnary.x_fcn.x_endndx.l = GET_FCN_ENDNDX (abfd, ext);
452     }
453   else
454     {
455 #if DIMNUM != E_DIMNUM
456 #error we need to cope with truncating or extending DIMNUM
457 #endif
458       in->x_sym.x_fcnary.x_ary.x_dimen[0] =
459 	H_GET_16 (abfd, ext->x_sym.x_fcnary.x_ary.x_dimen[0]);
460       in->x_sym.x_fcnary.x_ary.x_dimen[1] =
461 	H_GET_16 (abfd, ext->x_sym.x_fcnary.x_ary.x_dimen[1]);
462       in->x_sym.x_fcnary.x_ary.x_dimen[2] =
463 	H_GET_16 (abfd, ext->x_sym.x_fcnary.x_ary.x_dimen[2]);
464       in->x_sym.x_fcnary.x_ary.x_dimen[3] =
465 	H_GET_16 (abfd, ext->x_sym.x_fcnary.x_ary.x_dimen[3]);
466     }
467 
468   if (ISFCN (type))
469     in->x_sym.x_misc.x_fsize = H_GET_32 (abfd, ext->x_sym.x_misc.x_fsize);
470   else
471     {
472       in->x_sym.x_misc.x_lnsz.x_lnno = GET_LNSZ_LNNO (abfd, ext);
473       in->x_sym.x_misc.x_lnsz.x_size = GET_LNSZ_SIZE (abfd, ext);
474     }
475 
476  end: ;
477 
478 #ifdef COFF_ADJUST_AUX_IN_POST
479   COFF_ADJUST_AUX_IN_POST (abfd, ext1, type, in_class, indx, numaux, in1);
480 #endif
481 }
482 
483 static unsigned int
coff_swap_aux_out(bfd * abfd,void * inp,int type,int in_class,int indx ATTRIBUTE_UNUSED,int numaux ATTRIBUTE_UNUSED,void * extp)484 coff_swap_aux_out (bfd * abfd,
485 		   void * inp,
486 		   int type,
487 		   int in_class,
488 		   int indx ATTRIBUTE_UNUSED,
489 		   int numaux ATTRIBUTE_UNUSED,
490 		   void * extp)
491 {
492   union internal_auxent * in = (union internal_auxent *) inp;
493   AUXENT *ext = (AUXENT *) extp;
494 
495 #ifdef COFF_ADJUST_AUX_OUT_PRE
496   COFF_ADJUST_AUX_OUT_PRE (abfd, inp, type, in_class, indx, numaux, extp);
497 #endif
498 
499   memset (ext, 0, AUXESZ);
500 
501   switch (in_class)
502     {
503     case C_FILE:
504       if (in->x_file.x_fname[0] == 0)
505 	{
506 	  H_PUT_32 (abfd, 0, ext->x_file.x_n.x_zeroes);
507 	  H_PUT_32 (abfd, in->x_file.x_n.x_offset, ext->x_file.x_n.x_offset);
508 	}
509       else
510 	{
511 #if FILNMLEN != E_FILNMLEN
512 #error we need to cope with truncating or extending FILNMLEN
513 #else
514 	  memcpy (ext->x_file.x_fname, in->x_file.x_fname, FILNMLEN);
515 #endif
516 	}
517       goto end;
518 
519     case C_STAT:
520 #ifdef C_LEAFSTAT
521     case C_LEAFSTAT:
522 #endif
523     case C_HIDDEN:
524       if (type == T_NULL)
525 	{
526 	  PUT_SCN_SCNLEN (abfd, in->x_scn.x_scnlen, ext);
527 	  PUT_SCN_NRELOC (abfd, in->x_scn.x_nreloc, ext);
528 	  PUT_SCN_NLINNO (abfd, in->x_scn.x_nlinno, ext);
529 	  goto end;
530 	}
531       break;
532     }
533 
534   H_PUT_32 (abfd, in->x_sym.x_tagndx.l, ext->x_sym.x_tagndx);
535 #ifndef NO_TVNDX
536   H_PUT_16 (abfd, in->x_sym.x_tvndx, ext->x_sym.x_tvndx);
537 #endif
538 
539   if (in_class == C_BLOCK || in_class == C_FCN || ISFCN (type)
540       || ISTAG (in_class))
541     {
542       PUT_FCN_LNNOPTR (abfd, in->x_sym.x_fcnary.x_fcn.x_lnnoptr, ext);
543       PUT_FCN_ENDNDX (abfd, in->x_sym.x_fcnary.x_fcn.x_endndx.l, ext);
544     }
545   else
546     {
547 #if DIMNUM != E_DIMNUM
548 #error we need to cope with truncating or extending DIMNUM
549 #endif
550       H_PUT_16 (abfd, in->x_sym.x_fcnary.x_ary.x_dimen[0],
551 	       ext->x_sym.x_fcnary.x_ary.x_dimen[0]);
552       H_PUT_16 (abfd, in->x_sym.x_fcnary.x_ary.x_dimen[1],
553 	       ext->x_sym.x_fcnary.x_ary.x_dimen[1]);
554       H_PUT_16 (abfd, in->x_sym.x_fcnary.x_ary.x_dimen[2],
555 	       ext->x_sym.x_fcnary.x_ary.x_dimen[2]);
556       H_PUT_16 (abfd, in->x_sym.x_fcnary.x_ary.x_dimen[3],
557 	       ext->x_sym.x_fcnary.x_ary.x_dimen[3]);
558     }
559 
560   if (ISFCN (type))
561     H_PUT_32 (abfd, in->x_sym.x_misc.x_fsize, ext->x_sym.x_misc.x_fsize);
562   else
563     {
564       PUT_LNSZ_LNNO (abfd, in->x_sym.x_misc.x_lnsz.x_lnno, ext);
565       PUT_LNSZ_SIZE (abfd, in->x_sym.x_misc.x_lnsz.x_size, ext);
566     }
567 
568  end:
569 #ifdef COFF_ADJUST_AUX_OUT_POST
570   COFF_ADJUST_AUX_OUT_POST (abfd, inp, type, in_class, indx, numaux, extp);
571 #endif
572   return AUXESZ;
573 }
574 
575 #endif /* NO_COFF_SYMBOLS */
576 
577 #ifndef NO_COFF_LINENOS
578 
579 static void
coff_swap_lineno_in(bfd * abfd,void * ext1,void * in1)580 coff_swap_lineno_in (bfd * abfd, void * ext1, void * in1)
581 {
582   LINENO *ext = (LINENO *) ext1;
583   struct internal_lineno *in = (struct internal_lineno *) in1;
584 
585   in->l_addr.l_symndx = H_GET_32 (abfd, ext->l_addr.l_symndx);
586   in->l_lnno = GET_LINENO_LNNO (abfd, ext);
587 }
588 
589 static unsigned int
coff_swap_lineno_out(bfd * abfd,void * inp,void * outp)590 coff_swap_lineno_out (bfd * abfd, void * inp, void * outp)
591 {
592   struct internal_lineno *in = (struct internal_lineno *) inp;
593   struct external_lineno *ext = (struct external_lineno *) outp;
594   H_PUT_32 (abfd, in->l_addr.l_symndx, ext->l_addr.l_symndx);
595 
596   PUT_LINENO_LNNO (abfd, in->l_lnno, ext);
597   return LINESZ;
598 }
599 
600 #endif /* NO_COFF_LINENOS */
601 
602 static void
coff_swap_aouthdr_in(bfd * abfd,void * aouthdr_ext1,void * aouthdr_int1)603 coff_swap_aouthdr_in (bfd * abfd, void * aouthdr_ext1, void * aouthdr_int1)
604 {
605   AOUTHDR *aouthdr_ext;
606   struct internal_aouthdr *aouthdr_int;
607 
608   aouthdr_ext = (AOUTHDR *) aouthdr_ext1;
609   aouthdr_int = (struct internal_aouthdr *) aouthdr_int1;
610   aouthdr_int->magic = H_GET_16 (abfd, aouthdr_ext->magic);
611   aouthdr_int->vstamp = H_GET_16 (abfd, aouthdr_ext->vstamp);
612   aouthdr_int->tsize = GET_AOUTHDR_TSIZE (abfd, aouthdr_ext->tsize);
613   aouthdr_int->dsize = GET_AOUTHDR_DSIZE (abfd, aouthdr_ext->dsize);
614   aouthdr_int->bsize = GET_AOUTHDR_BSIZE (abfd, aouthdr_ext->bsize);
615   aouthdr_int->entry = GET_AOUTHDR_ENTRY (abfd, aouthdr_ext->entry);
616   aouthdr_int->text_start =
617     GET_AOUTHDR_TEXT_START (abfd, aouthdr_ext->text_start);
618   aouthdr_int->data_start =
619     GET_AOUTHDR_DATA_START (abfd, aouthdr_ext->data_start);
620 
621 #ifdef I960
622   aouthdr_int->tagentries = H_GET_32 (abfd, aouthdr_ext->tagentries);
623 #endif
624 
625 #ifdef APOLLO_M68
626   H_PUT_32 (abfd, aouthdr_int->o_inlib, aouthdr_ext->o_inlib);
627   H_PUT_32 (abfd, aouthdr_int->o_sri, aouthdr_ext->o_sri);
628   H_PUT_32 (abfd, aouthdr_int->vid[0], aouthdr_ext->vid);
629   H_PUT_32 (abfd, aouthdr_int->vid[1], aouthdr_ext->vid + 4);
630 #endif
631 
632 #ifdef RS6000COFF_C
633 #ifdef XCOFF64
634   aouthdr_int->o_toc = H_GET_64 (abfd, aouthdr_ext->o_toc);
635 #else
636   aouthdr_int->o_toc = H_GET_32 (abfd, aouthdr_ext->o_toc);
637 #endif
638   aouthdr_int->o_snentry  = H_GET_16 (abfd, aouthdr_ext->o_snentry);
639   aouthdr_int->o_sntext   = H_GET_16 (abfd, aouthdr_ext->o_sntext);
640   aouthdr_int->o_sndata   = H_GET_16 (abfd, aouthdr_ext->o_sndata);
641   aouthdr_int->o_sntoc    = H_GET_16 (abfd, aouthdr_ext->o_sntoc);
642   aouthdr_int->o_snloader = H_GET_16 (abfd, aouthdr_ext->o_snloader);
643   aouthdr_int->o_snbss    = H_GET_16 (abfd, aouthdr_ext->o_snbss);
644   aouthdr_int->o_algntext = H_GET_16 (abfd, aouthdr_ext->o_algntext);
645   aouthdr_int->o_algndata = H_GET_16 (abfd, aouthdr_ext->o_algndata);
646   aouthdr_int->o_modtype  = H_GET_16 (abfd, aouthdr_ext->o_modtype);
647   aouthdr_int->o_cputype  = H_GET_16 (abfd, aouthdr_ext->o_cputype);
648 #ifdef XCOFF64
649   aouthdr_int->o_maxstack = H_GET_64 (abfd, aouthdr_ext->o_maxstack);
650   aouthdr_int->o_maxdata  = H_GET_64 (abfd, aouthdr_ext->o_maxdata);
651 #else
652   aouthdr_int->o_maxstack = H_GET_32 (abfd, aouthdr_ext->o_maxstack);
653   aouthdr_int->o_maxdata  = H_GET_32 (abfd, aouthdr_ext->o_maxdata);
654 #endif
655 #endif
656 
657 #ifdef MIPSECOFF
658   aouthdr_int->bss_start  = H_GET_32 (abfd, aouthdr_ext->bss_start);
659   aouthdr_int->gp_value   = H_GET_32 (abfd, aouthdr_ext->gp_value);
660   aouthdr_int->gprmask    = H_GET_32 (abfd, aouthdr_ext->gprmask);
661   aouthdr_int->cprmask[0] = H_GET_32 (abfd, aouthdr_ext->cprmask[0]);
662   aouthdr_int->cprmask[1] = H_GET_32 (abfd, aouthdr_ext->cprmask[1]);
663   aouthdr_int->cprmask[2] = H_GET_32 (abfd, aouthdr_ext->cprmask[2]);
664   aouthdr_int->cprmask[3] = H_GET_32 (abfd, aouthdr_ext->cprmask[3]);
665 #endif
666 
667 #ifdef ALPHAECOFF
668   aouthdr_int->bss_start = H_GET_64 (abfd, aouthdr_ext->bss_start);
669   aouthdr_int->gp_value  = H_GET_64 (abfd, aouthdr_ext->gp_value);
670   aouthdr_int->gprmask   = H_GET_32 (abfd, aouthdr_ext->gprmask);
671   aouthdr_int->fprmask   = H_GET_32 (abfd, aouthdr_ext->fprmask);
672 #endif
673 }
674 
675 static unsigned int
coff_swap_aouthdr_out(bfd * abfd,void * in,void * out)676 coff_swap_aouthdr_out (bfd * abfd, void * in, void * out)
677 {
678   struct internal_aouthdr *aouthdr_in = (struct internal_aouthdr *) in;
679   AOUTHDR *aouthdr_out = (AOUTHDR *) out;
680 
681   H_PUT_16 (abfd, aouthdr_in->magic, aouthdr_out->magic);
682   H_PUT_16 (abfd, aouthdr_in->vstamp, aouthdr_out->vstamp);
683   PUT_AOUTHDR_TSIZE (abfd, aouthdr_in->tsize, aouthdr_out->tsize);
684   PUT_AOUTHDR_DSIZE (abfd, aouthdr_in->dsize, aouthdr_out->dsize);
685   PUT_AOUTHDR_BSIZE (abfd, aouthdr_in->bsize, aouthdr_out->bsize);
686   PUT_AOUTHDR_ENTRY (abfd, aouthdr_in->entry, aouthdr_out->entry);
687   PUT_AOUTHDR_TEXT_START (abfd, aouthdr_in->text_start,
688 			  aouthdr_out->text_start);
689   PUT_AOUTHDR_DATA_START (abfd, aouthdr_in->data_start,
690 			  aouthdr_out->data_start);
691 
692 #ifdef I960
693   H_PUT_32 (abfd, aouthdr_in->tagentries, aouthdr_out->tagentries);
694 #endif
695 
696 #ifdef RS6000COFF_C
697 #ifdef XCOFF64
698   H_PUT_64 (abfd, aouthdr_in->o_toc, aouthdr_out->o_toc);
699 #else
700   H_PUT_32 (abfd, aouthdr_in->o_toc, aouthdr_out->o_toc);
701 #endif
702   H_PUT_16 (abfd, aouthdr_in->o_snentry, aouthdr_out->o_snentry);
703   H_PUT_16 (abfd, aouthdr_in->o_sntext, aouthdr_out->o_sntext);
704   H_PUT_16 (abfd, aouthdr_in->o_sndata, aouthdr_out->o_sndata);
705   H_PUT_16 (abfd, aouthdr_in->o_sntoc, aouthdr_out->o_sntoc);
706   H_PUT_16 (abfd, aouthdr_in->o_snloader, aouthdr_out->o_snloader);
707   H_PUT_16 (abfd, aouthdr_in->o_snbss, aouthdr_out->o_snbss);
708   H_PUT_16 (abfd, aouthdr_in->o_algntext, aouthdr_out->o_algntext);
709   H_PUT_16 (abfd, aouthdr_in->o_algndata, aouthdr_out->o_algndata);
710   H_PUT_16 (abfd, aouthdr_in->o_modtype, aouthdr_out->o_modtype);
711   H_PUT_16 (abfd, aouthdr_in->o_cputype, aouthdr_out->o_cputype);
712 #ifdef XCOFF64
713   H_PUT_64 (abfd, aouthdr_in->o_maxstack, aouthdr_out->o_maxstack);
714   H_PUT_64 (abfd, aouthdr_in->o_maxdata, aouthdr_out->o_maxdata);
715 #else
716   H_PUT_32 (abfd, aouthdr_in->o_maxstack, aouthdr_out->o_maxstack);
717   H_PUT_32 (abfd, aouthdr_in->o_maxdata, aouthdr_out->o_maxdata);
718 #endif
719   memset (aouthdr_out->o_resv2, 0, sizeof aouthdr_out->o_resv2);
720 #ifdef XCOFF64
721   memset (aouthdr_out->o_debugger, 0, sizeof aouthdr_out->o_debugger);
722   memset (aouthdr_out->o_resv3, 0, sizeof aouthdr_out->o_resv3);
723 #endif
724 #endif
725 
726 #ifdef MIPSECOFF
727   H_PUT_32 (abfd, aouthdr_in->bss_start, aouthdr_out->bss_start);
728   H_PUT_32 (abfd, aouthdr_in->gp_value, aouthdr_out->gp_value);
729   H_PUT_32 (abfd, aouthdr_in->gprmask, aouthdr_out->gprmask);
730   H_PUT_32 (abfd, aouthdr_in->cprmask[0], aouthdr_out->cprmask[0]);
731   H_PUT_32 (abfd, aouthdr_in->cprmask[1], aouthdr_out->cprmask[1]);
732   H_PUT_32 (abfd, aouthdr_in->cprmask[2], aouthdr_out->cprmask[2]);
733   H_PUT_32 (abfd, aouthdr_in->cprmask[3], aouthdr_out->cprmask[3]);
734 #endif
735 
736 #ifdef ALPHAECOFF
737   /* FIXME: What does bldrev mean?  */
738   H_PUT_16 (abfd, 2, aouthdr_out->bldrev);
739   H_PUT_16 (abfd, 0, aouthdr_out->padding);
740   H_PUT_64 (abfd, aouthdr_in->bss_start, aouthdr_out->bss_start);
741   H_PUT_64 (abfd, aouthdr_in->gp_value, aouthdr_out->gp_value);
742   H_PUT_32 (abfd, aouthdr_in->gprmask, aouthdr_out->gprmask);
743   H_PUT_32 (abfd, aouthdr_in->fprmask, aouthdr_out->fprmask);
744 #endif
745 
746   return AOUTSZ;
747 }
748 
749 static void
coff_swap_scnhdr_in(bfd * abfd,void * ext,void * in)750 coff_swap_scnhdr_in (bfd * abfd, void * ext, void * in)
751 {
752   SCNHDR *scnhdr_ext = (SCNHDR *) ext;
753   struct internal_scnhdr *scnhdr_int = (struct internal_scnhdr *) in;
754 
755 #ifdef COFF_ADJUST_SCNHDR_IN_PRE
756   COFF_ADJUST_SCNHDR_IN_PRE (abfd, ext, in);
757 #endif
758   memcpy (scnhdr_int->s_name, scnhdr_ext->s_name, sizeof (scnhdr_int->s_name));
759 
760   scnhdr_int->s_vaddr = GET_SCNHDR_VADDR (abfd, scnhdr_ext->s_vaddr);
761   scnhdr_int->s_paddr = GET_SCNHDR_PADDR (abfd, scnhdr_ext->s_paddr);
762   scnhdr_int->s_size = GET_SCNHDR_SIZE (abfd, scnhdr_ext->s_size);
763 
764   scnhdr_int->s_scnptr = GET_SCNHDR_SCNPTR (abfd, scnhdr_ext->s_scnptr);
765   scnhdr_int->s_relptr = GET_SCNHDR_RELPTR (abfd, scnhdr_ext->s_relptr);
766   scnhdr_int->s_lnnoptr = GET_SCNHDR_LNNOPTR (abfd, scnhdr_ext->s_lnnoptr);
767   scnhdr_int->s_flags = GET_SCNHDR_FLAGS (abfd, scnhdr_ext->s_flags);
768   scnhdr_int->s_nreloc = GET_SCNHDR_NRELOC (abfd, scnhdr_ext->s_nreloc);
769   scnhdr_int->s_nlnno = GET_SCNHDR_NLNNO (abfd, scnhdr_ext->s_nlnno);
770 #ifdef I960
771   scnhdr_int->s_align = GET_SCNHDR_ALIGN (abfd, scnhdr_ext->s_align);
772 #endif
773 #ifdef COFF_ADJUST_SCNHDR_IN_POST
774   COFF_ADJUST_SCNHDR_IN_POST (abfd, ext, in);
775 #endif
776 }
777 
778 static unsigned int
coff_swap_scnhdr_out(bfd * abfd,void * in,void * out)779 coff_swap_scnhdr_out (bfd * abfd, void * in, void * out)
780 {
781   struct internal_scnhdr *scnhdr_int = (struct internal_scnhdr *) in;
782   SCNHDR *scnhdr_ext = (SCNHDR *) out;
783   unsigned int ret = bfd_coff_scnhsz (abfd);
784 
785 #ifdef COFF_ADJUST_SCNHDR_OUT_PRE
786   COFF_ADJUST_SCNHDR_OUT_PRE (abfd, in, out);
787 #endif
788   memcpy (scnhdr_ext->s_name, scnhdr_int->s_name, sizeof (scnhdr_int->s_name));
789 
790   PUT_SCNHDR_VADDR (abfd, scnhdr_int->s_vaddr, scnhdr_ext->s_vaddr);
791   PUT_SCNHDR_PADDR (abfd, scnhdr_int->s_paddr, scnhdr_ext->s_paddr);
792   PUT_SCNHDR_SIZE (abfd, scnhdr_int->s_size, scnhdr_ext->s_size);
793   PUT_SCNHDR_SCNPTR (abfd, scnhdr_int->s_scnptr, scnhdr_ext->s_scnptr);
794   PUT_SCNHDR_RELPTR (abfd, scnhdr_int->s_relptr, scnhdr_ext->s_relptr);
795   PUT_SCNHDR_LNNOPTR (abfd, scnhdr_int->s_lnnoptr, scnhdr_ext->s_lnnoptr);
796   PUT_SCNHDR_FLAGS (abfd, scnhdr_int->s_flags, scnhdr_ext->s_flags);
797 #if defined(M88)
798   H_PUT_32 (abfd, scnhdr_int->s_nlnno, scnhdr_ext->s_nlnno);
799   H_PUT_32 (abfd, scnhdr_int->s_nreloc, scnhdr_ext->s_nreloc);
800 #else
801   if (scnhdr_int->s_nlnno <= MAX_SCNHDR_NLNNO)
802     PUT_SCNHDR_NLNNO (abfd, scnhdr_int->s_nlnno, scnhdr_ext->s_nlnno);
803   else
804     {
805       char buf[sizeof (scnhdr_int->s_name) + 1];
806 
807       memcpy (buf, scnhdr_int->s_name, sizeof (scnhdr_int->s_name));
808       buf[sizeof (scnhdr_int->s_name)] = '\0';
809       (*_bfd_error_handler)
810 	(_("%s: warning: %s: line number overflow: 0x%lx > 0xffff"),
811 	 bfd_get_filename (abfd),
812 	 buf, scnhdr_int->s_nlnno);
813       PUT_SCNHDR_NLNNO (abfd, 0xffff, scnhdr_ext->s_nlnno);
814     }
815 
816   if (scnhdr_int->s_nreloc <= MAX_SCNHDR_NRELOC)
817     PUT_SCNHDR_NRELOC (abfd, scnhdr_int->s_nreloc, scnhdr_ext->s_nreloc);
818   else
819     {
820       char buf[sizeof (scnhdr_int->s_name) + 1];
821 
822       memcpy (buf, scnhdr_int->s_name, sizeof (scnhdr_int->s_name));
823       buf[sizeof (scnhdr_int->s_name)] = '\0';
824       (*_bfd_error_handler) (_("%s: %s: reloc overflow: 0x%lx > 0xffff"),
825 			     bfd_get_filename (abfd),
826 			     buf, scnhdr_int->s_nreloc);
827       bfd_set_error (bfd_error_file_truncated);
828       PUT_SCNHDR_NRELOC (abfd, 0xffff, scnhdr_ext->s_nreloc);
829       ret = 0;
830     }
831 #endif
832 
833 #ifdef I960
834   PUT_SCNHDR_ALIGN (abfd, scnhdr_int->s_align, scnhdr_ext->s_align);
835 #endif
836 #ifdef COFF_ADJUST_SCNHDR_OUT_POST
837   COFF_ADJUST_SCNHDR_OUT_POST (abfd, in, out);
838 #endif
839   return ret;
840 }
841