diff options
Diffstat (limited to 'arch')
-rw-r--r-- | arch/x86/include/asm/hypervisor.h | 3 | ||||
-rw-r--r-- | arch/x86/include/asm/xen/hypervisor.h | 35 | ||||
-rw-r--r-- | arch/x86/xen/enlighten.c | 31 |
3 files changed, 50 insertions, 19 deletions
diff --git a/arch/x86/include/asm/hypervisor.h b/arch/x86/include/asm/hypervisor.h index 0c6f7af7fda8..7a15153c675d 100644 --- a/arch/x86/include/asm/hypervisor.h +++ b/arch/x86/include/asm/hypervisor.h | |||
@@ -21,6 +21,7 @@ | |||
21 | #define _ASM_X86_HYPERVISOR_H | 21 | #define _ASM_X86_HYPERVISOR_H |
22 | 22 | ||
23 | #include <asm/kvm_para.h> | 23 | #include <asm/kvm_para.h> |
24 | #include <asm/xen/hypervisor.h> | ||
24 | 25 | ||
25 | extern void init_hypervisor(struct cpuinfo_x86 *c); | 26 | extern void init_hypervisor(struct cpuinfo_x86 *c); |
26 | extern void init_hypervisor_platform(void); | 27 | extern void init_hypervisor_platform(void); |
@@ -53,6 +54,8 @@ static inline bool hypervisor_x2apic_available(void) | |||
53 | { | 54 | { |
54 | if (kvm_para_available()) | 55 | if (kvm_para_available()) |
55 | return true; | 56 | return true; |
57 | if (xen_x2apic_para_available()) | ||
58 | return true; | ||
56 | return false; | 59 | return false; |
57 | } | 60 | } |
58 | 61 | ||
diff --git a/arch/x86/include/asm/xen/hypervisor.h b/arch/x86/include/asm/xen/hypervisor.h index 396ff4cc8ed4..66d0fff1ee84 100644 --- a/arch/x86/include/asm/xen/hypervisor.h +++ b/arch/x86/include/asm/xen/hypervisor.h | |||
@@ -37,4 +37,39 @@ | |||
37 | extern struct shared_info *HYPERVISOR_shared_info; | 37 | extern struct shared_info *HYPERVISOR_shared_info; |
38 | extern struct start_info *xen_start_info; | 38 | extern struct start_info *xen_start_info; |
39 | 39 | ||
40 | #include <asm/processor.h> | ||
41 | |||
42 | static inline uint32_t xen_cpuid_base(void) | ||
43 | { | ||
44 | uint32_t base, eax, ebx, ecx, edx; | ||
45 | char signature[13]; | ||
46 | |||
47 | for (base = 0x40000000; base < 0x40010000; base += 0x100) { | ||
48 | cpuid(base, &eax, &ebx, &ecx, &edx); | ||
49 | *(uint32_t *)(signature + 0) = ebx; | ||
50 | *(uint32_t *)(signature + 4) = ecx; | ||
51 | *(uint32_t *)(signature + 8) = edx; | ||
52 | signature[12] = 0; | ||
53 | |||
54 | if (!strcmp("XenVMMXenVMM", signature) && ((eax - base) >= 2)) | ||
55 | return base; | ||
56 | } | ||
57 | |||
58 | return 0; | ||
59 | } | ||
60 | |||
61 | #ifdef CONFIG_XEN | ||
62 | extern bool xen_hvm_need_lapic(void); | ||
63 | |||
64 | static inline bool xen_x2apic_para_available(void) | ||
65 | { | ||
66 | return xen_hvm_need_lapic(); | ||
67 | } | ||
68 | #else | ||
69 | static inline bool xen_x2apic_para_available(void) | ||
70 | { | ||
71 | return (xen_cpuid_base() != 0); | ||
72 | } | ||
73 | #endif | ||
74 | |||
40 | #endif /* _ASM_X86_XEN_HYPERVISOR_H */ | 75 | #endif /* _ASM_X86_XEN_HYPERVISOR_H */ |
diff --git a/arch/x86/xen/enlighten.c b/arch/x86/xen/enlighten.c index 44dcad43989d..5b8986fe3371 100644 --- a/arch/x86/xen/enlighten.c +++ b/arch/x86/xen/enlighten.c | |||
@@ -1256,25 +1256,6 @@ asmlinkage void __init xen_start_kernel(void) | |||
1256 | #endif | 1256 | #endif |
1257 | } | 1257 | } |
1258 | 1258 | ||
1259 | static uint32_t xen_cpuid_base(void) | ||
1260 | { | ||
1261 | uint32_t base, eax, ebx, ecx, edx; | ||
1262 | char signature[13]; | ||
1263 | |||
1264 | for (base = 0x40000000; base < 0x40010000; base += 0x100) { | ||
1265 | cpuid(base, &eax, &ebx, &ecx, &edx); | ||
1266 | *(uint32_t *)(signature + 0) = ebx; | ||
1267 | *(uint32_t *)(signature + 4) = ecx; | ||
1268 | *(uint32_t *)(signature + 8) = edx; | ||
1269 | signature[12] = 0; | ||
1270 | |||
1271 | if (!strcmp("XenVMMXenVMM", signature) && ((eax - base) >= 2)) | ||
1272 | return base; | ||
1273 | } | ||
1274 | |||
1275 | return 0; | ||
1276 | } | ||
1277 | |||
1278 | static int init_hvm_pv_info(int *major, int *minor) | 1259 | static int init_hvm_pv_info(int *major, int *minor) |
1279 | { | 1260 | { |
1280 | uint32_t eax, ebx, ecx, edx, pages, msr, base; | 1261 | uint32_t eax, ebx, ecx, edx, pages, msr, base; |
@@ -1384,6 +1365,18 @@ static bool __init xen_hvm_platform(void) | |||
1384 | return true; | 1365 | return true; |
1385 | } | 1366 | } |
1386 | 1367 | ||
1368 | bool xen_hvm_need_lapic(void) | ||
1369 | { | ||
1370 | if (xen_pv_domain()) | ||
1371 | return false; | ||
1372 | if (!xen_hvm_domain()) | ||
1373 | return false; | ||
1374 | if (xen_feature(XENFEAT_hvm_pirqs) && xen_have_vector_callback) | ||
1375 | return false; | ||
1376 | return true; | ||
1377 | } | ||
1378 | EXPORT_SYMBOL_GPL(xen_hvm_need_lapic); | ||
1379 | |||
1387 | const __refconst struct hypervisor_x86 x86_hyper_xen_hvm = { | 1380 | const __refconst struct hypervisor_x86 x86_hyper_xen_hvm = { |
1388 | .name = "Xen HVM", | 1381 | .name = "Xen HVM", |
1389 | .detect = xen_hvm_platform, | 1382 | .detect = xen_hvm_platform, |