#define ELF_TARGET_SPARC
#include <stdio.h>
#include <string.h>
#include <alloca.h>
#include <sys/types.h>
#include <debug.h>
#include "msg.h"
#include "_libld.h"
#include "machsym.sparc.h"
int
ld_reg_check_sparc(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;
Conv_inv_buf_t inv_buf1, inv_buf2;
if ((osym->st_name == 0) && (nsym->st_name == 0))
return (0);
if ((ELF_ST_BIND(osym->st_info) == STB_LOCAL) ||
(ELF_ST_BIND(nsym->st_info) == STB_LOCAL)) {
if (osym->st_value == nsym->st_value) {
ld_eprintf(ofl, ERR_FATAL,
MSG_INTL(MSG_SYM_INCOMPREG3),
conv_sym_SPARC_value(osym->st_value, 0, &inv_buf1),
sdp->sd_file->ifl_name, demangle(oname),
ifl->ifl_name, demangle(nname));
return (1);
}
return (0);
}
if (osym->st_value == nsym->st_value) {
if (((osym->st_name == 0) || (nsym->st_name == 0)) ||
(strcmp(oname, nname) != 0)) {
ld_eprintf(ofl, ERR_FATAL, MSG_INTL(MSG_SYM_INCOMPREG1),
conv_sym_SPARC_value(osym->st_value, 0, &inv_buf1),
sdp->sd_file->ifl_name, demangle(oname),
ifl->ifl_name, demangle(nname));
return (1);
}
if ((osym->st_shndx == SHN_ABS) &&
(nsym->st_shndx == SHN_ABS)) {
ld_eprintf(ofl, ERR_FATAL, MSG_INTL(MSG_SYM_MULTINIREG),
conv_sym_SPARC_value(osym->st_value, 0, &inv_buf1),
demangle(nname), sdp->sd_file->ifl_name,
ifl->ifl_name);
return (1);
}
} else if (strcmp(oname, nname) == 0) {
ld_eprintf(ofl, ERR_FATAL, MSG_INTL(MSG_SYM_INCOMPREG2),
demangle(sdp->sd_name), sdp->sd_file->ifl_name,
conv_sym_SPARC_value(osym->st_value, 0, &inv_buf1),
ifl->ifl_name,
conv_sym_SPARC_value(nsym->st_value, 0, &inv_buf2));
return (1);
}
return (0);
}
int
ld_mach_sym_typecheck_sparc(Sym_desc *sdp, Sym *nsym, Ifl_desc *ifl,
Ofl_desc *ofl)
{
Conv_inv_buf_t inv_buf1, inv_buf2;
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)) {
ld_eprintf(ofl, ERR_FATAL, MSG_INTL(MSG_SYM_DIFFTYPE),
demangle(sdp->sd_name));
ld_eprintf(ofl, ERR_NONE, MSG_INTL(MSG_SYM_FILETYPES),
sdp->sd_file->ifl_name, conv_sym_info_type(
sdp->sd_file->ifl_ehdr->e_machine, otype,
0, &inv_buf1), ifl->ifl_name,
conv_sym_info_type(ifl->ifl_ehdr->e_machine,
ntype, 0, &inv_buf2));
return (1);
}
} else if (otype == STT_SPARC_REGISTER)
return (ld_reg_check_sparc(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 *
ld_is_regsym_sparc(Ofl_desc *ofl, Ifl_desc *ifl, Sym *sym, const char *strs,
int symndx, Word shndx, const char *symsecname, sd_flag_t *flags)
{
const char *name;
if (ELF_ST_TYPE(sym->st_info) != STT_SPARC_REGISTER)
return (0);
if ((sym->st_value < STO_SPARC_REGISTER_G1) ||
(sym->st_value > STO_SPARC_REGISTER_G7)) {
ld_eprintf(ofl, ERR_FATAL, MSG_INTL(MSG_SYM_BADREG),
ifl->ifl_name, symsecname, symndx, EC_XWORD(sym->st_value));
return ((const char *)S_ERROR);
}
if ((shndx != SHN_ABS) && (shndx != SHN_UNDEF)) {
ld_eprintf(ofl, ERR_FATAL, MSG_INTL(MSG_SYM_BADREG),
ifl->ifl_name, symsecname, symndx, EC_XWORD(sym->st_value));
return ((const char *)S_ERROR);
}
if (sym->st_name == 0) {
if ((ELF_ST_BIND(sym->st_info) != STB_GLOBAL) ||
(shndx != SHN_UNDEF)) {
Conv_inv_buf_t inv_buf;
ld_eprintf(ofl, ERR_FATAL, MSG_INTL(MSG_SYM_BADSCRATCH),
ifl->ifl_name, symsecname, symndx,
conv_sym_SPARC_value(sym->st_value, 0, &inv_buf));
return ((const char *)S_ERROR);
}
name = registers[sym->st_value];
} else
name = strs + sym->st_name;
*flags |= FLG_SY_REGSYM;
return (name);
}
Sym_desc *
ld_reg_find_sparc(Sym *sym, Ofl_desc *ofl)
{
if (ofl->ofl_regsyms == NULL)
return (NULL);
return (ofl->ofl_regsyms[sym->st_value]);
}
int
ld_reg_enter_sparc(Sym_desc *sdp, Ofl_desc *ofl)
{
if (ofl->ofl_regsyms == NULL) {
ofl->ofl_regsymsno = STO_SPARC_REGISTER_G7 + 1;
if ((ofl->ofl_regsyms = libld_calloc(ofl->ofl_regsymsno,
sizeof (Sym_desc *))) == NULL) {
ofl->ofl_flags |= FLG_OF_FATAL;
return (0);
}
}
ofl->ofl_regsyms[sdp->sd_sym->st_value] = sdp;
return (1);
}