diff options
-rw-r--r-- | arch/x86/include/asm/paravirt.h | 6 | ||||
-rw-r--r-- | arch/x86/include/asm/paravirt_types.h | 5 | ||||
-rw-r--r-- | arch/x86/include/asm/processor.h | 1 | ||||
-rw-r--r-- | arch/x86/kernel/rtc.c | 3 | ||||
-rw-r--r-- | arch/x86/lguest/boot.c | 1 | ||||
-rw-r--r-- | arch/x86/xen/enlighten.c | 4 |
6 files changed, 19 insertions, 1 deletions
diff --git a/arch/x86/include/asm/paravirt.h b/arch/x86/include/asm/paravirt.h index 10d0596433f8..c759b3cca663 100644 --- a/arch/x86/include/asm/paravirt.h +++ b/arch/x86/include/asm/paravirt.h | |||
@@ -19,6 +19,12 @@ static inline int paravirt_enabled(void) | |||
19 | return pv_info.paravirt_enabled; | 19 | return pv_info.paravirt_enabled; |
20 | } | 20 | } |
21 | 21 | ||
22 | static inline int paravirt_has_feature(unsigned int feature) | ||
23 | { | ||
24 | WARN_ON_ONCE(!pv_info.paravirt_enabled); | ||
25 | return (pv_info.features & feature); | ||
26 | } | ||
27 | |||
22 | static inline void load_sp0(struct tss_struct *tss, | 28 | static inline void load_sp0(struct tss_struct *tss, |
23 | struct thread_struct *thread) | 29 | struct thread_struct *thread) |
24 | { | 30 | { |
diff --git a/arch/x86/include/asm/paravirt_types.h b/arch/x86/include/asm/paravirt_types.h index 31247b5bff7c..3d44191185f8 100644 --- a/arch/x86/include/asm/paravirt_types.h +++ b/arch/x86/include/asm/paravirt_types.h | |||
@@ -70,9 +70,14 @@ struct pv_info { | |||
70 | #endif | 70 | #endif |
71 | 71 | ||
72 | int paravirt_enabled; | 72 | int paravirt_enabled; |
73 | unsigned int features; /* valid only if paravirt_enabled is set */ | ||
73 | const char *name; | 74 | const char *name; |
74 | }; | 75 | }; |
75 | 76 | ||
77 | #define paravirt_has(x) paravirt_has_feature(PV_SUPPORTED_##x) | ||
78 | /* Supported features */ | ||
79 | #define PV_SUPPORTED_RTC (1<<0) | ||
80 | |||
76 | struct pv_init_ops { | 81 | struct pv_init_ops { |
77 | /* | 82 | /* |
78 | * Patch may replace one of the defined code sequences with | 83 | * Patch may replace one of the defined code sequences with |
diff --git a/arch/x86/include/asm/processor.h b/arch/x86/include/asm/processor.h index 67522256c7ff..2d5a50cb61a2 100644 --- a/arch/x86/include/asm/processor.h +++ b/arch/x86/include/asm/processor.h | |||
@@ -472,6 +472,7 @@ static inline unsigned long current_top_of_stack(void) | |||
472 | #else | 472 | #else |
473 | #define __cpuid native_cpuid | 473 | #define __cpuid native_cpuid |
474 | #define paravirt_enabled() 0 | 474 | #define paravirt_enabled() 0 |
475 | #define paravirt_has(x) 0 | ||
475 | 476 | ||
476 | static inline void load_sp0(struct tss_struct *tss, | 477 | static inline void load_sp0(struct tss_struct *tss, |
477 | struct thread_struct *thread) | 478 | struct thread_struct *thread) |
diff --git a/arch/x86/kernel/rtc.c b/arch/x86/kernel/rtc.c index cd9685235df9..4af8d063fb36 100644 --- a/arch/x86/kernel/rtc.c +++ b/arch/x86/kernel/rtc.c | |||
@@ -200,6 +200,9 @@ static __init int add_rtc_cmos(void) | |||
200 | } | 200 | } |
201 | #endif | 201 | #endif |
202 | 202 | ||
203 | if (paravirt_enabled() && !paravirt_has(RTC)) | ||
204 | return -ENODEV; | ||
205 | |||
203 | platform_device_register(&rtc_device); | 206 | platform_device_register(&rtc_device); |
204 | dev_info(&rtc_device.dev, | 207 | dev_info(&rtc_device.dev, |
205 | "registered platform RTC device (no PNP device found)\n"); | 208 | "registered platform RTC device (no PNP device found)\n"); |
diff --git a/arch/x86/lguest/boot.c b/arch/x86/lguest/boot.c index a0d09f6c6533..a43b2eafc466 100644 --- a/arch/x86/lguest/boot.c +++ b/arch/x86/lguest/boot.c | |||
@@ -1414,6 +1414,7 @@ __init void lguest_init(void) | |||
1414 | pv_info.kernel_rpl = 1; | 1414 | pv_info.kernel_rpl = 1; |
1415 | /* Everyone except Xen runs with this set. */ | 1415 | /* Everyone except Xen runs with this set. */ |
1416 | pv_info.shared_kernel_pmd = 1; | 1416 | pv_info.shared_kernel_pmd = 1; |
1417 | pv_info.features = 0; | ||
1417 | 1418 | ||
1418 | /* | 1419 | /* |
1419 | * We set up all the lguest overrides for sensitive operations. These | 1420 | * We set up all the lguest overrides for sensitive operations. These |
diff --git a/arch/x86/xen/enlighten.c b/arch/x86/xen/enlighten.c index d315151411e5..b7de78bdc09c 100644 --- a/arch/x86/xen/enlighten.c +++ b/arch/x86/xen/enlighten.c | |||
@@ -1192,7 +1192,7 @@ static const struct pv_info xen_info __initconst = { | |||
1192 | #ifdef CONFIG_X86_64 | 1192 | #ifdef CONFIG_X86_64 |
1193 | .extra_user_64bit_cs = FLAT_USER_CS64, | 1193 | .extra_user_64bit_cs = FLAT_USER_CS64, |
1194 | #endif | 1194 | #endif |
1195 | 1195 | .features = 0, | |
1196 | .name = "Xen", | 1196 | .name = "Xen", |
1197 | }; | 1197 | }; |
1198 | 1198 | ||
@@ -1535,6 +1535,8 @@ asmlinkage __visible void __init xen_start_kernel(void) | |||
1535 | 1535 | ||
1536 | /* Install Xen paravirt ops */ | 1536 | /* Install Xen paravirt ops */ |
1537 | pv_info = xen_info; | 1537 | pv_info = xen_info; |
1538 | if (xen_initial_domain()) | ||
1539 | pv_info.features |= PV_SUPPORTED_RTC; | ||
1538 | pv_init_ops = xen_init_ops; | 1540 | pv_init_ops = xen_init_ops; |
1539 | pv_apic_ops = xen_apic_ops; | 1541 | pv_apic_ops = xen_apic_ops; |
1540 | if (!xen_pvh_domain()) { | 1542 | if (!xen_pvh_domain()) { |