Commit fe76fe49 authored by Ben Skeggs's avatar Ben Skeggs

drm/nouveau/mc: implement intr handling on top of nvkm_intr

- new-style handlers can now be used here too
- decent clean-up
Signed-off-by: default avatarBen Skeggs <bskeggs@redhat.com>
Reviewed-by: default avatarLyude Paul <lyude@redhat.com>
parent e5f92c87
...@@ -6,15 +6,14 @@ ...@@ -6,15 +6,14 @@
struct nvkm_mc { struct nvkm_mc {
const struct nvkm_mc_func *func; const struct nvkm_mc_func *func;
struct nvkm_subdev subdev; struct nvkm_subdev subdev;
struct nvkm_intr intr;
}; };
void nvkm_mc_enable(struct nvkm_device *, enum nvkm_subdev_type, int); void nvkm_mc_enable(struct nvkm_device *, enum nvkm_subdev_type, int);
void nvkm_mc_disable(struct nvkm_device *, enum nvkm_subdev_type, int); void nvkm_mc_disable(struct nvkm_device *, enum nvkm_subdev_type, int);
bool nvkm_mc_enabled(struct nvkm_device *, enum nvkm_subdev_type, int); bool nvkm_mc_enabled(struct nvkm_device *, enum nvkm_subdev_type, int);
void nvkm_mc_reset(struct nvkm_device *, enum nvkm_subdev_type, int); void nvkm_mc_reset(struct nvkm_device *, enum nvkm_subdev_type, int);
void nvkm_mc_intr(struct nvkm_device *, bool *handled);
void nvkm_mc_intr_unarm(struct nvkm_device *);
void nvkm_mc_intr_rearm(struct nvkm_device *);
void nvkm_mc_intr_mask(struct nvkm_device *, enum nvkm_subdev_type, int, bool enable); void nvkm_mc_intr_mask(struct nvkm_device *, enum nvkm_subdev_type, int, bool enable);
void nvkm_mc_unk260(struct nvkm_device *, u32 data); void nvkm_mc_unk260(struct nvkm_device *, u32 data);
...@@ -31,6 +30,5 @@ int gk104_mc_new(struct nvkm_device *, enum nvkm_subdev_type, int inst, struct n ...@@ -31,6 +30,5 @@ int gk104_mc_new(struct nvkm_device *, enum nvkm_subdev_type, int inst, struct n
int gk20a_mc_new(struct nvkm_device *, enum nvkm_subdev_type, int inst, struct nvkm_mc **); int gk20a_mc_new(struct nvkm_device *, enum nvkm_subdev_type, int inst, struct nvkm_mc **);
int gp100_mc_new(struct nvkm_device *, enum nvkm_subdev_type, int inst, struct nvkm_mc **); int gp100_mc_new(struct nvkm_device *, enum nvkm_subdev_type, int inst, struct nvkm_mc **);
int gp10b_mc_new(struct nvkm_device *, enum nvkm_subdev_type, int inst, struct nvkm_mc **); int gp10b_mc_new(struct nvkm_device *, enum nvkm_subdev_type, int inst, struct nvkm_mc **);
int tu102_mc_new(struct nvkm_device *, enum nvkm_subdev_type, int inst, struct nvkm_mc **);
int ga100_mc_new(struct nvkm_device *, enum nvkm_subdev_type, int inst, struct nvkm_mc **); int ga100_mc_new(struct nvkm_device *, enum nvkm_subdev_type, int inst, struct nvkm_mc **);
#endif #endif
...@@ -25,8 +25,6 @@ ...@@ -25,8 +25,6 @@
#include <subdev/pci.h> #include <subdev/pci.h>
#include <subdev/top.h> #include <subdev/top.h>
#include <subdev/mc.h>
static int static int
nvkm_intr_xlat(struct nvkm_subdev *subdev, struct nvkm_intr *intr, nvkm_intr_xlat(struct nvkm_subdev *subdev, struct nvkm_intr *intr,
enum nvkm_intr_type type, int *leaf, u32 *mask) enum nvkm_intr_type type, int *leaf, u32 *mask)
...@@ -151,7 +149,6 @@ nvkm_intr_rearm_locked(struct nvkm_device *device) ...@@ -151,7 +149,6 @@ nvkm_intr_rearm_locked(struct nvkm_device *device)
list_for_each_entry(intr, &device->intr.intr, head) list_for_each_entry(intr, &device->intr.intr, head)
intr->func->rearm(intr); intr->func->rearm(intr);
nvkm_mc_intr_rearm(device);
} }
static void static void
...@@ -161,7 +158,6 @@ nvkm_intr_unarm_locked(struct nvkm_device *device) ...@@ -161,7 +158,6 @@ nvkm_intr_unarm_locked(struct nvkm_device *device)
list_for_each_entry(intr, &device->intr.intr, head) list_for_each_entry(intr, &device->intr.intr, head)
intr->func->unarm(intr); intr->func->unarm(intr);
nvkm_mc_intr_unarm(device);
} }
static irqreturn_t static irqreturn_t
...@@ -171,7 +167,7 @@ nvkm_intr(int irq, void *arg) ...@@ -171,7 +167,7 @@ nvkm_intr(int irq, void *arg)
struct nvkm_intr *intr; struct nvkm_intr *intr;
struct nvkm_inth *inth; struct nvkm_inth *inth;
irqreturn_t ret = IRQ_NONE; irqreturn_t ret = IRQ_NONE;
bool pending = false, handled; bool pending = false;
int prio, leaf; int prio, leaf;
/* Disable all top-level interrupt sources, and re-arm MSI interrupts. */ /* Disable all top-level interrupt sources, and re-arm MSI interrupts. */
...@@ -188,10 +184,6 @@ nvkm_intr(int irq, void *arg) ...@@ -188,10 +184,6 @@ nvkm_intr(int irq, void *arg)
pending = true; pending = true;
} }
nvkm_mc_intr(device, &handled);
if (handled)
ret = IRQ_HANDLED;
if (!pending) if (!pending)
goto done; goto done;
......
...@@ -2412,7 +2412,7 @@ nv162_chipset = { ...@@ -2412,7 +2412,7 @@ nv162_chipset = {
.i2c = { 0x00000001, gm200_i2c_new }, .i2c = { 0x00000001, gm200_i2c_new },
.imem = { 0x00000001, nv50_instmem_new }, .imem = { 0x00000001, nv50_instmem_new },
.ltc = { 0x00000001, gp102_ltc_new }, .ltc = { 0x00000001, gp102_ltc_new },
.mc = { 0x00000001, tu102_mc_new }, .mc = { 0x00000001, gp100_mc_new },
.mmu = { 0x00000001, tu102_mmu_new }, .mmu = { 0x00000001, tu102_mmu_new },
.pci = { 0x00000001, gp100_pci_new }, .pci = { 0x00000001, gp100_pci_new },
.pmu = { 0x00000001, gp102_pmu_new }, .pmu = { 0x00000001, gp102_pmu_new },
...@@ -2447,7 +2447,7 @@ nv164_chipset = { ...@@ -2447,7 +2447,7 @@ nv164_chipset = {
.i2c = { 0x00000001, gm200_i2c_new }, .i2c = { 0x00000001, gm200_i2c_new },
.imem = { 0x00000001, nv50_instmem_new }, .imem = { 0x00000001, nv50_instmem_new },
.ltc = { 0x00000001, gp102_ltc_new }, .ltc = { 0x00000001, gp102_ltc_new },
.mc = { 0x00000001, tu102_mc_new }, .mc = { 0x00000001, gp100_mc_new },
.mmu = { 0x00000001, tu102_mmu_new }, .mmu = { 0x00000001, tu102_mmu_new },
.pci = { 0x00000001, gp100_pci_new }, .pci = { 0x00000001, gp100_pci_new },
.pmu = { 0x00000001, gp102_pmu_new }, .pmu = { 0x00000001, gp102_pmu_new },
...@@ -2482,7 +2482,7 @@ nv166_chipset = { ...@@ -2482,7 +2482,7 @@ nv166_chipset = {
.i2c = { 0x00000001, gm200_i2c_new }, .i2c = { 0x00000001, gm200_i2c_new },
.imem = { 0x00000001, nv50_instmem_new }, .imem = { 0x00000001, nv50_instmem_new },
.ltc = { 0x00000001, gp102_ltc_new }, .ltc = { 0x00000001, gp102_ltc_new },
.mc = { 0x00000001, tu102_mc_new }, .mc = { 0x00000001, gp100_mc_new },
.mmu = { 0x00000001, tu102_mmu_new }, .mmu = { 0x00000001, tu102_mmu_new },
.pci = { 0x00000001, gp100_pci_new }, .pci = { 0x00000001, gp100_pci_new },
.pmu = { 0x00000001, gp102_pmu_new }, .pmu = { 0x00000001, gp102_pmu_new },
...@@ -2517,7 +2517,7 @@ nv167_chipset = { ...@@ -2517,7 +2517,7 @@ nv167_chipset = {
.i2c = { 0x00000001, gm200_i2c_new }, .i2c = { 0x00000001, gm200_i2c_new },
.imem = { 0x00000001, nv50_instmem_new }, .imem = { 0x00000001, nv50_instmem_new },
.ltc = { 0x00000001, gp102_ltc_new }, .ltc = { 0x00000001, gp102_ltc_new },
.mc = { 0x00000001, tu102_mc_new }, .mc = { 0x00000001, gp100_mc_new },
.mmu = { 0x00000001, tu102_mmu_new }, .mmu = { 0x00000001, tu102_mmu_new },
.pci = { 0x00000001, gp100_pci_new }, .pci = { 0x00000001, gp100_pci_new },
.pmu = { 0x00000001, gp102_pmu_new }, .pmu = { 0x00000001, gp102_pmu_new },
...@@ -2552,7 +2552,7 @@ nv168_chipset = { ...@@ -2552,7 +2552,7 @@ nv168_chipset = {
.i2c = { 0x00000001, gm200_i2c_new }, .i2c = { 0x00000001, gm200_i2c_new },
.imem = { 0x00000001, nv50_instmem_new }, .imem = { 0x00000001, nv50_instmem_new },
.ltc = { 0x00000001, gp102_ltc_new }, .ltc = { 0x00000001, gp102_ltc_new },
.mc = { 0x00000001, tu102_mc_new }, .mc = { 0x00000001, gp100_mc_new },
.mmu = { 0x00000001, tu102_mmu_new }, .mmu = { 0x00000001, tu102_mmu_new },
.pci = { 0x00000001, gp100_pci_new }, .pci = { 0x00000001, gp100_pci_new },
.pmu = { 0x00000001, gp102_pmu_new }, .pmu = { 0x00000001, gp102_pmu_new },
......
...@@ -13,5 +13,4 @@ nvkm-y += nvkm/subdev/mc/gk104.o ...@@ -13,5 +13,4 @@ nvkm-y += nvkm/subdev/mc/gk104.o
nvkm-y += nvkm/subdev/mc/gk20a.o nvkm-y += nvkm/subdev/mc/gk20a.o
nvkm-y += nvkm/subdev/mc/gp100.o nvkm-y += nvkm/subdev/mc/gp100.o
nvkm-y += nvkm/subdev/mc/gp10b.o nvkm-y += nvkm/subdev/mc/gp10b.o
nvkm-y += nvkm/subdev/mc/tu102.o
nvkm-y += nvkm/subdev/mc/ga100.o nvkm-y += nvkm/subdev/mc/ga100.o
...@@ -37,84 +37,14 @@ nvkm_mc_unk260(struct nvkm_device *device, u32 data) ...@@ -37,84 +37,14 @@ nvkm_mc_unk260(struct nvkm_device *device, u32 data)
void void
nvkm_mc_intr_mask(struct nvkm_device *device, enum nvkm_subdev_type type, int inst, bool en) nvkm_mc_intr_mask(struct nvkm_device *device, enum nvkm_subdev_type type, int inst, bool en)
{ {
struct nvkm_mc *mc = device->mc; struct nvkm_subdev *subdev = nvkm_device_subdev(device, type, inst);
const struct nvkm_mc_map *map;
if (likely(mc) && mc->func->intr_mask) {
u32 mask = nvkm_top_intr_mask(device, type, inst);
for (map = mc->func->intr; !mask && map->stat; map++) {
if (map->type == type && map->inst == inst)
mask = map->stat;
}
mc->func->intr_mask(mc, mask, en ? mask : 0);
}
}
void
nvkm_mc_intr_unarm(struct nvkm_device *device)
{
struct nvkm_mc *mc = device->mc;
if (likely(mc))
mc->func->intr_unarm(mc);
}
void
nvkm_mc_intr_rearm(struct nvkm_device *device)
{
struct nvkm_mc *mc = device->mc;
if (likely(mc))
mc->func->intr_rearm(mc);
}
static u32
nvkm_mc_intr_stat(struct nvkm_mc *mc)
{
u32 intr = mc->func->intr_stat(mc);
if (WARN_ON_ONCE(intr == 0xffffffff))
intr = 0; /* likely fallen off the bus */
return intr;
}
void
nvkm_mc_intr(struct nvkm_device *device, bool *handled)
{
struct nvkm_mc *mc = device->mc;
struct nvkm_top *top = device->top;
struct nvkm_top_device *tdev;
struct nvkm_subdev *subdev;
const struct nvkm_mc_map *map;
u32 stat, intr;
if (unlikely(!mc))
return;
stat = intr = nvkm_mc_intr_stat(mc);
if (top) { if (subdev) {
list_for_each_entry(tdev, &top->device, head) { if (en)
if (tdev->intr >= 0 && (stat & BIT(tdev->intr))) { nvkm_intr_allow(subdev, NVKM_INTR_SUBDEV);
subdev = nvkm_device_subdev(device, tdev->type, tdev->inst); else
if (subdev) { nvkm_intr_block(subdev, NVKM_INTR_SUBDEV);
nvkm_subdev_intr(subdev);
stat &= ~BIT(tdev->intr);
if (!stat)
break;
}
}
}
}
for (map = mc->func->intr; map->stat; map++) {
if (intr & map->stat) {
subdev = nvkm_device_subdev(device, map->type, map->inst);
if (subdev)
nvkm_subdev_intr(subdev);
stat &= ~map->stat;
}
} }
if (stat)
nvkm_error(&mc->subdev, "intr %08x\n", stat);
*handled = intr != 0;
} }
static u32 static u32
...@@ -177,20 +107,12 @@ nvkm_mc_enabled(struct nvkm_device *device, enum nvkm_subdev_type type, int inst ...@@ -177,20 +107,12 @@ nvkm_mc_enabled(struct nvkm_device *device, enum nvkm_subdev_type type, int inst
} }
static int
nvkm_mc_fini(struct nvkm_subdev *subdev, bool suspend)
{
nvkm_mc_intr_unarm(subdev->device);
return 0;
}
static int static int
nvkm_mc_init(struct nvkm_subdev *subdev) nvkm_mc_init(struct nvkm_subdev *subdev)
{ {
struct nvkm_mc *mc = nvkm_mc(subdev); struct nvkm_mc *mc = nvkm_mc(subdev);
if (mc->func->init) if (mc->func->init)
mc->func->init(mc); mc->func->init(mc);
nvkm_mc_intr_rearm(subdev->device);
return 0; return 0;
} }
...@@ -204,24 +126,27 @@ static const struct nvkm_subdev_func ...@@ -204,24 +126,27 @@ static const struct nvkm_subdev_func
nvkm_mc = { nvkm_mc = {
.dtor = nvkm_mc_dtor, .dtor = nvkm_mc_dtor,
.init = nvkm_mc_init, .init = nvkm_mc_init,
.fini = nvkm_mc_fini,
}; };
void
nvkm_mc_ctor(const struct nvkm_mc_func *func, struct nvkm_device *device,
enum nvkm_subdev_type type, int inst, struct nvkm_mc *mc)
{
nvkm_subdev_ctor(&nvkm_mc, device, type, inst, &mc->subdev);
mc->func = func;
}
int int
nvkm_mc_new_(const struct nvkm_mc_func *func, struct nvkm_device *device, nvkm_mc_new_(const struct nvkm_mc_func *func, struct nvkm_device *device,
enum nvkm_subdev_type type, int inst, struct nvkm_mc **pmc) enum nvkm_subdev_type type, int inst, struct nvkm_mc **pmc)
{ {
struct nvkm_mc *mc; struct nvkm_mc *mc;
int ret;
if (!(mc = *pmc = kzalloc(sizeof(*mc), GFP_KERNEL))) if (!(mc = *pmc = kzalloc(sizeof(*mc), GFP_KERNEL)))
return -ENOMEM; return -ENOMEM;
nvkm_mc_ctor(func, device, type, inst, *pmc);
nvkm_subdev_ctor(&nvkm_mc, device, type, inst, &mc->subdev);
mc->func = func;
if (mc->func->intr) {
ret = nvkm_intr_add(mc->func->intr, mc->func->intrs, &mc->subdev,
mc->func->intr_nonstall ? 2 : 1, &mc->intr);
if (ret)
return ret;
}
return 0; return 0;
} }
...@@ -34,30 +34,28 @@ g84_mc_reset[] = { ...@@ -34,30 +34,28 @@ g84_mc_reset[] = {
{} {}
}; };
static const struct nvkm_mc_map static const struct nvkm_intr_data
g84_mc_intr[] = { g84_mc_intrs[] = {
{ 0x04000000, NVKM_ENGINE_DISP }, { NVKM_ENGINE_DISP , 0, 0, 0x04000000, true },
{ 0x00020000, NVKM_ENGINE_VP }, { NVKM_ENGINE_VP , 0, 0, 0x00020000, true },
{ 0x00008000, NVKM_ENGINE_BSP }, { NVKM_ENGINE_BSP , 0, 0, 0x00008000, true },
{ 0x00004000, NVKM_ENGINE_CIPHER }, { NVKM_ENGINE_CIPHER, 0, 0, 0x00004000, true },
{ 0x00001000, NVKM_ENGINE_GR }, { NVKM_ENGINE_GR , 0, 0, 0x00001000, true },
{ 0x00000100, NVKM_ENGINE_FIFO }, { NVKM_ENGINE_FIFO , 0, 0, 0x00000100, true },
{ 0x00000001, NVKM_ENGINE_MPEG }, { NVKM_ENGINE_MPEG , 0, 0, 0x00000001, true },
{ 0x0002d101, NVKM_SUBDEV_FB }, { NVKM_SUBDEV_FB , 0, 0, 0x0002d101, true },
{ 0x10000000, NVKM_SUBDEV_BUS }, { NVKM_SUBDEV_BUS , 0, 0, 0x10000000, true },
{ 0x00200000, NVKM_SUBDEV_GPIO }, { NVKM_SUBDEV_GPIO , 0, 0, 0x00200000, true },
{ 0x00200000, NVKM_SUBDEV_I2C }, { NVKM_SUBDEV_I2C , 0, 0, 0x00200000, true },
{ 0x00100000, NVKM_SUBDEV_TIMER }, { NVKM_SUBDEV_TIMER , 0, 0, 0x00100000, true },
{}, {},
}; };
static const struct nvkm_mc_func static const struct nvkm_mc_func
g84_mc = { g84_mc = {
.init = nv50_mc_init, .init = nv50_mc_init,
.intr = g84_mc_intr, .intr = &nv04_mc_intr,
.intr_unarm = nv04_mc_intr_unarm, .intrs = g84_mc_intrs,
.intr_rearm = nv04_mc_intr_rearm,
.intr_stat = nv04_mc_intr_stat,
.reset = g84_mc_reset, .reset = g84_mc_reset,
}; };
......
...@@ -34,30 +34,28 @@ g98_mc_reset[] = { ...@@ -34,30 +34,28 @@ g98_mc_reset[] = {
{} {}
}; };
static const struct nvkm_mc_map static const struct nvkm_intr_data
g98_mc_intr[] = { g98_mc_intrs[] = {
{ 0x04000000, NVKM_ENGINE_DISP }, { NVKM_ENGINE_DISP , 0, 0, 0x04000000, true },
{ 0x00020000, NVKM_ENGINE_MSPDEC }, { NVKM_ENGINE_MSPDEC, 0, 0, 0x00020000, true },
{ 0x00008000, NVKM_ENGINE_MSVLD }, { NVKM_ENGINE_MSVLD , 0, 0, 0x00008000, true },
{ 0x00004000, NVKM_ENGINE_SEC }, { NVKM_ENGINE_SEC , 0, 0, 0x00004000, true },
{ 0x00001000, NVKM_ENGINE_GR }, { NVKM_ENGINE_GR , 0, 0, 0x00001000, true },
{ 0x00000100, NVKM_ENGINE_FIFO }, { NVKM_ENGINE_FIFO , 0, 0, 0x00000100, true },
{ 0x00000001, NVKM_ENGINE_MSPPP }, { NVKM_ENGINE_MSPPP , 0, 0, 0x00000001, true },
{ 0x0002d101, NVKM_SUBDEV_FB }, { NVKM_SUBDEV_FB , 0, 0, 0x0002d101, true },
{ 0x10000000, NVKM_SUBDEV_BUS }, { NVKM_SUBDEV_BUS , 0, 0, 0x10000000, true },
{ 0x00200000, NVKM_SUBDEV_GPIO }, { NVKM_SUBDEV_GPIO , 0, 0, 0x00200000, true },
{ 0x00200000, NVKM_SUBDEV_I2C }, { NVKM_SUBDEV_I2C , 0, 0, 0x00200000, true },
{ 0x00100000, NVKM_SUBDEV_TIMER }, { NVKM_SUBDEV_TIMER , 0, 0, 0x00100000, true },
{}, {},
}; };
static const struct nvkm_mc_func static const struct nvkm_mc_func
g98_mc = { g98_mc = {
.init = nv50_mc_init, .init = nv50_mc_init,
.intr = g98_mc_intr, .intr = &nv04_mc_intr,
.intr_unarm = nv04_mc_intr_unarm, .intrs = g98_mc_intrs,
.intr_rearm = nv04_mc_intr_rearm,
.intr_stat = nv04_mc_intr_stat,
.reset = g98_mc_reset, .reset = g98_mc_reset,
}; };
......
...@@ -21,35 +21,9 @@ ...@@ -21,35 +21,9 @@
*/ */
#include "priv.h" #include "priv.h"
static void
ga100_mc_intr_unarm(struct nvkm_mc *mc)
{
}
static void
ga100_mc_intr_rearm(struct nvkm_mc *mc)
{
}
static u32
ga100_mc_intr_stat(struct nvkm_mc *mc)
{
return 0;
}
static void
ga100_mc_init(struct nvkm_mc *mc)
{
nv50_mc_init(mc);
}
static const struct nvkm_mc_func static const struct nvkm_mc_func
ga100_mc = { ga100_mc = {
.init = ga100_mc_init, .init = nv50_mc_init,
.intr = gp100_mc_intr,
.intr_unarm = ga100_mc_intr_unarm,
.intr_rearm = ga100_mc_intr_rearm,
.intr_stat = ga100_mc_intr_stat,
.reset = gk104_mc_reset, .reset = gk104_mc_reset,
}; };
......
...@@ -36,63 +36,28 @@ gf100_mc_reset[] = { ...@@ -36,63 +36,28 @@ gf100_mc_reset[] = {
{} {}
}; };
static const struct nvkm_mc_map static const struct nvkm_intr_data
gf100_mc_intr[] = { gf100_mc_intrs[] = {
{ 0x04000000, NVKM_ENGINE_DISP }, { NVKM_ENGINE_DISP , 0, 0, 0x04000000, true },
{ 0x00020000, NVKM_ENGINE_MSPDEC }, { NVKM_ENGINE_MSPDEC , 0, 0, 0x00020000, true },
{ 0x00008000, NVKM_ENGINE_MSVLD }, { NVKM_ENGINE_MSVLD , 0, 0, 0x00008000, true },
{ 0x00001000, NVKM_ENGINE_GR }, { NVKM_ENGINE_GR , 0, 0, 0x00001000, true },
{ 0x00000100, NVKM_ENGINE_FIFO }, { NVKM_ENGINE_FIFO , 0, 0, 0x00000100, true },
{ 0x00000040, NVKM_ENGINE_CE, 1 }, { NVKM_ENGINE_CE , 1, 0, 0x00000040, true },
{ 0x00000020, NVKM_ENGINE_CE, 0 }, { NVKM_ENGINE_CE , 0, 0, 0x00000020, true },
{ 0x00000001, NVKM_ENGINE_MSPPP }, { NVKM_ENGINE_MSPPP , 0, 0, 0x00000001, true },
{ 0x40000000, NVKM_SUBDEV_PRIVRING }, { NVKM_SUBDEV_PRIVRING, 0, 0, 0x40000000, true },
{ 0x10000000, NVKM_SUBDEV_BUS }, { NVKM_SUBDEV_BUS , 0, 0, 0x10000000, true },
{ 0x08000000, NVKM_SUBDEV_FB }, { NVKM_SUBDEV_FB , 0, 0, 0x08002000, true },
{ 0x02000000, NVKM_SUBDEV_LTC }, { NVKM_SUBDEV_LTC , 0, 0, 0x02000000, true },
{ 0x01000000, NVKM_SUBDEV_PMU }, { NVKM_SUBDEV_PMU , 0, 0, 0x01000000, true },
{ 0x00200000, NVKM_SUBDEV_GPIO }, { NVKM_SUBDEV_GPIO , 0, 0, 0x00200000, true },
{ 0x00200000, NVKM_SUBDEV_I2C }, { NVKM_SUBDEV_I2C , 0, 0, 0x00200000, true },
{ 0x00100000, NVKM_SUBDEV_TIMER }, { NVKM_SUBDEV_TIMER , 0, 0, 0x00100000, true },
{ 0x00040000, NVKM_SUBDEV_THERM }, { NVKM_SUBDEV_THERM , 0, 0, 0x00040000, true },
{ 0x00002000, NVKM_SUBDEV_FB },
{}, {},
}; };
void
gf100_mc_intr_unarm(struct nvkm_mc *mc)
{
struct nvkm_device *device = mc->subdev.device;
nvkm_wr32(device, 0x000140, 0x00000000);
nvkm_wr32(device, 0x000144, 0x00000000);
nvkm_rd32(device, 0x000140);
}
void
gf100_mc_intr_rearm(struct nvkm_mc *mc)
{
struct nvkm_device *device = mc->subdev.device;
nvkm_wr32(device, 0x000140, 0x00000001);
nvkm_wr32(device, 0x000144, 0x00000001);
}
u32
gf100_mc_intr_stat(struct nvkm_mc *mc)
{
struct nvkm_device *device = mc->subdev.device;
u32 intr0 = nvkm_rd32(device, 0x000100);
u32 intr1 = nvkm_rd32(device, 0x000104);
return intr0 | intr1;
}
void
gf100_mc_intr_mask(struct nvkm_mc *mc, u32 mask, u32 stat)
{
struct nvkm_device *device = mc->subdev.device;
nvkm_mask(device, 0x000640, mask, stat);
nvkm_mask(device, 0x000644, mask, stat);
}
void void
gf100_mc_unk260(struct nvkm_mc *mc, u32 data) gf100_mc_unk260(struct nvkm_mc *mc, u32 data)
{ {
...@@ -102,11 +67,9 @@ gf100_mc_unk260(struct nvkm_mc *mc, u32 data) ...@@ -102,11 +67,9 @@ gf100_mc_unk260(struct nvkm_mc *mc, u32 data)
static const struct nvkm_mc_func static const struct nvkm_mc_func
gf100_mc = { gf100_mc = {
.init = nv50_mc_init, .init = nv50_mc_init,
.intr = gf100_mc_intr, .intr = &gt215_mc_intr,
.intr_unarm = gf100_mc_intr_unarm, .intrs = gf100_mc_intrs,
.intr_rearm = gf100_mc_intr_rearm, .intr_nonstall = true,
.intr_mask = gf100_mc_intr_mask,
.intr_stat = gf100_mc_intr_stat,
.reset = gf100_mc_reset, .reset = gf100_mc_reset,
.unk260 = gf100_mc_unk260, .unk260 = gf100_mc_unk260,
}; };
......
...@@ -30,31 +30,29 @@ gk104_mc_reset[] = { ...@@ -30,31 +30,29 @@ gk104_mc_reset[] = {
{} {}
}; };
const struct nvkm_mc_map const struct nvkm_intr_data
gk104_mc_intr[] = { gk104_mc_intrs[] = {
{ 0x04000000, NVKM_ENGINE_DISP }, { NVKM_ENGINE_DISP , 0, 0, 0x04000000, true },
{ 0x00000100, NVKM_ENGINE_FIFO }, { NVKM_ENGINE_FIFO , 0, 0, 0x00000100, true },
{ 0x40000000, NVKM_SUBDEV_PRIVRING }, { NVKM_SUBDEV_PRIVRING, 0, 0, 0x40000000, true },
{ 0x10000000, NVKM_SUBDEV_BUS }, { NVKM_SUBDEV_BUS , 0, 0, 0x10000000, true },
{ 0x08000000, NVKM_SUBDEV_FB }, { NVKM_SUBDEV_FB , 0, 0, 0x08002000, true },
{ 0x02000000, NVKM_SUBDEV_LTC }, { NVKM_SUBDEV_LTC , 0, 0, 0x02000000, true },
{ 0x01000000, NVKM_SUBDEV_PMU }, { NVKM_SUBDEV_PMU , 0, 0, 0x01000000, true },
{ 0x00200000, NVKM_SUBDEV_GPIO }, { NVKM_SUBDEV_GPIO , 0, 0, 0x00200000, true },
{ 0x00200000, NVKM_SUBDEV_I2C }, { NVKM_SUBDEV_I2C , 0, 0, 0x00200000, true },
{ 0x00100000, NVKM_SUBDEV_TIMER }, { NVKM_SUBDEV_TIMER , 0, 0, 0x00100000, true },
{ 0x00040000, NVKM_SUBDEV_THERM }, { NVKM_SUBDEV_THERM , 0, 0, 0x00040000, true },
{ 0x00002000, NVKM_SUBDEV_FB }, { NVKM_SUBDEV_TOP , 0, 0, 0xffffffff, true },
{}, {},
}; };
static const struct nvkm_mc_func static const struct nvkm_mc_func
gk104_mc = { gk104_mc = {
.init = nv50_mc_init, .init = nv50_mc_init,
.intr = gk104_mc_intr, .intr = &gt215_mc_intr,
.intr_unarm = gf100_mc_intr_unarm, .intrs = gk104_mc_intrs,
.intr_rearm = gf100_mc_intr_rearm, .intr_nonstall = true,
.intr_mask = gf100_mc_intr_mask,
.intr_stat = gf100_mc_intr_stat,
.reset = gk104_mc_reset, .reset = gk104_mc_reset,
.unk260 = gf100_mc_unk260, .unk260 = gf100_mc_unk260,
}; };
......
...@@ -26,11 +26,9 @@ ...@@ -26,11 +26,9 @@
static const struct nvkm_mc_func static const struct nvkm_mc_func
gk20a_mc = { gk20a_mc = {
.init = nv50_mc_init, .init = nv50_mc_init,
.intr = gk104_mc_intr, .intr = &gt215_mc_intr,
.intr_unarm = gf100_mc_intr_unarm, .intrs = gk104_mc_intrs,
.intr_rearm = gf100_mc_intr_rearm, .intr_nonstall = true,
.intr_mask = gf100_mc_intr_mask,
.intr_stat = gf100_mc_intr_stat,
.reset = gk104_mc_reset, .reset = gk104_mc_reset,
}; };
......
...@@ -21,108 +21,80 @@ ...@@ -21,108 +21,80 @@
* *
* Authors: Ben Skeggs * Authors: Ben Skeggs
*/ */
#define gp100_mc(p) container_of((p), struct gp100_mc, base)
#include "priv.h" #include "priv.h"
struct gp100_mc { const struct nvkm_intr_data
struct nvkm_mc base; gp100_mc_intrs[] = {
spinlock_t lock; { NVKM_ENGINE_DISP , 0, 0, 0x04000000, true },
bool intr; { NVKM_ENGINE_FIFO , 0, 0, 0x00000100, true },
u32 mask; { NVKM_SUBDEV_FAULT , 0, 0, 0x00000200, true },
{ NVKM_SUBDEV_PRIVRING, 0, 0, 0x40000000, true },
{ NVKM_SUBDEV_BUS , 0, 0, 0x10000000, true },
{ NVKM_SUBDEV_FB , 0, 0, 0x08002000, true },
{ NVKM_SUBDEV_LTC , 0, 0, 0x02000000, true },
{ NVKM_SUBDEV_PMU , 0, 0, 0x01000000, true },
{ NVKM_SUBDEV_GPIO , 0, 0, 0x00200000, true },
{ NVKM_SUBDEV_I2C , 0, 0, 0x00200000, true },
{ NVKM_SUBDEV_TIMER , 0, 0, 0x00100000, true },
{ NVKM_SUBDEV_THERM , 0, 0, 0x00040000, true },
{ NVKM_SUBDEV_TOP , 0, 0, 0xffffffff, true },
{},
}; };
static void static void
gp100_mc_intr_update(struct gp100_mc *mc) gp100_mc_intr_allow(struct nvkm_intr *intr, int leaf, u32 mask)
{ {
struct nvkm_device *device = mc->base.subdev.device; struct nvkm_mc *mc = container_of(intr, typeof(*mc), intr);
u32 mask = mc->intr ? mc->mask : 0, i;
for (i = 0; i < 2; i++) { nvkm_wr32(mc->subdev.device, 0x000160 + (leaf * 4), mask);
nvkm_wr32(device, 0x000180 + (i * 0x04), ~mask);
nvkm_wr32(device, 0x000160 + (i * 0x04), mask);
}
} }
void static void
gp100_mc_intr_unarm(struct nvkm_mc *base) gp100_mc_intr_block(struct nvkm_intr *intr, int leaf, u32 mask)
{ {
struct gp100_mc *mc = gp100_mc(base); struct nvkm_mc *mc = container_of(intr, typeof(*mc), intr);
unsigned long flags;
spin_lock_irqsave(&mc->lock, flags); nvkm_wr32(mc->subdev.device, 0x000180 + (leaf * 4), mask);
mc->intr = false;
gp100_mc_intr_update(mc);
spin_unlock_irqrestore(&mc->lock, flags);
} }
void static void
gp100_mc_intr_rearm(struct nvkm_mc *base) gp100_mc_intr_rearm(struct nvkm_intr *intr)
{ {
struct gp100_mc *mc = gp100_mc(base); int i;
unsigned long flags;
spin_lock_irqsave(&mc->lock, flags); for (i = 0; i < intr->leaves; i++)
mc->intr = true; intr->func->allow(intr, i, intr->mask[i]);
gp100_mc_intr_update(mc);
spin_unlock_irqrestore(&mc->lock, flags);
} }
void static void
gp100_mc_intr_mask(struct nvkm_mc *base, u32 mask, u32 intr) gp100_mc_intr_unarm(struct nvkm_intr *intr)
{ {
struct gp100_mc *mc = gp100_mc(base); int i;
unsigned long flags;
spin_lock_irqsave(&mc->lock, flags); for (i = 0; i < intr->leaves; i++)
mc->mask = (mc->mask & ~mask) | intr; intr->func->block(intr, i, 0xffffffff);
gp100_mc_intr_update(mc);
spin_unlock_irqrestore(&mc->lock, flags);
} }
const struct nvkm_mc_map const struct nvkm_intr_func
gp100_mc_intr[] = { gp100_mc_intr = {
{ 0x04000000, NVKM_ENGINE_DISP }, .pending = nv04_mc_intr_pending,
{ 0x00000100, NVKM_ENGINE_FIFO }, .unarm = gp100_mc_intr_unarm,
{ 0x00000200, NVKM_SUBDEV_FAULT }, .rearm = gp100_mc_intr_rearm,
{ 0x40000000, NVKM_SUBDEV_PRIVRING }, .block = gp100_mc_intr_block,
{ 0x10000000, NVKM_SUBDEV_BUS }, .allow = gp100_mc_intr_allow,
{ 0x08000000, NVKM_SUBDEV_FB },
{ 0x02000000, NVKM_SUBDEV_LTC },
{ 0x01000000, NVKM_SUBDEV_PMU },
{ 0x00200000, NVKM_SUBDEV_GPIO },
{ 0x00200000, NVKM_SUBDEV_I2C },
{ 0x00100000, NVKM_SUBDEV_TIMER },
{ 0x00040000, NVKM_SUBDEV_THERM },
{ 0x00002000, NVKM_SUBDEV_FB },
{},
}; };
static const struct nvkm_mc_func static const struct nvkm_mc_func
gp100_mc = { gp100_mc = {
.init = nv50_mc_init, .init = nv50_mc_init,
.intr = gp100_mc_intr, .intr = &gp100_mc_intr,
.intr_unarm = gp100_mc_intr_unarm, .intrs = gp100_mc_intrs,
.intr_rearm = gp100_mc_intr_rearm, .intr_nonstall = true,
.intr_mask = gp100_mc_intr_mask,
.intr_stat = gf100_mc_intr_stat,
.reset = gk104_mc_reset, .reset = gk104_mc_reset,
}; };
int
gp100_mc_new_(const struct nvkm_mc_func *func, struct nvkm_device *device,
enum nvkm_subdev_type type, int inst, struct nvkm_mc **pmc)
{
struct gp100_mc *mc;
if (!(mc = kzalloc(sizeof(*mc), GFP_KERNEL)))
return -ENOMEM;
nvkm_mc_ctor(func, device, type, inst, &mc->base);
*pmc = &mc->base;
spin_lock_init(&mc->lock);
mc->intr = false;
mc->mask = 0x7fffffff;
return 0;
}
int int
gp100_mc_new(struct nvkm_device *device, enum nvkm_subdev_type type, int inst, struct nvkm_mc **pmc) gp100_mc_new(struct nvkm_device *device, enum nvkm_subdev_type type, int inst, struct nvkm_mc **pmc)
{ {
return gp100_mc_new_(&gp100_mc, device, type, inst, pmc); return nvkm_mc_new_(&gp100_mc, device, type, inst, pmc);
} }
...@@ -34,16 +34,14 @@ gp10b_mc_init(struct nvkm_mc *mc) ...@@ -34,16 +34,14 @@ gp10b_mc_init(struct nvkm_mc *mc)
static const struct nvkm_mc_func static const struct nvkm_mc_func
gp10b_mc = { gp10b_mc = {
.init = gp10b_mc_init, .init = gp10b_mc_init,
.intr = gp100_mc_intr, .intr = &gp100_mc_intr,
.intr_unarm = gp100_mc_intr_unarm, .intrs = gp100_mc_intrs,
.intr_rearm = gp100_mc_intr_rearm, .intr_nonstall = true,
.intr_mask = gp100_mc_intr_mask,
.intr_stat = gf100_mc_intr_stat,
.reset = gk104_mc_reset, .reset = gk104_mc_reset,
}; };
int int
gp10b_mc_new(struct nvkm_device *device, enum nvkm_subdev_type type, int inst, struct nvkm_mc **pmc) gp10b_mc_new(struct nvkm_device *device, enum nvkm_subdev_type type, int inst, struct nvkm_mc **pmc)
{ {
return gp100_mc_new_(&gp10b_mc, device, type, inst, pmc); return nvkm_mc_new_(&gp10b_mc, device, type, inst, pmc);
} }
...@@ -34,39 +34,55 @@ gt215_mc_reset[] = { ...@@ -34,39 +34,55 @@ gt215_mc_reset[] = {
{} {}
}; };
static const struct nvkm_mc_map static const struct nvkm_intr_data
gt215_mc_intr[] = { gt215_mc_intrs[] = {
{ 0x04000000, NVKM_ENGINE_DISP }, { NVKM_ENGINE_DISP , 0, 0, 0x04000000, true },
{ 0x00400000, NVKM_ENGINE_CE, 0 }, { NVKM_ENGINE_CE , 0, 0, 0x00400000, true },
{ 0x00020000, NVKM_ENGINE_MSPDEC }, { NVKM_ENGINE_MSPDEC, 0, 0, 0x00020000, true },
{ 0x00008000, NVKM_ENGINE_MSVLD }, { NVKM_ENGINE_MSVLD , 0, 0, 0x00008000, true },
{ 0x00001000, NVKM_ENGINE_GR }, { NVKM_ENGINE_GR , 0, 0, 0x00001000, true },
{ 0x00000100, NVKM_ENGINE_FIFO }, { NVKM_ENGINE_FIFO , 0, 0, 0x00000100, true },
{ 0x00000001, NVKM_ENGINE_MSPPP }, { NVKM_ENGINE_MSPPP , 0, 0, 0x00000001, true },
{ 0x00429101, NVKM_SUBDEV_FB }, { NVKM_SUBDEV_FB , 0, 0, 0x00429101, true },
{ 0x10000000, NVKM_SUBDEV_BUS }, { NVKM_SUBDEV_BUS , 0, 0, 0x10000000, true },
{ 0x00200000, NVKM_SUBDEV_GPIO }, { NVKM_SUBDEV_GPIO , 0, 0, 0x00200000, true },
{ 0x00200000, NVKM_SUBDEV_I2C }, { NVKM_SUBDEV_I2C , 0, 0, 0x00200000, true },
{ 0x00100000, NVKM_SUBDEV_TIMER }, { NVKM_SUBDEV_TIMER , 0, 0, 0x00100000, true },
{ 0x00080000, NVKM_SUBDEV_THERM }, { NVKM_SUBDEV_THERM , 0, 0, 0x00080000, true },
{ 0x00040000, NVKM_SUBDEV_PMU }, { NVKM_SUBDEV_PMU , 0, 0, 0x00040000, true },
{}, {},
}; };
static void static void
gt215_mc_intr_mask(struct nvkm_mc *mc, u32 mask, u32 stat) gt215_mc_intr_allow(struct nvkm_intr *intr, int leaf, u32 mask)
{
struct nvkm_mc *mc = container_of(intr, typeof(*mc), intr);
nvkm_mask(mc->subdev.device, 0x000640 + (leaf * 4), mask, mask);
}
static void
gt215_mc_intr_block(struct nvkm_intr *intr, int leaf, u32 mask)
{ {
nvkm_mask(mc->subdev.device, 0x000640, mask, stat); struct nvkm_mc *mc = container_of(intr, typeof(*mc), intr);
nvkm_mask(mc->subdev.device, 0x000640 + (leaf * 4), mask, 0);
} }
const struct nvkm_intr_func
gt215_mc_intr = {
.pending = nv04_mc_intr_pending,
.unarm = nv04_mc_intr_unarm,
.rearm = nv04_mc_intr_rearm,
.block = gt215_mc_intr_block,
.allow = gt215_mc_intr_allow,
};
static const struct nvkm_mc_func static const struct nvkm_mc_func
gt215_mc = { gt215_mc = {
.init = nv50_mc_init, .init = nv50_mc_init,
.intr = gt215_mc_intr, .intr = &nv04_mc_intr,
.intr_unarm = nv04_mc_intr_unarm, .intrs = gt215_mc_intrs,
.intr_rearm = nv04_mc_intr_rearm,
.intr_mask = gt215_mc_intr_mask,
.intr_stat = nv04_mc_intr_stat,
.reset = gt215_mc_reset, .reset = gt215_mc_reset,
}; };
......
...@@ -30,37 +30,61 @@ nv04_mc_reset[] = { ...@@ -30,37 +30,61 @@ nv04_mc_reset[] = {
{} {}
}; };
static const struct nvkm_mc_map static const struct nvkm_intr_data
nv04_mc_intr[] = { nv04_mc_intrs[] = {
{ 0x01010000, NVKM_ENGINE_DISP }, { NVKM_ENGINE_DISP , 0, 0, 0x01010000, true },
{ 0x00001000, NVKM_ENGINE_GR }, { NVKM_ENGINE_GR , 0, 0, 0x00001000, true },
{ 0x00000100, NVKM_ENGINE_FIFO }, { NVKM_ENGINE_FIFO , 0, 0, 0x00000100, true },
{ 0x10000000, NVKM_SUBDEV_BUS }, { NVKM_SUBDEV_BUS , 0, 0, 0x10000000, true },
{ 0x00100000, NVKM_SUBDEV_TIMER }, { NVKM_SUBDEV_TIMER, 0, 0, 0x00100000, true },
{} {}
}; };
void void
nv04_mc_intr_unarm(struct nvkm_mc *mc) nv04_mc_intr_rearm(struct nvkm_intr *intr)
{ {
struct nvkm_device *device = mc->subdev.device; struct nvkm_mc *mc = container_of(intr, typeof(*mc), intr);
nvkm_wr32(device, 0x000140, 0x00000000); int leaf;
nvkm_rd32(device, 0x000140);
for (leaf = 0; leaf < intr->leaves; leaf++)
nvkm_wr32(mc->subdev.device, 0x000140 + (leaf * 4), 0x00000001);
} }
void void
nv04_mc_intr_rearm(struct nvkm_mc *mc) nv04_mc_intr_unarm(struct nvkm_intr *intr)
{ {
struct nvkm_device *device = mc->subdev.device; struct nvkm_mc *mc = container_of(intr, typeof(*mc), intr);
nvkm_wr32(device, 0x000140, 0x00000001); int leaf;
for (leaf = 0; leaf < intr->leaves; leaf++)
nvkm_wr32(mc->subdev.device, 0x000140 + (leaf * 4), 0x00000000);
nvkm_rd32(mc->subdev.device, 0x000140);
} }
u32 bool
nv04_mc_intr_stat(struct nvkm_mc *mc) nv04_mc_intr_pending(struct nvkm_intr *intr)
{ {
return nvkm_rd32(mc->subdev.device, 0x000100); struct nvkm_mc *mc = container_of(intr, typeof(*mc), intr);
bool pending = false;
int leaf;
for (leaf = 0; leaf < intr->leaves; leaf++) {
intr->stat[leaf] = nvkm_rd32(mc->subdev.device, 0x000100 + (leaf * 4));
if (intr->stat[leaf])
pending = true;
}
return pending;
} }
const struct nvkm_intr_func
nv04_mc_intr = {
.pending = nv04_mc_intr_pending,
.unarm = nv04_mc_intr_unarm,
.rearm = nv04_mc_intr_rearm,
};
void void
nv04_mc_init(struct nvkm_mc *mc) nv04_mc_init(struct nvkm_mc *mc)
{ {
...@@ -72,10 +96,8 @@ nv04_mc_init(struct nvkm_mc *mc) ...@@ -72,10 +96,8 @@ nv04_mc_init(struct nvkm_mc *mc)
static const struct nvkm_mc_func static const struct nvkm_mc_func
nv04_mc = { nv04_mc = {
.init = nv04_mc_init, .init = nv04_mc_init,
.intr = nv04_mc_intr, .intr = &nv04_mc_intr,
.intr_unarm = nv04_mc_intr_unarm, .intrs = nv04_mc_intrs,
.intr_rearm = nv04_mc_intr_rearm,
.intr_stat = nv04_mc_intr_stat,
.reset = nv04_mc_reset, .reset = nv04_mc_reset,
}; };
......
...@@ -23,23 +23,21 @@ ...@@ -23,23 +23,21 @@
*/ */
#include "priv.h" #include "priv.h"
static const struct nvkm_mc_map static const struct nvkm_intr_data
nv11_mc_intr[] = { nv11_mc_intrs[] = {
{ 0x03010000, NVKM_ENGINE_DISP }, { NVKM_ENGINE_DISP , 0, 0, 0x03010000, true },
{ 0x00001000, NVKM_ENGINE_GR }, { NVKM_ENGINE_GR , 0, 0, 0x00001000, true },
{ 0x00000100, NVKM_ENGINE_FIFO }, { NVKM_ENGINE_FIFO , 0, 0, 0x00000100, true },
{ 0x10000000, NVKM_SUBDEV_BUS }, { NVKM_SUBDEV_BUS , 0, 0, 0x10000000, true },
{ 0x00100000, NVKM_SUBDEV_TIMER }, { NVKM_SUBDEV_TIMER, 0, 0, 0x00100000, true },
{} {}
}; };
static const struct nvkm_mc_func static const struct nvkm_mc_func
nv11_mc = { nv11_mc = {
.init = nv04_mc_init, .init = nv04_mc_init,
.intr = nv11_mc_intr, .intr = &nv04_mc_intr,
.intr_unarm = nv04_mc_intr_unarm, .intrs = nv11_mc_intrs,
.intr_rearm = nv04_mc_intr_rearm,
.intr_stat = nv04_mc_intr_stat,
.reset = nv04_mc_reset, .reset = nv04_mc_reset,
}; };
......
...@@ -31,24 +31,22 @@ nv17_mc_reset[] = { ...@@ -31,24 +31,22 @@ nv17_mc_reset[] = {
{} {}
}; };
const struct nvkm_mc_map const struct nvkm_intr_data
nv17_mc_intr[] = { nv17_mc_intrs[] = {
{ 0x03010000, NVKM_ENGINE_DISP }, { NVKM_ENGINE_DISP , 0, 0, 0x03010000, true },
{ 0x00001000, NVKM_ENGINE_GR }, { NVKM_ENGINE_GR , 0, 0, 0x00001000, true },
{ 0x00000100, NVKM_ENGINE_FIFO }, { NVKM_ENGINE_FIFO , 0, 0, 0x00000100, true },
{ 0x00000001, NVKM_ENGINE_MPEG }, { NVKM_ENGINE_MPEG , 0, 0, 0x00000001, true },
{ 0x10000000, NVKM_SUBDEV_BUS }, { NVKM_SUBDEV_BUS , 0, 0, 0x10000000, true },
{ 0x00100000, NVKM_SUBDEV_TIMER }, { NVKM_SUBDEV_TIMER, 0, 0, 0x00100000, true },
{} {}
}; };
static const struct nvkm_mc_func static const struct nvkm_mc_func
nv17_mc = { nv17_mc = {
.init = nv04_mc_init, .init = nv04_mc_init,
.intr = nv17_mc_intr, .intr = &nv04_mc_intr,
.intr_unarm = nv04_mc_intr_unarm, .intrs = nv17_mc_intrs,
.intr_rearm = nv04_mc_intr_rearm,
.intr_stat = nv04_mc_intr_stat,
.reset = nv17_mc_reset, .reset = nv17_mc_reset,
}; };
......
...@@ -40,10 +40,8 @@ nv44_mc_init(struct nvkm_mc *mc) ...@@ -40,10 +40,8 @@ nv44_mc_init(struct nvkm_mc *mc)
static const struct nvkm_mc_func static const struct nvkm_mc_func
nv44_mc = { nv44_mc = {
.init = nv44_mc_init, .init = nv44_mc_init,
.intr = nv17_mc_intr, .intr = &nv04_mc_intr,
.intr_unarm = nv04_mc_intr_unarm, .intrs = nv17_mc_intrs,
.intr_rearm = nv04_mc_intr_rearm,
.intr_stat = nv04_mc_intr_stat,
.reset = nv17_mc_reset, .reset = nv17_mc_reset,
}; };
......
...@@ -23,17 +23,17 @@ ...@@ -23,17 +23,17 @@
*/ */
#include "priv.h" #include "priv.h"
static const struct nvkm_mc_map static const struct nvkm_intr_data
nv50_mc_intr[] = { nv50_mc_intrs[] = {
{ 0x04000000, NVKM_ENGINE_DISP }, { NVKM_ENGINE_DISP , 0, 0, 0x04000000, true },
{ 0x00001000, NVKM_ENGINE_GR }, { NVKM_ENGINE_GR , 0, 0, 0x00001000, true },
{ 0x00000100, NVKM_ENGINE_FIFO }, { NVKM_ENGINE_FIFO , 0, 0, 0x00000100, true },
{ 0x00000001, NVKM_ENGINE_MPEG }, { NVKM_ENGINE_MPEG , 0, 0, 0x00000001, true },
{ 0x00001101, NVKM_SUBDEV_FB }, { NVKM_SUBDEV_FB , 0, 0, 0x00001101, true },
{ 0x10000000, NVKM_SUBDEV_BUS }, { NVKM_SUBDEV_BUS , 0, 0, 0x10000000, true },
{ 0x00200000, NVKM_SUBDEV_GPIO }, { NVKM_SUBDEV_GPIO , 0, 0, 0x00200000, true },
{ 0x00200000, NVKM_SUBDEV_I2C }, { NVKM_SUBDEV_I2C , 0, 0, 0x00200000, true },
{ 0x00100000, NVKM_SUBDEV_TIMER }, { NVKM_SUBDEV_TIMER, 0, 0, 0x00100000, true },
{}, {},
}; };
...@@ -47,10 +47,8 @@ nv50_mc_init(struct nvkm_mc *mc) ...@@ -47,10 +47,8 @@ nv50_mc_init(struct nvkm_mc *mc)
static const struct nvkm_mc_func static const struct nvkm_mc_func
nv50_mc = { nv50_mc = {
.init = nv50_mc_init, .init = nv50_mc_init,
.intr = nv50_mc_intr, .intr = &nv04_mc_intr,
.intr_unarm = nv04_mc_intr_unarm, .intrs = nv50_mc_intrs,
.intr_rearm = nv04_mc_intr_rearm,
.intr_stat = nv04_mc_intr_stat,
.reset = nv17_mc_reset, .reset = nv17_mc_reset,
}; };
......
...@@ -4,8 +4,6 @@ ...@@ -4,8 +4,6 @@
#define nvkm_mc(p) container_of((p), struct nvkm_mc, subdev) #define nvkm_mc(p) container_of((p), struct nvkm_mc, subdev)
#include <subdev/mc.h> #include <subdev/mc.h>
void nvkm_mc_ctor(const struct nvkm_mc_func *, struct nvkm_device *, enum nvkm_subdev_type, int,
struct nvkm_mc *);
int nvkm_mc_new_(const struct nvkm_mc_func *, struct nvkm_device *, enum nvkm_subdev_type, int, int nvkm_mc_new_(const struct nvkm_mc_func *, struct nvkm_device *, enum nvkm_subdev_type, int,
struct nvkm_mc **); struct nvkm_mc **);
...@@ -18,46 +16,36 @@ struct nvkm_mc_map { ...@@ -18,46 +16,36 @@ struct nvkm_mc_map {
struct nvkm_mc_func { struct nvkm_mc_func {
void (*init)(struct nvkm_mc *); void (*init)(struct nvkm_mc *);
const struct nvkm_mc_map *intr;
/* disable reporting of interrupts to host */ const struct nvkm_intr_func *intr;
void (*intr_unarm)(struct nvkm_mc *); const struct nvkm_intr_data *intrs;
/* enable reporting of interrupts to host */ bool intr_nonstall;
void (*intr_rearm)(struct nvkm_mc *);
/* (un)mask delivery of specific interrupts */
void (*intr_mask)(struct nvkm_mc *, u32 mask, u32 stat);
/* retrieve pending interrupt mask (NV_PMC_INTR) */
u32 (*intr_stat)(struct nvkm_mc *);
const struct nvkm_mc_map *reset; const struct nvkm_mc_map *reset;
void (*unk260)(struct nvkm_mc *, u32); void (*unk260)(struct nvkm_mc *, u32);
}; };
void nv04_mc_init(struct nvkm_mc *); void nv04_mc_init(struct nvkm_mc *);
void nv04_mc_intr_unarm(struct nvkm_mc *); extern const struct nvkm_intr_func nv04_mc_intr;
void nv04_mc_intr_rearm(struct nvkm_mc *); bool nv04_mc_intr_pending(struct nvkm_intr *);
u32 nv04_mc_intr_stat(struct nvkm_mc *); void nv04_mc_intr_unarm(struct nvkm_intr *);
void nv04_mc_intr_rearm(struct nvkm_intr *);
extern const struct nvkm_mc_map nv04_mc_reset[]; extern const struct nvkm_mc_map nv04_mc_reset[];
extern const struct nvkm_mc_map nv17_mc_intr[]; extern const struct nvkm_intr_data nv17_mc_intrs[];
extern const struct nvkm_mc_map nv17_mc_reset[]; extern const struct nvkm_mc_map nv17_mc_reset[];
void nv44_mc_init(struct nvkm_mc *); void nv44_mc_init(struct nvkm_mc *);
void nv50_mc_init(struct nvkm_mc *); void nv50_mc_init(struct nvkm_mc *);
void gk104_mc_init(struct nvkm_mc *);
void gf100_mc_intr_unarm(struct nvkm_mc *); extern const struct nvkm_intr_func gt215_mc_intr;
void gf100_mc_intr_rearm(struct nvkm_mc *);
void gf100_mc_intr_mask(struct nvkm_mc *, u32, u32);
u32 gf100_mc_intr_stat(struct nvkm_mc *);
void gf100_mc_unk260(struct nvkm_mc *, u32); void gf100_mc_unk260(struct nvkm_mc *, u32);
void gp100_mc_intr_unarm(struct nvkm_mc *);
void gp100_mc_intr_rearm(struct nvkm_mc *);
void gp100_mc_intr_mask(struct nvkm_mc *, u32, u32);
int gp100_mc_new_(const struct nvkm_mc_func *, struct nvkm_device *, enum nvkm_subdev_type, int,
struct nvkm_mc **);
extern const struct nvkm_mc_map gk104_mc_intr[]; void gk104_mc_init(struct nvkm_mc *);
extern const struct nvkm_intr_data gk104_mc_intrs[];
extern const struct nvkm_mc_map gk104_mc_reset[]; extern const struct nvkm_mc_map gk104_mc_reset[];
extern const struct nvkm_mc_map gp100_mc_intr[]; extern const struct nvkm_intr_func gp100_mc_intr;
extern const struct nvkm_intr_data gp100_mc_intrs[];
#endif #endif
/*
* Copyright 2018 Red Hat Inc.
*
* Permission is hereby granted, free of charge, to any person obtaining a
* copy of this software and associated documentation files (the "Software"),
* to deal in the Software without restriction, including without limitation
* the rights to use, copy, modify, merge, publish, distribute, sublicense,
* and/or sell copies of the Software, and to permit persons to whom the
* Software is furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in
* all copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
* THE COPYRIGHT HOLDER(S) OR AUTHOR(S) BE LIABLE FOR ANY CLAIM, DAMAGES OR
* OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
* ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
* OTHER DEALINGS IN THE SOFTWARE.
*/
#define tu102_mc(p) container_of((p), struct tu102_mc, base)
#include "priv.h"
struct tu102_mc {
struct nvkm_mc base;
spinlock_t lock;
bool intr;
u32 mask;
};
static void
tu102_mc_intr_update(struct tu102_mc *mc)
{
struct nvkm_device *device = mc->base.subdev.device;
u32 mask = mc->intr ? mc->mask : 0, i;
for (i = 0; i < 2; i++) {
nvkm_wr32(device, 0x000180 + (i * 0x04), ~mask);
nvkm_wr32(device, 0x000160 + (i * 0x04), mask);
}
}
static void
tu102_mc_intr_unarm(struct nvkm_mc *base)
{
struct tu102_mc *mc = tu102_mc(base);
unsigned long flags;
spin_lock_irqsave(&mc->lock, flags);
mc->intr = false;
tu102_mc_intr_update(mc);
spin_unlock_irqrestore(&mc->lock, flags);
}
static void
tu102_mc_intr_rearm(struct nvkm_mc *base)
{
struct tu102_mc *mc = tu102_mc(base);
unsigned long flags;
spin_lock_irqsave(&mc->lock, flags);
mc->intr = true;
tu102_mc_intr_update(mc);
spin_unlock_irqrestore(&mc->lock, flags);
}
static void
tu102_mc_intr_mask(struct nvkm_mc *base, u32 mask, u32 intr)
{
struct tu102_mc *mc = tu102_mc(base);
unsigned long flags;
spin_lock_irqsave(&mc->lock, flags);
mc->mask = (mc->mask & ~mask) | intr;
tu102_mc_intr_update(mc);
spin_unlock_irqrestore(&mc->lock, flags);
}
static u32
tu102_mc_intr_stat(struct nvkm_mc *mc)
{
struct nvkm_device *device = mc->subdev.device;
u32 intr0 = nvkm_rd32(device, 0x000100);
u32 intr1 = nvkm_rd32(device, 0x000104);
return intr0 | intr1;
}
static const struct nvkm_mc_func
tu102_mc = {
.init = nv50_mc_init,
.intr = gp100_mc_intr,
.intr_unarm = tu102_mc_intr_unarm,
.intr_rearm = tu102_mc_intr_rearm,
.intr_mask = tu102_mc_intr_mask,
.intr_stat = tu102_mc_intr_stat,
.reset = gk104_mc_reset,
};
static int
tu102_mc_new_(const struct nvkm_mc_func *func, struct nvkm_device *device,
enum nvkm_subdev_type type, int inst, struct nvkm_mc **pmc)
{
struct tu102_mc *mc;
if (!(mc = kzalloc(sizeof(*mc), GFP_KERNEL)))
return -ENOMEM;
nvkm_mc_ctor(func, device, type, inst, &mc->base);
*pmc = &mc->base;
spin_lock_init(&mc->lock);
mc->intr = false;
mc->mask = 0x7fffffff;
return 0;
}
int
tu102_mc_new(struct nvkm_device *device, enum nvkm_subdev_type type, int inst, struct nvkm_mc **pmc)
{
return tu102_mc_new_(&tu102_mc, device, type, inst, pmc);
}
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment