|
@@ -24,6 +24,7 @@
|
|
#include "nv50.h"
|
|
#include "nv50.h"
|
|
|
|
|
|
#include <subdev/bar.h>
|
|
#include <subdev/bar.h>
|
|
|
|
+#include <engine/disp.h>
|
|
|
|
|
|
#include <nvif/ioctl.h>
|
|
#include <nvif/ioctl.h>
|
|
|
|
|
|
@@ -31,65 +32,9 @@
|
|
* software object classes
|
|
* software object classes
|
|
******************************************************************************/
|
|
******************************************************************************/
|
|
|
|
|
|
-static int
|
|
|
|
-gf100_sw_mthd_vblsem_offset(struct nvkm_object *object, u32 mthd,
|
|
|
|
- void *args, u32 size)
|
|
|
|
-{
|
|
|
|
- struct nv50_sw_chan *chan = (void *)nv_engctx(object->parent);
|
|
|
|
- u64 data = *(u32 *)args;
|
|
|
|
- if (mthd == 0x0400) {
|
|
|
|
- chan->vblank.offset &= 0x00ffffffffULL;
|
|
|
|
- chan->vblank.offset |= data << 32;
|
|
|
|
- } else {
|
|
|
|
- chan->vblank.offset &= 0xff00000000ULL;
|
|
|
|
- chan->vblank.offset |= data;
|
|
|
|
- }
|
|
|
|
- return 0;
|
|
|
|
-}
|
|
|
|
-
|
|
|
|
-static int
|
|
|
|
-gf100_sw_mthd_mp_control(struct nvkm_object *object, u32 mthd,
|
|
|
|
- void *args, u32 size)
|
|
|
|
-{
|
|
|
|
- struct nv50_sw_chan *chan = (void *)nv_engctx(object->parent);
|
|
|
|
- struct nvkm_sw *sw = (void *)nv_object(chan)->engine;
|
|
|
|
- struct nvkm_device *device = sw->engine.subdev.device;
|
|
|
|
- u32 data = *(u32 *)args;
|
|
|
|
-
|
|
|
|
- switch (mthd) {
|
|
|
|
- case 0x600:
|
|
|
|
- nvkm_wr32(device, 0x419e00, data); /* MP.PM_UNK000 */
|
|
|
|
- break;
|
|
|
|
- case 0x644:
|
|
|
|
- if (data & ~0x1ffffe)
|
|
|
|
- return -EINVAL;
|
|
|
|
- nvkm_wr32(device, 0x419e44, data); /* MP.TRAP_WARP_ERROR_EN */
|
|
|
|
- break;
|
|
|
|
- case 0x6ac:
|
|
|
|
- nvkm_wr32(device, 0x419eac, data); /* MP.PM_UNK0AC */
|
|
|
|
- break;
|
|
|
|
- default:
|
|
|
|
- return -EINVAL;
|
|
|
|
- }
|
|
|
|
- return 0;
|
|
|
|
-}
|
|
|
|
-
|
|
|
|
-static struct nvkm_omthds
|
|
|
|
-gf100_sw_omthds[] = {
|
|
|
|
- { 0x0400, 0x0400, gf100_sw_mthd_vblsem_offset },
|
|
|
|
- { 0x0404, 0x0404, gf100_sw_mthd_vblsem_offset },
|
|
|
|
- { 0x0408, 0x0408, nv50_sw_mthd_vblsem_value },
|
|
|
|
- { 0x040c, 0x040c, nv50_sw_mthd_vblsem_release },
|
|
|
|
- { 0x0500, 0x0500, nv50_sw_mthd_flip },
|
|
|
|
- { 0x0600, 0x0600, gf100_sw_mthd_mp_control },
|
|
|
|
- { 0x0644, 0x0644, gf100_sw_mthd_mp_control },
|
|
|
|
- { 0x06ac, 0x06ac, gf100_sw_mthd_mp_control },
|
|
|
|
- {}
|
|
|
|
-};
|
|
|
|
-
|
|
|
|
static struct nvkm_oclass
|
|
static struct nvkm_oclass
|
|
gf100_sw_sclass[] = {
|
|
gf100_sw_sclass[] = {
|
|
- { NVIF_IOCTL_NEW_V0_SW_GF100, &nvkm_nvsw_ofuncs, gf100_sw_omthds },
|
|
|
|
|
|
+ { NVIF_IOCTL_NEW_V0_SW_GF100, &nvkm_nvsw_ofuncs },
|
|
{}
|
|
{}
|
|
};
|
|
};
|
|
|
|
|
|
@@ -115,6 +60,53 @@ gf100_sw_vblsem_release(struct nvkm_notify *notify)
|
|
return NVKM_NOTIFY_DROP;
|
|
return NVKM_NOTIFY_DROP;
|
|
}
|
|
}
|
|
|
|
|
|
|
|
+static bool
|
|
|
|
+gf100_sw_chan_mthd(struct nvkm_sw_chan *base, int subc, u32 mthd, u32 data)
|
|
|
|
+{
|
|
|
|
+ struct nv50_sw_chan *chan = nv50_sw_chan(base);
|
|
|
|
+ struct nvkm_engine *engine = chan->base.base.gpuobj.object.engine;
|
|
|
|
+ struct nvkm_device *device = engine->subdev.device;
|
|
|
|
+ switch (mthd) {
|
|
|
|
+ case 0x0400:
|
|
|
|
+ chan->vblank.offset &= 0x00ffffffffULL;
|
|
|
|
+ chan->vblank.offset |= (u64)data << 32;
|
|
|
|
+ return true;
|
|
|
|
+ case 0x0404:
|
|
|
|
+ chan->vblank.offset &= 0xff00000000ULL;
|
|
|
|
+ chan->vblank.offset |= data;
|
|
|
|
+ return true;
|
|
|
|
+ case 0x0408:
|
|
|
|
+ chan->vblank.value = data;
|
|
|
|
+ return true;
|
|
|
|
+ case 0x040c:
|
|
|
|
+ if (data < device->disp->vblank.index_nr) {
|
|
|
|
+ nvkm_notify_get(&chan->vblank.notify[data]);
|
|
|
|
+ return true;
|
|
|
|
+ }
|
|
|
|
+ break;
|
|
|
|
+ case 0x600: /* MP.PM_UNK000 */
|
|
|
|
+ nvkm_wr32(device, 0x419e00, data);
|
|
|
|
+ return true;
|
|
|
|
+ case 0x644: /* MP.TRAP_WARP_ERROR_EN */
|
|
|
|
+ if (!(data & ~0x001ffffe)) {
|
|
|
|
+ nvkm_wr32(device, 0x419e44, data);
|
|
|
|
+ return true;
|
|
|
|
+ }
|
|
|
|
+ break;
|
|
|
|
+ case 0x6ac: /* MP.PM_UNK0AC */
|
|
|
|
+ nvkm_wr32(device, 0x419eac, data);
|
|
|
|
+ return true;
|
|
|
|
+ default:
|
|
|
|
+ break;
|
|
|
|
+ }
|
|
|
|
+ return false;
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+static const struct nvkm_sw_chan_func
|
|
|
|
+gf100_sw_chan_func = {
|
|
|
|
+ .mthd = gf100_sw_chan_mthd,
|
|
|
|
+};
|
|
|
|
+
|
|
static struct nv50_sw_cclass
|
|
static struct nv50_sw_cclass
|
|
gf100_sw_cclass = {
|
|
gf100_sw_cclass = {
|
|
.base.handle = NV_ENGCTX(SW, 0xc0),
|
|
.base.handle = NV_ENGCTX(SW, 0xc0),
|
|
@@ -125,6 +117,7 @@ gf100_sw_cclass = {
|
|
.fini = _nvkm_sw_context_fini,
|
|
.fini = _nvkm_sw_context_fini,
|
|
},
|
|
},
|
|
.vblank = gf100_sw_vblsem_release,
|
|
.vblank = gf100_sw_vblsem_release,
|
|
|
|
+ .chan = &gf100_sw_chan_func,
|
|
};
|
|
};
|
|
|
|
|
|
/*******************************************************************************
|
|
/*******************************************************************************
|