Re: [PATCH 1/2] arm/psci: Test that CPU 1 has been successfully brought online

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



Hi,

On Mon, Dec 05, 2022 at 12:10:52PM +0000, Nikita Venkatesh wrote:
> From: Alexandru Elisei <alexandru.elisei@xxxxxxx>
> 
> For the PSCI CPU_ON function test, all other CPUs perform a CPU_ON call
> that target CPU 1. The test is considered a success if CPU_ON returns
> SUCCESS exactly once, and for the rest of the calls ALREADY_ON.
> 
> Enhance the test by making sure that CPU 1 is actually online and able to
> execute code. Since the CPU 1 thread is now being set up properly by
> kvm-unit-tests when being brought online, it becomes possible to add other
> tests in the future that require all CPUs.
> 
> Signed-off-by: Alexandru Elisei <alexandru.elisei@xxxxxxx>
> Signed-off-by: Nikita Venkatesh <Nikita.Venkatesh@xxxxxxx>
> ---
>  arm/psci.c        | 30 +++++++++++++++++++++---------
>  lib/arm/asm/smp.h |  1 +
>  lib/arm/smp.c     | 12 +++++++++---
>  3 files changed, 31 insertions(+), 12 deletions(-)
> 
> diff --git a/arm/psci.c b/arm/psci.c
> index efa0722..0b9834c 100644
> --- a/arm/psci.c
> +++ b/arm/psci.c
> @@ -72,14 +72,23 @@ static int cpu_on_ret[NR_CPUS];
>  static cpumask_t cpu_on_ready, cpu_on_done;
>  static volatile int cpu_on_start;
>  
> -static void cpu_on_secondary_entry(void)
> +extern void secondary_entry(void);
> +static void cpu_on_wake_target(void)
>  {
>  	int cpu = smp_processor_id();
>  
>  	cpumask_set_cpu(cpu, &cpu_on_ready);
>  	while (!cpu_on_start)
>  		cpu_relax();
> -	cpu_on_ret[cpu] = psci_cpu_on(cpus[1], __pa(halt));
> +	cpu_on_ret[cpu] = psci_cpu_on(cpus[1], __pa(secondary_entry));
> +	cpumask_set_cpu(cpu, &cpu_on_done);
> +}
> +
> +static void cpu_on_target(void)
> +{
> +	int cpu = smp_processor_id();
> +
> +	cpu_on_ret[cpu] = PSCI_RET_ALREADY_ON;
>  	cpumask_set_cpu(cpu, &cpu_on_done);
>  }
>  
> @@ -89,31 +98,34 @@ static bool psci_cpu_on_test(void)
>  	int ret_success = 0;
>  	int cpu;
>  
> -	cpumask_set_cpu(1, &cpu_on_ready);
> -	cpumask_set_cpu(1, &cpu_on_done);
> -
>  	for_each_present_cpu(cpu) {
>  		if (cpu < 2)
>  			continue;
> -		smp_boot_secondary(cpu, cpu_on_secondary_entry);
> +		smp_boot_secondary(cpu, cpu_on_wake_target);
>  	}
>  
>  	cpumask_set_cpu(0, &cpu_on_ready);
> +	cpumask_set_cpu(1, &cpu_on_ready);
>  	while (!cpumask_full(&cpu_on_ready))
>  		cpu_relax();
>  
> +	/*
> +	 * Wait for all other CPUs to be online before configuring the thread
> +	 * for the target CPU, as all secondaries are set up using the same
> +	 * global variable.
> +	 */
> +	smp_prepare_secondary(1, cpu_on_target);
> +
>  	cpu_on_start = 1;
>  	smp_mb();
>  
> -	cpu_on_ret[0] = psci_cpu_on(cpus[1], __pa(halt));
> +	cpu_on_ret[0] = psci_cpu_on(cpus[1], __pa(secondary_entry));
>  	cpumask_set_cpu(0, &cpu_on_done);
>  
>  	while (!cpumask_full(&cpu_on_done))
>  		cpu_relax();

I've realized that with this change the test hangs if CPU 1 doesn't come
online, which wasn't the case before.

I think adding a timeout here would be the best way to fix it, something
like this:

--- a/arm/psci.c
+++ b/arm/psci.c
@@ -9,6 +9,7 @@
  */
 #include <libcflat.h>
 #include <errata.h>
+#include <asm/delay.h>
 #include <asm/processor.h>
 #include <asm/smp.h>
 #include <asm/psci.h>
@@ -96,7 +97,7 @@ static bool psci_cpu_on_test(void)
 {
        bool failed = false;
        int ret_success = 0;
-       int cpu;
+       int i, cpu;

        for_each_present_cpu(cpu) {
                if (cpu < 2)
@@ -122,8 +123,24 @@ static bool psci_cpu_on_test(void)
        cpu_on_ret[0] = psci_cpu_on(cpus[1], __pa(secondary_entry));
        cpumask_set_cpu(0, &cpu_on_done);

-       while (!cpumask_full(&cpu_on_done))
-               cpu_relax();
+       report_info("waiting for CPU1 to come online...");
+       for (i = 0; i < 10; i++) {
+               mdelay(100);
+               if (cpumask_full(&cpu_on_done))
+                       break;
+       }
+
+       if (!cpumask_full(&cpu_on_done)) {
+               for_each_present_cpu(cpu) {
+                       if (!cpumask_test_cpu(cpu, &cpu_on_done)) {
+                               if (cpu == 1)
+                                       report_info("CPU1 failed to come online");
+                               else
+                                       report_info("CPU%d failed to online CPU1", cpu);
+                       }
+               }
+               return false;
+       }

        for_each_present_cpu(cpu) {
                if (cpu_on_ret[cpu] == PSCI_RET_SUCCESS) {

What do you think?

Thanks,
Alex

>  
>  	for_each_present_cpu(cpu) {
> -		if (cpu == 1)
> -			continue;
>  		if (cpu_on_ret[cpu] == PSCI_RET_SUCCESS) {
>  			ret_success++;
>  		} else if (cpu_on_ret[cpu] != PSCI_RET_ALREADY_ON) {
> diff --git a/lib/arm/asm/smp.h b/lib/arm/asm/smp.h
> index 077afde..ff2ef8f 100644
> --- a/lib/arm/asm/smp.h
> +++ b/lib/arm/asm/smp.h
> @@ -49,6 +49,7 @@ static inline void set_cpu_idle(int cpu, bool idle)
>  }
>  
>  typedef void (*secondary_entry_fn)(void);
> +extern void smp_prepare_secondary(int cpu, secondary_entry_fn entry);
>  extern void smp_boot_secondary(int cpu, secondary_entry_fn entry);
>  extern void on_cpu_async(int cpu, void (*func)(void *data), void *data);
>  extern void on_cpu(int cpu, void (*func)(void *data), void *data);
> diff --git a/lib/arm/smp.c b/lib/arm/smp.c
> index 98a5054..947f417 100644
> --- a/lib/arm/smp.c
> +++ b/lib/arm/smp.c
> @@ -58,13 +58,19 @@ secondary_entry_fn secondary_cinit(void)
>  	return entry;
>  }
>  
> -static void __smp_boot_secondary(int cpu, secondary_entry_fn entry)
> +void smp_prepare_secondary(int cpu, secondary_entry_fn entry)
>  {
> -	int ret;
> -
>  	secondary_data.stack = thread_stack_alloc();
>  	secondary_data.entry = entry;
>  	mmu_mark_disabled(cpu);
> +}
> +
> +static void __smp_boot_secondary(int cpu, secondary_entry_fn entry)
> +{
> +	int ret;
> +
> +	smp_prepare_secondary(cpu, entry);
> +
>  	ret = cpu_psci_cpu_boot(cpu);
>  	assert(ret == 0);
>  
> -- 
> 2.25.1
> 



[Index of Archives]     [KVM ARM]     [KVM ia64]     [KVM ppc]     [Virtualization Tools]     [Spice Development]     [Libvirt]     [Libvirt Users]     [Linux USB Devel]     [Linux Audio Users]     [Yosemite Questions]     [Linux Kernel]     [Linux SCSI]     [XFree86]

  Powered by Linux