#include <fcntl.h>
#include <stdio.h>
#include <libelf.h>
#include <link.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <locale.h>
#include <errno.h>
#include <sgs.h>
#include <conv.h>
#include <gelf.h>
#include <debug.h>
#include <ctype.h>
#include <alist.h>
#include "msg.h"
#define AL_CNT_MATCH_LIST 5
#define AL_CNT_GVER_DESC 25
typedef struct cache {
Elf_Scn *c_scn;
Elf_Data *c_data;
char *c_name;
} Cache;
typedef struct gver_desc {
const char *vd_name;
unsigned long vd_hash;
GElf_Half vd_ndx;
GElf_Half vd_flags;
APlist *vd_deps;
} GVer_desc;
typedef struct {
GElf_Versym *vsd_vsp;
Elf_Data *vsd_sym_data;
Word vsd_symn;
const char *vsd_strs;
} Gver_sym_data;
typedef enum {
MATCH_OPT_NAME,
MATCH_OPT_NEED_VER,
MATCH_OPT_NDX,
MATCH_OPT_RANGE,
} match_opt_t;
typedef struct {
match_opt_t opt_type;
union {
struct {
const char *version;
const char *needobj;
} name;
struct {
int start;
int end;
} ndx;
} value;
} match_rec_t;
static const char *cname;
static int Cflag, dflag, lflag, nflag, oflag, rflag, sflag, vflag;
static Alist *match_list;
#define DEF_DEFINED 1
#define USR_DEFINED 2
static const char *
demangle(const char *name)
{
if (Cflag)
return (Elf_demangle_name(name));
else
return (name);
}
static void
pvs_aplist_append(APlist **lst, const void *item, const char *file)
{
if (aplist_append(lst, item, AL_CNT_GVER_DESC) == NULL) {
int err = errno;
(void) fprintf(stderr, MSG_INTL(MSG_SYS_MALLOC), cname, file,
strerror(err));
exit(1);
}
}
static void
add_match_record(int opt, const char *str)
{
#define WS_SKIP(_str) for (; *(_str) && isspace(*(_str)); (_str)++)
#define WS_SKIP_LIMIT(_str, _limit) \
while (((_str) < s2) && isspace(*(_str))) \
(_str)++
#define WS_RSKIP_LIMIT(_str, _tail) \
while (((_tail) > (_str)) && isspace(*((_tail) - 1))) \
(_tail)--; \
*(_tail) = '\0'
match_rec_t *rec;
char *lstr, *s1, *s2;
rec = alist_append(&match_list, NULL, sizeof (match_rec_t),
AL_CNT_MATCH_LIST);
if (rec == NULL) {
int err = errno;
(void) fprintf(stderr, MSG_INTL(MSG_SYS_MALLOC), cname,
MSG_INTL(MSG_STR_MATCH_RECORD), strerror(err));
exit(1);
}
if (opt == 'N') {
if ((lstr = strdup(str)) == NULL) {
int err = errno;
(void) fprintf(stderr, MSG_INTL(MSG_SYS_MALLOC),
cname, MSG_INTL(MSG_STR_MATCH_RECORD),
strerror(err));
exit(1);
}
s2 = lstr + strlen(lstr);
WS_SKIP_LIMIT(lstr, s2);
WS_RSKIP_LIMIT(lstr, s2);
rec->opt_type = MATCH_OPT_NAME;
rec->value.name.version = lstr;
if ((s2 == lstr) || (*(s2 - 1) != ')'))
return;
for (s1 = lstr; *s1 && (*s1 != '('); s1++)
;
if (*s1 != '(')
return;
rec->opt_type = MATCH_OPT_NEED_VER;
rec->value.name.needobj = lstr;
rec->value.name.version = s1 + 1;
s2--;
WS_SKIP_LIMIT(rec->value.name.version, s2);
WS_RSKIP_LIMIT(rec->value.name.version, s2);
WS_RSKIP_LIMIT(rec->value.name.needobj, s1);
return;
}
rec->value.ndx.start = strtol(str, &s2, 10);
if ((str == s2) || (rec->value.ndx.start < 1))
goto syntax_error;
str = s2;
WS_SKIP(str);
if (*str != ':') {
rec->opt_type = MATCH_OPT_NDX;
} else {
str++;
rec->opt_type = MATCH_OPT_RANGE;
WS_SKIP(str);
if (*str == '\0') {
rec->value.ndx.end = -1;
} else {
rec->value.ndx.end = strtol(str, &s2, 10);
if ((str == s2) || (rec->value.ndx.end < 0))
goto syntax_error;
str = s2;
WS_SKIP(str);
}
}
if (*str == '\0')
return;
syntax_error:
(void) fprintf(stderr, MSG_INTL(MSG_USAGE_BRIEF), cname);
exit(1);
#undef WS_SKIP
#undef WS_SKIP_LIMIT
#undef WS_RSKIP_LIMIT
}
int
match(const char *needobj, const char *version, int ndx)
{
Aliste _idx;
match_rec_t *rec;
const char *str;
if (alist_nitems(match_list) == 0)
return (1);
for (ALIST_TRAVERSE(match_list, _idx, rec)) {
switch (rec->opt_type) {
case MATCH_OPT_NAME:
if (needobj)
str = needobj;
else if (version)
str = version;
else
break;
if (strcmp(rec->value.name.version, str) == 0)
return (1);
break;
case MATCH_OPT_NEED_VER:
if (needobj && version &&
(strcmp(rec->value.name.needobj, needobj) == 0) &&
(strcmp(rec->value.name.version, version) == 0))
return (1);
break;
case MATCH_OPT_NDX:
if ((ndx > 0) && (ndx == rec->value.ndx.start))
return (1);
break;
case MATCH_OPT_RANGE:
if ((ndx > 0) &&
(ndx >= rec->value.ndx.start) &&
((rec->value.ndx.end < 0) ||
(ndx <= rec->value.ndx.end)))
return (1);
break;
}
}
return (0);
}
static void
gvers_syms(const Gver_sym_data *vsdata, GElf_Half vd_ndx,
const char *vd_name, const char *needobj, const char *file)
{
GElf_Sym sym;
int _symn;
for (_symn = 0; _symn < vsdata->vsd_symn; _symn++) {
size_t size = 0;
const char *name;
if (vsdata->vsd_vsp[_symn] != vd_ndx)
continue;
(void) gelf_getsym(vsdata->vsd_sym_data, _symn, &sym);
name = demangle(vsdata->vsd_strs + sym.st_name);
if (needobj == NULL) {
if ((GELF_ST_TYPE(sym.st_info) == STT_OBJECT) ||
(GELF_ST_TYPE(sym.st_info) == STT_COMMON) ||
(GELF_ST_TYPE(sym.st_info) == STT_TLS))
size = (size_t)sym.st_size;
if (!vflag && (sym.st_shndx == SHN_ABS) &&
(strcmp(name, vd_name) == 0))
continue;
}
if (oflag) {
if (needobj == NULL)
(void) printf(MSG_ORIG(MSG_FMT_SYM_OFIL),
file, vd_name);
else
(void) printf(MSG_ORIG(MSG_FMT_SYM_NEED_OFIL),
file, needobj, vd_name);
if (size)
(void) printf(MSG_ORIG(MSG_FMT_SYM_SZ_OFLG),
name, (ulong_t)size);
else
(void) printf(MSG_ORIG(MSG_FMT_SYM_OFLG), name);
} else {
if (size)
(void) printf(MSG_ORIG(MSG_FMT_SYM_SZ), name,
(ulong_t)size);
else
(void) printf(MSG_ORIG(MSG_FMT_SYM), name);
}
}
}
static void
sym_local(Cache *cache, Cache *csym, const char *file)
{
int symn, _symn, found = 0;
GElf_Shdr shdr;
GElf_Sym sym;
char *strs;
(void) gelf_getshdr(csym->c_scn, &shdr);
strs = (char *)cache[shdr.sh_link].c_data->d_buf;
symn = shdr.sh_info;
(void) gelf_getsym(csym->c_data, 1, &sym);
if (GELF_ST_TYPE(sym.st_info) != STT_FILE) {
(void) fprintf(stderr, MSG_INTL(MSG_VER_UNREDSYMS), cname,
file);
(void) fprintf(stderr, MSG_INTL(MSG_VER_NOTSTTFILE),
csym->c_name);
return;
}
for (_symn = 2; _symn < symn; _symn++) {
const char *name;
(void) gelf_getsym(csym->c_data, _symn, &sym);
if (GELF_ST_TYPE(sym.st_info) == STT_SECTION)
continue;
if (GELF_ST_TYPE(sym.st_info) == STT_FILE)
break;
if (GELF_ST_BIND(sym.st_info) != STB_LOCAL)
break;
name = demangle(strs + sym.st_name);
if (oflag) {
(void) printf(MSG_ORIG(MSG_FMT_LOCSYM_OFLG),
file, name);
} else {
if (found == 0) {
found = 1;
(void) printf(MSG_ORIG(MSG_FMT_LOCSYM_HDR));
}
(void) printf(MSG_ORIG(MSG_FMT_LOCSYM), name);
}
}
}
static int
gvers_need(Cache *cache, Cache *need, const Gver_sym_data *vsdata,
const char *file)
{
unsigned int num, _num;
char *strs;
GElf_Verneed *vnd = need->c_data->d_buf;
GElf_Shdr shdr;
int error = 0;
int show = vflag || (vsdata == NULL) || !oflag;
(void) gelf_getshdr(need->c_scn, &shdr);
if (vnd->vn_version > VER_DEF_CURRENT)
(void) fprintf(stderr, MSG_INTL(MSG_VER_HIGHREV), cname, file,
vnd->vn_version, VER_DEF_CURRENT);
strs = (char *)cache[shdr.sh_link].c_data->d_buf;
num = shdr.sh_info;
for (_num = 1; _num <= num; _num++,
vnd = (GElf_Verneed *)((uintptr_t)vnd + vnd->vn_next)) {
GElf_Vernaux *vnap;
Word ndx;
const char *needobj, *dep;
int started = 0, listcnt = 0;
vnap = (GElf_Vernaux *) ((uintptr_t)vnd + vnd->vn_aux);
needobj = (char *)(strs + vnd->vn_file);
error = 1;
for (ndx = 0; ndx < vnd->vn_cnt; ndx++,
vnap = (GElf_Vernaux *)((uintptr_t)vnap + vnap->vna_next)) {
Conv_ver_flags_buf_t ver_flags_buf;
dep = (char *)(strs + vnap->vna_name);
if (!match(needobj, dep, vnap->vna_other))
continue;
if (show) {
if ((started == 0) || (vsdata != NULL)) {
if (oflag && show)
(void) printf(
MSG_ORIG(MSG_FMT_OFIL),
file);
(void) printf(
MSG_ORIG(MSG_FMT_LIST_BEGIN),
needobj);
started = 1;
}
if (vflag || (vsdata != NULL) ||
(alist_nitems(match_list) != 0) ||
!(vnap->vna_flags & VER_FLG_INFO)) {
const char *fmt = (listcnt == 0) ?
MSG_ORIG(MSG_FMT_LIST_FIRST) :
MSG_ORIG(MSG_FMT_LIST_NEXT);
if (vsdata == NULL)
listcnt++;
(void) printf(fmt, dep);
if (vflag && (vnap->vna_flags != 0))
(void) printf(
MSG_ORIG(MSG_FMT_VER_FLG),
conv_ver_flags(
vnap->vna_flags,
CONV_FMT_NOBKT,
&ver_flags_buf));
}
if (vsdata != NULL)
(void) printf(oflag ?
MSG_ORIG(MSG_FMT_LIST_END_SEM) :
MSG_ORIG(MSG_FMT_LIST_END_COL));
}
if (vsdata && (vnap->vna_other > 0))
gvers_syms(vsdata, vnap->vna_other,
dep, needobj, file);
}
if (show && started && (vsdata == NULL))
(void) printf(MSG_ORIG(MSG_FMT_LIST_END_SEM));
}
return (error);
}
static GVer_desc *
gvers_find(const char *name, unsigned long hash, APlist *lst)
{
Aliste idx;
GVer_desc *vdp;
for (APLIST_TRAVERSE(lst, idx, vdp))
if ((vdp->vd_hash == hash) &&
(strcmp(vdp->vd_name, name) == 0))
return (vdp);
return (NULL);
}
static GVer_desc *
gvers_desc(const char *name, unsigned long hash, APlist **lst, const char *file)
{
GVer_desc *vdp;
if ((vdp = gvers_find(name, hash, *lst)) == NULL) {
if ((vdp = calloc(1, sizeof (GVer_desc))) == NULL) {
int err = errno;
(void) fprintf(stderr, MSG_INTL(MSG_SYS_MALLOC), cname,
file, strerror(err));
exit(1);
}
vdp->vd_name = name;
vdp->vd_hash = hash;
pvs_aplist_append(lst, vdp, file);
}
return (vdp);
}
static GVer_desc *
gvers_depend(const char *name, unsigned long hash, GVer_desc *vdp, APlist **lst,
const char *file)
{
GVer_desc *_vdp;
_vdp = gvers_desc(name, hash, lst, file);
pvs_aplist_append(&vdp->vd_deps, _vdp, file);
return (vdp);
}
static void
gvers_derefer(GVer_desc *vdp, int weak)
{
Aliste idx;
GVer_desc *_vdp;
if ((weak && (vdp->vd_flags & VER_FLG_WEAK)) || (!weak))
vdp->vd_flags &= ~FLG_VER_AVAIL;
for (APLIST_TRAVERSE(vdp->vd_deps, idx, _vdp))
gvers_derefer(_vdp, weak);
}
static void
recurse_syms(const Gver_sym_data *vsdata, GVer_desc *vdp, const char *file)
{
Aliste idx;
GVer_desc *_vdp;
for (APLIST_TRAVERSE(vdp->vd_deps, idx, _vdp)) {
if (!oflag)
(void) printf(MSG_ORIG(MSG_FMT_TNCO), _vdp->vd_name);
gvers_syms(vsdata, _vdp->vd_ndx, _vdp->vd_name, NULL, file);
if (aplist_nitems(_vdp->vd_deps) != 0)
recurse_syms(vsdata, _vdp, file);
}
}
static int
gvers_def(Cache *cache, Cache *def, const Gver_sym_data *vsdata,
const char *file)
{
unsigned int num, _num;
char *strs;
GElf_Verdef *vdf = def->c_data->d_buf;
GElf_Shdr shdr;
GVer_desc *vdp, *bvdp = NULL;
Aliste idx1;
APlist *verdefs = NULL;
int error = 0;
if (vdf->vd_version > VER_DEF_CURRENT) {
(void) fprintf(stderr, MSG_INTL(MSG_VER_HIGHREV), cname, file,
vdf->vd_version, VER_DEF_CURRENT);
}
(void) gelf_getshdr(def->c_scn, &shdr);
strs = (char *)cache[shdr.sh_link].c_data->d_buf;
num = shdr.sh_info;
for (_num = 1; _num <= num; _num++,
vdf = (GElf_Verdef *)((uintptr_t)vdf + vdf->vd_next)) {
GElf_Half cnt = vdf->vd_cnt;
GElf_Half ndx = vdf->vd_ndx;
GElf_Verdaux *vdap;
const char *_name;
vdap = (GElf_Verdaux *)((uintptr_t)vdf + vdf->vd_aux);
_name = (char *)(strs + vdap->vda_name);
vdp = gvers_desc(_name, elf_hash(_name), &verdefs, file);
vdp->vd_ndx = ndx;
vdp->vd_flags = vdf->vd_flags | FLG_VER_AVAIL;
vdap = (GElf_Verdaux *)((uintptr_t)vdap + vdap->vda_next);
for (cnt--; cnt; cnt--,
vdap = (GElf_Verdaux *)((uintptr_t)vdap + vdap->vda_next)) {
_name = (char *)(strs + vdap->vda_name);
if (gvers_depend(_name, elf_hash(_name), vdp,
&verdefs, file) == NULL)
return (0);
}
if (ndx == VER_NDX_GLOBAL)
bvdp = vdp;
}
if (nflag) {
for (APLIST_TRAVERSE(verdefs, idx1, vdp)) {
Aliste idx2;
GVer_desc *_vdp;
int type = vdp->vd_flags & VER_FLG_WEAK;
for (APLIST_TRAVERSE(vdp->vd_deps, idx2, _vdp))
gvers_derefer(_vdp, type);
}
if (bvdp)
bvdp->vd_flags &= ~FLG_VER_AVAIL;
}
for (APLIST_TRAVERSE(verdefs, idx1, vdp)) {
Aliste idx2;
GVer_desc *_vdp;
int count;
if (!match(NULL, vdp->vd_name, vdp->vd_ndx))
continue;
if ((alist_nitems(match_list) == 0) &&
!(vdp->vd_flags & FLG_VER_AVAIL))
continue;
error = 1;
if (vflag) {
if (oflag)
(void) printf(MSG_ORIG(MSG_FMT_OFIL), file);
(void) printf(MSG_ORIG(MSG_FMT_VER_NAME), vdp->vd_name);
if ((vdp->vd_flags & MSK_VER_USER) != 0) {
Conv_ver_flags_buf_t ver_flags_buf;
(void) printf(MSG_ORIG(MSG_FMT_VER_FLG),
conv_ver_flags(
vdp->vd_flags & MSK_VER_USER,
CONV_FMT_NOBKT, &ver_flags_buf));
}
count = 1;
for (APLIST_TRAVERSE(vdp->vd_deps, idx2, _vdp)) {
const char *_name = _vdp->vd_name;
if (count++ == 1) {
if (oflag)
(void) printf(
MSG_ORIG(MSG_FMT_IN_OFLG),
_name);
else if (vdp->vd_flags & VER_FLG_WEAK)
(void) printf(
MSG_ORIG(MSG_FMT_IN_WEAK),
_name);
else
(void) printf(
MSG_ORIG(MSG_FMT_IN),
_name);
} else
(void) printf(
MSG_ORIG(MSG_FMT_LIST_NEXT), _name);
}
if (count != 1)
(void) printf(MSG_ORIG(MSG_FMT_IN_END));
if (vsdata && !oflag)
(void) printf(MSG_ORIG(MSG_FMT_COL_NL));
else
(void) printf(MSG_ORIG(MSG_FMT_SEM_NL));
} else {
if (vsdata && !oflag)
(void) printf(MSG_ORIG(MSG_FMT_TNCO),
vdp->vd_name);
else if (!vsdata) {
if (oflag)
(void) printf(MSG_ORIG(MSG_FMT_OFIL),
file);
(void) printf(MSG_ORIG(MSG_FMT_TNSE),
vdp->vd_name);
}
}
if (vsdata == NULL)
continue;
gvers_syms(vsdata, vdp->vd_ndx, vdp->vd_name, NULL, file);
if (alist_nitems(match_list) != 0) {
recurse_syms(vsdata, vdp, file);
if (vflag && bvdp &&
!match(NULL, bvdp->vd_name, bvdp->vd_ndx)) {
if (!oflag)
(void) printf(MSG_ORIG(MSG_FMT_TNCO),
bvdp->vd_name);
gvers_syms(vsdata, bvdp->vd_ndx,
bvdp->vd_name, NULL, file);
}
}
}
return (error);
}
int
main(int argc, char **argv, char **envp)
{
GElf_Shdr shdr;
Elf *elf;
Elf_Scn *scn;
Elf_Data *data;
GElf_Ehdr ehdr;
int nfile, var;
char *names;
Cache *cache, *_cache;
Cache *_cache_def, *_cache_need, *_cache_sym, *_cache_loc;
int error = 0;
Gver_sym_data vsdata_s;
const Gver_sym_data *vsdata = NULL;
(void) setlocale(LC_MESSAGES, MSG_ORIG(MSG_STR_EMPTY));
(void) textdomain(MSG_ORIG(MSG_SUNW_OST_SGS));
cname = argv[0];
Cflag = dflag = lflag = nflag = oflag = rflag = sflag = vflag = 0;
opterr = 0;
while ((var = getopt(argc, argv, MSG_ORIG(MSG_STR_OPTIONS))) != EOF) {
switch (var) {
case 'C':
Cflag = USR_DEFINED;
break;
case 'd':
dflag = USR_DEFINED;
break;
case 'l':
lflag = sflag = USR_DEFINED;
break;
case 'n':
nflag = USR_DEFINED;
break;
case 'o':
oflag = USR_DEFINED;
break;
case 'r':
rflag = USR_DEFINED;
break;
case 's':
sflag = USR_DEFINED;
break;
case 'v':
vflag = USR_DEFINED;
break;
case 'I':
case 'N':
add_match_record(var, optarg);
break;
case '?':
(void) fprintf(stderr, MSG_INTL(MSG_USAGE_BRIEF),
cname);
(void) fprintf(stderr, MSG_INTL(MSG_USAGE_DETAIL));
exit(1);
default:
break;
}
}
if ((nfile = argc - optind) == 0) {
(void) fprintf(stderr, MSG_INTL(MSG_USAGE_BRIEF), cname);
exit(1);
}
if ((dflag == 0) && (rflag == 0) && (lflag == 0))
dflag = rflag = DEF_DEFINED;
for (; optind < argc; optind++) {
int derror = 0, nerror = 0, err;
const char *file = argv[optind];
size_t shnum = 0;
if ((var = open(file, O_RDONLY)) == -1) {
err = errno;
(void) fprintf(stderr, MSG_INTL(MSG_SYS_OPEN),
cname, file, strerror(err));
error = 1;
continue;
}
(void) elf_version(EV_CURRENT);
if ((elf = elf_begin(var, ELF_C_READ, NULL)) == NULL) {
(void) fprintf(stderr, MSG_ORIG(MSG_ELF_BEGIN), cname,
file, elf_errmsg(elf_errno()));
error = 1;
(void) close(var);
continue;
}
if (elf_kind(elf) != ELF_K_ELF) {
(void) fprintf(stderr, MSG_INTL(MSG_ELF_NOTELF), cname,
file);
error = 1;
(void) close(var);
(void) elf_end(elf);
continue;
}
if (gelf_getehdr(elf, &ehdr) == NULL) {
(void) fprintf(stderr, MSG_ORIG(MSG_ELF_GETEHDR), cname,
file, elf_errmsg(elf_errno()));
error = 1;
(void) close(var);
(void) elf_end(elf);
continue;
}
if ((scn = elf_getscn(elf, ehdr.e_shstrndx)) == NULL) {
(void) fprintf(stderr, MSG_ORIG(MSG_ELF_GETSCN), cname,
file, elf_errmsg(elf_errno()));
error = 1;
(void) close(var);
(void) elf_end(elf);
continue;
}
if ((data = elf_getdata(scn, NULL)) == NULL) {
(void) fprintf(stderr, MSG_ORIG(MSG_ELF_GETDATA), cname,
file, elf_errmsg(elf_errno()));
error = 1;
(void) close(var);
(void) elf_end(elf);
continue;
}
names = data->d_buf;
if (elf_getshdrnum(elf, &shnum) == -1) {
(void) fprintf(stderr, MSG_ORIG(MSG_ELF_GETSHDRNUM),
cname, file, elf_errmsg(elf_errno()));
exit(1);
}
if ((cache = calloc(shnum, sizeof (Cache))) == NULL) {
int err = errno;
(void) fprintf(stderr, MSG_INTL(MSG_SYS_MALLOC), cname,
file, strerror(err));
exit(1);
}
_cache_def = _cache_need = _cache_sym = _cache_loc = NULL;
_cache = cache;
_cache++;
for (scn = NULL; (scn = elf_nextscn(elf, scn)) != NULL;
_cache++) {
if (gelf_getshdr(scn, &shdr) == NULL) {
(void) fprintf(stderr,
MSG_ORIG(MSG_ELF_GETSHDR), cname, file,
elf_errmsg(elf_errno()));
error = 1;
continue;
}
if ((_cache->c_data = elf_getdata(scn, NULL)) ==
NULL) {
(void) fprintf(stderr,
MSG_ORIG(MSG_ELF_GETDATA), cname, file,
elf_errmsg(elf_errno()));
error = 1;
continue;
}
_cache->c_scn = scn;
_cache->c_name = names + shdr.sh_name;
switch (shdr.sh_type) {
case SHT_SUNW_verdef:
if (dflag)
_cache_def = _cache;
break;
case SHT_SUNW_verneed:
if (rflag)
_cache_need = _cache;
break;
case SHT_SUNW_versym:
if (sflag)
_cache_sym = _cache;
break;
case SHT_SYMTAB:
if (lflag)
_cache_loc = _cache;
break;
}
}
if (lflag && (_cache_loc == NULL)) {
(void) fprintf(stderr, MSG_INTL(MSG_VER_UNREDSYMS),
cname, file);
(void) fprintf(stderr, MSG_INTL(MSG_VER_NOSYMTAB));
}
if ((nfile > 1) && !oflag)
(void) printf(MSG_ORIG(MSG_FMT_FILE), file);
if (_cache_sym != NULL) {
vsdata = &vsdata_s;
(void) gelf_getshdr(_cache_sym->c_scn, &shdr);
vsdata_s.vsd_vsp =
(GElf_Versym *)_cache_sym->c_data->d_buf;
vsdata_s.vsd_sym_data = cache[shdr.sh_link].c_data;
(void) gelf_getshdr(cache[shdr.sh_link].c_scn, &shdr);
vsdata_s.vsd_symn = shdr.sh_size / shdr.sh_entsize;
vsdata_s.vsd_strs =
(const char *)cache[shdr.sh_link].c_data->d_buf;
}
if (_cache_need)
nerror = gvers_need(cache, _cache_need, vsdata, file);
if (_cache_def)
derror = gvers_def(cache, _cache_def, vsdata, file);
if (_cache_loc)
sym_local(cache, _cache_loc, file);
if (((dflag == USR_DEFINED) && (derror == 0)) ||
((rflag == USR_DEFINED) && (nerror == 0)) ||
(rflag && dflag && (derror == 0) && (nerror == 0)))
error = 1;
(void) close(var);
(void) elf_end(elf);
free(cache);
}
return (error);
}
const char *
_pvs_msg(Msg mid)
{
return (gettext(MSG_ORIG(mid)));
}