Commit | Line | Data |
---|---|---|
e98e6ec1 SC |
1 | /* This module is part of BFD */ |
2 | ||
3 | ||
4 | /* The intention is that one day, all the code which uses sections | |
5 | will change and use seclets instead - maybe seglet would have been | |
6 | a better name.. | |
7 | ||
8 | Anyway, a seclet contains enough info to be able to describe an | |
9 | area of output memory in one go. | |
10 | ||
11 | The only description so far catered for is that of the | |
12 | <<bfd_indirect_seclet>>, which is a select which points to a | |
13 | <<section>> and the <<asymbols>> associated with the section, so | |
14 | that relocation can be done when needed. | |
15 | ||
16 | One day there will be more types - they will at least migrate from | |
17 | the linker's data structures - also there could be extra stuff, | |
18 | like a bss seclet, which descibes a lump of memory as containing | |
19 | zeros compactly, without the horrible SEC_* flag cruft. | |
20 | ||
21 | ||
22 | */ | |
23 | ||
24 | #include "bfd.h" | |
25 | #include "sysdep.h" | |
26 | #include "libbfd.h" | |
27 | #include "seclet.h" | |
28 | #include "coff/internal.h" | |
29 | bfd_seclet_type * | |
30 | DEFUN(bfd_new_seclet,(abfd, section), | |
31 | bfd *abfd AND | |
32 | asection *section) | |
33 | { | |
34 | bfd_seclet_type *n = (bfd_seclet_type *)bfd_alloc(abfd, sizeof(bfd_seclet_type)); | |
35 | if (section->seclets_tail != (bfd_seclet_type *)NULL) { | |
36 | section->seclets_tail->next = n; | |
37 | } | |
38 | else | |
39 | { | |
40 | section->seclets_head = n; | |
41 | } | |
42 | section->seclets_tail = n; | |
43 | ||
44 | return n; | |
e98e6ec1 SC |
45 | } |
46 | ||
47 | ||
48 | ||
49 | ||
50 | #define MAX_ERRORS_IN_A_ROW 10 | |
51 | extern bfd_error_vector_type bfd_error_vector; | |
e98e6ec1 | 52 | |
e98e6ec1 SC |
53 | |
54 | void | |
55 | DEFUN(rel,(abfd, seclet, output_section), | |
56 | bfd *abfd AND | |
57 | bfd_seclet_type *seclet AND | |
58 | asection *output_section) | |
59 | { | |
2cfd0562 | 60 | |
ab98fd5d SC |
61 | if (output_section->flags & SEC_HAS_CONTENTS |
62 | && !(output_section->flags & SEC_NEVER_LOAD)) | |
e98e6ec1 | 63 | { |
ae6e60b5 | 64 | bfd_byte *data = (bfd_byte *)alloca(seclet->size); |
2cfd0562 | 65 | data = bfd_get_relocated_section_contents(abfd, seclet, data); |
e5932011 SC |
66 | if(bfd_set_section_contents(abfd, |
67 | output_section, | |
68 | data, | |
69 | seclet->offset, | |
70 | seclet->size) == false) | |
71 | { | |
72 | abort(); | |
73 | } | |
2cfd0562 | 74 | |
e98e6ec1 | 75 | } |
e98e6ec1 SC |
76 | } |
77 | ||
78 | void | |
79 | DEFUN(seclet_dump_seclet,(abfd, seclet, section), | |
80 | bfd *abfd AND | |
81 | bfd_seclet_type *seclet AND | |
82 | asection *section) | |
83 | { | |
84 | switch (seclet->type) | |
85 | { | |
e5932011 | 86 | case bfd_indirect_seclet: |
e98e6ec1 SC |
87 | /* The contents of this section come from another one somewhere |
88 | else */ | |
89 | rel(abfd, seclet, section); | |
e98e6ec1 | 90 | break; |
e5932011 SC |
91 | case bfd_fill_seclet: |
92 | /* Fill in the section with us */ | |
93 | { | |
94 | char *d = malloc(seclet->size); | |
95 | unsigned int i; | |
96 | for (i =0; i < seclet->size; i+=2) { | |
97 | d[i] = seclet->u.fill.value >> 8; | |
98 | } | |
99 | for (i = 1; i < seclet->size; i+=2) { | |
100 | d[i] = seclet->u.fill.value ; | |
101 | } | |
102 | bfd_set_section_contents(abfd, section, d, seclet->offset, seclet->size); | |
103 | ||
104 | } | |
105 | break; | |
106 | default: | |
e98e6ec1 SC |
107 | abort(); |
108 | } | |
e98e6ec1 SC |
109 | } |
110 | ||
111 | void | |
112 | DEFUN(seclet_dump,(abfd), | |
113 | bfd *abfd) | |
114 | { | |
115 | /* Write all the seclets on the bfd out, relocate etc according to the | |
116 | rules */ | |
117 | ||
118 | asection *o = abfd->sections; | |
119 | while (o != (asection *)NULL) | |
120 | { | |
121 | bfd_seclet_type *p = o->seclets_head; | |
122 | while (p != (bfd_seclet_type *)NULL) | |
123 | { | |
124 | seclet_dump_seclet(abfd, p, o); | |
125 | p = p ->next; | |
126 | } | |
e98e6ec1 SC |
127 | o = o->next; |
128 | } | |
e98e6ec1 | 129 | } |