Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
L
linux
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
Analytics
Analytics
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Commits
Issue Boards
Open sidebar
nexedi
linux
Commits
7a3182b1
Commit
7a3182b1
authored
Aug 02, 2002
by
Anton Blanchard
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
ppc64: hotplug cpu changes
parent
1a84d369
Changes
6
Hide whitespace changes
Inline
Side-by-side
Showing
6 changed files
with
190 additions
and
240 deletions
+190
-240
arch/ppc64/kernel/open_pic.c
arch/ppc64/kernel/open_pic.c
+1
-1
arch/ppc64/kernel/prom.c
arch/ppc64/kernel/prom.c
+1
-0
arch/ppc64/kernel/smp.c
arch/ppc64/kernel/smp.c
+171
-229
arch/ppc64/kernel/xics.c
arch/ppc64/kernel/xics.c
+1
-1
include/asm-ppc64/machdep.h
include/asm-ppc64/machdep.h
+16
-8
include/asm-ppc64/smp.h
include/asm-ppc64/smp.h
+0
-1
No files found.
arch/ppc64/kernel/open_pic.c
View file @
7a3182b1
...
@@ -576,7 +576,7 @@ void openpic_request_IPIs(void)
...
@@ -576,7 +576,7 @@ void openpic_request_IPIs(void)
*/
*/
static
spinlock_t
openpic_setup_lock
__initdata
=
SPIN_LOCK_UNLOCKED
;
static
spinlock_t
openpic_setup_lock
__initdata
=
SPIN_LOCK_UNLOCKED
;
void
__init
do_openpic_setup_cpu
(
void
)
void
__
dev
init
do_openpic_setup_cpu
(
void
)
{
{
#ifdef CONFIG_IRQ_ALL_CPUS
#ifdef CONFIG_IRQ_ALL_CPUS
int
i
;
int
i
;
...
...
arch/ppc64/kernel/prom.c
View file @
7a3182b1
...
@@ -144,6 +144,7 @@ static interpret_func interpret_root_props;
...
@@ -144,6 +144,7 @@ static interpret_func interpret_root_props;
#define FB_MAX 8
#define FB_MAX 8
#endif
#endif
static
int
ppc64_is_smp
;
struct
prom_t
prom
=
{
struct
prom_t
prom
=
{
0
,
/* entry */
0
,
/* entry */
...
...
arch/ppc64/kernel/smp.c
View file @
7a3182b1
...
@@ -25,8 +25,6 @@
...
@@ -25,8 +25,6 @@
#include <linux/interrupt.h>
#include <linux/interrupt.h>
#include <linux/kernel_stat.h>
#include <linux/kernel_stat.h>
#include <linux/delay.h>
#include <linux/delay.h>
#define __KERNEL_SYSCALLS__
#include <linux/unistd.h>
#include <linux/init.h>
#include <linux/init.h>
#include <linux/spinlock.h>
#include <linux/spinlock.h>
#include <linux/cache.h>
#include <linux/cache.h>
...
@@ -53,22 +51,19 @@
...
@@ -53,22 +51,19 @@
#include <asm/machdep.h>
#include <asm/machdep.h>
int
smp_threads_ready
=
0
;
int
smp_threads_ready
=
0
;
volatile
int
smp_commenced
=
0
;
int
smp_tb_synchronized
=
0
;
spinlock_t
kernel_flag
__cacheline_aligned
=
SPIN_LOCK_UNLOCKED
;
spinlock_t
kernel_flag
__cacheline_aligned
=
SPIN_LOCK_UNLOCKED
;
unsigned
long
cache_decay_ticks
;
unsigned
long
cache_decay_ticks
;
static
int
max_cpus
__initdata
=
NR_CPUS
;
/* initialised so it doesnt end up in bss */
/* initialised so it doesnt end up in bss */
unsigned
long
cpu_online_map
=
0
;
unsigned
long
cpu_online_map
=
0
;
int
boot_cpuid
=
0
;
int
boot_cpuid
=
0
;
int
ppc64_is_smp
=
0
;
volatile
unsigned
long
cpu_callin_map
[
NR_CPUS
]
=
{
0
,};
static
struct
smp_ops_t
*
smp_ops
;
volatile
unsigned
long
cpu_callin_map
[
NR_CPUS
];
extern
unsigned
char
stab_array
[];
extern
unsigned
char
stab_array
[];
int
start_secondary
(
void
*
);
extern
int
cpu_idle
(
void
*
unused
);
extern
int
cpu_idle
(
void
*
unused
);
void
smp_call_function_interrupt
(
void
);
void
smp_call_function_interrupt
(
void
);
void
smp_message_pass
(
int
target
,
int
msg
,
unsigned
long
data
,
int
wait
);
void
smp_message_pass
(
int
target
,
int
msg
,
unsigned
long
data
,
int
wait
);
...
@@ -86,7 +81,7 @@ struct xics_ipi_struct {
...
@@ -86,7 +81,7 @@ struct xics_ipi_struct {
struct
xics_ipi_struct
xics_ipi_message
[
NR_CPUS
]
__cacheline_aligned
;
struct
xics_ipi_struct
xics_ipi_message
[
NR_CPUS
]
__cacheline_aligned
;
#define smp_message_pass(t,m,d,w)
ppc_md.smp_
message_pass((t),(m),(d),(w))
#define smp_message_pass(t,m,d,w)
smp_ops->
message_pass((t),(m),(d),(w))
static
inline
void
set_tb
(
unsigned
int
upper
,
unsigned
int
lower
)
static
inline
void
set_tb
(
unsigned
int
upper
,
unsigned
int
lower
)
{
{
...
@@ -106,7 +101,6 @@ void iSeries_smp_message_recv( struct pt_regs * regs )
...
@@ -106,7 +101,6 @@ void iSeries_smp_message_recv( struct pt_regs * regs )
for
(
msg
=
0
;
msg
<
4
;
++
msg
)
for
(
msg
=
0
;
msg
<
4
;
++
msg
)
if
(
test_and_clear_bit
(
msg
,
&
iSeries_smp_message
[
cpu
]
)
)
if
(
test_and_clear_bit
(
msg
,
&
iSeries_smp_message
[
cpu
]
)
)
smp_message_recv
(
msg
,
regs
);
smp_message_recv
(
msg
,
regs
);
}
}
static
void
smp_iSeries_message_pass
(
int
target
,
int
msg
,
unsigned
long
data
,
int
wait
)
static
void
smp_iSeries_message_pass
(
int
target
,
int
msg
,
unsigned
long
data
,
int
wait
)
...
@@ -127,6 +121,7 @@ static void smp_iSeries_message_pass(int target, int msg, unsigned long data, in
...
@@ -127,6 +121,7 @@ static void smp_iSeries_message_pass(int target, int msg, unsigned long data, in
}
}
}
}
#ifdef CONFIG_PPC_ISERIES
static
int
smp_iSeries_numProcs
(
void
)
static
int
smp_iSeries_numProcs
(
void
)
{
{
unsigned
np
,
i
;
unsigned
np
,
i
;
...
@@ -141,24 +136,23 @@ static int smp_iSeries_numProcs(void)
...
@@ -141,24 +136,23 @@ static int smp_iSeries_numProcs(void)
}
}
return
np
;
return
np
;
}
}
#endif
static
void
smp_iSeries_probe
(
void
)
static
int
smp_iSeries_probe
(
void
)
{
{
unsigned
i
;
unsigned
i
;
unsigned
np
;
unsigned
np
=
0
;
struct
ItLpPaca
*
lpPaca
;
struct
ItLpPaca
*
lpPaca
;
np
=
0
;
for
(
i
=
0
;
i
<
MAX_PACAS
;
++
i
)
{
for
(
i
=
0
;
i
<
MAX_PACAS
;
++
i
)
{
lpPaca
=
paca
[
i
].
xLpPacaPtr
;
lpPaca
=
paca
[
i
].
xLpPacaPtr
;
if
(
lpPaca
->
xDynProcStatus
<
2
)
{
if
(
lpPaca
->
xDynProcStatus
<
2
)
{
paca
[
i
].
active
=
1
;
paca
[
i
].
active
=
1
;
++
np
;
++
np
;
paca
[
i
].
next_jiffy_update_tb
=
paca
[
0
].
next_jiffy_update_tb
;
}
}
}
}
smp_tb_synchronized
=
1
;
return
np
;
}
}
static
void
smp_iSeries_kick_cpu
(
int
nr
)
static
void
smp_iSeries_kick_cpu
(
int
nr
)
...
@@ -194,10 +188,11 @@ static void smp_iSeries_setup_cpu(int nr)
...
@@ -194,10 +188,11 @@ static void smp_iSeries_setup_cpu(int nr)
/* This is called very early. */
/* This is called very early. */
void
smp_init_iSeries
(
void
)
void
smp_init_iSeries
(
void
)
{
{
ppc_md
.
smp_message_pass
=
smp_iSeries_message_pass
;
smp_ops
=
&
ppc_md
.
smp_ops
;
ppc_md
.
smp_probe
=
smp_iSeries_probe
;
smp_ops
->
message_pass
=
smp_iSeries_message_pass
;
ppc_md
.
smp_kick_cpu
=
smp_iSeries_kick_cpu
;
smp_ops
->
probe
=
smp_iSeries_probe
;
ppc_md
.
smp_setup_cpu
=
smp_iSeries_setup_cpu
;
smp_ops
->
kick_cpu
=
smp_iSeries_kick_cpu
;
smp_ops
->
setup_cpu
=
smp_iSeries_setup_cpu
;
#ifdef CONFIG_PPC_ISERIES
#ifdef CONFIG_PPC_ISERIES
#warning fix for iseries
#warning fix for iseries
naca
->
processorCount
=
smp_iSeries_numProcs
();
naca
->
processorCount
=
smp_iSeries_numProcs
();
...
@@ -229,10 +224,20 @@ smp_openpic_message_pass(int target, int msg, unsigned long data, int wait)
...
@@ -229,10 +224,20 @@ smp_openpic_message_pass(int target, int msg, unsigned long data, int wait)
}
}
}
}
static
void
smp_chrp_probe
(
void
)
static
int
smp_chrp_probe
(
void
)
{
{
if
(
ppc64_is_smp
)
int
i
;
int
nr_cpus
=
0
;
for
(
i
=
0
;
i
<
NR_CPUS
;
i
++
)
{
if
(
cpu_possible
(
i
))
nr_cpus
++
;
}
if
(
nr_cpus
>
1
)
openpic_request_IPIs
();
openpic_request_IPIs
();
return
nr_cpus
;
}
}
static
void
static
void
...
@@ -252,78 +257,32 @@ smp_kick_cpu(int nr)
...
@@ -252,78 +257,32 @@ smp_kick_cpu(int nr)
/* The processor is currently spinning, waiting
/* The processor is currently spinning, waiting
* for the xProcStart field to become non-zero
* for the xProcStart field to become non-zero
* After we set xProcStart, the processor will
* After we set xProcStart, the processor will
* continue on to secondary_start
in iSeries_head.S
* continue on to secondary_start
*/
*/
paca
[
nr
].
xProcStart
=
1
;
paca
[
nr
].
xProcStart
=
1
;
}
}
extern
struct
gettimeofday_struct
do_gtod
;
static
void
smp_space_timers
(
unsigned
int
max_cpus
)
static
void
smp_space_timers
()
{
{
int
i
;
int
i
;
unsigned
long
offset
=
tb_ticks_per_jiffy
/
NR_CPUS
;
unsigned
long
offset
=
tb_ticks_per_jiffy
/
max_cpus
;
unsigned
long
previous_tb
=
paca
[
boot_cpuid
].
next_jiffy_update_tb
;
for
(
i
=
1
;
i
<
NR_CPUS
;
++
i
)
for
(
i
=
0
;
i
<
NR_CPUS
;
i
++
)
{
paca
[
i
].
next_jiffy_update_tb
=
if
(
cpu_possible
(
i
)
&&
i
!=
boot_cpuid
)
{
paca
[
i
-
1
].
next_jiffy_update_tb
+
offset
;
paca
[
i
].
next_jiffy_update_tb
=
}
previous_tb
+
offset
;
previous_tb
=
paca
[
i
].
next_jiffy_update_tb
;
static
void
smp_chrp_setup_cpu
(
int
cpu_nr
)
{
static
atomic_t
ready
=
ATOMIC_INIT
(
1
);
static
volatile
int
frozen
=
0
;
if
(
naca
->
platform
==
PLATFORM_PSERIES_LPAR
)
{
/* timebases already synced under the hypervisor. */
paca
[
cpu_nr
].
next_jiffy_update_tb
=
tb_last_stamp
=
get_tb
();
if
(
cpu_nr
==
boot_cpuid
)
{
do_gtod
.
tb_orig_stamp
=
tb_last_stamp
;
/* Should update do_gtod.stamp_xsec.
* For now we leave it which means the time can be some
* number of msecs off until someone does a settimeofday()
*/
}
smp_tb_synchronized
=
1
;
}
else
{
if
(
cpu_nr
==
boot_cpuid
)
{
/* wait for all the others */
while
(
atomic_read
(
&
ready
)
<
num_online_cpus
())
barrier
();
atomic_set
(
&
ready
,
1
);
/* freeze the timebase */
rtas_call
(
rtas_token
(
"freeze-time-base"
),
0
,
1
,
NULL
);
mb
();
frozen
=
1
;
set_tb
(
0
,
0
);
paca
[
boot_cpuid
].
next_jiffy_update_tb
=
0
;
smp_space_timers
();
while
(
atomic_read
(
&
ready
)
<
num_online_cpus
())
barrier
();
/* thaw the timebase again */
rtas_call
(
rtas_token
(
"thaw-time-base"
),
0
,
1
,
NULL
);
mb
();
frozen
=
0
;
tb_last_stamp
=
get_tb
();
do_gtod
.
tb_orig_stamp
=
tb_last_stamp
;
smp_tb_synchronized
=
1
;
}
else
{
atomic_inc
(
&
ready
);
while
(
!
frozen
)
barrier
();
set_tb
(
0
,
0
);
mb
();
atomic_inc
(
&
ready
);
while
(
frozen
)
barrier
();
}
}
}
}
}
static
void
__devinit
pSeries_setup_cpu
(
int
cpu
)
{
if
(
OpenPIC_Addr
)
{
if
(
OpenPIC_Addr
)
{
do_openpic_setup_cpu
();
do_openpic_setup_cpu
();
}
else
{
}
else
{
if
(
cpu
_nr
!=
boot_cpuid
)
if
(
cpu
!=
boot_cpuid
)
xics_setup_cpu
();
xics_setup_cpu
();
}
}
}
}
...
@@ -347,26 +306,65 @@ smp_xics_message_pass(int target, int msg, unsigned long data, int wait)
...
@@ -347,26 +306,65 @@ smp_xics_message_pass(int target, int msg, unsigned long data, int wait)
}
}
}
}
static
void
smp_xics_probe
(
void
)
static
int
smp_xics_probe
(
void
)
{
{
int
i
;
int
nr_cpus
=
0
;
for
(
i
=
0
;
i
<
NR_CPUS
;
i
++
)
{
if
(
cpu_possible
(
i
))
nr_cpus
++
;
}
return
nr_cpus
;
}
static
spinlock_t
timebase_lock
=
SPIN_LOCK_UNLOCKED
;
static
unsigned
long
timebase
=
0
;
static
void
__devinit
pSeries_give_timebase
(
void
)
{
spin_lock
(
&
timebase_lock
);
rtas_call
(
rtas_token
(
"freeze-time-base"
),
0
,
1
,
NULL
);
timebase
=
get_tb
();
spin_unlock
(
&
timebase_lock
);
while
(
timebase
)
barrier
();
rtas_call
(
rtas_token
(
"thaw-time-base"
),
0
,
1
,
NULL
);
}
static
void
__devinit
pSeries_take_timebase
(
void
)
{
while
(
!
timebase
)
barrier
();
spin_lock
(
&
timebase_lock
);
set_tb
(
timebase
,
timebase
>>
32
);
timebase
=
0
;
spin_unlock
(
&
timebase_lock
);
}
}
/* This is called very early */
/* This is called very early */
void
smp_init_pSeries
(
void
)
void
__init
smp_init_pSeries
(
void
)
{
{
if
(
naca
->
interrupt_controller
==
IC_OPEN_PIC
)
{
smp_ops
=
&
ppc_md
.
smp_ops
;
ppc_md
.
smp_message_pass
=
smp_openpic_message_pass
;
ppc_md
.
smp_probe
=
smp_chrp_probe
;
if
(
naca
->
interrupt_controller
==
IC_OPEN_PIC
)
{
ppc_md
.
smp_kick_cpu
=
smp_kick_cpu
;
smp_ops
->
message_pass
=
smp_openpic_message_pass
;
ppc_md
.
smp_setup_cpu
=
smp_chrp_setup_cpu
;
smp_ops
->
probe
=
smp_chrp_probe
;
}
else
{
}
else
{
ppc_md
.
smp_message_pass
=
smp_xics_message_pass
;
smp_ops
->
message_pass
=
smp_xics_message_pass
;
ppc_md
.
smp_probe
=
smp_xics_probe
;
smp_ops
->
probe
=
smp_xics_probe
;
ppc_md
.
smp_kick_cpu
=
smp_kick_cpu
;
}
ppc_md
.
smp_setup_cpu
=
smp_chrp_setup_cpu
;
if
(
naca
->
platform
==
PLATFORM_PSERIES
)
{
smp_ops
->
give_timebase
=
pSeries_give_timebase
;
smp_ops
->
take_timebase
=
pSeries_take_timebase
;
}
}
}
smp_ops
->
kick_cpu
=
smp_kick_cpu
;
smp_ops
->
setup_cpu
=
pSeries_setup_cpu
;
}
void
smp_local_timer_interrupt
(
struct
pt_regs
*
regs
)
void
smp_local_timer_interrupt
(
struct
pt_regs
*
regs
)
{
{
...
@@ -469,8 +467,7 @@ static struct call_data_struct {
...
@@ -469,8 +467,7 @@ static struct call_data_struct {
* hardware interrupt handler or from a bottom half handler.
* hardware interrupt handler or from a bottom half handler.
*/
*/
int
smp_call_function
(
void
(
*
func
)
(
void
*
info
),
void
*
info
,
int
nonatomic
,
int
smp_call_function
(
void
(
*
func
)
(
void
*
info
),
void
*
info
,
int
nonatomic
,
int
wait
)
int
wait
)
{
{
struct
call_data_struct
data
;
struct
call_data_struct
data
;
int
ret
=
-
1
,
cpus
=
num_online_cpus
()
-
1
;
int
ret
=
-
1
,
cpus
=
num_online_cpus
()
-
1
;
...
@@ -553,38 +550,41 @@ void smp_call_function_interrupt(void)
...
@@ -553,38 +550,41 @@ void smp_call_function_interrupt(void)
atomic_inc
(
&
call_data
->
finished
);
atomic_inc
(
&
call_data
->
finished
);
}
}
extern
unsigned
long
decr_overclock
;
extern
unsigned
long
decr_overclock
;
extern
struct
gettimeofday_struct
do_gtod
;
struct
thread_
struct
*
current_set
[
NR_CPUS
]
=
{
&
init_thread_union
,
0
}
;
struct
thread_
info
*
current_set
[
NR_CPUS
]
;
void
__init
smp_boot_cpus
(
vo
id
)
static
void
__devinit
smp_store_cpu_info
(
int
id
)
{
{
int
i
,
cpu_nr
=
0
;
paca
[
id
].
pvr
=
_get_PVR
();
struct
task_struct
*
p
;
}
printk
(
"Entering SMP Mode...
\n
"
);
smp_store_cpu_info
(
boot_cpuid
);
void
__init
smp_prepare_cpus
(
unsigned
int
max_cpus
)
cpu_callin_map
[
boot_cpuid
]
=
1
;
{
int
i
;
/* XXX buggy - Anton */
/* Fixup boot cpu */
current_thread_info
()
->
cpu
=
0
;
smp_store_cpu_info
(
smp_processor_id
());
cpu_callin_map
[
smp_processor_id
()]
=
1
;
for
(
i
=
0
;
i
<
NR_CPUS
;
i
++
)
{
for
(
i
=
0
;
i
<
NR_CPUS
;
i
++
)
{
paca
[
i
].
prof_counter
=
1
;
paca
[
i
].
prof_counter
=
1
;
paca
[
i
].
prof_multiplier
=
1
;
paca
[
i
].
prof_multiplier
=
1
;
if
(
i
!=
boot_cpuid
)
{
if
(
i
!=
boot_cpuid
)
{
void
*
tmp
;
/*
/*
* the boot cpu segment table is statically
* the boot cpu segment table is statically
* initialized to real address 0x5000. The
* initialized to real address 0x5000. The
* Other processor's tables are created and
* Other processor's tables are created and
* initialized here.
* initialized here.
*/
*/
paca
[
i
].
xStab_data
.
virt
=
(
unsigned
long
)
&
stab_array
[
PAGE_SIZE
*
(
i
-
1
)];
tmp
=
&
stab_array
[
PAGE_SIZE
*
(
i
-
1
)];
memset
((
void
*
)
paca
[
i
].
xStab_data
.
virt
,
0
,
PAGE_SIZE
);
memset
(
tmp
,
0
,
PAGE_SIZE
);
paca
[
i
].
xStab_data
.
real
=
__v2a
(
paca
[
i
].
xStab_data
.
virt
);
paca
[
i
].
xStab_data
.
virt
=
(
unsigned
long
)
tmp
;
paca
[
i
].
default_decr
=
tb_ticks_per_jiffy
/
decr_overclock
;
paca
[
i
].
xStab_data
.
real
=
(
unsigned
long
)
__v2a
(
tmp
);
paca
[
i
].
default_decr
=
tb_ticks_per_jiffy
/
decr_overclock
;
}
}
}
}
...
@@ -593,135 +593,84 @@ void __init smp_boot_cpus(void)
...
@@ -593,135 +593,84 @@ void __init smp_boot_cpus(void)
*/
*/
cache_decay_ticks
=
HZ
/
100
;
cache_decay_ticks
=
HZ
/
100
;
ppc_md
.
smp_probe
();
#ifndef CONFIG_PPC_ISERIES
paca
[
boot_cpuid
].
next_jiffy_update_tb
=
tb_last_stamp
=
get_tb
();
for
(
i
=
0
;
i
<
NR_CPUS
;
i
++
)
{
/*
if
(
paca
[
i
].
active
)
* Should update do_gtod.stamp_xsec.
cpu_nr
++
;
* For now we leave it which means the time can be some
}
* number of msecs off until someone does a settimeofday()
printk
(
"Probe found %d CPUs
\n
"
,
cpu_nr
);
*/
do_gtod
.
tb_orig_stamp
=
tb_last_stamp
;
#ifdef CONFIG_ISERIES
smp_space_timers
();
#endif
#endif
printk
(
"Waiting for %d CPUs
\n
"
,
cpu_nr
-
1
);
max_cpus
=
smp_ops
->
probe
();
smp_space_timers
(
max_cpus
);
for
(
i
=
1
;
i
<
NR_CPUS
;
i
++
)
{
}
int
c
;
struct
pt_regs
regs
;
if
(
!
paca
[
i
].
active
)
int
__cpu_up
(
unsigned
int
cpu
)
continue
;
{
struct
pt_regs
regs
;
struct
task_struct
*
p
;
int
c
;
if
(
i
==
boot_cpuid
)
/* create a process for the processor */
continue
;
/* only regs.msr is actually used, and 0 is OK for it */
memset
(
&
regs
,
0
,
sizeof
(
struct
pt_regs
));
p
=
do_fork
(
CLONE_VM
|
CLONE_IDLETASK
,
0
,
&
regs
,
0
);
if
(
IS_ERR
(
p
))
panic
(
"failed fork for CPU %u: %li"
,
cpu
,
PTR_ERR
(
p
));
if
(
num_online_cpus
()
>=
max_cpus
)
init_idle
(
p
,
cpu
);
break
;
unhash_process
(
p
);
/* create a process for the processor */
/* we don't care about the values in regs since we'll
never reschedule the forked task. */
/* We DO care about one bit in the pt_regs we
pass to do_fork. That is the MSR_FP bit in
regs.msr. If that bit is on, then do_fork
(via copy_thread) will call giveup_fpu.
giveup_fpu will get a pointer to our (current's)
last register savearea via current->thread.regs
and using that pointer will turn off the MSR_FP,
MSR_FE0 and MSR_FE1 bits. At this point, this
pointer is pointing to some arbitrary point within
our stack */
memset
(
&
regs
,
0
,
sizeof
(
struct
pt_regs
));
p
=
do_fork
(
CLONE_VM
|
CLONE_IDLETASK
,
0
,
&
regs
,
0
);
if
(
IS_ERR
(
p
))
panic
(
"failed fork for CPU %d"
,
i
);
init_idle
(
p
,
i
);
unhash_process
(
p
);
paca
[
i
].
xCurrent
=
(
u64
)
p
;
current_set
[
i
]
=
p
->
thread_info
;
/* wake up cpus */
ppc_md
.
smp_kick_cpu
(
i
);
/*
* wait to see if the cpu made a callin (is actually up).
* use this value that I found through experimentation.
* -- Cort
*/
for
(
c
=
5000
;
c
&&
!
cpu_callin_map
[
i
]
;
c
--
)
{
udelay
(
100
);
}
if
(
cpu_callin_map
[
i
]
)
{
printk
(
"Processor %d found.
\n
"
,
i
);
/* this sync's the decr's -- Cort */
}
else
{
printk
(
"Processor %d is stuck.
\n
"
,
i
);
}
}
/* Setup boot cpu last (important) */
paca
[
cpu
].
xCurrent
=
(
u64
)
p
;
ppc_md
.
smp_setup_cpu
(
boot_cpuid
)
;
current_set
[
cpu
]
=
p
->
thread_info
;
if
(
num_online_cpus
()
<
2
)
{
/* wake up cpus */
tb_last_stamp
=
get_tb
();
smp_ops
->
kick_cpu
(
cpu
);
smp_tb_synchronized
=
1
;
}
}
void
__init
smp_commence
(
void
)
{
/*
/*
* Lets the callin's below out of their loop.
* wait to see if the cpu made a callin (is actually up).
* use this value that I found through experimentation.
* -- Cort
*/
*/
PPCDBG
(
PPCDBG_SMP
,
"smp_commence: start
\n
"
);
for
(
c
=
5000
;
c
&&
!
cpu_callin_map
[
cpu
];
c
--
)
wmb
();
udelay
(
100
);
smp_commenced
=
1
;
}
void
__init
smp_callin
(
void
)
{
int
cpu
=
smp_processor_id
();
smp_store_cpu_info
(
cpu
);
if
(
!
cpu_callin_map
[
cpu
])
{
set_dec
(
paca
[
cpu
].
default_decr
);
printk
(
"Processor %u is stuck.
\n
"
,
cpu
);
set_bit
(
smp_processor_id
(),
&
cpu_online_map
);
return
-
ENOENT
;
smp_mb
();
cpu_callin_map
[
cpu
]
=
1
;
ppc_md
.
smp_setup_cpu
(
cpu
);
while
(
!
smp_commenced
)
{
barrier
();
}
}
local_irq_enable
();
}
/* intel needs this */
printk
(
"Processor %u found.
\n
"
,
cpu
);
void
__init
initialize_secondary
(
void
)
{
if
(
smp_ops
->
give_timebase
)
smp_ops
->
give_timebase
();
set_bit
(
cpu
,
&
cpu_online_map
);
return
0
;
}
}
/* Activate a secondary processor. */
/* Activate a secondary processor. */
int
start_secondary
(
void
*
unused
)
int
__devinit
start_secondary
(
void
*
unused
)
{
{
unsigned
int
cpu
=
smp_processor_id
();
atomic_inc
(
&
init_mm
.
mm_count
);
atomic_inc
(
&
init_mm
.
mm_count
);
current
->
active_mm
=
&
init_mm
;
current
->
active_mm
=
&
init_mm
;
smp_callin
();
return
cpu_idle
(
NULL
);
smp_store_cpu_info
(
cpu
);
}
set_dec
(
paca
[
cpu
].
default_decr
);
cpu_callin_map
[
cpu
]
=
1
;
void
__init
smp_setup
(
char
*
str
,
int
*
ints
)
smp_ops
->
setup_cpu
(
cpu
);
{
if
(
smp_ops
->
take_timebase
)
smp_ops
->
take_timebase
();
/* XXX required? */
local_irq_enable
();
return
cpu_idle
(
NULL
);
}
}
int
setup_profiling_timer
(
unsigned
int
multiplier
)
int
setup_profiling_timer
(
unsigned
int
multiplier
)
...
@@ -729,17 +678,10 @@ int setup_profiling_timer(unsigned int multiplier)
...
@@ -729,17 +678,10 @@ int setup_profiling_timer(unsigned int multiplier)
return
0
;
return
0
;
}
}
/* this function is called for each processor
void
smp_cpus_done
(
unsigned
int
max_cpus
)
*/
void
__init
smp_store_cpu_info
(
int
id
)
{
{
paca
[
id
].
pvr
=
_get_PVR
();
smp_ops
->
setup_cpu
(
boot_cpuid
);
}
static
int
__init
maxcpus
(
char
*
str
)
/* XXX fix this, xics currently relies on it - Anton */
{
smp_threads_ready
=
1
;
get_option
(
&
str
,
&
max_cpus
);
return
1
;
}
}
__setup
(
"maxcpus="
,
maxcpus
);
arch/ppc64/kernel/xics.c
View file @
7a3182b1
...
@@ -373,7 +373,7 @@ xics_init_IRQ( void )
...
@@ -373,7 +373,7 @@ xics_init_IRQ( void )
if
(
naca
->
platform
==
PLATFORM_PSERIES
)
{
if
(
naca
->
platform
==
PLATFORM_PSERIES
)
{
#ifdef CONFIG_SMP
#ifdef CONFIG_SMP
for
(
i
=
0
;
i
<
NR_CPUS
;
++
i
)
{
for
(
i
=
0
;
i
<
NR_CPUS
;
++
i
)
{
if
(
!
paca
[
i
].
active
)
if
(
!
cpu_possible
(
i
)
)
continue
;
continue
;
xics_info
.
per_cpu
[
i
]
=
xics_info
.
per_cpu
[
i
]
=
__ioremap
((
ulong
)
inodes
[
i
].
addr
,
__ioremap
((
ulong
)
inodes
[
i
].
addr
,
...
...
include/asm-ppc64/machdep.h
View file @
7a3182b1
...
@@ -20,6 +20,17 @@ struct device_node;
...
@@ -20,6 +20,17 @@ struct device_node;
struct
TceTable
;
struct
TceTable
;
struct
rtc_time
;
struct
rtc_time
;
#ifdef CONFIG_SMP
struct
smp_ops_t
{
void
(
*
message_pass
)(
int
target
,
int
msg
,
unsigned
long
data
,
int
wait
);
int
(
*
probe
)(
void
);
void
(
*
kick_cpu
)(
int
nr
);
void
(
*
setup_cpu
)(
int
nr
);
void
(
*
take_timebase
)(
void
);
void
(
*
give_timebase
)(
void
);
};
#endif
struct
machdep_calls
{
struct
machdep_calls
{
/* High use functions in the first cachelines, low use functions
/* High use functions in the first cachelines, low use functions
* follow. DRENG collect profile data.
* follow. DRENG collect profile data.
...
@@ -58,14 +69,6 @@ struct machdep_calls {
...
@@ -58,14 +69,6 @@ struct machdep_calls {
void
(
*
tce_free_one
)(
struct
TceTable
*
tbl
,
void
(
*
tce_free_one
)(
struct
TceTable
*
tbl
,
long
tcenum
);
long
tcenum
);
void
(
*
smp_message_pass
)(
int
target
,
int
msg
,
unsigned
long
data
,
int
wait
);
void
(
*
smp_probe
)(
void
);
void
(
*
smp_kick_cpu
)(
int
nr
);
void
(
*
smp_setup_cpu
)(
int
nr
);
void
(
*
setup_arch
)(
void
);
void
(
*
setup_arch
)(
void
);
/* Optional, may be NULL. */
/* Optional, may be NULL. */
void
(
*
setup_residual
)(
struct
seq_file
*
m
,
int
cpu_id
);
void
(
*
setup_residual
)(
struct
seq_file
*
m
,
int
cpu_id
);
...
@@ -145,6 +148,11 @@ struct machdep_calls {
...
@@ -145,6 +148,11 @@ struct machdep_calls {
/* this is for modules, since _machine can be a define -- Cort */
/* this is for modules, since _machine can be a define -- Cort */
int
ppc_machine
;
int
ppc_machine
;
#ifdef CONFIG_SMP
/* functions for dealing with other cpus */
struct
smp_ops_t
smp_ops
;
#endif
/* CONFIG_SMP */
};
};
extern
struct
machdep_calls
ppc_md
;
extern
struct
machdep_calls
ppc_md
;
...
...
include/asm-ppc64/smp.h
View file @
7a3182b1
...
@@ -30,7 +30,6 @@
...
@@ -30,7 +30,6 @@
extern
unsigned
long
cpu_online_map
;
extern
unsigned
long
cpu_online_map
;
extern
void
smp_message_pass
(
int
target
,
int
msg
,
unsigned
long
data
,
int
wait
);
extern
void
smp_message_pass
(
int
target
,
int
msg
,
unsigned
long
data
,
int
wait
);
extern
void
smp_store_cpu_info
(
int
id
);
extern
void
smp_send_tlb_invalidate
(
int
);
extern
void
smp_send_tlb_invalidate
(
int
);
extern
void
smp_send_xmon_break
(
int
cpu
);
extern
void
smp_send_xmon_break
(
int
cpu
);
struct
pt_regs
;
struct
pt_regs
;
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment