123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350 |
- /* hw_ops.c - query/set operations on active SPU context.
- *
- * Copyright (C) IBM 2005
- * Author: Mark Nutter <mnutter@us.ibm.com>
- *
- * This program is free software; you can redistribute it and/or modify
- * it under the terms of the GNU General Public License as published by
- * the Free Software Foundation; either version 2, or (at your option)
- * any later version.
- *
- * This program is distributed in the hope that it will be useful,
- * but WITHOUT ANY WARRANTY; without even the implied warranty of
- * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
- * GNU General Public License for more details.
- *
- * You should have received a copy of the GNU General Public License
- * along with this program; if not, write to the Free Software
- * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
- */
- #include <linux/errno.h>
- #include <linux/sched.h>
- #include <linux/kernel.h>
- #include <linux/mm.h>
- #include <linux/poll.h>
- #include <linux/smp.h>
- #include <linux/stddef.h>
- #include <linux/unistd.h>
- #include <asm/io.h>
- #include <asm/spu.h>
- #include <asm/spu_priv1.h>
- #include <asm/spu_csa.h>
- #include <asm/mmu_context.h>
- #include "spufs.h"
- static int spu_hw_mbox_read(struct spu_context *ctx, u32 * data)
- {
- struct spu *spu = ctx->spu;
- struct spu_problem __iomem *prob = spu->problem;
- u32 mbox_stat;
- int ret = 0;
- spin_lock_irq(&spu->register_lock);
- mbox_stat = in_be32(&prob->mb_stat_R);
- if (mbox_stat & 0x0000ff) {
- *data = in_be32(&prob->pu_mb_R);
- ret = 4;
- }
- spin_unlock_irq(&spu->register_lock);
- return ret;
- }
- static u32 spu_hw_mbox_stat_read(struct spu_context *ctx)
- {
- return in_be32(&ctx->spu->problem->mb_stat_R);
- }
- static unsigned int spu_hw_mbox_stat_poll(struct spu_context *ctx,
- unsigned int events)
- {
- struct spu *spu = ctx->spu;
- int ret = 0;
- u32 stat;
- spin_lock_irq(&spu->register_lock);
- stat = in_be32(&spu->problem->mb_stat_R);
- /* if the requested event is there, return the poll
- mask, otherwise enable the interrupt to get notified,
- but first mark any pending interrupts as done so
- we don't get woken up unnecessarily */
- if (events & (POLLIN | POLLRDNORM)) {
- if (stat & 0xff0000)
- ret |= POLLIN | POLLRDNORM;
- else {
- spu_int_stat_clear(spu, 2, CLASS2_MAILBOX_INTR);
- spu_int_mask_or(spu, 2, CLASS2_ENABLE_MAILBOX_INTR);
- }
- }
- if (events & (POLLOUT | POLLWRNORM)) {
- if (stat & 0x00ff00)
- ret = POLLOUT | POLLWRNORM;
- else {
- spu_int_stat_clear(spu, 2,
- CLASS2_MAILBOX_THRESHOLD_INTR);
- spu_int_mask_or(spu, 2,
- CLASS2_ENABLE_MAILBOX_THRESHOLD_INTR);
- }
- }
- spin_unlock_irq(&spu->register_lock);
- return ret;
- }
- static int spu_hw_ibox_read(struct spu_context *ctx, u32 * data)
- {
- struct spu *spu = ctx->spu;
- struct spu_problem __iomem *prob = spu->problem;
- struct spu_priv2 __iomem *priv2 = spu->priv2;
- int ret;
- spin_lock_irq(&spu->register_lock);
- if (in_be32(&prob->mb_stat_R) & 0xff0000) {
- /* read the first available word */
- *data = in_be64(&priv2->puint_mb_R);
- ret = 4;
- } else {
- /* make sure we get woken up by the interrupt */
- spu_int_mask_or(spu, 2, CLASS2_ENABLE_MAILBOX_INTR);
- ret = 0;
- }
- spin_unlock_irq(&spu->register_lock);
- return ret;
- }
- static int spu_hw_wbox_write(struct spu_context *ctx, u32 data)
- {
- struct spu *spu = ctx->spu;
- struct spu_problem __iomem *prob = spu->problem;
- int ret;
- spin_lock_irq(&spu->register_lock);
- if (in_be32(&prob->mb_stat_R) & 0x00ff00) {
- /* we have space to write wbox_data to */
- out_be32(&prob->spu_mb_W, data);
- ret = 4;
- } else {
- /* make sure we get woken up by the interrupt when space
- becomes available */
- spu_int_mask_or(spu, 2, CLASS2_ENABLE_MAILBOX_THRESHOLD_INTR);
- ret = 0;
- }
- spin_unlock_irq(&spu->register_lock);
- return ret;
- }
- static void spu_hw_signal1_write(struct spu_context *ctx, u32 data)
- {
- out_be32(&ctx->spu->problem->signal_notify1, data);
- }
- static void spu_hw_signal2_write(struct spu_context *ctx, u32 data)
- {
- out_be32(&ctx->spu->problem->signal_notify2, data);
- }
- static void spu_hw_signal1_type_set(struct spu_context *ctx, u64 val)
- {
- struct spu *spu = ctx->spu;
- struct spu_priv2 __iomem *priv2 = spu->priv2;
- u64 tmp;
- spin_lock_irq(&spu->register_lock);
- tmp = in_be64(&priv2->spu_cfg_RW);
- if (val)
- tmp |= 1;
- else
- tmp &= ~1;
- out_be64(&priv2->spu_cfg_RW, tmp);
- spin_unlock_irq(&spu->register_lock);
- }
- static u64 spu_hw_signal1_type_get(struct spu_context *ctx)
- {
- return ((in_be64(&ctx->spu->priv2->spu_cfg_RW) & 1) != 0);
- }
- static void spu_hw_signal2_type_set(struct spu_context *ctx, u64 val)
- {
- struct spu *spu = ctx->spu;
- struct spu_priv2 __iomem *priv2 = spu->priv2;
- u64 tmp;
- spin_lock_irq(&spu->register_lock);
- tmp = in_be64(&priv2->spu_cfg_RW);
- if (val)
- tmp |= 2;
- else
- tmp &= ~2;
- out_be64(&priv2->spu_cfg_RW, tmp);
- spin_unlock_irq(&spu->register_lock);
- }
- static u64 spu_hw_signal2_type_get(struct spu_context *ctx)
- {
- return ((in_be64(&ctx->spu->priv2->spu_cfg_RW) & 2) != 0);
- }
- static u32 spu_hw_npc_read(struct spu_context *ctx)
- {
- return in_be32(&ctx->spu->problem->spu_npc_RW);
- }
- static void spu_hw_npc_write(struct spu_context *ctx, u32 val)
- {
- out_be32(&ctx->spu->problem->spu_npc_RW, val);
- }
- static u32 spu_hw_status_read(struct spu_context *ctx)
- {
- return in_be32(&ctx->spu->problem->spu_status_R);
- }
- static char *spu_hw_get_ls(struct spu_context *ctx)
- {
- return ctx->spu->local_store;
- }
- static void spu_hw_privcntl_write(struct spu_context *ctx, u64 val)
- {
- out_be64(&ctx->spu->priv2->spu_privcntl_RW, val);
- }
- static u32 spu_hw_runcntl_read(struct spu_context *ctx)
- {
- return in_be32(&ctx->spu->problem->spu_runcntl_RW);
- }
- static void spu_hw_runcntl_write(struct spu_context *ctx, u32 val)
- {
- spin_lock_irq(&ctx->spu->register_lock);
- if (val & SPU_RUNCNTL_ISOLATE)
- spu_hw_privcntl_write(ctx,
- SPU_PRIVCNT_LOAD_REQUEST_ENABLE_MASK);
- out_be32(&ctx->spu->problem->spu_runcntl_RW, val);
- spin_unlock_irq(&ctx->spu->register_lock);
- }
- static void spu_hw_runcntl_stop(struct spu_context *ctx)
- {
- spin_lock_irq(&ctx->spu->register_lock);
- out_be32(&ctx->spu->problem->spu_runcntl_RW, SPU_RUNCNTL_STOP);
- while (in_be32(&ctx->spu->problem->spu_status_R) & SPU_STATUS_RUNNING)
- cpu_relax();
- spin_unlock_irq(&ctx->spu->register_lock);
- }
- static void spu_hw_master_start(struct spu_context *ctx)
- {
- struct spu *spu = ctx->spu;
- u64 sr1;
- spin_lock_irq(&spu->register_lock);
- sr1 = spu_mfc_sr1_get(spu) | MFC_STATE1_MASTER_RUN_CONTROL_MASK;
- spu_mfc_sr1_set(spu, sr1);
- spin_unlock_irq(&spu->register_lock);
- }
- static void spu_hw_master_stop(struct spu_context *ctx)
- {
- struct spu *spu = ctx->spu;
- u64 sr1;
- spin_lock_irq(&spu->register_lock);
- sr1 = spu_mfc_sr1_get(spu) & ~MFC_STATE1_MASTER_RUN_CONTROL_MASK;
- spu_mfc_sr1_set(spu, sr1);
- spin_unlock_irq(&spu->register_lock);
- }
- static int spu_hw_set_mfc_query(struct spu_context * ctx, u32 mask, u32 mode)
- {
- struct spu_problem __iomem *prob = ctx->spu->problem;
- int ret;
- spin_lock_irq(&ctx->spu->register_lock);
- ret = -EAGAIN;
- if (in_be32(&prob->dma_querytype_RW))
- goto out;
- ret = 0;
- out_be32(&prob->dma_querymask_RW, mask);
- out_be32(&prob->dma_querytype_RW, mode);
- out:
- spin_unlock_irq(&ctx->spu->register_lock);
- return ret;
- }
- static u32 spu_hw_read_mfc_tagstatus(struct spu_context * ctx)
- {
- return in_be32(&ctx->spu->problem->dma_tagstatus_R);
- }
- static u32 spu_hw_get_mfc_free_elements(struct spu_context *ctx)
- {
- return in_be32(&ctx->spu->problem->dma_qstatus_R);
- }
- static int spu_hw_send_mfc_command(struct spu_context *ctx,
- struct mfc_dma_command *cmd)
- {
- u32 status;
- struct spu_problem __iomem *prob = ctx->spu->problem;
- spin_lock_irq(&ctx->spu->register_lock);
- out_be32(&prob->mfc_lsa_W, cmd->lsa);
- out_be64(&prob->mfc_ea_W, cmd->ea);
- out_be32(&prob->mfc_union_W.by32.mfc_size_tag32,
- cmd->size << 16 | cmd->tag);
- out_be32(&prob->mfc_union_W.by32.mfc_class_cmd32,
- cmd->class << 16 | cmd->cmd);
- status = in_be32(&prob->mfc_union_W.by32.mfc_class_cmd32);
- spin_unlock_irq(&ctx->spu->register_lock);
- switch (status & 0xffff) {
- case 0:
- return 0;
- case 2:
- return -EAGAIN;
- default:
- return -EINVAL;
- }
- }
- static void spu_hw_restart_dma(struct spu_context *ctx)
- {
- struct spu_priv2 __iomem *priv2 = ctx->spu->priv2;
- if (!test_bit(SPU_CONTEXT_SWITCH_PENDING, &ctx->spu->flags))
- out_be64(&priv2->mfc_control_RW, MFC_CNTL_RESTART_DMA_COMMAND);
- }
- struct spu_context_ops spu_hw_ops = {
- .mbox_read = spu_hw_mbox_read,
- .mbox_stat_read = spu_hw_mbox_stat_read,
- .mbox_stat_poll = spu_hw_mbox_stat_poll,
- .ibox_read = spu_hw_ibox_read,
- .wbox_write = spu_hw_wbox_write,
- .signal1_write = spu_hw_signal1_write,
- .signal2_write = spu_hw_signal2_write,
- .signal1_type_set = spu_hw_signal1_type_set,
- .signal1_type_get = spu_hw_signal1_type_get,
- .signal2_type_set = spu_hw_signal2_type_set,
- .signal2_type_get = spu_hw_signal2_type_get,
- .npc_read = spu_hw_npc_read,
- .npc_write = spu_hw_npc_write,
- .status_read = spu_hw_status_read,
- .get_ls = spu_hw_get_ls,
- .privcntl_write = spu_hw_privcntl_write,
- .runcntl_read = spu_hw_runcntl_read,
- .runcntl_write = spu_hw_runcntl_write,
- .runcntl_stop = spu_hw_runcntl_stop,
- .master_start = spu_hw_master_start,
- .master_stop = spu_hw_master_stop,
- .set_mfc_query = spu_hw_set_mfc_query,
- .read_mfc_tagstatus = spu_hw_read_mfc_tagstatus,
- .get_mfc_free_elements = spu_hw_get_mfc_free_elements,
- .send_mfc_command = spu_hw_send_mfc_command,
- .restart_dma = spu_hw_restart_dma,
- };
|