aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/kvm/x86.c
diff options
context:
space:
mode:
Diffstat (limited to 'drivers/kvm/x86.c')
-rw-r--r--drivers/kvm/x86.c102
1 files changed, 102 insertions, 0 deletions
diff --git a/drivers/kvm/x86.c b/drivers/kvm/x86.c
new file mode 100644
index 000000000000..437902cf178d
--- /dev/null
+++ b/drivers/kvm/x86.c
@@ -0,0 +1,102 @@
1/*
2 * Kernel-based Virtual Machine driver for Linux
3 *
4 * derived from drivers/kvm/kvm_main.c
5 *
6 * Copyright (C) 2006 Qumranet, Inc.
7 *
8 * Authors:
9 * Avi Kivity <avi@qumranet.com>
10 * Yaniv Kamay <yaniv@qumranet.com>
11 *
12 * This work is licensed under the terms of the GNU GPL, version 2. See
13 * the COPYING file in the top-level directory.
14 *
15 */
16
17#include "x86.h"
18
19#include <asm/uaccess.h>
20
21/*
22 * List of msr numbers which we expose to userspace through KVM_GET_MSRS
23 * and KVM_SET_MSRS, and KVM_GET_MSR_INDEX_LIST.
24 *
25 * This list is modified at module load time to reflect the
26 * capabilities of the host cpu.
27 */
28static u32 msrs_to_save[] = {
29 MSR_IA32_SYSENTER_CS, MSR_IA32_SYSENTER_ESP, MSR_IA32_SYSENTER_EIP,
30 MSR_K6_STAR,
31#ifdef CONFIG_X86_64
32 MSR_CSTAR, MSR_KERNEL_GS_BASE, MSR_SYSCALL_MASK, MSR_LSTAR,
33#endif
34 MSR_IA32_TIME_STAMP_COUNTER,
35};
36
37static unsigned num_msrs_to_save;
38
39static u32 emulated_msrs[] = {
40 MSR_IA32_MISC_ENABLE,
41};
42
43long kvm_arch_dev_ioctl(struct file *filp,
44 unsigned int ioctl, unsigned long arg)
45{
46 void __user *argp = (void __user *)arg;
47 long r;
48
49 switch (ioctl) {
50 case KVM_GET_MSR_INDEX_LIST: {
51 struct kvm_msr_list __user *user_msr_list = argp;
52 struct kvm_msr_list msr_list;
53 unsigned n;
54
55 r = -EFAULT;
56 if (copy_from_user(&msr_list, user_msr_list, sizeof msr_list))
57 goto out;
58 n = msr_list.nmsrs;
59 msr_list.nmsrs = num_msrs_to_save + ARRAY_SIZE(emulated_msrs);
60 if (copy_to_user(user_msr_list, &msr_list, sizeof msr_list))
61 goto out;
62 r = -E2BIG;
63 if (n < num_msrs_to_save)
64 goto out;
65 r = -EFAULT;
66 if (copy_to_user(user_msr_list->indices, &msrs_to_save,
67 num_msrs_to_save * sizeof(u32)))
68 goto out;
69 if (copy_to_user(user_msr_list->indices
70 + num_msrs_to_save * sizeof(u32),
71 &emulated_msrs,
72 ARRAY_SIZE(emulated_msrs) * sizeof(u32)))
73 goto out;
74 r = 0;
75 break;
76 }
77 default:
78 r = -EINVAL;
79 }
80out:
81 return r;
82}
83
84static __init void kvm_init_msr_list(void)
85{
86 u32 dummy[2];
87 unsigned i, j;
88
89 for (i = j = 0; i < ARRAY_SIZE(msrs_to_save); i++) {
90 if (rdmsr_safe(msrs_to_save[i], &dummy[0], &dummy[1]) < 0)
91 continue;
92 if (j < i)
93 msrs_to_save[j] = msrs_to_save[i];
94 j++;
95 }
96 num_msrs_to_save = j;
97}
98
99__init void kvm_arch_init(void)
100{
101 kvm_init_msr_list();
102}