#pragma ident "%Z%%M% %I% %E% SMI"
#include <sys/types.h>
#include <sys/param.h>
#include <sys/fpu/fpu_simulator.h>
#include <sys/fpu/globals.h>
#include <sys/systm.h>
#include <vm/seg.h>
#include <sys/privregs.h>
#include <sys/stack.h>
#include <sys/debug.h>
#include <sys/model.h>
enum ftt_type
_fp_read_inst(
const uint32_t *address,
uint32_t *pvalue,
fp_simd_type *pfpsd)
{
if (((uintptr_t)address & 0x3) != 0)
return (ftt_alignment);
if (get_udatamodel() == DATAMODEL_ILP32) {
caddr32_t address32 = (caddr32_t)(uintptr_t)address;
address = (uint32_t *)(uintptr_t)address32;
}
if (fuword32(address, pvalue) == -1) {
pfpsd->fp_trapaddr = (caddr_t)address;
pfpsd->fp_traprw = S_READ;
return (ftt_fault);
}
return (ftt_none);
}
enum ftt_type
_fp_read_extword(
const uint64_t *address,
uint64_t *pvalue,
fp_simd_type *pfpsd)
{
if (((uintptr_t)address & 0x7) != 0)
return (ftt_alignment);
if (get_udatamodel() == DATAMODEL_ILP32) {
caddr32_t address32 = (caddr32_t)(uintptr_t)address;
address = (uint64_t *)(uintptr_t)address32;
}
if (fuword64(address, pvalue) == -1) {
pfpsd->fp_trapaddr = (caddr_t)address;
pfpsd->fp_traprw = S_READ;
return (ftt_fault);
}
return (ftt_none);
}
enum ftt_type
_fp_read_word(
const uint32_t *address,
uint32_t *pvalue,
fp_simd_type *pfpsd)
{
if (((uintptr_t)address & 0x3) != 0)
return (ftt_alignment);
if (get_udatamodel() == DATAMODEL_ILP32) {
caddr32_t address32 = (caddr32_t)(uintptr_t)address;
address = (uint32_t *)(uintptr_t)address32;
}
if (fuword32(address, pvalue) == -1) {
pfpsd->fp_trapaddr = (caddr_t)address;
pfpsd->fp_traprw = S_READ;
return (ftt_fault);
}
return (ftt_none);
}
enum ftt_type
_fp_write_extword(
uint64_t *address,
uint64_t value,
fp_simd_type *pfpsd)
{
if (((uintptr_t)address & 0x7) != 0)
return (ftt_alignment);
if (get_udatamodel() == DATAMODEL_ILP32) {
caddr32_t address32 = (caddr32_t)(uintptr_t)address;
address = (uint64_t *)(uintptr_t)address32;
}
if (suword64(address, value) == -1) {
pfpsd->fp_trapaddr = (caddr_t)address;
pfpsd->fp_traprw = S_WRITE;
return (ftt_fault);
}
return (ftt_none);
}
enum ftt_type
_fp_write_word(
uint32_t *address,
uint32_t value,
fp_simd_type *pfpsd)
{
if (((uintptr_t)address & 0x3) != 0)
return (ftt_alignment);
if (get_udatamodel() == DATAMODEL_ILP32) {
caddr32_t address32 = (caddr32_t)(uintptr_t)address;
address = (uint32_t *)(uintptr_t)address32;
}
if (suword32(address, value) == -1) {
pfpsd->fp_trapaddr = (caddr_t)address;
pfpsd->fp_traprw = S_WRITE;
return (ftt_fault);
}
return (ftt_none);
}
enum ftt_type
read_iureg(
fp_simd_type *pfpsd,
uint_t n,
struct regs *pregs,
void *prw,
uint64_t *pvalue)
{
enum ftt_type ftt;
if (n == 0) {
*pvalue = 0;
return (ftt_none);
} else if (n < 16) {
long long *preg;
preg = &pregs->r_ps;
*pvalue = preg[n];
return (ftt_none);
} else if (USERMODE(pregs->r_tstate)) {
if (lwp_getdatamodel(curthread->t_lwp) == DATAMODEL_ILP32) {
uint32_t res, *addr, *rw;
caddr32_t rw32;
rw32 = (caddr32_t)(uintptr_t)prw;
rw = (uint32_t *)(uintptr_t)rw32;
addr = (uint32_t *)&rw[n - 16];
ftt = _fp_read_word(addr, &res, pfpsd);
*pvalue = (uint64_t)res;
} else {
uint64_t res, *addr, *rw = (uint64_t *)
((uintptr_t)prw + STACK_BIAS);
addr = (uint64_t *)&rw[n - 16];
ftt = _fp_read_extword(addr, &res, pfpsd);
*pvalue = res;
}
return (ftt);
} else {
ulong_t *addr, *rw = (ulong_t *)((uintptr_t)prw + STACK_BIAS);
ulong_t res;
addr = (ulong_t *)&rw[n - 16];
res = *addr;
*pvalue = res;
return (ftt_none);
}
}
enum ftt_type
write_iureg(
fp_simd_type *pfpsd,
uint_t n,
struct regs *pregs,
void *prw,
uint64_t *pvalue)
{
long long *preg;
enum ftt_type ftt;
if (n == 0) {
return (ftt_none);
} else if (n < 16) {
preg = &pregs->r_ps;
preg[n] = *pvalue;
return (ftt_none);
} else if (USERMODE(pregs->r_tstate)) {
if (lwp_getdatamodel(curthread->t_lwp) == DATAMODEL_ILP32) {
uint32_t res, *addr, *rw;
caddr32_t rw32;
rw32 = (caddr32_t)(uintptr_t)prw;
rw = (uint32_t *)(uintptr_t)rw32;
addr = &rw[n - 16];
res = (uint_t)*pvalue;
ftt = _fp_write_word(addr, res, pfpsd);
} else {
uint64_t *addr, *rw = (uint64_t *)
((uintptr_t)prw + STACK_BIAS);
uint64_t res;
addr = &rw[n - 16];
res = *pvalue;
ftt = _fp_write_extword(addr, res, pfpsd);
}
return (ftt);
} else {
ulong_t *addr, *rw = (ulong_t *)((uintptr_t)prw + STACK_BIAS);
ulong_t res = *pvalue;
addr = &rw[n - 16];
*addr = res;
return (ftt_none);
}
}