Commit 1e947bad authored by Marc Zyngier's avatar Marc Zyngier

arm64: KVM: Skip HYP setup when already running in HYP

With the kernel running at EL2, there is no point trying to
configure page tables for HYP, as the kernel is already mapped.

Take this opportunity to refactor the whole init a bit, allowing
the various parts of the hypervisor bringup to be split across
multiple functions.
Reviewed-by: default avatarChristoffer Dall <christoffer.dall@linaro.org>
Signed-off-by: default avatarMarc Zyngier <marc.zyngier@arm.com>
parent d88701be
...@@ -967,6 +967,11 @@ long kvm_arch_vm_ioctl(struct file *filp, ...@@ -967,6 +967,11 @@ long kvm_arch_vm_ioctl(struct file *filp,
} }
} }
static void cpu_init_stage2(void *dummy)
{
__cpu_init_stage2();
}
static void cpu_init_hyp_mode(void *dummy) static void cpu_init_hyp_mode(void *dummy)
{ {
phys_addr_t boot_pgd_ptr; phys_addr_t boot_pgd_ptr;
...@@ -1036,6 +1041,82 @@ static inline void hyp_cpu_pm_init(void) ...@@ -1036,6 +1041,82 @@ static inline void hyp_cpu_pm_init(void)
} }
#endif #endif
static void teardown_common_resources(void)
{
free_percpu(kvm_host_cpu_state);
}
static int init_common_resources(void)
{
kvm_host_cpu_state = alloc_percpu(kvm_cpu_context_t);
if (!kvm_host_cpu_state) {
kvm_err("Cannot allocate host CPU state\n");
return -ENOMEM;
}
return 0;
}
static int init_subsystems(void)
{
int err;
/*
* Init HYP view of VGIC
*/
err = kvm_vgic_hyp_init();
switch (err) {
case 0:
vgic_present = true;
break;
case -ENODEV:
case -ENXIO:
vgic_present = false;
break;
default:
return err;
}
/*
* Init HYP architected timer support
*/
err = kvm_timer_hyp_init();
if (err)
return err;
kvm_perf_init();
kvm_coproc_table_init();
return 0;
}
static void teardown_hyp_mode(void)
{
int cpu;
if (is_kernel_in_hyp_mode())
return;
free_hyp_pgds();
for_each_possible_cpu(cpu)
free_page(per_cpu(kvm_arm_hyp_stack_page, cpu));
}
static int init_vhe_mode(void)
{
/*
* Execute the init code on each CPU.
*/
on_each_cpu(cpu_init_stage2, NULL, 1);
/* set size of VMID supported by CPU */
kvm_vmid_bits = kvm_get_vmid_bits();
kvm_info("%d-bit VMID\n", kvm_vmid_bits);
kvm_info("VHE mode initialized successfully\n");
return 0;
}
/** /**
* Inits Hyp-mode on all online CPUs * Inits Hyp-mode on all online CPUs
*/ */
...@@ -1066,7 +1147,7 @@ static int init_hyp_mode(void) ...@@ -1066,7 +1147,7 @@ static int init_hyp_mode(void)
stack_page = __get_free_page(GFP_KERNEL); stack_page = __get_free_page(GFP_KERNEL);
if (!stack_page) { if (!stack_page) {
err = -ENOMEM; err = -ENOMEM;
goto out_free_stack_pages; goto out_err;
} }
per_cpu(kvm_arm_hyp_stack_page, cpu) = stack_page; per_cpu(kvm_arm_hyp_stack_page, cpu) = stack_page;
...@@ -1078,13 +1159,13 @@ static int init_hyp_mode(void) ...@@ -1078,13 +1159,13 @@ static int init_hyp_mode(void)
err = create_hyp_mappings(__hyp_text_start, __hyp_text_end); err = create_hyp_mappings(__hyp_text_start, __hyp_text_end);
if (err) { if (err) {
kvm_err("Cannot map world-switch code\n"); kvm_err("Cannot map world-switch code\n");
goto out_free_mappings; goto out_err;
} }
err = create_hyp_mappings(__start_rodata, __end_rodata); err = create_hyp_mappings(__start_rodata, __end_rodata);
if (err) { if (err) {
kvm_err("Cannot map rodata section\n"); kvm_err("Cannot map rodata section\n");
goto out_free_mappings; goto out_err;
} }
/* /*
...@@ -1096,20 +1177,10 @@ static int init_hyp_mode(void) ...@@ -1096,20 +1177,10 @@ static int init_hyp_mode(void)
if (err) { if (err) {
kvm_err("Cannot map hyp stack\n"); kvm_err("Cannot map hyp stack\n");
goto out_free_mappings; goto out_err;
} }
} }
/*
* Map the host CPU structures
*/
kvm_host_cpu_state = alloc_percpu(kvm_cpu_context_t);
if (!kvm_host_cpu_state) {
err = -ENOMEM;
kvm_err("Cannot allocate host CPU state\n");
goto out_free_mappings;
}
for_each_possible_cpu(cpu) { for_each_possible_cpu(cpu) {
kvm_cpu_context_t *cpu_ctxt; kvm_cpu_context_t *cpu_ctxt;
...@@ -1118,7 +1189,7 @@ static int init_hyp_mode(void) ...@@ -1118,7 +1189,7 @@ static int init_hyp_mode(void)
if (err) { if (err) {
kvm_err("Cannot map host CPU state: %d\n", err); kvm_err("Cannot map host CPU state: %d\n", err);
goto out_free_context; goto out_err;
} }
} }
...@@ -1127,34 +1198,22 @@ static int init_hyp_mode(void) ...@@ -1127,34 +1198,22 @@ static int init_hyp_mode(void)
*/ */
on_each_cpu(cpu_init_hyp_mode, NULL, 1); on_each_cpu(cpu_init_hyp_mode, NULL, 1);
/*
* Init HYP view of VGIC
*/
err = kvm_vgic_hyp_init();
switch (err) {
case 0:
vgic_present = true;
break;
case -ENODEV:
case -ENXIO:
vgic_present = false;
break;
default:
goto out_free_context;
}
/*
* Init HYP architected timer support
*/
err = kvm_timer_hyp_init();
if (err)
goto out_free_context;
#ifndef CONFIG_HOTPLUG_CPU #ifndef CONFIG_HOTPLUG_CPU
free_boot_hyp_pgd(); free_boot_hyp_pgd();
#endif #endif
kvm_perf_init(); cpu_notifier_register_begin();
err = __register_cpu_notifier(&hyp_init_cpu_nb);
cpu_notifier_register_done();
if (err) {
kvm_err("Cannot register HYP init CPU notifier (%d)\n", err);
goto out_err;
}
hyp_cpu_pm_init();
/* set size of VMID supported by CPU */ /* set size of VMID supported by CPU */
kvm_vmid_bits = kvm_get_vmid_bits(); kvm_vmid_bits = kvm_get_vmid_bits();
...@@ -1163,14 +1222,9 @@ static int init_hyp_mode(void) ...@@ -1163,14 +1222,9 @@ static int init_hyp_mode(void)
kvm_info("Hyp mode initialized successfully\n"); kvm_info("Hyp mode initialized successfully\n");
return 0; return 0;
out_free_context:
free_percpu(kvm_host_cpu_state);
out_free_mappings:
free_hyp_pgds();
out_free_stack_pages:
for_each_possible_cpu(cpu)
free_page(per_cpu(kvm_arm_hyp_stack_page, cpu));
out_err: out_err:
teardown_hyp_mode();
kvm_err("error initializing Hyp mode: %d\n", err); kvm_err("error initializing Hyp mode: %d\n", err);
return err; return err;
} }
...@@ -1214,26 +1268,27 @@ int kvm_arch_init(void *opaque) ...@@ -1214,26 +1268,27 @@ int kvm_arch_init(void *opaque)
} }
} }
cpu_notifier_register_begin(); err = init_common_resources();
err = init_hyp_mode();
if (err) if (err)
goto out_err; return err;
err = __register_cpu_notifier(&hyp_init_cpu_nb); if (is_kernel_in_hyp_mode())
if (err) { err = init_vhe_mode();
kvm_err("Cannot register HYP init CPU notifier (%d)\n", err); else
err = init_hyp_mode();
if (err)
goto out_err; goto out_err;
}
cpu_notifier_register_done();
hyp_cpu_pm_init(); err = init_subsystems();
if (err)
goto out_hyp;
kvm_coproc_table_init();
return 0; return 0;
out_hyp:
teardown_hyp_mode();
out_err: out_err:
cpu_notifier_register_done(); teardown_common_resources();
return err; return err;
} }
......
...@@ -28,6 +28,7 @@ ...@@ -28,6 +28,7 @@
#include <asm/kvm_mmio.h> #include <asm/kvm_mmio.h>
#include <asm/kvm_asm.h> #include <asm/kvm_asm.h>
#include <asm/kvm_emulate.h> #include <asm/kvm_emulate.h>
#include <asm/virt.h>
#include "trace.h" #include "trace.h"
...@@ -598,6 +599,9 @@ int create_hyp_mappings(void *from, void *to) ...@@ -598,6 +599,9 @@ int create_hyp_mappings(void *from, void *to)
unsigned long start = KERN_TO_HYP((unsigned long)from); unsigned long start = KERN_TO_HYP((unsigned long)from);
unsigned long end = KERN_TO_HYP((unsigned long)to); unsigned long end = KERN_TO_HYP((unsigned long)to);
if (is_kernel_in_hyp_mode())
return 0;
start = start & PAGE_MASK; start = start & PAGE_MASK;
end = PAGE_ALIGN(end); end = PAGE_ALIGN(end);
...@@ -630,6 +634,9 @@ int create_hyp_io_mappings(void *from, void *to, phys_addr_t phys_addr) ...@@ -630,6 +634,9 @@ int create_hyp_io_mappings(void *from, void *to, phys_addr_t phys_addr)
unsigned long start = KERN_TO_HYP((unsigned long)from); unsigned long start = KERN_TO_HYP((unsigned long)from);
unsigned long end = KERN_TO_HYP((unsigned long)to); unsigned long end = KERN_TO_HYP((unsigned long)to);
if (is_kernel_in_hyp_mode())
return 0;
/* Check for a valid kernel IO mapping */ /* Check for a valid kernel IO mapping */
if (!is_vmalloc_addr(from) || !is_vmalloc_addr(to - 1)) if (!is_vmalloc_addr(from) || !is_vmalloc_addr(to - 1))
return -EINVAL; return -EINVAL;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment