x86: mp: Park CPUs before running the OS
authorSimon Glass <sjg@chromium.org>
Fri, 17 Jul 2020 14:48:20 +0000 (08:48 -0600)
committerBin Meng <bmeng.cn@gmail.com>
Mon, 20 Jul 2020 01:46:46 +0000 (09:46 +0800)
With the new MP features the CPUs are no-longer parked when the OS is run.
Fix this by calling a special function to park them, just before the OS is
started.

Signed-off-by: Simon Glass <sjg@chromium.org>
Reviewed-by: Wolfgang Wallner <wolfgang.wallner@br-automation.com>
Reviewed-by: Bin Meng <bmeng.cn@gmail.com>
arch/x86/cpu/cpu.c
arch/x86/cpu/mp_init.c
arch/x86/include/asm/mp.h

index c343586..69c1418 100644 (file)
@@ -67,6 +67,11 @@ static const char *const x86_vendor_name[] = {
 
 int __weak x86_cleanup_before_linux(void)
 {
+       int ret;
+
+       ret = mp_park_aps();
+       if (ret)
+               return log_msg_ret("park", ret);
        bootstage_stash((void *)CONFIG_BOOTSTAGE_STASH_ADDR,
                        CONFIG_BOOTSTAGE_STASH_SIZE);
 
index dd6d6bf..427ec8f 100644 (file)
@@ -668,6 +668,22 @@ int mp_run_on_cpus(int cpu_select, mp_run_func func, void *arg)
        return 0;
 }
 
+static void park_this_cpu(void *unused)
+{
+       stop_this_cpu();
+}
+
+int mp_park_aps(void)
+{
+       int ret;
+
+       ret = mp_run_on_cpus(MP_SELECT_APS, park_this_cpu, NULL);
+       if (ret)
+               return log_ret(ret);
+
+       return 0;
+}
+
 int mp_init(void)
 {
        int num_aps, num_cpus;
index eb49e69..f9d6c8e 100644 (file)
@@ -109,6 +109,15 @@ typedef void (*mp_run_func)(void *arg);
  * @return 0 on success, -ve on error
  */
 int mp_run_on_cpus(int cpu_select, mp_run_func func, void *arg);
+
+/**
+ * mp_park_aps() - Park the APs ready for the OS
+ *
+ * This halts all CPUs except the main one, ready for the OS to use them
+ *
+ * @return 0 if OK, -ve on error
+ */
+int mp_park_aps(void);
 #else
 static inline int mp_run_on_cpus(int cpu_select, mp_run_func func, void *arg)
 {
@@ -117,6 +126,14 @@ static inline int mp_run_on_cpus(int cpu_select, mp_run_func func, void *arg)
 
        return 0;
 }
+
+static inline int mp_park_aps(void)
+{
+       /* No APs to park */
+
+       return 0;
+}
+
 #endif
 
 #endif /* _X86_MP_H_ */