1 /* backing_ops.c - query/set operations on saved SPU context.
3 * Copyright (C) IBM 2005
4 * Author: Mark Nutter <mnutter@us.ibm.com>
6 * These register operations allow SPUFS to operate on saved
7 * SPU contexts rather than hardware.
9 * This program is free software; you can redistribute it and/or modify
10 * it under the terms of the GNU General Public License as published by
11 * the Free Software Foundation; either version 2, or (at your option)
14 * This program is distributed in the hope that it will be useful,
15 * but WITHOUT ANY WARRANTY; without even the implied warranty of
16 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
17 * GNU General Public License for more details.
19 * You should have received a copy of the GNU General Public License
20 * along with this program; if not, write to the Free Software
21 * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
24 #include <linux/errno.h>
25 #include <linux/sched.h>
26 #include <linux/kernel.h>
28 #include <linux/vmalloc.h>
29 #include <linux/smp.h>
30 #include <linux/stddef.h>
31 #include <linux/unistd.h>
32 #include <linux/poll.h>
36 #include <asm/spu_csa.h>
37 #include <asm/spu_info.h>
38 #include <asm/mmu_context.h>
42 * Reads/writes to various problem and priv2 registers require
43 * state changes, i.e. generate SPU events, modify channel
47 static void gen_spu_event(struct spu_context
*ctx
, u32 event
)
53 ch0_cnt
= ctx
->csa
.spu_chnlcnt_RW
[0];
54 ch0_data
= ctx
->csa
.spu_chnldata_RW
[0];
55 ch1_data
= ctx
->csa
.spu_chnldata_RW
[1];
56 ctx
->csa
.spu_chnldata_RW
[0] |= event
;
57 if ((ch0_cnt
== 0) && !(ch0_data
& event
) && (ch1_data
& event
)) {
58 ctx
->csa
.spu_chnlcnt_RW
[0] = 1;
62 static int spu_backing_mbox_read(struct spu_context
*ctx
, u32
* data
)
67 spin_lock(&ctx
->csa
.register_lock
);
68 mbox_stat
= ctx
->csa
.prob
.mb_stat_R
;
69 if (mbox_stat
& 0x0000ff) {
70 /* Read the first available word.
71 * Implementation note: the depth
72 * of pu_mb_R is currently 1.
74 *data
= ctx
->csa
.prob
.pu_mb_R
;
75 ctx
->csa
.prob
.mb_stat_R
&= ~(0x0000ff);
76 ctx
->csa
.spu_chnlcnt_RW
[28] = 1;
77 gen_spu_event(ctx
, MFC_PU_MAILBOX_AVAILABLE_EVENT
);
80 spin_unlock(&ctx
->csa
.register_lock
);
84 static u32
spu_backing_mbox_stat_read(struct spu_context
*ctx
)
86 return ctx
->csa
.prob
.mb_stat_R
;
89 static unsigned int spu_backing_mbox_stat_poll(struct spu_context
*ctx
,
96 spin_lock_irq(&ctx
->csa
.register_lock
);
97 stat
= ctx
->csa
.prob
.mb_stat_R
;
99 /* if the requested event is there, return the poll
100 mask, otherwise enable the interrupt to get notified,
101 but first mark any pending interrupts as done so
102 we don't get woken up unnecessarily */
104 if (events
& (POLLIN
| POLLRDNORM
)) {
106 ret
|= POLLIN
| POLLRDNORM
;
108 ctx
->csa
.priv1
.int_stat_class2_RW
&=
109 ~CLASS2_MAILBOX_INTR
;
110 ctx
->csa
.priv1
.int_mask_class2_RW
|=
111 CLASS2_ENABLE_MAILBOX_INTR
;
114 if (events
& (POLLOUT
| POLLWRNORM
)) {
116 ret
= POLLOUT
| POLLWRNORM
;
118 ctx
->csa
.priv1
.int_stat_class2_RW
&=
119 ~CLASS2_MAILBOX_THRESHOLD_INTR
;
120 ctx
->csa
.priv1
.int_mask_class2_RW
|=
121 CLASS2_ENABLE_MAILBOX_THRESHOLD_INTR
;
124 spin_unlock_irq(&ctx
->csa
.register_lock
);
128 static int spu_backing_ibox_read(struct spu_context
*ctx
, u32
* data
)
132 spin_lock(&ctx
->csa
.register_lock
);
133 if (ctx
->csa
.prob
.mb_stat_R
& 0xff0000) {
134 /* Read the first available word.
135 * Implementation note: the depth
136 * of puint_mb_R is currently 1.
138 *data
= ctx
->csa
.priv2
.puint_mb_R
;
139 ctx
->csa
.prob
.mb_stat_R
&= ~(0xff0000);
140 ctx
->csa
.spu_chnlcnt_RW
[30] = 1;
141 gen_spu_event(ctx
, MFC_PU_INT_MAILBOX_AVAILABLE_EVENT
);
144 /* make sure we get woken up by the interrupt */
145 ctx
->csa
.priv1
.int_mask_class2_RW
|= CLASS2_ENABLE_MAILBOX_INTR
;
148 spin_unlock(&ctx
->csa
.register_lock
);
152 static int spu_backing_wbox_write(struct spu_context
*ctx
, u32 data
)
156 spin_lock(&ctx
->csa
.register_lock
);
157 if ((ctx
->csa
.prob
.mb_stat_R
) & 0x00ff00) {
158 int slot
= ctx
->csa
.spu_chnlcnt_RW
[29];
159 int avail
= (ctx
->csa
.prob
.mb_stat_R
& 0x00ff00) >> 8;
161 /* We have space to write wbox_data.
162 * Implementation note: the depth
163 * of spu_mb_W is currently 4.
165 BUG_ON(avail
!= (4 - slot
));
166 ctx
->csa
.spu_mailbox_data
[slot
] = data
;
167 ctx
->csa
.spu_chnlcnt_RW
[29] = ++slot
;
168 ctx
->csa
.prob
.mb_stat_R
&= ~(0x00ff00);
169 ctx
->csa
.prob
.mb_stat_R
|= (((4 - slot
) & 0xff) << 8);
170 gen_spu_event(ctx
, MFC_SPU_MAILBOX_WRITTEN_EVENT
);
173 /* make sure we get woken up by the interrupt when space
175 ctx
->csa
.priv1
.int_mask_class2_RW
|=
176 CLASS2_ENABLE_MAILBOX_THRESHOLD_INTR
;
179 spin_unlock(&ctx
->csa
.register_lock
);
183 static u32
spu_backing_signal1_read(struct spu_context
*ctx
)
185 return ctx
->csa
.spu_chnldata_RW
[3];
188 static void spu_backing_signal1_write(struct spu_context
*ctx
, u32 data
)
190 spin_lock(&ctx
->csa
.register_lock
);
191 if (ctx
->csa
.priv2
.spu_cfg_RW
& 0x1)
192 ctx
->csa
.spu_chnldata_RW
[3] |= data
;
194 ctx
->csa
.spu_chnldata_RW
[3] = data
;
195 ctx
->csa
.spu_chnlcnt_RW
[3] = 1;
196 gen_spu_event(ctx
, MFC_SIGNAL_1_EVENT
);
197 spin_unlock(&ctx
->csa
.register_lock
);
200 static u32
spu_backing_signal2_read(struct spu_context
*ctx
)
202 return ctx
->csa
.spu_chnldata_RW
[4];
205 static void spu_backing_signal2_write(struct spu_context
*ctx
, u32 data
)
207 spin_lock(&ctx
->csa
.register_lock
);
208 if (ctx
->csa
.priv2
.spu_cfg_RW
& 0x2)
209 ctx
->csa
.spu_chnldata_RW
[4] |= data
;
211 ctx
->csa
.spu_chnldata_RW
[4] = data
;
212 ctx
->csa
.spu_chnlcnt_RW
[4] = 1;
213 gen_spu_event(ctx
, MFC_SIGNAL_2_EVENT
);
214 spin_unlock(&ctx
->csa
.register_lock
);
217 static void spu_backing_signal1_type_set(struct spu_context
*ctx
, u64 val
)
221 spin_lock(&ctx
->csa
.register_lock
);
222 tmp
= ctx
->csa
.priv2
.spu_cfg_RW
;
227 ctx
->csa
.priv2
.spu_cfg_RW
= tmp
;
228 spin_unlock(&ctx
->csa
.register_lock
);
231 static u64
spu_backing_signal1_type_get(struct spu_context
*ctx
)
233 return ((ctx
->csa
.priv2
.spu_cfg_RW
& 1) != 0);
236 static void spu_backing_signal2_type_set(struct spu_context
*ctx
, u64 val
)
240 spin_lock(&ctx
->csa
.register_lock
);
241 tmp
= ctx
->csa
.priv2
.spu_cfg_RW
;
246 ctx
->csa
.priv2
.spu_cfg_RW
= tmp
;
247 spin_unlock(&ctx
->csa
.register_lock
);
250 static u64
spu_backing_signal2_type_get(struct spu_context
*ctx
)
252 return ((ctx
->csa
.priv2
.spu_cfg_RW
& 2) != 0);
255 static u32
spu_backing_npc_read(struct spu_context
*ctx
)
257 return ctx
->csa
.prob
.spu_npc_RW
;
260 static void spu_backing_npc_write(struct spu_context
*ctx
, u32 val
)
262 ctx
->csa
.prob
.spu_npc_RW
= val
;
265 static u32
spu_backing_status_read(struct spu_context
*ctx
)
267 return ctx
->csa
.prob
.spu_status_R
;
270 static char *spu_backing_get_ls(struct spu_context
*ctx
)
272 return ctx
->csa
.lscsa
->ls
;
275 static void spu_backing_privcntl_write(struct spu_context
*ctx
, u64 val
)
277 ctx
->csa
.priv2
.spu_privcntl_RW
= val
;
280 static u32
spu_backing_runcntl_read(struct spu_context
*ctx
)
282 return ctx
->csa
.prob
.spu_runcntl_RW
;
285 static void spu_backing_runcntl_write(struct spu_context
*ctx
, u32 val
)
287 spin_lock(&ctx
->csa
.register_lock
);
288 ctx
->csa
.prob
.spu_runcntl_RW
= val
;
289 if (val
& SPU_RUNCNTL_RUNNABLE
) {
290 ctx
->csa
.prob
.spu_status_R
&=
291 ~SPU_STATUS_STOPPED_BY_STOP
&
292 ~SPU_STATUS_STOPPED_BY_HALT
&
293 ~SPU_STATUS_SINGLE_STEP
&
294 ~SPU_STATUS_INVALID_INSTR
&
295 ~SPU_STATUS_INVALID_CH
;
296 ctx
->csa
.prob
.spu_status_R
|= SPU_STATUS_RUNNING
;
298 ctx
->csa
.prob
.spu_status_R
&= ~SPU_STATUS_RUNNING
;
300 spin_unlock(&ctx
->csa
.register_lock
);
303 static void spu_backing_runcntl_stop(struct spu_context
*ctx
)
305 spu_backing_runcntl_write(ctx
, SPU_RUNCNTL_STOP
);
308 static void spu_backing_master_start(struct spu_context
*ctx
)
310 struct spu_state
*csa
= &ctx
->csa
;
313 spin_lock(&csa
->register_lock
);
314 sr1
= csa
->priv1
.mfc_sr1_RW
| MFC_STATE1_MASTER_RUN_CONTROL_MASK
;
315 csa
->priv1
.mfc_sr1_RW
= sr1
;
316 spin_unlock(&csa
->register_lock
);
319 static void spu_backing_master_stop(struct spu_context
*ctx
)
321 struct spu_state
*csa
= &ctx
->csa
;
324 spin_lock(&csa
->register_lock
);
325 sr1
= csa
->priv1
.mfc_sr1_RW
& ~MFC_STATE1_MASTER_RUN_CONTROL_MASK
;
326 csa
->priv1
.mfc_sr1_RW
= sr1
;
327 spin_unlock(&csa
->register_lock
);
330 static int spu_backing_set_mfc_query(struct spu_context
* ctx
, u32 mask
,
333 struct spu_problem_collapsed
*prob
= &ctx
->csa
.prob
;
336 spin_lock(&ctx
->csa
.register_lock
);
338 if (prob
->dma_querytype_RW
)
341 /* FIXME: what are the side-effects of this? */
342 prob
->dma_querymask_RW
= mask
;
343 prob
->dma_querytype_RW
= mode
;
344 /* In the current implementation, the SPU context is always
345 * acquired in runnable state when new bits are added to the
346 * mask (tagwait), so it's sufficient just to mask
347 * dma_tagstatus_R with the 'mask' parameter here.
349 ctx
->csa
.prob
.dma_tagstatus_R
&= mask
;
351 spin_unlock(&ctx
->csa
.register_lock
);
356 static u32
spu_backing_read_mfc_tagstatus(struct spu_context
* ctx
)
358 return ctx
->csa
.prob
.dma_tagstatus_R
;
361 static u32
spu_backing_get_mfc_free_elements(struct spu_context
*ctx
)
363 return ctx
->csa
.prob
.dma_qstatus_R
;
366 static int spu_backing_send_mfc_command(struct spu_context
*ctx
,
367 struct mfc_dma_command
*cmd
)
371 spin_lock(&ctx
->csa
.register_lock
);
373 /* FIXME: set up priv2->puq */
374 spin_unlock(&ctx
->csa
.register_lock
);
379 static void spu_backing_restart_dma(struct spu_context
*ctx
)
381 ctx
->csa
.priv2
.mfc_control_RW
|= MFC_CNTL_RESTART_DMA_COMMAND
;
384 struct spu_context_ops spu_backing_ops
= {
385 .mbox_read
= spu_backing_mbox_read
,
386 .mbox_stat_read
= spu_backing_mbox_stat_read
,
387 .mbox_stat_poll
= spu_backing_mbox_stat_poll
,
388 .ibox_read
= spu_backing_ibox_read
,
389 .wbox_write
= spu_backing_wbox_write
,
390 .signal1_read
= spu_backing_signal1_read
,
391 .signal1_write
= spu_backing_signal1_write
,
392 .signal2_read
= spu_backing_signal2_read
,
393 .signal2_write
= spu_backing_signal2_write
,
394 .signal1_type_set
= spu_backing_signal1_type_set
,
395 .signal1_type_get
= spu_backing_signal1_type_get
,
396 .signal2_type_set
= spu_backing_signal2_type_set
,
397 .signal2_type_get
= spu_backing_signal2_type_get
,
398 .npc_read
= spu_backing_npc_read
,
399 .npc_write
= spu_backing_npc_write
,
400 .status_read
= spu_backing_status_read
,
401 .get_ls
= spu_backing_get_ls
,
402 .privcntl_write
= spu_backing_privcntl_write
,
403 .runcntl_read
= spu_backing_runcntl_read
,
404 .runcntl_write
= spu_backing_runcntl_write
,
405 .runcntl_stop
= spu_backing_runcntl_stop
,
406 .master_start
= spu_backing_master_start
,
407 .master_stop
= spu_backing_master_stop
,
408 .set_mfc_query
= spu_backing_set_mfc_query
,
409 .read_mfc_tagstatus
= spu_backing_read_mfc_tagstatus
,
410 .get_mfc_free_elements
= spu_backing_get_mfc_free_elements
,
411 .send_mfc_command
= spu_backing_send_mfc_command
,
412 .restart_dma
= spu_backing_restart_dma
,