2019-05-06 08:34:47 +00:00
|
|
|
|
/*
|
|
|
|
|
* \brief Genode VirtualBox SUPLib supplements
|
|
|
|
|
* \author Alexander Boettcher
|
|
|
|
|
* \author Norman Feske
|
|
|
|
|
* \author Christian Helmuth
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
/*
|
2020-12-18 13:08:06 +00:00
|
|
|
|
* Copyright (C) 2013-2021 Genode Labs GmbH
|
2019-05-06 08:34:47 +00:00
|
|
|
|
*
|
|
|
|
|
* This file is distributed under the terms of the GNU General Public License
|
|
|
|
|
* version 2.
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
#ifndef _VIRTUALBOX__VCPU_H_
|
|
|
|
|
#define _VIRTUALBOX__VCPU_H_
|
|
|
|
|
|
|
|
|
|
/* Genode includes */
|
|
|
|
|
#include <base/log.h>
|
|
|
|
|
#include <rom_session/connection.h>
|
|
|
|
|
#include <timer_session/connection.h>
|
|
|
|
|
#include <vm_session/connection.h>
|
2020-12-18 13:08:06 +00:00
|
|
|
|
#include <cpu/vcpu_state.h>
|
2019-05-06 08:34:47 +00:00
|
|
|
|
|
|
|
|
|
/* VirtualBox includes */
|
|
|
|
|
#include "PGMInternal.h" /* enable access to pgm.s.* */
|
|
|
|
|
|
|
|
|
|
#include "HMInternal.h" /* enable access to hm.s.* */
|
|
|
|
|
#include "CPUMInternal.h" /* enable access to cpum.s.* */
|
|
|
|
|
|
|
|
|
|
#include <VBox/vmm/vm.h>
|
|
|
|
|
#include <VBox/vmm/hm_svm.h>
|
|
|
|
|
#include <VBox/err.h>
|
|
|
|
|
|
|
|
|
|
#include <VBox/vmm/pdmapi.h>
|
|
|
|
|
|
|
|
|
|
#include <iprt/time.h>
|
|
|
|
|
|
|
|
|
|
/* Genode's VirtualBox includes */
|
|
|
|
|
#include "sup.h"
|
|
|
|
|
|
|
|
|
|
/* Genode libc pthread binding */
|
libc: split task.cc into multiple files
This patch is the first step of re-organizing the internal structure of
the libc. The original version involved many direct calls of global
functions (often with side effects) across compilation units, which
made the control flow (e.g., the initialization sequence) hard to
follow.
The new version replaces those ad-hoc interactions with dedicated
interfaces (like suspend.h, resume.h, select.h, current_time.h). The
underlying facilities are provided by the central Libc::Kernel and
selectively propagated to the various compilation units. The latter is
done by a sequence of 'init_*' calls, which eventually will be replaced
by constructor calls.
The addition of new headers increases the chance for name clashes with
existing (public) headers. To disambiguate libc-internal header files
from public headers, this patch moves the former into a new 'internal/'
subdirectory. This makes the include directives easier to follow and the
libc's source-tree structure more tidy.
There are still a few legacies left, which cannot easily be removed
right now (e.g., because noux relies on them). However, the patch moves
those bad apples to legacy.h and legacy.cc, which highlights the
deprecation of those functions.
Issue #3497
2019-09-18 18:19:10 +00:00
|
|
|
|
#include <internal/pthread.h>
|
2019-05-06 08:34:47 +00:00
|
|
|
|
|
|
|
|
|
#include <VBox/vmm/rem.h>
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* VirtualBox stores segment attributes in Intel format using a 32-bit
|
|
|
|
|
* value. Genode represents the attributes in packed format using a 16-bit
|
|
|
|
|
* value.
|
|
|
|
|
*/
|
|
|
|
|
static inline Genode::uint16_t sel_ar_conv_to_genode(Genode::uint32_t v)
|
|
|
|
|
{
|
|
|
|
|
return (v & 0xff) | ((v & 0x1f000) >> 4);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
static inline Genode::uint32_t sel_ar_conv_from_genode(Genode::uint16_t v)
|
|
|
|
|
{
|
|
|
|
|
return (v & 0xff) | (((uint32_t )v << 4) & 0x1f000);
|
|
|
|
|
}
|
|
|
|
|
|
2019-05-22 08:41:30 +00:00
|
|
|
|
class Vcpu_handler : public Genode::List<Vcpu_handler>::Element
|
2019-05-06 08:34:47 +00:00
|
|
|
|
{
|
|
|
|
|
protected:
|
|
|
|
|
|
2020-12-18 13:08:06 +00:00
|
|
|
|
static Genode::Vm_connection::Exit_config const _exit_config;
|
|
|
|
|
|
2019-05-06 08:34:47 +00:00
|
|
|
|
Genode::Entrypoint _ep;
|
2020-07-06 09:05:38 +00:00
|
|
|
|
Genode::Blockade _blockade_emt { };
|
2019-05-22 08:41:30 +00:00
|
|
|
|
Genode::Semaphore _sem_handler;
|
2020-12-18 13:08:06 +00:00
|
|
|
|
Genode::Vcpu_state *_state { nullptr };
|
2019-05-06 08:34:47 +00:00
|
|
|
|
|
2020-07-21 09:47:25 +00:00
|
|
|
|
pthread_cond_t _cond_wait;
|
|
|
|
|
pthread_mutex_t _mutex;
|
|
|
|
|
|
2019-05-06 08:34:47 +00:00
|
|
|
|
|
|
|
|
|
/* information used for NPT/EPT handling */
|
|
|
|
|
Genode::addr_t npt_ept_exit_addr { 0 };
|
|
|
|
|
RTGCUINT npt_ept_errorcode { 0 };
|
|
|
|
|
bool npt_ept_unmap { false };
|
|
|
|
|
|
|
|
|
|
/* state machine between EMT and EP thread of a vCPU */
|
2019-05-22 08:41:30 +00:00
|
|
|
|
enum { RUNNING, PAUSED, IRQ_WIN, NPT_EPT } _vm_state { PAUSED };
|
|
|
|
|
enum { PAUSE_EXIT, RUN } _next_state { RUN };
|
2019-05-06 08:34:47 +00:00
|
|
|
|
|
|
|
|
|
private:
|
|
|
|
|
|
|
|
|
|
bool _irq_win = false;
|
|
|
|
|
|
|
|
|
|
unsigned const _cpu_id;
|
|
|
|
|
PVM _vm { nullptr };
|
|
|
|
|
PVMCPU _vcpu { nullptr };
|
|
|
|
|
|
|
|
|
|
unsigned int _last_inj_info = 0;
|
|
|
|
|
unsigned int _last_inj_error = 0;
|
|
|
|
|
|
|
|
|
|
enum {
|
|
|
|
|
REQ_IRQWIN_EXIT = 0x1000U,
|
|
|
|
|
IRQ_INJ_VALID_MASK = 0x80000000UL,
|
|
|
|
|
IRQ_INJ_NONE = 0U,
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* Intel® 64 and IA-32 Architectures Software Developer’s Manual
|
|
|
|
|
* Volume 3C, Chapter 24.4.2.
|
|
|
|
|
* May 2012
|
|
|
|
|
*/
|
|
|
|
|
BLOCKING_BY_STI = 1U << 0,
|
|
|
|
|
BLOCKING_BY_MOV_SS = 1U << 1,
|
|
|
|
|
ACTIVITY_STATE_ACTIVE = 0U,
|
|
|
|
|
INTERRUPT_STATE_NONE = 0U,
|
|
|
|
|
};
|
|
|
|
|
|
2020-07-21 09:47:25 +00:00
|
|
|
|
timespec add_timespec_ns(timespec a, uint64_t ns) const
|
|
|
|
|
{
|
|
|
|
|
enum { NSEC_PER_SEC = 1'000'000'000ull };
|
|
|
|
|
|
|
|
|
|
long sec = a.tv_sec;
|
|
|
|
|
|
|
|
|
|
while (a.tv_nsec >= NSEC_PER_SEC) {
|
|
|
|
|
a.tv_nsec -= NSEC_PER_SEC;
|
|
|
|
|
sec++;
|
|
|
|
|
}
|
|
|
|
|
while (ns >= NSEC_PER_SEC) {
|
|
|
|
|
ns -= NSEC_PER_SEC;
|
|
|
|
|
sec++;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
long nsec = a.tv_nsec + ns;
|
|
|
|
|
while (nsec >= NSEC_PER_SEC) {
|
|
|
|
|
nsec -= NSEC_PER_SEC;
|
|
|
|
|
sec++;
|
|
|
|
|
}
|
|
|
|
|
return timespec { sec, nsec };
|
|
|
|
|
}
|
|
|
|
|
|
2019-05-06 08:34:47 +00:00
|
|
|
|
protected:
|
|
|
|
|
|
|
|
|
|
int map_memory(Genode::Vm_connection &vm_session,
|
|
|
|
|
RTGCPHYS GCPhys, RTGCUINT vbox_fault_reason);
|
|
|
|
|
|
|
|
|
|
Genode::addr_t _vm_exits = 0;
|
|
|
|
|
Genode::addr_t _recall_skip = 0;
|
|
|
|
|
Genode::addr_t _recall_req = 0;
|
|
|
|
|
Genode::addr_t _recall_inv = 0;
|
|
|
|
|
Genode::addr_t _recall_drop = 0;
|
|
|
|
|
Genode::addr_t _irq_request = 0;
|
|
|
|
|
Genode::addr_t _irq_inject = 0;
|
|
|
|
|
Genode::addr_t _irq_drop = 0;
|
|
|
|
|
|
|
|
|
|
struct {
|
|
|
|
|
unsigned intr_state;
|
|
|
|
|
unsigned ctrl[2];
|
|
|
|
|
} next_utcb;
|
|
|
|
|
|
|
|
|
|
unsigned _ept_fault_addr_type;
|
|
|
|
|
|
|
|
|
|
Genode::uint64_t * pdpte_map(VM *pVM, RTGCPHYS cr3);
|
|
|
|
|
|
2019-05-21 07:25:17 +00:00
|
|
|
|
void switch_to_hw(PCPUMCTX pCtx)
|
2019-05-06 08:34:47 +00:00
|
|
|
|
{
|
2020-12-18 13:08:06 +00:00
|
|
|
|
using Genode::Vcpu_state;
|
|
|
|
|
|
2019-05-06 08:34:47 +00:00
|
|
|
|
again:
|
|
|
|
|
|
2019-05-21 07:25:17 +00:00
|
|
|
|
/* write FPU state */
|
2020-12-18 13:08:06 +00:00
|
|
|
|
AssertCompile(sizeof(Vcpu_state::Fpu::State) >= sizeof(X86FXSTATE));
|
|
|
|
|
_state->fpu.charge([&] (Vcpu_state::Fpu::State &fpu) {
|
|
|
|
|
::memcpy(&fpu, pCtx->pXStateR3, sizeof(X86FXSTATE));
|
2019-05-21 07:25:17 +00:00
|
|
|
|
});
|
|
|
|
|
|
2019-05-22 08:41:30 +00:00
|
|
|
|
Assert(_vm_state == IRQ_WIN || _vm_state == PAUSED || _vm_state == NPT_EPT);
|
|
|
|
|
Assert(_next_state == PAUSE_EXIT || _next_state == RUN);
|
|
|
|
|
|
|
|
|
|
/* wake up vcpu ep handler */
|
|
|
|
|
_sem_handler.up();
|
2019-05-06 08:34:47 +00:00
|
|
|
|
|
|
|
|
|
/* wait for next exit */
|
2020-07-06 09:05:38 +00:00
|
|
|
|
_blockade_emt.block();
|
2019-05-22 08:41:30 +00:00
|
|
|
|
|
|
|
|
|
/* next time run - recall() may change this */
|
|
|
|
|
_next_state = RUN;
|
2019-05-06 08:34:47 +00:00
|
|
|
|
|
2019-05-21 07:25:17 +00:00
|
|
|
|
/* write FPU state of vCPU to pCtx */
|
2020-12-18 13:08:06 +00:00
|
|
|
|
_state->fpu.with_state([&] (Vcpu_state::Fpu::State const &fpu) {
|
|
|
|
|
::memcpy(pCtx->pXStateR3, &fpu, sizeof(X86FXSTATE));
|
2019-05-21 07:25:17 +00:00
|
|
|
|
});
|
|
|
|
|
|
2019-05-06 08:34:47 +00:00
|
|
|
|
if (_vm_state == IRQ_WIN) {
|
2020-12-18 13:08:06 +00:00
|
|
|
|
_state->discharge();
|
2019-05-06 08:34:47 +00:00
|
|
|
|
_irq_window_pthread();
|
|
|
|
|
goto again;
|
|
|
|
|
} else
|
|
|
|
|
if (_vm_state == NPT_EPT) {
|
|
|
|
|
if (npt_ept_unmap) {
|
|
|
|
|
Genode::error("NPT/EPT unmap not supported - stop");
|
|
|
|
|
while (true) {
|
2020-07-06 09:05:38 +00:00
|
|
|
|
_blockade_emt.block();
|
2019-05-06 08:34:47 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Genode::addr_t const gp_map_addr = npt_ept_exit_addr & ~((1UL << 12) - 1);
|
|
|
|
|
int res = attach_memory_to_vm(gp_map_addr, npt_ept_errorcode);
|
|
|
|
|
if (res == VINF_SUCCESS) {
|
2020-12-18 13:08:06 +00:00
|
|
|
|
_state->discharge();
|
2019-05-06 08:34:47 +00:00
|
|
|
|
goto again;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!(_vm_state == PAUSED || _vm_state == NPT_EPT))
|
|
|
|
|
Genode::error("which state we are ? ", (int)_vm_state, " ", Genode::Thread::myself()->name());
|
|
|
|
|
|
|
|
|
|
Assert(_vm_state == PAUSED || _vm_state == NPT_EPT);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void _default_handler()
|
|
|
|
|
{
|
|
|
|
|
if (_vm_state != RUNNING)
|
|
|
|
|
Genode::error(__func__, " _vm_state=", (int)_vm_state, " exit_reason=", Genode::Hex(_state->exit_reason));
|
|
|
|
|
Assert(_vm_state == RUNNING);
|
|
|
|
|
|
|
|
|
|
Assert(_state->actv_state.value() == ACTIVITY_STATE_ACTIVE);
|
|
|
|
|
Assert(!(_state->inj_info.value() & IRQ_INJ_VALID_MASK));
|
|
|
|
|
|
|
|
|
|
_vm_exits ++;
|
|
|
|
|
|
|
|
|
|
_vm_state = PAUSED;
|
|
|
|
|
|
2020-07-06 09:05:38 +00:00
|
|
|
|
_blockade_emt.wakeup();
|
2019-05-06 08:34:47 +00:00
|
|
|
|
}
|
|
|
|
|
|
2019-05-22 08:41:30 +00:00
|
|
|
|
bool _recall_handler()
|
2019-05-06 08:34:47 +00:00
|
|
|
|
{
|
|
|
|
|
if (_vm_state != RUNNING)
|
|
|
|
|
Genode::error(__func__, " _vm_state=", (int)_vm_state, " exit_reason=", Genode::Hex(_state->exit_reason));
|
|
|
|
|
Assert(_vm_state == RUNNING);
|
|
|
|
|
|
|
|
|
|
_vm_exits ++;
|
|
|
|
|
_recall_inv ++;
|
|
|
|
|
|
|
|
|
|
Assert(_state->actv_state.value() == ACTIVITY_STATE_ACTIVE);
|
|
|
|
|
|
|
|
|
|
if (_state->inj_info.value() & IRQ_INJ_VALID_MASK) {
|
|
|
|
|
|
|
|
|
|
Assert(_state->flags.value() & X86_EFL_IF);
|
|
|
|
|
|
|
|
|
|
if (_state->intr_state.value() != INTERRUPT_STATE_NONE)
|
|
|
|
|
Genode::log("intr state ", Genode::Hex(_state->intr_state.value()),
|
|
|
|
|
" ", Genode::Hex(_state->intr_state.value() & 0xf));
|
|
|
|
|
|
|
|
|
|
Assert(_state->intr_state.value() == INTERRUPT_STATE_NONE);
|
|
|
|
|
|
|
|
|
|
if (!continue_hw_accelerated())
|
|
|
|
|
_recall_drop ++;
|
|
|
|
|
|
|
|
|
|
/* got recall during irq injection and the guest is ready for
|
|
|
|
|
* delivery of IRQ - just continue */
|
2019-05-22 08:41:30 +00:00
|
|
|
|
return /* no-wait */ false;
|
2019-05-06 08:34:47 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* are we forced to go back to emulation mode ? */
|
|
|
|
|
if (!continue_hw_accelerated()) {
|
|
|
|
|
/* go back to emulation mode */
|
|
|
|
|
_default_handler();
|
2019-05-22 08:41:30 +00:00
|
|
|
|
return /* wait */ true;
|
2019-05-06 08:34:47 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* check whether we have to request irq injection window */
|
|
|
|
|
if (check_to_request_irq_window(_vcpu)) {
|
2020-12-18 13:08:06 +00:00
|
|
|
|
_state->discharge();
|
|
|
|
|
_state->inj_info.charge(_state->inj_info.value());
|
2019-05-06 08:34:47 +00:00
|
|
|
|
_irq_win = true;
|
2019-05-22 08:41:30 +00:00
|
|
|
|
return /* no-wait */ false;
|
2019-05-06 08:34:47 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
_default_handler();
|
2019-05-22 08:41:30 +00:00
|
|
|
|
return /* wait */ true;
|
2019-05-06 08:34:47 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
inline bool vbox_to_state(VM *pVM, PVMCPU pVCpu)
|
|
|
|
|
{
|
2020-12-18 13:08:06 +00:00
|
|
|
|
typedef Genode::Vcpu_state::Range Range;
|
2019-05-06 08:34:47 +00:00
|
|
|
|
|
|
|
|
|
PCPUMCTX pCtx = CPUMQueryGuestCtxPtr(pVCpu);
|
|
|
|
|
|
2020-12-18 13:08:06 +00:00
|
|
|
|
_state->ip.charge(pCtx->rip);
|
|
|
|
|
_state->sp.charge(pCtx->rsp);
|
2019-05-06 08:34:47 +00:00
|
|
|
|
|
2020-12-18 13:08:06 +00:00
|
|
|
|
_state->ax.charge(pCtx->rax);
|
|
|
|
|
_state->bx.charge(pCtx->rbx);
|
|
|
|
|
_state->cx.charge(pCtx->rcx);
|
|
|
|
|
_state->dx.charge(pCtx->rdx);
|
2019-05-06 08:34:47 +00:00
|
|
|
|
|
2020-12-18 13:08:06 +00:00
|
|
|
|
_state->bp.charge(pCtx->rbp);
|
|
|
|
|
_state->si.charge(pCtx->rsi);
|
|
|
|
|
_state->di.charge(pCtx->rdi);
|
2019-05-06 08:34:47 +00:00
|
|
|
|
|
2020-12-18 13:08:06 +00:00
|
|
|
|
_state->r8.charge(pCtx->r8);
|
|
|
|
|
_state->r9.charge(pCtx->r9);
|
|
|
|
|
_state->r10.charge(pCtx->r10);
|
|
|
|
|
_state->r11.charge(pCtx->r11);
|
|
|
|
|
_state->r12.charge(pCtx->r12);
|
|
|
|
|
_state->r13.charge(pCtx->r13);
|
|
|
|
|
_state->r14.charge(pCtx->r14);
|
|
|
|
|
_state->r15.charge(pCtx->r15);
|
2019-05-06 08:34:47 +00:00
|
|
|
|
|
2020-12-18 13:08:06 +00:00
|
|
|
|
_state->flags.charge(pCtx->rflags.u);
|
2019-05-06 08:34:47 +00:00
|
|
|
|
|
2020-12-18 13:08:06 +00:00
|
|
|
|
_state->sysenter_cs.charge(pCtx->SysEnter.cs);
|
|
|
|
|
_state->sysenter_sp.charge(pCtx->SysEnter.esp);
|
|
|
|
|
_state->sysenter_ip.charge(pCtx->SysEnter.eip);
|
2019-05-06 08:34:47 +00:00
|
|
|
|
|
2020-12-18 13:08:06 +00:00
|
|
|
|
_state->dr7.charge(pCtx->dr[7]);
|
2019-05-06 08:34:47 +00:00
|
|
|
|
|
2020-12-18 13:08:06 +00:00
|
|
|
|
_state->cr0.charge(pCtx->cr0);
|
|
|
|
|
_state->cr2.charge(pCtx->cr2);
|
|
|
|
|
_state->cr3.charge(pCtx->cr3);
|
|
|
|
|
_state->cr4.charge(pCtx->cr4);
|
2019-05-06 08:34:47 +00:00
|
|
|
|
|
2020-12-18 13:08:06 +00:00
|
|
|
|
_state->idtr.charge(Range { .limit = pCtx->idtr.cbIdt,
|
|
|
|
|
.base = pCtx->idtr.pIdt });
|
|
|
|
|
_state->gdtr.charge(Range { .limit = pCtx->gdtr.cbGdt,
|
|
|
|
|
.base = pCtx->gdtr.pGdt });
|
2019-05-06 08:34:47 +00:00
|
|
|
|
|
2020-12-18 13:08:06 +00:00
|
|
|
|
_state->efer.charge(CPUMGetGuestEFER(pVCpu));
|
2019-05-06 08:34:47 +00:00
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* Update the PDPTE registers if necessary
|
|
|
|
|
*
|
|
|
|
|
* Intel manual sections 4.4.1 of Vol. 3A and 26.3.2.4 of Vol. 3C
|
|
|
|
|
* indicate the conditions when this is the case. The following
|
|
|
|
|
* code currently does not check if the recompiler modified any
|
|
|
|
|
* CR registers, which means the update can happen more often
|
|
|
|
|
* than really necessary.
|
|
|
|
|
*/
|
|
|
|
|
if (pVM->hm.s.vmx.fSupported &&
|
|
|
|
|
CPUMIsGuestPagingEnabledEx(pCtx) &&
|
|
|
|
|
CPUMIsGuestInPAEModeEx(pCtx)) {
|
|
|
|
|
|
|
|
|
|
Genode::uint64_t *pdpte = pdpte_map(pVM, pCtx->cr3);
|
|
|
|
|
|
2020-12-18 13:08:06 +00:00
|
|
|
|
_state->pdpte_0.charge(pdpte[0]);
|
|
|
|
|
_state->pdpte_1.charge(pdpte[1]);
|
|
|
|
|
_state->pdpte_2.charge(pdpte[2]);
|
|
|
|
|
_state->pdpte_3.charge(pdpte[3]);
|
2019-05-06 08:34:47 +00:00
|
|
|
|
}
|
|
|
|
|
|
2020-12-18 13:08:06 +00:00
|
|
|
|
_state->star.charge(pCtx->msrSTAR);
|
|
|
|
|
_state->lstar.charge(pCtx->msrLSTAR);
|
|
|
|
|
_state->cstar.charge(pCtx->msrCSTAR);
|
|
|
|
|
_state->fmask.charge(pCtx->msrSFMASK);
|
|
|
|
|
_state->kernel_gs_base.charge(pCtx->msrKERNELGSBASE);
|
2019-05-06 08:34:47 +00:00
|
|
|
|
|
|
|
|
|
/* from HMVMXR0.cpp */
|
|
|
|
|
bool interrupt_pending = false;
|
|
|
|
|
uint8_t tpr = 0;
|
|
|
|
|
uint8_t pending_interrupt = 0;
|
|
|
|
|
PDMApicGetTPR(pVCpu, &tpr, &interrupt_pending, &pending_interrupt);
|
|
|
|
|
|
2020-12-18 13:08:06 +00:00
|
|
|
|
_state->tpr.charge(tpr);
|
|
|
|
|
_state->tpr_threshold.charge(0);
|
2019-05-06 08:34:47 +00:00
|
|
|
|
|
|
|
|
|
if (interrupt_pending) {
|
|
|
|
|
const uint8_t pending_priority = (pending_interrupt >> 4) & 0xf;
|
|
|
|
|
const uint8_t tpr_priority = (tpr >> 4) & 0xf;
|
|
|
|
|
if (pending_priority <= tpr_priority)
|
2020-12-18 13:08:06 +00:00
|
|
|
|
_state->tpr_threshold.charge(pending_priority);
|
2019-05-06 08:34:47 +00:00
|
|
|
|
else
|
2020-12-18 13:08:06 +00:00
|
|
|
|
_state->tpr_threshold.charge(tpr_priority);
|
2019-05-06 08:34:47 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
inline bool state_to_vbox(VM *pVM, PVMCPU pVCpu)
|
|
|
|
|
{
|
|
|
|
|
PCPUMCTX pCtx = CPUMQueryGuestCtxPtr(pVCpu);
|
|
|
|
|
|
|
|
|
|
pCtx->rip = _state->ip.value();
|
|
|
|
|
pCtx->rsp = _state->sp.value();
|
|
|
|
|
|
|
|
|
|
pCtx->rax = _state->ax.value();
|
|
|
|
|
pCtx->rbx = _state->bx.value();
|
|
|
|
|
pCtx->rcx = _state->cx.value();
|
|
|
|
|
pCtx->rdx = _state->dx.value();
|
|
|
|
|
|
|
|
|
|
pCtx->rbp = _state->bp.value();
|
|
|
|
|
pCtx->rsi = _state->si.value();
|
|
|
|
|
pCtx->rdi = _state->di.value();
|
|
|
|
|
pCtx->rflags.u = _state->flags.value();
|
|
|
|
|
|
|
|
|
|
pCtx->r8 = _state->r8.value();
|
|
|
|
|
pCtx->r9 = _state->r9.value();
|
|
|
|
|
pCtx->r10 = _state->r10.value();
|
|
|
|
|
pCtx->r11 = _state->r11.value();
|
|
|
|
|
pCtx->r12 = _state->r12.value();
|
|
|
|
|
pCtx->r13 = _state->r13.value();
|
|
|
|
|
pCtx->r14 = _state->r14.value();
|
|
|
|
|
pCtx->r15 = _state->r15.value();
|
|
|
|
|
|
|
|
|
|
pCtx->dr[7] = _state->dr7.value();
|
|
|
|
|
|
|
|
|
|
if (pCtx->SysEnter.cs != _state->sysenter_cs.value())
|
|
|
|
|
CPUMSetGuestMsr(pVCpu, MSR_IA32_SYSENTER_CS, _state->sysenter_cs.value());
|
|
|
|
|
|
|
|
|
|
if (pCtx->SysEnter.esp != _state->sysenter_sp.value())
|
|
|
|
|
CPUMSetGuestMsr(pVCpu, MSR_IA32_SYSENTER_ESP, _state->sysenter_sp.value());
|
|
|
|
|
|
|
|
|
|
if (pCtx->SysEnter.eip != _state->sysenter_ip.value())
|
|
|
|
|
CPUMSetGuestMsr(pVCpu, MSR_IA32_SYSENTER_EIP, _state->sysenter_ip.value());
|
|
|
|
|
|
|
|
|
|
if (pCtx->idtr.cbIdt != _state->idtr.value().limit ||
|
|
|
|
|
pCtx->idtr.pIdt != _state->idtr.value().base)
|
|
|
|
|
CPUMSetGuestIDTR(pVCpu, _state->idtr.value().base, _state->idtr.value().limit);
|
|
|
|
|
|
|
|
|
|
if (pCtx->gdtr.cbGdt != _state->gdtr.value().limit ||
|
|
|
|
|
pCtx->gdtr.pGdt != _state->gdtr.value().base)
|
|
|
|
|
CPUMSetGuestGDTR(pVCpu, _state->gdtr.value().base, _state->gdtr.value().limit);
|
|
|
|
|
|
|
|
|
|
CPUMSetGuestEFER(pVCpu, _state->efer.value());
|
|
|
|
|
|
|
|
|
|
if (pCtx->cr0 != _state->cr0.value())
|
|
|
|
|
CPUMSetGuestCR0(pVCpu, _state->cr0.value());
|
|
|
|
|
|
|
|
|
|
if (pCtx->cr2 != _state->cr2.value())
|
|
|
|
|
CPUMSetGuestCR2(pVCpu, _state->cr2.value());
|
|
|
|
|
|
|
|
|
|
if (pCtx->cr3 != _state->cr3.value()) {
|
|
|
|
|
CPUMSetGuestCR3(pVCpu, _state->cr3.value());
|
|
|
|
|
VMCPU_FF_SET(pVCpu, VMCPU_FF_HM_UPDATE_CR3);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (pCtx->cr4 != _state->cr4.value())
|
|
|
|
|
CPUMSetGuestCR4(pVCpu, _state->cr4.value());
|
|
|
|
|
|
|
|
|
|
if (pCtx->msrSTAR != _state->star.value())
|
|
|
|
|
CPUMSetGuestMsr(pVCpu, MSR_K6_STAR, _state->star.value());
|
|
|
|
|
|
|
|
|
|
if (pCtx->msrLSTAR != _state->lstar.value())
|
|
|
|
|
CPUMSetGuestMsr(pVCpu, MSR_K8_LSTAR, _state->lstar.value());
|
|
|
|
|
|
2020-11-28 21:05:47 +00:00
|
|
|
|
if (pCtx->msrCSTAR != _state->cstar.value())
|
|
|
|
|
CPUMSetGuestMsr(pVCpu, MSR_K8_CSTAR, _state->cstar.value());
|
|
|
|
|
|
2019-05-06 08:34:47 +00:00
|
|
|
|
if (pCtx->msrSFMASK != _state->fmask.value())
|
|
|
|
|
CPUMSetGuestMsr(pVCpu, MSR_K8_SF_MASK, _state->fmask.value());
|
|
|
|
|
|
|
|
|
|
if (pCtx->msrKERNELGSBASE != _state->kernel_gs_base.value())
|
|
|
|
|
CPUMSetGuestMsr(pVCpu, MSR_K8_KERNEL_GS_BASE, _state->kernel_gs_base.value());
|
|
|
|
|
|
|
|
|
|
const uint32_t tpr = _state->tpr.value();
|
|
|
|
|
|
|
|
|
|
/* reset message transfer descriptor for next invocation */
|
|
|
|
|
Assert (!(_state->inj_info.value() & IRQ_INJ_VALID_MASK));
|
|
|
|
|
next_utcb.intr_state = _state->intr_state.value();
|
|
|
|
|
next_utcb.ctrl[0] = _state->ctrl_primary.value();
|
|
|
|
|
next_utcb.ctrl[1] = _state->ctrl_secondary.value();
|
|
|
|
|
|
|
|
|
|
if (next_utcb.intr_state & 3) {
|
|
|
|
|
next_utcb.intr_state &= ~3U;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
VMCPU_FF_CLEAR(pVCpu, VMCPU_FF_TO_R3);
|
|
|
|
|
|
|
|
|
|
CPUMSetChangedFlags(pVCpu, CPUM_CHANGED_FPU_REM);
|
|
|
|
|
pVCpu->cpum.s.fUseFlags |= (CPUM_USED_FPU_GUEST | CPUM_USED_FPU_SINCE_REM);
|
|
|
|
|
|
|
|
|
|
if (_state->intr_state.value() != 0) {
|
|
|
|
|
Assert(_state->intr_state.value() == BLOCKING_BY_STI ||
|
|
|
|
|
_state->intr_state.value() == BLOCKING_BY_MOV_SS);
|
|
|
|
|
EMSetInhibitInterruptsPC(pVCpu, pCtx->rip);
|
|
|
|
|
} else
|
|
|
|
|
VMCPU_FF_CLEAR(pVCpu, VMCPU_FF_INHIBIT_INTERRUPTS);
|
|
|
|
|
|
|
|
|
|
PDMApicSetTPR(pVCpu, tpr);
|
|
|
|
|
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
inline bool check_to_request_irq_window(PVMCPU pVCpu)
|
|
|
|
|
{
|
|
|
|
|
if (VMCPU_FF_IS_SET(pVCpu, VMCPU_FF_INHIBIT_INTERRUPTS))
|
|
|
|
|
return false;
|
|
|
|
|
|
|
|
|
|
if (!TRPMHasTrap(pVCpu) &&
|
|
|
|
|
!VMCPU_FF_IS_PENDING(pVCpu, (VMCPU_FF_INTERRUPT_APIC |
|
|
|
|
|
VMCPU_FF_INTERRUPT_PIC)))
|
|
|
|
|
return false;
|
|
|
|
|
|
|
|
|
|
_irq_request++;
|
|
|
|
|
|
|
|
|
|
unsigned const vector = 0;
|
2020-12-18 13:08:06 +00:00
|
|
|
|
_state->inj_info.charge(REQ_IRQWIN_EXIT | vector);
|
2019-05-06 08:34:47 +00:00
|
|
|
|
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void _irq_window()
|
|
|
|
|
{
|
|
|
|
|
if (_vm_state != RUNNING)
|
|
|
|
|
Genode::error(__func__, " _vm_state=", (int)_vm_state, " exit_reason=", Genode::Hex(_state->exit_reason));
|
|
|
|
|
Assert(_vm_state == RUNNING);
|
|
|
|
|
|
|
|
|
|
_vm_exits ++;
|
|
|
|
|
|
|
|
|
|
_vm_state = IRQ_WIN;
|
2020-07-06 09:05:38 +00:00
|
|
|
|
_blockade_emt.wakeup();
|
2019-05-06 08:34:47 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void _npt_ept()
|
|
|
|
|
{
|
|
|
|
|
if (_vm_state != RUNNING)
|
|
|
|
|
Genode::error(__func__, " _vm_state=", (int)_vm_state, " exit_reason=", Genode::Hex(_state->exit_reason));
|
|
|
|
|
Assert(_vm_state == RUNNING);
|
|
|
|
|
|
|
|
|
|
_vm_exits ++;
|
|
|
|
|
|
|
|
|
|
_vm_state = NPT_EPT;
|
2020-07-06 09:05:38 +00:00
|
|
|
|
_blockade_emt.wakeup();
|
2019-05-06 08:34:47 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void _irq_window_pthread()
|
|
|
|
|
{
|
|
|
|
|
PVMCPU pVCpu = _vcpu;
|
|
|
|
|
|
|
|
|
|
Assert(_state->intr_state.value() == INTERRUPT_STATE_NONE);
|
|
|
|
|
Assert(_state->flags.value() & X86_EFL_IF);
|
|
|
|
|
Assert(!VMCPU_FF_IS_SET(pVCpu, VMCPU_FF_INHIBIT_INTERRUPTS));
|
|
|
|
|
Assert(!(_state->inj_info.value() & IRQ_INJ_VALID_MASK));
|
|
|
|
|
|
|
|
|
|
Assert(_irq_win);
|
|
|
|
|
|
|
|
|
|
_irq_win = false;
|
|
|
|
|
|
|
|
|
|
/* request current tpr state from guest, it may block IRQs */
|
|
|
|
|
PDMApicSetTPR(pVCpu, _state->tpr_threshold.value());
|
|
|
|
|
|
|
|
|
|
if (!TRPMHasTrap(pVCpu)) {
|
|
|
|
|
|
|
|
|
|
bool res = VMCPU_FF_TEST_AND_CLEAR(pVCpu, VMCPU_FF_INTERRUPT_NMI);
|
|
|
|
|
if (res)
|
|
|
|
|
Genode::log("NMI was set");
|
|
|
|
|
|
|
|
|
|
if (VMCPU_FF_IS_PENDING(pVCpu, (VMCPU_FF_INTERRUPT_APIC |
|
|
|
|
|
VMCPU_FF_INTERRUPT_PIC))) {
|
|
|
|
|
|
|
|
|
|
uint8_t irq;
|
|
|
|
|
int rc = PDMGetInterrupt(pVCpu, &irq);
|
|
|
|
|
Assert(RT_SUCCESS(rc));
|
|
|
|
|
|
|
|
|
|
rc = TRPMAssertTrap(pVCpu, irq, TRPM_HARDWARE_INT);
|
|
|
|
|
Assert(RT_SUCCESS(rc));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!TRPMHasTrap(pVCpu)) {
|
|
|
|
|
_irq_drop++;
|
|
|
|
|
/* happens if PDMApicSetTPR (see above) mask IRQ */
|
2020-12-18 13:08:06 +00:00
|
|
|
|
_state->inj_info.charge(IRQ_INJ_NONE);
|
2019-05-06 08:34:47 +00:00
|
|
|
|
Genode::error("virq window pthread aaaaaaa while loop");
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
_irq_inject++;
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* If we have no IRQ for injection, something with requesting the
|
|
|
|
|
* IRQ window went wrong. Probably it was forgotten to be reset.
|
|
|
|
|
*/
|
|
|
|
|
Assert(TRPMHasTrap(pVCpu));
|
|
|
|
|
|
|
|
|
|
/* interrupt can be dispatched */
|
|
|
|
|
uint8_t u8Vector;
|
|
|
|
|
TRPMEVENT enmType;
|
|
|
|
|
SVMEVENT Event;
|
|
|
|
|
RTGCUINT u32ErrorCode;
|
|
|
|
|
RTGCUINTPTR GCPtrFaultAddress;
|
|
|
|
|
uint8_t cbInstr;
|
|
|
|
|
|
|
|
|
|
Event.u = 0;
|
|
|
|
|
|
|
|
|
|
/* If a new event is pending, then dispatch it now. */
|
|
|
|
|
int rc = TRPMQueryTrapAll(pVCpu, &u8Vector, &enmType, &u32ErrorCode, 0, 0);
|
|
|
|
|
AssertRC(rc);
|
|
|
|
|
Assert(enmType == TRPM_HARDWARE_INT);
|
|
|
|
|
Assert(u8Vector != X86_XCPT_NMI);
|
|
|
|
|
|
|
|
|
|
/* Clear the pending trap. */
|
|
|
|
|
rc = TRPMResetTrap(pVCpu);
|
|
|
|
|
AssertRC(rc);
|
|
|
|
|
|
|
|
|
|
Event.n.u8Vector = u8Vector;
|
|
|
|
|
Event.n.u1Valid = 1;
|
|
|
|
|
Event.n.u32ErrorCode = u32ErrorCode;
|
|
|
|
|
|
|
|
|
|
Event.n.u3Type = SVM_EVENT_EXTERNAL_IRQ;
|
|
|
|
|
|
2020-12-18 13:08:06 +00:00
|
|
|
|
_state->inj_info.charge(Event.u);
|
|
|
|
|
_state->inj_error.charge(Event.n.u32ErrorCode);
|
2019-05-06 08:34:47 +00:00
|
|
|
|
|
|
|
|
|
_last_inj_info = _state->inj_info.value();
|
|
|
|
|
_last_inj_error = _state->inj_error.value();
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
Genode::log("type:info:vector ", Genode::Hex(Event.n.u3Type),
|
|
|
|
|
Genode::Hex(utcb->inj_info), Genode::Hex(u8Vector),
|
|
|
|
|
" intr:actv - ", Genode::Hex(utcb->intr_state),
|
|
|
|
|
Genode::Hex(utcb->actv_state), " mtd ",
|
|
|
|
|
Genode::Hex(utcb->mtd));
|
|
|
|
|
*/
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
inline bool continue_hw_accelerated(bool verbose = false)
|
|
|
|
|
{
|
|
|
|
|
uint32_t check_vm = VM_FF_HM_TO_R3_MASK | VM_FF_REQUEST
|
|
|
|
|
| VM_FF_PGM_POOL_FLUSH_PENDING
|
|
|
|
|
| VM_FF_PDM_DMA;
|
|
|
|
|
uint32_t check_vcpu = VMCPU_FF_HM_TO_R3_MASK
|
|
|
|
|
| VMCPU_FF_PGM_SYNC_CR3
|
|
|
|
|
| VMCPU_FF_PGM_SYNC_CR3_NON_GLOBAL
|
|
|
|
|
| VMCPU_FF_REQUEST;
|
|
|
|
|
|
|
|
|
|
if (!VM_FF_IS_PENDING(_vm, check_vm) &&
|
|
|
|
|
!VMCPU_FF_IS_PENDING(_vcpu, check_vcpu))
|
|
|
|
|
return true;
|
|
|
|
|
|
|
|
|
|
Assert(!(VM_FF_IS_PENDING(_vm, VM_FF_PGM_NO_MEMORY)));
|
|
|
|
|
|
|
|
|
|
#define VERBOSE_VM(flag) \
|
|
|
|
|
do { \
|
|
|
|
|
if (VM_FF_IS_PENDING(_vm, flag)) \
|
|
|
|
|
Genode::log("flag ", flag, " pending"); \
|
|
|
|
|
} while (0)
|
|
|
|
|
|
|
|
|
|
#define VERBOSE_VMCPU(flag) \
|
|
|
|
|
do { \
|
|
|
|
|
if (VMCPU_FF_IS_PENDING(_vcpu, flag)) \
|
|
|
|
|
Genode::log("flag ", flag, " pending"); \
|
|
|
|
|
} while (0)
|
|
|
|
|
|
|
|
|
|
if (verbose) {
|
|
|
|
|
/*
|
|
|
|
|
* VM_FF_HM_TO_R3_MASK
|
|
|
|
|
*/
|
|
|
|
|
VERBOSE_VM(VM_FF_TM_VIRTUAL_SYNC);
|
|
|
|
|
VERBOSE_VM(VM_FF_PGM_NEED_HANDY_PAGES);
|
|
|
|
|
/* handled by the assertion above */
|
|
|
|
|
/* VERBOSE_VM(VM_FF_PGM_NO_MEMORY); */
|
|
|
|
|
VERBOSE_VM(VM_FF_PDM_QUEUES);
|
|
|
|
|
VERBOSE_VM(VM_FF_EMT_RENDEZVOUS);
|
|
|
|
|
|
|
|
|
|
VERBOSE_VM(VM_FF_REQUEST);
|
|
|
|
|
VERBOSE_VM(VM_FF_PGM_POOL_FLUSH_PENDING);
|
|
|
|
|
VERBOSE_VM(VM_FF_PDM_DMA);
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* VMCPU_FF_HM_TO_R3_MASK
|
|
|
|
|
*/
|
|
|
|
|
VERBOSE_VMCPU(VMCPU_FF_TO_R3);
|
|
|
|
|
/* when this flag gets set, a recall request follows */
|
|
|
|
|
/* VERBOSE_VMCPU(VMCPU_FF_TIMER); */
|
|
|
|
|
VERBOSE_VMCPU(VMCPU_FF_PDM_CRITSECT);
|
|
|
|
|
|
|
|
|
|
VERBOSE_VMCPU(VMCPU_FF_PGM_SYNC_CR3);
|
|
|
|
|
VERBOSE_VMCPU(VMCPU_FF_PGM_SYNC_CR3_NON_GLOBAL);
|
|
|
|
|
VERBOSE_VMCPU(VMCPU_FF_REQUEST);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#undef VERBOSE_VMCPU
|
|
|
|
|
#undef VERBOSE_VM
|
|
|
|
|
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
2020-12-18 13:08:06 +00:00
|
|
|
|
virtual bool hw_load_state(Genode::Vcpu_state *, VM *, PVMCPU) = 0;
|
|
|
|
|
virtual bool hw_save_state(Genode::Vcpu_state *, VM *, PVMCPU) = 0;
|
2019-05-06 08:34:47 +00:00
|
|
|
|
virtual int vm_exit_requires_instruction_emulation(PCPUMCTX) = 0;
|
|
|
|
|
|
|
|
|
|
virtual void pause_vm() = 0;
|
|
|
|
|
virtual int attach_memory_to_vm(RTGCPHYS const,
|
|
|
|
|
RTGCUINT vbox_fault_reason) = 0;
|
|
|
|
|
|
|
|
|
|
public:
|
|
|
|
|
|
|
|
|
|
enum Exit_condition
|
|
|
|
|
{
|
|
|
|
|
SVM_NPT = 0xfc,
|
|
|
|
|
SVM_INVALID = 0xfd,
|
|
|
|
|
|
|
|
|
|
VCPU_STARTUP = 0xfe,
|
|
|
|
|
|
|
|
|
|
RECALL = 0xff,
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
Vcpu_handler(Genode::Env &env, size_t stack_size,
|
|
|
|
|
Genode::Affinity::Location location,
|
|
|
|
|
unsigned int cpu_id)
|
|
|
|
|
:
|
|
|
|
|
_ep(env, stack_size,
|
|
|
|
|
Genode::String<12>("EP-EMT-", cpu_id).string(), location),
|
|
|
|
|
_cpu_id(cpu_id)
|
2020-07-21 09:47:25 +00:00
|
|
|
|
{
|
|
|
|
|
pthread_mutexattr_t _attr;
|
|
|
|
|
pthread_mutexattr_init(&_attr);
|
|
|
|
|
|
|
|
|
|
pthread_cond_init(&_cond_wait, nullptr);
|
|
|
|
|
|
|
|
|
|
pthread_mutexattr_settype(&_attr, PTHREAD_MUTEX_ERRORCHECK);
|
|
|
|
|
pthread_mutex_init(&_mutex, &_attr);
|
|
|
|
|
}
|
2019-05-06 08:34:47 +00:00
|
|
|
|
|
|
|
|
|
unsigned int cpu_id() { return _cpu_id; }
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void recall(PVM vm)
|
|
|
|
|
{
|
|
|
|
|
if (!_vm || !_vcpu) {
|
|
|
|
|
_vm = vm;
|
|
|
|
|
_vcpu = &vm->aCpus[_cpu_id];
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (_vm != vm || _vcpu != &vm->aCpus[_cpu_id])
|
|
|
|
|
Genode::error("wrong CPU !?");
|
|
|
|
|
|
2019-05-22 08:41:30 +00:00
|
|
|
|
_recall_req ++;
|
|
|
|
|
|
|
|
|
|
if (_irq_win) {
|
|
|
|
|
_recall_skip ++;
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
asm volatile ("":::"memory");
|
|
|
|
|
|
|
|
|
|
if (_vm_state != PAUSED)
|
|
|
|
|
pause_vm();
|
|
|
|
|
|
|
|
|
|
_next_state = PAUSE_EXIT;
|
2019-05-06 08:34:47 +00:00
|
|
|
|
|
|
|
|
|
#if 0
|
|
|
|
|
if (_recall_req % 1000 == 0) {
|
|
|
|
|
using Genode::log;
|
|
|
|
|
|
|
|
|
|
while (other) {
|
|
|
|
|
log(other->_cpu_id, " exits=", other->_vm_exits,
|
|
|
|
|
" req:skip:drop,inv recall=", other->_recall_req, ":",
|
|
|
|
|
other->_recall_skip, ":", other->_recall_drop, ":",
|
|
|
|
|
other->_recall_inv, " req:inj:drop irq=",
|
|
|
|
|
other->_irq_request, ":", other->_irq_inject, ":",
|
|
|
|
|
other->_irq_drop);
|
|
|
|
|
|
|
|
|
|
other = other->next();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
#endif
|
|
|
|
|
}
|
|
|
|
|
|
2020-07-21 09:47:25 +00:00
|
|
|
|
void halt(Genode::uint64_t const wait_ns)
|
2019-05-06 08:34:47 +00:00
|
|
|
|
{
|
2020-07-21 09:47:25 +00:00
|
|
|
|
/* calculate timeout */
|
|
|
|
|
timespec ts { 0, 0 };
|
|
|
|
|
clock_gettime(CLOCK_REALTIME, &ts);
|
|
|
|
|
ts = add_timespec_ns(ts, wait_ns);
|
|
|
|
|
|
|
|
|
|
/* wait for condition or timeout */
|
|
|
|
|
pthread_mutex_lock(&_mutex);
|
|
|
|
|
pthread_cond_timedwait(&_cond_wait, &_mutex, &ts);
|
|
|
|
|
pthread_mutex_unlock(&_mutex);
|
2019-05-06 08:34:47 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void wake_up()
|
|
|
|
|
{
|
2020-07-21 09:47:25 +00:00
|
|
|
|
pthread_mutex_lock(&_mutex);
|
|
|
|
|
pthread_cond_signal(&_cond_wait);
|
|
|
|
|
pthread_mutex_unlock(&_mutex);
|
2019-05-06 08:34:47 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int run_hw(PVMR0 pVMR0)
|
|
|
|
|
{
|
|
|
|
|
VM * pVM = reinterpret_cast<VM *>(pVMR0);
|
|
|
|
|
PVMCPU pVCpu = &pVM->aCpus[_cpu_id];
|
|
|
|
|
PCPUMCTX pCtx = CPUMQueryGuestCtxPtr(pVCpu);
|
|
|
|
|
|
|
|
|
|
if (!_vm || !_vcpu) {
|
|
|
|
|
_vm = pVM;
|
|
|
|
|
_vcpu = &pVM->aCpus[_cpu_id];
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (_vm != pVM || _vcpu != &pVM->aCpus[_cpu_id])
|
|
|
|
|
Genode::error("wrong CPU !?");
|
|
|
|
|
|
|
|
|
|
/* take the utcb state prepared during the last exit */
|
2020-12-18 13:08:06 +00:00
|
|
|
|
_state->inj_info.charge(IRQ_INJ_NONE);
|
|
|
|
|
_state->intr_state.charge(next_utcb.intr_state);
|
|
|
|
|
_state->actv_state.charge(ACTIVITY_STATE_ACTIVE);
|
|
|
|
|
_state->ctrl_primary.charge(next_utcb.ctrl[0]);
|
|
|
|
|
_state->ctrl_secondary.charge(next_utcb.ctrl[1]);
|
2019-05-06 08:34:47 +00:00
|
|
|
|
|
|
|
|
|
/* Transfer vCPU state from vbox to Genode format */
|
|
|
|
|
if (!vbox_to_state(pVM, pVCpu) ||
|
|
|
|
|
!hw_load_state(_state, pVM, pVCpu)) {
|
|
|
|
|
|
|
|
|
|
Genode::error("loading vCPU state failed");
|
|
|
|
|
return VERR_INTERNAL_ERROR;
|
|
|
|
|
}
|
2019-05-21 07:25:17 +00:00
|
|
|
|
|
2019-05-06 08:34:47 +00:00
|
|
|
|
/* check whether to request interrupt window for injection */
|
|
|
|
|
_irq_win = check_to_request_irq_window(pVCpu);
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* Flag vCPU to be "pokeable" by external events such as interrupts
|
|
|
|
|
* from virtual devices. Only if this flag is set, the
|
|
|
|
|
* 'vmR3HaltGlobal1NotifyCpuFF' function calls 'SUPR3CallVMMR0Ex'
|
|
|
|
|
* with VMMR0_DO_GVMM_SCHED_POKE as argument to indicate such
|
|
|
|
|
* events. This function, in turn, will recall the vCPU.
|
|
|
|
|
*/
|
|
|
|
|
VMCPU_SET_STATE(pVCpu, VMCPUSTATE_STARTED_EXEC);
|
|
|
|
|
|
|
|
|
|
/* switch to hardware accelerated mode */
|
2019-05-21 07:25:17 +00:00
|
|
|
|
switch_to_hw(pCtx);
|
2019-05-06 08:34:47 +00:00
|
|
|
|
|
|
|
|
|
Assert(_state->actv_state.value() == ACTIVITY_STATE_ACTIVE);
|
|
|
|
|
|
|
|
|
|
/* see hmR0VmxExitToRing3 - sync recompiler state */
|
|
|
|
|
CPUMSetChangedFlags(pVCpu, CPUM_CHANGED_SYSENTER_MSR |
|
|
|
|
|
CPUM_CHANGED_LDTR | CPUM_CHANGED_GDTR |
|
|
|
|
|
CPUM_CHANGED_IDTR | CPUM_CHANGED_TR |
|
|
|
|
|
CPUM_CHANGED_HIDDEN_SEL_REGS |
|
|
|
|
|
CPUM_CHANGED_GLOBAL_TLB_FLUSH);
|
|
|
|
|
|
|
|
|
|
VMCPU_SET_STATE(pVCpu, VMCPUSTATE_STARTED);
|
|
|
|
|
|
|
|
|
|
/* Transfer vCPU state from Genode to vbox format */
|
|
|
|
|
if (!state_to_vbox(pVM, pVCpu) ||
|
|
|
|
|
!hw_save_state(_state, pVM, pVCpu)) {
|
|
|
|
|
|
|
|
|
|
Genode::error("saving vCPU state failed");
|
|
|
|
|
return VERR_INTERNAL_ERROR;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#ifdef VBOX_WITH_REM
|
|
|
|
|
REMFlushTBs(pVM);
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
/* track guest mode changes - see VMM/VMMAll/IEMAllCImpl.cpp.h */
|
|
|
|
|
PGMChangeMode(pVCpu, pCtx->cr0, pCtx->cr4, pCtx->msrEFER);
|
|
|
|
|
|
|
|
|
|
int rc = vm_exit_requires_instruction_emulation(pCtx);
|
|
|
|
|
|
|
|
|
|
/* evaluated in VMM/include/EMHandleRCTmpl.h */
|
|
|
|
|
return rc;
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
#endif /* _VIRTUALBOX__VCPU_H_ */
|