2018-04-23 00:02:09 +02:00
// SPDX-License-Identifier: GPL-2.0
2017-11-27 09:11:46 +01:00
/*
* Jailhouse paravirt_ops implementation
*
* Copyright ( c ) Siemens AG , 2015 - 2017
*
* Authors :
* Jan Kiszka < jan . kiszka @ siemens . com >
*/
2017-11-27 09:11:48 +01:00
# include <linux/acpi_pmtmr.h>
2017-11-27 09:11:46 +01:00
# include <linux/kernel.h>
2017-11-27 09:11:52 +01:00
# include <linux/reboot.h>
2017-11-27 09:11:47 +01:00
# include <asm/apic.h>
2017-11-27 09:11:46 +01:00
# include <asm/cpu.h>
# include <asm/hypervisor.h>
2017-11-27 09:11:50 +01:00
# include <asm/i8259.h>
2017-11-27 09:11:53 +01:00
# include <asm/irqdomain.h>
2017-11-27 09:11:54 +01:00
# include <asm/pci_x86.h>
2017-11-27 09:11:52 +01:00
# include <asm/reboot.h>
2017-11-27 09:11:46 +01:00
# include <asm/setup.h>
static __initdata struct jailhouse_setup_data setup_data ;
2017-11-27 09:11:49 +01:00
static unsigned int precalibrated_tsc_khz ;
2017-11-27 09:11:46 +01:00
static uint32_t jailhouse_cpuid_base ( void )
{
if ( boot_cpu_data . cpuid_level < 0 | |
! boot_cpu_has ( X86_FEATURE_HYPERVISOR ) )
return 0 ;
return hypervisor_cpuid_base ( " Jailhouse \0 \0 \0 " , 0 ) ;
}
static uint32_t __init jailhouse_detect ( void )
{
return jailhouse_cpuid_base ( ) ;
}
2017-11-27 09:11:50 +01:00
static void jailhouse_get_wallclock ( struct timespec * now )
{
memset ( now , 0 , sizeof ( * now ) ) ;
}
2017-11-27 09:11:49 +01:00
static void __init jailhouse_timer_init ( void )
{
lapic_timer_frequency = setup_data . apic_khz * ( 1000 / HZ ) ;
}
static unsigned long jailhouse_get_tsc ( void )
{
return precalibrated_tsc_khz ;
}
2018-01-15 10:24:34 +01:00
static void __init jailhouse_x2apic_init ( void )
{
# ifdef CONFIG_X86_X2APIC
if ( ! x2apic_enabled ( ) )
return ;
/*
* We do not have access to IR inside Jailhouse non - root cells . So
* we have to run in physical mode .
*/
x2apic_phys = 1 ;
/*
* This will trigger the switch to apic_x2apic_phys . Empty OEM IDs
* ensure that only this APIC driver picks up the call .
*/
default_acpi_madt_oem_check ( " " , " " ) ;
# endif
}
2017-11-27 09:11:47 +01:00
static void __init jailhouse_get_smp_config ( unsigned int early )
{
2017-11-27 09:11:53 +01:00
struct ioapic_domain_cfg ioapic_cfg = {
. type = IOAPIC_DOMAIN_STRICT ,
. ops = & mp_ioapic_irqdomain_ops ,
} ;
struct mpc_intsrc mp_irq = {
. type = MP_INTSRC ,
. irqtype = mp_INT ,
. irqflag = MP_IRQPOL_ACTIVE_HIGH | MP_IRQTRIG_EDGE ,
} ;
2017-11-27 09:11:47 +01:00
unsigned int cpu ;
2018-01-15 10:24:34 +01:00
jailhouse_x2apic_init ( ) ;
2017-11-27 09:11:47 +01:00
register_lapic_address ( 0xfee00000 ) ;
for ( cpu = 0 ; cpu < setup_data . num_cpus ; cpu + + ) {
generic_processor_info ( setup_data . cpu_ids [ cpu ] ,
boot_cpu_apic_version ) ;
}
smp_found_config = 1 ;
2017-11-27 09:11:53 +01:00
if ( setup_data . standard_ioapic ) {
mp_register_ioapic ( 0 , 0xfec00000 , gsi_top , & ioapic_cfg ) ;
/* Register 1:1 mapping for legacy UART IRQs 3 and 4 */
mp_irq . srcbusirq = mp_irq . dstirq = 3 ;
mp_save_irq ( & mp_irq ) ;
mp_irq . srcbusirq = mp_irq . dstirq = 4 ;
mp_save_irq ( & mp_irq ) ;
}
2017-11-27 09:11:47 +01:00
}
2017-11-27 09:11:52 +01:00
static void jailhouse_no_restart ( void )
{
pr_notice ( " Jailhouse: Restart not supported, halting \n " ) ;
machine_halt ( ) ;
}
2017-11-27 09:11:54 +01:00
static int __init jailhouse_pci_arch_init ( void )
{
pci_direct_init ( 1 ) ;
/*
* There are no bridges on the virtual PCI root bus under Jailhouse ,
* thus no other way to discover all devices than a full scan .
2018-01-19 11:06:30 +01:00
* Respect any overrides via the command line , though .
2017-11-27 09:11:54 +01:00
*/
2018-01-19 11:06:30 +01:00
if ( pcibios_last_bus < 0 )
pcibios_last_bus = 0xff ;
2017-11-27 09:11:54 +01:00
2018-03-07 08:39:14 +01:00
# ifdef CONFIG_PCI_MMCONFIG
if ( setup_data . pci_mmconfig_base ) {
pci_mmconfig_add ( 0 , 0 , pcibios_last_bus ,
setup_data . pci_mmconfig_base ) ;
pci_mmcfg_arch_init ( ) ;
}
# endif
2017-11-27 09:11:54 +01:00
return 0 ;
}
2017-11-27 09:11:46 +01:00
static void __init jailhouse_init_platform ( void )
{
u64 pa_data = boot_params . hdr . setup_data ;
struct setup_data header ;
void * mapping ;
2017-11-27 09:11:50 +01:00
x86_init . irqs . pre_vector_init = x86_init_noop ;
2017-11-27 09:11:49 +01:00
x86_init . timers . timer_init = jailhouse_timer_init ;
2017-11-27 09:11:47 +01:00
x86_init . mpparse . get_smp_config = jailhouse_get_smp_config ;
2017-11-27 09:11:54 +01:00
x86_init . pci . arch_init = jailhouse_pci_arch_init ;
2017-11-27 09:11:47 +01:00
2017-11-27 09:11:49 +01:00
x86_platform . calibrate_cpu = jailhouse_get_tsc ;
x86_platform . calibrate_tsc = jailhouse_get_tsc ;
2017-11-27 09:11:50 +01:00
x86_platform . get_wallclock = jailhouse_get_wallclock ;
x86_platform . legacy . rtc = 0 ;
x86_platform . legacy . warm_reset = 0 ;
x86_platform . legacy . i8042 = X86_LEGACY_I8042_PLATFORM_ABSENT ;
legacy_pic = & null_legacy_pic ;
2017-11-27 09:11:49 +01:00
2017-11-27 09:11:52 +01:00
machine_ops . emergency_restart = jailhouse_no_restart ;
2017-11-27 09:11:46 +01:00
while ( pa_data ) {
mapping = early_memremap ( pa_data , sizeof ( header ) ) ;
memcpy ( & header , mapping , sizeof ( header ) ) ;
early_memunmap ( mapping , sizeof ( header ) ) ;
if ( header . type = = SETUP_JAILHOUSE & &
header . len > = sizeof ( setup_data ) ) {
pa_data + = offsetof ( struct setup_data , data ) ;
mapping = early_memremap ( pa_data , sizeof ( setup_data ) ) ;
memcpy ( & setup_data , mapping , sizeof ( setup_data ) ) ;
early_memunmap ( mapping , sizeof ( setup_data ) ) ;
break ;
}
pa_data = header . next ;
}
if ( ! pa_data )
panic ( " Jailhouse: No valid setup data found " ) ;
if ( setup_data . compatible_version > JAILHOUSE_SETUP_REQUIRED_VERSION )
panic ( " Jailhouse: Unsupported setup data structure " ) ;
2017-11-27 09:11:48 +01:00
pmtmr_ioport = setup_data . pm_timer_address ;
pr_debug ( " Jailhouse: PM-Timer IO Port: %#x \n " , pmtmr_ioport ) ;
2017-11-27 09:11:49 +01:00
precalibrated_tsc_khz = setup_data . tsc_khz ;
2018-01-19 11:06:17 +01:00
setup_force_cpu_cap ( X86_FEATURE_TSC_KNOWN_FREQ ) ;
2017-11-27 09:11:51 +01:00
2017-11-27 09:11:54 +01:00
pci_probe = 0 ;
2017-11-27 09:11:51 +01:00
/*
* Avoid that the kernel complains about missing ACPI tables - there
* are none in a non - root cell .
*/
disable_acpi ( ) ;
2017-11-27 09:11:46 +01:00
}
bool jailhouse_paravirt ( void )
{
return jailhouse_cpuid_base ( ) ! = 0 ;
}
2017-11-27 09:11:47 +01:00
static bool jailhouse_x2apic_available ( void )
{
/*
* The x2APIC is only available if the root cell enabled it . Jailhouse
* does not support switching between xAPIC and x2APIC .
*/
return x2apic_enabled ( ) ;
}
2017-11-27 09:11:46 +01:00
const struct hypervisor_x86 x86_hyper_jailhouse __refconst = {
. name = " Jailhouse " ,
. detect = jailhouse_detect ,
. init . init_platform = jailhouse_init_platform ,
2017-11-27 09:11:47 +01:00
. init . x2apic_available = jailhouse_x2apic_available ,
2017-11-27 09:11:46 +01:00
} ;