8ddef5528d
It wasn't documented (or likely used) and wastes time. (try_open): If EXTEN is empty, don't try it. * ldctor.c, lderror.c, ldexp.c, ldfile.c, ldindr.c, ldlang.c, ldlex.l, ldmain.c, ldmisc.c, ldsym.c, ldver.c, ldwarn.c, ldwrite.c, lexsup.c, mri.c, relax.c: Replace DEFUN macro calls with normal function declarations. * Move *.em to emultempl/*.em. Move *.sh to emulparams/*.sh. Move *.sc-sh to scripttempl/*.sc. * {emultempl,emulparams,scripttempl}/README: New files. * sh.em, st2000.em, z8ksim.em, h8300hms.em, h8500hms.em: Files removed, replaced with generic.em. * h8300.sh, h8500.sh, h8300.sc, h8500.sc: Renamed from h8[35]00hms.s[ch]. Change their contents to omit the "hms". * *.em (*_get_script): Return script name instead of script contents. * ldlang.c (lang_process): Change caller. * ldlex.l, ldgram.y: Recognize -m option. Check for input files after *all* options in grammar. * ldmain.c (main): Check for -m options. Add default directory for -m. * mkscript.c: File removed. * genscripts.sh: Take two more parameters, tooldirlib and libdir, to add to the default LIB_PATH. Look for input files in the new subdirectories. Create the scripts in emulations subdirectory and don't filter them through mkscript. * configure.in: Make the emulations subdirectory. * Makefile.in: Account for all of the above changes. Remove unused .SUFFIXES. Get libgcc.a path with gcc -print-libgcc-file-name instead of $(libdir)/libgcc.a. Put CFLAGS last in the compilation rules. Add -I../bfd to INCLUDES so sysdep.h is found. * ldfile.c (try_open): If opening without the extension fails, try with the extension even if -v or -V was given. had_script is imported (from ldgram.y), not exported.
125 lines
3.3 KiB
C
125 lines
3.3 KiB
C
|
|
/* ldindr.c
|
|
Handle indirect symbols.
|
|
|
|
Copyright (C) 1991 Free Software Foundation, Inc.
|
|
Written by Steve Chamberlain steve@cygnus.com
|
|
|
|
This file is part of GLD, the Gnu Linker.
|
|
|
|
GLD 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.
|
|
|
|
GLD 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 GLD; see the file COPYING. If not, write to
|
|
the Free Software Foundation, 675 Mass Ave, Cambridge, MA 02139, USA.
|
|
|
|
*/
|
|
|
|
/*
|
|
An indirect symbol is where a global symbol in one file say's that
|
|
all refs like it should be turned into refs of the symbol pointed
|
|
at by the value of the indirect symbol.
|
|
|
|
BFD supplies symbols to be indirected with the BFD_INDIRECT bit
|
|
set. Whenever the linker gets one of these, it calls add_indirect
|
|
with the symbol. We look up the symbol which this one dereferneces,
|
|
and stop if they are the same. If they are not the same, copy all
|
|
the information from the current to the dereffed symbol. Set the
|
|
indirect bit in the flag. From now on the ldsym_get stuff will
|
|
perform the indirection for us, at no charge.
|
|
*/
|
|
|
|
|
|
|
|
#include "bfd.h"
|
|
#include "sysdep.h"
|
|
#include "ld.h"
|
|
#include "ldsym.h"
|
|
#include "ldmisc.h"
|
|
|
|
|
|
|
|
static asymbol **
|
|
move_it (a_list, b_list)
|
|
asymbol **a_list;
|
|
asymbol **b_list;
|
|
{
|
|
asymbol **head = a_list;
|
|
asymbol **cursor = head;
|
|
|
|
if (a_list == 0) return b_list;
|
|
if (b_list == 0) return a_list;
|
|
|
|
while (1) {
|
|
asymbol *ptr = cursor[0];
|
|
asymbol **next = (asymbol **)(ptr->udata);
|
|
if (next == 0) {
|
|
ptr->udata = (PTR) b_list;
|
|
return head;
|
|
}
|
|
cursor = next;
|
|
}
|
|
}
|
|
|
|
#if 0
|
|
void
|
|
copy_over (ldsym, bfdsym)
|
|
ldsym_type *ldsym;
|
|
asymbol **bfdsym;
|
|
{
|
|
while (list && *list)
|
|
{
|
|
refize(Q_enter_global_ref(list, name));
|
|
list = (asymbol **)((*list)->udata);
|
|
}
|
|
}
|
|
#endif
|
|
|
|
/* This call allows us to change the symbol table so that all future
|
|
refs to the symbol are patched to know the alias - but we still
|
|
have to fix all the old ones */
|
|
void
|
|
add_indirect (ptr)
|
|
asymbol **ptr;
|
|
{
|
|
asymbol **p;
|
|
ldsym_type *lgs = ldsym_get((*ptr)->name);
|
|
ldsym_type *new = ldsym_get(((asymbol *)((*ptr)->value))->name);
|
|
|
|
/* If the mapping has already been done, stop now */
|
|
if (lgs == new) return;
|
|
|
|
lgs->flags |= SYM_INDIRECT;
|
|
|
|
if (lgs->sdefs_chain && lgs->sdefs_chain[0])
|
|
{
|
|
einfo("indirect symbol already has definition %s", lgs->sdefs_chain[0]);
|
|
}
|
|
new->scoms_chain = move_it(new->scoms_chain, lgs->scoms_chain);
|
|
lgs->scoms_chain = 0;
|
|
new->srefs_chain = move_it(new->srefs_chain, lgs->srefs_chain);
|
|
lgs->srefs_chain = 0;
|
|
new->sdefs_chain = move_it(new->sdefs_chain, lgs->sdefs_chain);
|
|
lgs->sdefs_chain = 0;
|
|
|
|
/* If the result has any commons they should be turned into refs */
|
|
|
|
if (new->sdefs_chain && new->scoms_chain)
|
|
{
|
|
refize(new, new->scoms_chain);
|
|
}
|
|
lgs->sdefs_chain = (asymbol **)new;
|
|
lgs->srefs_chain = ptr;
|
|
}
|
|
|
|
|
|
|