blob: 5083cb8845448c4bd8f82b7391e3ba0dc00cb8ed [file] [log] [blame]
// Copyright 2017 The Fuchsia Authors
//
// Use of this source code is governed by a MIT-style
// license that can be found in the LICENSE file or at
// https://opensource.org/licenses/MIT
#include <object/vcpu_dispatcher.h>
#include <arch/hypervisor.h>
#include <hypervisor/guest_physical_address_space.h>
#include <vm/vm_object.h>
#include <zircon/rights.h>
#include <zircon/types.h>
#include <fbl/alloc_checker.h>
#include <object/guest_dispatcher.h>
zx_status_t VcpuDispatcher::Create(fbl::RefPtr<GuestDispatcher> guest_dispatcher, zx_vaddr_t ip,
zx_vaddr_t cr3, fbl::RefPtr<VmObject> apic_vmo,
fbl::RefPtr<Dispatcher>* dispatcher, zx_rights_t* rights) {
Guest* guest = guest_dispatcher->guest();
GuestPhysicalAddressSpace* gpas = guest->AddressSpace();
if (ip >= gpas->size())
return ZX_ERR_INVALID_ARGS;
if (cr3 >= gpas->size() - PAGE_SIZE)
return ZX_ERR_INVALID_ARGS;
fbl::unique_ptr<Vcpu> vcpu;
#if ARCH_X86_64
zx_status_t status = x86_vcpu_create(ip, cr3, apic_vmo, guest->ApicAccessAddress(),
guest->MsrBitmapsAddress(), gpas, guest->Mux(), &vcpu);
#else // ARCH_X86_64
zx_status_t status = ZX_ERR_NOT_SUPPORTED;
#endif
if (status != ZX_OK)
return status;
fbl::AllocChecker ac;
auto disp = new (&ac) VcpuDispatcher(guest_dispatcher, fbl::move(vcpu));
if (!ac.check())
return ZX_ERR_NO_MEMORY;
*rights = ZX_DEFAULT_VCPU_RIGHTS;
*dispatcher = fbl::AdoptRef<Dispatcher>(disp);
return ZX_OK;
}
VcpuDispatcher::VcpuDispatcher(fbl::RefPtr<GuestDispatcher> guest, fbl::unique_ptr<Vcpu> vcpu)
: guest_(guest), vcpu_(fbl::move(vcpu)) {}
VcpuDispatcher::~VcpuDispatcher() {}
zx_status_t VcpuDispatcher::Resume(zx_port_packet_t* packet) {
canary_.Assert();
return arch_vcpu_resume(vcpu_.get(), packet);
}
zx_status_t VcpuDispatcher::Interrupt(uint32_t vector) {
canary_.Assert();
return arch_vcpu_interrupt(vcpu_.get(), vector);
}
zx_status_t VcpuDispatcher::ReadState(uint32_t kind, void* buffer, uint32_t len) const {
canary_.Assert();
return arch_vcpu_read_state(vcpu_.get(), kind, buffer, len);
}
zx_status_t VcpuDispatcher::WriteState(uint32_t kind, const void* buffer, uint32_t len) {
canary_.Assert();
return arch_vcpu_write_state(vcpu_.get(), kind, buffer, len);
}