source: trunk/src/binutils/bfd/nlm32-ppc.c@ 323

Last change on this file since 323 was 10, checked in by bird, 23 years ago

Initial revision

  • Property cvs2svn:cvs-rev set to 1.1
  • Property svn:eol-style set to native
  • Property svn:executable set to *
File size: 30.2 KB
Line 
1/* Support for 32-bit PowerPC NLM (NetWare Loadable Module)
2 Copyright 1994, 1995, 2000 Free Software Foundation, Inc.
3
4This file is part of BFD, the Binary File Descriptor library.
5
6This program is free software; you can redistribute it and/or modify
7it under the terms of the GNU General Public License as published by
8the Free Software Foundation; either version 2 of the License, or
9(at your option) any later version.
10
11This program is distributed in the hope that it will be useful,
12but WITHOUT ANY WARRANTY; without even the implied warranty of
13MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14GNU General Public License for more details.
15
16You should have received a copy of the GNU General Public License
17along with this program; if not, write to the Free Software
18Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. */
19
20#include "bfd.h"
21#include "sysdep.h"
22#include "libbfd.h"
23
24/* The format of a PowerPC NLM changed. Define OLDFORMAT to get the
25 old format. */
26
27#define ARCH_SIZE 32
28
29#include "nlm/ppc-ext.h"
30#define Nlm_External_Fixed_Header Nlm32_powerpc_External_Fixed_Header
31
32#include "libnlm.h"
33
34#ifdef OLDFORMAT
35static boolean nlm_powerpc_backend_object_p
36 PARAMS ((bfd *));
37static boolean nlm_powerpc_write_prefix
38 PARAMS ((bfd *));
39#endif
40
41static boolean nlm_powerpc_read_reloc
42 PARAMS ((bfd *, nlmNAME(symbol_type) *, asection **, arelent *));
43static boolean nlm_powerpc_mangle_relocs
44 PARAMS ((bfd *, asection *, PTR, bfd_vma, bfd_size_type));
45static boolean nlm_powerpc_read_import
46 PARAMS ((bfd *, nlmNAME(symbol_type) *));
47
48#ifdef OLDFORMAT
49static boolean nlm_powerpc_write_reloc
50 PARAMS ((bfd *, asection *, arelent *, int));
51#endif
52
53static boolean nlm_powerpc_write_import
54 PARAMS ((bfd *, asection *, arelent *));
55static boolean nlm_powerpc_write_external
56 PARAMS ((bfd *, bfd_size_type, asymbol *, struct reloc_and_sec *));
57
58#ifndef OLDFORMAT
59static boolean nlm_powerpc_set_public_section
60 PARAMS ((bfd *, nlmNAME(symbol_type) *));
61static bfd_vma nlm_powerpc_get_public_offset
62 PARAMS ((bfd *, asymbol *));
63#endif
64
65
66#ifdef OLDFORMAT
67
68/* The prefix header is only used in the old format. */
69
70/* PowerPC NLM's have a prefix header before the standard NLM. This
71 function reads it in, verifies the version, and seeks the bfd to
72 the location before the regular NLM header. */
73
74static boolean
75nlm_powerpc_backend_object_p (abfd)
76 bfd *abfd;
77{
78 struct nlm32_powerpc_external_prefix_header s;
79
80 if (bfd_read ((PTR) &s, sizeof s, 1, abfd) != sizeof s)
81 return false;
82
83 if (memcmp (s.signature, NLM32_POWERPC_SIGNATURE, sizeof s.signature) != 0
84 || bfd_h_get_32 (abfd, s.headerVersion) != NLM32_POWERPC_HEADER_VERSION)
85 return false;
86
87 return true;
88}
89
90/* Write out the prefix. */
91
92static boolean
93nlm_powerpc_write_prefix (abfd)
94 bfd *abfd;
95{
96 struct nlm32_powerpc_external_prefix_header s;
97
98 memset (&s, 0, sizeof s);
99 memcpy (s.signature, NLM32_POWERPC_SIGNATURE, sizeof s.signature);
100 bfd_h_put_32 (abfd, (bfd_vma) NLM32_POWERPC_HEADER_VERSION, s.headerVersion);
101 bfd_h_put_32 (abfd, (bfd_vma) 0, s.origins);
102
103 /* FIXME: What should we do about the date? */
104
105 if (bfd_write ((PTR) &s, sizeof s, 1, abfd) != sizeof s)
106 return false;
107
108 return true;
109}
110
111#endif /* OLDFORMAT */
112
113
114#ifndef OLDFORMAT
115
116/* There is only one type of reloc in a PowerPC NLM. */
117
118static reloc_howto_type nlm_powerpc_howto =
119 HOWTO (0, /* type */
120 0, /* rightshift */
121 2, /* size (0 = byte, 1 = short, 2 = long) */
122 32, /* bitsize */
123 false, /* pc_relative */
124 0, /* bitpos */
125 complain_overflow_bitfield, /* complain_on_overflow */
126 0, /* special_function */
127 "32", /* name */
128 true, /* partial_inplace */
129 0xffffffff, /* src_mask */
130 0xffffffff, /* dst_mask */
131 false); /* pcrel_offset */
132
133/* Read a PowerPC NLM reloc. */
134
135static boolean
136nlm_powerpc_read_reloc (abfd, sym, secp, rel)
137 bfd *abfd;
138 nlmNAME(symbol_type) *sym;
139 asection **secp;
140 arelent *rel;
141{
142 bfd_byte temp[4];
143 bfd_vma val;
144 const char *name;
145
146 if (bfd_read (temp, sizeof (temp), 1, abfd) != sizeof (temp))
147 return false;
148
149 val = bfd_get_32 (abfd, temp);
150
151 /* The value is a word offset into either the code or data segment.
152 This is the location which needs to be adjusted.
153
154 The high bit is 0 if the value is an offset into the data
155 segment, or 1 if the value is an offset into the text segment.
156
157 If this is a relocation fixup rather than an imported symbol (the
158 sym argument is NULL), then the second most significant bit is 0
159 if the address of the data segment should be added to the
160 location addressed by the value, or 1 if the address of the text
161 segment should be added.
162
163 If this is an imported symbol, the second most significant bit is
164 not used and must be 0. */
165
166 if ((val & NLM_HIBIT) == 0)
167 name = NLM_INITIALIZED_DATA_NAME;
168 else
169 {
170 name = NLM_CODE_NAME;
171 val &=~ NLM_HIBIT;
172 }
173 *secp = bfd_get_section_by_name (abfd, name);
174
175 if (sym == NULL)
176 {
177 if ((val & (NLM_HIBIT >> 1)) == 0)
178 name = NLM_INITIALIZED_DATA_NAME;
179 else
180 {
181 name = NLM_CODE_NAME;
182 val &=~ (NLM_HIBIT >> 1);
183 }
184 rel->sym_ptr_ptr = bfd_get_section_by_name (abfd, name)->symbol_ptr_ptr;
185 }
186
187 rel->howto = &nlm_powerpc_howto;
188
189 rel->address = val << 2;
190 rel->addend = 0;
191
192 return true;
193}
194
195#else /* OLDFORMAT */
196
197/* This reloc handling is only applicable to the old format. */
198
199/* How to process the various reloc types. PowerPC NLMs use XCOFF
200 reloc types, and I have just copied the XCOFF reloc table here. */
201
202static reloc_howto_type nlm_powerpc_howto_table[] =
203{
204 /* Standard 32 bit relocation. */
205 HOWTO (0, /* type */
206 0, /* rightshift */
207 2, /* size (0 = byte, 1 = short, 2 = long) */
208 32, /* bitsize */
209 false, /* pc_relative */
210 0, /* bitpos */
211 complain_overflow_bitfield, /* complain_on_overflow */
212 0, /* special_function */
213 "R_POS", /* name */
214 true, /* partial_inplace */
215 0xffffffff, /* src_mask */
216 0xffffffff, /* dst_mask */
217 false), /* pcrel_offset */
218
219 /* 32 bit relocation, but store negative value. */
220 HOWTO (1, /* type */
221 0, /* rightshift */
222 -2, /* size (0 = byte, 1 = short, 2 = long) */
223 32, /* bitsize */
224 false, /* pc_relative */
225 0, /* bitpos */
226 complain_overflow_bitfield, /* complain_on_overflow */
227 0, /* special_function */
228 "R_NEG", /* name */
229 true, /* partial_inplace */
230 0xffffffff, /* src_mask */
231 0xffffffff, /* dst_mask */
232 false), /* pcrel_offset */
233
234 /* 32 bit PC relative relocation. */
235 HOWTO (2, /* type */
236 0, /* rightshift */
237 2, /* size (0 = byte, 1 = short, 2 = long) */
238 32, /* bitsize */
239 true, /* pc_relative */
240 0, /* bitpos */
241 complain_overflow_signed, /* complain_on_overflow */
242 0, /* special_function */
243 "R_REL", /* name */
244 true, /* partial_inplace */
245 0xffffffff, /* src_mask */
246 0xffffffff, /* dst_mask */
247 false), /* pcrel_offset */
248
249 /* 16 bit TOC relative relocation. */
250 HOWTO (3, /* type */
251 0, /* rightshift */
252 1, /* size (0 = byte, 1 = short, 2 = long) */
253 16, /* bitsize */
254 false, /* pc_relative */
255 0, /* bitpos */
256 complain_overflow_signed, /* complain_on_overflow */
257 0, /* special_function */
258 "R_TOC", /* name */
259 true, /* partial_inplace */
260 0xffff, /* src_mask */
261 0xffff, /* dst_mask */
262 false), /* pcrel_offset */
263
264 /* I don't really know what this is. */
265 HOWTO (4, /* type */
266 1, /* rightshift */
267 2, /* size (0 = byte, 1 = short, 2 = long) */
268 32, /* bitsize */
269 false, /* pc_relative */
270 0, /* bitpos */
271 complain_overflow_bitfield, /* complain_on_overflow */
272 0, /* special_function */
273 "R_RTB", /* name */
274 true, /* partial_inplace */
275 0xffffffff, /* src_mask */
276 0xffffffff, /* dst_mask */
277 false), /* pcrel_offset */
278
279 /* External TOC relative symbol. */
280 HOWTO (5, /* type */
281 0, /* rightshift */
282 2, /* size (0 = byte, 1 = short, 2 = long) */
283 16, /* bitsize */
284 false, /* pc_relative */
285 0, /* bitpos */
286 complain_overflow_bitfield, /* complain_on_overflow */
287 0, /* special_function */
288 "R_GL", /* name */
289 true, /* partial_inplace */
290 0xffff, /* src_mask */
291 0xffff, /* dst_mask */
292 false), /* pcrel_offset */
293
294 /* Local TOC relative symbol. */
295 HOWTO (6, /* type */
296 0, /* rightshift */
297 2, /* size (0 = byte, 1 = short, 2 = long) */
298 16, /* bitsize */
299 false, /* pc_relative */
300 0, /* bitpos */
301 complain_overflow_bitfield, /* complain_on_overflow */
302 0, /* special_function */
303 "R_TCL", /* name */
304 true, /* partial_inplace */
305 0xffff, /* src_mask */
306 0xffff, /* dst_mask */
307 false), /* pcrel_offset */
308
309 { 7 },
310
311 /* Non modifiable absolute branch. */
312 HOWTO (8, /* type */
313 0, /* rightshift */
314 2, /* size (0 = byte, 1 = short, 2 = long) */
315 26, /* bitsize */
316 false, /* pc_relative */
317 0, /* bitpos */
318 complain_overflow_bitfield, /* complain_on_overflow */
319 0, /* special_function */
320 "R_BA", /* name */
321 true, /* partial_inplace */
322 0x3fffffc, /* src_mask */
323 0x3fffffc, /* dst_mask */
324 false), /* pcrel_offset */
325
326 { 9 },
327
328 /* Non modifiable relative branch. */
329 HOWTO (0xa, /* type */
330 0, /* rightshift */
331 2, /* size (0 = byte, 1 = short, 2 = long) */
332 26, /* bitsize */
333 true, /* pc_relative */
334 0, /* bitpos */
335 complain_overflow_signed, /* complain_on_overflow */
336 0, /* special_function */
337 "R_BR", /* name */
338 true, /* partial_inplace */
339 0x3fffffc, /* src_mask */
340 0x3fffffc, /* dst_mask */
341 false), /* pcrel_offset */
342
343 { 0xb },
344
345 /* Indirect load. */
346 HOWTO (0xc, /* type */
347 0, /* rightshift */
348 2, /* size (0 = byte, 1 = short, 2 = long) */
349 16, /* bitsize */
350 false, /* pc_relative */
351 0, /* bitpos */
352 complain_overflow_bitfield, /* complain_on_overflow */
353 0, /* special_function */
354 "R_RL", /* name */
355 true, /* partial_inplace */
356 0xffff, /* src_mask */
357 0xffff, /* dst_mask */
358 false), /* pcrel_offset */
359
360 /* Load address. */
361 HOWTO (0xd, /* type */
362 0, /* rightshift */
363 2, /* size (0 = byte, 1 = short, 2 = long) */
364 16, /* bitsize */
365 false, /* pc_relative */
366 0, /* bitpos */
367 complain_overflow_bitfield, /* complain_on_overflow */
368 0, /* special_function */
369 "R_RLA", /* name */
370 true, /* partial_inplace */
371 0xffff, /* src_mask */
372 0xffff, /* dst_mask */
373 false), /* pcrel_offset */
374
375 { 0xe },
376
377 /* Non-relocating reference. */
378 HOWTO (0xf, /* type */
379 0, /* rightshift */
380 2, /* size (0 = byte, 1 = short, 2 = long) */
381 32, /* bitsize */
382 false, /* pc_relative */
383 0, /* bitpos */
384 complain_overflow_bitfield, /* complain_on_overflow */
385 0, /* special_function */
386 "R_REF", /* name */
387 false, /* partial_inplace */
388 0, /* src_mask */
389 0, /* dst_mask */
390 false), /* pcrel_offset */
391
392 { 0x10 },
393 { 0x11 },
394
395 /* TOC relative indirect load. */
396 HOWTO (0x12, /* type */
397 0, /* rightshift */
398 2, /* size (0 = byte, 1 = short, 2 = long) */
399 16, /* bitsize */
400 false, /* pc_relative */
401 0, /* bitpos */
402 complain_overflow_bitfield, /* complain_on_overflow */
403 0, /* special_function */
404 "R_TRL", /* name */
405 true, /* partial_inplace */
406 0xffff, /* src_mask */
407 0xffff, /* dst_mask */
408 false), /* pcrel_offset */
409
410 /* TOC relative load address. */
411 HOWTO (0x13, /* type */
412 0, /* rightshift */
413 2, /* size (0 = byte, 1 = short, 2 = long) */
414 16, /* bitsize */
415 false, /* pc_relative */
416 0, /* bitpos */
417 complain_overflow_bitfield, /* complain_on_overflow */
418 0, /* special_function */
419 "R_TRLA", /* name */
420 true, /* partial_inplace */
421 0xffff, /* src_mask */
422 0xffff, /* dst_mask */
423 false), /* pcrel_offset */
424
425 /* Modifiable relative branch. */
426 HOWTO (0x14, /* type */
427 1, /* rightshift */
428 2, /* size (0 = byte, 1 = short, 2 = long) */
429 32, /* bitsize */
430 false, /* pc_relative */
431 0, /* bitpos */
432 complain_overflow_bitfield, /* complain_on_overflow */
433 0, /* special_function */
434 "R_RRTBI", /* name */
435 true, /* partial_inplace */
436 0xffffffff, /* src_mask */
437 0xffffffff, /* dst_mask */
438 false), /* pcrel_offset */
439
440 /* Modifiable absolute branch. */
441 HOWTO (0x15, /* type */
442 1, /* rightshift */
443 2, /* size (0 = byte, 1 = short, 2 = long) */
444 32, /* bitsize */
445 false, /* pc_relative */
446 0, /* bitpos */
447 complain_overflow_bitfield, /* complain_on_overflow */
448 0, /* special_function */
449 "R_RRTBA", /* name */
450 true, /* partial_inplace */
451 0xffffffff, /* src_mask */
452 0xffffffff, /* dst_mask */
453 false), /* pcrel_offset */
454
455 /* Modifiable call absolute indirect. */
456 HOWTO (0x16, /* type */
457 0, /* rightshift */
458 2, /* size (0 = byte, 1 = short, 2 = long) */
459 16, /* bitsize */
460 false, /* pc_relative */
461 0, /* bitpos */
462 complain_overflow_bitfield, /* complain_on_overflow */
463 0, /* special_function */
464 "R_CAI", /* name */
465 true, /* partial_inplace */
466 0xffff, /* src_mask */
467 0xffff, /* dst_mask */
468 false), /* pcrel_offset */
469
470 /* Modifiable call relative. */
471 HOWTO (0x17, /* type */
472 0, /* rightshift */
473 2, /* size (0 = byte, 1 = short, 2 = long) */
474 16, /* bitsize */
475 false, /* pc_relative */
476 0, /* bitpos */
477 complain_overflow_bitfield, /* complain_on_overflow */
478 0, /* special_function */
479 "R_REL", /* name */
480 true, /* partial_inplace */
481 0xffff, /* src_mask */
482 0xffff, /* dst_mask */
483 false), /* pcrel_offset */
484
485 /* Modifiable branch absolute. */
486 HOWTO (0x18, /* type */
487 0, /* rightshift */
488 2, /* size (0 = byte, 1 = short, 2 = long) */
489 16, /* bitsize */
490 false, /* pc_relative */
491 0, /* bitpos */
492 complain_overflow_bitfield, /* complain_on_overflow */
493 0, /* special_function */
494 "R_RBA", /* name */
495 true, /* partial_inplace */
496 0xffff, /* src_mask */
497 0xffff, /* dst_mask */
498 false), /* pcrel_offset */
499
500 /* Modifiable branch absolute. */
501 HOWTO (0x19, /* type */
502 0, /* rightshift */
503 2, /* size (0 = byte, 1 = short, 2 = long) */
504 16, /* bitsize */
505 false, /* pc_relative */
506 0, /* bitpos */
507 complain_overflow_bitfield, /* complain_on_overflow */
508 0, /* special_function */
509 "R_RBAC", /* name */
510 true, /* partial_inplace */
511 0xffff, /* src_mask */
512 0xffff, /* dst_mask */
513 false), /* pcrel_offset */
514
515 /* Modifiable branch relative. */
516 HOWTO (0x1a, /* type */
517 0, /* rightshift */
518 2, /* size (0 = byte, 1 = short, 2 = long) */
519 26, /* bitsize */
520 false, /* pc_relative */
521 0, /* bitpos */
522 complain_overflow_signed, /* complain_on_overflow */
523 0, /* special_function */
524 "R_REL", /* name */
525 true, /* partial_inplace */
526 0xffff, /* src_mask */
527 0xffff, /* dst_mask */
528 false), /* pcrel_offset */
529
530 /* Modifiable branch absolute. */
531 HOWTO (0x1b, /* type */
532 0, /* rightshift */
533 2, /* size (0 = byte, 1 = short, 2 = long) */
534 16, /* bitsize */
535 false, /* pc_relative */
536 0, /* bitpos */
537 complain_overflow_bitfield, /* complain_on_overflow */
538 0, /* special_function */
539 "R_REL", /* name */
540 true, /* partial_inplace */
541 0xffff, /* src_mask */
542 0xffff, /* dst_mask */
543 false) /* pcrel_offset */
544};
545
546#define HOWTO_COUNT (sizeof nlm_powerpc_howto_table \
547 / sizeof nlm_powerpc_howto_table[0])
548
549/* Read a PowerPC NLM reloc. */
550
551static boolean
552nlm_powerpc_read_reloc (abfd, sym, secp, rel)
553 bfd *abfd;
554 nlmNAME(symbol_type) *sym;
555 asection **secp;
556 arelent *rel;
557{
558 struct nlm32_powerpc_external_reloc ext;
559 bfd_vma l_vaddr;
560 unsigned long l_symndx;
561 int l_rtype;
562 int l_rsecnm;
563 asection *code_sec, *data_sec, *bss_sec;
564
565 /* Read the reloc from the file. */
566 if (bfd_read (&ext, sizeof ext, 1, abfd) != sizeof ext)
567 return false;
568
569 /* Swap in the fields. */
570 l_vaddr = bfd_h_get_32 (abfd, ext.l_vaddr);
571 l_symndx = bfd_h_get_32 (abfd, ext.l_symndx);
572 l_rtype = bfd_h_get_16 (abfd, ext.l_rtype);
573 l_rsecnm = bfd_h_get_16 (abfd, ext.l_rsecnm);
574
575 /* Get the sections now, for convenience. */
576 code_sec = bfd_get_section_by_name (abfd, NLM_CODE_NAME);
577 data_sec = bfd_get_section_by_name (abfd, NLM_INITIALIZED_DATA_NAME);
578 bss_sec = bfd_get_section_by_name (abfd, NLM_UNINITIALIZED_DATA_NAME);
579
580 /* Work out the arelent fields. */
581 if (sym != NULL)
582 {
583 /* This is an import. sym_ptr_ptr is filled in by
584 nlm_canonicalize_reloc. */
585 rel->sym_ptr_ptr = NULL;
586 }
587 else
588 {
589 asection *sec;
590
591 if (l_symndx == 0)
592 sec = code_sec;
593 else if (l_symndx == 1)
594 sec = data_sec;
595 else if (l_symndx == 2)
596 sec = bss_sec;
597 else
598 {
599 bfd_set_error (bfd_error_bad_value);
600 return false;
601 }
602
603 rel->sym_ptr_ptr = sec->symbol_ptr_ptr;
604 }
605
606 rel->addend = 0;
607
608 BFD_ASSERT ((l_rtype & 0xff) < HOWTO_COUNT);
609
610 rel->howto = nlm_powerpc_howto_table + (l_rtype & 0xff);
611
612 BFD_ASSERT (rel->howto->name != NULL
613 && ((l_rtype & 0x8000) != 0
614 ? (rel->howto->complain_on_overflow
615 == complain_overflow_signed)
616 : (rel->howto->complain_on_overflow
617 == complain_overflow_bitfield))
618 && ((l_rtype >> 8) & 0x1f) == rel->howto->bitsize - 1);
619
620 if (l_rsecnm == 0)
621 *secp = code_sec;
622 else if (l_rsecnm == 1)
623 {
624 *secp = data_sec;
625 l_vaddr -= bfd_section_size (abfd, code_sec);
626 }
627 else
628 {
629 bfd_set_error (bfd_error_bad_value);
630 return false;
631 }
632
633 rel->address = l_vaddr;
634
635 return true;
636}
637
638#endif /* OLDFORMAT */
639
640/* Mangle PowerPC NLM relocs for output. */
641
642static boolean
643nlm_powerpc_mangle_relocs (abfd, sec, data, offset, count)
644 bfd *abfd ATTRIBUTE_UNUSED;
645 asection *sec ATTRIBUTE_UNUSED;
646 PTR data ATTRIBUTE_UNUSED;
647 bfd_vma offset ATTRIBUTE_UNUSED;
648 bfd_size_type count ATTRIBUTE_UNUSED;
649{
650 return true;
651}
652
653/* Read a PowerPC NLM import record */
654
655static boolean
656nlm_powerpc_read_import (abfd, sym)
657 bfd *abfd;
658 nlmNAME(symbol_type) *sym;
659{
660 struct nlm_relent *nlm_relocs; /* relocation records for symbol */
661 bfd_size_type rcount; /* number of relocs */
662 bfd_byte temp[NLM_TARGET_LONG_SIZE]; /* temporary 32-bit value */
663 unsigned char symlength; /* length of symbol name */
664 char *name;
665
666 if (bfd_read ((PTR) &symlength, sizeof (symlength), 1, abfd)
667 != sizeof (symlength))
668 return (false);
669 sym -> symbol.the_bfd = abfd;
670 name = bfd_alloc (abfd, symlength + 1);
671 if (name == NULL)
672 return false;
673 if (bfd_read (name, symlength, 1, abfd) != symlength)
674 return (false);
675 name[symlength] = '\0';
676 sym -> symbol.name = name;
677 sym -> symbol.flags = 0;
678 sym -> symbol.value = 0;
679 sym -> symbol.section = bfd_und_section_ptr;
680 if (bfd_read ((PTR) temp, sizeof (temp), 1, abfd) != sizeof (temp))
681 return (false);
682 rcount = bfd_h_get_32 (abfd, temp);
683 nlm_relocs = ((struct nlm_relent *)
684 bfd_alloc (abfd, rcount * sizeof (struct nlm_relent)));
685 if (nlm_relocs == (struct nlm_relent *) NULL)
686 return false;
687 sym -> relocs = nlm_relocs;
688 sym -> rcnt = 0;
689 while (sym -> rcnt < rcount)
690 {
691 asection *section;
692
693 if (nlm_powerpc_read_reloc (abfd, sym, &section,
694 &nlm_relocs -> reloc)
695 == false)
696 return false;
697 nlm_relocs -> section = section;
698 nlm_relocs++;
699 sym -> rcnt++;
700 }
701 return true;
702}
703
704#ifndef OLDFORMAT
705
706/* Write a PowerPC NLM reloc. */
707
708static boolean
709nlm_powerpc_write_import (abfd, sec, rel)
710 bfd *abfd;
711 asection *sec;
712 arelent *rel;
713{
714 asymbol *sym;
715 bfd_vma val;
716 bfd_byte temp[4];
717
718 /* PowerPC NetWare only supports one kind of reloc. */
719 if (rel->addend != 0
720 || rel->howto == NULL
721 || rel->howto->rightshift != 0
722 || rel->howto->size != 2
723 || rel->howto->bitsize != 32
724 || rel->howto->bitpos != 0
725 || rel->howto->pc_relative
726 || (rel->howto->src_mask != 0xffffffff && rel->addend != 0)
727 || rel->howto->dst_mask != 0xffffffff)
728 {
729 bfd_set_error (bfd_error_invalid_operation);
730 return false;
731 }
732
733 sym = *rel->sym_ptr_ptr;
734
735 /* The value we write out is the offset into the appropriate
736 segment, rightshifted by two. This offset is the section vma,
737 adjusted by the vma of the lowest section in that segment, plus
738 the address of the relocation. */
739 val = bfd_get_section_vma (abfd, sec) + rel->address;
740 if ((val & 3) != 0)
741 {
742 bfd_set_error (bfd_error_bad_value);
743 return false;
744 }
745 val >>= 2;
746
747 /* The high bit is 0 if the reloc is in the data section, or 1 if
748 the reloc is in the code section. */
749 if (bfd_get_section_flags (abfd, sec) & SEC_DATA)
750 val -= nlm_get_data_low (abfd);
751 else
752 {
753 val -= nlm_get_text_low (abfd);
754 val |= NLM_HIBIT;
755 }
756
757 if (! bfd_is_und_section (bfd_get_section (sym)))
758 {
759 /* This is an internal relocation fixup. The second most
760 significant bit is 0 if this is a reloc against the data
761 segment, or 1 if it is a reloc against the text segment. */
762 if (bfd_get_section_flags (abfd, bfd_get_section (sym)) & SEC_CODE)
763 val |= NLM_HIBIT >> 1;
764 }
765
766 bfd_put_32 (abfd, val, temp);
767 if (bfd_write (temp, sizeof (temp), 1, abfd) != sizeof (temp))
768 return false;
769
770 return true;
771}
772
773#else /* OLDFORMAT */
774
775/* This is used for the reloc handling in the old format. */
776
777/* Write a PowerPC NLM reloc. */
778
779static boolean
780nlm_powerpc_write_reloc (abfd, sec, rel, indx)
781 bfd *abfd;
782 asection *sec;
783 arelent *rel;
784 int indx;
785{
786 struct nlm32_powerpc_external_reloc ext;
787 asection *code_sec, *data_sec, *bss_sec;
788 asymbol *sym;
789 asection *symsec;
790 unsigned long l_symndx;
791 int l_rtype;
792 int l_rsecnm;
793 reloc_howto_type *howto;
794 bfd_size_type address;
795
796 /* Get the sections now, for convenience. */
797 code_sec = bfd_get_section_by_name (abfd, NLM_CODE_NAME);
798 data_sec = bfd_get_section_by_name (abfd, NLM_INITIALIZED_DATA_NAME);
799 bss_sec = bfd_get_section_by_name (abfd, NLM_UNINITIALIZED_DATA_NAME);
800
801 sym = *rel->sym_ptr_ptr;
802 symsec = bfd_get_section (sym);
803 if (indx != -1)
804 {
805 BFD_ASSERT (bfd_is_und_section (symsec));
806 l_symndx = indx + 3;
807 }
808 else
809 {
810 if (symsec == code_sec)
811 l_symndx = 0;
812 else if (symsec == data_sec)
813 l_symndx = 1;
814 else if (symsec == bss_sec)
815 l_symndx = 2;
816 else
817 {
818 bfd_set_error (bfd_error_bad_value);
819 return false;
820 }
821 }
822
823 bfd_h_put_32 (abfd, (bfd_vma) l_symndx, ext.l_symndx);
824
825 for (howto = nlm_powerpc_howto_table;
826 howto < nlm_powerpc_howto_table + HOWTO_COUNT;
827 howto++)
828 {
829 if (howto->rightshift == rel->howto->rightshift
830 && howto->size == rel->howto->size
831 && howto->bitsize == rel->howto->bitsize
832 && howto->pc_relative == rel->howto->pc_relative
833 && howto->bitpos == rel->howto->bitpos
834 && (howto->partial_inplace == rel->howto->partial_inplace
835 || (! rel->howto->partial_inplace
836 && rel->addend == 0))
837 && (howto->src_mask == rel->howto->src_mask
838 || (rel->howto->src_mask == 0
839 && rel->addend == 0))
840 && howto->dst_mask == rel->howto->dst_mask
841 && howto->pcrel_offset == rel->howto->pcrel_offset)
842 break;
843 }
844 if (howto >= nlm_powerpc_howto_table + HOWTO_COUNT)
845 {
846 bfd_set_error (bfd_error_bad_value);
847 return false;
848 }
849
850 l_rtype = howto->type;
851 if (howto->complain_on_overflow == complain_overflow_signed)
852 l_rtype |= 0x8000;
853 l_rtype |= (howto->bitsize - 1) << 8;
854 bfd_h_put_16 (abfd, (bfd_vma) l_rtype, ext.l_rtype);
855
856 address = rel->address;
857
858 if (sec == code_sec)
859 l_rsecnm = 0;
860 else if (sec == data_sec)
861 {
862 l_rsecnm = 1;
863 address += bfd_section_size (abfd, code_sec);
864 }
865 else
866 {
867 bfd_set_error (bfd_error_bad_value);
868 return false;
869 }
870
871 bfd_h_put_16 (abfd, (bfd_vma) l_rsecnm, ext.l_rsecnm);
872 bfd_h_put_32 (abfd, (bfd_vma) address, ext.l_vaddr);
873
874 if (bfd_write (&ext, sizeof ext, 1, abfd) != sizeof ext)
875 return false;
876
877 return true;
878}
879
880/* Write a PowerPC NLM import. */
881
882static boolean
883nlm_powerpc_write_import (abfd, sec, rel)
884 bfd *abfd;
885 asection *sec;
886 arelent *rel;
887{
888 return nlm_powerpc_write_reloc (abfd, sec, rel, -1);
889}
890
891#endif /* OLDFORMAT */
892
893/* Write a PowerPC NLM external symbol. This routine keeps a static
894 count of the symbol index. FIXME: I don't know if this is
895 necessary, and the index never gets reset. */
896
897static boolean
898nlm_powerpc_write_external (abfd, count, sym, relocs)
899 bfd *abfd;
900 bfd_size_type count;
901 asymbol *sym;
902 struct reloc_and_sec *relocs;
903{
904 unsigned int i;
905 bfd_byte len;
906 unsigned char temp[NLM_TARGET_LONG_SIZE];
907#ifdef OLDFORMAT
908 static int indx;
909#endif
910
911 len = strlen (sym->name);
912 if ((bfd_write (&len, sizeof (bfd_byte), 1, abfd) != sizeof (bfd_byte))
913 || bfd_write (sym->name, len, 1, abfd) != len)
914 return false;
915
916 bfd_put_32 (abfd, count, temp);
917 if (bfd_write (temp, sizeof (temp), 1, abfd) != sizeof (temp))
918 return false;
919
920 for (i = 0; i < count; i++)
921 {
922#ifndef OLDFORMAT
923 if (! nlm_powerpc_write_import (abfd, relocs[i].sec, relocs[i].rel))
924 return false;
925#else
926 if (! nlm_powerpc_write_reloc (abfd, relocs[i].sec,
927 relocs[i].rel, indx))
928 return false;
929#endif
930 }
931
932#ifdef OLDFORMAT
933 ++indx;
934#endif
935
936 return true;
937}
938
939
940#ifndef OLDFORMAT
941
942/* PowerPC Netware uses a word offset, not a byte offset, for public
943 symbols. */
944
945/* Set the section for a public symbol. */
946
947static boolean
948nlm_powerpc_set_public_section (abfd, sym)
949 bfd *abfd;
950 nlmNAME(symbol_type) *sym;
951{
952 if (sym->symbol.value & NLM_HIBIT)
953 {
954 sym->symbol.value &= ~NLM_HIBIT;
955 sym->symbol.flags |= BSF_FUNCTION;
956 sym->symbol.section =
957 bfd_get_section_by_name (abfd, NLM_CODE_NAME);
958 }
959 else
960 {
961 sym->symbol.section =
962 bfd_get_section_by_name (abfd, NLM_INITIALIZED_DATA_NAME);
963 }
964
965 sym->symbol.value <<= 2;
966
967 return true;
968}
969
970/* Get the offset to write out for a public symbol. */
971
972static bfd_vma
973nlm_powerpc_get_public_offset (abfd, sym)
974 bfd *abfd;
975 asymbol *sym;
976{
977 bfd_vma offset;
978 asection *sec;
979
980 offset = bfd_asymbol_value (sym);
981 sec = bfd_get_section (sym);
982 if (sec->flags & SEC_CODE)
983 {
984 offset -= nlm_get_text_low (abfd);
985 offset |= NLM_HIBIT;
986 }
987 else if (sec->flags & (SEC_DATA | SEC_ALLOC))
988 {
989 /* SEC_ALLOC is for the .bss section. */
990 offset -= nlm_get_data_low (abfd);
991 }
992 else
993 {
994 /* We can't handle an exported symbol that is not in the code or
995 data segment. */
996 bfd_set_error (bfd_error_invalid_operation);
997 /* FIXME: No way to return error. */
998 abort ();
999 }
1000
1001 return offset;
1002}
1003
1004#endif /* ! defined (OLDFORMAT) */
1005
1006
1007#include "nlmswap.h"
1008
1009static const struct nlm_backend_data nlm32_powerpc_backend =
1010{
1011 "NetWare PowerPC Module \032",
1012 sizeof (Nlm32_powerpc_External_Fixed_Header),
1013#ifndef OLDFORMAT
1014 0, /* optional_prefix_size */
1015#else
1016 sizeof (struct nlm32_powerpc_external_prefix_header),
1017#endif
1018 bfd_arch_powerpc,
1019 0,
1020 false,
1021#ifndef OLDFORMAT
1022 0, /* backend_object_p */
1023 0, /* write_prefix */
1024#else
1025 nlm_powerpc_backend_object_p,
1026 nlm_powerpc_write_prefix,
1027#endif
1028 nlm_powerpc_read_reloc,
1029 nlm_powerpc_mangle_relocs,
1030 nlm_powerpc_read_import,
1031 nlm_powerpc_write_import,
1032#ifndef OLDFORMAT
1033 nlm_powerpc_set_public_section,
1034 nlm_powerpc_get_public_offset,
1035#else
1036 0, /* set_public_section */
1037 0, /* get_public_offset */
1038#endif
1039 nlm_swap_fixed_header_in,
1040 nlm_swap_fixed_header_out,
1041 nlm_powerpc_write_external,
1042 0, /* write_export */
1043};
1044
1045#define TARGET_BIG_NAME "nlm32-powerpc"
1046#define TARGET_BIG_SYM nlmNAME(powerpc_vec)
1047#define TARGET_BACKEND_DATA &nlm32_powerpc_backend
1048
1049#include "nlm-target.h"
Note: See TracBrowser for help on using the repository browser.