machsym.c revision 7c478bd95313f5f23a4c958a745db2134aa03244
/*
* CDDL HEADER START
*
* The contents of this file are subject to the terms of the
* Common Development and Distribution License, Version 1.0 only
* (the "License"). You may not use this file except in compliance
* with the License.
*
* You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
* or http://www.opensolaris.org/os/licensing.
* See the License for the specific language governing permissions
* and limitations under the License.
*
* When distributing Covered Code, include this CDDL HEADER in each
* file and include the License file at usr/src/OPENSOLARIS.LICENSE.
* If applicable, add the following below this CDDL HEADER, with the
* fields enclosed by brackets "[]" replaced with your own identifying
* information: Portions Copyright [yyyy] [name of copyright owner]
*
* CDDL HEADER END
*/
/*
* Copyright 2000-2003 Sun Microsystems, Inc. All rights reserved.
* Use is subject to license terms.
*/
#pragma ident "%Z%%M% %I% %E% SMI"
#include <stdio.h>
#include <string.h>
#include <alloca.h>
#include <sys/types.h>
#include "debug.h"
#include "msg.h"
#include "_libld.h"
/*
* Matrix of legal combinations of usage of a given register:
*
* Obj1 \ Obj2 Scratch Named
* Scratch OK NO
* Named NO *
*
* (*) OK if the symbols are identical, NO if they are not. Two symbols
* are identical if and only if one of the following is true:
* A. They are both global and have the same name.
* B. They are both local, have the same name, and are defined in the same
* object. (Note that a local symbol in one object is never identical to
* a local symbol in another object, even if the name is the same.)
*
* Matrix of legal combinations of st_shndx for the same register symbol:
*
* Obj1 \ Obj2 UNDEF ABS
* UNDEF OK OK
* ABS OK NO
*
*/
int
reg_check(Sym_desc *sdp, Sym *nsym, const char *nname, Ifl_desc *ifl,
Ofl_desc * ofl)
{
Sym *osym = sdp->sd_sym;
const char *oname = sdp->sd_name;
/*
* Scratch register definitions are compatible.
*/
if ((osym->st_name == 0) && (nsym->st_name == 0))
return (0);
/*
* A local and a global, or another local is incompatible.
*/
if ((ELF_ST_BIND(osym->st_info) == STB_LOCAL) ||
(ELF_ST_BIND(nsym->st_info) == STB_LOCAL)) {
if (osym->st_value == nsym->st_value) {
eprintf(ERR_FATAL, MSG_INTL(MSG_SYM_INCOMPREG3),
conv_sym_SPARC_value_str((Lword)osym->st_value),
sdp->sd_file->ifl_name, demangle(oname),
ifl->ifl_name, demangle(nname));
ofl->ofl_flags |= FLG_OF_FATAL;
return (1);
}
return (0);
}
if (osym->st_value == nsym->st_value) {
/*
* A scratch register and a named register are incompatible.
* So are two different named registers.
*/
if (((osym->st_name == 0) || (nsym->st_name == 0)) ||
(strcmp(oname, nname) != 0)) {
eprintf(ERR_FATAL, MSG_INTL(MSG_SYM_INCOMPREG1),
conv_sym_SPARC_value_str((Lword)osym->st_value),
sdp->sd_file->ifl_name, demangle(oname),
ifl->ifl_name, demangle(nname));
ofl->ofl_flags |= FLG_OF_FATAL;
return (1);
}
/*
* A multiply initialized symbol is also illegal.
*/
if ((osym->st_shndx == SHN_ABS) &&
(nsym->st_shndx == SHN_ABS)) {
eprintf(ERR_FATAL, MSG_INTL(MSG_SYM_MULTINIREG),
conv_sym_SPARC_value_str((Lword)osym->st_value),
demangle(nname), sdp->sd_file->ifl_name,
ifl->ifl_name);
ofl->ofl_flags |= FLG_OF_FATAL;
return (1);
}
} else if (strcmp(oname, nname) == 0) {
eprintf(ERR_FATAL, MSG_INTL(MSG_SYM_INCOMPREG2),
demangle(sdp->sd_name), sdp->sd_file->ifl_name,
conv_sym_SPARC_value_str((Lword)osym->st_value),
ifl->ifl_name,
conv_sym_SPARC_value_str((Lword)nsym->st_value));
ofl->ofl_flags |= FLG_OF_FATAL;
return (1);
}
return (0);
}
int
mach_sym_typecheck(Sym_desc *sdp, Sym *nsym, Ifl_desc *ifl, Ofl_desc *ofl)
{
Sym *osym = sdp->sd_sym;
Byte otype = ELF_ST_TYPE(osym->st_info);
Byte ntype = ELF_ST_TYPE(nsym->st_info);
if (otype != ntype) {
if ((otype == STT_SPARC_REGISTER) ||
(ntype == STT_SPARC_REGISTER)) {
eprintf(ERR_FATAL, MSG_INTL(MSG_SYM_DIFFTYPE),
demangle(sdp->sd_name));
eprintf(ERR_NONE, MSG_INTL(MSG_SYM_FILETYPES),
sdp->sd_file->ifl_name,
conv_info_type_str(ofl->ofl_e_machine, otype),
ifl->ifl_name,
conv_info_type_str(ofl->ofl_e_machine, ntype));
ofl->ofl_flags |= FLG_OF_FATAL;
return (1);
}
} else if (otype == STT_SPARC_REGISTER)
return (reg_check(sdp, nsym, sdp->sd_name, ifl, ofl));
return (0);
}
static const char *registers[] = { 0,
MSG_ORIG(MSG_STO_REGISTERG1), MSG_ORIG(MSG_STO_REGISTERG2),
MSG_ORIG(MSG_STO_REGISTERG3), MSG_ORIG(MSG_STO_REGISTERG4),
MSG_ORIG(MSG_STO_REGISTERG5), MSG_ORIG(MSG_STO_REGISTERG6),
MSG_ORIG(MSG_STO_REGISTERG7)
};
const char *
is_regsym(Ifl_desc *ifl, Sym *sym, const char *strs, int symndx, Word shndx,
const char *symsecname, Word * flags)
{
const char *name;
/*
* Only do something if this is a register symbol.
*/
if (ELF_ST_TYPE(sym->st_info) != STT_SPARC_REGISTER)
return (0);
/*
* Check for bogus register number.
*/
if ((sym->st_value < STO_SPARC_REGISTER_G1) ||
(sym->st_value > STO_SPARC_REGISTER_G7)) {
eprintf(ERR_FATAL, MSG_INTL(MSG_SYM_BADREG), ifl->ifl_name,
symsecname, symndx, EC_XWORD(sym->st_value));
return ((const char *)S_ERROR);
}
/*
* A register symbol can only be undefined or defined (absolute).
*/
if ((shndx != SHN_ABS) && (shndx != SHN_UNDEF)) {
eprintf(ERR_FATAL, MSG_INTL(MSG_SYM_BADREG), ifl->ifl_name,
symsecname, symndx, EC_XWORD(sym->st_value));
return ((const char *)S_ERROR);
}
/*
* Determine whether this is a scratch (unnamed) definition.
*/
if (sym->st_name == 0) {
/*
* Check for bogus scratch register definitions.
*/
if ((ELF_ST_BIND(sym->st_info) != STB_GLOBAL) ||
(shndx != SHN_UNDEF)) {
eprintf(ERR_FATAL, MSG_INTL(MSG_SYM_BADSCRATCH),
ifl->ifl_name, symsecname, symndx,
conv_sym_SPARC_value_str((Lword)sym->st_value));
return ((const char *)S_ERROR);
}
/*
* Fabricate a name for this register so that this definition
* can be processed through the symbol resolution engine.
*/
name = registers[sym->st_value];
} else
name = strs + sym->st_name;
/*
* Indicate we're dealing with a register and return its name.
*/
*flags |= FLG_SY_REGSYM;
return (name);
}
Sym_desc *
reg_find(Sym * sym, Ofl_desc * ofl)
{
if (ofl->ofl_regsyms == 0)
return (0);
return (ofl->ofl_regsyms[sym->st_value]);
}
int
reg_enter(Sym_desc * sdp, Ofl_desc * ofl)
{
if (ofl->ofl_regsyms == 0) {
ofl->ofl_regsymsno = STO_SPARC_REGISTER_G7 + 1;
if ((ofl->ofl_regsyms = libld_calloc(sizeof (Sym_desc *),
ofl->ofl_regsymsno)) == 0) {
ofl->ofl_flags |= FLG_OF_FATAL;
return (0);
}
}
ofl->ofl_regsyms[sdp->sd_sym->st_value] = sdp;
return (1);
}