1 /* outaout.c output routines for the Netwide Assembler to produce
2 * Linux a.out object files
4 * The Netwide Assembler is copyright (C) 1996 Simon Tatham and
5 * Julian Hall. All rights reserved. The software is
6 * redistributable under the licence given in the file "Licence"
7 * distributed in the NASM archive.
23 long address
; /* relative to _start_ of section */
24 long symbol
; /* symbol number or -ve section id */
25 int bytes
; /* 2 or 4 */
26 int relative
; /* TRUE or FALSE */
30 long strpos
; /* string table position of name */
31 int type
; /* symbol type - see flags below */
32 long value
; /* address, or COMMON variable size */
36 * Section IDs - used in Reloc.symbol when negative, and in
37 * Symbol.type when positive.
39 #define SECT_ABS 2 /* absolute value */
40 #define SECT_TEXT 4 /* text section */
41 #define SECT_DATA 6 /* data section */
42 #define SECT_BSS 8 /* bss section */
43 #define SECT_MASK 0xE /* mask out any of the above */
46 * Another flag used in Symbol.type.
48 #define SYM_GLOBAL 1 /* it's a global symbol */
51 * Bit more explanation of symbol types: SECT_xxx denotes a local
52 * symbol. SECT_xxx|SYM_GLOBAL denotes a global symbol, defined in
53 * this module. Just SYM_GLOBAL, with zero value, denotes an
54 * external symbol referenced in this module. And just SYM_GLOBAL,
55 * but with a non-zero value, declares a C `common' variable, of
61 unsigned long len
, size
, nrelocs
;
63 struct Reloc
*head
, **tail
;
66 static struct Section stext
, sdata
;
67 static unsigned long bsslen
;
70 static struct SAA
*syms
;
71 static unsigned long nsyms
;
73 static struct RAA
*bsym
;
75 static struct SAA
*strs
;
76 static unsigned long strslen
;
81 static void aout_write(void);
82 static void aout_write_relocs(struct Reloc
*);
83 static void aout_write_syms(void);
84 static void aout_sect_write(struct Section
*, unsigned char *, unsigned long);
85 static void aout_pad_sections(void);
86 static void aout_fixup_relocs(struct Section
*);
88 static void aout_init(FILE *fp
, efunc errfunc
, ldfunc ldef
) {
91 (void) ldef
; /* placate optimisers */
92 stext
.data
= saa_init(1L); stext
.head
= NULL
; stext
.tail
= &stext
.head
;
93 sdata
.data
= saa_init(1L); sdata
.head
= NULL
; sdata
.tail
= &sdata
.head
;
94 stext
.len
= stext
.size
= sdata
.len
= sdata
.size
= bsslen
= 0;
95 stext
.nrelocs
= sdata
.nrelocs
= 0;
96 stext
.index
= seg_alloc();
97 sdata
.index
= seg_alloc();
98 bssindex
= seg_alloc();
99 syms
= saa_init((long)sizeof(struct Symbol
));
106 static void aout_cleanup(void) {
110 aout_fixup_relocs(&stext
);
111 aout_fixup_relocs(&sdata
);
114 saa_free (stext
.data
);
117 stext
.head
= stext
.head
->next
;
120 saa_free (sdata
.data
);
123 sdata
.head
= sdata
.head
->next
;
131 static long aout_section_names (char *name
, int pass
, int *bits
) {
133 * Default to 32 bits.
141 if (!strcmp(name
, ".text"))
143 else if (!strcmp(name
, ".data"))
145 else if (!strcmp(name
, ".bss"))
151 static void aout_deflabel (char *name
, long segment
, long offset
,
156 if (name
[0] == '.' && name
[1] == '.' && name
[2] != '@') {
157 error (ERR_NONFATAL
, "unrecognised special symbol `%s'", name
);
161 saa_wbytes (strs
, name
, (long)(1+strlen(name
)));
162 strslen
+= 1+strlen(name
);
164 sym
= saa_wstruct (syms
);
167 sym
->type
= is_global
? SYM_GLOBAL
: 0;
168 if (segment
== NO_SEG
)
169 sym
->type
|= SECT_ABS
;
170 else if (segment
== stext
.index
)
171 sym
->type
|= SECT_TEXT
;
172 else if (segment
== sdata
.index
)
173 sym
->type
|= SECT_DATA
;
174 else if (segment
== bssindex
)
175 sym
->type
|= SECT_BSS
;
177 sym
->type
= SYM_GLOBAL
;
181 sym
->value
= (sym
->type
== SYM_GLOBAL
? 0 : offset
);
184 * define the references from external-symbol segment numbers
185 * to these symbol records.
187 if (segment
!= NO_SEG
&& segment
!= stext
.index
&&
188 segment
!= sdata
.index
&& segment
!= bssindex
)
189 bsym
= raa_write (bsym
, segment
, nsyms
);
194 static void aout_add_reloc (struct Section
*sect
, long segment
,
195 int relative
, int bytes
) {
198 r
= *sect
->tail
= nasm_malloc(sizeof(struct Reloc
));
199 sect
->tail
= &r
->next
;
202 r
->address
= sect
->len
;
203 r
->symbol
= (segment
== NO_SEG
? -SECT_ABS
:
204 segment
== stext
.index
? -SECT_TEXT
:
205 segment
== sdata
.index
? -SECT_DATA
:
206 segment
== bssindex
? -SECT_BSS
:
207 raa_read(bsym
, segment
));
208 r
->relative
= relative
;
214 static void aout_out (long segto
, void *data
, unsigned long type
,
215 long segment
, long wrt
) {
217 long realbytes
= type
& OUT_SIZMASK
;
218 unsigned char mydata
[4], *p
;
221 wrt
= NO_SEG
; /* continue to do _something_ */
222 error (ERR_NONFATAL
, "WRT not supported by a.out output format");
228 * handle absolute-assembly (structure definitions)
230 if (segto
== NO_SEG
) {
231 if (type
!= OUT_RESERVE
)
232 error (ERR_NONFATAL
, "attempt to assemble code in [ABSOLUTE]"
237 if (segto
== stext
.index
)
239 else if (segto
== sdata
.index
)
241 else if (segto
== bssindex
)
244 error(ERR_WARNING
, "attempt to assemble code in"
245 " segment %d: defaulting to `.text'", segto
);
249 if (!s
&& type
!= OUT_RESERVE
) {
250 error(ERR_WARNING
, "attempt to initialise memory in the"
251 " BSS section: ignored");
252 if (type
== OUT_REL2ADR
)
254 else if (type
== OUT_REL4ADR
)
260 if (type
== OUT_RESERVE
) {
262 error(ERR_WARNING
, "uninitialised space declared in"
263 " %s section: zeroing",
264 (segto
== stext
.index
? "code" : "data"));
265 aout_sect_write (s
, NULL
, realbytes
);
268 } else if (type
== OUT_RAWDATA
) {
269 if (segment
!= NO_SEG
)
270 error(ERR_PANIC
, "OUT_RAWDATA with other than NO_SEG");
271 aout_sect_write (s
, data
, realbytes
);
272 } else if (type
== OUT_ADDRESS
) {
273 if (segment
!= NO_SEG
) {
275 error(ERR_NONFATAL
, "a.out format does not support"
276 " segment base references");
278 aout_add_reloc (s
, segment
, FALSE
, realbytes
);
282 WRITESHORT (p
, *(long *)data
);
284 WRITELONG (p
, *(long *)data
);
285 aout_sect_write (s
, mydata
, realbytes
);
286 } else if (type
== OUT_REL2ADR
) {
287 if (segment
== segto
)
288 error(ERR_PANIC
, "intra-segment OUT_REL2ADR");
289 if (segment
!= NO_SEG
&& segment
% 2) {
290 error(ERR_NONFATAL
, "a.out format does not support"
291 " segment base references");
293 aout_add_reloc (s
, segment
, TRUE
, 2);
295 WRITESHORT (p
, *(long*)data
-(realbytes
+ s
->len
));
296 aout_sect_write (s
, mydata
, 2L);
297 } else if (type
== OUT_REL4ADR
) {
298 if (segment
== segto
)
299 error(ERR_PANIC
, "intra-segment OUT_REL4ADR");
300 if (segment
!= NO_SEG
&& segment
% 2) {
301 error(ERR_NONFATAL
, "a.out format does not support"
302 " segment base references");
304 aout_add_reloc (s
, segment
, TRUE
, 4);
306 WRITELONG (p
, *(long*)data
-(realbytes
+ s
->len
));
307 aout_sect_write (s
, mydata
, 4L);
311 static void aout_pad_sections(void) {
312 static unsigned char pad
[] = { 0x90, 0x90, 0x90, 0x90 };
314 * Pad each of the text and data sections with NOPs until their
315 * length is a multiple of four. (NOP == 0x90.) Also increase
316 * the length of the BSS section similarly.
318 aout_sect_write (&stext
, pad
, (-stext
.len
) & 3);
319 aout_sect_write (&sdata
, pad
, (-sdata
.len
) & 3);
320 bsslen
= (bsslen
+ 3) & ~3;
324 * a.out files have the curious property that all references to
325 * things in the data or bss sections are done by addresses which
326 * are actually relative to the start of the _text_ section, in the
327 * _file_. (No relation to what happens after linking. No idea why
328 * this should be so. It's very strange.) So we have to go through
329 * the relocation table, _after_ the final size of each section is
330 * known, and fix up the relocations pointed to.
332 static void aout_fixup_relocs(struct Section
*sect
) {
335 saa_rewind (sect
->data
);
336 for (r
= sect
->head
; r
; r
= r
->next
) {
337 unsigned char *p
, *q
, blk
[4];
340 saa_fread (sect
->data
, r
->address
, blk
, (long)r
->bytes
);
343 l
+= ((long)*p
++) << 8;
345 l
+= ((long)*p
++) << 16;
346 l
+= ((long)*p
++) << 24;
348 if (r
->symbol
== -SECT_DATA
)
350 else if (r
->symbol
== -SECT_BSS
)
351 l
+= stext
.len
+ sdata
.len
;
356 saa_fwrite (sect
->data
, r
->address
, blk
, (long)r
->bytes
);
360 static void aout_write(void) {
362 * Emit the a.out header.
364 fwritelong (0x640107L
, aoutfp
); /* OMAGIC, M_386, no flags */
365 fwritelong (stext
.len
, aoutfp
);
366 fwritelong (sdata
.len
, aoutfp
);
367 fwritelong (bsslen
, aoutfp
);
368 fwritelong (nsyms
* 12, aoutfp
); /* length of symbol table */
369 fwritelong (0L, aoutfp
); /* object files have no entry point */
370 fwritelong (stext
.nrelocs
* 8, aoutfp
); /* size of text relocs */
371 fwritelong (sdata
.nrelocs
* 8, aoutfp
); /* size of data relocs */
374 * Write out the code section and the data section.
376 saa_fpwrite (stext
.data
, aoutfp
);
377 saa_fpwrite (sdata
.data
, aoutfp
);
380 * Write out the relocations.
382 aout_write_relocs (stext
.head
);
383 aout_write_relocs (sdata
.head
);
386 * Write the symbol table.
391 * And the string table.
393 fwritelong (strslen
+4, aoutfp
); /* length includes length count */
394 saa_fpwrite (strs
, aoutfp
);
397 static void aout_write_relocs (struct Reloc
*r
) {
401 fwritelong (r
->address
, aoutfp
);
404 word2
= r
->symbol
| 0x8000000L
;
409 word2
|= (r
->bytes
== 2 ? 0x2000000L
: 0x4000000L
);
410 fwritelong (word2
, aoutfp
);
416 static void aout_write_syms (void) {
420 for (i
=0; i
<nsyms
; i
++) {
421 struct Symbol
*sym
= saa_rstruct(syms
);
422 fwritelong (sym
->strpos
, aoutfp
);
423 fwritelong ((long)sym
->type
, aoutfp
);
425 * Fix up the symbol value now we know the final section
428 if ((sym
->type
& SECT_MASK
) == SECT_DATA
)
429 sym
->value
+= stext
.len
;
430 if ((sym
->type
& SECT_MASK
) == SECT_BSS
)
431 sym
->value
+= stext
.len
+ sdata
.len
;
432 fwritelong (sym
->value
, aoutfp
);
436 static void aout_sect_write (struct Section
*sect
,
437 unsigned char *data
, unsigned long len
) {
438 saa_wbytes (sect
->data
, data
, len
);
442 static long aout_segbase (long segment
) {
446 static int aout_directive (char *directive
, char *value
, int pass
) {
450 static void aout_filename (char *inname
, char *outname
, efunc error
) {
451 standard_extension (inname
, outname
, ".o", error
);
454 struct ofmt of_aout
= {
455 "GNU a.out (i386) object files (e.g. Linux)",