blob: c131ba4e70ef8229d2c9f1722f88d100f942489e [file] [log] [blame]
Thomas Gleixnerfd534e92019-05-23 11:14:39 +02001// SPDX-License-Identifier: GPL-2.0-or-later
Rusty Russelld3561b72006-12-07 02:14:07 +01002/* Paravirtualization interfaces
3 Copyright (C) 2006 Rusty Russell IBM Corporation
4
Glauber de Oliveira Costab1df07b2008-01-30 13:32:04 +01005
6 2007 - x86_64 support added by Glauber de Oliveira Costa, Red Hat Inc
Rusty Russelld3561b72006-12-07 02:14:07 +01007*/
Glauber de Oliveira Costab1df07b2008-01-30 13:32:04 +01008
Rusty Russelld3561b72006-12-07 02:14:07 +01009#include <linux/errno.h>
Paul Gortmaker186f4362016-07-13 20:18:56 -040010#include <linux/init.h>
11#include <linux/export.h>
Rusty Russelld3561b72006-12-07 02:14:07 +010012#include <linux/efi.h>
13#include <linux/bcd.h>
Jeremy Fitzhardingece6234b2007-05-02 19:27:15 +020014#include <linux/highmem.h>
Masami Hiramatsu376e2422014-04-17 17:17:05 +090015#include <linux/kprobes.h>
Rusty Russelld3561b72006-12-07 02:14:07 +010016
17#include <asm/bug.h>
18#include <asm/paravirt.h>
Paul Gortmaker50af5ea2012-01-20 18:35:53 -050019#include <asm/debugreg.h>
Rusty Russelld3561b72006-12-07 02:14:07 +010020#include <asm/desc.h>
21#include <asm/setup.h>
Eduardo Habkosta312b372008-07-08 15:06:23 -070022#include <asm/pgtable.h>
Rusty Russelld3561b72006-12-07 02:14:07 +010023#include <asm/time.h>
Jeremy Fitzhardingeeba00452008-06-25 00:19:12 -040024#include <asm/pgalloc.h>
Rusty Russelld3561b72006-12-07 02:14:07 +010025#include <asm/irq.h>
26#include <asm/delay.h>
Rusty Russell13623d72006-12-07 02:14:08 +010027#include <asm/fixmap.h>
28#include <asm/apic.h>
Rusty Russellda181a82006-12-07 02:14:08 +010029#include <asm/tlbflush.h>
Zachary Amsden6cb9a832007-03-05 00:30:35 -080030#include <asm/timer.h>
David Howellsf05e7982012-03-28 18:11:12 +010031#include <asm/special_insns.h>
Peter Zijlstra48a8b972018-08-22 17:30:16 +020032#include <asm/tlb.h>
Juergen Gross99bcd4a2020-02-18 16:47:12 +010033#include <asm/io_bitmap.h>
Rusty Russelld3561b72006-12-07 02:14:07 +010034
Andy Lutomirskifc57a7c2015-09-20 16:32:04 -070035/*
36 * nop stub, which must not clobber anything *including the stack* to
37 * avoid confusing the entry prologues.
38 */
39extern void _paravirt_nop(void);
40asm (".pushsection .entry.text, \"ax\"\n"
41 ".global _paravirt_nop\n"
42 "_paravirt_nop:\n\t"
43 "ret\n\t"
44 ".size _paravirt_nop, . - _paravirt_nop\n\t"
45 ".type _paravirt_nop, @function\n\t"
46 ".popsection");
Rusty Russelld3561b72006-12-07 02:14:07 +010047
Thomas Gleixner6f30c1a2009-08-20 13:19:57 +020048void __init default_banner(void)
Rusty Russelld3561b72006-12-07 02:14:07 +010049{
50 printk(KERN_INFO "Booting paravirtualized kernel on %s\n",
Jeremy Fitzhardinge93b1eab2007-10-16 11:51:29 -070051 pv_info.name);
Rusty Russelld3561b72006-12-07 02:14:07 +010052}
53
Jeremy Fitzhardinge93b1eab2007-10-16 11:51:29 -070054/* Undefined instruction for dealing with missing ops pointers. */
55static const unsigned char ud2a[] = { 0x0f, 0x0b };
Rusty Russell139ec7c2006-12-07 02:14:08 +010056
Andi Kleen19d36cc2007-07-22 11:12:31 +020057struct branch {
58 unsigned char opcode;
59 u32 delta;
60} __attribute__((packed));
61
Ingo Molnar1fc654c2019-04-25 13:03:31 +020062static unsigned paravirt_patch_call(void *insn_buff, const void *target,
Juergen Grossabc745f2018-08-28 09:40:17 +020063 unsigned long addr, unsigned len)
Jeremy Fitzhardinge63f70272007-05-02 19:27:14 +020064{
Ingo Molnar11e86dc2019-04-25 11:50:39 +020065 const int call_len = 5;
Ingo Molnar1fc654c2019-04-25 13:03:31 +020066 struct branch *b = insn_buff;
Ingo Molnar11e86dc2019-04-25 11:50:39 +020067 unsigned long delta = (unsigned long)target - (addr+call_len);
Jeremy Fitzhardinge63f70272007-05-02 19:27:14 +020068
Ingo Molnar11e86dc2019-04-25 11:50:39 +020069 if (len < call_len) {
70 pr_warn("paravirt: Failed to patch indirect CALL at %ps\n", (void *)addr);
71 /* Kernel might not be viable if patching fails, bail out: */
72 BUG_ON(1);
Peter Zijlstra5800dc52018-08-03 16:41:39 +020073 }
Jeremy Fitzhardinge63f70272007-05-02 19:27:14 +020074
Andi Kleenab144f52007-08-10 22:31:03 +020075 b->opcode = 0xe8; /* call */
76 b->delta = delta;
Ingo Molnar11e86dc2019-04-25 11:50:39 +020077 BUILD_BUG_ON(sizeof(*b) != call_len);
Jeremy Fitzhardinge63f70272007-05-02 19:27:14 +020078
Ingo Molnar11e86dc2019-04-25 11:50:39 +020079 return call_len;
Jeremy Fitzhardinge63f70272007-05-02 19:27:14 +020080}
81
Juergen Gross9bad5652018-08-28 09:40:23 +020082#ifdef CONFIG_PARAVIRT_XXL
Juergen Gross7847c7b2018-10-30 07:33:01 +010083/* identity function, which can be inlined */
84u64 notrace _paravirt_ident_64(u64 x)
85{
86 return x;
87}
88
Ingo Molnar1fc654c2019-04-25 13:03:31 +020089static unsigned paravirt_patch_jmp(void *insn_buff, const void *target,
Juergen Gross7e437202018-08-28 09:40:16 +020090 unsigned long addr, unsigned len)
Jeremy Fitzhardinge63f70272007-05-02 19:27:14 +020091{
Ingo Molnar1fc654c2019-04-25 13:03:31 +020092 struct branch *b = insn_buff;
Andi Kleenab144f52007-08-10 22:31:03 +020093 unsigned long delta = (unsigned long)target - (addr+5);
Jeremy Fitzhardinge63f70272007-05-02 19:27:14 +020094
Peter Zijlstra5800dc52018-08-03 16:41:39 +020095 if (len < 5) {
96#ifdef CONFIG_RETPOLINE
Dan Carpenter571d0562018-09-19 13:35:53 +030097 WARN_ONCE(1, "Failing to patch indirect JMP in %ps\n", (void *)addr);
Peter Zijlstra5800dc52018-08-03 16:41:39 +020098#endif
Jeremy Fitzhardinge63f70272007-05-02 19:27:14 +020099 return len; /* call too long for patch site */
Peter Zijlstra5800dc52018-08-03 16:41:39 +0200100 }
Jeremy Fitzhardinge63f70272007-05-02 19:27:14 +0200101
Andi Kleenab144f52007-08-10 22:31:03 +0200102 b->opcode = 0xe9; /* jmp */
103 b->delta = delta;
Jeremy Fitzhardinge63f70272007-05-02 19:27:14 +0200104
105 return 5;
106}
Juergen Gross9bad5652018-08-28 09:40:23 +0200107#endif
Jeremy Fitzhardinge63f70272007-05-02 19:27:14 +0200108
Juergen Gross90434422017-09-06 19:36:24 +0200109DEFINE_STATIC_KEY_TRUE(virt_spin_lock_key);
110
111void __init native_pv_lock_init(void)
112{
Borislav Petkov67e87d42019-03-29 19:52:59 +0100113 if (!boot_cpu_has(X86_FEATURE_HYPERVISOR))
Juergen Gross90434422017-09-06 19:36:24 +0200114 static_branch_disable(&virt_spin_lock_key);
115}
116
Ingo Molnar1fc654c2019-04-25 13:03:31 +0200117unsigned paravirt_patch_default(u8 type, void *insn_buff,
Andi Kleenab144f52007-08-10 22:31:03 +0200118 unsigned long addr, unsigned len)
Jeremy Fitzhardinge63f70272007-05-02 19:27:14 +0200119{
Juergen Gross5c835112018-08-28 09:40:19 +0200120 /*
121 * Neat trick to map patch type back to the call within the
122 * corresponding structure.
123 */
124 void *opfunc = *((void **)&pv_ops + type);
Jeremy Fitzhardinge63f70272007-05-02 19:27:14 +0200125 unsigned ret;
126
127 if (opfunc == NULL)
128 /* If there's no function, patch it with a ud2a (BUG) */
Ingo Molnar1fc654c2019-04-25 13:03:31 +0200129 ret = paravirt_patch_insns(insn_buff, len, ud2a, ud2a+sizeof(ud2a));
Jeremy Fitzhardinge41edafd2009-01-28 14:35:02 -0800130 else if (opfunc == _paravirt_nop)
Borislav Petkov79f1d832015-11-03 10:18:49 +0100131 ret = 0;
Jeremy Fitzhardinge41edafd2009-01-28 14:35:02 -0800132
Juergen Gross7847c7b2018-10-30 07:33:01 +0100133#ifdef CONFIG_PARAVIRT_XXL
Jeremy Fitzhardinge41edafd2009-01-28 14:35:02 -0800134 /* identity functions just return their single argument */
Jeremy Fitzhardinge41edafd2009-01-28 14:35:02 -0800135 else if (opfunc == _paravirt_ident_64)
Ingo Molnar1fc654c2019-04-25 13:03:31 +0200136 ret = paravirt_patch_ident_64(insn_buff, len);
Jeremy Fitzhardinge41edafd2009-01-28 14:35:02 -0800137
Juergen Gross5c835112018-08-28 09:40:19 +0200138 else if (type == PARAVIRT_PATCH(cpu.iret) ||
139 type == PARAVIRT_PATCH(cpu.usergs_sysret64))
Jeremy Fitzhardinge63f70272007-05-02 19:27:14 +0200140 /* If operation requires a jmp, then jmp */
Ingo Molnar1fc654c2019-04-25 13:03:31 +0200141 ret = paravirt_patch_jmp(insn_buff, opfunc, addr, len);
Juergen Gross9bad5652018-08-28 09:40:23 +0200142#endif
Jeremy Fitzhardinge63f70272007-05-02 19:27:14 +0200143 else
Juergen Grossabc745f2018-08-28 09:40:17 +0200144 /* Otherwise call the function. */
Ingo Molnar1fc654c2019-04-25 13:03:31 +0200145 ret = paravirt_patch_call(insn_buff, opfunc, addr, len);
Jeremy Fitzhardinge63f70272007-05-02 19:27:14 +0200146
147 return ret;
148}
149
Ingo Molnar1fc654c2019-04-25 13:03:31 +0200150unsigned paravirt_patch_insns(void *insn_buff, unsigned len,
Jeremy Fitzhardinge63f70272007-05-02 19:27:14 +0200151 const char *start, const char *end)
152{
153 unsigned insn_len = end - start;
154
Ingo Molnar2777cae2019-04-25 11:17:17 +0200155 /* Alternative instruction is too large for the patch site and we cannot continue: */
156 BUG_ON(insn_len > len || start == NULL);
157
Ingo Molnar1fc654c2019-04-25 13:03:31 +0200158 memcpy(insn_buff, start, insn_len);
Jeremy Fitzhardinge63f70272007-05-02 19:27:14 +0200159
Rusty Russell139ec7c2006-12-07 02:14:08 +0100160 return insn_len;
161}
162
Andi Kleen1a1eecd2007-02-13 13:26:25 +0100163static void native_flush_tlb(void)
Rusty Russellda181a82006-12-07 02:14:08 +0100164{
165 __native_flush_tlb();
166}
167
168/*
169 * Global pages have to be flushed a bit differently. Not a real
170 * performance problem because this does not happen often.
171 */
Andi Kleen1a1eecd2007-02-13 13:26:25 +0100172static void native_flush_tlb_global(void)
Rusty Russellda181a82006-12-07 02:14:08 +0100173{
174 __native_flush_tlb_global();
175}
176
Andy Lutomirski1299ef12018-01-31 08:03:10 -0800177static void native_flush_tlb_one_user(unsigned long addr)
Rusty Russellda181a82006-12-07 02:14:08 +0100178{
Andy Lutomirski1299ef12018-01-31 08:03:10 -0800179 __native_flush_tlb_one_user(addr);
Rusty Russellda181a82006-12-07 02:14:08 +0100180}
181
Ingo Molnarc5905af2012-02-24 08:31:31 +0100182struct static_key paravirt_steal_enabled;
183struct static_key paravirt_steal_rq_enabled;
Glauber Costa3c404b52011-07-11 15:28:15 -0400184
185static u64 native_steal_clock(int cpu)
186{
187 return 0;
188}
189
Rusty Russelld3561b72006-12-07 02:14:07 +0100190/* These are in entry.S */
Andi Kleen1a1eecd2007-02-13 13:26:25 +0100191extern void native_iret(void);
Jeremy Fitzhardinge2be29982008-06-25 00:19:28 -0400192extern void native_usergs_sysret64(void);
Rusty Russelld3561b72006-12-07 02:14:07 +0100193
Jeremy Fitzhardinged5729292007-07-17 18:37:04 -0700194static struct resource reserve_ioports = {
195 .start = 0,
196 .end = IO_SPACE_LIMIT,
197 .name = "paravirt-ioport",
198 .flags = IORESOURCE_IO | IORESOURCE_BUSY,
199};
200
Jeremy Fitzhardinged5729292007-07-17 18:37:04 -0700201/*
202 * Reserve the whole legacy IO space to prevent any legacy drivers
203 * from wasting time probing for their hardware. This is a fairly
204 * brute-force approach to disabling all non-virtual drivers.
205 *
206 * Note that this must be called very early to have any effect.
207 */
208int paravirt_disable_iospace(void)
209{
Jeremy Fitzhardingef7743fe2008-03-27 17:28:40 -0700210 return request_resource(&ioport_resource, &reserve_ioports);
Jeremy Fitzhardinged5729292007-07-17 18:37:04 -0700211}
212
Jeremy Fitzhardinge8965c1c02007-10-16 11:51:29 -0700213static DEFINE_PER_CPU(enum paravirt_lazy_mode, paravirt_lazy_mode) = PARAVIRT_LAZY_NONE;
214
215static inline void enter_lazy(enum paravirt_lazy_mode mode)
216{
Alex Shic6ae41e2012-05-11 15:35:27 +0800217 BUG_ON(this_cpu_read(paravirt_lazy_mode) != PARAVIRT_LAZY_NONE);
Jeremy Fitzhardinge8965c1c02007-10-16 11:51:29 -0700218
Alex Shic6ae41e2012-05-11 15:35:27 +0800219 this_cpu_write(paravirt_lazy_mode, mode);
Jeremy Fitzhardinge8965c1c02007-10-16 11:51:29 -0700220}
221
Jeremy Fitzhardingeb407fc52009-02-17 23:46:21 -0800222static void leave_lazy(enum paravirt_lazy_mode mode)
Jeremy Fitzhardinge8965c1c02007-10-16 11:51:29 -0700223{
Alex Shic6ae41e2012-05-11 15:35:27 +0800224 BUG_ON(this_cpu_read(paravirt_lazy_mode) != mode);
Jeremy Fitzhardinge8965c1c02007-10-16 11:51:29 -0700225
Alex Shic6ae41e2012-05-11 15:35:27 +0800226 this_cpu_write(paravirt_lazy_mode, PARAVIRT_LAZY_NONE);
Jeremy Fitzhardinge8965c1c02007-10-16 11:51:29 -0700227}
228
229void paravirt_enter_lazy_mmu(void)
230{
231 enter_lazy(PARAVIRT_LAZY_MMU);
232}
233
234void paravirt_leave_lazy_mmu(void)
235{
Jeremy Fitzhardingeb407fc52009-02-17 23:46:21 -0800236 leave_lazy(PARAVIRT_LAZY_MMU);
Jeremy Fitzhardinge8965c1c02007-10-16 11:51:29 -0700237}
238
Boris Ostrovsky511ba862013-03-23 09:36:36 -0400239void paravirt_flush_lazy_mmu(void)
240{
241 preempt_disable();
242
243 if (paravirt_get_lazy_mode() == PARAVIRT_LAZY_MMU) {
244 arch_leave_lazy_mmu_mode();
245 arch_enter_lazy_mmu_mode();
246 }
247
248 preempt_enable();
249}
250
Juergen Gross9bad5652018-08-28 09:40:23 +0200251#ifdef CONFIG_PARAVIRT_XXL
Jeremy Fitzhardinge224101e2009-02-18 11:18:57 -0800252void paravirt_start_context_switch(struct task_struct *prev)
Jeremy Fitzhardinge8965c1c02007-10-16 11:51:29 -0700253{
Jeremy Fitzhardinge2829b442009-02-17 23:53:19 -0800254 BUG_ON(preemptible());
255
Alex Shic6ae41e2012-05-11 15:35:27 +0800256 if (this_cpu_read(paravirt_lazy_mode) == PARAVIRT_LAZY_MMU) {
Jeremy Fitzhardingeb407fc52009-02-17 23:46:21 -0800257 arch_leave_lazy_mmu_mode();
Jeremy Fitzhardinge224101e2009-02-18 11:18:57 -0800258 set_ti_thread_flag(task_thread_info(prev), TIF_LAZY_MMU_UPDATES);
Jeremy Fitzhardingeb407fc52009-02-17 23:46:21 -0800259 }
Jeremy Fitzhardinge8965c1c02007-10-16 11:51:29 -0700260 enter_lazy(PARAVIRT_LAZY_CPU);
261}
262
Jeremy Fitzhardinge224101e2009-02-18 11:18:57 -0800263void paravirt_end_context_switch(struct task_struct *next)
Jeremy Fitzhardinge8965c1c02007-10-16 11:51:29 -0700264{
Jeremy Fitzhardinge2829b442009-02-17 23:53:19 -0800265 BUG_ON(preemptible());
266
Jeremy Fitzhardingeb407fc52009-02-17 23:46:21 -0800267 leave_lazy(PARAVIRT_LAZY_CPU);
268
Jeremy Fitzhardinge224101e2009-02-18 11:18:57 -0800269 if (test_and_clear_ti_thread_flag(task_thread_info(next), TIF_LAZY_MMU_UPDATES))
Jeremy Fitzhardingeb407fc52009-02-17 23:46:21 -0800270 arch_enter_lazy_mmu_mode();
Jeremy Fitzhardinge8965c1c02007-10-16 11:51:29 -0700271}
Juergen Gross9bad5652018-08-28 09:40:23 +0200272#endif
Jeremy Fitzhardinge8965c1c02007-10-16 11:51:29 -0700273
274enum paravirt_lazy_mode paravirt_get_lazy_mode(void)
275{
Jeremy Fitzhardingeb8bcfe92009-02-17 23:05:19 -0800276 if (in_interrupt())
277 return PARAVIRT_LAZY_NONE;
278
Alex Shic6ae41e2012-05-11 15:35:27 +0800279 return this_cpu_read(paravirt_lazy_mode);
Jeremy Fitzhardinge8965c1c02007-10-16 11:51:29 -0700280}
281
Jeremy Fitzhardinge93b1eab2007-10-16 11:51:29 -0700282struct pv_info pv_info = {
Rusty Russelld3561b72006-12-07 02:14:07 +0100283 .name = "bare hardware",
Juergen Gross40181642018-08-28 09:40:22 +0200284#ifdef CONFIG_PARAVIRT_XXL
Rusty Russelld3561b72006-12-07 02:14:07 +0100285 .kernel_rpl = 0,
Jeremy Fitzhardinge5311ab62007-05-02 19:27:13 +0200286 .shared_kernel_pmd = 1, /* Only used when CONFIG_X86_PAE is set */
Andy Lutomirski318f5a22011-08-03 09:31:53 -0400287
288#ifdef CONFIG_X86_64
289 .extra_user_64bit_cs = __USER_CS,
290#endif
Glauber de Oliveira Costa88b47552008-01-30 13:33:19 +0100291#endif
Jeremy Fitzhardinge93b1eab2007-10-16 11:51:29 -0700292};
Rusty Russelld3561b72006-12-07 02:14:07 +0100293
Jeremy Fitzhardinge41edafd2009-01-28 14:35:02 -0800294/* 64-bit pagetable entries */
Jeremy Fitzhardingeda5de7c2009-01-28 14:35:07 -0800295#define PTE_IDENT __PV_IS_CALLEE_SAVE(_paravirt_ident_64)
Jeremy Fitzhardinge41edafd2009-01-28 14:35:02 -0800296
Juergen Gross5c835112018-08-28 09:40:19 +0200297struct paravirt_patch_template pv_ops = {
298 /* Init ops. */
299 .init.patch = native_patch,
Jeremy Fitzhardingeb239fb22007-05-02 19:27:13 +0200300
Juergen Gross5c835112018-08-28 09:40:19 +0200301 /* Time ops. */
302 .time.sched_clock = native_sched_clock,
303 .time.steal_clock = native_steal_clock,
Jeremy Fitzhardinge93b1eab2007-10-16 11:51:29 -0700304
Juergen Gross5c835112018-08-28 09:40:19 +0200305 /* Cpu ops. */
Juergen Gross9bad5652018-08-28 09:40:23 +0200306 .cpu.io_delay = native_io_delay,
Rusty Russellda181a82006-12-07 02:14:08 +0100307
Juergen Gross9bad5652018-08-28 09:40:23 +0200308#ifdef CONFIG_PARAVIRT_XXL
Juergen Gross5c835112018-08-28 09:40:19 +0200309 .cpu.cpuid = native_cpuid,
310 .cpu.get_debugreg = native_get_debugreg,
311 .cpu.set_debugreg = native_set_debugreg,
312 .cpu.read_cr0 = native_read_cr0,
313 .cpu.write_cr0 = native_write_cr0,
314 .cpu.write_cr4 = native_write_cr4,
Juergen Gross5c835112018-08-28 09:40:19 +0200315 .cpu.wbinvd = native_wbinvd,
316 .cpu.read_msr = native_read_msr,
317 .cpu.write_msr = native_write_msr,
318 .cpu.read_msr_safe = native_read_msr_safe,
319 .cpu.write_msr_safe = native_write_msr_safe,
320 .cpu.read_pmc = native_read_pmc,
321 .cpu.load_tr_desc = native_load_tr_desc,
322 .cpu.set_ldt = native_set_ldt,
323 .cpu.load_gdt = native_load_gdt,
324 .cpu.load_idt = native_load_idt,
325 .cpu.store_tr = native_store_tr,
326 .cpu.load_tls = native_load_tls,
327#ifdef CONFIG_X86_64
328 .cpu.load_gs_index = native_load_gs_index,
329#endif
330 .cpu.write_ldt_entry = native_write_ldt_entry,
331 .cpu.write_gdt_entry = native_write_gdt_entry,
332 .cpu.write_idt_entry = native_write_idt_entry,
Jeremy Fitzhardingeeba00452008-06-25 00:19:12 -0400333
Juergen Gross5c835112018-08-28 09:40:19 +0200334 .cpu.alloc_ldt = paravirt_nop,
335 .cpu.free_ldt = paravirt_nop,
Zachary Amsdenc119ecc2007-02-13 13:26:21 +0100336
Juergen Gross5c835112018-08-28 09:40:19 +0200337 .cpu.load_sp0 = native_load_sp0,
Jeremy Fitzhardinge3dc494e2007-05-02 19:27:13 +0200338
Juergen Gross5c835112018-08-28 09:40:19 +0200339#ifdef CONFIG_X86_64
340 .cpu.usergs_sysret64 = native_usergs_sysret64,
341#endif
342 .cpu.iret = native_iret,
343 .cpu.swapgs = native_swapgs,
Jeremy Fitzhardinge3dc494e2007-05-02 19:27:13 +0200344
Juergen Gross99bcd4a2020-02-18 16:47:12 +0100345#ifdef CONFIG_X86_IOPL_IOPERM
346 .cpu.update_io_bitmap = native_tss_update_io_bitmap,
347#endif
348
Juergen Gross5c835112018-08-28 09:40:19 +0200349 .cpu.start_context_switch = paravirt_nop,
350 .cpu.end_context_switch = paravirt_nop,
351
352 /* Irq ops. */
353 .irq.save_fl = __PV_IS_CALLEE_SAVE(native_save_fl),
354 .irq.restore_fl = __PV_IS_CALLEE_SAVE(native_restore_fl),
355 .irq.irq_disable = __PV_IS_CALLEE_SAVE(native_irq_disable),
356 .irq.irq_enable = __PV_IS_CALLEE_SAVE(native_irq_enable),
357 .irq.safe_halt = native_safe_halt,
358 .irq.halt = native_halt,
Juergen Gross6da63eb2018-08-28 09:40:24 +0200359#endif /* CONFIG_PARAVIRT_XXL */
Juergen Gross5c835112018-08-28 09:40:19 +0200360
361 /* Mmu ops. */
Juergen Gross5c835112018-08-28 09:40:19 +0200362 .mmu.flush_tlb_user = native_flush_tlb,
363 .mmu.flush_tlb_kernel = native_flush_tlb_global,
364 .mmu.flush_tlb_one_user = native_flush_tlb_one_user,
365 .mmu.flush_tlb_others = native_flush_tlb_others,
366 .mmu.tlb_remove_table =
367 (void (*)(struct mmu_gather *, void *))tlb_remove_page,
368
Juergen Grossfdc02692018-08-28 09:40:25 +0200369 .mmu.exit_mmap = paravirt_nop,
370
371#ifdef CONFIG_PARAVIRT_XXL
Peter Zijlstra55aeddd2019-07-11 13:40:55 +0200372 .mmu.read_cr2 = __PV_IS_CALLEE_SAVE(native_read_cr2),
Juergen Grossfdc02692018-08-28 09:40:25 +0200373 .mmu.write_cr2 = native_write_cr2,
374 .mmu.read_cr3 = __native_read_cr3,
375 .mmu.write_cr3 = native_write_cr3,
376
Juergen Gross5c835112018-08-28 09:40:19 +0200377 .mmu.pgd_alloc = __paravirt_pgd_alloc,
378 .mmu.pgd_free = paravirt_nop,
379
380 .mmu.alloc_pte = paravirt_nop,
381 .mmu.alloc_pmd = paravirt_nop,
382 .mmu.alloc_pud = paravirt_nop,
383 .mmu.alloc_p4d = paravirt_nop,
384 .mmu.release_pte = paravirt_nop,
385 .mmu.release_pmd = paravirt_nop,
386 .mmu.release_pud = paravirt_nop,
387 .mmu.release_p4d = paravirt_nop,
388
389 .mmu.set_pte = native_set_pte,
390 .mmu.set_pte_at = native_set_pte_at,
391 .mmu.set_pmd = native_set_pmd,
392
393 .mmu.ptep_modify_prot_start = __ptep_modify_prot_start,
394 .mmu.ptep_modify_prot_commit = __ptep_modify_prot_commit,
Jeremy Fitzhardinge08b882c2008-06-16 04:30:01 -0700395
Kirill A. Shutemov98233362015-04-14 15:46:14 -0700396#if CONFIG_PGTABLE_LEVELS >= 3
Rusty Russellda181a82006-12-07 02:14:08 +0100397#ifdef CONFIG_X86_PAE
Juergen Gross5c835112018-08-28 09:40:19 +0200398 .mmu.set_pte_atomic = native_set_pte_atomic,
399 .mmu.pte_clear = native_pte_clear,
400 .mmu.pmd_clear = native_pmd_clear,
Eduardo Habkostf95f2f72008-01-30 13:33:20 +0100401#endif
Juergen Gross5c835112018-08-28 09:40:19 +0200402 .mmu.set_pud = native_set_pud,
Jeremy Fitzhardingeda5de7c2009-01-28 14:35:07 -0800403
Juergen Gross5c835112018-08-28 09:40:19 +0200404 .mmu.pmd_val = PTE_IDENT,
405 .mmu.make_pmd = PTE_IDENT,
Eduardo Habkostf95f2f72008-01-30 13:33:20 +0100406
Kirill A. Shutemovf2a6a702017-03-17 21:55:15 +0300407#if CONFIG_PGTABLE_LEVELS >= 4
Juergen Gross5c835112018-08-28 09:40:19 +0200408 .mmu.pud_val = PTE_IDENT,
409 .mmu.make_pud = PTE_IDENT,
Jeremy Fitzhardingeda5de7c2009-01-28 14:35:07 -0800410
Juergen Gross5c835112018-08-28 09:40:19 +0200411 .mmu.set_p4d = native_set_p4d,
Kirill A. Shutemovf2a6a702017-03-17 21:55:15 +0300412
413#if CONFIG_PGTABLE_LEVELS >= 5
Juergen Gross5c835112018-08-28 09:40:19 +0200414 .mmu.p4d_val = PTE_IDENT,
415 .mmu.make_p4d = PTE_IDENT,
Kirill A. Shutemov335437f2017-03-30 11:07:28 +0300416
Juergen Gross5c835112018-08-28 09:40:19 +0200417 .mmu.set_pgd = native_set_pgd,
Kirill A. Shutemov335437f2017-03-30 11:07:28 +0300418#endif /* CONFIG_PGTABLE_LEVELS >= 5 */
Kirill A. Shutemovf2a6a702017-03-17 21:55:15 +0300419#endif /* CONFIG_PGTABLE_LEVELS >= 4 */
Kirill A. Shutemov98233362015-04-14 15:46:14 -0700420#endif /* CONFIG_PGTABLE_LEVELS >= 3 */
Rusty Russellda181a82006-12-07 02:14:08 +0100421
Juergen Gross5c835112018-08-28 09:40:19 +0200422 .mmu.pte_val = PTE_IDENT,
423 .mmu.pgd_val = PTE_IDENT,
Jeremy Fitzhardinge3dc494e2007-05-02 19:27:13 +0200424
Juergen Gross5c835112018-08-28 09:40:19 +0200425 .mmu.make_pte = PTE_IDENT,
426 .mmu.make_pgd = PTE_IDENT,
Jeremy Fitzhardinge3dc494e2007-05-02 19:27:13 +0200427
Juergen Gross5c835112018-08-28 09:40:19 +0200428 .mmu.dup_mmap = paravirt_nop,
Juergen Gross5c835112018-08-28 09:40:19 +0200429 .mmu.activate_mm = paravirt_nop,
Jeremy Fitzhardinge8965c1c02007-10-16 11:51:29 -0700430
Juergen Gross5c835112018-08-28 09:40:19 +0200431 .mmu.lazy_mode = {
432 .enter = paravirt_nop,
433 .leave = paravirt_nop,
434 .flush = paravirt_nop,
Jeremy Fitzhardinge8965c1c02007-10-16 11:51:29 -0700435 },
Jeremy Fitzhardingeaeaaa592008-06-17 11:42:01 -0700436
Juergen Gross5c835112018-08-28 09:40:19 +0200437 .mmu.set_fixmap = native_set_fixmap,
Juergen Grossfdc02692018-08-28 09:40:25 +0200438#endif /* CONFIG_PARAVIRT_XXL */
Juergen Gross5c835112018-08-28 09:40:19 +0200439
440#if defined(CONFIG_PARAVIRT_SPINLOCKS)
441 /* Lock ops. */
442#ifdef CONFIG_SMP
443 .lock.queued_spin_lock_slowpath = native_queued_spin_lock_slowpath,
444 .lock.queued_spin_unlock =
445 PV_CALLEE_SAVE(__native_queued_spin_unlock),
446 .lock.wait = paravirt_nop,
447 .lock.kick = paravirt_nop,
448 .lock.vcpu_is_preempted =
449 PV_CALLEE_SAVE(__native_vcpu_is_preempted),
450#endif /* SMP */
451#endif
Rusty Russelld3561b72006-12-07 02:14:07 +0100452};
Ingo Molnar0dbe5a12007-01-22 20:40:36 -0800453
Juergen Gross9bad5652018-08-28 09:40:23 +0200454#ifdef CONFIG_PARAVIRT_XXL
Juergen Gross5c835112018-08-28 09:40:19 +0200455/* At this point, native_get/set_debugreg has real function entries */
456NOKPROBE_SYMBOL(native_get_debugreg);
457NOKPROBE_SYMBOL(native_set_debugreg);
458NOKPROBE_SYMBOL(native_load_idt);
Juergen Gross9bad5652018-08-28 09:40:23 +0200459#endif
Juergen Gross5c835112018-08-28 09:40:19 +0200460
Juergen Gross8af19092018-10-29 16:01:16 +0100461EXPORT_SYMBOL(pv_ops);
Jeremy Fitzhardinge93b1eab2007-10-16 11:51:29 -0700462EXPORT_SYMBOL_GPL(pv_info);