linux/arch/x86/kvm/svm/svm.c

#define pr_fmt(fmt)

#include <linux/kvm_host.h>

#include "irq.h"
#include "mmu.h"
#include "kvm_cache_regs.h"
#include "x86.h"
#include "smm.h"
#include "cpuid.h"
#include "pmu.h"

#include <linux/module.h>
#include <linux/mod_devicetable.h>
#include <linux/kernel.h>
#include <linux/vmalloc.h>
#include <linux/highmem.h>
#include <linux/amd-iommu.h>
#include <linux/sched.h>
#include <linux/trace_events.h>
#include <linux/slab.h>
#include <linux/hashtable.h>
#include <linux/objtool.h>
#include <linux/psp-sev.h>
#include <linux/file.h>
#include <linux/pagemap.h>
#include <linux/swap.h>
#include <linux/rwsem.h>
#include <linux/cc_platform.h>
#include <linux/smp.h>

#include <asm/apic.h>
#include <asm/perf_event.h>
#include <asm/tlbflush.h>
#include <asm/desc.h>
#include <asm/debugreg.h>
#include <asm/kvm_para.h>
#include <asm/irq_remapping.h>
#include <asm/spec-ctrl.h>
#include <asm/cpu_device_id.h>
#include <asm/traps.h>
#include <asm/reboot.h>
#include <asm/fpu/api.h>

#include <trace/events/ipi.h>

#include "trace.h"

#include "svm.h"
#include "svm_ops.h"

#include "kvm_onhyperv.h"
#include "svm_onhyperv.h"

MODULE_AUTHOR();
MODULE_DESCRIPTION();
MODULE_LICENSE();

#ifdef MODULE
static const struct x86_cpu_id svm_cpu_id[] = {
	X86_MATCH_FEATURE(X86_FEATURE_SVM, NULL),
	{}
};
MODULE_DEVICE_TABLE(x86cpu, svm_cpu_id);
#endif

#define SEG_TYPE_LDT
#define SEG_TYPE_BUSY_TSS16

static bool erratum_383_found __read_mostly;

u32 msrpm_offsets[MSRPM_OFFSETS] __read_mostly;

/*
 * Set osvw_len to higher value when updated Revision Guides
 * are published and we know what the new status bits are
 */
static uint64_t osvw_len =, osvw_status;

static DEFINE_PER_CPU(u64, current_tsc_ratio);

#define X2APIC_MSR(x)

static const struct svm_direct_access_msrs {} direct_access_msrs[MAX_DIRECT_ACCESS_MSRS] =;

/*
 * These 2 parameters are used to config the controls for Pause-Loop Exiting:
 * pause_filter_count: On processors that support Pause filtering(indicated
 *	by CPUID Fn8000_000A_EDX), the VMCB provides a 16 bit pause filter
 *	count value. On VMRUN this value is loaded into an internal counter.
 *	Each time a pause instruction is executed, this counter is decremented
 *	until it reaches zero at which time a #VMEXIT is generated if pause
 *	intercept is enabled. Refer to  AMD APM Vol 2 Section 15.14.4 Pause
 *	Intercept Filtering for more details.
 *	This also indicate if ple logic enabled.
 *
 * pause_filter_thresh: In addition, some processor families support advanced
 *	pause filtering (indicated by CPUID Fn8000_000A_EDX) upper bound on
 *	the amount of time a guest is allowed to execute in a pause loop.
 *	In this mode, a 16-bit pause filter threshold field is added in the
 *	VMCB. The threshold value is a cycle count that is used to reset the
 *	pause counter. As with simple pause filtering, VMRUN loads the pause
 *	count value from VMCB into an internal counter. Then, on each pause
 *	instruction the hardware checks the elapsed number of cycles since
 *	the most recent pause instruction against the pause filter threshold.
 *	If the elapsed cycle count is greater than the pause filter threshold,
 *	then the internal pause count is reloaded from the VMCB and execution
 *	continues. If the elapsed cycle count is less than the pause filter
 *	threshold, then the internal pause count is decremented. If the count
 *	value is less than zero and PAUSE intercept is enabled, a #VMEXIT is
 *	triggered. If advanced pause filtering is supported and pause filter
 *	threshold field is set to zero, the filter will operate in the simpler,
 *	count only mode.
 */

static unsigned short pause_filter_thresh =;
module_param(pause_filter_thresh, ushort, 0444);

static unsigned short pause_filter_count =;
module_param(pause_filter_count, ushort, 0444);

/* Default doubles per-vcpu window every exit. */
static unsigned short pause_filter_count_grow =;
module_param(pause_filter_count_grow, ushort, 0444);

/* Default resets per-vcpu window every exit to pause_filter_count. */
static unsigned short pause_filter_count_shrink =;
module_param(pause_filter_count_shrink, ushort, 0444);

/* Default is to compute the maximum so we can never overflow. */
static unsigned short pause_filter_count_max =;
module_param(pause_filter_count_max, ushort, 0444);

/*
 * Use nested page tables by default.  Note, NPT may get forced off by
 * svm_hardware_setup() if it's unsupported by hardware or the host kernel.
 */
bool npt_enabled =;
module_param_named(npt, npt_enabled, bool, 0444);

/* allow nested virtualization in KVM/SVM */
static int nested =;
module_param(nested, int, 0444);

/* enable/disable Next RIP Save */
int nrips =;
module_param(nrips, int, 0444);

/* enable/disable Virtual VMLOAD VMSAVE */
static int vls =;
module_param(vls, int, 0444);

/* enable/disable Virtual GIF */
int vgif =;
module_param(vgif, int, 0444);

/* enable/disable LBR virtualization */
int lbrv =;
module_param(lbrv, int, 0444);

static int tsc_scaling =;
module_param(tsc_scaling, int, 0444);

/*
 * enable / disable AVIC.  Because the defaults differ for APICv
 * support between VMX and SVM we cannot use module_param_named.
 */
static bool avic;
module_param(avic, bool, 0444);

bool __read_mostly dump_invalid_vmcb;
module_param(dump_invalid_vmcb, bool, 0644);


bool intercept_smi =;
module_param(intercept_smi, bool, 0444);

bool vnmi =;
module_param(vnmi, bool, 0444);

static bool svm_gp_erratum_intercept =;

static u8 rsm_ins_bytes[] =;

static unsigned long iopm_base;

DEFINE_PER_CPU(struct svm_cpu_data, svm_data);

/*
 * Only MSR_TSC_AUX is switched via the user return hook.  EFER is switched via
 * the VMCB, and the SYSCALL/SYSENTER MSRs are handled by VMLOAD/VMSAVE.
 *
 * RDTSCP and RDPID are not used in the kernel, specifically to allow KVM to
 * defer the restoration of TSC_AUX until the CPU returns to userspace.
 */
static int tsc_aux_uret_slot __read_mostly =;

static const u32 msrpm_ranges[] =;

#define NUM_MSR_MAPS
#define MSRS_RANGE_SIZE
#define MSRS_IN_RANGE

u32 svm_msrpm_offset(u32 msr)
{}

static void svm_flush_tlb_current(struct kvm_vcpu *vcpu);

static int get_npt_level(void)
{}

int svm_set_efer(struct kvm_vcpu *vcpu, u64 efer)
{}

static u32 svm_get_interrupt_shadow(struct kvm_vcpu *vcpu)
{}

static void svm_set_interrupt_shadow(struct kvm_vcpu *vcpu, int mask)
{}

static int __svm_skip_emulated_instruction(struct kvm_vcpu *vcpu,
					   bool commit_side_effects)
{}

static int svm_skip_emulated_instruction(struct kvm_vcpu *vcpu)
{}

static int svm_update_soft_interrupt_rip(struct kvm_vcpu *vcpu)
{}

static void svm_inject_exception(struct kvm_vcpu *vcpu)
{}

static void svm_init_erratum_383(void)
{}

static void svm_init_osvw(struct kvm_vcpu *vcpu)
{}

static bool __kvm_is_svm_supported(void)
{}

static bool kvm_is_svm_supported(void)
{}

static int svm_check_processor_compat(void)
{}

static void __svm_write_tsc_multiplier(u64 multiplier)
{}

static __always_inline struct sev_es_save_area *sev_es_host_save_area(struct svm_cpu_data *sd)
{}

static inline void kvm_cpu_svm_disable(void)
{}

static void svm_emergency_disable(void)
{}

static void svm_hardware_disable(void)
{}

static int svm_hardware_enable(void)
{}

static void svm_cpu_uninit(int cpu)
{}

static int svm_cpu_init(int cpu)
{}

static void set_dr_intercepts(struct vcpu_svm *svm)
{}

static void clr_dr_intercepts(struct vcpu_svm *svm)
{}

static int direct_access_msr_slot(u32 msr)
{}

static void set_shadow_msr_intercept(struct kvm_vcpu *vcpu, u32 msr, int read,
				     int write)
{}

static bool valid_msr_intercept(u32 index)
{}

static bool msr_write_intercepted(struct kvm_vcpu *vcpu, u32 msr)
{}

static void set_msr_interception_bitmap(struct kvm_vcpu *vcpu, u32 *msrpm,
					u32 msr, int read, int write)
{}

void set_msr_interception(struct kvm_vcpu *vcpu, u32 *msrpm, u32 msr,
			  int read, int write)
{}

u32 *svm_vcpu_alloc_msrpm(void)
{}

void svm_vcpu_init_msrpm(struct kvm_vcpu *vcpu, u32 *msrpm)
{}

void svm_set_x2apic_msr_interception(struct vcpu_svm *svm, bool intercept)
{}

void svm_vcpu_free_msrpm(u32 *msrpm)
{}

static void svm_msr_filter_changed(struct kvm_vcpu *vcpu)
{}

static void add_msr_offset(u32 offset)
{}

static void init_msrpm_offsets(void)
{}

void svm_copy_lbrs(struct vmcb *to_vmcb, struct vmcb *from_vmcb)
{}

void svm_enable_lbrv(struct kvm_vcpu *vcpu)
{}

static void svm_disable_lbrv(struct kvm_vcpu *vcpu)
{}

static struct vmcb *svm_get_lbr_vmcb(struct vcpu_svm *svm)
{}

void svm_update_lbrv(struct kvm_vcpu *vcpu)
{}

void disable_nmi_singlestep(struct vcpu_svm *svm)
{}

static void grow_ple_window(struct kvm_vcpu *vcpu)
{}

static void shrink_ple_window(struct kvm_vcpu *vcpu)
{}

static void svm_hardware_unsetup(void)
{}

static void init_seg(struct vmcb_seg *seg)
{}

static void init_sys_seg(struct vmcb_seg *seg, uint32_t type)
{}

static u64 svm_get_l2_tsc_offset(struct kvm_vcpu *vcpu)
{}

static u64 svm_get_l2_tsc_multiplier(struct kvm_vcpu *vcpu)
{}

static void svm_write_tsc_offset(struct kvm_vcpu *vcpu)
{}

void svm_write_tsc_multiplier(struct kvm_vcpu *vcpu)
{}

/* Evaluate instruction intercepts that depend on guest CPUID features. */
static void svm_recalc_instruction_intercepts(struct kvm_vcpu *vcpu,
					      struct vcpu_svm *svm)
{}

static inline void init_vmcb_after_set_cpuid(struct kvm_vcpu *vcpu)
{}

static void init_vmcb(struct kvm_vcpu *vcpu)
{}

static void __svm_vcpu_reset(struct kvm_vcpu *vcpu)
{}

static void svm_vcpu_reset(struct kvm_vcpu *vcpu, bool init_event)
{}

void svm_switch_vmcb(struct vcpu_svm *svm, struct kvm_vmcb_info *target_vmcb)
{}

static int svm_vcpu_create(struct kvm_vcpu *vcpu)
{}

static void svm_clear_current_vmcb(struct vmcb *vmcb)
{}

static void svm_vcpu_free(struct kvm_vcpu *vcpu)
{}

static void svm_prepare_switch_to_guest(struct kvm_vcpu *vcpu)
{}

static void svm_prepare_host_switch(struct kvm_vcpu *vcpu)
{}

static void svm_vcpu_load(struct kvm_vcpu *vcpu, int cpu)
{}

static void svm_vcpu_put(struct kvm_vcpu *vcpu)
{}

static unsigned long svm_get_rflags(struct kvm_vcpu *vcpu)
{}

static void svm_set_rflags(struct kvm_vcpu *vcpu, unsigned long rflags)
{}

static bool svm_get_if_flag(struct kvm_vcpu *vcpu)
{}

static void svm_cache_reg(struct kvm_vcpu *vcpu, enum kvm_reg reg)
{}

static void svm_set_vintr(struct vcpu_svm *svm)
{}

static void svm_clear_vintr(struct vcpu_svm *svm)
{}

static struct vmcb_seg *svm_seg(struct kvm_vcpu *vcpu, int seg)
{}

static u64 svm_get_segment_base(struct kvm_vcpu *vcpu, int seg)
{}

static void svm_get_segment(struct kvm_vcpu *vcpu,
			    struct kvm_segment *var, int seg)
{}

static int svm_get_cpl(struct kvm_vcpu *vcpu)
{}

static void svm_get_cs_db_l_bits(struct kvm_vcpu *vcpu, int *db, int *l)
{}

static void svm_get_idt(struct kvm_vcpu *vcpu, struct desc_ptr *dt)
{}

static void svm_set_idt(struct kvm_vcpu *vcpu, struct desc_ptr *dt)
{}

static void svm_get_gdt(struct kvm_vcpu *vcpu, struct desc_ptr *dt)
{}

static void svm_set_gdt(struct kvm_vcpu *vcpu, struct desc_ptr *dt)
{}

static void sev_post_set_cr3(struct kvm_vcpu *vcpu, unsigned long cr3)
{}

static bool svm_is_valid_cr0(struct kvm_vcpu *vcpu, unsigned long cr0)
{}

void svm_set_cr0(struct kvm_vcpu *vcpu, unsigned long cr0)
{}

static bool svm_is_valid_cr4(struct kvm_vcpu *vcpu, unsigned long cr4)
{}

void svm_set_cr4(struct kvm_vcpu *vcpu, unsigned long cr4)
{}

static void svm_set_segment(struct kvm_vcpu *vcpu,
			    struct kvm_segment *var, int seg)
{}

static void svm_update_exception_bitmap(struct kvm_vcpu *vcpu)
{}

static void new_asid(struct vcpu_svm *svm, struct svm_cpu_data *sd)
{}

static void svm_set_dr6(struct vcpu_svm *svm, unsigned long value)
{}

static void svm_sync_dirty_debug_regs(struct kvm_vcpu *vcpu)
{}

static void svm_set_dr7(struct kvm_vcpu *vcpu, unsigned long value)
{}

static int pf_interception(struct kvm_vcpu *vcpu)
{}

static int npf_interception(struct kvm_vcpu *vcpu)
{}

static int db_interception(struct kvm_vcpu *vcpu)
{}

static int bp_interception(struct kvm_vcpu *vcpu)
{}

static int ud_interception(struct kvm_vcpu *vcpu)
{}

static int ac_interception(struct kvm_vcpu *vcpu)
{}

static bool is_erratum_383(void)
{}

static void svm_handle_mce(struct kvm_vcpu *vcpu)
{}

static int mc_interception(struct kvm_vcpu *vcpu)
{}

static int shutdown_interception(struct kvm_vcpu *vcpu)
{}

static int io_interception(struct kvm_vcpu *vcpu)
{}

static int nmi_interception(struct kvm_vcpu *vcpu)
{}

static int smi_interception(struct kvm_vcpu *vcpu)
{}

static int intr_interception(struct kvm_vcpu *vcpu)
{}

static int vmload_vmsave_interception(struct kvm_vcpu *vcpu, bool vmload)
{}

static int vmload_interception(struct kvm_vcpu *vcpu)
{}

static int vmsave_interception(struct kvm_vcpu *vcpu)
{}

static int vmrun_interception(struct kvm_vcpu *vcpu)
{}

enum {};

/* Return NONE_SVM_INSTR if not SVM instrs, otherwise return decode result */
static int svm_instr_opcode(struct kvm_vcpu *vcpu)
{}

static int emulate_svm_instr(struct kvm_vcpu *vcpu, int opcode)
{}

/*
 * #GP handling code. Note that #GP can be triggered under the following two
 * cases:
 *   1) SVM VM-related instructions (VMRUN/VMSAVE/VMLOAD) that trigger #GP on
 *      some AMD CPUs when EAX of these instructions are in the reserved memory
 *      regions (e.g. SMM memory on host).
 *   2) VMware backdoor
 */
static int gp_interception(struct kvm_vcpu *vcpu)
{}

void svm_set_gif(struct vcpu_svm *svm, bool value)
{}

static int stgi_interception(struct kvm_vcpu *vcpu)
{}

static int clgi_interception(struct kvm_vcpu *vcpu)
{}

static int invlpga_interception(struct kvm_vcpu *vcpu)
{}

static int skinit_interception(struct kvm_vcpu *vcpu)
{}

static int task_switch_interception(struct kvm_vcpu *vcpu)
{}

static void svm_clr_iret_intercept(struct vcpu_svm *svm)
{}

static void svm_set_iret_intercept(struct vcpu_svm *svm)
{}

static int iret_interception(struct kvm_vcpu *vcpu)
{}

static int invlpg_interception(struct kvm_vcpu *vcpu)
{}

static int emulate_on_interception(struct kvm_vcpu *vcpu)
{}

static int rsm_interception(struct kvm_vcpu *vcpu)
{}

static bool check_selective_cr0_intercepted(struct kvm_vcpu *vcpu,
					    unsigned long val)
{}

#define CR_VALID

static int cr_interception(struct kvm_vcpu *vcpu)
{}

static int cr_trap(struct kvm_vcpu *vcpu)
{}

static int dr_interception(struct kvm_vcpu *vcpu)
{}

static int cr8_write_interception(struct kvm_vcpu *vcpu)
{}

static int efer_trap(struct kvm_vcpu *vcpu)
{}

static int svm_get_msr_feature(struct kvm_msr_entry *msr)
{}

static bool
sev_es_prevent_msr_access(struct kvm_vcpu *vcpu, struct msr_data *msr_info)
{}

static int svm_get_msr(struct kvm_vcpu *vcpu, struct msr_data *msr_info)
{}

static int svm_complete_emulated_msr(struct kvm_vcpu *vcpu, int err)
{}

static int svm_set_vm_cr(struct kvm_vcpu *vcpu, u64 data)
{}

static int svm_set_msr(struct kvm_vcpu *vcpu, struct msr_data *msr)
{}

static int msr_interception(struct kvm_vcpu *vcpu)
{}

static int interrupt_window_interception(struct kvm_vcpu *vcpu)
{}

static int pause_interception(struct kvm_vcpu *vcpu)
{}

static int invpcid_interception(struct kvm_vcpu *vcpu)
{}

static int (*const svm_exit_handlers[])(struct kvm_vcpu *vcpu) =;

static void dump_vmcb(struct kvm_vcpu *vcpu)
{}

static bool svm_check_exit_valid(u64 exit_code)
{}

static int svm_handle_invalid_exit(struct kvm_vcpu *vcpu, u64 exit_code)
{}

int svm_invoke_exit_handler(struct kvm_vcpu *vcpu, u64 exit_code)
{}

static void svm_get_exit_info(struct kvm_vcpu *vcpu, u32 *reason,
			      u64 *info1, u64 *info2,
			      u32 *intr_info, u32 *error_code)
{}

static int svm_handle_exit(struct kvm_vcpu *vcpu, fastpath_t exit_fastpath)
{}

static void pre_svm_run(struct kvm_vcpu *vcpu)
{}

static void svm_inject_nmi(struct kvm_vcpu *vcpu)
{}

static bool svm_is_vnmi_pending(struct kvm_vcpu *vcpu)
{}

static bool svm_set_vnmi_pending(struct kvm_vcpu *vcpu)
{}

static void svm_inject_irq(struct kvm_vcpu *vcpu, bool reinjected)
{}

void svm_complete_interrupt_delivery(struct kvm_vcpu *vcpu, int delivery_mode,
				     int trig_mode, int vector)
{}

static void svm_deliver_interrupt(struct kvm_lapic *apic,  int delivery_mode,
				  int trig_mode, int vector)
{}

static void svm_update_cr8_intercept(struct kvm_vcpu *vcpu, int tpr, int irr)
{}

static bool svm_get_nmi_mask(struct kvm_vcpu *vcpu)
{}

static void svm_set_nmi_mask(struct kvm_vcpu *vcpu, bool masked)
{}

bool svm_nmi_blocked(struct kvm_vcpu *vcpu)
{}

static int svm_nmi_allowed(struct kvm_vcpu *vcpu, bool for_injection)
{}

bool svm_interrupt_blocked(struct kvm_vcpu *vcpu)
{}

static int svm_interrupt_allowed(struct kvm_vcpu *vcpu, bool for_injection)
{}

static void svm_enable_irq_window(struct kvm_vcpu *vcpu)
{}

static void svm_enable_nmi_window(struct kvm_vcpu *vcpu)
{}

static void svm_flush_tlb_asid(struct kvm_vcpu *vcpu)
{}

static void svm_flush_tlb_current(struct kvm_vcpu *vcpu)
{}

static void svm_flush_tlb_all(struct kvm_vcpu *vcpu)
{}

static void svm_flush_tlb_gva(struct kvm_vcpu *vcpu, gva_t gva)
{}

static inline void sync_cr8_to_lapic(struct kvm_vcpu *vcpu)
{}

static inline void sync_lapic_to_cr8(struct kvm_vcpu *vcpu)
{}

static void svm_complete_soft_interrupt(struct kvm_vcpu *vcpu, u8 vector,
					int type)
{}

static void svm_complete_interrupts(struct kvm_vcpu *vcpu)
{}

static void svm_cancel_injection(struct kvm_vcpu *vcpu)
{}

static int svm_vcpu_pre_run(struct kvm_vcpu *vcpu)
{}

static fastpath_t svm_exit_handlers_fastpath(struct kvm_vcpu *vcpu)
{}

static noinstr void svm_vcpu_enter_exit(struct kvm_vcpu *vcpu, bool spec_ctrl_intercepted)
{}

static __no_kcsan fastpath_t svm_vcpu_run(struct kvm_vcpu *vcpu,
					  bool force_immediate_exit)
{}

static void svm_load_mmu_pgd(struct kvm_vcpu *vcpu, hpa_t root_hpa,
			     int root_level)
{}

static void
svm_patch_hypercall(struct kvm_vcpu *vcpu, unsigned char *hypercall)
{}

/*
 * The kvm parameter can be NULL (module initialization, or invocation before
 * VM creation). Be sure to check the kvm parameter before using it.
 */
static bool svm_has_emulated_msr(struct kvm *kvm, u32 index)
{}

static void svm_vcpu_after_set_cpuid(struct kvm_vcpu *vcpu)
{}

static bool svm_has_wbinvd_exit(void)
{}

#define PRE_EX
#define POST_EX
#define POST_MEM

static const struct __x86_intercept {} x86_intercept_map[] =;

#undef PRE_EX
#undef POST_EX
#undef POST_MEM

static int svm_check_intercept(struct kvm_vcpu *vcpu,
			       struct x86_instruction_info *info,
			       enum x86_intercept_stage stage,
			       struct x86_exception *exception)
{}

static void svm_handle_exit_irqoff(struct kvm_vcpu *vcpu)
{}

static void svm_setup_mce(struct kvm_vcpu *vcpu)
{}

#ifdef CONFIG_KVM_SMM
bool svm_smi_blocked(struct kvm_vcpu *vcpu)
{}

static int svm_smi_allowed(struct kvm_vcpu *vcpu, bool for_injection)
{}

static int svm_enter_smm(struct kvm_vcpu *vcpu, union kvm_smram *smram)
{}

static int svm_leave_smm(struct kvm_vcpu *vcpu, const union kvm_smram *smram)
{}

static void svm_enable_smi_window(struct kvm_vcpu *vcpu)
{}
#endif

static int svm_check_emulate_instruction(struct kvm_vcpu *vcpu, int emul_type,
					 void *insn, int insn_len)
{}

static bool svm_apic_init_signal_blocked(struct kvm_vcpu *vcpu)
{}

static void svm_vcpu_deliver_sipi_vector(struct kvm_vcpu *vcpu, u8 vector)
{}

static void svm_vm_destroy(struct kvm *kvm)
{}

static int svm_vm_init(struct kvm *kvm)
{}

static void *svm_alloc_apic_backing_page(struct kvm_vcpu *vcpu)
{}

static struct kvm_x86_ops svm_x86_ops __initdata =;

/*
 * The default MMIO mask is a single bit (excluding the present bit),
 * which could conflict with the memory encryption bit. Check for
 * memory encryption support and override the default MMIO mask if
 * memory encryption is enabled.
 */
static __init void svm_adjust_mmio_mask(void)
{}

static __init void svm_set_cpu_caps(void)
{}

static __init int svm_hardware_setup(void)
{}


static struct kvm_x86_init_ops svm_init_ops __initdata =;

static void __svm_exit(void)
{}

static int __init svm_init(void)
{}

static void __exit svm_exit(void)
{}

module_init()
module_exit()