hc
2023-11-06 e3e12f52b214121840b44c91de5b3e5af5d3eb84
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
/*
 *  linux/arch/arm/plat-versatile/platsmp.c
 *
 *  Copyright (C) 2002 ARM Ltd.
 *  All Rights Reserved
 *
 * This program is free software; you can redistribute it and/or modify
 * it under the terms of the GNU General Public License version 2 as
 * published by the Free Software Foundation.
 */
#include <linux/init.h>
#include <linux/errno.h>
#include <linux/delay.h>
#include <linux/device.h>
#include <linux/jiffies.h>
#include <linux/smp.h>
 
#include <asm/cacheflush.h>
#include <asm/smp_plat.h>
 
#include <plat/platsmp.h>
 
/*
 * Write pen_release in a way that is guaranteed to be visible to all
 * observers, irrespective of whether they're taking part in coherency
 * or not.  This is necessary for the hotplug code to work reliably.
 */
static void write_pen_release(int val)
{
   pen_release = val;
   smp_wmb();
   sync_cache_w(&pen_release);
}
 
static DEFINE_RAW_SPINLOCK(boot_lock);
 
void versatile_secondary_init(unsigned int cpu)
{
   /*
    * let the primary processor know we're out of the
    * pen, then head off into the C entry point
    */
   write_pen_release(-1);
 
   /*
    * Synchronise with the boot thread.
    */
   raw_spin_lock(&boot_lock);
   raw_spin_unlock(&boot_lock);
}
 
int versatile_boot_secondary(unsigned int cpu, struct task_struct *idle)
{
   unsigned long timeout;
 
   /*
    * Set synchronisation state between this boot processor
    * and the secondary one
    */
   raw_spin_lock(&boot_lock);
 
   /*
    * This is really belt and braces; we hold unintended secondary
    * CPUs in the holding pen until we're ready for them.  However,
    * since we haven't sent them a soft interrupt, they shouldn't
    * be there.
    */
   write_pen_release(cpu_logical_map(cpu));
 
   /*
    * Send the secondary CPU a soft interrupt, thereby causing
    * the boot monitor to read the system wide flags register,
    * and branch to the address found there.
    */
   arch_send_wakeup_ipi_mask(cpumask_of(cpu));
 
   timeout = jiffies + (1 * HZ);
   while (time_before(jiffies, timeout)) {
       smp_rmb();
       if (pen_release == -1)
           break;
 
       udelay(10);
   }
 
   /*
    * now the secondary core is starting up let it run its
    * calibrations, then wait for it to finish
    */
   raw_spin_unlock(&boot_lock);
 
   return pen_release != -1 ? -ENOSYS : 0;
}