#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
#include <fcntl.h>
#include <libgen.h>
#include <sys/stat.h>
#include <errno.h>
#include <err.h>
#include <assert.h>
#include <sys/vmm.h>
#include <sys/vmm_dev.h>
#include <vmmapi.h>
#include "common.h"
int
main(int argc, char *argv[])
{
const char *suite_name = basename(argv[0]);
struct vmctx *ctx;
struct vcpu *vcpu;
ctx = create_test_vm(suite_name);
if (ctx == NULL) {
errx(EXIT_FAILURE, "could open test VM");
}
assert(check_instance_usable(suite_name));
if ((vcpu = vm_vcpu_open(ctx, 0)) == NULL) {
err(EXIT_FAILURE, "Could not open vcpu0");
}
if (ioctl(vm_get_device_fd(ctx), VM_SET_AUTODESTRUCT, 0) != 0) {
errx(EXIT_FAILURE, "could not disable auto-destruct");
}
if (ioctl(vm_get_device_fd(ctx), VM_DESTROY_SELF, 0) != 0) {
errx(EXIT_FAILURE, "ioctl(VM_DESTROY_SELF) failed");
}
if (!check_instance_exists(suite_name)) {
err(EXIT_FAILURE,
"instance missing after unfinished destroy");
}
uint64_t reg = 0;
if (vm_get_register(vcpu, VM_REG_GUEST_RAX, ®) == 0) {
err(EXIT_FAILURE,
"VM_GET_REGISTER succeeded despite instance destruction");
}
if (check_instance_usable(suite_name)) {
err(EXIT_FAILURE,
"instance not reporting in-progress destruction");
}
vm_vcpu_close(vcpu);
vm_close(ctx);
ctx = NULL;
if (check_instance_exists(suite_name)) {
err(EXIT_FAILURE, "instance still accessible after destroy");
}
(void) printf("%s\tPASS\n", suite_name);
return (EXIT_SUCCESS);
}