mirror of
https://github.com/projectacrn/acrn-hypervisor.git
synced 2025-09-28 20:16:09 +00:00
Current option of removing vGSI capability is global, which exposes vIOAPIC link for all ptdev even only one need this. This patch makes it as ptdev local option to lower the system level impact. To keep vGSI for MSI capable ptdev, just explicitly append ",keep_gsi" in option list, like "-s 14,passthru,0/e/0,keep_gsi" Signed-off-by: Edwin Zhai <edwin.zhai@intel.com> Acked-by: Anthony Xu <anthony.xu@intel.com>
926 lines
20 KiB
C
926 lines
20 KiB
C
/*-
|
|
* Copyright (c) 2011 NetApp, Inc.
|
|
* All rights reserved.
|
|
*
|
|
* Redistribution and use in source and binary forms, with or without
|
|
* modification, are permitted provided that the following conditions
|
|
* are met:
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
* notice, this list of conditions and the following disclaimer.
|
|
* 2. Redistributions in binary form must reproduce the above copyright
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
* documentation and/or other materials provided with the distribution.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY NETAPP, INC ``AS IS'' AND
|
|
* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
|
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
|
* ARE DISCLAIMED. IN NO EVENT SHALL NETAPP, INC OR CONTRIBUTORS BE LIABLE
|
|
* FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
|
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
|
|
* OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
|
|
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
|
|
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
|
|
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
|
|
* SUCH DAMAGE.
|
|
*
|
|
* $FreeBSD$
|
|
*/
|
|
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <err.h>
|
|
#include <errno.h>
|
|
#include <libgen.h>
|
|
#include <unistd.h>
|
|
#include <assert.h>
|
|
#include <pthread.h>
|
|
#include <sysexits.h>
|
|
#include <stdbool.h>
|
|
#include <getopt.h>
|
|
|
|
#include "vmmapi.h"
|
|
#include "sw_load.h"
|
|
#include "cpuset.h"
|
|
#include "dm.h"
|
|
#include "acpi.h"
|
|
#include "atkbdc.h"
|
|
#include "inout.h"
|
|
#include "ioapic.h"
|
|
#include "mem.h"
|
|
#include "mevent.h"
|
|
#include "mptbl.h"
|
|
#include "pci_core.h"
|
|
#include "irq.h"
|
|
#include "lpc.h"
|
|
#include "smbiostbl.h"
|
|
#include "rtc.h"
|
|
#include "version.h"
|
|
#include "sw_load.h"
|
|
#include "monitor.h"
|
|
#include "ioc.h"
|
|
|
|
#define GUEST_NIO_PORT 0x488 /* guest upcalls via i/o port */
|
|
|
|
typedef int (*vmexit_handler_t)(struct vmctx *,
|
|
struct vhm_request *, int *vcpu);
|
|
|
|
char *vmname;
|
|
|
|
int guest_ncpus;
|
|
char *guest_uuid_str;
|
|
char *vsbl_file_name;
|
|
uint8_t trusty_enabled;
|
|
bool stdio_in_use;
|
|
bool hugetlb;
|
|
|
|
static int guest_vmexit_on_hlt, guest_vmexit_on_pause;
|
|
static int virtio_msix = 1;
|
|
static int x2apic_mode; /* default is xAPIC */
|
|
|
|
static int strictio;
|
|
static int strictmsr = 1;
|
|
|
|
static int acpi;
|
|
|
|
static char *progname;
|
|
static const int BSP;
|
|
|
|
static cpuset_t cpumask;
|
|
|
|
static void vm_loop(struct vmctx *ctx);
|
|
|
|
static int quit_vm_loop;
|
|
|
|
static char vhm_request_page[4096] __attribute__ ((aligned(4096)));
|
|
|
|
static struct vhm_request *vhm_req_buf =
|
|
(struct vhm_request *)&vhm_request_page;
|
|
|
|
struct dmstats {
|
|
uint64_t vmexit_bogus;
|
|
uint64_t vmexit_reqidle;
|
|
uint64_t vmexit_hlt;
|
|
uint64_t vmexit_pause;
|
|
uint64_t vmexit_mtrap;
|
|
uint64_t cpu_switch_rotate;
|
|
uint64_t cpu_switch_direct;
|
|
uint64_t vmexit_mmio_emul;
|
|
} stats;
|
|
|
|
struct mt_vmm_info {
|
|
pthread_t mt_thr;
|
|
struct vmctx *mt_ctx;
|
|
int mt_vcpu;
|
|
} mt_vmm_info[VM_MAXCPU];
|
|
|
|
static cpuset_t *vcpumap[VM_MAXCPU] = { NULL };
|
|
|
|
static struct vmctx *_ctx;
|
|
|
|
static void
|
|
usage(int code)
|
|
{
|
|
fprintf(stderr,
|
|
"Usage: %s [-abehuwxACHPSTWY] [-c vcpus] [-g <gdb port>] [-l <lpc>]\n"
|
|
" %*s [-m mem] [-p vcpu:hostcpu] [-s <pci>] [-U uuid] \n"
|
|
" %*s [--vsbl vsbl_file_name] [--part_info part_info_name]\n"
|
|
" %*s [--enable_trusty] <vm>\n"
|
|
" -a: local apic is in xAPIC mode (deprecated)\n"
|
|
" -A: create ACPI tables\n"
|
|
" -b: enable bvmcons\n"
|
|
" -c: # cpus (default 1)\n"
|
|
" -C: include guest memory in core file\n"
|
|
" -e: exit on unhandled I/O access\n"
|
|
" -g: gdb port\n"
|
|
" -h: help\n"
|
|
" -H: vmexit from the guest on hlt\n"
|
|
" -l: LPC device configuration\n"
|
|
" -m: memory size in MB\n"
|
|
" -p: pin 'vcpu' to 'hostcpu'\n"
|
|
" -P: vmexit from the guest on pause\n"
|
|
" -s: <slot,driver,configinfo> PCI slot config\n"
|
|
" -S: guest memory cannot be swapped\n"
|
|
" -u: RTC keeps UTC time\n"
|
|
" -U: uuid\n"
|
|
" -w: ignore unimplemented MSRs\n"
|
|
" -W: force virtio to use single-vector MSI\n"
|
|
" -T: use hugetlb for memory allocation\n"
|
|
" -x: local apic is in x2APIC mode\n"
|
|
" -Y: disable MPtable generation\n"
|
|
" -k: kernel image path\n"
|
|
" -r: ramdisk image path\n"
|
|
" -B: bootargs for kernel\n"
|
|
" -G: GVT args: low_gm_size, high_gm_size, fence_sz\n"
|
|
" -v: version\n"
|
|
" -i: ioc boot parameters\n"
|
|
" --vsbl: vsbl file path\n"
|
|
" --part_info: guest partition info file path\n"
|
|
" --enable_trusty: enable trusty for guest\n"
|
|
" --ptdev_no_reset: disable reset check for ptdev\n",
|
|
progname, (int)strlen(progname), "", (int)strlen(progname), "",
|
|
(int)strlen(progname), "");
|
|
|
|
exit(code);
|
|
}
|
|
|
|
static void
|
|
print_version(void)
|
|
{
|
|
if (DM_RC_VERSION)
|
|
fprintf(stderr, "DM version is: %d.%d-%d-%s, build by %s@%s\n",
|
|
DM_MAJOR_VERSION, DM_MINOR_VERSION, DM_RC_VERSION,
|
|
DM_BUILD_VERSION, DM_BUILD_USER, DM_BUILD_TIME);
|
|
else
|
|
fprintf(stderr, "DM version is: %d.%d-%s, build by %s@%s\n",
|
|
DM_MAJOR_VERSION, DM_MINOR_VERSION, DM_BUILD_VERSION,
|
|
DM_BUILD_USER, DM_BUILD_TIME);
|
|
|
|
exit(0);
|
|
}
|
|
|
|
static int
|
|
pincpu_parse(const char *opt)
|
|
{
|
|
int vcpu, pcpu;
|
|
|
|
if (sscanf(opt, "%d:%d", &vcpu, &pcpu) != 2) {
|
|
fprintf(stderr, "invalid format: %s\n", opt);
|
|
return -1;
|
|
}
|
|
|
|
if (vcpu < 0 || vcpu >= VM_MAXCPU) {
|
|
fprintf(stderr, "vcpu '%d' outside valid range from 0 to %d\n",
|
|
vcpu, VM_MAXCPU - 1);
|
|
return -1;
|
|
}
|
|
|
|
if (pcpu < 0 || pcpu >= CPU_SETSIZE) {
|
|
fprintf(stderr,
|
|
"hostcpu '%d' outside valid range from 0 to %d\n",
|
|
pcpu, CPU_SETSIZE - 1);
|
|
return -1;
|
|
}
|
|
|
|
if (vcpumap[vcpu] == NULL) {
|
|
vcpumap[vcpu] = malloc(sizeof(cpuset_t));
|
|
if (vcpumap[vcpu] == NULL) {
|
|
perror("malloc");
|
|
return -1;
|
|
}
|
|
CPU_ZERO(vcpumap[vcpu]);
|
|
}
|
|
CPU_SET(pcpu, vcpumap[vcpu]);
|
|
return 0;
|
|
}
|
|
|
|
void *
|
|
paddr_guest2host(struct vmctx *ctx, uintptr_t gaddr, size_t len)
|
|
{
|
|
return vm_map_gpa(ctx, gaddr, len);
|
|
}
|
|
|
|
void *
|
|
dm_gpa2hva(uint64_t gpa, size_t size)
|
|
{
|
|
return vm_map_gpa(_ctx, gpa, size);
|
|
}
|
|
|
|
int
|
|
virtio_uses_msix(void)
|
|
{
|
|
return virtio_msix;
|
|
}
|
|
|
|
static void *
|
|
start_thread(void *param)
|
|
{
|
|
char tname[MAXCOMLEN + 1];
|
|
struct mt_vmm_info *mtp;
|
|
int vcpu;
|
|
|
|
mtp = param;
|
|
vcpu = mtp->mt_vcpu;
|
|
|
|
snprintf(tname, sizeof(tname), "vcpu %d", vcpu);
|
|
pthread_setname_np(mtp->mt_thr, tname);
|
|
|
|
vm_loop(mtp->mt_ctx);
|
|
|
|
/* reset or halt */
|
|
return NULL;
|
|
}
|
|
|
|
static int
|
|
add_cpu(struct vmctx *ctx, int guest_ncpus)
|
|
{
|
|
int i;
|
|
int error;
|
|
|
|
for (i = 0; i < guest_ncpus; i++) {
|
|
error = vm_create_vcpu(ctx, i);
|
|
if (error != 0) {
|
|
fprintf(stderr, "ERROR: could not create VCPU %d\n", i);
|
|
return error;
|
|
}
|
|
|
|
CPU_SET_ATOMIC(i, &cpumask);
|
|
|
|
mt_vmm_info[i].mt_ctx = ctx;
|
|
mt_vmm_info[i].mt_vcpu = i;
|
|
}
|
|
|
|
error = pthread_create(&mt_vmm_info[0].mt_thr, NULL,
|
|
start_thread, &mt_vmm_info[0]);
|
|
|
|
return error;
|
|
}
|
|
|
|
static int
|
|
delete_cpu(struct vmctx *ctx, int vcpu)
|
|
{
|
|
if (!CPU_ISSET(vcpu, &cpumask)) {
|
|
fprintf(stderr, "Attempting to delete unknown cpu %d\n", vcpu);
|
|
exit(1);
|
|
}
|
|
|
|
/* wait for vm_loop cleanup */
|
|
quit_vm_loop = 1;
|
|
vm_destroy_ioreq_client(ctx);
|
|
while (quit_vm_loop)
|
|
usleep(10000);
|
|
|
|
CPU_CLR_ATOMIC(vcpu, &cpumask);
|
|
return CPU_EMPTY(&cpumask);
|
|
}
|
|
|
|
static int
|
|
vmexit_inout(struct vmctx *ctx, struct vhm_request *vhm_req, int *pvcpu)
|
|
{
|
|
int error;
|
|
int bytes, port, in;
|
|
|
|
port = vhm_req->reqs.pio_request.address;
|
|
bytes = vhm_req->reqs.pio_request.size;
|
|
in = (vhm_req->reqs.pio_request.direction == REQUEST_READ);
|
|
|
|
error = emulate_inout(ctx, pvcpu, &vhm_req->reqs.pio_request, strictio);
|
|
if (error) {
|
|
fprintf(stderr, "Unhandled %s%c 0x%04x\n",
|
|
in ? "in" : "out",
|
|
bytes == 1 ? 'b' : (bytes == 2 ? 'w' : 'l'),
|
|
port);
|
|
return VMEXIT_ABORT;
|
|
} else {
|
|
return VMEXIT_CONTINUE;
|
|
}
|
|
}
|
|
|
|
static int
|
|
vmexit_mmio_emul(struct vmctx *ctx, struct vhm_request *vhm_req, int *pvcpu)
|
|
{
|
|
int err;
|
|
|
|
stats.vmexit_mmio_emul++;
|
|
err = emulate_mem(ctx, &vhm_req->reqs.mmio_request);
|
|
|
|
if (err) {
|
|
if (err == -ESRCH)
|
|
fprintf(stderr, "Unhandled memory access to 0x%lx\n",
|
|
vhm_req->reqs.mmio_request.address);
|
|
|
|
fprintf(stderr, "Failed to emulate instruction [");
|
|
fprintf(stderr, "mmio address 0x%lx, size %ld",
|
|
vhm_req->reqs.mmio_request.address,
|
|
vhm_req->reqs.mmio_request.size);
|
|
vhm_req->processed = REQ_STATE_FAILED;
|
|
return VMEXIT_ABORT;
|
|
}
|
|
vhm_req->processed = REQ_STATE_SUCCESS;
|
|
return VMEXIT_CONTINUE;
|
|
}
|
|
|
|
static int
|
|
vmexit_pci_emul(struct vmctx *ctx, struct vhm_request *vhm_req, int *pvcpu)
|
|
{
|
|
int err, in = (vhm_req->reqs.pci_request.direction == REQUEST_READ);
|
|
|
|
err = emulate_pci_cfgrw(ctx, *pvcpu, in,
|
|
vhm_req->reqs.pci_request.bus,
|
|
vhm_req->reqs.pci_request.dev,
|
|
vhm_req->reqs.pci_request.func,
|
|
vhm_req->reqs.pci_request.reg,
|
|
vhm_req->reqs.pci_request.size,
|
|
&vhm_req->reqs.pci_request.value);
|
|
if (err) {
|
|
fprintf(stderr, "Unhandled pci cfg rw at %x:%x.%x reg 0x%x\n",
|
|
vhm_req->reqs.pci_request.bus,
|
|
vhm_req->reqs.pci_request.dev,
|
|
vhm_req->reqs.pci_request.func,
|
|
vhm_req->reqs.pci_request.reg);
|
|
return VMEXIT_ABORT;
|
|
}
|
|
|
|
vhm_req->processed = REQ_STATE_SUCCESS;
|
|
return VMEXIT_CONTINUE;
|
|
}
|
|
|
|
#define DEBUG_EPT_MISCONFIG
|
|
|
|
#ifdef DEBUG_EPT_MISCONFIG
|
|
|
|
#define EXIT_REASON_EPT_MISCONFIG 49
|
|
#define VMCS_GUEST_PHYSICAL_ADDRESS 0x00002400
|
|
#define VMCS_IDENT(x) ((x) | 0x80000000)
|
|
|
|
#endif /* #ifdef DEBUG_EPT_MISCONFIG */
|
|
|
|
static int
|
|
vmexit_bogus(struct vmctx *ctx, struct vhm_request *vhm_req, int *pvcpu)
|
|
{
|
|
stats.vmexit_bogus++;
|
|
|
|
return VMEXIT_CONTINUE;
|
|
}
|
|
|
|
static int
|
|
vmexit_reqidle(struct vmctx *ctx, struct vhm_request *vhm_req, int *pvcpu)
|
|
{
|
|
stats.vmexit_reqidle++;
|
|
|
|
return VMEXIT_CONTINUE;
|
|
}
|
|
|
|
static int
|
|
vmexit_hlt(struct vmctx *ctx, struct vhm_request *vhm_req, int *pvcpu)
|
|
{
|
|
stats.vmexit_hlt++;
|
|
|
|
/*
|
|
* Just continue execution with the next instruction. We use
|
|
* the HLT VM exit as a way to be friendly with the host
|
|
* scheduler.
|
|
*/
|
|
return VMEXIT_CONTINUE;
|
|
}
|
|
|
|
static int
|
|
vmexit_pause(struct vmctx *ctx, struct vhm_request *vhm_req, int *pvcpu)
|
|
{
|
|
stats.vmexit_pause++;
|
|
|
|
return VMEXIT_CONTINUE;
|
|
}
|
|
|
|
static int
|
|
vmexit_mtrap(struct vmctx *ctx, struct vhm_request *vhm_req, int *pvcpu)
|
|
{
|
|
stats.vmexit_mtrap++;
|
|
|
|
return VMEXIT_CONTINUE;
|
|
}
|
|
|
|
static vmexit_handler_t handler[VM_EXITCODE_MAX] = {
|
|
[VM_EXITCODE_INOUT] = vmexit_inout,
|
|
[VM_EXITCODE_MMIO_EMUL] = vmexit_mmio_emul,
|
|
[VM_EXITCODE_PCI_CFG] = vmexit_pci_emul,
|
|
[VM_EXITCODE_BOGUS] = vmexit_bogus,
|
|
[VM_EXITCODE_REQIDLE] = vmexit_reqidle,
|
|
[VM_EXITCODE_MTRAP] = vmexit_mtrap,
|
|
[VM_EXITCODE_HLT] = vmexit_hlt,
|
|
[VM_EXITCODE_PAUSE] = vmexit_pause,
|
|
};
|
|
|
|
static void
|
|
handle_vmexit(struct vmctx *ctx, struct vhm_request *vhm_req, int vcpu)
|
|
{
|
|
int rc;
|
|
enum vm_exitcode exitcode;
|
|
|
|
exitcode = vhm_req->type;
|
|
if (exitcode >= VM_EXITCODE_MAX || handler[exitcode] == NULL) {
|
|
fprintf(stderr, "handle vmexit: unexpected exitcode 0x%x\n",
|
|
exitcode);
|
|
exit(1);
|
|
}
|
|
|
|
rc = (*handler[exitcode])(ctx, vhm_req, &vcpu);
|
|
switch (rc) {
|
|
case VMEXIT_CONTINUE:
|
|
vhm_req->processed = REQ_STATE_SUCCESS;
|
|
break;
|
|
case VMEXIT_ABORT:
|
|
vhm_req->processed = REQ_STATE_FAILED;
|
|
abort();
|
|
default:
|
|
exit(1);
|
|
}
|
|
vm_notify_request_done(ctx, vcpu);
|
|
}
|
|
|
|
static int
|
|
vm_init_vdevs(struct vmctx *ctx)
|
|
{
|
|
int ret;
|
|
|
|
init_mem();
|
|
init_inout();
|
|
pci_irq_init(ctx);
|
|
atkbdc_init(ctx);
|
|
ioapic_init(ctx);
|
|
|
|
/*
|
|
* We don't care ioc_init return value so far.
|
|
* Will add return value check once ioc is full function.
|
|
*/
|
|
ret = ioc_init(ctx);
|
|
|
|
ret = vrtc_init(ctx);
|
|
if (ret < 0)
|
|
goto vrtc_fail;
|
|
|
|
sci_init(ctx);
|
|
init_bvmcons();
|
|
|
|
ret = monitor_init(ctx);
|
|
if (ret < 0)
|
|
goto monitor_fail;
|
|
|
|
ret = init_pci(ctx);
|
|
if (ret < 0)
|
|
goto pci_fail;
|
|
|
|
return 0;
|
|
pci_fail:
|
|
monitor_close();
|
|
monitor_fail:
|
|
deinit_bvmcons();
|
|
vrtc_deinit(ctx);
|
|
vrtc_fail:
|
|
ioc_deinit(ctx);
|
|
atkbdc_deinit(ctx);
|
|
pci_irq_deinit(ctx);
|
|
ioapic_deinit();
|
|
return -1;
|
|
}
|
|
|
|
static void
|
|
vm_deinit_vdevs(struct vmctx *ctx)
|
|
{
|
|
deinit_pci(ctx);
|
|
monitor_close();
|
|
deinit_bvmcons();
|
|
vrtc_deinit(ctx);
|
|
ioc_deinit(ctx);
|
|
atkbdc_deinit(ctx);
|
|
pci_irq_deinit(ctx);
|
|
ioapic_deinit();
|
|
}
|
|
|
|
static void
|
|
vm_loop(struct vmctx *ctx)
|
|
{
|
|
int error;
|
|
|
|
ctx->ioreq_client = vm_create_ioreq_client(ctx);
|
|
assert(ctx->ioreq_client > 0);
|
|
|
|
error = vm_run(ctx);
|
|
assert(error == 0);
|
|
|
|
while (1) {
|
|
int vcpu;
|
|
struct vhm_request *vhm_req;
|
|
|
|
error = vm_attach_ioreq_client(ctx);
|
|
if (error)
|
|
break;
|
|
|
|
for (vcpu = 0; vcpu < 4; vcpu++) {
|
|
vhm_req = &vhm_req_buf[vcpu];
|
|
if (vhm_req->valid
|
|
&& (vhm_req->processed == REQ_STATE_PROCESSING)
|
|
&& (vhm_req->client == ctx->ioreq_client))
|
|
handle_vmexit(ctx, vhm_req, vcpu);
|
|
}
|
|
}
|
|
quit_vm_loop = 0;
|
|
printf("VM loop exit\n");
|
|
}
|
|
|
|
static int
|
|
num_vcpus_allowed(struct vmctx *ctx)
|
|
{
|
|
/* TODO: add ioctl to get gerneric information including
|
|
* virtual cpus, now hardcode
|
|
*/
|
|
return VM_MAXCPU;
|
|
}
|
|
|
|
static struct vmctx *
|
|
do_open(const char *vmname)
|
|
{
|
|
struct vmctx *ctx;
|
|
int error;
|
|
|
|
error = vm_create(vmname);
|
|
if (error) {
|
|
perror("vm_create");
|
|
exit(1);
|
|
|
|
}
|
|
|
|
ctx = vm_open(vmname);
|
|
if (ctx == NULL) {
|
|
perror("vm_open");
|
|
exit(1);
|
|
}
|
|
|
|
return ctx;
|
|
}
|
|
|
|
static void
|
|
sig_handler_term(int signo)
|
|
{
|
|
printf("Receive SIGINT to terminate application...\n");
|
|
vm_set_suspend_mode(VM_SUSPEND_POWEROFF);
|
|
mevent_notify();
|
|
}
|
|
|
|
enum {
|
|
CMD_OPT_VSBL = 1000,
|
|
CMD_OPT_PART_INFO,
|
|
CMD_OPT_TRUSTY_ENABLE,
|
|
CMD_OPT_PTDEV_NO_RESET,
|
|
};
|
|
|
|
static struct option long_options[] = {
|
|
{"no_x2apic_mode", no_argument, 0, 'a' },
|
|
{"acpi", no_argument, 0, 'A' },
|
|
{"bvmcons", no_argument, 0, 'b' },
|
|
{"pincpu", required_argument, 0, 'p' },
|
|
{"ncpus", required_argument, 0, 'c' },
|
|
{"memflags_incore", no_argument, 0, 'C' },
|
|
{"gdb_port", required_argument, 0, 'g' },
|
|
{"lpc", required_argument, 0, 'l' },
|
|
{"pci_slot", required_argument, 0, 's' },
|
|
{"memflags_wired", no_argument, 0, 'S' },
|
|
{"memsize", required_argument, 0, 'm' },
|
|
{"ioapic", no_argument, 0, 'I' },
|
|
{"vmexit_pause", no_argument, 0, 'P' },
|
|
{"strictio", no_argument, 0, 'e' },
|
|
{"rtc_localtime", no_argument, 0, 'u' },
|
|
{"uuid", required_argument, 0, 'U' },
|
|
{"strictmsr", no_argument, 0, 'w' },
|
|
{"virtio_msix", no_argument, 0, 'W' },
|
|
{"x2apic_mode", no_argument, 0, 'x' },
|
|
{"mptgen", no_argument, 0, 'Y' },
|
|
{"kernel", required_argument, 0, 'k' },
|
|
{"ramdisk", required_argument, 0, 'r' },
|
|
{"bootargs", required_argument, 0, 'B' },
|
|
{"version", no_argument, 0, 'v' },
|
|
{"gvtargs", required_argument, 0, 'G' },
|
|
{"help", no_argument, 0, 'h' },
|
|
|
|
/* Following cmd option only has long option */
|
|
{"vsbl", required_argument, 0, CMD_OPT_VSBL},
|
|
{"part_info", required_argument, 0, CMD_OPT_PART_INFO},
|
|
{"enable_trusty", no_argument, 0,
|
|
CMD_OPT_TRUSTY_ENABLE},
|
|
{"ptdev_no_reset", no_argument, 0,
|
|
CMD_OPT_PTDEV_NO_RESET},
|
|
{0, 0, 0, 0 },
|
|
};
|
|
|
|
int
|
|
main(int argc, char *argv[])
|
|
{
|
|
int c, error, gdb_port, err;
|
|
int max_vcpus, mptgen, memflags;
|
|
struct vmctx *ctx;
|
|
size_t memsize;
|
|
char *optstr;
|
|
int option_idx = 0;
|
|
|
|
progname = basename(argv[0]);
|
|
gdb_port = 0;
|
|
guest_ncpus = 1;
|
|
memsize = 256 * MB;
|
|
mptgen = 1;
|
|
memflags = 0;
|
|
quit_vm_loop = 0;
|
|
hugetlb = 0;
|
|
|
|
if (signal(SIGHUP, sig_handler_term) == SIG_ERR)
|
|
fprintf(stderr, "cannot register handler for SIGHUP\n");
|
|
if (signal(SIGINT, sig_handler_term) == SIG_ERR)
|
|
fprintf(stderr, "cannot register handler for SIGINT\n");
|
|
|
|
optstr = "abehuwxACHIPSTWYvk:r:B:p:g:c:s:m:l:U:G:i:";
|
|
while ((c = getopt_long(argc, argv, optstr, long_options,
|
|
&option_idx)) != -1) {
|
|
switch (c) {
|
|
case 'a':
|
|
x2apic_mode = 0;
|
|
break;
|
|
case 'A':
|
|
acpi = 1;
|
|
break;
|
|
case 'b':
|
|
enable_bvmcons();
|
|
break;
|
|
case 'p':
|
|
if (pincpu_parse(optarg) != 0) {
|
|
errx(EX_USAGE,
|
|
"invalid vcpu pinning configuration '%s'",
|
|
optarg);
|
|
}
|
|
break;
|
|
case 'c':
|
|
guest_ncpus = atoi(optarg);
|
|
break;
|
|
case 'C':
|
|
memflags |= VM_MEM_F_INCORE;
|
|
break;
|
|
case 'g':
|
|
gdb_port = atoi(optarg);
|
|
break;
|
|
|
|
case 'i':
|
|
ioc_parse(optarg);
|
|
break;
|
|
|
|
case 'l':
|
|
if (lpc_device_parse(optarg) != 0) {
|
|
errx(EX_USAGE,
|
|
"invalid lpc device configuration '%s'",
|
|
optarg);
|
|
}
|
|
break;
|
|
case 's':
|
|
if (pci_parse_slot(optarg) != 0)
|
|
exit(1);
|
|
else
|
|
break;
|
|
case 'S':
|
|
memflags |= VM_MEM_F_WIRED;
|
|
break;
|
|
case 'm':
|
|
error = vm_parse_memsize(optarg, &memsize);
|
|
if (error)
|
|
errx(EX_USAGE, "invalid memsize '%s'", optarg);
|
|
break;
|
|
case 'H':
|
|
guest_vmexit_on_hlt = 1;
|
|
break;
|
|
case 'I':
|
|
/*
|
|
* The "-I" option was used to add an ioapic to the
|
|
* virtual machine.
|
|
*
|
|
* An ioapic is now provided unconditionally for each
|
|
* virtual machine and this option is now deprecated.
|
|
*/
|
|
break;
|
|
case 'P':
|
|
guest_vmexit_on_pause = 1;
|
|
break;
|
|
case 'e':
|
|
strictio = 1;
|
|
break;
|
|
case 'u':
|
|
vrtc_enable_localtime(0);
|
|
break;
|
|
case 'U':
|
|
guest_uuid_str = optarg;
|
|
break;
|
|
case 'w':
|
|
strictmsr = 0;
|
|
break;
|
|
case 'W':
|
|
virtio_msix = 0;
|
|
break;
|
|
case 'T':
|
|
if (check_hugetlb_support())
|
|
hugetlb = 1;
|
|
break;
|
|
case 'x':
|
|
x2apic_mode = 1;
|
|
break;
|
|
case 'Y':
|
|
mptgen = 0;
|
|
break;
|
|
case 'k':
|
|
if (acrn_parse_kernel(optarg) != 0)
|
|
exit(1);
|
|
else
|
|
break;
|
|
case 'r':
|
|
if (acrn_parse_ramdisk(optarg) != 0)
|
|
exit(1);
|
|
else
|
|
break;
|
|
case 'B':
|
|
if (acrn_parse_bootargs(optarg) != 0)
|
|
exit(1);
|
|
else
|
|
break;
|
|
break;
|
|
case 'G':
|
|
if (acrn_parse_gvtargs(optarg) != 0) {
|
|
errx(EX_USAGE, "invalid GVT param %s", optarg);
|
|
exit(1);
|
|
}
|
|
break;
|
|
case 'v':
|
|
print_version();
|
|
break;
|
|
case CMD_OPT_VSBL:
|
|
if (acrn_parse_vsbl(optarg) != 0) {
|
|
errx(EX_USAGE, "invalid vsbl param %s", optarg);
|
|
exit(1);
|
|
}
|
|
break;
|
|
case CMD_OPT_PART_INFO:
|
|
if (acrn_parse_guest_part_info(optarg) != 0) {
|
|
errx(EX_USAGE,
|
|
"invalid guest partition info param %s",
|
|
optarg);
|
|
exit(1);
|
|
}
|
|
break;
|
|
case CMD_OPT_TRUSTY_ENABLE:
|
|
trusty_enabled = 1;
|
|
break;
|
|
case CMD_OPT_PTDEV_NO_RESET:
|
|
ptdev_no_reset(true);
|
|
break;
|
|
case 'h':
|
|
usage(0);
|
|
default:
|
|
usage(1);
|
|
}
|
|
}
|
|
argc -= optind;
|
|
argv += optind;
|
|
|
|
if (argc != 1)
|
|
usage(1);
|
|
|
|
vmname = argv[0];
|
|
|
|
for (;;) {
|
|
ctx = do_open(vmname);
|
|
|
|
/* set IOReq buffer page */
|
|
error = vm_set_shared_io_page(ctx, (unsigned long)vhm_req_buf);
|
|
if (error)
|
|
goto fail;
|
|
|
|
if (guest_ncpus < 1) {
|
|
fprintf(stderr, "Invalid guest vCPUs (%d)\n",
|
|
guest_ncpus);
|
|
goto fail;
|
|
}
|
|
|
|
max_vcpus = num_vcpus_allowed(ctx);
|
|
if (guest_ncpus > max_vcpus) {
|
|
fprintf(stderr, "%d vCPUs requested but %d available\n",
|
|
guest_ncpus, max_vcpus);
|
|
goto fail;
|
|
}
|
|
|
|
vm_set_memflags(ctx, memflags);
|
|
err = vm_setup_memory(ctx, memsize, VM_MMAP_ALL);
|
|
if (err) {
|
|
fprintf(stderr, "Unable to setup memory (%d)\n", errno);
|
|
goto fail;
|
|
}
|
|
|
|
err = mevent_init();
|
|
if (err) {
|
|
fprintf(stderr, "Unable to initialize mevent (%d)\n",
|
|
errno);
|
|
goto mevent_fail;
|
|
}
|
|
|
|
if (gdb_port != 0)
|
|
fprintf(stderr, "dbgport not supported\n");
|
|
|
|
if (vm_init_vdevs(ctx) < 0) {
|
|
fprintf(stderr, "Unable to init vdev (%d)\n", errno);
|
|
goto dev_fail;
|
|
}
|
|
|
|
/*
|
|
* build the guest tables, MP etc.
|
|
*/
|
|
if (mptgen) {
|
|
error = mptable_build(ctx, guest_ncpus);
|
|
if (error) {
|
|
goto vm_fail;
|
|
}
|
|
}
|
|
|
|
error = smbios_build(ctx);
|
|
if (error)
|
|
goto vm_fail;
|
|
|
|
if (acpi) {
|
|
error = acpi_build(ctx, guest_ncpus);
|
|
if (error)
|
|
goto vm_fail;
|
|
}
|
|
|
|
error = acrn_sw_load(ctx);
|
|
if (error)
|
|
goto vm_fail;
|
|
|
|
/*
|
|
* Change the proc title to include the VM name.
|
|
*/
|
|
/*setproctitle("%s", vmname);*/
|
|
|
|
/*
|
|
* Add CPU 0
|
|
*/
|
|
error = add_cpu(ctx, guest_ncpus);
|
|
if (error)
|
|
goto vm_fail;
|
|
|
|
/* Make a copy for ctx */
|
|
_ctx = ctx;
|
|
|
|
/*
|
|
* Head off to the main event dispatch loop
|
|
*/
|
|
mevent_dispatch();
|
|
|
|
vm_pause(ctx);
|
|
delete_cpu(ctx, BSP);
|
|
|
|
if (vm_get_suspend_mode() != VM_SUSPEND_RESET)
|
|
break;
|
|
|
|
vm_deinit_vdevs(ctx);
|
|
mevent_deinit();
|
|
vm_unsetup_memory(ctx);
|
|
vm_destroy(ctx);
|
|
vm_close(ctx);
|
|
_ctx = 0;
|
|
|
|
vm_set_suspend_mode(VM_SUSPEND_NONE);
|
|
}
|
|
|
|
vm_fail:
|
|
vm_deinit_vdevs(ctx);
|
|
dev_fail:
|
|
mevent_deinit();
|
|
mevent_fail:
|
|
vm_unsetup_memory(ctx);
|
|
fail:
|
|
vm_destroy(ctx);
|
|
vm_close(ctx);
|
|
exit(0);
|
|
}
|