Skip to content
GitLab
Projects
Groups
Snippets
/
Help
Help
Support
Community forum
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
Menu
Open sidebar
xcap
xcap-capability-linux
Commits
944b380e
Commit
944b380e
authored
Feb 14, 2007
by
Paul Mackerras
Browse files
Merge branch 'cell-merge' of
git+ssh://master.kernel.org/pub/scm/linux/kernel/git/arnd/cell-2.6
parents
fff5f528
c7eb7347
Changes
10
Hide whitespace changes
Inline
Side-by-side
arch/powerpc/oprofile/op_model_cell.c
View file @
944b380e
...
...
@@ -39,10 +39,17 @@
#include
"../platforms/cell/interrupt.h"
#define PPU_CYCLES_EVENT_NUM 1
/* event number for CYCLES */
#define PPU_CYCLES_GRP_NUM 1
/* special group number for identifying
* PPU_CYCLES event
*/
#define CBE_COUNT_ALL_CYCLES 0x42800000
/* PPU cycle event specifier */
#define NUM_THREADS 2
#define VIRT_CNTR_SW_TIME_NS 100000000 // 0.5 seconds
#define NUM_THREADS 2
/* number of physical threads in
* physical processor
*/
#define NUM_TRACE_BUS_WORDS 4
#define NUM_INPUT_BUS_WORDS 2
struct
pmc_cntrl_data
{
unsigned
long
vcntr
;
...
...
@@ -58,7 +65,7 @@ struct pmc_cntrl_data {
struct
pm_signal
{
u16
cpu
;
/* Processor to modify */
u16
sub_unit
;
/* hw subunit this applies to (if applicable) */
u16
signal_group
;
/* Signal Group to Enable/Disable */
short
int
signal_group
;
/* Signal Group to Enable/Disable */
u8
bus_word
;
/* Enable/Disable on this Trace/Trigger/Event
* Bus Word(s) (bitmask)
*/
...
...
@@ -93,7 +100,6 @@ static struct {
u32
pm07_cntrl
[
NR_PHYS_CTRS
];
}
pm_regs
;
#define GET_SUB_UNIT(x) ((x & 0x0000f000) >> 12)
#define GET_BUS_WORD(x) ((x & 0x000000f0) >> 4)
#define GET_BUS_TYPE(x) ((x & 0x00000300) >> 8)
...
...
@@ -101,7 +107,6 @@ static struct {
#define GET_COUNT_CYCLES(x) (x & 0x00000001)
#define GET_INPUT_CONTROL(x) ((x & 0x00000004) >> 2)
static
DEFINE_PER_CPU
(
unsigned
long
[
NR_PHYS_CTRS
],
pmc_values
);
static
struct
pmc_cntrl_data
pmc_cntrl
[
NUM_THREADS
][
NR_PHYS_CTRS
];
...
...
@@ -129,8 +134,8 @@ static spinlock_t virt_cntr_lock = SPIN_LOCK_UNLOCKED;
static
u32
ctr_enabled
;
static
unsigned
char
trace_bus
[
4
];
static
unsigned
char
input_bus
[
2
];
static
unsigned
char
trace_bus
[
NUM_TRACE_BUS_WORDS
];
static
unsigned
char
input_bus
[
NUM_INPUT_BUS_WORDS
];
/*
* Firmware interface functions
...
...
@@ -177,25 +182,40 @@ static void pm_rtas_reset_signals(u32 node)
static
void
pm_rtas_activate_signals
(
u32
node
,
u32
count
)
{
int
ret
;
int
j
;
int
i
,
j
;
struct
pm_signal
pm_signal_local
[
NR_PHYS_CTRS
];
/* There is no debug setup required for the cycles event.
* Note that only events in the same group can be used.
* Otherwise, there will be conflicts in correctly routing
* the signals on the debug bus. It is the responsiblity
* of the OProfile user tool to check the events are in
* the same group.
*/
i
=
0
;
for
(
j
=
0
;
j
<
count
;
j
++
)
{
/* fw expects physical cpu # */
pm_signal_local
[
j
].
cpu
=
node
;
pm_signal_local
[
j
].
signal_group
=
pm_signal
[
j
].
signal_group
;
pm_signal_local
[
j
].
bus_word
=
pm_signal
[
j
].
bus_word
;
pm_signal_local
[
j
].
sub_unit
=
pm_signal
[
j
].
sub_unit
;
pm_signal_local
[
j
].
bit
=
pm_signal
[
j
].
bit
;
if
(
pm_signal
[
j
].
signal_group
!=
PPU_CYCLES_GRP_NUM
)
{
/* fw expects physical cpu # */
pm_signal_local
[
i
].
cpu
=
node
;
pm_signal_local
[
i
].
signal_group
=
pm_signal
[
j
].
signal_group
;
pm_signal_local
[
i
].
bus_word
=
pm_signal
[
j
].
bus_word
;
pm_signal_local
[
i
].
sub_unit
=
pm_signal
[
j
].
sub_unit
;
pm_signal_local
[
i
].
bit
=
pm_signal
[
j
].
bit
;
i
++
;
}
}
ret
=
rtas_ibm_cbe_perftools
(
SUBFUNC_ACTIVATE
,
PASSTHRU_ENABLE
,
pm_signal_local
,
count
*
sizeof
(
struct
pm_signal
));
if
(
i
!=
0
)
{
ret
=
rtas_ibm_cbe_perftools
(
SUBFUNC_ACTIVATE
,
PASSTHRU_ENABLE
,
pm_signal_local
,
i
*
sizeof
(
struct
pm_signal
));
if
(
ret
)
printk
(
KERN_WARNING
"%s: rtas returned: %d
\n
"
,
__FUNCTION__
,
ret
);
if
(
ret
)
printk
(
KERN_WARNING
"%s: rtas returned: %d
\n
"
,
__FUNCTION__
,
ret
);
}
}
/*
...
...
@@ -212,7 +232,7 @@ static void set_pm_event(u32 ctr, int event, u32 unit_mask)
/* Special Event: Count all cpu cycles */
pm_regs
.
pm07_cntrl
[
ctr
]
=
CBE_COUNT_ALL_CYCLES
;
p
=
&
(
pm_signal
[
ctr
]);
p
->
signal_group
=
21
;
p
->
signal_group
=
PPU_CYCLES_GRP_NUM
;
p
->
bus_word
=
1
;
p
->
sub_unit
=
0
;
p
->
bit
=
0
;
...
...
@@ -232,13 +252,21 @@ static void set_pm_event(u32 ctr, int event, u32 unit_mask)
p
->
signal_group
=
event
/
100
;
p
->
bus_word
=
bus_word
;
p
->
sub_unit
=
unit_mask
&
0x0000f000
;
p
->
sub_unit
=
(
unit_mask
&
0x0000f000
)
>>
12
;
pm_regs
.
pm07_cntrl
[
ctr
]
=
0
;
pm_regs
.
pm07_cntrl
[
ctr
]
|=
PM07_CTR_COUNT_CYCLES
(
count_cycles
);
pm_regs
.
pm07_cntrl
[
ctr
]
|=
PM07_CTR_POLARITY
(
polarity
);
pm_regs
.
pm07_cntrl
[
ctr
]
|=
PM07_CTR_INPUT_CONTROL
(
input_control
);
/* Some of the islands signal selection is based on 64 bit words.
* The debug bus words are 32 bits, the input words to the performance
* counters are defined as 32 bits. Need to convert the 64 bit island
* specification to the appropriate 32 input bit and bus word for the
* performance counter event selection. See the CELL Performance
* monitoring signals manual and the Perf cntr hardware descriptions
* for the details.
*/
if
(
input_control
==
0
)
{
if
(
signal_bit
>
31
)
{
signal_bit
-=
32
;
...
...
@@ -259,12 +287,12 @@ static void set_pm_event(u32 ctr, int event, u32 unit_mask)
p
->
bit
=
signal_bit
;
}
for
(
i
=
0
;
i
<
4
;
i
++
)
{
for
(
i
=
0
;
i
<
NUM_TRACE_BUS_WORDS
;
i
++
)
{
if
(
bus_word
&
(
1
<<
i
))
{
pm_regs
.
debug_bus_control
|=
(
bus_type
<<
(
31
-
(
2
*
i
)
+
1
));
for
(
j
=
0
;
j
<
2
;
j
++
)
{
for
(
j
=
0
;
j
<
NUM_INPUT_BUS_WORDS
;
j
++
)
{
if
(
input_bus
[
j
]
==
0xff
)
{
input_bus
[
j
]
=
i
;
pm_regs
.
group_control
|=
...
...
@@ -278,52 +306,58 @@ out:
;
}
static
void
write_pm_cntrl
(
int
cpu
,
struct
pm_cntrl
*
pm_cntrl
)
static
void
write_pm_cntrl
(
int
cpu
)
{
/* Oprofile will use 32 bit counters, set bits 7:10 to 0 */
/* Oprofile will use 32 bit counters, set bits 7:10 to 0
* pmregs.pm_cntrl is a global
*/
u32
val
=
0
;
if
(
pm_cntrl
->
enable
==
1
)
if
(
pm_
regs
.
pm_
cntrl
.
enable
==
1
)
val
|=
CBE_PM_ENABLE_PERF_MON
;
if
(
pm_cntrl
->
stop_at_max
==
1
)
if
(
pm_
regs
.
pm_
cntrl
.
stop_at_max
==
1
)
val
|=
CBE_PM_STOP_AT_MAX
;
if
(
pm_cntrl
->
trace_mode
==
1
)
val
|=
CBE_PM_TRACE_MODE_SET
(
pm_cntrl
->
trace_mode
);
if
(
pm_
regs
.
pm_
cntrl
.
trace_mode
==
1
)
val
|=
CBE_PM_TRACE_MODE_SET
(
pm_
regs
.
pm_
cntrl
.
trace_mode
);
if
(
pm_cntrl
->
freeze
==
1
)
if
(
pm_
regs
.
pm_
cntrl
.
freeze
==
1
)
val
|=
CBE_PM_FREEZE_ALL_CTRS
;
/* Routine set_count_mode must be called previously to set
* the count mode based on the user selection of user and kernel.
*/
val
|=
CBE_PM_COUNT_MODE_SET
(
pm_cntrl
->
count_mode
);
val
|=
CBE_PM_COUNT_MODE_SET
(
pm_
regs
.
pm_
cntrl
.
count_mode
);
cbe_write_pm
(
cpu
,
pm_control
,
val
);
}
static
inline
void
set_count_mode
(
u32
kernel
,
u32
user
,
struct
pm_cntrl
*
pm_cntrl
)
set_count_mode
(
u32
kernel
,
u32
user
)
{
/* The user must specify user and kernel if they want them. If
* neither is specified, OProfile will count in hypervisor mode
* neither is specified, OProfile will count in hypervisor mode.
* pm_regs.pm_cntrl is a global
*/
if
(
kernel
)
{
if
(
user
)
pm_cntrl
->
count_mode
=
CBE_COUNT_ALL_MODES
;
pm_regs
.
pm_cntrl
.
count_mode
=
CBE_COUNT_ALL_MODES
;
else
pm_cntrl
->
count_mode
=
CBE_COUNT_SUPERVISOR_MODE
;
pm_regs
.
pm_cntrl
.
count_mode
=
CBE_COUNT_SUPERVISOR_MODE
;
}
else
{
if
(
user
)
pm_cntrl
->
count_mode
=
CBE_COUNT_PROBLEM_MODE
;
pm_regs
.
pm_cntrl
.
count_mode
=
CBE_COUNT_PROBLEM_MODE
;
else
pm_cntrl
->
count_mode
=
CBE_COUNT_HYPERVISOR_MODE
;
pm_regs
.
pm_cntrl
.
count_mode
=
CBE_COUNT_HYPERVISOR_MODE
;
}
}
static
inline
void
enable_ctr
(
u32
cpu
,
u32
ctr
,
u32
*
pm07_cntrl
)
{
pm07_cntrl
[
ctr
]
|=
PM
07
_CTR_ENABLE
(
1
)
;
pm07_cntrl
[
ctr
]
|=
CBE_
PM_CTR_ENABLE
;
cbe_write_pm07_control
(
cpu
,
ctr
,
pm07_cntrl
[
ctr
]);
}
...
...
@@ -365,6 +399,14 @@ static void cell_virtual_cntr(unsigned long data)
hdw_thread
=
1
^
hdw_thread
;
next_hdw_thread
=
hdw_thread
;
for
(
i
=
0
;
i
<
num_counters
;
i
++
)
/* There are some per thread events. Must do the
* set event, for the thread that is being started
*/
set_pm_event
(
i
,
pmc_cntrl
[
next_hdw_thread
][
i
].
evnts
,
pmc_cntrl
[
next_hdw_thread
][
i
].
masks
);
/* The following is done only once per each node, but
* we need cpu #, not node #, to pass to the cbe_xxx functions.
*/
...
...
@@ -385,12 +427,13 @@ static void cell_virtual_cntr(unsigned long data)
==
0xFFFFFFFF
)
/* If the cntr value is 0xffffffff, we must
* reset that to 0xfffffff0 when the current
* thread is restarted. This will generate a new
* interrupt and make sure that we never restore
* the counters to the max value. If the counters
* were restored to the max value, they do not
* increment and no interrupts are generated. Hence
* no more samples will be collected on that cpu.
* thread is restarted. This will generate a
* new interrupt and make sure that we never
* restore the counters to the max value. If
* the counters were restored to the max value,
* they do not increment and no interrupts are
* generated. Hence no more samples will be
* collected on that cpu.
*/
cbe_write_ctr
(
cpu
,
i
,
0xFFFFFFF0
);
else
...
...
@@ -410,9 +453,6 @@ static void cell_virtual_cntr(unsigned long data)
* Must do the set event, enable_cntr
* for each cpu.
*/
set_pm_event
(
i
,
pmc_cntrl
[
next_hdw_thread
][
i
].
evnts
,
pmc_cntrl
[
next_hdw_thread
][
i
].
masks
);
enable_ctr
(
cpu
,
i
,
pm_regs
.
pm07_cntrl
);
}
else
{
...
...
@@ -465,8 +505,7 @@ cell_reg_setup(struct op_counter_config *ctr,
pm_regs
.
pm_cntrl
.
trace_mode
=
0
;
pm_regs
.
pm_cntrl
.
freeze
=
1
;
set_count_mode
(
sys
->
enable_kernel
,
sys
->
enable_user
,
&
pm_regs
.
pm_cntrl
);
set_count_mode
(
sys
->
enable_kernel
,
sys
->
enable_user
);
/* Setup the thread 0 events */
for
(
i
=
0
;
i
<
num_ctrs
;
++
i
)
{
...
...
@@ -498,10 +537,10 @@ cell_reg_setup(struct op_counter_config *ctr,
pmc_cntrl
[
1
][
i
].
vcntr
=
i
;
}
for
(
i
=
0
;
i
<
4
;
i
++
)
for
(
i
=
0
;
i
<
NUM_TRACE_BUS_WORDS
;
i
++
)
trace_bus
[
i
]
=
0xff
;
for
(
i
=
0
;
i
<
2
;
i
++
)
for
(
i
=
0
;
i
<
NUM_INPUT_BUS_WORDS
;
i
++
)
input_bus
[
i
]
=
0xff
;
/* Our counters count up, and "count" refers to
...
...
@@ -560,7 +599,7 @@ static void cell_cpu_setup(struct op_counter_config *cntr)
cbe_write_pm
(
cpu
,
pm_start_stop
,
0
);
cbe_write_pm
(
cpu
,
group_control
,
pm_regs
.
group_control
);
cbe_write_pm
(
cpu
,
debug_bus_control
,
pm_regs
.
debug_bus_control
);
write_pm_cntrl
(
cpu
,
&
pm_regs
.
pm_cntrl
);
write_pm_cntrl
(
cpu
);
for
(
i
=
0
;
i
<
num_counters
;
++
i
)
{
if
(
ctr_enabled
&
(
1
<<
i
))
{
...
...
@@ -602,7 +641,7 @@ static void cell_global_start(struct op_counter_config *ctr)
}
}
cbe_clear_pm_interrupts
(
cpu
);
cbe_
get_and_
clear_pm_interrupts
(
cpu
);
cbe_enable_pm_interrupts
(
cpu
,
hdw_thread
,
interrupt_mask
);
cbe_enable_pm
(
cpu
);
}
...
...
@@ -672,7 +711,7 @@ cell_handle_interrupt(struct pt_regs *regs, struct op_counter_config *ctr)
cbe_disable_pm
(
cpu
);
interrupt_mask
=
cbe_clear_pm_interrupts
(
cpu
);
interrupt_mask
=
cbe_
get_and_
clear_pm_interrupts
(
cpu
);
/* If the interrupt mask has been cleared, then the virt cntr
* has cleared the interrupt. When the thread that generated
...
...
arch/powerpc/platforms/cell/pmu.c
View file @
944b380e
...
...
@@ -345,18 +345,12 @@ EXPORT_SYMBOL_GPL(cbe_read_trace_buffer);
* Enabling/disabling interrupts for the entire performance monitoring unit.
*/
u32
cbe_query_pm_interrupts
(
u32
cpu
)
{
return
cbe_read_pm
(
cpu
,
pm_status
);
}
EXPORT_SYMBOL_GPL
(
cbe_query_pm_interrupts
);
u32
cbe_clear_pm_interrupts
(
u32
cpu
)
u32
cbe_get_and_clear_pm_interrupts
(
u32
cpu
)
{
/* Reading pm_status clears the interrupt bits. */
return
cbe_
query_pm_interrupts
(
cpu
);
return
cbe_
read_pm
(
cpu
,
pm_status
);
}
EXPORT_SYMBOL_GPL
(
cbe_clear_pm_interrupts
);
EXPORT_SYMBOL_GPL
(
cbe_
get_and_
clear_pm_interrupts
);
void
cbe_enable_pm_interrupts
(
u32
cpu
,
u32
thread
,
u32
mask
)
{
...
...
@@ -371,7 +365,7 @@ EXPORT_SYMBOL_GPL(cbe_enable_pm_interrupts);
void
cbe_disable_pm_interrupts
(
u32
cpu
)
{
cbe_clear_pm_interrupts
(
cpu
);
cbe_
get_and_
clear_pm_interrupts
(
cpu
);
cbe_write_pm
(
cpu
,
pm_status
,
0
);
}
EXPORT_SYMBOL_GPL
(
cbe_disable_pm_interrupts
);
...
...
arch/powerpc/platforms/cell/spufs/context.c
View file @
944b380e
...
...
@@ -42,7 +42,7 @@ struct spu_context *alloc_spu_context(struct spu_gang *gang)
}
spin_lock_init
(
&
ctx
->
mmio_lock
);
kref_init
(
&
ctx
->
kref
);
init_rwsem
(
&
ctx
->
state_
sema
);
mutex_init
(
&
ctx
->
state_
mutex
);
init_MUTEX
(
&
ctx
->
run_sema
);
init_waitqueue_head
(
&
ctx
->
ibox_wq
);
init_waitqueue_head
(
&
ctx
->
wbox_wq
);
...
...
@@ -53,6 +53,10 @@ struct spu_context *alloc_spu_context(struct spu_gang *gang)
ctx
->
owner
=
get_task_mm
(
current
);
if
(
gang
)
spu_gang_add_ctx
(
gang
,
ctx
);
ctx
->
rt_priority
=
current
->
rt_priority
;
ctx
->
policy
=
current
->
policy
;
ctx
->
prio
=
current
->
prio
;
INIT_DELAYED_WORK
(
&
ctx
->
sched_work
,
spu_sched_tick
);
goto
out
;
out_free:
kfree
(
ctx
);
...
...
@@ -65,9 +69,9 @@ void destroy_spu_context(struct kref *kref)
{
struct
spu_context
*
ctx
;
ctx
=
container_of
(
kref
,
struct
spu_context
,
kref
);
down_write
(
&
ctx
->
state_
sema
);
mutex_lock
(
&
ctx
->
state_
mutex
);
spu_deactivate
(
ctx
);
up_write
(
&
ctx
->
state_
sema
);
mutex_unlock
(
&
ctx
->
state_
mutex
);
spu_fini_csa
(
&
ctx
->
csa
);
if
(
ctx
->
gang
)
spu_gang_remove_ctx
(
ctx
->
gang
,
ctx
);
...
...
@@ -96,16 +100,6 @@ void spu_forget(struct spu_context *ctx)
spu_release
(
ctx
);
}
void
spu_acquire
(
struct
spu_context
*
ctx
)
{
down_read
(
&
ctx
->
state_sema
);
}
void
spu_release
(
struct
spu_context
*
ctx
)
{
up_read
(
&
ctx
->
state_sema
);
}
void
spu_unmap_mappings
(
struct
spu_context
*
ctx
)
{
if
(
ctx
->
local_store
)
...
...
@@ -124,83 +118,84 @@ void spu_unmap_mappings(struct spu_context *ctx)
unmap_mapping_range
(
ctx
->
psmap
,
0
,
0x20000
,
1
);
}
/**
* spu_acquire_exclusive - lock spu contex and protect against userspace access
* @ctx: spu contex to lock
*
* Note:
* Returns 0 and with the context locked on success
* Returns negative error and with the context _unlocked_ on failure.
*/
int
spu_acquire_exclusive
(
struct
spu_context
*
ctx
)
{
int
ret
=
0
;
int
ret
=
-
EINVAL
;
down_writ
e
(
&
ctx
->
state_sema
);
/*
ctx is about to be freed, can't acquire any more */
if
(
!
ctx
->
owner
)
{
ret
=
-
EINVAL
;
goto
out
;
}
spu_acquir
e
(
ctx
);
/*
* Context is about to be freed, so we can't acquire it anymore.
*/
if
(
!
ctx
->
owner
)
goto
out_unlock
;
if
(
ctx
->
state
==
SPU_STATE_SAVED
)
{
ret
=
spu_activate
(
ctx
,
0
);
if
(
ret
)
goto
out
;
ctx
->
state
=
SPU_STATE_RUNNABLE
;
goto
out_unlock
;
}
else
{
/* We need to exclude userspace access to the context. */
/*
* We need to exclude userspace access to the context.
*
* To protect against memory access we invalidate all ptes
* and make sure the pagefault handlers block on the mutex.
*/
spu_unmap_mappings
(
ctx
);
}
out:
if
(
ret
)
up_write
(
&
ctx
->
state_sema
);
return
0
;
out_unlock:
spu_release
(
ctx
);
return
ret
;
}
int
spu_acquire_runnable
(
struct
spu_context
*
ctx
)
/**
* spu_acquire_runnable - lock spu contex and make sure it is in runnable state
* @ctx: spu contex to lock
*
* Note:
* Returns 0 and with the context locked on success
* Returns negative error and with the context _unlocked_ on failure.
*/
int
spu_acquire_runnable
(
struct
spu_context
*
ctx
,
unsigned
long
flags
)
{
int
ret
=
0
;
down_read
(
&
ctx
->
state_sema
);
if
(
ctx
->
state
==
SPU_STATE_RUNNABLE
)
{
ctx
->
spu
->
prio
=
current
->
prio
;
return
0
;
}
up_read
(
&
ctx
->
state_sema
);
down_write
(
&
ctx
->
state_sema
);
/* ctx is about to be freed, can't acquire any more */
if
(
!
ctx
->
owner
)
{
ret
=
-
EINVAL
;
goto
out
;
}
int
ret
=
-
EINVAL
;
spu_acquire
(
ctx
);
if
(
ctx
->
state
==
SPU_STATE_SAVED
)
{
ret
=
spu_activate
(
ctx
,
0
);
/*
* Context is about to be freed, so we can't acquire it anymore.
*/
if
(
!
ctx
->
owner
)
goto
out_unlock
;
ret
=
spu_activate
(
ctx
,
flags
);
if
(
ret
)
goto
out
;
ctx
->
state
=
SPU_STATE_RUNNABLE
;
goto
out_unlock
;
}
downgrade_write
(
&
ctx
->
state_sema
);
/* On success, we return holding the lock */
return
ret
;
out:
/* Release here, to simplify calling code. */
up_write
(
&
ctx
->
state_sema
);
return
0
;
out_unlock:
spu_release
(
ctx
);
return
ret
;
}
/**
* spu_acquire_saved - lock spu contex and make sure it is in saved state
* @ctx: spu contex to lock
*/
void
spu_acquire_saved
(
struct
spu_context
*
ctx
)
{
down_read
(
&
ctx
->
state_sema
);
if
(
ctx
->
state
==
SPU_STATE_SAVED
)
return
;
up_read
(
&
ctx
->
state_sema
);
down_write
(
&
ctx
->
state_sema
);
if
(
ctx
->
state
==
SPU_STATE_RUNNABLE
)
{
spu_acquire
(
ctx
);
if
(
ctx
->
state
!=
SPU_STATE_SAVED
)
spu_deactivate
(
ctx
);
ctx
->
state
=
SPU_STATE_SAVED
;
}
downgrade_write
(
&
ctx
->
state_sema
);
}
arch/powerpc/platforms/cell/spufs/file.c
View file @
944b380e
...
...
@@ -103,6 +103,9 @@ static unsigned long spufs_mem_mmap_nopfn(struct vm_area_struct *vma,
offset
+=
vma
->
vm_pgoff
<<
PAGE_SHIFT
;
if
(
offset
>=
LS_SIZE
)
return
NOPFN_SIGBUS
;
spu_acquire
(
ctx
);
if
(
ctx
->
state
==
SPU_STATE_SAVED
)
{
...
...
@@ -164,7 +167,7 @@ static unsigned long spufs_ps_nopfn(struct vm_area_struct *vma,
/* error here usually means a signal.. we might want to test
* the error code more precisely though
*/
ret
=
spu_acquire_runnable
(
ctx
);
ret
=
spu_acquire_runnable
(
ctx
,
0
);
if
(
ret
)
return
NOPFN_REFAULT
;
...
...
@@ -1306,7 +1309,7 @@ static ssize_t spufs_mfc_write(struct file *file, const char __user *buffer,
if
(
ret
)
goto
out
;
spu_acquire_runnable
(
ctx
);
spu_acquire_runnable
(
ctx
,
0
);
if
(
file
->
f_flags
&
O_NONBLOCK
)
{
ret
=
ctx
->
ops
->
send_mfc_command
(
ctx
,
&
cmd
);
}
else
{
...
...
arch/powerpc/platforms/cell/spufs/run.c
View file @
944b380e
...
...
@@ -133,7 +133,7 @@ out_drop_priv:
spu_mfc_sr1_set
(
ctx
->
spu
,
sr1
);
out_unlock:
spu_release
_exclusive
(
ctx
);
spu_release
(
ctx
);
out:
return
ret
;
}
...
...
@@ -143,7 +143,7 @@ static inline int spu_run_init(struct spu_context *ctx, u32 * npc)
int
ret
;
unsigned
long
runcntl
=
SPU_RUNCNTL_RUNNABLE
;
ret
=
spu_acquire_runnable
(
ctx
);
ret
=
spu_acquire_runnable
(
ctx
,
SPU_ACTIVATE_NOWAKE
);
if
(
ret
)
return
ret
;
...
...
@@ -155,7 +155,7 @@ static inline int spu_run_init(struct spu_context *ctx, u32 * npc)
spu_release
(
ctx
);
ret
=
spu_setup_isolated
(
ctx
);
if
(
!
ret
)
ret
=
spu_acquire_runnable
(
ctx
);
ret
=
spu_acquire_runnable
(
ctx
,
SPU_ACTIVATE_NOWAKE
);
}
/* if userspace has set the runcntrl register (eg, to issue an
...
...
@@ -164,8 +164,10 @@ static inline int spu_run_init(struct spu_context *ctx, u32 * npc)
(
SPU_RUNCNTL_RUNNABLE
|
SPU_RUNCNTL_ISOLATE
);
if
(
runcntl
==
0
)
runcntl
=
SPU_RUNCNTL_RUNNABLE
;
}
else
}
else
{
spu_start_tick
(
ctx
);
ctx
->
ops
->
npc_write
(
ctx
,
*
npc
);
}
ctx
->
ops
->
runcntl_write
(
ctx
,
runcntl
);
return
ret
;
...
...
@@ -176,6 +178,7 @@ static inline int spu_run_fini(struct spu_context *ctx, u32 * npc,
{
int
ret
=
0
;
spu_stop_tick
(
ctx
);
*
status
=
ctx
->
ops
->
status_read
(
ctx
);
*
npc
=
ctx
->
ops
->
npc_read
(
ctx
);
spu_release
(
ctx
);
...
...
@@ -329,8 +332,10 @@ long spufs_run_spu(struct file *file, struct spu_context *ctx,
}
if
(
unlikely
(
ctx
->
state
!=
SPU_STATE_RUNNABLE
))
{
ret
=
spu_reacquire_runnable
(
ctx
,
npc
,
&
status
);
if
(
ret
)
if
(
ret
)
{
spu_stop_tick
(
ctx
);
goto
out2
;
}
continue
;