Merge branch 'master' of git://git.kernel.org/pub/scm/linux/kernel/git/jk/spufs into merge
This commit is contained in:
Коммит
b9c64498f5
|
@ -81,9 +81,12 @@ struct spu_slb {
|
|||
void spu_invalidate_slbs(struct spu *spu)
|
||||
{
|
||||
struct spu_priv2 __iomem *priv2 = spu->priv2;
|
||||
unsigned long flags;
|
||||
|
||||
spin_lock_irqsave(&spu->register_lock, flags);
|
||||
if (spu_mfc_sr1_get(spu) & MFC_STATE1_RELOCATE_MASK)
|
||||
out_be64(&priv2->slb_invalidate_all_W, 0UL);
|
||||
spin_unlock_irqrestore(&spu->register_lock, flags);
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(spu_invalidate_slbs);
|
||||
|
||||
|
@ -148,7 +151,11 @@ static inline void spu_load_slb(struct spu *spu, int slbe, struct spu_slb *slb)
|
|||
__func__, slbe, slb->vsid, slb->esid);
|
||||
|
||||
out_be64(&priv2->slb_index_W, slbe);
|
||||
/* set invalid before writing vsid */
|
||||
out_be64(&priv2->slb_esid_RW, 0);
|
||||
/* now it's safe to write the vsid */
|
||||
out_be64(&priv2->slb_vsid_RW, slb->vsid);
|
||||
/* setting the new esid makes the entry valid again */
|
||||
out_be64(&priv2->slb_esid_RW, slb->esid);
|
||||
}
|
||||
|
||||
|
@ -290,9 +297,11 @@ void spu_setup_kernel_slbs(struct spu *spu, struct spu_lscsa *lscsa,
|
|||
nr_slbs++;
|
||||
}
|
||||
|
||||
spin_lock_irq(&spu->register_lock);
|
||||
/* Add the set of SLBs */
|
||||
for (i = 0; i < nr_slbs; i++)
|
||||
spu_load_slb(spu, i, &slbs[i]);
|
||||
spin_unlock_irq(&spu->register_lock);
|
||||
}
|
||||
EXPORT_SYMBOL_GPL(spu_setup_kernel_slbs);
|
||||
|
||||
|
@ -337,13 +346,14 @@ spu_irq_class_1(int irq, void *data)
|
|||
if (stat & CLASS1_STORAGE_FAULT_INTR)
|
||||
spu_mfc_dsisr_set(spu, 0ul);
|
||||
spu_int_stat_clear(spu, 1, stat);
|
||||
spin_unlock(&spu->register_lock);
|
||||
pr_debug("%s: %lx %lx %lx %lx\n", __FUNCTION__, mask, stat,
|
||||
dar, dsisr);
|
||||
|
||||
if (stat & CLASS1_SEGMENT_FAULT_INTR)
|
||||
__spu_trap_data_seg(spu, dar);
|
||||
|
||||
spin_unlock(&spu->register_lock);
|
||||
pr_debug("%s: %lx %lx %lx %lx\n", __FUNCTION__, mask, stat,
|
||||
dar, dsisr);
|
||||
|
||||
if (stat & CLASS1_STORAGE_FAULT_INTR)
|
||||
__spu_trap_data_map(spu, dar, dsisr);
|
||||
|
||||
|
|
|
@ -109,13 +109,12 @@ void spu_forget(struct spu_context *ctx)
|
|||
|
||||
/*
|
||||
* This is basically an open-coded spu_acquire_saved, except that
|
||||
* we don't acquire the state mutex interruptible.
|
||||
* we don't acquire the state mutex interruptible, and we don't
|
||||
* want this context to be rescheduled on release.
|
||||
*/
|
||||
mutex_lock(&ctx->state_mutex);
|
||||
if (ctx->state != SPU_STATE_SAVED) {
|
||||
set_bit(SPU_SCHED_WAS_ACTIVE, &ctx->sched_flags);
|
||||
if (ctx->state != SPU_STATE_SAVED)
|
||||
spu_deactivate(ctx);
|
||||
}
|
||||
|
||||
mm = ctx->owner;
|
||||
ctx->owner = NULL;
|
||||
|
|
|
@ -366,6 +366,13 @@ static unsigned long spufs_ps_nopfn(struct vm_area_struct *vma,
|
|||
if (offset >= ps_size)
|
||||
return NOPFN_SIGBUS;
|
||||
|
||||
/*
|
||||
* Because we release the mmap_sem, the context may be destroyed while
|
||||
* we're in spu_wait. Grab an extra reference so it isn't destroyed
|
||||
* in the meantime.
|
||||
*/
|
||||
get_spu_context(ctx);
|
||||
|
||||
/*
|
||||
* We have to wait for context to be loaded before we have
|
||||
* pages to hand out to the user, but we don't want to wait
|
||||
|
@ -375,7 +382,7 @@ static unsigned long spufs_ps_nopfn(struct vm_area_struct *vma,
|
|||
* hanged.
|
||||
*/
|
||||
if (spu_acquire(ctx))
|
||||
return NOPFN_REFAULT;
|
||||
goto refault;
|
||||
|
||||
if (ctx->state == SPU_STATE_SAVED) {
|
||||
up_read(¤t->mm->mmap_sem);
|
||||
|
@ -391,6 +398,9 @@ static unsigned long spufs_ps_nopfn(struct vm_area_struct *vma,
|
|||
|
||||
if (!ret)
|
||||
spu_release(ctx);
|
||||
|
||||
refault:
|
||||
put_spu_context(ctx);
|
||||
return NOPFN_REFAULT;
|
||||
}
|
||||
|
||||
|
|
|
@ -246,7 +246,7 @@ static void spu_bind_context(struct spu *spu, struct spu_context *ctx)
|
|||
spu_switch_notify(spu, ctx);
|
||||
ctx->state = SPU_STATE_RUNNABLE;
|
||||
|
||||
spuctx_switch_state(ctx, SPU_UTIL_IDLE_LOADED);
|
||||
spuctx_switch_state(ctx, SPU_UTIL_USER);
|
||||
}
|
||||
|
||||
/*
|
||||
|
|
|
@ -58,12 +58,12 @@ static int sputrace_sprint(char *tbuf, int n)
|
|||
ktime_to_timespec(ktime_sub(t->tstamp, sputrace_start));
|
||||
|
||||
return snprintf(tbuf, n,
|
||||
"[%lu.%09lu] %d: %s (thread = %d, spu = %d)\n",
|
||||
"[%lu.%09lu] %d: %s (ctxthread = %d, spu = %d)\n",
|
||||
(unsigned long) tv.tv_sec,
|
||||
(unsigned long) tv.tv_nsec,
|
||||
t->owner_tid,
|
||||
t->name,
|
||||
t->curr_tid,
|
||||
t->name,
|
||||
t->owner_tid,
|
||||
t->number);
|
||||
}
|
||||
|
||||
|
@ -188,6 +188,7 @@ struct spu_probe spu_probes[] = {
|
|||
{ "spufs_ps_nopfn__insert", "%p %p", spu_context_event },
|
||||
{ "spu_acquire_saved__enter", "%p", spu_context_nospu_event },
|
||||
{ "destroy_spu_context__enter", "%p", spu_context_nospu_event },
|
||||
{ "spufs_stop_callback__enter", "%p %p", spu_context_event },
|
||||
};
|
||||
|
||||
static int __init sputrace_init(void)
|
||||
|
|
|
@ -34,6 +34,7 @@
|
|||
|
||||
#include <linux/module.h>
|
||||
#include <linux/errno.h>
|
||||
#include <linux/hardirq.h>
|
||||
#include <linux/sched.h>
|
||||
#include <linux/kernel.h>
|
||||
#include <linux/mm.h>
|
||||
|
@ -117,6 +118,8 @@ static inline void disable_interrupts(struct spu_state *csa, struct spu *spu)
|
|||
* Write INT_MASK_class1 with value of 0.
|
||||
* Save INT_Mask_class2 in CSA.
|
||||
* Write INT_MASK_class2 with value of 0.
|
||||
* Synchronize all three interrupts to be sure
|
||||
* we no longer execute a handler on another CPU.
|
||||
*/
|
||||
spin_lock_irq(&spu->register_lock);
|
||||
if (csa) {
|
||||
|
@ -129,6 +132,9 @@ static inline void disable_interrupts(struct spu_state *csa, struct spu *spu)
|
|||
spu_int_mask_set(spu, 2, 0ul);
|
||||
eieio();
|
||||
spin_unlock_irq(&spu->register_lock);
|
||||
synchronize_irq(spu->irqs[0]);
|
||||
synchronize_irq(spu->irqs[1]);
|
||||
synchronize_irq(spu->irqs[2]);
|
||||
}
|
||||
|
||||
static inline void set_watchdog_timer(struct spu_state *csa, struct spu *spu)
|
||||
|
|
Загрузка…
Ссылка в новой задаче