kvmclock.c 2.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120
  1. /*
  2. * QEMU KVM support, paravirtual clock device
  3. *
  4. * Copyright (C) 2011 Siemens AG
  5. *
  6. * Authors:
  7. * Jan Kiszka <jan.kiszka@siemens.com>
  8. *
  9. * This work is licensed under the terms of the GNU GPL version 2.
  10. * See the COPYING file in the top-level directory.
  11. *
  12. */
  13. #include "qemu-common.h"
  14. #include "sysemu.h"
  15. #include "sysbus.h"
  16. #include "kvm.h"
  17. #include "kvmclock.h"
  18. #include <linux/kvm.h>
  19. #include <linux/kvm_para.h>
  20. typedef struct KVMClockState {
  21. SysBusDevice busdev;
  22. uint64_t clock;
  23. bool clock_valid;
  24. } KVMClockState;
  25. static void kvmclock_pre_save(void *opaque)
  26. {
  27. KVMClockState *s = opaque;
  28. struct kvm_clock_data data;
  29. int ret;
  30. if (s->clock_valid) {
  31. return;
  32. }
  33. ret = kvm_vm_ioctl(kvm_state, KVM_GET_CLOCK, &data);
  34. if (ret < 0) {
  35. fprintf(stderr, "KVM_GET_CLOCK failed: %s\n", strerror(ret));
  36. data.clock = 0;
  37. }
  38. s->clock = data.clock;
  39. /*
  40. * If the VM is stopped, declare the clock state valid to avoid re-reading
  41. * it on next vmsave (which would return a different value). Will be reset
  42. * when the VM is continued.
  43. */
  44. s->clock_valid = !vm_running;
  45. }
  46. static int kvmclock_post_load(void *opaque, int version_id)
  47. {
  48. KVMClockState *s = opaque;
  49. struct kvm_clock_data data;
  50. data.clock = s->clock;
  51. data.flags = 0;
  52. return kvm_vm_ioctl(kvm_state, KVM_SET_CLOCK, &data);
  53. }
  54. static void kvmclock_vm_state_change(void *opaque, int running, int reason)
  55. {
  56. KVMClockState *s = opaque;
  57. if (running) {
  58. s->clock_valid = false;
  59. }
  60. }
  61. static int kvmclock_init(SysBusDevice *dev)
  62. {
  63. KVMClockState *s = FROM_SYSBUS(KVMClockState, dev);
  64. qemu_add_vm_change_state_handler(kvmclock_vm_state_change, s);
  65. return 0;
  66. }
  67. static const VMStateDescription kvmclock_vmsd = {
  68. .name = "kvmclock",
  69. .version_id = 1,
  70. .minimum_version_id = 1,
  71. .minimum_version_id_old = 1,
  72. .pre_save = kvmclock_pre_save,
  73. .post_load = kvmclock_post_load,
  74. .fields = (VMStateField[]) {
  75. VMSTATE_UINT64(clock, KVMClockState),
  76. VMSTATE_END_OF_LIST()
  77. }
  78. };
  79. static SysBusDeviceInfo kvmclock_info = {
  80. .qdev.name = "kvmclock",
  81. .qdev.size = sizeof(KVMClockState),
  82. .qdev.vmsd = &kvmclock_vmsd,
  83. .qdev.no_user = 1,
  84. .init = kvmclock_init,
  85. };
  86. /* Note: Must be called after VCPU initialization. */
  87. void kvmclock_create(void)
  88. {
  89. if (kvm_enabled() &&
  90. first_cpu->cpuid_kvm_features & ((1ULL << KVM_FEATURE_CLOCKSOURCE)
  91. #ifdef KVM_FEATURE_CLOCKSOURCE2
  92. || (1ULL << KVM_FEATURE_CLOCKSOURCE2)
  93. #endif
  94. )) {
  95. sysbus_create_simple("kvmclock", -1, NULL);
  96. }
  97. }
  98. static void kvmclock_register_device(void)
  99. {
  100. if (kvm_enabled()) {
  101. sysbus_register_withprop(&kvmclock_info);
  102. }
  103. }
  104. device_init(kvmclock_register_device);