aboutsummaryrefslogtreecommitdiff
path: root/bfd/nlm32-alpha.c
diff options
context:
space:
mode:
Diffstat (limited to 'bfd/nlm32-alpha.c')
-rw-r--r--bfd/nlm32-alpha.c154
1 files changed, 106 insertions, 48 deletions
diff --git a/bfd/nlm32-alpha.c b/bfd/nlm32-alpha.c
index 358bc40..17d3a1a 100644
--- a/bfd/nlm32-alpha.c
+++ b/bfd/nlm32-alpha.c
@@ -96,6 +96,7 @@ nlm_alpha_write_prefix (abfd)
bfd_error = system_call_error;
return false;
}
+ return true;
}
/* How to process the various reloc types. */
@@ -469,13 +470,13 @@ nlm_alpha_read_reloc (abfd, sym, secp, rel)
rel->sym_ptr_ptr = bfd_abs_section.symbol_ptr_ptr;
rel->addend = 0;
}
- else if (r_symndx == RELOC_SECTION_TEXT)
+ else if (r_symndx == ALPHA_RELOC_SECTION_TEXT)
{
rel->sym_ptr_ptr = code_sec->symbol_ptr_ptr;
BFD_ASSERT (bfd_get_section_vma (abfd, code_sec) == 0);
rel->addend = 0;
}
- else if (r_symndx == RELOC_SECTION_DATA)
+ else if (r_symndx == ALPHA_RELOC_SECTION_DATA)
{
rel->sym_ptr_ptr = data_sec->symbol_ptr_ptr;
rel->addend = - bfd_get_section_vma (abfd, data_sec);
@@ -573,14 +574,23 @@ nlm_alpha_read_reloc (abfd, sym, secp, rel)
break;
case ALPHA_R_NW_RELOC:
+ /* If this is SETGP, we set the addend to 0. Otherwise we set
+ the addend to the size of the .lita section (this is
+ r_symndx) plus 1. We have already set the address of the
+ reloc to r_vaddr. */
if (r_size == ALPHA_R_NW_RELOC_SETGP)
- gp_value = r_vaddr;
+ {
+ gp_value = r_vaddr;
+ rel->addend = 0;
+ }
else if (r_size == ALPHA_R_NW_RELOC_LITA)
- lita_address = r_vaddr;
+ {
+ lita_address = r_vaddr;
+ rel->addend = r_symndx + 1;
+ }
else
BFD_ASSERT (0);
rel->sym_ptr_ptr = bfd_abs_section.symbol_ptr_ptr;
- rel->addend = r_size;
break;
default:
@@ -628,6 +638,11 @@ nlm_alpha_read_import (abfd, sym)
}
sym -> symbol.the_bfd = abfd;
sym -> symbol.name = bfd_alloc (abfd, symlength + 1);
+ if (!sym -> symbol.name)
+ {
+ bfd_error = no_memory;
+ return false;
+ }
if (bfd_read ((PTR) sym -> symbol.name, symlength, 1, abfd)
!= symlength)
{
@@ -645,6 +660,11 @@ nlm_alpha_read_import (abfd, sym)
rcount = bfd_h_get_32 (abfd, temp);
nlm_relocs = ((struct nlm_relent *)
bfd_alloc (abfd, rcount * sizeof (struct nlm_relent)));
+ if (!nlm_relocs)
+ {
+ bfd_error = no_memory;
+ return false;
+ }
sym -> relocs = nlm_relocs;
sym -> rcnt = 0;
while (sym -> rcnt < rcount)
@@ -680,52 +700,72 @@ nlm_alpha_write_import (abfd, sec, rel)
sym = *rel->sym_ptr_ptr;
/* Get values for the relocation fields. */
- r_vaddr = bfd_get_section_vma (abfd, sec) + rel->address;
- if (bfd_get_section (sym) == &bfd_und_section)
- {
- r_extern = 1;
- r_symndx = 0;
- }
- else
+ r_type = rel->howto->type;
+ if (r_type != ALPHA_R_NW_RELOC)
{
- r_extern = 0;
- if (bfd_get_section_flags (abfd, bfd_get_section (sym)) & SEC_CODE)
- r_symndx = RELOC_SECTION_TEXT;
+ r_vaddr = bfd_get_section_vma (abfd, sec) + rel->address;
+ if ((sec->flags & SEC_CODE) == 0)
+ r_vaddr += bfd_section_size (abfd,
+ bfd_get_section_by_name (abfd,
+ NLM_CODE_NAME));
+ if (bfd_get_section (sym) == &bfd_und_section)
+ {
+ r_extern = 1;
+ r_symndx = 0;
+ }
else
- r_symndx = RELOC_SECTION_DATA;
- }
- r_type = rel->howto->type;
- r_offset = 0;
- r_size = 0;
+ {
+ r_extern = 0;
+ if (bfd_get_section_flags (abfd, bfd_get_section (sym)) & SEC_CODE)
+ r_symndx = ALPHA_RELOC_SECTION_TEXT;
+ else
+ r_symndx = ALPHA_RELOC_SECTION_DATA;
+ }
+ r_offset = 0;
+ r_size = 0;
- switch (r_type)
+ switch (r_type)
+ {
+ case ALPHA_R_LITUSE:
+ case ALPHA_R_GPDISP:
+ r_symndx = rel->addend;
+ break;
+
+ case ALPHA_R_OP_STORE:
+ r_size = rel->addend & 0xff;
+ r_offset = (rel->addend >> 8) & 0xff;
+ break;
+
+ case ALPHA_R_OP_PUSH:
+ case ALPHA_R_OP_PSUB:
+ case ALPHA_R_OP_PRSHIFT:
+ r_vaddr = rel->addend;
+ break;
+
+ case ALPHA_R_IGNORE:
+ r_vaddr = rel->address;
+ break;
+
+ default:
+ break;
+ }
+ }
+ else
{
- case ALPHA_R_LITUSE:
- case ALPHA_R_GPDISP:
- r_symndx = rel->addend;
- break;
-
- case ALPHA_R_OP_STORE:
- r_size = rel->addend & 0xff;
- r_offset = (rel->addend >> 8) & 0xff;
- break;
-
- case ALPHA_R_OP_PUSH:
- case ALPHA_R_OP_PSUB:
- case ALPHA_R_OP_PRSHIFT:
- r_vaddr = rel->addend;
- break;
-
- case ALPHA_R_IGNORE:
+ /* r_type == ALPHA_R_NW_RELOC */
r_vaddr = rel->address;
- break;
-
- case ALPHA_R_NW_RELOC:
- r_size = rel->addend;
- break;
-
- default:
- break;
+ if (rel->addend == 0)
+ {
+ r_symndx = 0;
+ r_size = ALPHA_R_NW_RELOC_SETGP;
+ }
+ else
+ {
+ r_symndx = rel->addend - 1;
+ r_size = ALPHA_R_NW_RELOC_LITA;
+ }
+ r_extern = 0;
+ r_offset = 0;
}
/* Swap out the relocation fields. */
@@ -800,13 +840,14 @@ nlm_alpha_get_public_offset (abfd, sym)
static boolean
nlm_alpha_write_external (abfd, count, sym, relocs)
bfd *abfd;
- bfd_vma count;
+ bfd_size_type count;
asymbol *sym;
struct reloc_and_sec *relocs;
{
int i;
bfd_byte len;
unsigned char temp[NLM_TARGET_LONG_SIZE];
+ arelent r;
len = strlen (sym->name);
if ((bfd_write (&len, sizeof (bfd_byte), 1, abfd) != sizeof(bfd_byte))
@@ -816,13 +857,28 @@ nlm_alpha_write_external (abfd, count, sym, relocs)
return false;
}
- bfd_put_32 (abfd, count, temp);
+ bfd_put_32 (abfd, count + 2, temp);
if (bfd_write (temp, sizeof (temp), 1, abfd) != sizeof (temp))
{
bfd_error = system_call_error;
return false;
}
+ /* The first two relocs for each external symbol are the .lita
+ address and the GP value. */
+ r.sym_ptr_ptr = bfd_abs_section.symbol_ptr_ptr;
+ r.howto = &nlm32_alpha_nw_howto;
+
+ r.address = nlm_alpha_backend_data (abfd)->lita_address;
+ r.addend = nlm_alpha_backend_data (abfd)->lita_size + 1;
+ if (nlm_alpha_write_import (abfd, (asection *) NULL, &r) == false)
+ return false;
+
+ r.address = nlm_alpha_backend_data (abfd)->gp;
+ r.addend = 0;
+ if (nlm_alpha_write_import (abfd, (asection *) NULL, &r) == false)
+ return false;
+
for (i = 0; i < count; i++)
{
if (nlm_alpha_write_import (abfd, relocs[i].sec,
@@ -842,6 +898,7 @@ static const struct nlm_backend_data nlm32_alpha_backend =
sizeof (struct nlm32_alpha_external_prefix_header),
bfd_arch_alpha,
0,
+ true, /* no uninitialized data permitted by Alpha NetWare. */
nlm_alpha_backend_object_p,
nlm_alpha_write_prefix,
nlm_alpha_read_reloc,
@@ -853,6 +910,7 @@ static const struct nlm_backend_data nlm32_alpha_backend =
nlm_swap_fixed_header_in,
nlm_swap_fixed_header_out,
nlm_alpha_write_external,
+ 0, /* write_export */
};
#define TARGET_LITTLE_NAME "nlm32-alpha"