#include <signal.h>
#include <stdlib.h>
#include <sys/auxv.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <link.h>
#include <dlfcn.h>
#include "_rtld.h"
#include "_audit.h"
#include "msg.h"
uint_t _plt_save_size;
void (*_plt_fp_save)(void *);
void (*_plt_fp_restore)(void *);
extern void _elf_rtbndr_fp_save_orig(void *);
extern void _elf_rtbndr_fp_restore_orig(void *);
extern void _elf_rtbndr_fp_fxsave(void *);
extern void _elf_rtbndr_fp_fxrestore(void *);
extern void _elf_rtbndr_fp_xsave(void *);
extern void _elf_rtbndr_fp_xrestore(void *);
static void
_setup_plt_fpu(int kind, size_t len)
{
if (len == 0)
kind = -1;
switch (kind) {
case AT_386_FPINFO_FXSAVE:
_plt_fp_save = _elf_rtbndr_fp_fxsave;
_plt_fp_restore = _elf_rtbndr_fp_fxrestore;
_plt_save_size = len;
break;
case AT_386_FPINFO_XSAVE:
case AT_386_FPINFO_XSAVE_AMD:
_plt_fp_save = _elf_rtbndr_fp_xsave;
_plt_fp_restore = _elf_rtbndr_fp_xrestore;
_plt_save_size = len;
break;
default:
_plt_fp_save = _elf_rtbndr_fp_save_orig;
_plt_fp_restore = _elf_rtbndr_fp_restore_orig;
_plt_save_size = 64 * 8;
break;
}
}
unsigned long
_setup(Boot *ebp, Dyn *ld_dyn)
{
ulong_t reladdr, relacount, ld_base = 0;
ulong_t relaent = 0, pltrelsz = 0;
ulong_t strtab, soname, interp_base = 0;
char *_rt_name, **_envp, **_argv;
int _syspagsz = 0, fd = -1;
uint_t _flags = 0;
uint_t hwcap[3] = { 0, 0, 0 };
Dyn *dyn_ptr;
Phdr *phdr = NULL;
Rt_map *lmp;
auxv_t *auxv, *_auxv;
uid_t uid = (uid_t)-1, euid = (uid_t)-1;
gid_t gid = (gid_t)-1, egid = (gid_t)-1;
char *_platform = NULL, *_execname = NULL, *_emulator = NULL;
int auxflags = -1, fpkind = -1;
size_t fpsize = 0;
for (; ebp->eb_tag != EB_NULL; ebp++)
switch (ebp->eb_tag) {
case EB_LDSO_BASE:
ld_base = (unsigned long)ebp->eb_un.eb_val;
break;
case EB_ARGV:
_argv = (char **)ebp->eb_un.eb_ptr;
break;
case EB_ENVP:
_envp = (char **)ebp->eb_un.eb_ptr;
break;
case EB_AUXV:
_auxv = (auxv_t *)ebp->eb_un.eb_ptr;
break;
case EB_PAGESIZE:
_syspagsz = (int)ebp->eb_un.eb_val;
break;
}
for (auxv = _auxv; auxv->a_type != AT_NULL; auxv++) {
switch (auxv->a_type) {
case AT_EXECFD:
fd = (int)auxv->a_un.a_val;
break;
case AT_FLAGS:
_flags = auxv->a_un.a_val;
break;
case AT_PAGESZ:
_syspagsz = (int)auxv->a_un.a_val;
break;
case AT_PHDR:
phdr = (Phdr *)auxv->a_un.a_ptr;
break;
case AT_BASE:
if (ld_base == 0)
ld_base = auxv->a_un.a_val;
interp_base = auxv->a_un.a_val;
break;
case AT_SUN_UID:
euid = (uid_t)auxv->a_un.a_val;
break;
case AT_SUN_RUID:
uid = (uid_t)auxv->a_un.a_val;
break;
case AT_SUN_GID:
egid = (gid_t)auxv->a_un.a_val;
break;
case AT_SUN_RGID:
gid = (gid_t)auxv->a_un.a_val;
break;
case AT_SUN_PLATFORM:
_platform = auxv->a_un.a_ptr;
break;
case AT_SUN_EXECNAME:
_execname = auxv->a_un.a_ptr;
break;
case AT_SUN_AUXFLAGS:
auxflags = (int)auxv->a_un.a_val;
break;
case AT_SUN_HWCAP:
hwcap[0] = (uint_t)auxv->a_un.a_val;
break;
case AT_SUN_HWCAP2:
hwcap[1] = (uint_t)auxv->a_un.a_val;
break;
case AT_SUN_HWCAP3:
hwcap[2] = (uint_t)auxv->a_un.a_val;
break;
case AT_SUN_EMULATOR:
_emulator = auxv->a_un.a_ptr;
break;
case AT_SUN_FPTYPE:
fpkind = (int)auxv->a_un.a_val;
break;
case AT_SUN_FPSIZE:
fpsize = (size_t)auxv->a_un.a_val;
break;
}
}
dyn_ptr = (Dyn *)((char *)ld_dyn + ld_base);
for (ld_dyn = dyn_ptr; ld_dyn->d_tag != DT_NULL; ld_dyn++) {
switch (ld_dyn->d_tag) {
case DT_RELA:
reladdr = ld_dyn->d_un.d_ptr + ld_base;
break;
case DT_RELACOUNT:
relacount = ld_dyn->d_un.d_val;
break;
case DT_RELAENT:
relaent = ld_dyn->d_un.d_val;
break;
case DT_PLTRELSZ:
pltrelsz = ld_dyn->d_un.d_val;
break;
case DT_STRTAB:
strtab = ld_dyn->d_un.d_ptr + ld_base;
break;
case DT_SONAME:
soname = ld_dyn->d_un.d_val;
break;
}
}
_rt_name = (char *)strtab + soname;
if (relaent == 0)
relaent = sizeof (Rela);
relacount += (pltrelsz / relaent);
for (; relacount; relacount--) {
ulong_t roffset;
roffset = ((Rela *)reladdr)->r_offset + ld_base;
*((ulong_t *)roffset) += ld_base +
((Rela *)reladdr)->r_addend;
reladdr += relaent;
}
if (_emulator != NULL) {
_execname = _emulator;
rtld_flags2 |= RT_FL2_BRANDED;
}
dyn_plt_ent_size = ROUND(dyn_plt_ent_size, M_WORD_ALIGN) +
sizeof (uintptr_t) + sizeof (uintptr_t) + sizeof (ulong_t) +
sizeof (ulong_t) + sizeof (Sym);
_setup_plt_fpu(fpkind, fpsize);
if ((lmp = setup((char **)_envp, (auxv_t *)_auxv, _flags, _platform,
_syspagsz, _rt_name, ld_base, interp_base, fd, phdr,
_execname, _argv, uid, euid, gid, egid, auxflags,
hwcap)) == NULL) {
rtldexit(&lml_main, 1);
}
return (LM_ENTRY_PT(lmp)());
}