2018-08-28 10:40:15 +03:00
// SPDX-License-Identifier: GPL-2.0
2017-11-09 16:27:39 +03:00
# include <linux/acpi.h>
2017-03-14 20:35:40 +03:00
# include <linux/cpu.h>
# include <linux/kexec.h>
2017-06-12 14:53:56 +03:00
# include <linux/memblock.h>
2017-03-14 20:35:40 +03:00
# include <xen/features.h>
# include <xen/events.h>
# include <xen/interface/memory.h>
# include <asm/cpu.h>
# include <asm/smp.h>
# include <asm/reboot.h>
# include <asm/setup.h>
# include <asm/hypervisor.h>
2017-06-12 14:53:56 +03:00
# include <asm/e820/api.h>
2017-07-28 13:23:14 +03:00
# include <asm/early_ioremap.h>
2017-03-14 20:35:40 +03:00
# include <asm/xen/cpuid.h>
# include <asm/xen/hypervisor.h>
2017-06-12 14:53:56 +03:00
# include <asm/xen/page.h>
2017-03-14 20:35:40 +03:00
# include "xen-ops.h"
# include "mmu.h"
# include "smp.h"
2017-07-28 13:23:14 +03:00
static unsigned long shared_info_pfn ;
2017-07-28 13:23:13 +03:00
void xen_hvm_init_shared_info ( void )
2017-03-14 20:35:40 +03:00
{
struct xen_add_to_physmap xatp ;
xatp . domid = DOMID_SELF ;
xatp . idx = 0 ;
xatp . space = XENMAPSPACE_shared_info ;
2017-07-28 13:23:14 +03:00
xatp . gpfn = shared_info_pfn ;
2017-03-14 20:35:40 +03:00
if ( HYPERVISOR_memory_op ( XENMEM_add_to_physmap , & xatp ) )
BUG ( ) ;
}
2017-07-28 13:23:13 +03:00
static void __init reserve_shared_info ( void )
{
u64 pa ;
/*
* Search for a free page starting at 4 kB physical address .
* Low memory is preferred to avoid an EPT large page split up
* by the mapping .
* Starting below X86_RESERVE_LOW ( usually 64 kB ) is fine as
* the BIOS used for HVM guests is well behaved and won ' t
* clobber memory other than the first 4 kB .
*/
for ( pa = PAGE_SIZE ;
! e820__mapped_all ( pa , pa + PAGE_SIZE , E820_TYPE_RAM ) | |
memblock_is_reserved ( pa ) ;
pa + = PAGE_SIZE )
;
2017-07-28 13:23:14 +03:00
shared_info_pfn = PHYS_PFN ( pa ) ;
2017-07-28 13:23:13 +03:00
memblock_reserve ( pa , PAGE_SIZE ) ;
2017-07-28 13:23:14 +03:00
HYPERVISOR_shared_info = early_memremap ( pa , PAGE_SIZE ) ;
}
static void __init xen_hvm_init_mem_mapping ( void )
{
early_memunmap ( HYPERVISOR_shared_info , PAGE_SIZE ) ;
HYPERVISOR_shared_info = __va ( PFN_PHYS ( shared_info_pfn ) ) ;
2018-05-04 23:11:00 +03:00
/*
* The virtual address of the shared_info page has changed , so
* the vcpu_info pointer for VCPU 0 is now stale .
*
* The prepare_boot_cpu callback will re - initialize it via
* xen_vcpu_setup , but we can ' t rely on that to be called for
* old Xen versions ( xen_have_vector_callback = = 0 ) .
*
* It is , in any case , bad to have a stale vcpu_info pointer
* so reset it now .
*/
xen_vcpu_info_reset ( 0 ) ;
2017-07-28 13:23:13 +03:00
}
2017-03-14 20:35:40 +03:00
static void __init init_hvm_pv_info ( void )
{
int major , minor ;
uint32_t eax , ebx , ecx , edx , base ;
base = xen_cpuid_base ( ) ;
eax = cpuid_eax ( base + 1 ) ;
major = eax > > 16 ;
minor = eax & 0xffff ;
printk ( KERN_INFO " Xen version %d.%d. \n " , major , minor ) ;
xen_domain_type = XEN_HVM_DOMAIN ;
/* PVH set up hypercall page in xen_prepare_pvh(). */
if ( xen_pvh_domain ( ) )
pv_info . name = " Xen PVH " ;
else {
u64 pfn ;
uint32_t msr ;
pv_info . name = " Xen HVM " ;
msr = cpuid_ebx ( base + 2 ) ;
pfn = __pa ( hypercall_page ) ;
wrmsr_safe ( msr , ( u32 ) pfn , ( u32 ) ( pfn > > 32 ) ) ;
}
xen_setup_features ( ) ;
cpuid ( base + 4 , & eax , & ebx , & ecx , & edx ) ;
if ( eax & XEN_HVM_CPUID_VCPU_ID_PRESENT )
this_cpu_write ( xen_vcpu_id , ebx ) ;
else
this_cpu_write ( xen_vcpu_id , smp_processor_id ( ) ) ;
}
# ifdef CONFIG_KEXEC_CORE
static void xen_hvm_shutdown ( void )
{
native_machine_shutdown ( ) ;
if ( kexec_in_progress )
xen_reboot ( SHUTDOWN_soft_reset ) ;
}
static void xen_hvm_crash_shutdown ( struct pt_regs * regs )
{
native_machine_crash_shutdown ( regs ) ;
xen_reboot ( SHUTDOWN_soft_reset ) ;
}
# endif
static int xen_cpu_up_prepare_hvm ( unsigned int cpu )
{
2017-06-03 03:06:01 +03:00
int rc = 0 ;
2017-03-14 20:35:40 +03:00
/*
* This can happen if CPU was offlined earlier and
* offlining timed out in common_cpu_die ( ) .
*/
if ( cpu_report_state ( cpu ) = = CPU_DEAD_FROZEN ) {
xen_smp_intr_free ( cpu ) ;
xen_uninit_lock_cpu ( cpu ) ;
}
if ( cpu_acpi_id ( cpu ) ! = U32_MAX )
per_cpu ( xen_vcpu_id , cpu ) = cpu_acpi_id ( cpu ) ;
else
per_cpu ( xen_vcpu_id , cpu ) = cpu ;
2017-06-03 03:06:01 +03:00
rc = xen_vcpu_setup ( cpu ) ;
if ( rc )
return rc ;
2017-03-14 20:35:40 +03:00
2017-04-24 22:04:53 +03:00
if ( xen_have_vector_callback & & xen_feature ( XENFEAT_hvm_safe_pvclock ) )
2017-03-14 20:35:40 +03:00
xen_setup_timer ( cpu ) ;
rc = xen_smp_intr_init ( cpu ) ;
if ( rc ) {
WARN ( 1 , " xen_smp_intr_init() for CPU %d failed: %d \n " ,
cpu , rc ) ;
}
2017-06-03 03:06:01 +03:00
return rc ;
2017-03-14 20:35:40 +03:00
}
static int xen_cpu_dead_hvm ( unsigned int cpu )
{
xen_smp_intr_free ( cpu ) ;
2017-04-24 22:04:53 +03:00
if ( xen_have_vector_callback & & xen_feature ( XENFEAT_hvm_safe_pvclock ) )
2017-03-14 20:35:40 +03:00
xen_teardown_timer ( cpu ) ;
return 0 ;
}
static void __init xen_hvm_guest_init ( void )
{
if ( xen_pv_domain ( ) )
return ;
init_hvm_pv_info ( ) ;
2017-07-28 13:23:13 +03:00
reserve_shared_info ( ) ;
2017-03-14 20:35:40 +03:00
xen_hvm_init_shared_info ( ) ;
2017-06-03 03:05:59 +03:00
/*
* xen_vcpu is a pointer to the vcpu_info struct in the shared_info
* page , we use it in the event channel upcall and in some pvclock
* related functions .
*/
xen_vcpu_info_reset ( 0 ) ;
2017-03-14 20:35:40 +03:00
xen_panic_handler_init ( ) ;
2017-04-24 22:04:53 +03:00
if ( xen_feature ( XENFEAT_hvm_callback_vector ) )
xen_have_vector_callback = 1 ;
2017-03-14 20:35:40 +03:00
xen_hvm_smp_init ( ) ;
WARN_ON ( xen_cpuhp_setup ( xen_cpu_up_prepare_hvm , xen_cpu_dead_hvm ) ) ;
xen_unplug_emulated_devices ( ) ;
x86_init . irqs . intr_init = xen_init_IRQ ;
xen_hvm_init_time_ops ( ) ;
xen_hvm_init_mmu_ops ( ) ;
# ifdef CONFIG_KEXEC_CORE
machine_ops . shutdown = xen_hvm_shutdown ;
machine_ops . crash_shutdown = xen_hvm_crash_shutdown ;
# endif
}
static bool xen_nopv ;
static __init int xen_parse_nopv ( char * arg )
{
xen_nopv = true ;
return 0 ;
}
early_param ( " xen_nopv " , xen_parse_nopv ) ;
bool xen_hvm_need_lapic ( void )
{
if ( xen_nopv )
return false ;
if ( xen_pv_domain ( ) )
return false ;
if ( ! xen_hvm_domain ( ) )
return false ;
2017-04-24 22:04:53 +03:00
if ( xen_feature ( XENFEAT_hvm_pirqs ) & & xen_have_vector_callback )
2017-03-14 20:35:40 +03:00
return false ;
return true ;
}
EXPORT_SYMBOL_GPL ( xen_hvm_need_lapic ) ;
static uint32_t __init xen_platform_hvm ( void )
{
if ( xen_pv_domain ( ) | | xen_nopv )
return 0 ;
return xen_cpuid_base ( ) ;
}
2017-11-09 16:27:39 +03:00
static __init void xen_hvm_guest_late_init ( void )
{
# ifdef CONFIG_XEN_PVH
/* Test for PVH domain (PVH boot path taken overrides ACPI flags). */
if ( ! xen_pvh & &
( x86_platform . legacy . rtc | | ! x86_platform . legacy . no_vga ) )
return ;
/* PVH detected. */
xen_pvh = true ;
/* Make sure we don't fall back to (default) ACPI_IRQ_MODEL_PIC. */
if ( ! nr_ioapics & & acpi_irq_model = = ACPI_IRQ_MODEL_PIC )
acpi_irq_model = ACPI_IRQ_MODEL_PLATFORM ;
machine_ops . emergency_restart = xen_emergency_restart ;
pv_info . name = " Xen PVH " ;
# endif
}
2017-11-09 16:27:36 +03:00
const __initconst struct hypervisor_x86 x86_hyper_xen_hvm = {
2017-03-14 20:35:40 +03:00
. name = " Xen HVM " ,
. detect = xen_platform_hvm ,
2017-11-09 16:27:36 +03:00
. type = X86_HYPER_XEN_HVM ,
2017-11-09 16:27:35 +03:00
. init . init_platform = xen_hvm_guest_init ,
. init . x2apic_available = xen_x2apic_para_available ,
. init . init_mem_mapping = xen_hvm_init_mem_mapping ,
2017-11-09 16:27:39 +03:00
. init . guest_late_init = xen_hvm_guest_late_init ,
2017-11-09 16:27:35 +03:00
. runtime . pin_vcpu = xen_pin_vcpu ,
2017-03-14 20:35:40 +03:00
} ;