summaryrefslogtreecommitdiff
path: root/gnu/usr.bin/binutils/bfd/evax-etir.c
diff options
context:
space:
mode:
Diffstat (limited to 'gnu/usr.bin/binutils/bfd/evax-etir.c')
-rw-r--r--gnu/usr.bin/binutils/bfd/evax-etir.c1553
1 files changed, 1553 insertions, 0 deletions
diff --git a/gnu/usr.bin/binutils/bfd/evax-etir.c b/gnu/usr.bin/binutils/bfd/evax-etir.c
new file mode 100644
index 00000000000..cb186f11e96
--- /dev/null
+++ b/gnu/usr.bin/binutils/bfd/evax-etir.c
@@ -0,0 +1,1553 @@
+/* evax-etir.c -- BFD back-end for ALPHA EVAX (openVMS/AXP) files.
+ Copyright 1996 Free Software Foundation, Inc.
+ ETIR record handling functions
+
+ go and read the openVMS linker manual (esp. appendix B)
+ if you don't know what's going on here :-)
+
+ Written by Klaus Kämpf (kkaempf@progis.de)
+ of proGIS Softwareentwicklung, Aachen, Germany
+
+This program 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 of the License, or
+(at your option) any later version.
+
+This program 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 this program; if not, write to the Free Software
+Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. */
+
+
+/* The following type abbreviations are used:
+
+ cs counted string (ascii string with length byte)
+ by byte (1 byte)
+ sh short (2 byte, 16 bit)
+ lw longword (4 byte, 32 bit)
+ qw quadword (8 byte, 64 bit)
+ da data stream */
+
+#include <stdio.h>
+#include <ctype.h>
+
+#include "bfd.h"
+#include "sysdep.h"
+#include "bfdlink.h"
+#include "libbfd.h"
+
+#include "evax.h"
+
+#if 0
+static void location_save
+ PARAMS ((bfd *abfd, unsigned long index, unsigned long loc, int section));
+static unsigned long location_restore
+ PARAMS ((bfd *abfd, unsigned long index, int *section));
+#endif /* 0 */
+
+static void image_set_ptr PARAMS ((bfd *abfd, int psect, uquad offset));
+static void image_inc_ptr PARAMS ((bfd *abfd, uquad offset));
+static void image_dump PARAMS ((bfd *abfd, unsigned char *ptr, int size, int offset));
+static void image_write_b PARAMS ((bfd *abfd, unsigned int value));
+static void image_write_w PARAMS ((bfd *abfd, unsigned int value));
+static void image_write_l PARAMS ((bfd *abfd, unsigned long value));
+static void image_write_q PARAMS ((bfd *abfd, uquad value));
+
+/*-----------------------------------------------------------------------------*/
+
+#if 0
+
+/* Save location counter at index */
+
+static void
+location_save (abfd, index, loc, section)
+ bfd *abfd;
+ unsigned long index;
+ unsigned long loc;
+ int section;
+{
+ PRIV(location_stack)[index].value = loc;
+ PRIV(location_stack)[index].psect = section;
+
+ return;
+}
+
+/* Restore location counter from index */
+
+static unsigned long
+location_restore (abfd, index, section)
+ bfd *abfd;
+ unsigned long index;
+ int *section;
+{
+ if (section != NULL)
+ *section = PRIV(location_stack)[index].psect;
+ return PRIV(location_stack)[index].value;
+}
+
+#endif /* 0 */
+
+/* routines to fill sections contents during etir read */
+
+/* Initialize image buffer pointer to be filled */
+
+static void
+image_set_ptr (abfd, psect, offset)
+ bfd *abfd;
+ int psect;
+ uquad offset;
+{
+#if EVAX_DEBUG
+ evax_debug (4, "image_set_ptr(%d=%s, %d)\n",
+ psect, PRIV(sections)[psect]->name, offset);
+#endif
+
+ PRIV(image_ptr) = PRIV(sections)[psect]->contents + offset;
+ return;
+}
+
+
+/* Increment image buffer pointer by offset */
+
+static void
+image_inc_ptr (abfd, offset)
+ bfd *abfd;
+ uquad offset;
+{
+#if EVAX_DEBUG
+ evax_debug (4, "image_inc_ptr(%d)\n", offset);
+#endif
+
+ PRIV(image_ptr) += offset;
+
+ return;
+}
+
+
+/* Dump multiple bytes to section image */
+
+static void
+image_dump (abfd, ptr, size, offset)
+ bfd *abfd;
+ unsigned char *ptr;
+ int size;
+ int offset;
+{
+#if EVAX_DEBUG
+ evax_debug (6, "image_dump from (%p, %d) to (%p)\n", ptr, size, PRIV(image_ptr));
+ _bfd_hexdump (7, ptr, size, offset);
+#endif
+
+ while (size-- > 0)
+ *PRIV(image_ptr)++ = *ptr++;
+ return;
+}
+
+
+/* Write byte to section image */
+
+static void
+image_write_b (abfd, value)
+ bfd *abfd;
+ unsigned int value;
+{
+#if EVAX_DEBUG
+ evax_debug (6, "image_write_b(%02x)\n", (int)value);
+#endif
+
+ *PRIV(image_ptr)++ = (value & 0xff);
+ return;
+}
+
+
+/* Write 2-byte word to image */
+
+static void
+image_write_w (abfd, value)
+ bfd *abfd;
+ unsigned int value;
+{
+#if EVAX_DEBUG
+ evax_debug (6, "image_write_w(%04x)\n", (int)value);
+#endif
+
+ bfd_putl16 (value, PRIV(image_ptr));
+ PRIV(image_ptr) += 2;
+
+ return;
+}
+
+
+/* Write 4-byte long to image */
+
+static void
+image_write_l (abfd, value)
+ bfd *abfd;
+ unsigned long value;
+{
+#if EVAX_DEBUG
+ evax_debug (6, "image_write_l(%08lx)\n", value);
+#endif
+
+ bfd_putl32 (value, PRIV(image_ptr));
+ PRIV(image_ptr) += 4;
+
+ return;
+}
+
+
+/* Write 4-byte long to image */
+
+static void
+image_write_q (abfd, value)
+ bfd *abfd;
+ uquad value;
+{
+#if EVAX_DEBUG
+ evax_debug (6, "image_write_q(%016lx)\n", value);
+#endif
+
+ bfd_putl64 (value, PRIV(image_ptr));
+ PRIV(image_ptr) += 8;
+
+ return;
+}
+
+
+#define HIGHBIT(op) ((op & 0x80000000L) == 0x80000000L)
+
+/* etir_sta
+
+ evax stack commands
+
+ handle sta_xxx commands in etir section
+ ptr points to data area in record
+
+ see table B-8 of the openVMS linker manual */
+
+static boolean
+etir_sta (abfd, cmd, ptr)
+ bfd *abfd;
+ int cmd;
+ unsigned char *ptr;
+{
+
+ switch (cmd)
+ {
+ /* stack */
+
+ /* stack global
+ arg: cs symbol name
+
+ stack 32 bit value of symbol (high bits set to 0) */
+
+ case ETIR_S_C_STA_GBL:
+ {
+ char *name;
+ evax_symbol_entry *entry;
+
+ name = _bfd_evax_save_counted_string ((char *)ptr);
+ entry = (evax_symbol_entry *)
+ bfd_hash_lookup (PRIV(evax_symbol_table), name, false, false);
+ if (entry == (evax_symbol_entry *)NULL)
+ {
+#if EVAX_DEBUG
+ evax_debug (3, "ETIR_S_C_STA_GBL: no symbol \"%s\"\n", name);
+#endif
+ return false;
+ }
+ else
+ {
+ _bfd_evax_push (abfd, (uquad)(entry->symbol->value), -1);
+ }
+ }
+ break;
+
+ /* stack longword
+ arg: lw value
+
+ stack 32 bit value, sign extend to 64 bit */
+
+ case ETIR_S_C_STA_LW:
+ _bfd_evax_push (abfd, (uquad)bfd_getl32 (ptr), -1);
+ break;
+
+ /* stack global
+ arg: qw value
+
+ stack 64 bit value of symbol */
+
+ case ETIR_S_C_STA_QW:
+ _bfd_evax_push (abfd, (uquad)bfd_getl64(ptr), -1);
+ break;
+
+ /* stack psect base plus quadword offset
+ arg: lw section index
+ qw signed quadword offset (low 32 bits)
+
+ stack qw argument and section index
+ (see ETIR_S_C_STO_OFF, ETIR_S_C_CTL_SETRB) */
+
+ case ETIR_S_C_STA_PQ:
+ {
+ uquad dummy;
+ int psect;
+
+ psect = bfd_getl32 (ptr);
+ if (psect >= PRIV(egsd_sec_count))
+ {
+ (*_bfd_error_handler) ("Bad section index in ETIR_S_C_STA_PQ");
+ bfd_set_error (bfd_error_bad_value);
+ return false;
+ }
+ dummy = bfd_getl64 (ptr+4);
+ _bfd_evax_push (abfd, dummy, psect);
+ }
+ break;
+
+ /* all not supported */
+
+ case ETIR_S_C_STA_LI:
+ case ETIR_S_C_STA_MOD:
+ case ETIR_S_C_STA_CKARG:
+
+ (*_bfd_error_handler) ("Unsupported STA cmd %d", cmd);
+ return false;
+ break;
+
+ default:
+ (*_bfd_error_handler) ("Reserved STA cmd %d", cmd);
+ return false;
+ break;
+ }
+ return true;
+}
+
+
+/*
+ etir_sto
+
+ evax store commands
+
+ handle sto_xxx commands in etir section
+ ptr points to data area in record
+
+ see table B-9 of the openVMS linker manual */
+
+static boolean
+etir_sto (abfd, cmd, ptr)
+ bfd *abfd;
+ int cmd;
+ unsigned char *ptr;
+{
+ uquad dummy;
+ int psect;
+
+ switch (cmd)
+ {
+
+ /* store byte: pop stack, write byte
+ arg: - */
+
+ case ETIR_S_C_STO_B:
+ dummy = _bfd_evax_pop (abfd, &psect);
+#if 0
+ if (is_share) /* FIXME */
+ (*_bfd_error_handler) ("ETIR_S_C_STO_B: byte fixups not supported");
+#endif
+ image_write_b (abfd, dummy & 0xff); /* FIXME: check top bits */
+ break;
+
+ /* store word: pop stack, write word
+ arg: - */
+
+ case ETIR_S_C_STO_W:
+ dummy = _bfd_evax_pop (abfd, &psect);
+#if 0
+ if (is_share) /* FIXME */
+ (*_bfd_error_handler) ("ETIR_S_C_STO_B: word fixups not supported");
+#endif
+ image_write_w (abfd, dummy & 0xffff); /* FIXME: check top bits */
+ break;
+
+ /* store longword: pop stack, write longword
+ arg: - */
+
+ case ETIR_S_C_STO_LW:
+ dummy = _bfd_evax_pop (abfd, &psect);
+ dummy += (PRIV(sections)[psect])->vma;
+ image_write_l (abfd, dummy & 0xffffffff);/* FIXME: check top bits */
+#if 0 /* FIXME */
+ if (is_rel)
+ evax_debug (3, "ETIR_S_C_STO_LW: Relocation !\n");
+ if (is_share)
+ evax_debug (3, "ETIR_S_C_STO_LW: Fix-up share !\n");
+#endif
+ break;
+
+ /* store quadword: pop stack, write quadword
+ arg: - */
+
+ case ETIR_S_C_STO_QW:
+ dummy = _bfd_evax_pop (abfd, &psect);
+ dummy += (PRIV(sections)[psect])->vma;
+ image_write_q(abfd, dummy); /* FIXME: check top bits */
+#if 0 /* FIXME */
+ if (is_rel)
+ evax_debug (3, "ETIR_S_C_STO_LW: Relocation !\n");
+ if (is_share)
+ evax_debug (3, "ETIR_S_C_STO_LW: Fix-up share !\n");
+#endif
+ break;
+
+ /* store immediate repeated: pop stack for repeat count
+ arg: lw byte count
+ da data */
+
+ case ETIR_S_C_STO_IMMR:
+ {
+ unsigned long size;
+
+ size = bfd_getl32 (ptr);
+ dummy = (unsigned long)_bfd_evax_pop (abfd, NULL);
+ while (dummy-- > 0L)
+ image_dump (abfd, ptr+4, size, 0);
+ }
+ break;
+
+ /* store global: write symbol value
+ arg: cs global symbol name */
+
+ case ETIR_S_C_STO_GBL:
+ {
+ evax_symbol_entry *entry;
+ char *name;
+
+ name = _bfd_evax_save_counted_string ((char *)ptr);
+ entry = (evax_symbol_entry *)bfd_hash_lookup (PRIV(evax_symbol_table), name, false, false);
+ if (entry == (evax_symbol_entry *)NULL)
+ {
+ (*_bfd_error_handler) ("ETIR_S_C_STO_GBL: no symbol \"%s\"",
+ name);
+ return false;
+ }
+ else
+ image_write_q (abfd, (uquad)(entry->symbol->value)); /* FIXME, reloc */
+ }
+ break;
+
+ /* store code address: write address of entry point
+ arg: cs global symbol name (procedure) */
+
+ case ETIR_S_C_STO_CA:
+ {
+ evax_symbol_entry *entry;
+ char *name;
+
+ name = _bfd_evax_save_counted_string ((char *)ptr);
+ entry = (evax_symbol_entry *) bfd_hash_lookup (PRIV(evax_symbol_table), name, false, false);
+ if (entry == (evax_symbol_entry *)NULL)
+ {
+ (*_bfd_error_handler) ("ETIR_S_C_STO_CA: no symbol \"%s\"",
+ name);
+ return false;
+ }
+ else
+ image_write_q (abfd, (uquad)(entry->symbol->value)); /* FIXME, reloc */
+ }
+ break;
+
+ /* not supported */
+
+ case ETIR_S_C_STO_RB:
+ case ETIR_S_C_STO_AB:
+ (*_bfd_error_handler) ("ETIR_S_C_STO_RB/AB: Not supported");
+ break;
+
+ /* store offset to psect: pop stack, add low 32 bits to base of psect
+ arg: - */
+
+ case ETIR_S_C_STO_OFF:
+ {
+ uquad q;
+ int psect;
+
+ q = _bfd_evax_pop (abfd, &psect);
+ q += (PRIV(sections)[psect])->vma;
+ image_write_q (abfd, q);
+ }
+ break;
+
+ /* store immediate
+ arg: lw count of bytes
+ da data */
+
+ case ETIR_S_C_STO_IMM:
+ {
+ int size;
+
+ size = bfd_getl32 (ptr);
+ image_dump (abfd, ptr+4, size, 0);
+ }
+ break;
+
+ /* this code is 'reserved to digital' according to the openVMS linker manual,
+ however it is generated by the DEC C compiler and defined in the include file.
+ FIXME, since the following is just a guess
+ store global longword: store 32bit value of symbol
+ arg: cs symbol name */
+
+ case ETIR_S_C_STO_GBL_LW:
+ {
+ evax_symbol_entry *entry;
+ char *name;
+
+ name = _bfd_evax_save_counted_string ((char *)ptr);
+ entry = (evax_symbol_entry *)bfd_hash_lookup (PRIV(evax_symbol_table), name, false, false);
+ if (entry == (evax_symbol_entry *)NULL)
+ {
+#if EVAX_DEBUG
+ evax_debug (3, "ETIR_S_C_STO_GBL_LW: no symbol \"%s\"\n", name);
+#endif
+ return false;
+ }
+ else
+ image_write_l (abfd, (unsigned long)(entry->symbol->value)); /* FIXME, reloc */
+ }
+ break;
+
+ /* not supported */
+
+ case ETIR_S_C_STO_LP_PSB:
+ (*_bfd_error_handler) ("ETIR_S_C_STO_LP_PSB: Not supported");
+ break;
+
+ /* */
+
+ case ETIR_S_C_STO_HINT_GBL:
+ (*_bfd_error_handler) ("ETIR_S_C_STO_HINT_GBL: not implemented");
+ break;
+
+ /* */
+
+ case ETIR_S_C_STO_HINT_PS:
+ (*_bfd_error_handler) ("ETIR_S_C_STO_HINT_PS: not implemented");
+ break;
+
+ default:
+ (*_bfd_error_handler) ("Reserved STO cmd %d", cmd);
+ break;
+ }
+
+ return true;
+}
+
+/* stack operator commands
+ all 32 bit signed arithmetic
+ all word just like a stack calculator
+ arguments are popped from stack, results are pushed on stack
+
+ see table B-10 of the openVMS linker manual */
+
+static boolean
+etir_opr (abfd, cmd, ptr)
+ bfd *abfd;
+ int cmd;
+ unsigned char *ptr;
+{
+ long op1, op2;
+
+ switch (cmd)
+ {
+ /* operation */
+
+ /* no-op */
+
+ case ETIR_S_C_OPR_NOP:
+ break;
+
+ /* add */
+
+ case ETIR_S_C_OPR_ADD:
+ op1 = (long)_bfd_evax_pop (abfd, NULL);
+ op2 = (long)_bfd_evax_pop (abfd, NULL);
+ _bfd_evax_push (abfd, (uquad)(op1 + op2), -1);
+ break;
+
+ /* subtract */
+
+ case ETIR_S_C_OPR_SUB:
+ op1 = (long)_bfd_evax_pop (abfd, NULL);
+ op2 = (long)_bfd_evax_pop (abfd, NULL);
+ _bfd_evax_push (abfd, (uquad)(op2 - op1), -1);
+ break;
+
+ /* multiply */
+
+ case ETIR_S_C_OPR_MUL:
+ op1 = (long)_bfd_evax_pop (abfd, NULL);
+ op2 = (long)_bfd_evax_pop (abfd, NULL);
+ _bfd_evax_push (abfd, (uquad)(op1 * op2), -1);
+ break;
+
+ /* divide */
+
+ case ETIR_S_C_OPR_DIV:
+ op1 = (long)_bfd_evax_pop (abfd, NULL);
+ op2 = (long)_bfd_evax_pop (abfd, NULL);
+ if (op2 == 0)
+ _bfd_evax_push (abfd, (uquad)0L, -1);
+ else
+ _bfd_evax_push (abfd, (uquad)(op2 / op1), -1);
+ break;
+
+ /* logical and */
+
+ case ETIR_S_C_OPR_AND:
+ op1 = (long)_bfd_evax_pop (abfd, NULL);
+ op2 = (long)_bfd_evax_pop (abfd, NULL);
+ _bfd_evax_push (abfd, (uquad)(op1 & op2), -1);
+ break;
+
+ /* logical inclusive or */
+
+ case ETIR_S_C_OPR_IOR:
+ op1 = (long)_bfd_evax_pop (abfd, NULL);
+ op2 = (long)_bfd_evax_pop (abfd, NULL);
+ _bfd_evax_push (abfd, (uquad)(op1 | op2), -1);
+ break;
+
+ /* logical exclusive or */
+
+ case ETIR_S_C_OPR_EOR:
+ op1 = (long)_bfd_evax_pop (abfd, NULL);
+ op2 = (long)_bfd_evax_pop (abfd, NULL);
+ _bfd_evax_push (abfd, (uquad)(op1 ^ op2), -1);
+ break;
+
+ /* negate */
+
+ case ETIR_S_C_OPR_NEG:
+ op1 = (long)_bfd_evax_pop (abfd, NULL);
+ _bfd_evax_push (abfd, (uquad)(-op1), -1);
+ break;
+
+ /* complement */
+
+ case ETIR_S_C_OPR_COM:
+ op1 = (long)_bfd_evax_pop (abfd, NULL);
+ _bfd_evax_push (abfd, (uquad)(op1 ^ -1L), -1);
+ break;
+
+ /* insert field */
+
+ case ETIR_S_C_OPR_INSV:
+ (void)_bfd_evax_pop (abfd, NULL);
+ (*_bfd_error_handler) ("ETIR_S_C_OPR_INSV: Not supported");
+ break;
+
+ /* arithmetic shift */
+
+ case ETIR_S_C_OPR_ASH:
+ op1 = (long)_bfd_evax_pop (abfd, NULL);
+ op2 = (long)_bfd_evax_pop (abfd, NULL);
+ if (op2 < 0) /* shift right */
+ op1 >>= -op2;
+ else /* shift left */
+ op1 <<= op2;
+ _bfd_evax_push (abfd, (uquad)op1, -1);
+ break;
+
+ /* unsigned shift */
+
+ case ETIR_S_C_OPR_USH:
+ (*_bfd_error_handler) ("ETIR_S_C_OPR_USH: Not supported");
+ break;
+
+ /* rotate */
+
+ case ETIR_S_C_OPR_ROT:
+ (*_bfd_error_handler) ("ETIR_S_C_OPR_ROT: Not supported");
+ break;
+
+ /* select */
+
+ case ETIR_S_C_OPR_SEL:
+ if ((long)_bfd_evax_pop (abfd, NULL) & 0x01L)
+ (void)_bfd_evax_pop (abfd, NULL);
+ else
+ {
+ op1 = (long)_bfd_evax_pop (abfd, NULL);
+ (void)_bfd_evax_pop (abfd, NULL);
+ _bfd_evax_push (abfd, (uquad)op1, -1);
+ }
+ break;
+
+ /* redefine symbol to current location */
+
+ case ETIR_S_C_OPR_REDEF:
+ (*_bfd_error_handler) ("ETIR_S_C_OPR_REDEF: Not supported");
+ break;
+
+ /* define a literal */
+
+ case ETIR_S_C_OPR_DFLIT:
+ (*_bfd_error_handler) ("ETIR_S_C_OPR_DFLIT: Not supported");
+ break;
+
+ default:
+ (*_bfd_error_handler) ("Reserved OPR cmd %d", cmd);
+ break;
+ }
+
+ return true;
+}
+
+
+/* control commands
+
+ see table B-11 of the openVMS linker manual */
+
+static boolean
+etir_ctl (abfd, cmd, ptr)
+ bfd *abfd;
+ int cmd;
+ unsigned char *ptr;
+{
+ uquad dummy;
+ int psect;
+
+ switch (cmd)
+ {
+ /* set relocation base: pop stack, set image location counter
+ arg: - */
+
+ case ETIR_S_C_CTL_SETRB:
+ dummy = _bfd_evax_pop (abfd, &psect);
+ image_set_ptr (abfd, psect, dummy);
+ break;
+
+ /* augment relocation base: increment image location counter by offset
+ arg: lw offset value */
+
+ case ETIR_S_C_CTL_AUGRB:
+ dummy = bfd_getl32 (ptr);
+ image_inc_ptr (abfd, dummy);
+ break;
+
+ /* define location: pop index, save location counter under index
+ arg: - */
+
+ case ETIR_S_C_CTL_DFLOC:
+ dummy = _bfd_evax_pop (abfd, NULL);
+ /* FIXME */
+ break;
+
+ /* set location: pop index, restore location counter from index
+ arg: - */
+
+ case ETIR_S_C_CTL_STLOC:
+ dummy = _bfd_evax_pop (abfd, &psect);
+ /* FIXME */
+ break;
+
+ /* stack defined location: pop index, push location counter from index
+ arg: - */
+
+ case ETIR_S_C_CTL_STKDL:
+ dummy = _bfd_evax_pop (abfd, &psect);
+ /* FIXME */
+ break;
+
+ default:
+ (*_bfd_error_handler) ("Reserved CTL cmd %d", cmd);
+ break;
+ }
+ return true;
+}
+
+
+/* store conditional commands
+
+ see table B-12 and B-13 of the openVMS linker manual */
+
+static boolean
+etir_stc (abfd, cmd, ptr)
+ bfd *abfd;
+ int cmd;
+ unsigned char *ptr;
+{
+
+ switch (cmd)
+ {
+ /* 200 Store-conditional Linkage Pair
+ arg: */
+
+ case ETIR_S_C_STC_LP:
+ (*_bfd_error_handler) ("ETIR_S_C_STC_LP: not supported");
+ break;
+
+ /* 201 Store-conditional Linkage Pair with Procedure Signature
+ arg: lw linkage index
+ cs procedure name
+ by signature length
+ da signature */
+
+ case ETIR_S_C_STC_LP_PSB:
+ image_inc_ptr (abfd, 16); /* skip entry,procval */
+ break;
+
+ /* 202 Store-conditional Address at global address
+ arg: lw linkage index
+ cs global name */
+
+ case ETIR_S_C_STC_GBL:
+ (*_bfd_error_handler) ("ETIR_S_C_STC_GBL: not supported");
+ break;
+
+ /* 203 Store-conditional Code Address at global address
+ arg: lw linkage index
+ cs procedure name */
+
+ case ETIR_S_C_STC_GCA:
+ (*_bfd_error_handler) ("ETIR_S_C_STC_GCA: not supported");
+ break;
+
+ /* 204 Store-conditional Address at psect + offset
+ arg: lw linkage index
+ lw psect index
+ qw offset */
+
+ case ETIR_S_C_STC_PS:
+ (*_bfd_error_handler) ("ETIR_S_C_STC_PS: not supported");
+ break;
+
+ /* 205 Store-conditional NOP at address of global
+ arg: */
+
+ case ETIR_S_C_STC_NOP_GBL:
+
+ /* 206 Store-conditional NOP at pect + offset
+ arg: */
+
+ case ETIR_S_C_STC_NOP_PS:
+
+ /* 207 Store-conditional BSR at global address
+ arg: */
+
+ case ETIR_S_C_STC_BSR_GBL:
+
+ /* 208 Store-conditional BSR at pect + offset
+ arg: */
+
+ case ETIR_S_C_STC_BSR_PS:
+
+ /* 209 Store-conditional LDA at global address
+ arg: */
+
+ case ETIR_S_C_STC_LDA_GBL:
+
+ /* 210 Store-conditional LDA at psect + offset
+ arg: */
+
+ case ETIR_S_C_STC_LDA_PS:
+
+ /* 211 Store-conditional BSR or Hint at global address
+ arg: */
+
+ case ETIR_S_C_STC_BOH_GBL:
+
+ /* 212 Store-conditional BSR or Hint at pect + offset
+ arg: */
+
+ case ETIR_S_C_STC_BOH_PS:
+
+ /* 213 Store-conditional NOP,BSR or HINT at global address
+ arg: */
+
+ case ETIR_S_C_STC_NBH_GBL:
+
+ /* 214 Store-conditional NOP,BSR or HINT at psect + offset
+ arg: */
+
+ case ETIR_S_C_STC_NBH_PS:
+/* FIXME (*_bfd_error_handler) ("ETIR_S_C_STC_xx: (%d) not supported", cmd); */
+ break;
+
+ default:
+#if EVAX_DEBUG
+ evax_debug (3, "Reserved STC cmd %d", cmd);
+#endif
+ break;
+ }
+ return true;
+}
+
+
+/* handle command from ETIR section */
+
+static boolean
+tir_cmd (abfd, cmd, ptr)
+ bfd *abfd;
+ int cmd;
+ unsigned char *ptr;
+{
+ static struct {
+ int mincod;
+ int maxcod;
+ boolean (*explain) PARAMS((bfd *, int, unsigned char *));
+ } tir_table[] = {
+ { ETIR_S_C_MINSTACOD, ETIR_S_C_MAXSTACOD, etir_sta },
+ { ETIR_S_C_MINSTOCOD, ETIR_S_C_MAXSTOCOD, etir_sto },
+ { ETIR_S_C_MINOPRCOD, ETIR_S_C_MAXOPRCOD, etir_opr },
+ { ETIR_S_C_MINCTLCOD, ETIR_S_C_MAXCTLCOD, etir_ctl },
+ { ETIR_S_C_MINSTCCOD, ETIR_S_C_MAXSTCCOD, etir_stc },
+ { -1, -1, NULL }
+ };
+
+ int i = 0;
+ boolean res = true;
+
+ while (tir_table[i].mincod >= 0)
+ {
+ if ( (tir_table[i].mincod <= cmd)
+ && (cmd <= tir_table[i].maxcod))
+ {
+ res = tir_table[i].explain (abfd, cmd, ptr);
+ break;
+ }
+ i++;
+ }
+
+ return res;
+}
+
+
+/* Text Information and Relocation Records (OBJ$C_TIR)
+ handle etir record */
+
+static boolean
+analyze_etir (abfd, ptr, length)
+ bfd *abfd;
+ unsigned char *ptr;
+ unsigned int length;
+{
+ int cmd;
+ unsigned char *maxptr;
+ boolean res = true;
+
+ maxptr = ptr + length;
+
+ while (ptr < maxptr)
+ {
+ cmd = bfd_getl16 (ptr);
+ length = bfd_getl16 (ptr + 2);
+ res = tir_cmd (abfd, cmd, ptr+4);
+ if (!res)
+ break;
+ ptr += length;
+ }
+ return res;
+}
+
+
+/* process ETIR record
+
+ return 0 on success, -1 on error */
+
+int
+_bfd_evax_slurp_etir (abfd)
+ bfd *abfd;
+{
+
+#if EVAX_DEBUG
+ evax_debug (2, "ETIR\n");
+#endif
+
+ PRIV(evax_rec) += 4; /* skip type, size */
+ PRIV(rec_size) -= 4;
+ if (analyze_etir (abfd, PRIV(evax_rec), PRIV(rec_size)))
+ return 0;
+
+ return -1;
+}
+
+
+/* process EDBG record
+ return 0 on success, -1 on error
+
+ not implemented yet */
+
+int
+_bfd_evax_slurp_edbg (abfd)
+ bfd *abfd;
+{
+#if EVAX_DEBUG
+ evax_debug (2, "EDBG\n");
+#endif
+
+ abfd->flags |= (HAS_DEBUG | HAS_LINENO);
+ return 0;
+}
+
+
+/* process ETBT record
+ return 0 on success, -1 on error
+
+ not implemented yet */
+
+int
+_bfd_evax_slurp_etbt (abfd)
+ bfd *abfd;
+{
+#if EVAX_DEBUG
+ evax_debug (2, "ETBT\n");
+#endif
+
+ return 0;
+}
+
+/*----------------------------------------------------------------------*/
+/* */
+/* WRITE ETIR SECTION */
+/* */
+/* this is still under construction and therefore not documented */
+/* */
+/*----------------------------------------------------------------------*/
+
+static void start_etir_record PARAMS ((bfd *abfd, int index, uquad offset, boolean justoffset));
+static void sto_imm PARAMS ((bfd *abfd, evax_section *sptr, bfd_vma vaddr, int index));
+static void end_etir_record PARAMS ((bfd *abfd));
+
+static void
+sto_imm (abfd, sptr, vaddr, index)
+ bfd *abfd;
+ evax_section *sptr;
+ bfd_vma vaddr;
+ int index;
+{
+ int size;
+ int ssize;
+ unsigned char *cptr;
+
+#if EVAX_DEBUG
+ evax_debug (8, "sto_imm %d bytes\n", sptr->size);
+ _bfd_hexdump (9, sptr->contents, (int)sptr->size, (int)vaddr);
+#endif
+
+ ssize = sptr->size;
+ cptr = sptr->contents;
+
+ while (ssize > 0)
+ {
+
+ size = ssize; /* try all the rest */
+
+ if (_bfd_evax_output_check (abfd, size) < 0)
+ { /* doesn't fit, split ! */
+ end_etir_record (abfd);
+ start_etir_record (abfd, index, vaddr, false);
+ size = _bfd_evax_output_check (abfd, 0); /* get max size */
+ if (size > ssize) /* more than what's left ? */
+ size = ssize;
+ }
+
+ _bfd_evax_output_begin (abfd, ETIR_S_C_STO_IMM, -1);
+ _bfd_evax_output_long (abfd, (unsigned long)(size));
+ _bfd_evax_output_dump (abfd, cptr, size);
+ _bfd_evax_output_flush (abfd);
+
+#if EVAX_DEBUG
+ evax_debug (10, "dumped %d bytes\n", size);
+ _bfd_hexdump (10, cptr, (int)size, (int)vaddr);
+#endif
+
+ vaddr += size;
+ ssize -= size;
+ cptr += size;
+ }
+
+ return;
+}
+
+/*-------------------------------------------------------------------*/
+
+/* start ETIR record for section #index at virtual addr offset. */
+
+static void
+start_etir_record (abfd, index, offset, justoffset)
+ bfd *abfd;
+ int index;
+ uquad offset;
+ boolean justoffset;
+{
+ if (!justoffset)
+ {
+ _bfd_evax_output_begin (abfd, EOBJ_S_C_ETIR, -1); /* one ETIR per section */
+ _bfd_evax_output_push (abfd);
+ }
+
+ _bfd_evax_output_begin (abfd, ETIR_S_C_STA_PQ, -1); /* push start offset */
+ _bfd_evax_output_long (abfd, (unsigned long)index);
+ _bfd_evax_output_quad (abfd, (uquad)offset);
+ _bfd_evax_output_flush (abfd);
+
+ _bfd_evax_output_begin (abfd, ETIR_S_C_CTL_SETRB, -1); /* start = pop() */
+ _bfd_evax_output_flush (abfd);
+
+ return;
+}
+
+
+/* end etir record */
+static void
+end_etir_record (abfd)
+ bfd *abfd;
+{
+ _bfd_evax_output_pop (abfd);
+ _bfd_evax_output_end (abfd);
+}
+
+/* write section contents for bfd abfd */
+
+int
+_bfd_evax_write_etir (abfd)
+ bfd *abfd;
+{
+ asection *section;
+ evax_section *sptr;
+ int nextoffset;
+ char uname[200];
+ char *nptr, *uptr;
+
+#if EVAX_DEBUG
+ evax_debug (2, "evax_write_etir(%p)\n", abfd);
+#endif
+
+ _bfd_evax_output_alignment (abfd, 4);
+
+ nextoffset = 0;
+ PRIV(evax_linkage_index) = 1;
+
+ /* dump all other sections */
+
+ section = abfd->sections;
+
+ while (section != NULL)
+ {
+
+#if EVAX_DEBUG
+ evax_debug (4, "writing %d. section '%s' (%d bytes)\n", section->index, section->name, (int)(section->_raw_size));
+#endif
+
+ if (section->flags & SEC_RELOC)
+ {
+ int i;
+
+ if ((i = section->reloc_count) <= 0)
+ {
+ (*_bfd_error_handler) ("SEC_RELOC with no relocs in section %s",
+ section->name);
+ }
+#if EVAX_DEBUG
+ else
+ {
+ arelent **rptr;
+ evax_debug (4, "%d relocations:\n", i);
+ rptr = section->orelocation;
+ while (i-- > 0)
+ {
+ evax_debug (4, "sym %s in sec %s, value %08lx, addr %08lx, off %08lx, len %d: %s\n",
+ (*(*rptr)->sym_ptr_ptr)->name,
+ (*(*rptr)->sym_ptr_ptr)->section->name,
+ (long)(*(*rptr)->sym_ptr_ptr)->value,
+ (*rptr)->address, (*rptr)->addend,
+ bfd_get_reloc_size((*rptr)->howto),
+ (*rptr)->howto->name);
+ rptr++;
+ }
+ }
+#endif
+ }
+
+ if (section->flags & SEC_HAS_CONTENTS)
+ {
+ bfd_vma vaddr; /* virtual addr in section */
+
+ sptr = _bfd_get_evax_section (abfd, section->index);
+ if (sptr == NULL)
+ {
+ bfd_set_error (bfd_error_no_contents);
+ return -1;
+ }
+
+ vaddr = (bfd_vma)(sptr->offset);
+
+ start_etir_record (abfd, section->index, (uquad) sptr->offset,
+ false);
+
+ while (sptr != NULL) /* one STA_PQ, CTL_SETRB per evax_section */
+ {
+
+ if (section->flags & SEC_RELOC) /* check for relocs */
+ {
+ arelent **rptr = section->orelocation;
+ int i = section->reloc_count;
+ for (;;)
+ {
+ bfd_size_type addr = (*rptr)->address;
+ int len = bfd_get_reloc_size ((*rptr)->howto);
+ if (sptr->offset < addr) /* sptr starts before reloc */
+ {
+ int before = addr - sptr->offset;
+ if (sptr->size <= before) /* complete before */
+ {
+ sto_imm (abfd, sptr, vaddr, section->index);
+ vaddr += sptr->size;
+ break;
+ }
+ else /* partly before */
+ {
+ int after = sptr->size - before;
+ sptr->size = before;
+ sto_imm (abfd, sptr, vaddr, section->index);
+ vaddr += sptr->size;
+ sptr->contents += before;
+ sptr->offset += before;
+ sptr->size = after;
+ }
+ }
+ else if (sptr->offset == addr) /* sptr starts at reloc */
+ {
+ asymbol *sym = *(*rptr)->sym_ptr_ptr;
+ asection *sec = sym->section;
+
+ switch ((*rptr)->howto->type)
+ {
+ case ALPHA_R_IGNORE:
+ break;
+
+ case ALPHA_R_REFLONG:
+ {
+ if (bfd_is_und_section (sym->section))
+ {
+ if (_bfd_evax_output_check (abfd,
+ strlen((char *)sym->name))
+ < 0)
+ {
+ end_etir_record (abfd);
+ start_etir_record (abfd,
+ section->index,
+ vaddr, false);
+ }
+ _bfd_evax_output_begin (abfd,
+ ETIR_S_C_STO_GBL_LW,
+ -1);
+ uptr = uname;
+ nptr = (char *)sym->name;
+ while (*nptr)
+ {
+ if (islower (*nptr))
+ *uptr = toupper (*nptr);
+ else
+ *uptr = *nptr;
+ nptr++;
+ uptr++;
+ }
+ *uptr = 0;
+ _bfd_evax_output_counted (abfd, uname);
+ _bfd_evax_output_flush (abfd);
+ }
+ else if (bfd_is_abs_section (sym->section))
+ {
+ if (_bfd_evax_output_check (abfd, 16) < 0)
+ {
+ end_etir_record (abfd);
+ start_etir_record (abfd,
+ section->index,
+ vaddr, false);
+ }
+ _bfd_evax_output_begin (abfd,
+ ETIR_S_C_STA_LW,
+ -1);
+ _bfd_evax_output_quad (abfd,
+ (uquad)sym->value);
+ _bfd_evax_output_flush (abfd);
+ _bfd_evax_output_begin (abfd,
+ ETIR_S_C_STO_LW,
+ -1);
+ _bfd_evax_output_flush (abfd);
+ }
+ else
+ {
+ if (_bfd_evax_output_check (abfd, 32) < 0)
+ {
+ end_etir_record (abfd);
+ start_etir_record (abfd,
+ section->index,
+ vaddr, false);
+ }
+ _bfd_evax_output_begin (abfd,
+ ETIR_S_C_STA_PQ,
+ -1);
+ _bfd_evax_output_long (abfd,
+ (unsigned long)(sec->index));
+ _bfd_evax_output_quad (abfd,
+ ((uquad)(*rptr)->addend
+ + (uquad)sym->value));
+ _bfd_evax_output_flush (abfd);
+ _bfd_evax_output_begin (abfd,
+ ETIR_S_C_STO_LW,
+ -1);
+ _bfd_evax_output_flush (abfd);
+ }
+ }
+ break;
+
+ case ALPHA_R_REFQUAD:
+ {
+ if (bfd_is_und_section (sym->section))
+ {
+ if (_bfd_evax_output_check (abfd,
+ strlen((char *)sym->name))
+ < 0)
+ {
+ end_etir_record (abfd);
+ start_etir_record (abfd,
+ section->index,
+ vaddr, false);
+ }
+ _bfd_evax_output_begin (abfd,
+ ETIR_S_C_STO_GBL,
+ -1);
+ uptr = uname;
+ nptr = (char *)sym->name;
+ while (*nptr)
+ {
+ if (islower (*nptr))
+ *uptr = toupper (*nptr);
+ else
+ *uptr = *nptr;
+ nptr++;
+ uptr++;
+ }
+ *uptr = 0;
+ _bfd_evax_output_counted (abfd, uname);
+ _bfd_evax_output_flush (abfd);
+ }
+ else if (bfd_is_abs_section (sym->section))
+ {
+ if (_bfd_evax_output_check (abfd, 16) < 0)
+ {
+ end_etir_record (abfd);
+ start_etir_record (abfd,
+ section->index,
+ vaddr, false);
+ }
+ _bfd_evax_output_begin (abfd,
+ ETIR_S_C_STA_QW,
+ -1);
+ _bfd_evax_output_quad (abfd,
+ (uquad)sym->value);
+ _bfd_evax_output_flush (abfd);
+ _bfd_evax_output_begin (abfd,
+ ETIR_S_C_STO_QW,
+ -1);
+ _bfd_evax_output_flush (abfd);
+ }
+ else
+ {
+ if (_bfd_evax_output_check (abfd, 32) < 0)
+ {
+ end_etir_record (abfd);
+ start_etir_record (abfd,
+ section->index,
+ vaddr, false);
+ }
+ _bfd_evax_output_begin (abfd,
+ ETIR_S_C_STA_PQ,
+ -1);
+ _bfd_evax_output_long (abfd,
+ (unsigned long)(sec->index));
+ _bfd_evax_output_quad (abfd,
+ ((uquad)(*rptr)->addend
+ + (uquad)sym->value));
+ _bfd_evax_output_flush (abfd);
+ _bfd_evax_output_begin (abfd,
+ ETIR_S_C_STO_OFF,
+ -1);
+ _bfd_evax_output_flush (abfd);
+ }
+ }
+ break;
+
+ case ALPHA_R_HINT:
+ {
+ int hint_size;
+
+ hint_size = sptr->size;
+ sptr->size = len;
+ sto_imm (abfd, sptr, vaddr, section->index);
+ sptr->size = hint_size;
+#if 0
+ evax_output_begin(abfd, ETIR_S_C_STO_HINT_GBL, -1);
+ evax_output_long(abfd, (unsigned long)(sec->index));
+ evax_output_quad(abfd, (uquad)addr);
+ uptr = uname;
+ nptr = (char *)(*(*rptr)->sym_ptr_ptr)->name;
+ while (*nptr)
+ {
+ if (islower (*nptr))
+ *uptr = toupper (*nptr);
+ else
+ *uptr = *nptr;
+ nptr++;
+ uptr++;
+ }
+ *uptr = 0;
+
+ evax_output_counted(abfd, uname);
+ evax_output_flush(abfd);
+#endif
+ }
+ break;
+#if 0
+ case ALPHA_R_BRADDR:
+ break;
+ case ALPHA_R_SREL16:
+ break;
+ case ALPHA_R_SREL32:
+ break;
+ case ALPHA_R_SREL64:
+ break;
+ case ALPHA_R_OP_PUSH:
+ break;
+ case ALPHA_R_OP_STORE:
+ break;
+ case ALPHA_R_OP_PSUB:
+ break;
+ case ALPHA_R_OP_PRSHIFT:
+ break;
+#endif
+ case ALPHA_R_LINKAGE:
+ {
+ if (_bfd_evax_output_check (abfd, 64) < 0)
+ {
+ end_etir_record (abfd);
+ start_etir_record (abfd, section->index,
+ vaddr, false);
+ }
+ _bfd_evax_output_begin (abfd,
+ ETIR_S_C_STC_LP_PSB,
+ -1);
+ _bfd_evax_output_long (abfd,
+ (unsigned long)PRIV(evax_linkage_index));
+ PRIV(evax_linkage_index) += 2;
+ uptr = uname;
+ nptr = (char *)(*(*rptr)->sym_ptr_ptr)->name;
+ while (*nptr)
+ {
+ if (islower (*nptr))
+ *uptr = toupper (*nptr);
+ else
+ *uptr = *nptr;
+ nptr++;
+ uptr++;
+ }
+ *uptr = 0;
+ _bfd_evax_output_counted (abfd, uname);
+ _bfd_evax_output_byte (abfd, 0);
+ _bfd_evax_output_flush (abfd);
+ }
+ break;
+
+ default:
+ (*_bfd_error_handler) ("Unhandled relocation %s",
+ (*rptr)->howto->name);
+ break;
+ }
+
+ vaddr += len;
+
+ if (len == sptr->size)
+ {
+ break;
+ }
+ else
+ {
+ sptr->contents += len;
+ sptr->offset += len;
+ sptr->size -= len;
+ i--;
+ rptr++;
+ }
+ }
+ else /* sptr starts after reloc */
+ {
+ i--; /* check next reloc */
+ rptr++;
+ }
+
+ if (i==0) /* all reloc checked */
+ {
+ if (sptr->size > 0)
+ {
+ sto_imm (abfd, sptr, vaddr, section->index); /* dump rest */
+ vaddr += sptr->size;
+ }
+ break;
+ }
+ } /* for (;;) */
+ } /* if SEC_RELOC */
+ else /* no relocs, just dump */
+ {
+ sto_imm (abfd, sptr, vaddr, section->index);
+ vaddr += sptr->size;
+ }
+
+ sptr = sptr->next;
+
+ } /* while (sptr != 0) */
+
+ end_etir_record (abfd);
+
+ } /* has_contents */
+
+ section = section->next;
+ }
+
+ _bfd_evax_output_alignment(abfd, 2);
+ return 0;
+}
+
+
+/* write traceback data for bfd abfd */
+
+int
+_bfd_evax_write_etbt (abfd)
+ bfd *abfd;
+{
+#if EVAX_DEBUG
+ evax_debug (2, "evax_write_etbt(%p)\n", abfd);
+#endif
+
+ return 0;
+}
+
+
+/* write debug info for bfd abfd */
+
+int
+_bfd_evax_write_edbg (abfd)
+ bfd *abfd;
+{
+#if EVAX_DEBUG
+ evax_debug (2, "evax_write_edbg(%p)\n", abfd);
+#endif
+
+ return 0;
+}