1993-05-28 03:42:23 +08:00
|
|
|
/* ELF object file format
|
|
|
|
Copyright (C) 1992, 1993 Free Software Foundation, Inc.
|
|
|
|
|
|
|
|
This file is part of GAS, the GNU Assembler.
|
|
|
|
|
|
|
|
GAS is free software; you can redistribute it and/or modify
|
|
|
|
it under the terms of the GNU General Public License as
|
|
|
|
published by the Free Software Foundation; either version 2,
|
|
|
|
or (at your option) any later version.
|
|
|
|
|
|
|
|
GAS is distributed in the hope that it will be useful, but
|
|
|
|
WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See
|
|
|
|
the GNU General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public
|
|
|
|
License along with GAS; see the file COPYING. If not, write
|
|
|
|
to the Free Software Foundation, 675 Mass Ave, Cambridge, MA 02139, USA. */
|
|
|
|
|
|
|
|
#include "as.h"
|
1993-07-08 00:40:30 +08:00
|
|
|
#include "subsegs.h"
|
1993-05-28 03:42:23 +08:00
|
|
|
#include "obstack.h"
|
|
|
|
|
1993-11-24 15:42:03 +08:00
|
|
|
#ifdef ECOFF_DEBUGGING
|
|
|
|
#include "ecoff.h"
|
|
|
|
#endif
|
|
|
|
|
1994-01-16 01:31:05 +08:00
|
|
|
#ifdef TC_MIPS
|
|
|
|
#include "elf/mips.h"
|
|
|
|
#endif
|
|
|
|
|
1993-11-24 15:42:03 +08:00
|
|
|
#ifdef ECOFF_DEBUGGING
|
|
|
|
static boolean elf_get_extr PARAMS ((asymbol *, EXTR *));
|
|
|
|
static void elf_set_index PARAMS ((asymbol *, bfd_size_type));
|
|
|
|
#endif
|
|
|
|
|
1993-09-14 05:32:07 +08:00
|
|
|
static void obj_elf_line PARAMS ((int));
|
|
|
|
void obj_elf_version PARAMS ((int));
|
|
|
|
static void obj_elf_size PARAMS ((int));
|
|
|
|
static void obj_elf_type PARAMS ((int));
|
|
|
|
static void obj_elf_ident PARAMS ((int));
|
|
|
|
static void obj_elf_weak PARAMS ((int));
|
|
|
|
static void obj_elf_local PARAMS ((int));
|
|
|
|
static void obj_elf_common PARAMS ((int));
|
1993-09-17 02:26:36 +08:00
|
|
|
static void obj_elf_data PARAMS ((int));
|
|
|
|
static void obj_elf_text PARAMS ((int));
|
1993-05-28 03:42:23 +08:00
|
|
|
|
1993-07-08 00:40:30 +08:00
|
|
|
const pseudo_typeS obj_pseudo_table[] =
|
|
|
|
{
|
1993-08-20 13:29:24 +08:00
|
|
|
{"comm", obj_elf_common, 0},
|
1993-07-08 00:40:30 +08:00
|
|
|
{"ident", obj_elf_ident, 0},
|
1993-08-20 13:29:24 +08:00
|
|
|
{"local", obj_elf_local, 0},
|
1993-07-20 03:49:34 +08:00
|
|
|
{"previous", obj_elf_previous, 0},
|
1993-07-08 00:40:30 +08:00
|
|
|
{"section", obj_elf_section, 0},
|
|
|
|
{"size", obj_elf_size, 0},
|
|
|
|
{"type", obj_elf_type, 0},
|
|
|
|
{"version", obj_elf_version, 0},
|
1993-08-19 05:09:23 +08:00
|
|
|
{"weak", obj_elf_weak, 0},
|
1993-07-08 00:40:30 +08:00
|
|
|
|
|
|
|
/* These are used for stabs-in-elf configurations. */
|
|
|
|
{"line", obj_elf_line, 0},
|
|
|
|
|
1993-07-20 03:49:34 +08:00
|
|
|
/* These are used for dwarf. */
|
|
|
|
{"2byte", cons, 2},
|
|
|
|
{"4byte", cons, 4},
|
1993-08-19 05:09:23 +08:00
|
|
|
{"8byte", cons, 8},
|
1993-07-20 03:49:34 +08:00
|
|
|
|
1993-09-17 02:26:36 +08:00
|
|
|
/* We need to trap the section changing calls to handle .previous. */
|
|
|
|
{"data", obj_elf_data, 0},
|
|
|
|
{"text", obj_elf_text, 0},
|
|
|
|
|
1993-11-24 15:42:03 +08:00
|
|
|
#ifdef ECOFF_DEBUGGING
|
|
|
|
/* COFF style debugging information for ECOFF. .ln is not used; .loc
|
|
|
|
is used instead. */
|
|
|
|
{ "def", ecoff_directive_def, 0 },
|
|
|
|
{ "dim", ecoff_directive_dim, 0 },
|
|
|
|
{ "endef", ecoff_directive_endef, 0 },
|
|
|
|
{ "file", ecoff_directive_file, 0 },
|
|
|
|
{ "scl", ecoff_directive_scl, 0 },
|
|
|
|
{ "tag", ecoff_directive_tag, 0 },
|
|
|
|
{ "val", ecoff_directive_val, 0 },
|
|
|
|
|
|
|
|
/* COFF debugging requires pseudo-ops .size and .type, but ELF
|
|
|
|
already has meanings for those. We use .esize and .etype
|
|
|
|
instead. These are only generated by gcc anyhow. */
|
|
|
|
{ "esize", ecoff_directive_size, 0 },
|
|
|
|
{ "etype", ecoff_directive_type, 0 },
|
|
|
|
|
|
|
|
/* ECOFF specific debugging information. */
|
|
|
|
{ "begin", ecoff_directive_begin, 0 },
|
|
|
|
{ "bend", ecoff_directive_bend, 0 },
|
|
|
|
{ "end", ecoff_directive_end, 0 },
|
|
|
|
{ "ent", ecoff_directive_ent, 0 },
|
|
|
|
{ "fmask", ecoff_directive_fmask, 0 },
|
|
|
|
{ "frame", ecoff_directive_frame, 0 },
|
|
|
|
{ "loc", ecoff_directive_loc, 0 },
|
|
|
|
{ "mask", ecoff_directive_mask, 0 },
|
|
|
|
|
|
|
|
/* These are used on Irix. I don't know how to implement them. */
|
|
|
|
{ "alias", s_ignore, 0 },
|
|
|
|
{ "bgnb", s_ignore, 0 },
|
|
|
|
{ "endb", s_ignore, 0 },
|
|
|
|
{ "lab", s_ignore, 0 },
|
|
|
|
{ "noalias", s_ignore, 0 },
|
|
|
|
{ "verstamp", s_ignore, 0 },
|
|
|
|
{ "vreg", s_ignore, 0 },
|
|
|
|
#endif /* ECOFF_DEBUGGING */
|
|
|
|
|
1993-07-08 00:40:30 +08:00
|
|
|
{NULL} /* end sentinel */
|
1993-05-28 03:42:23 +08:00
|
|
|
};
|
|
|
|
|
1993-08-19 05:09:23 +08:00
|
|
|
#undef NO_RELOC
|
1993-07-08 00:40:30 +08:00
|
|
|
#include "aout/aout64.h"
|
|
|
|
|
|
|
|
void
|
|
|
|
elf_file_symbol (s)
|
|
|
|
char *s;
|
|
|
|
{
|
|
|
|
symbolS *sym;
|
|
|
|
|
|
|
|
sym = symbol_new (s, absolute_section, (valueT) 0, (struct frag *) 0);
|
|
|
|
sym->sy_frag = &zero_address_frag;
|
|
|
|
sym->bsym->flags |= BSF_FILE;
|
|
|
|
|
|
|
|
if (symbol_rootP != sym)
|
|
|
|
{
|
|
|
|
symbol_remove (sym, &symbol_rootP, &symbol_lastP);
|
|
|
|
symbol_insert (sym, symbol_rootP, &symbol_rootP, &symbol_lastP);
|
|
|
|
#ifdef DEBUG
|
|
|
|
verify_symbol_chain (symbol_rootP, symbol_lastP);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
1993-08-20 13:29:24 +08:00
|
|
|
static void
|
1993-09-14 05:32:07 +08:00
|
|
|
obj_elf_common (ignore)
|
|
|
|
int ignore;
|
1993-08-20 13:29:24 +08:00
|
|
|
{
|
|
|
|
char *name;
|
|
|
|
char c;
|
|
|
|
char *p;
|
|
|
|
int temp, size;
|
|
|
|
symbolS *symbolP;
|
1993-10-27 05:01:15 +08:00
|
|
|
int have_align;
|
1993-08-20 13:29:24 +08:00
|
|
|
|
|
|
|
name = input_line_pointer;
|
|
|
|
c = get_symbol_end ();
|
|
|
|
/* just after name is now '\0' */
|
|
|
|
p = input_line_pointer;
|
|
|
|
*p = c;
|
|
|
|
SKIP_WHITESPACE ();
|
|
|
|
if (*input_line_pointer != ',')
|
|
|
|
{
|
|
|
|
as_bad ("Expected comma after symbol-name");
|
|
|
|
ignore_rest_of_line ();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
input_line_pointer++; /* skip ',' */
|
|
|
|
if ((temp = get_absolute_expression ()) < 0)
|
|
|
|
{
|
|
|
|
as_bad (".COMMon length (%d.) <0! Ignored.", temp);
|
|
|
|
ignore_rest_of_line ();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
size = temp;
|
|
|
|
*p = 0;
|
|
|
|
symbolP = symbol_find_or_make (name);
|
|
|
|
*p = c;
|
|
|
|
if (S_IS_DEFINED (symbolP))
|
|
|
|
{
|
|
|
|
as_bad ("Ignoring attempt to re-define symbol");
|
|
|
|
ignore_rest_of_line ();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (S_GET_VALUE (symbolP) != 0)
|
|
|
|
{
|
|
|
|
if (S_GET_VALUE (symbolP) != size)
|
|
|
|
{
|
|
|
|
as_warn ("Length of .comm \"%s\" is already %ld. Not changed to %d.",
|
|
|
|
S_GET_NAME (symbolP), (long) S_GET_VALUE (symbolP), size);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
know (symbolP->sy_frag == &zero_address_frag);
|
|
|
|
if (*input_line_pointer != ',')
|
1993-10-27 05:01:15 +08:00
|
|
|
have_align = 0;
|
|
|
|
else
|
1993-08-20 13:29:24 +08:00
|
|
|
{
|
1993-10-27 05:01:15 +08:00
|
|
|
have_align = 1;
|
|
|
|
input_line_pointer++;
|
|
|
|
SKIP_WHITESPACE ();
|
1993-08-20 13:29:24 +08:00
|
|
|
}
|
1993-10-27 05:01:15 +08:00
|
|
|
if (! have_align || *input_line_pointer != '"')
|
1993-08-20 13:29:24 +08:00
|
|
|
{
|
1993-10-27 05:01:15 +08:00
|
|
|
if (! have_align)
|
|
|
|
temp = 0;
|
|
|
|
else
|
1993-08-20 13:29:24 +08:00
|
|
|
{
|
1993-10-27 05:01:15 +08:00
|
|
|
temp = get_absolute_expression ();
|
|
|
|
if (temp < 0)
|
|
|
|
{
|
|
|
|
temp = 0;
|
|
|
|
as_warn ("Common alignment negative; 0 assumed");
|
|
|
|
}
|
1993-08-20 13:29:24 +08:00
|
|
|
}
|
|
|
|
if (symbolP->local)
|
|
|
|
{
|
|
|
|
segT old_sec;
|
|
|
|
int old_subsec;
|
1993-09-14 05:32:07 +08:00
|
|
|
char *pfrag;
|
1993-08-20 13:29:24 +08:00
|
|
|
int align;
|
|
|
|
|
1993-09-11 00:01:07 +08:00
|
|
|
/* allocate_bss: */
|
1993-08-20 13:29:24 +08:00
|
|
|
old_sec = now_seg;
|
|
|
|
old_subsec = now_subseg;
|
|
|
|
align = temp;
|
|
|
|
record_alignment (bss_section, align);
|
|
|
|
subseg_set (bss_section, 0);
|
|
|
|
if (align)
|
|
|
|
frag_align (align, 0);
|
|
|
|
if (S_GET_SEGMENT (symbolP) == bss_section)
|
|
|
|
symbolP->sy_frag->fr_symbol = 0;
|
|
|
|
symbolP->sy_frag = frag_now;
|
1993-09-14 05:32:07 +08:00
|
|
|
pfrag = frag_var (rs_org, 1, 1, (relax_substateT) 0, symbolP, size,
|
|
|
|
(char *) 0);
|
|
|
|
*pfrag = 0;
|
1993-08-20 13:29:24 +08:00
|
|
|
S_SET_SEGMENT (symbolP, bss_section);
|
|
|
|
S_CLEAR_EXTERNAL (symbolP);
|
|
|
|
subseg_set (old_sec, old_subsec);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
allocate_common:
|
1993-09-14 05:32:07 +08:00
|
|
|
S_SET_VALUE (symbolP, (valueT) size);
|
1993-08-20 13:29:24 +08:00
|
|
|
S_SET_EXTERNAL (symbolP);
|
|
|
|
/* should be common, but this is how gas does it for now */
|
|
|
|
S_SET_SEGMENT (symbolP, &bfd_und_section);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
input_line_pointer++;
|
|
|
|
/* @@ Some use the dot, some don't. Can we get some consistency?? */
|
|
|
|
if (*input_line_pointer == '.')
|
|
|
|
input_line_pointer++;
|
|
|
|
/* @@ Some say data, some say bss. */
|
|
|
|
if (strncmp (input_line_pointer, "bss\"", 4)
|
|
|
|
&& strncmp (input_line_pointer, "data\"", 5))
|
|
|
|
{
|
|
|
|
while (*--input_line_pointer != '"')
|
|
|
|
;
|
|
|
|
input_line_pointer--;
|
|
|
|
goto bad_common_segment;
|
|
|
|
}
|
|
|
|
while (*input_line_pointer++ != '"')
|
|
|
|
;
|
|
|
|
goto allocate_common;
|
|
|
|
}
|
|
|
|
demand_empty_rest_of_line ();
|
|
|
|
return;
|
|
|
|
|
|
|
|
{
|
|
|
|
bad_common_segment:
|
|
|
|
p = input_line_pointer;
|
|
|
|
while (*p && *p != '\n')
|
|
|
|
p++;
|
|
|
|
c = *p;
|
|
|
|
*p = '\0';
|
|
|
|
as_bad ("bad .common segment %s", input_line_pointer + 1);
|
|
|
|
*p = c;
|
|
|
|
input_line_pointer = p;
|
|
|
|
ignore_rest_of_line ();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
1993-09-14 05:32:07 +08:00
|
|
|
obj_elf_local (ignore)
|
|
|
|
int ignore;
|
1993-08-20 13:29:24 +08:00
|
|
|
{
|
|
|
|
char *name;
|
|
|
|
int c;
|
|
|
|
symbolS *symbolP;
|
|
|
|
|
|
|
|
do
|
|
|
|
{
|
|
|
|
name = input_line_pointer;
|
|
|
|
c = get_symbol_end ();
|
|
|
|
symbolP = symbol_find_or_make (name);
|
|
|
|
*input_line_pointer = c;
|
|
|
|
SKIP_WHITESPACE ();
|
|
|
|
S_CLEAR_EXTERNAL (symbolP);
|
|
|
|
symbolP->local = 1;
|
|
|
|
if (c == ',')
|
|
|
|
{
|
|
|
|
input_line_pointer++;
|
|
|
|
SKIP_WHITESPACE ();
|
|
|
|
if (*input_line_pointer == '\n')
|
|
|
|
c = '\n';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
while (c == ',');
|
|
|
|
demand_empty_rest_of_line ();
|
|
|
|
}
|
|
|
|
|
1993-08-19 05:09:23 +08:00
|
|
|
static void
|
1993-09-14 05:32:07 +08:00
|
|
|
obj_elf_weak (ignore)
|
|
|
|
int ignore;
|
1993-08-19 05:09:23 +08:00
|
|
|
{
|
|
|
|
char *name;
|
|
|
|
int c;
|
|
|
|
symbolS *symbolP;
|
|
|
|
|
|
|
|
do
|
|
|
|
{
|
|
|
|
name = input_line_pointer;
|
|
|
|
c = get_symbol_end ();
|
|
|
|
symbolP = symbol_find_or_make (name);
|
|
|
|
*input_line_pointer = c;
|
|
|
|
SKIP_WHITESPACE ();
|
|
|
|
S_SET_WEAK (symbolP);
|
|
|
|
symbolP->local = 1;
|
|
|
|
if (c == ',')
|
|
|
|
{
|
|
|
|
input_line_pointer++;
|
|
|
|
SKIP_WHITESPACE ();
|
|
|
|
if (*input_line_pointer == '\n')
|
|
|
|
c = '\n';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
while (c == ',');
|
|
|
|
demand_empty_rest_of_line ();
|
|
|
|
}
|
|
|
|
|
1993-07-20 03:49:34 +08:00
|
|
|
static segT previous_section;
|
|
|
|
static int previous_subsection;
|
|
|
|
|
1993-10-27 05:01:15 +08:00
|
|
|
/* Handle the .section pseudo-op. This code supports two different
|
|
|
|
syntaxes.
|
|
|
|
|
|
|
|
The first is found on Solaris, and looks like
|
|
|
|
.section ".sec1",#alloc,#execinstr,#write
|
|
|
|
Here the names after '#' are the SHF_* flags to turn on for the
|
|
|
|
section. I'm not sure how it determines the SHT_* type (BFD
|
|
|
|
doesn't really give us control over the type, anyhow).
|
|
|
|
|
|
|
|
The second format is found on UnixWare, and probably most SVR4
|
|
|
|
machines, and looks like
|
|
|
|
.section .sec1,"a",@progbits
|
|
|
|
The quoted string may contain any combination of a, w, x, and
|
|
|
|
represents the SHF_* flags to turn on for the section. The string
|
|
|
|
beginning with '@' can be progbits or nobits. There should be
|
|
|
|
other possibilities, but I don't know what they are. In any case,
|
|
|
|
BFD doesn't really let us set the section type. */
|
|
|
|
|
1994-01-16 01:31:05 +08:00
|
|
|
/* Certain named sections have particular defined types, listed on p.
|
|
|
|
4-19 of the ABI. */
|
|
|
|
struct special_section
|
|
|
|
{
|
|
|
|
const char *name;
|
|
|
|
int type;
|
|
|
|
int attributes;
|
|
|
|
};
|
|
|
|
|
|
|
|
static struct special_section special_sections[] =
|
|
|
|
{
|
|
|
|
{ ".bss", SHT_NOBITS, SHF_ALLOC + SHF_WRITE },
|
|
|
|
{ ".comment", SHT_PROGBITS, 0 },
|
|
|
|
{ ".data", SHT_PROGBITS, SHF_ALLOC + SHF_WRITE },
|
|
|
|
{ ".data1", SHT_PROGBITS, SHF_ALLOC + SHF_WRITE },
|
|
|
|
{ ".debug", SHT_PROGBITS, 0 },
|
|
|
|
{ ".fini", SHT_PROGBITS, SHF_ALLOC + SHF_EXECINSTR },
|
|
|
|
{ ".init", SHT_PROGBITS, SHF_ALLOC + SHF_EXECINSTR },
|
|
|
|
{ ".line", SHT_PROGBITS, 0 },
|
|
|
|
{ ".note", SHT_NOTE, 0 },
|
|
|
|
{ ".rodata", SHT_PROGBITS, SHF_ALLOC },
|
|
|
|
{ ".rodata1", SHT_PROGBITS, SHF_ALLOC },
|
|
|
|
{ ".text", SHT_PROGBITS, SHF_ALLOC + SHF_EXECINSTR },
|
|
|
|
|
|
|
|
#ifdef ELF_TC_SPECIAL_SECTIONS
|
|
|
|
ELF_TC_SPECIAL_SECTIONS
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if 0
|
|
|
|
/* The following section names are special, but they can not
|
|
|
|
reasonably appear in assembler code. Some of the attributes are
|
|
|
|
processor dependent. */
|
|
|
|
{ ".dynamic", SHT_DYNAMIC, SHF_ALLOC /* + SHF_WRITE */ },
|
|
|
|
{ ".dynstr", SHT_STRTAB, SHF_ALLOC },
|
|
|
|
{ ".dynsym", SHT_DYNSYM, SHF_ALLOC },
|
|
|
|
{ ".got", SHT_PROGBITS, 0 },
|
|
|
|
{ ".hash", SHT_HASH, SHF_ALLOC },
|
|
|
|
{ ".interp", SHT_PROGBITS, /* SHF_ALLOC */ },
|
|
|
|
{ ".plt", SHT_PROGBITS, 0 },
|
|
|
|
{ ".shstrtab",SHT_STRTAB, 0 },
|
|
|
|
{ ".strtab", SHT_STRTAB, /* SHF_ALLOC */ },
|
|
|
|
{ ".symtab", SHT_SYMTAB, /* SHF_ALLOC */ },
|
|
|
|
#endif
|
|
|
|
|
|
|
|
{ NULL, 0, 0 }
|
|
|
|
};
|
|
|
|
|
1993-08-05 07:10:43 +08:00
|
|
|
void
|
1993-05-28 03:42:23 +08:00
|
|
|
obj_elf_section (xxx)
|
|
|
|
int xxx;
|
|
|
|
{
|
|
|
|
char *string;
|
1993-09-17 02:26:36 +08:00
|
|
|
int new_sec;
|
1993-10-27 05:01:15 +08:00
|
|
|
segT sec;
|
1994-01-16 01:31:05 +08:00
|
|
|
int type, attr;
|
|
|
|
int i;
|
1993-10-27 05:01:15 +08:00
|
|
|
flagword flags;
|
1993-05-28 03:42:23 +08:00
|
|
|
|
1993-08-19 05:09:23 +08:00
|
|
|
/* Get name of section. */
|
1993-10-27 05:01:15 +08:00
|
|
|
SKIP_WHITESPACE ();
|
1993-07-20 03:49:34 +08:00
|
|
|
if (*input_line_pointer == '"')
|
1993-10-27 05:01:15 +08:00
|
|
|
{
|
|
|
|
string = demand_copy_C_string (&xxx);
|
|
|
|
if (string == NULL)
|
|
|
|
{
|
|
|
|
ignore_rest_of_line ();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
1993-07-20 03:49:34 +08:00
|
|
|
else
|
|
|
|
{
|
|
|
|
char *p = input_line_pointer;
|
|
|
|
char c;
|
|
|
|
while (0 == strchr ("\n\t,; ", *p))
|
|
|
|
p++;
|
1993-10-27 05:01:15 +08:00
|
|
|
if (p == input_line_pointer)
|
|
|
|
{
|
|
|
|
as_warn ("Missing section name");
|
|
|
|
ignore_rest_of_line ();
|
|
|
|
return;
|
|
|
|
}
|
1993-07-20 03:49:34 +08:00
|
|
|
c = *p;
|
|
|
|
*p = 0;
|
1993-09-14 05:32:07 +08:00
|
|
|
string = xmalloc ((unsigned long) (p - input_line_pointer + 1));
|
1993-07-20 03:49:34 +08:00
|
|
|
strcpy (string, input_line_pointer);
|
|
|
|
*p = c;
|
|
|
|
input_line_pointer = p;
|
|
|
|
}
|
1993-10-27 05:01:15 +08:00
|
|
|
|
|
|
|
/* Switch to the section, creating it if necessary. */
|
|
|
|
previous_section = now_seg;
|
|
|
|
previous_subsection = now_subseg;
|
|
|
|
|
|
|
|
new_sec = bfd_get_section_by_name (stdoutput, string) == NULL;
|
|
|
|
sec = subseg_new (string, 0);
|
|
|
|
|
|
|
|
/* If this section already existed, we don't bother to change the
|
|
|
|
flag values. */
|
|
|
|
if (! new_sec)
|
|
|
|
{
|
|
|
|
while (! is_end_of_line[(unsigned char) *input_line_pointer])
|
|
|
|
++input_line_pointer;
|
|
|
|
++input_line_pointer;
|
|
|
|
return;
|
|
|
|
}
|
1993-08-19 05:09:23 +08:00
|
|
|
|
1993-05-28 03:42:23 +08:00
|
|
|
SKIP_WHITESPACE ();
|
1994-01-16 01:31:05 +08:00
|
|
|
|
|
|
|
type = SHT_NULL;
|
|
|
|
attr = 0;
|
|
|
|
|
|
|
|
if (*input_line_pointer == ',')
|
1993-10-27 05:01:15 +08:00
|
|
|
{
|
|
|
|
/* Skip the comma. */
|
|
|
|
++input_line_pointer;
|
1993-08-19 05:09:23 +08:00
|
|
|
|
1993-10-27 05:01:15 +08:00
|
|
|
SKIP_WHITESPACE ();
|
1993-08-19 05:09:23 +08:00
|
|
|
if (*input_line_pointer == '"')
|
|
|
|
{
|
1993-10-27 05:01:15 +08:00
|
|
|
/* Pick up a string with a combination of a, w, x. */
|
|
|
|
++input_line_pointer;
|
|
|
|
while (*input_line_pointer != '"')
|
|
|
|
{
|
|
|
|
switch (*input_line_pointer)
|
|
|
|
{
|
|
|
|
case 'a':
|
1994-01-16 01:31:05 +08:00
|
|
|
attr |= SHF_ALLOC;
|
1993-10-27 05:01:15 +08:00
|
|
|
break;
|
|
|
|
case 'w':
|
1994-01-16 01:31:05 +08:00
|
|
|
attr |= SHF_WRITE;
|
1993-10-27 05:01:15 +08:00
|
|
|
break;
|
|
|
|
case 'x':
|
1994-01-16 01:31:05 +08:00
|
|
|
attr |= SHF_EXECINSTR;
|
1993-10-27 05:01:15 +08:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
as_warn ("Bad .section directive: want a,w,x in string");
|
|
|
|
ignore_rest_of_line ();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
++input_line_pointer;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Skip the closing quote. */
|
|
|
|
++input_line_pointer;
|
|
|
|
|
1993-08-19 05:09:23 +08:00
|
|
|
SKIP_WHITESPACE ();
|
1993-10-27 05:01:15 +08:00
|
|
|
if (*input_line_pointer == ',')
|
|
|
|
{
|
|
|
|
++input_line_pointer;
|
|
|
|
SKIP_WHITESPACE ();
|
|
|
|
if (*input_line_pointer == '@')
|
|
|
|
{
|
|
|
|
++input_line_pointer;
|
|
|
|
if (strncmp (input_line_pointer, "progbits",
|
|
|
|
sizeof "progbits" - 1) == 0)
|
|
|
|
{
|
1994-01-16 01:31:05 +08:00
|
|
|
type = SHT_PROGBITS;
|
1993-10-27 05:01:15 +08:00
|
|
|
input_line_pointer += sizeof "progbits" - 1;
|
|
|
|
}
|
|
|
|
else if (strncmp (input_line_pointer, "nobits",
|
|
|
|
sizeof "nobits" - 1) == 0)
|
|
|
|
{
|
1994-01-16 01:31:05 +08:00
|
|
|
type = SHT_NOBITS;
|
1993-10-27 05:01:15 +08:00
|
|
|
input_line_pointer += sizeof "nobits" - 1;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
as_warn ("Unrecognized section type");
|
|
|
|
ignore_rest_of_line ();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
1993-08-19 05:09:23 +08:00
|
|
|
}
|
1993-10-27 05:01:15 +08:00
|
|
|
else
|
1993-05-28 03:42:23 +08:00
|
|
|
{
|
1993-10-27 05:01:15 +08:00
|
|
|
do
|
|
|
|
{
|
|
|
|
SKIP_WHITESPACE ();
|
|
|
|
if (*input_line_pointer != '#')
|
|
|
|
{
|
|
|
|
as_warn ("Bad .section directive");
|
|
|
|
ignore_rest_of_line ();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
++input_line_pointer;
|
|
|
|
if (strncmp (input_line_pointer, "write",
|
|
|
|
sizeof "write" - 1) == 0)
|
|
|
|
{
|
1994-01-16 01:31:05 +08:00
|
|
|
attr |= SHF_WRITE;
|
1993-10-27 05:01:15 +08:00
|
|
|
input_line_pointer += sizeof "write" - 1;
|
|
|
|
}
|
|
|
|
else if (strncmp (input_line_pointer, "alloc",
|
|
|
|
sizeof "alloc" - 1) == 0)
|
|
|
|
{
|
1994-01-16 01:31:05 +08:00
|
|
|
attr |= SHF_ALLOC;
|
1993-10-27 05:01:15 +08:00
|
|
|
input_line_pointer += sizeof "alloc" - 1;
|
|
|
|
}
|
|
|
|
else if (strncmp (input_line_pointer, "execinstr",
|
|
|
|
sizeof "execinstr" - 1) == 0)
|
|
|
|
{
|
1994-01-16 01:31:05 +08:00
|
|
|
attr |= SHF_EXECINSTR;
|
1993-10-27 05:01:15 +08:00
|
|
|
input_line_pointer += sizeof "execinstr" - 1;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
as_warn ("Unrecognized section attribute");
|
|
|
|
ignore_rest_of_line ();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
SKIP_WHITESPACE ();
|
|
|
|
}
|
|
|
|
while (*input_line_pointer++ == ',');
|
|
|
|
--input_line_pointer;
|
1993-05-28 03:42:23 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
1994-01-16 01:31:05 +08:00
|
|
|
/* See if this is one of the special sections. */
|
|
|
|
for (i = 0; special_sections[i].name != NULL; i++)
|
|
|
|
{
|
|
|
|
if (string[1] == special_sections[i].name[1]
|
|
|
|
&& strcmp (string, special_sections[i].name) == 0)
|
|
|
|
{
|
|
|
|
if (type == SHT_NULL)
|
|
|
|
type = special_sections[i].type;
|
|
|
|
else if (type != special_sections[i].type)
|
|
|
|
as_warn ("Setting incorrect section type for %s", string);
|
|
|
|
|
|
|
|
if ((attr &~ special_sections[i].attributes) != 0)
|
|
|
|
as_warn ("Setting incorrect section attributes for %s", string);
|
|
|
|
attr |= special_sections[i].attributes;
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
flags = (SEC_RELOC
|
|
|
|
| ((attr & SHF_WRITE) ? 0 : SEC_READONLY)
|
|
|
|
| ((attr & SHF_ALLOC) ? SEC_ALLOC | SEC_LOAD : 0)
|
|
|
|
| ((attr & SHF_EXECINSTR) ? SEC_CODE : 0));
|
|
|
|
if (type == SHT_PROGBITS)
|
|
|
|
flags |= SEC_ALLOC | SEC_LOAD;
|
|
|
|
else if (type == SHT_NOBITS)
|
|
|
|
{
|
|
|
|
flags |= SEC_ALLOC;
|
|
|
|
flags &=~ SEC_LOAD;
|
|
|
|
}
|
|
|
|
|
1993-10-27 05:01:15 +08:00
|
|
|
bfd_set_section_flags (stdoutput, sec, flags);
|
1993-09-17 02:26:36 +08:00
|
|
|
|
1993-10-27 05:01:15 +08:00
|
|
|
demand_empty_rest_of_line ();
|
1993-09-17 02:26:36 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Change to the .data section. */
|
|
|
|
|
|
|
|
static void
|
|
|
|
obj_elf_data (i)
|
|
|
|
int i;
|
|
|
|
{
|
|
|
|
previous_section = now_seg;
|
|
|
|
previous_subsection = now_subseg;
|
|
|
|
s_data (i);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Change to the .text section. */
|
|
|
|
|
|
|
|
static void
|
|
|
|
obj_elf_text (i)
|
|
|
|
int i;
|
|
|
|
{
|
|
|
|
previous_section = now_seg;
|
|
|
|
previous_subsection = now_subseg;
|
|
|
|
s_text (i);
|
1993-05-28 03:42:23 +08:00
|
|
|
}
|
|
|
|
|
1993-08-05 07:10:43 +08:00
|
|
|
void
|
1993-09-14 05:32:07 +08:00
|
|
|
obj_elf_previous (ignore)
|
|
|
|
int ignore;
|
1993-07-20 03:49:34 +08:00
|
|
|
{
|
|
|
|
if (previous_section == 0)
|
|
|
|
{
|
|
|
|
as_bad (".previous without corresponding .section; ignored");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
subseg_set (previous_section, previous_subsection);
|
|
|
|
previous_section = 0;
|
|
|
|
}
|
|
|
|
|
1993-07-08 00:40:30 +08:00
|
|
|
static void
|
1993-09-14 05:32:07 +08:00
|
|
|
obj_elf_line (ignore)
|
|
|
|
int ignore;
|
1993-07-08 00:40:30 +08:00
|
|
|
{
|
|
|
|
/* Assume delimiter is part of expression. BSD4.2 as fails with
|
|
|
|
delightful bug, so we are not being incompatible here. */
|
|
|
|
new_logical_line ((char *) NULL, (int) (get_absolute_expression ()));
|
|
|
|
demand_empty_rest_of_line ();
|
|
|
|
}
|
1993-05-28 03:42:23 +08:00
|
|
|
|
1993-07-08 00:40:30 +08:00
|
|
|
void
|
|
|
|
obj_read_begin_hook ()
|
1993-05-28 03:42:23 +08:00
|
|
|
{
|
1993-11-24 15:42:03 +08:00
|
|
|
#ifdef ECOFF_DEBUGGING
|
|
|
|
ecoff_read_begin_hook ();
|
|
|
|
#endif
|
1993-05-28 03:42:23 +08:00
|
|
|
}
|
|
|
|
|
1993-07-08 00:40:30 +08:00
|
|
|
void
|
|
|
|
obj_symbol_new_hook (symbolP)
|
|
|
|
symbolS *symbolP;
|
1993-05-28 03:42:23 +08:00
|
|
|
{
|
1993-07-08 00:40:30 +08:00
|
|
|
#if 0 /* BFD already takes care of this */
|
|
|
|
elf32_symbol_type *esym = (elf32_symbol_type *) symbolP;
|
1993-05-28 03:42:23 +08:00
|
|
|
|
|
|
|
/* There is an Elf_Internal_Sym and an Elf_External_Sym. For now,
|
|
|
|
just zero them out. */
|
|
|
|
|
1993-07-08 00:40:30 +08:00
|
|
|
bzero ((char *) &esym->internal_elf_sym, sizeof (esym->internal_elf_sym));
|
|
|
|
bzero ((char *) &esym->native_elf_sym, sizeof (esym->native_elf_sym));
|
|
|
|
bzero ((char *) &esym->tc_data, sizeof (esym->tc_data));
|
|
|
|
#endif
|
1993-11-24 15:42:03 +08:00
|
|
|
#ifdef ECOFF_DEBUGGING
|
|
|
|
ecoff_symbol_new_hook (symbolP);
|
|
|
|
#endif
|
1993-05-28 03:42:23 +08:00
|
|
|
}
|
|
|
|
|
1993-07-08 00:40:30 +08:00
|
|
|
void
|
1993-09-14 05:32:07 +08:00
|
|
|
obj_elf_version (ignore)
|
|
|
|
int ignore;
|
1993-05-28 03:42:23 +08:00
|
|
|
{
|
1993-07-08 00:40:30 +08:00
|
|
|
char *name;
|
|
|
|
unsigned int c;
|
|
|
|
char ch;
|
|
|
|
char *p;
|
|
|
|
asection *seg = now_seg;
|
|
|
|
subsegT subseg = now_subseg;
|
|
|
|
Elf_Internal_Note i_note;
|
|
|
|
Elf_External_Note e_note;
|
|
|
|
asection *note_secp = (asection *) NULL;
|
|
|
|
int i, len;
|
|
|
|
|
|
|
|
SKIP_WHITESPACE ();
|
|
|
|
if (*input_line_pointer == '\"')
|
|
|
|
{
|
|
|
|
++input_line_pointer; /* -> 1st char of string. */
|
1993-05-28 03:42:23 +08:00
|
|
|
name = input_line_pointer;
|
|
|
|
|
1993-07-08 00:40:30 +08:00
|
|
|
while (is_a_char (c = next_char_of_string ()))
|
|
|
|
;
|
1993-05-28 03:42:23 +08:00
|
|
|
c = *input_line_pointer;
|
|
|
|
*input_line_pointer = '\0';
|
1993-07-08 00:40:30 +08:00
|
|
|
*(input_line_pointer - 1) = '\0';
|
1993-05-28 03:42:23 +08:00
|
|
|
*input_line_pointer = c;
|
|
|
|
|
Some more gcc lint, and:
* read.c (change_to_section): Removed. This is now done by
subseg_new.
(get_stab_string_offset): Rearranged somewhat. Create the section
using subseg_new. Store the string index in seg_info, rather than
in a static variable. Force the first string to be empty. Use
frag_more rather than FRAG_APPEND_1_CHAR.
(s_stab_generic): Rewrote.
* subsegs.h (segment_info_type): Added stabu union.
* subsegs.c (subseg_new): Initialize stab_string_size to 0.
* config/obj-aout.c: Don't include aout/stab_gnu.h.
(obj_aout_stab, obj_aout_desc): Removed.
(obj_pseudo_table): Removed desc and stabX entries.
* config/obj-bout.c: Same changes as config/obj-aout.c.
* config/obj-bout.h (S_SET_TYPE): Define.
(tc_bout_fix_to_chars): Declare.
* config/obj-coff.c (obj_coff_stab): Removed.
(obj_pseudo_table): Removed desc and stabX entries.
* config/obj-coff.h (SEPARATE_STAB_SECTIONS): Define.
* config/obj-coffbfd.c (current_stab_symbol): Removed.
* config/obj-coffbfd.h (obj_symbol_type): Removed n_strx, n_type,
n_other, n_desc and n_value fields.
(S_{S,G}ET_{OFFSET,OTHER,TYPE,DESC}): Removed.
(MAKE_STAB_SYMBOL): Removed.
* config/obj-ecoff.c (obj_ecoff_stab): Renamed to ecoff_stab.
Changed arguments and removed parsing code.
(obj_pseudo_table): Removed stabX entries.
* config/obj-ecoff.h (ecoff_stab): Declare.
(OBJ_PROCESS_STAB): Define.
* config/obj-elf.c: Don't include aout/stab_gnu.h.
(obj_elf_stab, obj_elf_xstab, obj_elf_desc,
elf_stab_symbol_string, elf_stab_symbol, obj_elf_stab_generic):
Removed.
(obj_pseudo_table): Removed desc, stabX and xstabs entries.
(obj_elf_version): Use subseg_new, not bfd_make_section. Don't
set SEC_LOAD for .note section.
(adjust_stab_sections): Get frag pointer from seg_info, rather
than looking through frags.
* config/obj-elf.h (S_{S,G}ET_{OTHER,TYPE,DESC}): Removed.
(SEPARATE_STAB_SECTIONS, INIT_STAB_SECTION, OBJ_PROCESS_STAB):
Define.
* config/obj-vms.c (obj_aout_stab): Removed.
(obj_pseudo_table): Removed stabX entries.
* config/obj-vms.h (S_SET_TYPE): Define.
1993-09-15 01:58:35 +08:00
|
|
|
/* create the .note section */
|
1993-05-28 03:42:23 +08:00
|
|
|
|
Some more gcc lint, and:
* read.c (change_to_section): Removed. This is now done by
subseg_new.
(get_stab_string_offset): Rearranged somewhat. Create the section
using subseg_new. Store the string index in seg_info, rather than
in a static variable. Force the first string to be empty. Use
frag_more rather than FRAG_APPEND_1_CHAR.
(s_stab_generic): Rewrote.
* subsegs.h (segment_info_type): Added stabu union.
* subsegs.c (subseg_new): Initialize stab_string_size to 0.
* config/obj-aout.c: Don't include aout/stab_gnu.h.
(obj_aout_stab, obj_aout_desc): Removed.
(obj_pseudo_table): Removed desc and stabX entries.
* config/obj-bout.c: Same changes as config/obj-aout.c.
* config/obj-bout.h (S_SET_TYPE): Define.
(tc_bout_fix_to_chars): Declare.
* config/obj-coff.c (obj_coff_stab): Removed.
(obj_pseudo_table): Removed desc and stabX entries.
* config/obj-coff.h (SEPARATE_STAB_SECTIONS): Define.
* config/obj-coffbfd.c (current_stab_symbol): Removed.
* config/obj-coffbfd.h (obj_symbol_type): Removed n_strx, n_type,
n_other, n_desc and n_value fields.
(S_{S,G}ET_{OFFSET,OTHER,TYPE,DESC}): Removed.
(MAKE_STAB_SYMBOL): Removed.
* config/obj-ecoff.c (obj_ecoff_stab): Renamed to ecoff_stab.
Changed arguments and removed parsing code.
(obj_pseudo_table): Removed stabX entries.
* config/obj-ecoff.h (ecoff_stab): Declare.
(OBJ_PROCESS_STAB): Define.
* config/obj-elf.c: Don't include aout/stab_gnu.h.
(obj_elf_stab, obj_elf_xstab, obj_elf_desc,
elf_stab_symbol_string, elf_stab_symbol, obj_elf_stab_generic):
Removed.
(obj_pseudo_table): Removed desc, stabX and xstabs entries.
(obj_elf_version): Use subseg_new, not bfd_make_section. Don't
set SEC_LOAD for .note section.
(adjust_stab_sections): Get frag pointer from seg_info, rather
than looking through frags.
* config/obj-elf.h (S_{S,G}ET_{OTHER,TYPE,DESC}): Removed.
(SEPARATE_STAB_SECTIONS, INIT_STAB_SECTION, OBJ_PROCESS_STAB):
Define.
* config/obj-vms.c (obj_aout_stab): Removed.
(obj_pseudo_table): Removed stabX entries.
* config/obj-vms.h (S_SET_TYPE): Define.
1993-09-15 01:58:35 +08:00
|
|
|
note_secp = subseg_new (".note", 0);
|
|
|
|
bfd_set_section_flags (stdoutput,
|
|
|
|
note_secp,
|
|
|
|
SEC_HAS_CONTENTS | SEC_READONLY);
|
1993-05-28 03:42:23 +08:00
|
|
|
|
|
|
|
/* process the version string */
|
|
|
|
|
1993-07-08 00:40:30 +08:00
|
|
|
len = strlen (name);
|
|
|
|
|
1993-08-05 07:10:43 +08:00
|
|
|
i_note.namesz = ((len + 1) + 3) & ~3; /* round this to word boundary */
|
|
|
|
i_note.descsz = 0; /* no description */
|
1993-07-08 00:40:30 +08:00
|
|
|
i_note.type = NT_VERSION;
|
|
|
|
p = frag_more (sizeof (e_note.namesz));
|
|
|
|
md_number_to_chars (p, (valueT) i_note.namesz, 4);
|
|
|
|
p = frag_more (sizeof (e_note.descsz));
|
|
|
|
md_number_to_chars (p, (valueT) i_note.descsz, 4);
|
|
|
|
p = frag_more (sizeof (e_note.type));
|
|
|
|
md_number_to_chars (p, (valueT) i_note.type, 4);
|
|
|
|
|
|
|
|
for (i = 0; i < len; i++)
|
|
|
|
{
|
|
|
|
ch = *(name + i);
|
|
|
|
{
|
|
|
|
FRAG_APPEND_1_CHAR (ch);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
frag_align (2, 0);
|
1993-05-28 03:42:23 +08:00
|
|
|
|
1993-09-14 05:32:07 +08:00
|
|
|
subseg_set (seg, subseg);
|
1993-05-28 03:42:23 +08:00
|
|
|
}
|
1993-07-08 00:40:30 +08:00
|
|
|
else
|
|
|
|
{
|
1993-08-05 07:10:43 +08:00
|
|
|
as_bad ("Expected quoted string");
|
1993-05-28 03:42:23 +08:00
|
|
|
}
|
1993-07-08 00:40:30 +08:00
|
|
|
demand_empty_rest_of_line ();
|
1993-05-28 03:42:23 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
1993-09-14 05:32:07 +08:00
|
|
|
obj_elf_size (ignore)
|
|
|
|
int ignore;
|
1993-05-28 03:42:23 +08:00
|
|
|
{
|
|
|
|
char *name = input_line_pointer;
|
|
|
|
char c = get_symbol_end ();
|
|
|
|
char *p;
|
|
|
|
expressionS exp;
|
|
|
|
symbolS *sym;
|
|
|
|
|
|
|
|
p = input_line_pointer;
|
|
|
|
*p = c;
|
|
|
|
SKIP_WHITESPACE ();
|
|
|
|
if (*input_line_pointer != ',')
|
|
|
|
{
|
|
|
|
*p = 0;
|
|
|
|
as_bad ("expected comma after name `%s' in .size directive", name);
|
|
|
|
*p = c;
|
|
|
|
ignore_rest_of_line ();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
input_line_pointer++;
|
* Extensive changes to permit symbols to contain any expression
type and to delay the computation of the expression until the
value is actually needed. This permits setting symbols to values
calculated based on object code size. Expressions were changed to
no longer be in a section, to stop the overloading of segment and
expression type that previously occurred.
* as.c (big_section, pass1_section, diff_section, absent_section):
Removed.
(expr_section): Added (used for dummy symbols which hold
intermediate expression values).
(perform_an_assembly_pass): Create expr_section, do not create the
sections now removed.
* as.h (segT): Removed SEG_ABSENT, SEG_PASS1, SEG_BIG, and
SEG_DIFFERENCE. Added SEG_EXPR.
(SEG_NORMAL): Corresponding changes.
* subsegs.c (seg_name, subsegs_begin): Changed accordingly.
* write.c (write_object_file): Ditto.
* config/obj-aout.c (seg_N_TYPE): Ditto.
* config/obj-bout.c (seg_N_TYPE): Ditto.
* config/obj-coff.c (seg_N_TYPE): Ditto.
* config/obj-coffbfd.c (seg_N_TYPE): Ditto.
* config/obj-vms.c (seg_N_TYPE): Ditto.
* expr.h (operatorT): Moved in from expr.c, added some values.
(expressionS): Added X_op field, removed X_seg field; renamed
X_subtract_symbol to X_op_symbol.
* expr.c: Extensive changes to assign expression types rather than
sections and to simplify the parsing.
* write.c (fix_new_internal): New static function.
(fix_new): Removed sub_symbol argument.
(fix_new_exp): New function, takes expression argument.
* write.h: Prototype changes for fix_new and fix_new_exp.
* cond.c (s_if): Changed accordingly.
* read.c (s_lsym, pseudo_set, emit_expr, parse_bitfield_cons,
parse_repeat_cons, get_segmented_expression,
get_known_segmented_expression, get_absolute_expression): Ditto.
* symbols.c (resolve_symbol_value, S_GET_VALUE, S_SET_VALUE):
Ditto.
* write.c (write_object_file): Ditto.
* config/obj-coff.c (obj_coff_def, obj_coff_val): Ditto.
* config/obj-coffbfd.c (obj_coff_def, obj_coff_val,
obj_coff_endef, yank_symbols): Ditto.
* config/obj-elf.c (obj_elf_stab_generic, obj_elf_size): Ditto.
* config/tc-a29k.c (md_assemble, parse_operand, machine_ip,
print_insn, md_operand): Ditto.
* config/tc-h8300.c (parse_exp, colonmod24, check_operand,
do_a_fix_imm, build_bytes): Ditto.
* config/tc-h8500.c (parse_exp, skip_colonthing, parse_reglist,
get_specific, check, insert, md_convert_frag): Ditto.
* config/tc-hppa.c (the_insn, fix_new_hppa, cons_fix_new_hppa,
md_assemble, pa_ip, getExpression, getAbsoluteExpression,
evaluateAbsolute, pa_build_unwind_subspace, pa_entry,
process_exit): Ditto.
* config/tc-hppa.h (STAB_FIXUP, is_DP_relative, is_PC_relative,
is_complex): Ditto.
* config/tc-i386.c (pe, md_assemble, i386_operand,
md_estimate_size_before_relax, md_create_long_jump): Ditto.
* config/tc-i860.c (md_assemble, getExpression, print_insn):
Ditto.
* config/tc-i960.c (parse_expr, subs, segs, md_convert_frag,
get_cdisp, mem_fmt, parse_ldconst, relax_cobr, s_sysproc,
i960_handle_align): Ditto.
* config/tc-m68k.c (struct m68k_exp, struct m68k_it, seg, op,
subs, add_fix, isvar, m68k_ip, md_assemble, md_convert_frag_1,
md_estimate_size_before_relax, md_create_long_jump, get_num):
Ditto.
* config/tc-m88k.c (md_assemble, get_imm16, get_pcr,
md_create_short_jump, md_create_long_jump): Ditto.
* config/tc-mips.c (md_assemble, append_insn, gp_reference,
macro_build, macro, my_getExpression): Ditto. Also removed
get_optional_absolute_expression; just use get_absolute_expression
instead.
* config/tc-ns32k.c (get_addr_mode, evaluate_expr, convert_iif,
fix_new_ns32k, fix_new_ns32k_exp, cons_fix_new_ns32k): Ditto.
* config/tc-ns32k.h (fix_new_ns32k prototype): Ditto.
* config/tc-sh.c (parse_exp, check, insert, md_convert_frag):
Ditto.
* config/tc-sparc.c (md_assemble, sparc_ip, getExpression,
print_insn): Ditto.
* config/tc-tahoe.c (struct top, md_estimate_size_before_relax,
tip_op, md_assemble): Ditto.
* config/tc-vax.c (seg_of_operand, md_assemble,
md_estimate_size_before_relax, md_create_long_jump): Ditto.
* config/tc-z8k.c (parse_exp, check_operand, newfix): Ditto.
1993-07-21 08:41:42 +08:00
|
|
|
expression (&exp);
|
|
|
|
if (exp.X_op == O_absent)
|
1993-05-28 03:42:23 +08:00
|
|
|
{
|
|
|
|
as_bad ("missing expression in .size directive");
|
* Extensive changes to permit symbols to contain any expression
type and to delay the computation of the expression until the
value is actually needed. This permits setting symbols to values
calculated based on object code size. Expressions were changed to
no longer be in a section, to stop the overloading of segment and
expression type that previously occurred.
* as.c (big_section, pass1_section, diff_section, absent_section):
Removed.
(expr_section): Added (used for dummy symbols which hold
intermediate expression values).
(perform_an_assembly_pass): Create expr_section, do not create the
sections now removed.
* as.h (segT): Removed SEG_ABSENT, SEG_PASS1, SEG_BIG, and
SEG_DIFFERENCE. Added SEG_EXPR.
(SEG_NORMAL): Corresponding changes.
* subsegs.c (seg_name, subsegs_begin): Changed accordingly.
* write.c (write_object_file): Ditto.
* config/obj-aout.c (seg_N_TYPE): Ditto.
* config/obj-bout.c (seg_N_TYPE): Ditto.
* config/obj-coff.c (seg_N_TYPE): Ditto.
* config/obj-coffbfd.c (seg_N_TYPE): Ditto.
* config/obj-vms.c (seg_N_TYPE): Ditto.
* expr.h (operatorT): Moved in from expr.c, added some values.
(expressionS): Added X_op field, removed X_seg field; renamed
X_subtract_symbol to X_op_symbol.
* expr.c: Extensive changes to assign expression types rather than
sections and to simplify the parsing.
* write.c (fix_new_internal): New static function.
(fix_new): Removed sub_symbol argument.
(fix_new_exp): New function, takes expression argument.
* write.h: Prototype changes for fix_new and fix_new_exp.
* cond.c (s_if): Changed accordingly.
* read.c (s_lsym, pseudo_set, emit_expr, parse_bitfield_cons,
parse_repeat_cons, get_segmented_expression,
get_known_segmented_expression, get_absolute_expression): Ditto.
* symbols.c (resolve_symbol_value, S_GET_VALUE, S_SET_VALUE):
Ditto.
* write.c (write_object_file): Ditto.
* config/obj-coff.c (obj_coff_def, obj_coff_val): Ditto.
* config/obj-coffbfd.c (obj_coff_def, obj_coff_val,
obj_coff_endef, yank_symbols): Ditto.
* config/obj-elf.c (obj_elf_stab_generic, obj_elf_size): Ditto.
* config/tc-a29k.c (md_assemble, parse_operand, machine_ip,
print_insn, md_operand): Ditto.
* config/tc-h8300.c (parse_exp, colonmod24, check_operand,
do_a_fix_imm, build_bytes): Ditto.
* config/tc-h8500.c (parse_exp, skip_colonthing, parse_reglist,
get_specific, check, insert, md_convert_frag): Ditto.
* config/tc-hppa.c (the_insn, fix_new_hppa, cons_fix_new_hppa,
md_assemble, pa_ip, getExpression, getAbsoluteExpression,
evaluateAbsolute, pa_build_unwind_subspace, pa_entry,
process_exit): Ditto.
* config/tc-hppa.h (STAB_FIXUP, is_DP_relative, is_PC_relative,
is_complex): Ditto.
* config/tc-i386.c (pe, md_assemble, i386_operand,
md_estimate_size_before_relax, md_create_long_jump): Ditto.
* config/tc-i860.c (md_assemble, getExpression, print_insn):
Ditto.
* config/tc-i960.c (parse_expr, subs, segs, md_convert_frag,
get_cdisp, mem_fmt, parse_ldconst, relax_cobr, s_sysproc,
i960_handle_align): Ditto.
* config/tc-m68k.c (struct m68k_exp, struct m68k_it, seg, op,
subs, add_fix, isvar, m68k_ip, md_assemble, md_convert_frag_1,
md_estimate_size_before_relax, md_create_long_jump, get_num):
Ditto.
* config/tc-m88k.c (md_assemble, get_imm16, get_pcr,
md_create_short_jump, md_create_long_jump): Ditto.
* config/tc-mips.c (md_assemble, append_insn, gp_reference,
macro_build, macro, my_getExpression): Ditto. Also removed
get_optional_absolute_expression; just use get_absolute_expression
instead.
* config/tc-ns32k.c (get_addr_mode, evaluate_expr, convert_iif,
fix_new_ns32k, fix_new_ns32k_exp, cons_fix_new_ns32k): Ditto.
* config/tc-ns32k.h (fix_new_ns32k prototype): Ditto.
* config/tc-sh.c (parse_exp, check, insert, md_convert_frag):
Ditto.
* config/tc-sparc.c (md_assemble, sparc_ip, getExpression,
print_insn): Ditto.
* config/tc-tahoe.c (struct top, md_estimate_size_before_relax,
tip_op, md_assemble): Ditto.
* config/tc-vax.c (seg_of_operand, md_assemble,
md_estimate_size_before_relax, md_create_long_jump): Ditto.
* config/tc-z8k.c (parse_exp, check_operand, newfix): Ditto.
1993-07-21 08:41:42 +08:00
|
|
|
exp.X_op = O_constant;
|
1993-05-28 03:42:23 +08:00
|
|
|
exp.X_add_number = 0;
|
|
|
|
}
|
|
|
|
*p = 0;
|
|
|
|
sym = symbol_find_or_make (name);
|
|
|
|
*p = c;
|
* Extensive changes to permit symbols to contain any expression
type and to delay the computation of the expression until the
value is actually needed. This permits setting symbols to values
calculated based on object code size. Expressions were changed to
no longer be in a section, to stop the overloading of segment and
expression type that previously occurred.
* as.c (big_section, pass1_section, diff_section, absent_section):
Removed.
(expr_section): Added (used for dummy symbols which hold
intermediate expression values).
(perform_an_assembly_pass): Create expr_section, do not create the
sections now removed.
* as.h (segT): Removed SEG_ABSENT, SEG_PASS1, SEG_BIG, and
SEG_DIFFERENCE. Added SEG_EXPR.
(SEG_NORMAL): Corresponding changes.
* subsegs.c (seg_name, subsegs_begin): Changed accordingly.
* write.c (write_object_file): Ditto.
* config/obj-aout.c (seg_N_TYPE): Ditto.
* config/obj-bout.c (seg_N_TYPE): Ditto.
* config/obj-coff.c (seg_N_TYPE): Ditto.
* config/obj-coffbfd.c (seg_N_TYPE): Ditto.
* config/obj-vms.c (seg_N_TYPE): Ditto.
* expr.h (operatorT): Moved in from expr.c, added some values.
(expressionS): Added X_op field, removed X_seg field; renamed
X_subtract_symbol to X_op_symbol.
* expr.c: Extensive changes to assign expression types rather than
sections and to simplify the parsing.
* write.c (fix_new_internal): New static function.
(fix_new): Removed sub_symbol argument.
(fix_new_exp): New function, takes expression argument.
* write.h: Prototype changes for fix_new and fix_new_exp.
* cond.c (s_if): Changed accordingly.
* read.c (s_lsym, pseudo_set, emit_expr, parse_bitfield_cons,
parse_repeat_cons, get_segmented_expression,
get_known_segmented_expression, get_absolute_expression): Ditto.
* symbols.c (resolve_symbol_value, S_GET_VALUE, S_SET_VALUE):
Ditto.
* write.c (write_object_file): Ditto.
* config/obj-coff.c (obj_coff_def, obj_coff_val): Ditto.
* config/obj-coffbfd.c (obj_coff_def, obj_coff_val,
obj_coff_endef, yank_symbols): Ditto.
* config/obj-elf.c (obj_elf_stab_generic, obj_elf_size): Ditto.
* config/tc-a29k.c (md_assemble, parse_operand, machine_ip,
print_insn, md_operand): Ditto.
* config/tc-h8300.c (parse_exp, colonmod24, check_operand,
do_a_fix_imm, build_bytes): Ditto.
* config/tc-h8500.c (parse_exp, skip_colonthing, parse_reglist,
get_specific, check, insert, md_convert_frag): Ditto.
* config/tc-hppa.c (the_insn, fix_new_hppa, cons_fix_new_hppa,
md_assemble, pa_ip, getExpression, getAbsoluteExpression,
evaluateAbsolute, pa_build_unwind_subspace, pa_entry,
process_exit): Ditto.
* config/tc-hppa.h (STAB_FIXUP, is_DP_relative, is_PC_relative,
is_complex): Ditto.
* config/tc-i386.c (pe, md_assemble, i386_operand,
md_estimate_size_before_relax, md_create_long_jump): Ditto.
* config/tc-i860.c (md_assemble, getExpression, print_insn):
Ditto.
* config/tc-i960.c (parse_expr, subs, segs, md_convert_frag,
get_cdisp, mem_fmt, parse_ldconst, relax_cobr, s_sysproc,
i960_handle_align): Ditto.
* config/tc-m68k.c (struct m68k_exp, struct m68k_it, seg, op,
subs, add_fix, isvar, m68k_ip, md_assemble, md_convert_frag_1,
md_estimate_size_before_relax, md_create_long_jump, get_num):
Ditto.
* config/tc-m88k.c (md_assemble, get_imm16, get_pcr,
md_create_short_jump, md_create_long_jump): Ditto.
* config/tc-mips.c (md_assemble, append_insn, gp_reference,
macro_build, macro, my_getExpression): Ditto. Also removed
get_optional_absolute_expression; just use get_absolute_expression
instead.
* config/tc-ns32k.c (get_addr_mode, evaluate_expr, convert_iif,
fix_new_ns32k, fix_new_ns32k_exp, cons_fix_new_ns32k): Ditto.
* config/tc-ns32k.h (fix_new_ns32k prototype): Ditto.
* config/tc-sh.c (parse_exp, check, insert, md_convert_frag):
Ditto.
* config/tc-sparc.c (md_assemble, sparc_ip, getExpression,
print_insn): Ditto.
* config/tc-tahoe.c (struct top, md_estimate_size_before_relax,
tip_op, md_assemble): Ditto.
* config/tc-vax.c (seg_of_operand, md_assemble,
md_estimate_size_before_relax, md_create_long_jump): Ditto.
* config/tc-z8k.c (parse_exp, check_operand, newfix): Ditto.
1993-07-21 08:41:42 +08:00
|
|
|
if (exp.X_op == O_constant)
|
1993-05-28 03:42:23 +08:00
|
|
|
S_SET_SIZE (sym, exp.X_add_number);
|
|
|
|
else
|
1993-07-08 00:40:30 +08:00
|
|
|
{
|
|
|
|
#if 0
|
|
|
|
static int warned;
|
|
|
|
if (!warned)
|
|
|
|
{
|
|
|
|
as_tsktsk (".size expressions not yet supported, ignored");
|
|
|
|
warned++;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|
1993-05-28 03:42:23 +08:00
|
|
|
demand_empty_rest_of_line ();
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
1993-09-14 05:32:07 +08:00
|
|
|
obj_elf_type (ignore)
|
|
|
|
int ignore;
|
1993-05-28 03:42:23 +08:00
|
|
|
{
|
|
|
|
char *name = input_line_pointer;
|
|
|
|
char c = get_symbol_end ();
|
|
|
|
char *p;
|
1993-07-08 00:40:30 +08:00
|
|
|
int type = 0;
|
1993-05-28 03:42:23 +08:00
|
|
|
symbolS *sym;
|
|
|
|
|
|
|
|
p = input_line_pointer;
|
|
|
|
*p = c;
|
|
|
|
SKIP_WHITESPACE ();
|
|
|
|
if (*input_line_pointer != ',')
|
|
|
|
{
|
|
|
|
as_bad ("expected comma after name in .type directive");
|
|
|
|
egress:
|
|
|
|
ignore_rest_of_line ();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
input_line_pointer++;
|
|
|
|
SKIP_WHITESPACE ();
|
1993-08-19 05:09:23 +08:00
|
|
|
if (*input_line_pointer != '#' && *input_line_pointer != '@')
|
1993-05-28 03:42:23 +08:00
|
|
|
{
|
1993-08-19 05:09:23 +08:00
|
|
|
as_bad ("expected `#' or `@' after comma in .type directive");
|
1993-05-28 03:42:23 +08:00
|
|
|
goto egress;
|
|
|
|
}
|
|
|
|
input_line_pointer++;
|
|
|
|
if (!strncmp ("function", input_line_pointer, sizeof ("function") - 1))
|
|
|
|
{
|
|
|
|
type = BSF_FUNCTION;
|
|
|
|
input_line_pointer += sizeof ("function") - 1;
|
|
|
|
}
|
1993-07-08 00:40:30 +08:00
|
|
|
else if (!strncmp ("object", input_line_pointer, sizeof ("object") - 1))
|
|
|
|
{
|
|
|
|
input_line_pointer += sizeof ("object") - 1;
|
|
|
|
}
|
1993-05-28 03:42:23 +08:00
|
|
|
else
|
|
|
|
{
|
|
|
|
as_bad ("unrecognized symbol type, ignored");
|
|
|
|
goto egress;
|
|
|
|
}
|
|
|
|
demand_empty_rest_of_line ();
|
|
|
|
*p = 0;
|
|
|
|
sym = symbol_find_or_make (name);
|
1993-07-08 00:40:30 +08:00
|
|
|
sym->bsym->flags |= type;
|
1993-05-28 03:42:23 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
1993-09-14 05:32:07 +08:00
|
|
|
obj_elf_ident (ignore)
|
|
|
|
int ignore;
|
1993-05-28 03:42:23 +08:00
|
|
|
{
|
1993-07-08 00:40:30 +08:00
|
|
|
static segT comment_section;
|
|
|
|
segT old_section = now_seg;
|
|
|
|
int old_subsection = now_subseg;
|
1993-05-28 03:42:23 +08:00
|
|
|
|
1993-07-08 00:40:30 +08:00
|
|
|
if (!comment_section)
|
|
|
|
{
|
|
|
|
char *p;
|
|
|
|
comment_section = subseg_new (".comment", 0);
|
1993-10-27 05:01:15 +08:00
|
|
|
bfd_set_section_flags (stdoutput, comment_section,
|
|
|
|
SEC_READONLY | SEC_HAS_CONTENTS);
|
1993-07-08 00:40:30 +08:00
|
|
|
p = frag_more (1);
|
|
|
|
*p = 0;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
subseg_set (comment_section, 0);
|
|
|
|
stringer (1);
|
|
|
|
subseg_set (old_section, old_subsection);
|
|
|
|
}
|
|
|
|
|
1994-01-16 01:31:05 +08:00
|
|
|
#ifdef INIT_STAB_SECTION
|
|
|
|
|
1993-09-21 07:29:42 +08:00
|
|
|
/* The first entry in a .stabs section is special. */
|
|
|
|
|
|
|
|
void
|
|
|
|
obj_elf_init_stab_section (seg)
|
|
|
|
segT seg;
|
|
|
|
{
|
1993-10-27 05:01:15 +08:00
|
|
|
char *file;
|
1993-09-21 07:29:42 +08:00
|
|
|
char *p;
|
1993-11-24 15:42:03 +08:00
|
|
|
char *stabstr_name;
|
1993-09-21 07:29:42 +08:00
|
|
|
unsigned int stroff;
|
|
|
|
|
1993-10-27 05:01:15 +08:00
|
|
|
/* Force the section to align to a longword boundary. Without this,
|
|
|
|
UnixWare ar crashes. */
|
|
|
|
bfd_set_section_alignment (stdoutput, seg, 2);
|
|
|
|
|
1993-09-21 07:29:42 +08:00
|
|
|
p = frag_more (12);
|
1993-10-27 05:01:15 +08:00
|
|
|
as_where (&file, (unsigned int *) NULL);
|
1993-11-24 15:42:03 +08:00
|
|
|
stabstr_name = (char *) alloca (strlen (segment_name (seg)) + 4);
|
|
|
|
strcpy (stabstr_name, segment_name (seg));
|
|
|
|
strcat (stabstr_name, "str");
|
|
|
|
stroff = get_stab_string_offset (file, stabstr_name);
|
1993-09-21 07:29:42 +08:00
|
|
|
know (stroff == 1);
|
|
|
|
md_number_to_chars (p, stroff, 4);
|
|
|
|
seg_info (seg)->stabu.p = p;
|
|
|
|
}
|
|
|
|
|
1994-01-16 01:31:05 +08:00
|
|
|
#endif
|
|
|
|
|
1993-09-21 07:29:42 +08:00
|
|
|
/* Fill in the counts in the first entry in a .stabs section. */
|
|
|
|
|
1993-08-05 07:10:43 +08:00
|
|
|
static void
|
|
|
|
adjust_stab_sections (abfd, sec, xxx)
|
|
|
|
bfd *abfd;
|
|
|
|
asection *sec;
|
|
|
|
PTR xxx;
|
|
|
|
{
|
|
|
|
char *name;
|
|
|
|
asection *strsec;
|
Some more gcc lint, and:
* read.c (change_to_section): Removed. This is now done by
subseg_new.
(get_stab_string_offset): Rearranged somewhat. Create the section
using subseg_new. Store the string index in seg_info, rather than
in a static variable. Force the first string to be empty. Use
frag_more rather than FRAG_APPEND_1_CHAR.
(s_stab_generic): Rewrote.
* subsegs.h (segment_info_type): Added stabu union.
* subsegs.c (subseg_new): Initialize stab_string_size to 0.
* config/obj-aout.c: Don't include aout/stab_gnu.h.
(obj_aout_stab, obj_aout_desc): Removed.
(obj_pseudo_table): Removed desc and stabX entries.
* config/obj-bout.c: Same changes as config/obj-aout.c.
* config/obj-bout.h (S_SET_TYPE): Define.
(tc_bout_fix_to_chars): Declare.
* config/obj-coff.c (obj_coff_stab): Removed.
(obj_pseudo_table): Removed desc and stabX entries.
* config/obj-coff.h (SEPARATE_STAB_SECTIONS): Define.
* config/obj-coffbfd.c (current_stab_symbol): Removed.
* config/obj-coffbfd.h (obj_symbol_type): Removed n_strx, n_type,
n_other, n_desc and n_value fields.
(S_{S,G}ET_{OFFSET,OTHER,TYPE,DESC}): Removed.
(MAKE_STAB_SYMBOL): Removed.
* config/obj-ecoff.c (obj_ecoff_stab): Renamed to ecoff_stab.
Changed arguments and removed parsing code.
(obj_pseudo_table): Removed stabX entries.
* config/obj-ecoff.h (ecoff_stab): Declare.
(OBJ_PROCESS_STAB): Define.
* config/obj-elf.c: Don't include aout/stab_gnu.h.
(obj_elf_stab, obj_elf_xstab, obj_elf_desc,
elf_stab_symbol_string, elf_stab_symbol, obj_elf_stab_generic):
Removed.
(obj_pseudo_table): Removed desc, stabX and xstabs entries.
(obj_elf_version): Use subseg_new, not bfd_make_section. Don't
set SEC_LOAD for .note section.
(adjust_stab_sections): Get frag pointer from seg_info, rather
than looking through frags.
* config/obj-elf.h (S_{S,G}ET_{OTHER,TYPE,DESC}): Removed.
(SEPARATE_STAB_SECTIONS, INIT_STAB_SECTION, OBJ_PROCESS_STAB):
Define.
* config/obj-vms.c (obj_aout_stab): Removed.
(obj_pseudo_table): Removed stabX entries.
* config/obj-vms.h (S_SET_TYPE): Define.
1993-09-15 01:58:35 +08:00
|
|
|
char *p;
|
1993-08-05 07:10:43 +08:00
|
|
|
int strsz, nsyms;
|
|
|
|
|
|
|
|
if (strncmp (".stab", sec->name, 5))
|
|
|
|
return;
|
|
|
|
if (!strcmp ("str", sec->name + strlen (sec->name) - 3))
|
|
|
|
return;
|
|
|
|
|
|
|
|
name = (char *) alloca (strlen (sec->name) + 4);
|
|
|
|
strcpy (name, sec->name);
|
|
|
|
strcat (name, "str");
|
|
|
|
strsec = bfd_get_section_by_name (abfd, name);
|
|
|
|
if (strsec)
|
|
|
|
strsz = bfd_section_size (abfd, strsec);
|
|
|
|
else
|
|
|
|
strsz = 0;
|
|
|
|
nsyms = bfd_section_size (abfd, sec) / 12 - 1;
|
|
|
|
|
Some more gcc lint, and:
* read.c (change_to_section): Removed. This is now done by
subseg_new.
(get_stab_string_offset): Rearranged somewhat. Create the section
using subseg_new. Store the string index in seg_info, rather than
in a static variable. Force the first string to be empty. Use
frag_more rather than FRAG_APPEND_1_CHAR.
(s_stab_generic): Rewrote.
* subsegs.h (segment_info_type): Added stabu union.
* subsegs.c (subseg_new): Initialize stab_string_size to 0.
* config/obj-aout.c: Don't include aout/stab_gnu.h.
(obj_aout_stab, obj_aout_desc): Removed.
(obj_pseudo_table): Removed desc and stabX entries.
* config/obj-bout.c: Same changes as config/obj-aout.c.
* config/obj-bout.h (S_SET_TYPE): Define.
(tc_bout_fix_to_chars): Declare.
* config/obj-coff.c (obj_coff_stab): Removed.
(obj_pseudo_table): Removed desc and stabX entries.
* config/obj-coff.h (SEPARATE_STAB_SECTIONS): Define.
* config/obj-coffbfd.c (current_stab_symbol): Removed.
* config/obj-coffbfd.h (obj_symbol_type): Removed n_strx, n_type,
n_other, n_desc and n_value fields.
(S_{S,G}ET_{OFFSET,OTHER,TYPE,DESC}): Removed.
(MAKE_STAB_SYMBOL): Removed.
* config/obj-ecoff.c (obj_ecoff_stab): Renamed to ecoff_stab.
Changed arguments and removed parsing code.
(obj_pseudo_table): Removed stabX entries.
* config/obj-ecoff.h (ecoff_stab): Declare.
(OBJ_PROCESS_STAB): Define.
* config/obj-elf.c: Don't include aout/stab_gnu.h.
(obj_elf_stab, obj_elf_xstab, obj_elf_desc,
elf_stab_symbol_string, elf_stab_symbol, obj_elf_stab_generic):
Removed.
(obj_pseudo_table): Removed desc, stabX and xstabs entries.
(obj_elf_version): Use subseg_new, not bfd_make_section. Don't
set SEC_LOAD for .note section.
(adjust_stab_sections): Get frag pointer from seg_info, rather
than looking through frags.
* config/obj-elf.h (S_{S,G}ET_{OTHER,TYPE,DESC}): Removed.
(SEPARATE_STAB_SECTIONS, INIT_STAB_SECTION, OBJ_PROCESS_STAB):
Define.
* config/obj-vms.c (obj_aout_stab): Removed.
(obj_pseudo_table): Removed stabX entries.
* config/obj-vms.h (S_SET_TYPE): Define.
1993-09-15 01:58:35 +08:00
|
|
|
p = seg_info (sec)->stabu.p;
|
|
|
|
assert (p != 0);
|
1993-08-05 07:10:43 +08:00
|
|
|
|
1993-09-21 07:29:42 +08:00
|
|
|
bfd_h_put_16 (abfd, (bfd_vma) nsyms, (bfd_byte *) p + 6);
|
|
|
|
bfd_h_put_32 (abfd, (bfd_vma) strsz, (bfd_byte *) p + 8);
|
1993-08-05 07:10:43 +08:00
|
|
|
}
|
|
|
|
|
1993-11-24 15:42:03 +08:00
|
|
|
#ifdef ECOFF_DEBUGGING
|
|
|
|
|
|
|
|
/* This function is called by the ECOFF code. It is supposed to
|
|
|
|
record the external symbol information so that the backend can
|
|
|
|
write it out correctly. The ELF backend doesn't actually handle
|
|
|
|
this at the moment, so we do it ourselves. We save the information
|
|
|
|
in the symbol. */
|
|
|
|
|
|
|
|
void
|
|
|
|
obj_ecoff_set_ext (sym, ext)
|
|
|
|
symbolS *sym;
|
|
|
|
EXTR *ext;
|
|
|
|
{
|
|
|
|
sym->bsym->udata = (PTR) ext;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* This function is called by bfd_ecoff_debug_externals. It is
|
|
|
|
supposed to *EXT to the external symbol information, and return
|
|
|
|
whether the symbol should be used at all. */
|
|
|
|
|
|
|
|
static boolean
|
|
|
|
elf_get_extr (sym, ext)
|
|
|
|
asymbol *sym;
|
|
|
|
EXTR *ext;
|
|
|
|
{
|
|
|
|
if (sym->udata == NULL)
|
|
|
|
return false;
|
|
|
|
*ext = *(EXTR *) sym->udata;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* This function is called by bfd_ecoff_debug_externals. It has
|
|
|
|
nothing to do for ELF. */
|
|
|
|
|
|
|
|
/*ARGSUSED*/
|
|
|
|
static void
|
|
|
|
elf_set_index (sym, indx)
|
|
|
|
asymbol *sym;
|
|
|
|
bfd_size_type indx;
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif /* ECOFF_DEBUGGING */
|
|
|
|
|
1993-07-08 00:40:30 +08:00
|
|
|
void
|
|
|
|
elf_frob_file ()
|
|
|
|
{
|
1993-08-05 07:10:43 +08:00
|
|
|
bfd_map_over_sections (stdoutput, adjust_stab_sections, (PTR) 0);
|
|
|
|
|
1993-07-08 00:40:30 +08:00
|
|
|
#ifdef elf_tc_symbol
|
1993-08-05 07:10:43 +08:00
|
|
|
{
|
|
|
|
int i;
|
1993-07-08 00:40:30 +08:00
|
|
|
|
1993-08-05 07:10:43 +08:00
|
|
|
for (i = 0; i < stdoutput->symcount; i++)
|
1993-09-04 06:36:26 +08:00
|
|
|
elf_tc_symbol (stdoutput, (PTR) (stdoutput->outsymbols[i]),
|
1993-08-05 07:10:43 +08:00
|
|
|
i + 1);
|
|
|
|
}
|
1993-07-08 00:40:30 +08:00
|
|
|
#endif
|
1993-08-05 07:10:43 +08:00
|
|
|
|
1993-07-08 00:40:30 +08:00
|
|
|
#ifdef elf_tc_final_processing
|
1993-09-04 06:36:26 +08:00
|
|
|
elf_tc_final_processing ();
|
1993-07-08 00:40:30 +08:00
|
|
|
#endif
|
|
|
|
|
|
|
|
/* Finally, we must make any target-specific sections. */
|
|
|
|
|
|
|
|
#ifdef elf_tc_make_sections
|
1993-09-04 06:36:26 +08:00
|
|
|
elf_tc_make_sections (stdoutput);
|
1993-07-08 00:40:30 +08:00
|
|
|
#endif
|
1993-11-24 15:42:03 +08:00
|
|
|
|
|
|
|
#ifdef ECOFF_DEBUGGING
|
|
|
|
/* Generate the ECOFF debugging information. */
|
|
|
|
{
|
|
|
|
const struct ecoff_debug_swap *debug_swap;
|
|
|
|
struct ecoff_debug_info debug;
|
|
|
|
char *buf;
|
|
|
|
asection *sec;
|
|
|
|
|
|
|
|
debug_swap
|
|
|
|
= get_elf_backend_data (stdoutput)->elf_backend_ecoff_debug_swap;
|
|
|
|
know (debug_swap != (const struct ecoff_debug_swap *) NULL);
|
|
|
|
ecoff_build_debug (&debug.symbolic_header, &buf, debug_swap);
|
|
|
|
|
|
|
|
/* Set up the pointers in debug. */
|
|
|
|
#define SET(ptr, offset, type) \
|
|
|
|
debug.ptr = (type) (buf + debug.symbolic_header.offset)
|
|
|
|
|
|
|
|
SET (line, cbLineOffset, unsigned char *);
|
|
|
|
SET (external_dnr, cbDnOffset, PTR);
|
|
|
|
SET (external_pdr, cbPdOffset, PTR);
|
|
|
|
SET (external_sym, cbSymOffset, PTR);
|
|
|
|
SET (external_opt, cbOptOffset, PTR);
|
|
|
|
SET (external_aux, cbAuxOffset, union aux_ext *);
|
|
|
|
SET (ss, cbSsOffset, char *);
|
|
|
|
SET (external_fdr, cbFdOffset, PTR);
|
|
|
|
SET (external_rfd, cbRfdOffset, PTR);
|
|
|
|
/* ssext and external_ext are set up just below. */
|
|
|
|
|
|
|
|
#undef SET
|
|
|
|
|
|
|
|
/* Set up the external symbols. */
|
|
|
|
debug.ssext = debug.ssext_end = NULL;
|
|
|
|
debug.external_ext = debug.external_ext_end = NULL;
|
|
|
|
if (! bfd_ecoff_debug_externals (stdoutput, &debug, debug_swap, true,
|
|
|
|
elf_get_extr, elf_set_index))
|
|
|
|
as_fatal ("Failed to set up debugging information: %s",
|
|
|
|
bfd_errmsg (bfd_error));
|
|
|
|
|
|
|
|
sec = bfd_get_section_by_name (stdoutput, ".mdebug");
|
|
|
|
assert (sec != NULL);
|
|
|
|
|
|
|
|
know (stdoutput->output_has_begun == false);
|
|
|
|
|
|
|
|
/* We set the size of the section, call bfd_set_section_contents
|
|
|
|
to force the ELF backend to allocate a file position, and then
|
|
|
|
write out the data. FIXME: Is this really the best way to do
|
|
|
|
this? */
|
|
|
|
sec->_raw_size = bfd_ecoff_debug_size (stdoutput, &debug, debug_swap);
|
|
|
|
|
|
|
|
if (! bfd_set_section_contents (stdoutput, sec, (PTR) NULL,
|
|
|
|
(file_ptr) 0, (bfd_size_type) 0))
|
|
|
|
as_fatal ("Can't start writing .mdebug section: %s",
|
|
|
|
bfd_errmsg (bfd_error));
|
|
|
|
|
|
|
|
know (stdoutput->output_has_begun == true);
|
|
|
|
know (sec->filepos != 0);
|
|
|
|
|
|
|
|
if (! bfd_ecoff_write_debug (stdoutput, &debug, debug_swap,
|
|
|
|
sec->filepos))
|
|
|
|
as_fatal ("Could not write .mdebug section: %s",
|
|
|
|
bfd_errmsg (bfd_error));
|
|
|
|
}
|
|
|
|
#endif /* ECOFF_DEBUGGING */
|
1993-05-28 03:42:23 +08:00
|
|
|
}
|