2018-03-27 00:14:19 +00:00
|
|
|
/*
|
2022-09-13 17:55:17 +00:00
|
|
|
* Copyright 2018 The ChromiumOS Authors
|
2018-03-27 00:14:19 +00:00
|
|
|
* Use of this source code is governed by a BSD-style license that can be
|
|
|
|
* found in the LICENSE file.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdint.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
|
|
|
|
#include "crosvm.h"
|
|
|
|
|
|
|
|
int main(int argc, char** argv) {
|
|
|
|
struct crosvm *crosvm;
|
|
|
|
int ret = crosvm_connect(&crosvm);
|
|
|
|
if (ret) {
|
|
|
|
fprintf(stderr, "failed to connect to crosvm: %d\n", ret);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
struct kvm_pic_state pic_state;
|
|
|
|
ret = crosvm_get_pic_state(crosvm, false, &pic_state);
|
|
|
|
if (ret < 0) {
|
|
|
|
fprintf(stderr, "failed to get initial PIC1 state: %d\n", ret);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (pic_state.auto_eoi) {
|
|
|
|
fprintf(stderr, "unexpected value of auto_eoi flag\n");
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
pic_state.auto_eoi = true;
|
|
|
|
ret = crosvm_set_pic_state(crosvm, false, &pic_state);
|
|
|
|
if (ret < 0) {
|
|
|
|
fprintf(stderr, "failed to update PIC1 state: %d\n", ret);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = crosvm_get_pic_state(crosvm, false, &pic_state);
|
|
|
|
if (ret < 0) {
|
|
|
|
fprintf(stderr, "failed to get updated PIC1 state: %d\n", ret);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!pic_state.auto_eoi) {
|
|
|
|
fprintf(stderr, "unexpected value of auto_eoi flag after update\n");
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Test retrieving and setting IOAPIC state.
|
|
|
|
struct kvm_ioapic_state ioapic_state;
|
|
|
|
ret = crosvm_get_ioapic_state(crosvm, &ioapic_state);
|
|
|
|
if (ret < 0) {
|
|
|
|
fprintf(stderr, "failed to get initial PIC1 state: %d\n", ret);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
fprintf(stderr, "IOAPIC ID: %d\n", ioapic_state.id);
|
|
|
|
|
|
|
|
if (ioapic_state.id != 0) {
|
|
|
|
fprintf(stderr, "unexpected value of IOAPIC ID: %d\n", ioapic_state.id);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
ioapic_state.id = 1;
|
|
|
|
ret = crosvm_set_ioapic_state(crosvm, &ioapic_state);
|
|
|
|
if (ret < 0) {
|
|
|
|
fprintf(stderr, "failed to update PIC1 state: %d\n", ret);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = crosvm_get_ioapic_state(crosvm, &ioapic_state);
|
|
|
|
if (ret < 0) {
|
|
|
|
fprintf(stderr, "failed to get updated PIC1 state: %d\n", ret);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ioapic_state.id != 1) {
|
|
|
|
fprintf(stderr, "unexpected value of IOAPIC ID after update: %d\n",
|
|
|
|
ioapic_state.id);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Test retrieving and setting PIT state.
|
|
|
|
struct kvm_pit_state2 pit_state;
|
|
|
|
ret = crosvm_get_pit_state(crosvm, &pit_state);
|
|
|
|
if (ret < 0) {
|
|
|
|
fprintf(stderr, "failed to get initial PIT state: %d\n", ret);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (pit_state.flags & KVM_PIT_FLAGS_HPET_LEGACY) {
|
|
|
|
fprintf(stderr, "unexpected value of KVM_PIT_FLAGS_HPET_LEGACY flag\n");
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
pit_state.flags |= KVM_PIT_FLAGS_HPET_LEGACY;
|
|
|
|
ret = crosvm_set_pit_state(crosvm, &pit_state);
|
|
|
|
if (ret < 0) {
|
|
|
|
fprintf(stderr, "failed to update PIT state: %d\n", ret);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = crosvm_get_pit_state(crosvm, &pit_state);
|
|
|
|
if (ret < 0) {
|
|
|
|
fprintf(stderr, "failed to get updated PIT state: %d\n", ret);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!(pit_state.flags & KVM_PIT_FLAGS_HPET_LEGACY)) {
|
|
|
|
fprintf(stderr,
|
|
|
|
"unexpected value of KVM_PIT_FLAGS_HPET_LEGACY after update\n");
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2019-01-18 00:20:33 +00:00
|
|
|
// Test retrieving and setting clock state.
|
|
|
|
struct kvm_clock_data clock_data = { .clock = 0, .flags = -1U, };
|
|
|
|
ret = crosvm_get_clock(crosvm, &clock_data);
|
|
|
|
if (ret < 0) {
|
|
|
|
fprintf(stderr, "failed to get initial clock state: %d\n", ret);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (clock_data.clock == 0 || clock_data.flags != 0) {
|
|
|
|
fprintf(stderr, "invalid clock data returned (%llu, %u)\n",
|
|
|
|
clock_data.clock, clock_data.flags);
|
|
|
|
}
|
|
|
|
|
|
|
|
clock_data.clock += 10000000;
|
|
|
|
|
|
|
|
ret = crosvm_set_clock(crosvm, &clock_data);
|
|
|
|
if (ret < 0) {
|
|
|
|
fprintf(stderr, "failed to update clock: %d\n", ret);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
clock_data.flags = -1U;
|
|
|
|
ret = crosvm_set_clock(crosvm, &clock_data);
|
|
|
|
if (ret >= 0) {
|
|
|
|
fprintf(stderr, "unexpected success updating clock\n");
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2018-03-27 00:14:19 +00:00
|
|
|
return 0;
|
|
|
|
}
|