diff options
author | Glauber de Oliveira Costa <gcosta@redhat.com> | 2008-01-30 07:32:10 -0500 |
---|---|---|
committer | Ingo Molnar <mingo@elte.hu> | 2008-01-30 07:32:10 -0500 |
commit | 2f485ef568372af4680c4e2f8490efb9f2523b05 (patch) | |
tree | d1a244d31333524d2636affe4a862b8058ae4f92 | |
parent | 21438f7c138f0b893a32df3cc77434e39a2145f8 (diff) |
x86: move patching code to arch-specific file.
The core patching code for paravirt is sufficiently different
among i386 and x86_64, and we move them to specific files.
Signed-off-by: Glauber de Oliveira Costa <gcosta@redhat.com>
Signed-off-by: Ingo Molnar <mingo@elte.hu>
Signed-off-by: Thomas Gleixner <tglx@linutronix.de>
-rw-r--r-- | arch/x86/kernel/Makefile_32 | 2 | ||||
-rw-r--r-- | arch/x86/kernel/paravirt.c | 50 | ||||
-rw-r--r-- | arch/x86/kernel/paravirt_patch_32.c | 49 | ||||
-rw-r--r-- | include/asm-x86/paravirt.h | 8 |
4 files changed, 58 insertions, 51 deletions
diff --git a/arch/x86/kernel/Makefile_32 b/arch/x86/kernel/Makefile_32 index cfb71a5deb19..86c6327798b2 100644 --- a/arch/x86/kernel/Makefile_32 +++ b/arch/x86/kernel/Makefile_32 | |||
@@ -48,7 +48,7 @@ obj-$(CONFIG_K8_NB) += k8.o | |||
48 | obj-$(CONFIG_MGEODE_LX) += geode_32.o mfgpt_32.o | 48 | obj-$(CONFIG_MGEODE_LX) += geode_32.o mfgpt_32.o |
49 | 49 | ||
50 | obj-$(CONFIG_VMI) += vmi_32.o vmiclock_32.o | 50 | obj-$(CONFIG_VMI) += vmi_32.o vmiclock_32.o |
51 | obj-$(CONFIG_PARAVIRT) += paravirt.o | 51 | obj-$(CONFIG_PARAVIRT) += paravirt.o paravirt_patch_32.o |
52 | obj-y += pcspeaker.o | 52 | obj-y += pcspeaker.o |
53 | 53 | ||
54 | obj-$(CONFIG_SCx200) += scx200_32.o | 54 | obj-$(CONFIG_SCx200) += scx200_32.o |
diff --git a/arch/x86/kernel/paravirt.c b/arch/x86/kernel/paravirt.c index e7c17cc4a99e..864be0498a32 100644 --- a/arch/x86/kernel/paravirt.c +++ b/arch/x86/kernel/paravirt.c | |||
@@ -58,59 +58,9 @@ char *memory_setup(void) | |||
58 | extern const char start_##ops##_##name[], end_##ops##_##name[]; \ | 58 | extern const char start_##ops##_##name[], end_##ops##_##name[]; \ |
59 | asm("start_" #ops "_" #name ": " code "; end_" #ops "_" #name ":") | 59 | asm("start_" #ops "_" #name ": " code "; end_" #ops "_" #name ":") |
60 | 60 | ||
61 | DEF_NATIVE(pv_irq_ops, irq_disable, "cli"); | ||
62 | DEF_NATIVE(pv_irq_ops, irq_enable, "sti"); | ||
63 | DEF_NATIVE(pv_irq_ops, restore_fl, "push %eax; popf"); | ||
64 | DEF_NATIVE(pv_irq_ops, save_fl, "pushf; pop %eax"); | ||
65 | DEF_NATIVE(pv_cpu_ops, iret, "iret"); | ||
66 | DEF_NATIVE(pv_cpu_ops, irq_enable_syscall_ret, "sti; sysexit"); | ||
67 | DEF_NATIVE(pv_mmu_ops, read_cr2, "mov %cr2, %eax"); | ||
68 | DEF_NATIVE(pv_mmu_ops, write_cr3, "mov %eax, %cr3"); | ||
69 | DEF_NATIVE(pv_mmu_ops, read_cr3, "mov %cr3, %eax"); | ||
70 | DEF_NATIVE(pv_cpu_ops, clts, "clts"); | ||
71 | DEF_NATIVE(pv_cpu_ops, read_tsc, "rdtsc"); | ||
72 | |||
73 | /* Undefined instruction for dealing with missing ops pointers. */ | 61 | /* Undefined instruction for dealing with missing ops pointers. */ |
74 | static const unsigned char ud2a[] = { 0x0f, 0x0b }; | 62 | static const unsigned char ud2a[] = { 0x0f, 0x0b }; |
75 | 63 | ||
76 | static unsigned native_patch(u8 type, u16 clobbers, void *ibuf, | ||
77 | unsigned long addr, unsigned len) | ||
78 | { | ||
79 | const unsigned char *start, *end; | ||
80 | unsigned ret; | ||
81 | |||
82 | switch(type) { | ||
83 | #define SITE(ops, x) \ | ||
84 | case PARAVIRT_PATCH(ops.x): \ | ||
85 | start = start_##ops##_##x; \ | ||
86 | end = end_##ops##_##x; \ | ||
87 | goto patch_site | ||
88 | |||
89 | SITE(pv_irq_ops, irq_disable); | ||
90 | SITE(pv_irq_ops, irq_enable); | ||
91 | SITE(pv_irq_ops, restore_fl); | ||
92 | SITE(pv_irq_ops, save_fl); | ||
93 | SITE(pv_cpu_ops, iret); | ||
94 | SITE(pv_cpu_ops, irq_enable_syscall_ret); | ||
95 | SITE(pv_mmu_ops, read_cr2); | ||
96 | SITE(pv_mmu_ops, read_cr3); | ||
97 | SITE(pv_mmu_ops, write_cr3); | ||
98 | SITE(pv_cpu_ops, clts); | ||
99 | SITE(pv_cpu_ops, read_tsc); | ||
100 | #undef SITE | ||
101 | |||
102 | patch_site: | ||
103 | ret = paravirt_patch_insns(ibuf, len, start, end); | ||
104 | break; | ||
105 | |||
106 | default: | ||
107 | ret = paravirt_patch_default(type, clobbers, ibuf, addr, len); | ||
108 | break; | ||
109 | } | ||
110 | |||
111 | return ret; | ||
112 | } | ||
113 | |||
114 | unsigned paravirt_patch_nop(void) | 64 | unsigned paravirt_patch_nop(void) |
115 | { | 65 | { |
116 | return 0; | 66 | return 0; |
diff --git a/arch/x86/kernel/paravirt_patch_32.c b/arch/x86/kernel/paravirt_patch_32.c new file mode 100644 index 000000000000..82fc5fcab4f4 --- /dev/null +++ b/arch/x86/kernel/paravirt_patch_32.c | |||
@@ -0,0 +1,49 @@ | |||
1 | #include <asm/paravirt.h> | ||
2 | |||
3 | DEF_NATIVE(pv_irq_ops, irq_disable, "cli"); | ||
4 | DEF_NATIVE(pv_irq_ops, irq_enable, "sti"); | ||
5 | DEF_NATIVE(pv_irq_ops, restore_fl, "push %eax; popf"); | ||
6 | DEF_NATIVE(pv_irq_ops, save_fl, "pushf; pop %eax"); | ||
7 | DEF_NATIVE(pv_cpu_ops, iret, "iret"); | ||
8 | DEF_NATIVE(pv_cpu_ops, irq_enable_syscall_ret, "sti; sysexit"); | ||
9 | DEF_NATIVE(pv_mmu_ops, read_cr2, "mov %cr2, %eax"); | ||
10 | DEF_NATIVE(pv_mmu_ops, write_cr3, "mov %eax, %cr3"); | ||
11 | DEF_NATIVE(pv_mmu_ops, read_cr3, "mov %cr3, %eax"); | ||
12 | DEF_NATIVE(pv_cpu_ops, clts, "clts"); | ||
13 | DEF_NATIVE(pv_cpu_ops, read_tsc, "rdtsc"); | ||
14 | |||
15 | unsigned native_patch(u8 type, u16 clobbers, void *ibuf, | ||
16 | unsigned long addr, unsigned len) | ||
17 | { | ||
18 | const unsigned char *start, *end; | ||
19 | unsigned ret; | ||
20 | |||
21 | #define PATCH_SITE(ops, x) \ | ||
22 | case PARAVIRT_PATCH(ops.x): \ | ||
23 | start = start_##ops##_##x; \ | ||
24 | end = end_##ops##_##x; \ | ||
25 | goto patch_site | ||
26 | switch(type) { | ||
27 | PATCH_SITE(pv_irq_ops, irq_disable); | ||
28 | PATCH_SITE(pv_irq_ops, irq_enable); | ||
29 | PATCH_SITE(pv_irq_ops, restore_fl); | ||
30 | PATCH_SITE(pv_irq_ops, save_fl); | ||
31 | PATCH_SITE(pv_cpu_ops, iret); | ||
32 | PATCH_SITE(pv_cpu_ops, irq_enable_syscall_ret); | ||
33 | PATCH_SITE(pv_mmu_ops, read_cr2); | ||
34 | PATCH_SITE(pv_mmu_ops, read_cr3); | ||
35 | PATCH_SITE(pv_mmu_ops, write_cr3); | ||
36 | PATCH_SITE(pv_cpu_ops, clts); | ||
37 | PATCH_SITE(pv_cpu_ops, read_tsc); | ||
38 | |||
39 | patch_site: | ||
40 | ret = paravirt_patch_insns(ibuf, len, start, end); | ||
41 | break; | ||
42 | |||
43 | default: | ||
44 | ret = paravirt_patch_default(type, clobbers, ibuf, addr, len); | ||
45 | break; | ||
46 | } | ||
47 | #undef PATCH_SITE | ||
48 | return ret; | ||
49 | } | ||
diff --git a/include/asm-x86/paravirt.h b/include/asm-x86/paravirt.h index 5935c273af1f..b35ed95166e4 100644 --- a/include/asm-x86/paravirt.h +++ b/include/asm-x86/paravirt.h | |||
@@ -308,6 +308,11 @@ extern struct pv_mmu_ops pv_mmu_ops; | |||
308 | #define paravirt_alt(insn_string) \ | 308 | #define paravirt_alt(insn_string) \ |
309 | _paravirt_alt(insn_string, "%c[paravirt_typenum]", "%c[paravirt_clobber]") | 309 | _paravirt_alt(insn_string, "%c[paravirt_typenum]", "%c[paravirt_clobber]") |
310 | 310 | ||
311 | /* Simple instruction patching code. */ | ||
312 | #define DEF_NATIVE(ops, name, code) \ | ||
313 | extern const char start_##ops##_##name[], end_##ops##_##name[]; \ | ||
314 | asm("start_" #ops "_" #name ": " code "; end_" #ops "_" #name ":") | ||
315 | |||
311 | unsigned paravirt_patch_nop(void); | 316 | unsigned paravirt_patch_nop(void); |
312 | unsigned paravirt_patch_ignore(unsigned len); | 317 | unsigned paravirt_patch_ignore(unsigned len); |
313 | unsigned paravirt_patch_call(void *insnbuf, | 318 | unsigned paravirt_patch_call(void *insnbuf, |
@@ -322,6 +327,9 @@ unsigned paravirt_patch_default(u8 type, u16 clobbers, void *insnbuf, | |||
322 | unsigned paravirt_patch_insns(void *insnbuf, unsigned len, | 327 | unsigned paravirt_patch_insns(void *insnbuf, unsigned len, |
323 | const char *start, const char *end); | 328 | const char *start, const char *end); |
324 | 329 | ||
330 | unsigned native_patch(u8 type, u16 clobbers, void *ibuf, | ||
331 | unsigned long addr, unsigned len); | ||
332 | |||
325 | int paravirt_disable_iospace(void); | 333 | int paravirt_disable_iospace(void); |
326 | 334 | ||
327 | /* | 335 | /* |