mirror of
https://github.com/genodelabs/genode.git
synced 2025-01-05 13:34:11 +00:00
104 lines
2.9 KiB
C
104 lines
2.9 KiB
C
|
/*
|
||
|
* \brief Genode specific VirtualBox SUPLib supplements
|
||
|
* \author Norman Feske
|
||
|
* \author Alexander Boettcher
|
||
|
*/
|
||
|
|
||
|
/*
|
||
|
* Copyright (C) 2013-2019 Genode Labs GmbH
|
||
|
*
|
||
|
* This file is distributed under the terms of the GNU General Public License
|
||
|
* version 2.
|
||
|
*/
|
||
|
|
||
|
#ifndef _VIRTUALBOX__VMX_H_
|
||
|
#define _VIRTUALBOX__VMX_H_
|
||
|
|
||
|
#define GENODE_READ_SELREG_REQUIRED(REG) \
|
||
|
(pCtx->REG.Sel != state->REG.value().sel) || \
|
||
|
(pCtx->REG.ValidSel != state->REG.value().sel) || \
|
||
|
(pCtx->REG.fFlags != CPUMSELREG_FLAGS_VALID) || \
|
||
|
(pCtx->REG.u32Limit != state->REG.value().limit) || \
|
||
|
(pCtx->REG.u64Base != state->REG.value().base) || \
|
||
|
(pCtx->REG.Attr.u != sel_ar_conv_from_genode(state->REG.value().ar))
|
||
|
|
||
|
#define GENODE_READ_SELREG(REG) \
|
||
|
pCtx->REG.Sel = state->REG.value().sel; \
|
||
|
pCtx->REG.ValidSel = state->REG.value().sel; \
|
||
|
pCtx->REG.fFlags = CPUMSELREG_FLAGS_VALID; \
|
||
|
pCtx->REG.u32Limit = state->REG.value().limit; \
|
||
|
pCtx->REG.u64Base = state->REG.value().base; \
|
||
|
pCtx->REG.Attr.u = sel_ar_conv_from_genode(state->REG.value().ar)
|
||
|
|
||
|
static inline bool vmx_save_state(Genode::Vm_state * state, VM * pVM, PVMCPU pVCpu)
|
||
|
{
|
||
|
PCPUMCTX pCtx = CPUMQueryGuestCtxPtr(pVCpu);
|
||
|
|
||
|
GENODE_READ_SELREG(cs);
|
||
|
GENODE_READ_SELREG(ds);
|
||
|
GENODE_READ_SELREG(es);
|
||
|
GENODE_READ_SELREG(fs);
|
||
|
GENODE_READ_SELREG(gs);
|
||
|
GENODE_READ_SELREG(ss);
|
||
|
|
||
|
if (GENODE_READ_SELREG_REQUIRED(ldtr)) {
|
||
|
GENODE_READ_SELREG(ldtr);
|
||
|
CPUMSetChangedFlags(pVCpu, CPUM_CHANGED_LDTR);
|
||
|
}
|
||
|
if (GENODE_READ_SELREG_REQUIRED(tr)) {
|
||
|
GENODE_READ_SELREG(tr);
|
||
|
CPUMSetChangedFlags(pVCpu, CPUM_CHANGED_TR);
|
||
|
}
|
||
|
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
#undef GENODE_READ_SELREG_REQUIRED
|
||
|
#undef GENODE_READ_SELREG
|
||
|
|
||
|
|
||
|
enum { VMCS_SEG_UNUSABLE = 0x10000 };
|
||
|
|
||
|
#define GENODE_WRITE_SELREG(REG) \
|
||
|
Assert(pCtx->REG.fFlags & CPUMSELREG_FLAGS_VALID); \
|
||
|
Assert(pCtx->REG.ValidSel == pCtx->REG.Sel); \
|
||
|
state->REG.value(Segment{pCtx->REG.Sel, \
|
||
|
sel_ar_conv_to_genode(pCtx->REG.Attr.u ? : VMCS_SEG_UNUSABLE), \
|
||
|
pCtx->REG.u32Limit, \
|
||
|
pCtx->REG.u64Base});
|
||
|
|
||
|
static inline bool vmx_load_state(Genode::Vm_state * state, VM * pVM, PVMCPU pVCpu)
|
||
|
{
|
||
|
PCPUMCTX pCtx = CPUMQueryGuestCtxPtr(pVCpu);
|
||
|
|
||
|
typedef Genode::Vm_state::Segment Segment;
|
||
|
|
||
|
GENODE_WRITE_SELREG(es);
|
||
|
GENODE_WRITE_SELREG(ds);
|
||
|
|
||
|
GENODE_WRITE_SELREG(fs);
|
||
|
GENODE_WRITE_SELREG(gs);
|
||
|
|
||
|
GENODE_WRITE_SELREG(cs);
|
||
|
GENODE_WRITE_SELREG(ss);
|
||
|
|
||
|
/* ldtr */
|
||
|
if (pCtx->ldtr.Sel == 0) {
|
||
|
state->ldtr.value(Segment{0, sel_ar_conv_to_genode(0x82), 0, 0});
|
||
|
} else {
|
||
|
state->ldtr.value(Segment{pCtx->ldtr.Sel,
|
||
|
sel_ar_conv_to_genode(pCtx->ldtr.Attr.u),
|
||
|
pCtx->ldtr.u32Limit, pCtx->ldtr.u64Base});
|
||
|
}
|
||
|
|
||
|
/* tr */
|
||
|
state->tr.value(Segment{pCtx->tr.Sel, sel_ar_conv_to_genode(pCtx->tr.Attr.u),
|
||
|
pCtx->tr.u32Limit, pCtx->tr.u64Base});
|
||
|
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
#undef GENODE_WRITE_SELREG
|
||
|
|
||
|
#endif /* _VIRTUALBOX__VMX_H_ */
|