CVSROOT: /cvs
Module name: src
Changes by: [email protected] 2015/12/08 12:48:06
Modified files:
sys/dev/pv : xenvar.h
Log message:
/*
* Copyright (c) 2015 Mike Belopuhov
*
* Permission to use, copy, modify, and distribute this software for any
* purpose with or without fee is hereby granted, provided that the above
* copyright notice and this permission notice appear in all copies.
*
* THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
* WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
* MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
* ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
* WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
* ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
* OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
*/
#include <sys/param.h>
#include <sys/systm.h>
#include <sys/atomic.h>
#include <sys/malloc.h>
#include <sys/kernel.h>
#include <sys/device.h>
#include <machine/bus.h>
#include <machine/cpu.h>
#include <machine/cpufunc.h>
#include <uvm/uvm_extern.h>
#include <machine/i82489var.h>
#include <dev/pv/pvvar.h>
#include <dev/pv/xenreg.h>
#include <dev/pv/xenvar.h>
struct xen_softc *xen_sc;
int xen_init_hypercall(struct xen_softc *);
int xen_getversion(struct xen_softc *);
int xen_getfeatures(struct xen_softc *);
int xen_init_info_page(struct xen_softc *);
int xen_init_cbvec(struct xen_softc *);
int xen_match(struct device *, void *, void *);
void xen_attach(struct device *, struct device *, void *);
void xen_resume(struct device *);
int xen_activate(struct device *, int);
const struct cfdriver xen_cd = {
NULL, "xen", DV_DULL
};
const struct cfattach xen_ca = {
sizeof(struct xen_softc), xen_match, xen_attach, NULL, xen_activate
};
int
xen_match(struct device *parent, void *match, void *aux)
{
struct pv_attach_args *pva = aux;
struct pvbus_hv *hv = &pva->pva_hv[PVBUS_XEN];
if (hv->hv_base == 0)
return (0);
return (1);
}
void
xen_attach(struct device *parent, struct device *self, void *aux)
{
struct pv_attach_args *pva = (struct pv_attach_args *)aux;
struct pvbus_hv *hv = &pva->pva_hv[PVBUS_XEN];
struct xen_softc *sc = (struct xen_softc *)self;
sc->sc_base = hv->hv_base;
printf("\n");
if (xen_init_hypercall(sc))
return;
/* Wire it up to the global */
xen_sc = sc;
if (xen_getversion(sc))
return;
if (xen_getfeatures(sc))
return;
if (xen_init_info_page(sc))
return;
xen_init_cbvec(sc);
}
void
xen_resume(struct device *self)
{
}
int
xen_activate(struct device *self, int act)
{
int rv = 0;
switch (act) {
case DVACT_RESUME:
xen_resume(self);
break;
}
return (rv);
}
int
xen_init_hypercall(struct xen_softc *sc)
{
extern void *xen_hypercall_page;
uint32_t regs[4];
paddr_t pa;
/* Get hypercall page configuration MSR */
CPUID(sc->sc_base + CPUID_OFFSET_XEN_HYPERCALL,
regs[0], regs[1], regs[2], regs[3]);
/* We don't support more than one hypercall page */
if (regs[0] != 1) {
printf("%s: requested %d hypercall pages\n",
sc->sc_dev.dv_xname, regs[0]);
return (-1);
}
sc->sc_hc = &xen_hypercall_page;
if (!pmap_extract(pmap_kernel(), (vaddr_t)sc->sc_hc, &pa)) {
printf("%s: hypercall page PA extraction failed\n",
sc->sc_dev.dv_xname);
return (-1);
}
wrmsr(regs[1], pa);
DPRINTF("%s: hypercall page at va %p pa %#lx\n", sc->sc_dev.dv_xname,
sc->sc_hc, pa);
return (0);
}
int
xen_hypercall(struct xen_softc *sc, int op, int argc, ...)
{
va_list ap;
ulong argv[5];
int i;
if (argc < 0 || argc > 5)
return (-1);
va_start(ap, argc);
for (i = 0; i < argc; i++)
argv[i] = (ulong)va_arg(ap, ulong);
return (xen_hypercallv(sc, op, argc, argv));
}
int
xen_hypercallv(struct xen_softc *sc, int op, int argc, ulong *argv)
{
ulong hcall;
int rv = 0;
hcall = (ulong)sc->sc_hc + op * 32;
#if defined(XEN_DEBUG) && disabled
{
int i;
printf("hypercall %d", op);
if (argc > 0) {
printf(", args {");
for (i = 0; i < argc; i++)
printf(" %#lx", argv[i]);
printf(" }\n");
} else
printf("\n");
}
#endif
switch (argc) {
case 0: {
HYPERCALL_RES1;
__asm__ volatile ( \
HYPERCALL_LABEL \
: HYPERCALL_OUT1 \
: HYPERCALL_PTR(hcall) \
: HYPERCALL_CLOBBER \
);
HYPERCALL_RET(rv);
break;
}
case 1: {
HYPERCALL_RES1; HYPERCALL_RES2;
HYPERCALL_ARG1(argv[0]);
__asm__ volatile ( \
HYPERCALL_LABEL \
: HYPERCALL_OUT1 HYPERCALL_OUT2 \
: HYPERCALL_IN1 \
, HYPERCALL_PTR(hcall) \
: HYPERCALL_CLOBBER \
);
HYPERCALL_RET(rv);
break;
}
case 2: {
HYPERCALL_RES1; HYPERCALL_RES2; HYPERCALL_RES3;
HYPERCALL_ARG1(argv[0]); HYPERCALL_ARG2(argv[1]);
__asm__ volatile ( \
HYPERCALL_LABEL \
: HYPERCALL_OUT1 HYPERCALL_OUT2 \
HYPERCALL_OUT3 \
: HYPERCALL_IN1 HYPERCALL_IN2 \
, HYPERCALL_PTR(hcall) \
: HYPERCALL_CLOBBER \
);
HYPERCALL_RET(rv);
break;
}
case 3: {
HYPERCALL_RES1; HYPERCALL_RES2; HYPERCALL_RES3;
HYPERCALL_RES4;
HYPERCALL_ARG1(argv[0]); HYPERCALL_ARG2(argv[1]);
HYPERCALL_ARG3(argv[2]);
__asm__ volatile ( \
HYPERCALL_LABEL \
: HYPERCALL_OUT1 HYPERCALL_OUT2 \
HYPERCALL_OUT3 HYPERCALL_OUT4 \
: HYPERCALL_IN1 HYPERCALL_IN2 \
HYPERCALL_IN3 \
, HYPERCALL_PTR(hcall) \
: HYPERCALL_CLOBBER \
);
HYPERCALL_RET(rv);
break;
}
case 4: {
HYPERCALL_RES1; HYPERCALL_RES2; HYPERCALL_RES3;
HYPERCALL_RES4; HYPERCALL_RES5;
HYPERCALL_ARG1(argv[0]); HYPERCALL_ARG2(argv[1]);
HYPERCALL_ARG3(argv[2]); HYPERCALL_ARG4(argv[3]);
__asm__ volatile ( \
HYPERCALL_LABEL \
: HYPERCALL_OUT1 HYPERCALL_OUT2 \
HYPERCALL_OUT3 HYPERCALL_OUT4 \
HYPERCALL_OUT5 \
: HYPERCALL_IN1 HYPERCALL_IN2 \
HYPERCALL_IN3 HYPERCALL_IN4 \
, HYPERCALL_PTR(hcall) \
: HYPERCALL_CLOBBER \
);
HYPERCALL_RET(rv);
break;
}
case 5: {
HYPERCALL_RES1; HYPERCALL_RES2; HYPERCALL_RES3;
HYPERCALL_RES4; HYPERCALL_RES5; HYPERCALL_RES6;
HYPERCALL_ARG1(argv[0]); HYPERCALL_ARG2(argv[1]);
HYPERCALL_ARG3(argv[2]); HYPERCALL_ARG4(argv[3]);
HYPERCALL_ARG5(argv[4]);
__asm__ volatile ( \
HYPERCALL_LABEL \
: HYPERCALL_OUT1 HYPERCALL_OUT2 \
HYPERCALL_OUT3 HYPERCALL_OUT4 \
HYPERCALL_OUT5 HYPERCALL_OUT6 \
: HYPERCALL_IN1 HYPERCALL_IN2 \
HYPERCALL_IN3 HYPERCALL_IN4 \
HYPERCALL_IN5 \
, HYPERCALL_PTR(hcall) \
: HYPERCALL_CLOBBER \
);
HYPERCALL_RET(rv);
break;
}
default:
DPRINTF("%s: wrong number of arguments: %d\n", __func__, argc);
rv = -1;
break;
}
return (rv);
}
int
xen_getversion(struct xen_softc *sc)
{
char buf[16];
int version;
ulong argv[2] = { XENVER_extraversion, (ulong)&buf[0] };
int argc = 2;
memset(buf, 0, sizeof(buf));
if ((version = xen_hypercall(sc, xen_version, 1, XENVER_version)) < 0) {
printf("%s: failed to fetch version\n", sc->sc_dev.dv_xname);
return (-1);
}
if (xen_hypercallv(sc, xen_version, argc, argv) < 0) {
printf("%s: failed to fetch extended version\n",
sc->sc_dev.dv_xname);
return (-1);
}
printf("%s: version %d.%d%s\n", sc->sc_dev.dv_xname,
version >> 16, version & 0xffff, buf);
return (0);
}
int
xen_getfeatures(struct xen_softc *sc)
{
struct xen_feature_info xfi;
ulong argv[2] = { XENVER_get_features, (ulong)&xfi };
int argc = 2;
memset(&xfi, 0, sizeof(xfi));
if (xen_hypercallv(sc, xen_version, argc, argv) < 0) {
printf("%s: failed to fetch features\n", sc->sc_dev.dv_xname);
return (-1);
}
sc->sc_features = xfi.submap;
printf("%s: features %b\n", sc->sc_dev.dv_xname, sc->sc_features,
"\20\014DOM0\013PIRQ\012PVCLOCK\011CBVEC\010GNTFLAGS\007HMA"
"\006PTUPD\005PAE4G\004SUPERVISOR\003AUTOPMAP\002WDT\001WPT");
return (0);
}
#ifdef XEN_DEBUG
void
xen_print_info_page(void)
{
struct xen_softc *sc = xen_sc;
struct shared_info *s = sc->sc_ipg;
struct vcpu_info *v;
int i;
membar_sync();
for (i = 0; i < XEN_LEGACY_MAX_VCPUS; i++) {
v = &s->vcpu_info[i];
if (!v->evtchn_upcall_pending && !v->evtchn_upcall_mask &&
!v->evtchn_pending_sel && !v->time.version &&
!v->time.tsc_timestamp && !v->time.system_time &&
!v->time.tsc_to_system_mul && !v->time.tsc_shift)
continue;
printf("vcpu%d:\n"
" upcall_pending=%02x upcall_mask=%02x pending_sel=%#lx\n"
" time version=%u tsc=%llu system=%llu\n"
" time mul=%u shift=%d\n"
, i, v->evtchn_upcall_pending, v->evtchn_upcall_mask,
v->evtchn_pending_sel, v->time.version,
v->time.tsc_timestamp, v->time.system_time,
v->time.tsc_to_system_mul, v->time.tsc_shift);
}
printf("pending events: ");
for (i = 0; i < nitems(s->evtchn_pending); i++) {
if (s->evtchn_pending[i] == 0)
continue;
printf(" %d:%#lx", i, s->evtchn_pending[i]);
}
printf("\nmasked events: ");
for (i = 0; i < nitems(s->evtchn_mask); i++) {
if (s->evtchn_mask[i] == 0xffffffffffffffffULL)
continue;
printf(" %d:%#lx", i, s->evtchn_mask[i]);
}
printf("\nwc ver=%u sec=%u nsec=%u\n", s->wc_version, s->wc_sec,
s->wc_nsec);
printf("arch maxpfn=%lu framelist=%lu nmi=%lu\n", s->arch.max_pfn,
s->arch.pfn_to_mfn_frame_list, s->arch.nmi_reason);
}
#endif /* XEN_DEBUG */
int
xen_init_info_page(struct xen_softc *sc)
{
struct xen_add_to_physmap xatp;
paddr_t pa;
sc->sc_ipg = malloc(PAGE_SIZE, M_DEVBUF, M_NOWAIT | M_ZERO);
if (sc->sc_ipg == NULL) {
printf("%s: failed to allocate shared info page\n",
sc->sc_dev.dv_xname);
return (-1);
}
if (!pmap_extract(pmap_kernel(), (vaddr_t)sc->sc_ipg, &pa)) {
printf("%s: shared info page PA extraction failed\n",
sc->sc_dev.dv_xname);
free(sc->sc_ipg, M_DEVBUF, PAGE_SIZE);
return (-1);
}
xatp.domid = DOMID_SELF;
xatp.idx = 0;
xatp.space = XENMAPSPACE_shared_info;
xatp.gpfn = atop(pa);
if (xen_hypercall(sc, memory_op, 2, XENMEM_add_to_physmap, &xatp)) {
printf("%s: failed to register shared info page\n",
sc->sc_dev.dv_xname);
free(sc->sc_ipg, M_DEVBUF, PAGE_SIZE);
return (-1);
}
DPRINTF("%s: shared info page at va %p pa %#lx\n", sc->sc_dev.dv_xname,
sc->sc_ipg, pa);
return (0);
}
int
xen_init_cbvec(struct xen_softc *sc)
{
struct xen_hvm_param xhp;
if ((sc->sc_features & XENFEAT_CBVEC) == 0)
return (ENOENT);
xhp.domid = DOMID_SELF;
xhp.index = HVM_PARAM_CALLBACK_IRQ;
xhp.value = HVM_CALLBACK_VECTOR(LAPIC_XEN_VECTOR);
if (xen_hypercall(sc, hvm_op, 2, HVMOP_set_param, &xhp)) {
/* Will retry with the xspd(4) PCI interrupt */
return (ENOENT);
}
DPRINTF("%s: registered callback IDT vector %d\n",
sc->sc_dev.dv_xname, LAPIC_XEN_VECTOR);
sc->sc_cbvec = 1;
return (0);
}
void
xen_intr(void)
{
/* stub */
}