2008-11-18 00:03:16 +03:00
/* CPU virtualization extensions handling
*
* This should carry the code for handling CPU virtualization extensions
* that needs to live in the kernel core .
*
* Author : Eduardo Habkost < ehabkost @ redhat . com >
*
* Copyright ( C ) 2008 , Red Hat Inc .
*
* Contains code from KVM , Copyright ( C ) 2006 Qumranet , Inc .
*
* This work is licensed under the terms of the GNU GPL , version 2. See
* the COPYING file in the top - level directory .
*/
# ifndef _ASM_X86_VIRTEX_H
# define _ASM_X86_VIRTEX_H
# include <asm/processor.h>
2008-11-18 00:03:17 +03:00
# include <asm/vmx.h>
2008-11-18 00:03:20 +03:00
# include <asm/svm.h>
2014-10-25 02:58:07 +04:00
# include <asm/tlbflush.h>
2008-11-18 00:03:17 +03:00
2008-11-18 00:03:16 +03:00
/*
* VMX functions :
*/
static inline int cpu_has_vmx ( void )
{
unsigned long ecx = cpuid_ecx ( 1 ) ;
return test_bit ( 5 , & ecx ) ; /* CPUID.1:ECX.VMX[bit 5] -> VT */
}
2008-11-18 00:03:17 +03:00
/** Disable VMX on the current CPU
*
* vmxoff causes a undefined - opcode exception if vmxon was not run
* on the CPU previously . Only call this function if you know VMX
* is enabled .
*/
static inline void cpu_vmxoff ( void )
{
asm volatile ( ASM_VMX_VMXOFF : : : " cc " ) ;
2014-10-25 02:58:07 +04:00
cr4_clear_bits ( X86_CR4_VMXE ) ;
2008-11-18 00:03:17 +03:00
}
2008-11-18 00:03:19 +03:00
static inline int cpu_vmx_enabled ( void )
{
2014-10-25 02:58:08 +04:00
return __read_cr4 ( ) & X86_CR4_VMXE ;
2008-11-18 00:03:19 +03:00
}
/** Disable VMX if it is enabled on the current CPU
*
* You shouldn ' t call this if cpu_has_vmx ( ) returns 0.
*/
static inline void __cpu_emergency_vmxoff ( void )
{
if ( cpu_vmx_enabled ( ) )
cpu_vmxoff ( ) ;
}
/** Disable VMX if it is supported and enabled on the current CPU
*/
static inline void cpu_emergency_vmxoff ( void )
{
if ( cpu_has_vmx ( ) )
__cpu_emergency_vmxoff ( ) ;
}
2008-11-18 00:03:20 +03:00
/*
* SVM functions :
*/
/** Check if the CPU has SVM support
*
* You can use the ' msg ' arg to get a message describing the problem ,
* if the function returns zero . Simply pass NULL if you are not interested
* on the messages ; gcc should take care of not generating code for
* the messages on this case .
*/
static inline int cpu_has_svm ( const char * * msg )
{
if ( boot_cpu_data . x86_vendor ! = X86_VENDOR_AMD ) {
if ( msg )
* msg = " not amd " ;
return 0 ;
}
2016-05-09 12:53:06 +03:00
if ( boot_cpu_data . extended_cpuid_level < SVM_CPUID_FUNC ) {
2008-11-18 00:03:20 +03:00
if ( msg )
* msg = " can't execute cpuid_8000000a " ;
return 0 ;
}
2016-05-09 12:53:06 +03:00
if ( ! boot_cpu_has ( X86_FEATURE_SVM ) ) {
2008-11-18 00:03:20 +03:00
if ( msg )
* msg = " svm not available " ;
return 0 ;
}
return 1 ;
}
2008-11-18 00:03:21 +03:00
/** Disable SVM on the current CPU
*
* You should call this only if cpu_has_svm ( ) returned true .
*/
static inline void cpu_svm_disable ( void )
{
uint64_t efer ;
wrmsrl ( MSR_VM_HSAVE_PA , 0 ) ;
rdmsrl ( MSR_EFER , efer ) ;
2008-11-25 22:17:02 +03:00
wrmsrl ( MSR_EFER , efer & ~ EFER_SVME ) ;
2008-11-18 00:03:21 +03:00
}
2008-11-18 00:03:22 +03:00
/** Makes sure SVM is disabled, if it is supported on the CPU
*/
static inline void cpu_emergency_svm_disable ( void )
{
if ( cpu_has_svm ( NULL ) )
cpu_svm_disable ( ) ;
}
2008-11-18 00:03:16 +03:00
# endif /* _ASM_X86_VIRTEX_H */