drm/nouveau/sw: remove dependence on namedb/engctx lookup

Signed-off-by: Ben Skeggs <bskeggs@redhat.com>
This commit is contained in:
Ben Skeggs 2015-08-20 14:54:18 +10:00
parent 590801c1a3
commit 6157091177
15 changed files with 251 additions and 301 deletions

View File

@ -109,7 +109,6 @@ struct nvkm_sclass {
struct nvkm_oclass { struct nvkm_oclass {
s32 handle; s32 handle;
struct nvkm_ofuncs * const ofuncs; struct nvkm_ofuncs * const ofuncs;
struct nvkm_omthds * const omthds;
int (*ctor)(const struct nvkm_oclass *, void *data, u32 size, int (*ctor)(const struct nvkm_oclass *, void *data, u32 size,
struct nvkm_object **); struct nvkm_object **);
@ -135,12 +134,6 @@ nv_pclass(struct nvkm_object *parent, u32 oclass)
return parent; return parent;
} }
struct nvkm_omthds {
u32 start;
u32 limit;
int (*call)(struct nvkm_object *, u32, void *, u32);
};
struct nvkm_ofuncs { struct nvkm_ofuncs {
int (*ctor)(struct nvkm_object *, struct nvkm_object *, int (*ctor)(struct nvkm_object *, struct nvkm_object *,
struct nvkm_oclass *, void *data, u32 size, struct nvkm_oclass *, void *data, u32 size,
@ -171,24 +164,4 @@ int nvkm_object_old(struct nvkm_object *, struct nvkm_object *,
void nvkm_object_ref(struct nvkm_object *, struct nvkm_object **); void nvkm_object_ref(struct nvkm_object *, struct nvkm_object **);
int nvkm_object_inc(struct nvkm_object *); int nvkm_object_inc(struct nvkm_object *);
int nvkm_object_dec(struct nvkm_object *, bool suspend); int nvkm_object_dec(struct nvkm_object *, bool suspend);
static inline int
nv_exec(void *obj, u32 mthd, void *data, u32 size)
{
struct nvkm_omthds *method = nv_oclass(obj)->omthds;
while (method && method->call) {
if (mthd >= method->start && mthd <= method->limit)
return method->call(obj, mthd, data, size);
method++;
}
return -EINVAL;
}
static inline int
nv_call(void *obj, u32 mthd, u32 data)
{
return nv_exec(obj, mthd, &data, sizeof(data));
}
#endif #endif

View File

@ -4,10 +4,16 @@
struct nvkm_sw { struct nvkm_sw {
struct nvkm_engine engine; struct nvkm_engine engine;
struct list_head chan;
}; };
bool nvkm_sw_mthd(struct nvkm_sw *sw, int chid, int subc, u32 mthd, u32 data);
#define nvkm_sw_create(p,e,c,d) \ #define nvkm_sw_create(p,e,c,d) \
nvkm_engine_create((p), (e), (c), true, "SW", "software", (d)) nvkm_sw_ctor((p), (e), (c), sizeof(**d), (void **)d)
int
nvkm_sw_ctor(struct nvkm_object *parent, struct nvkm_object *engine,
struct nvkm_oclass *oclass, int length, void **pobject);
#define nvkm_sw_destroy(d) \ #define nvkm_sw_destroy(d) \
nvkm_engine_destroy(&(d)->engine) nvkm_engine_destroy(&(d)->engine)
#define nvkm_sw_init(d) \ #define nvkm_sw_init(d) \

View File

@ -174,7 +174,6 @@ int nv50_disp_main_ctor(struct nvkm_object *, struct nvkm_object *,
struct nvkm_oclass *, void *, u32, struct nvkm_oclass *, void *, u32,
struct nvkm_object **); struct nvkm_object **);
void nv50_disp_main_dtor(struct nvkm_object *); void nv50_disp_main_dtor(struct nvkm_object *);
extern struct nvkm_omthds nv50_disp_main_omthds[];
extern struct nvkm_oclass nv50_disp_cclass; extern struct nvkm_oclass nv50_disp_cclass;
void nv50_disp_mthd_chan(struct nv50_disp *, int debug, int head, void nv50_disp_mthd_chan(struct nv50_disp *, int debug, int head,
const struct nv50_disp_mthd_chan *); const struct nv50_disp_mthd_chan *);

View File

@ -31,6 +31,7 @@
#include <subdev/fb.h> #include <subdev/fb.h>
#include <subdev/mmu.h> #include <subdev/mmu.h>
#include <subdev/timer.h> #include <subdev/timer.h>
#include <engine/sw.h>
#include <nvif/class.h> #include <nvif/class.h>
#include <nvif/ioctl.h> #include <nvif/ioctl.h>
@ -474,32 +475,6 @@ gf100_fifo_recover(struct gf100_fifo *fifo, struct nvkm_engine *engine,
schedule_work(&fifo->fault); schedule_work(&fifo->fault);
} }
static int
gf100_fifo_swmthd(struct gf100_fifo *fifo, u32 chid, u32 mthd, u32 data)
{
struct gf100_fifo_chan *chan = NULL;
struct nvkm_handle *bind;
unsigned long flags;
int ret = -EINVAL;
spin_lock_irqsave(&fifo->base.lock, flags);
if (likely(chid >= fifo->base.min && chid <= fifo->base.max))
chan = (void *)fifo->base.channel[chid];
if (unlikely(!chan))
goto out;
bind = nvkm_namedb_get_class(nv_namedb(chan), NVIF_IOCTL_NEW_V0_SW_GF100);
if (likely(bind)) {
if (!mthd || !nv_call(bind->object, mthd, data))
ret = 0;
nvkm_namedb_put(bind);
}
out:
spin_unlock_irqrestore(&fifo->base.lock, flags);
return ret;
}
static const struct nvkm_enum static const struct nvkm_enum
gf100_fifo_sched_reason[] = { gf100_fifo_sched_reason[] = {
{ 0x0a, "CTXSW_TIMEOUT" }, { 0x0a, "CTXSW_TIMEOUT" },
@ -701,8 +676,10 @@ gf100_fifo_intr_pbdma(struct gf100_fifo *fifo, int unit)
char msg[128]; char msg[128];
if (stat & 0x00800000) { if (stat & 0x00800000) {
if (!gf100_fifo_swmthd(fifo, chid, mthd, data)) if (device->sw) {
show &= ~0x00800000; if (nvkm_sw_mthd(device->sw, chid, subc, mthd, data))
show &= ~0x00800000;
}
} }
if (show) { if (show) {

View File

@ -31,6 +31,7 @@
#include <subdev/fb.h> #include <subdev/fb.h>
#include <subdev/mmu.h> #include <subdev/mmu.h>
#include <subdev/timer.h> #include <subdev/timer.h>
#include <engine/sw.h>
#include <nvif/class.h> #include <nvif/class.h>
#include <nvif/ioctl.h> #include <nvif/ioctl.h>
@ -520,32 +521,6 @@ gk104_fifo_recover(struct gk104_fifo *fifo, struct nvkm_engine *engine,
schedule_work(&fifo->fault); schedule_work(&fifo->fault);
} }
static int
gk104_fifo_swmthd(struct gk104_fifo *fifo, u32 chid, u32 mthd, u32 data)
{
struct gk104_fifo_chan *chan = NULL;
struct nvkm_handle *bind;
unsigned long flags;
int ret = -EINVAL;
spin_lock_irqsave(&fifo->base.lock, flags);
if (likely(chid >= fifo->base.min && chid <= fifo->base.max))
chan = (void *)fifo->base.channel[chid];
if (unlikely(!chan))
goto out;
bind = nvkm_namedb_get_class(nv_namedb(chan), NVIF_IOCTL_NEW_V0_SW_GF100);
if (likely(bind)) {
if (!mthd || !nv_call(bind->object, mthd, data))
ret = 0;
nvkm_namedb_put(bind);
}
out:
spin_unlock_irqrestore(&fifo->base.lock, flags);
return ret;
}
static const struct nvkm_enum static const struct nvkm_enum
gk104_fifo_bind_reason[] = { gk104_fifo_bind_reason[] = {
{ 0x01, "BIND_NOT_UNBOUND" }, { 0x01, "BIND_NOT_UNBOUND" },
@ -864,8 +839,10 @@ gk104_fifo_intr_pbdma_0(struct gk104_fifo *fifo, int unit)
char msg[128]; char msg[128];
if (stat & 0x00800000) { if (stat & 0x00800000) {
if (!gk104_fifo_swmthd(fifo, chid, mthd, data)) if (device->sw) {
show &= ~0x00800000; if (nvkm_sw_mthd(device->sw, chid, subc, mthd, data))
show &= ~0x00800000;
}
nvkm_wr32(device, 0x0400c0 + (unit * 0x2000), 0x80600008); nvkm_wr32(device, 0x0400c0 + (unit * 0x2000), 0x80600008);
} }

View File

@ -29,6 +29,7 @@
#include <core/ramht.h> #include <core/ramht.h>
#include <subdev/instmem.h> #include <subdev/instmem.h>
#include <subdev/timer.h> #include <subdev/timer.h>
#include <engine/sw.h>
#include <nvif/class.h> #include <nvif/class.h>
#include <nvif/unpack.h> #include <nvif/unpack.h>
@ -369,55 +370,29 @@ nv_dma_state_err(u32 state)
} }
static bool static bool
nv04_fifo_swmthd(struct nv04_fifo *fifo, u32 chid, u32 addr, u32 data) nv04_fifo_swmthd(struct nvkm_device *device, u32 chid, u32 addr, u32 data)
{ {
struct nvkm_device *device = fifo->base.engine.subdev.device; struct nvkm_sw *sw = device->sw;
struct nv04_fifo_chan *chan = NULL; const int subc = (addr & 0x0000e000) >> 13;
struct nvkm_handle *bind; const int mthd = (addr & 0x00001ffc);
const int subc = (addr >> 13) & 0x7; const u32 mask = 0x0000000f << (subc * 4);
const int mthd = addr & 0x1ffc; u32 engine = nvkm_rd32(device, 0x003280);
bool handled = false; bool handled = false;
unsigned long flags;
u32 engine;
spin_lock_irqsave(&fifo->base.lock, flags);
if (likely(chid >= fifo->base.min && chid <= fifo->base.max))
chan = (void *)fifo->base.channel[chid];
if (unlikely(!chan))
goto out;
switch (mthd) { switch (mthd) {
case 0x0000: case 0x0000 ... 0x0000: /* subchannel's engine -> software */
bind = nvkm_namedb_get(nv_namedb(chan), data); nvkm_wr32(device, 0x003280, (engine &= ~mask));
if (unlikely(!bind)) case 0x0180 ... 0x01fc: /* handle -> instance */
break; data = nvkm_rd32(device, 0x003258) & 0x0000ffff;
case 0x0100 ... 0x017c:
if (nv_engidx(bind->object->engine) == NVDEV_ENGINE_SW) { case 0x0200 ... 0x1ffc: /* pass method down to sw */
engine = 0x0000000f << (subc * 4); if (!(engine & mask) && sw)
chan->subc[subc] = data; handled = nvkm_sw_mthd(sw, chid, subc, mthd, data);
handled = true;
nvkm_mask(device, NV04_PFIFO_CACHE1_ENGINE, engine, 0);
}
nvkm_namedb_put(bind);
break; break;
default: default:
engine = nvkm_rd32(device, NV04_PFIFO_CACHE1_ENGINE);
if (unlikely(((engine >> (subc * 4)) & 0xf) != 0))
break;
bind = nvkm_namedb_get(nv_namedb(chan), chan->subc[subc]);
if (likely(bind)) {
if (!nv_call(bind->object, mthd, data))
handled = true;
nvkm_namedb_put(bind);
}
break; break;
} }
out:
spin_unlock_irqrestore(&fifo->base.lock, flags);
return handled; return handled;
} }
@ -426,6 +401,7 @@ nv04_fifo_cache_error(struct nv04_fifo *fifo, u32 chid, u32 get)
{ {
struct nvkm_subdev *subdev = &fifo->base.engine.subdev; struct nvkm_subdev *subdev = &fifo->base.engine.subdev;
struct nvkm_device *device = subdev->device; struct nvkm_device *device = subdev->device;
u32 pull0 = nvkm_rd32(device, 0x003250);
u32 mthd, data; u32 mthd, data;
int ptr; int ptr;
@ -444,7 +420,8 @@ nv04_fifo_cache_error(struct nv04_fifo *fifo, u32 chid, u32 get)
data = nvkm_rd32(device, NV40_PFIFO_CACHE1_DATA(ptr)); data = nvkm_rd32(device, NV40_PFIFO_CACHE1_DATA(ptr));
} }
if (!nv04_fifo_swmthd(fifo, chid, mthd, data)) { if (!(pull0 & 0x00000100) ||
!nv04_fifo_swmthd(device, chid, mthd, data)) {
const char *client_name = const char *client_name =
nvkm_client_name_for_fifo_chid(&fifo->base, chid); nvkm_client_name_for_fifo_chid(&fifo->base, chid);
nvkm_error(subdev, "CACHE_ERROR - " nvkm_error(subdev, "CACHE_ERROR - "

View File

@ -1,3 +1,4 @@
nvkm-y += nvkm/engine/sw/base.o
nvkm-y += nvkm/engine/sw/nv04.o nvkm-y += nvkm/engine/sw/nv04.o
nvkm-y += nvkm/engine/sw/nv10.o nvkm-y += nvkm/engine/sw/nv10.o
nvkm-y += nvkm/engine/sw/nv50.o nvkm-y += nvkm/engine/sw/nv50.o

View File

@ -0,0 +1,64 @@
/*
* Copyright 2015 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.
*
* Authors: Ben Skeggs <bskeggs@redhat.com>
*/
#include "priv.h"
#include "chan.h"
#include <engine/fifo.h>
bool
nvkm_sw_mthd(struct nvkm_sw *sw, int chid, int subc, u32 mthd, u32 data)
{
struct nvkm_sw_chan *chan;
bool handled = false;
unsigned long flags;
spin_lock_irqsave(&sw->engine.lock, flags);
list_for_each_entry(chan, &sw->chan, head) {
if (chan->fifo->chid == chid) {
handled = nvkm_sw_chan_mthd(chan, subc, mthd, data);
list_del(&chan->head);
list_add(&chan->head, &sw->chan);
break;
}
}
spin_unlock_irqrestore(&sw->engine.lock, flags);
return handled;
}
int
nvkm_sw_ctor(struct nvkm_object *parent, struct nvkm_object *engine,
struct nvkm_oclass *oclass, int length, void **pobject)
{
struct nvkm_sw *sw;
int ret;
ret = nvkm_engine_create_(parent, engine, oclass, true, "sw",
"sw", length, pobject);
sw = *pobject;
if (ret)
return ret;
INIT_LIST_HEAD(&sw->chan);
return 0;
}

View File

@ -24,10 +24,28 @@
#include "chan.h" #include "chan.h"
#include <core/notify.h> #include <core/notify.h>
#include <engine/fifo.h>
#include <nvif/event.h> #include <nvif/event.h>
#include <nvif/unpack.h> #include <nvif/unpack.h>
bool
nvkm_sw_chan_mthd(struct nvkm_sw_chan *chan, int subc, u32 mthd, u32 data)
{
switch (mthd) {
case 0x0000:
return true;
case 0x0500:
nvkm_event_send(&chan->event, 1, 0, NULL, 0);
return true;
default:
if (chan->func->mthd)
return chan->func->mthd(chan, subc, mthd, data);
break;
}
return false;
}
static int static int
nvkm_sw_chan_event_ctor(struct nvkm_object *object, void *data, u32 size, nvkm_sw_chan_event_ctor(struct nvkm_object *object, void *data, u32 size,
struct nvkm_notify *notify) struct nvkm_notify *notify)
@ -55,14 +73,17 @@ void
nvkm_sw_chan_dtor(struct nvkm_object *base) nvkm_sw_chan_dtor(struct nvkm_object *base)
{ {
struct nvkm_sw_chan *chan = (void *)base; struct nvkm_sw_chan *chan = (void *)base;
list_del(&chan->head);
nvkm_event_fini(&chan->event); nvkm_event_fini(&chan->event);
nvkm_engctx_destroy(&chan->base); nvkm_engctx_destroy(&chan->base);
} }
int int
nvkm_sw_chan_ctor(struct nvkm_object *parent, struct nvkm_object *engine, nvkm_sw_chan_ctor(const struct nvkm_sw_chan_func *func,
struct nvkm_object *parent, struct nvkm_object *engine,
struct nvkm_oclass *oclass, int length, void **pobject) struct nvkm_oclass *oclass, int length, void **pobject)
{ {
struct nvkm_sw *sw = (void *)engine;
struct nvkm_sw_chan *chan; struct nvkm_sw_chan *chan;
int ret; int ret;
@ -72,5 +93,9 @@ nvkm_sw_chan_ctor(struct nvkm_object *parent, struct nvkm_object *engine,
if (ret) if (ret)
return ret; return ret;
chan->func = func;
chan->fifo = nvkm_fifo_chan(parent);
list_add(&chan->head, &sw->chan);
return nvkm_event_init(&nvkm_sw_chan_event, 1, 1, &chan->event); return nvkm_event_init(&nvkm_sw_chan_event, 1, 1, &chan->event);
} }

View File

@ -6,12 +6,22 @@
struct nvkm_sw_chan { struct nvkm_sw_chan {
struct nvkm_engctx base; struct nvkm_engctx base;
const struct nvkm_sw_chan_func *func;
struct nvkm_fifo_chan *fifo;
struct list_head head;
struct nvkm_event event; struct nvkm_event event;
}; };
#define nvkm_sw_context_create(p,e,c,d) \ struct nvkm_sw_chan_func {
nvkm_sw_chan_ctor((p), (e), (c), sizeof(**d), (void **)d) bool (*mthd)(struct nvkm_sw_chan *, int subc, u32 mthd, u32 data);
int nvkm_sw_chan_ctor(struct nvkm_object *, struct nvkm_object *, };
bool nvkm_sw_chan_mthd(struct nvkm_sw_chan *, int subc, u32 mthd, u32 data);
#define nvkm_sw_context_create(f,p,e,c,d) \
nvkm_sw_chan_ctor((f), (p), (e), (c), sizeof(**d), (void **)d)
int nvkm_sw_chan_ctor(const struct nvkm_sw_chan_func *,
struct nvkm_object *, struct nvkm_object *,
struct nvkm_oclass *, int, void **); struct nvkm_oclass *, int, void **);
void nvkm_sw_chan_dtor(struct nvkm_object *); void nvkm_sw_chan_dtor(struct nvkm_object *);
#define nvkm_sw_context_init(d) \ #define nvkm_sw_context_init(d) \

View File

@ -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,
}; };
/******************************************************************************* /*******************************************************************************

View File

@ -21,6 +21,7 @@
* *
* Authors: Ben Skeggs * Authors: Ben Skeggs
*/ */
#define nv04_sw_chan(p) container_of((p), struct nv04_sw_chan, base)
#include "priv.h" #include "priv.h"
#include "chan.h" #include "chan.h"
#include "nvsw.h" #include "nvsw.h"
@ -38,29 +39,6 @@ struct nv04_sw_chan {
* software object classes * software object classes
******************************************************************************/ ******************************************************************************/
static int
nv04_sw_set_ref(struct nvkm_object *object, u32 mthd, void *data, u32 size)
{
struct nv04_sw_chan *chan = (void *)nv_engctx(object->parent);
atomic_set(&chan->ref, *(u32*)data);
return 0;
}
static int
nv04_sw_flip(struct nvkm_object *object, u32 mthd, void *args, u32 size)
{
struct nvkm_sw_chan *chan = (void *)nv_engctx(object->parent);
nvkm_event_send(&chan->event, 1, 0, NULL, 0);
return 0;
}
static struct nvkm_omthds
nv04_sw_omthds[] = {
{ 0x0150, 0x0150, nv04_sw_set_ref },
{ 0x0500, 0x0500, nv04_sw_flip },
{}
};
static int static int
nv04_sw_mthd_get_ref(struct nvkm_object *object, void *data, u32 size) nv04_sw_mthd_get_ref(struct nvkm_object *object, void *data, u32 size)
{ {
@ -100,7 +78,7 @@ nv04_sw_ofuncs = {
static struct nvkm_oclass static struct nvkm_oclass
nv04_sw_sclass[] = { nv04_sw_sclass[] = {
{ NVIF_IOCTL_NEW_V0_SW_NV04, &nv04_sw_ofuncs, nv04_sw_omthds }, { NVIF_IOCTL_NEW_V0_SW_NV04, &nv04_sw_ofuncs },
{} {}
}; };
@ -108,6 +86,27 @@ nv04_sw_sclass[] = {
* software context * software context
******************************************************************************/ ******************************************************************************/
static bool
nv04_sw_chan_mthd(struct nvkm_sw_chan *base, int subc, u32 mthd, u32 data)
{
struct nv04_sw_chan *chan = nv04_sw_chan(base);
switch (mthd) {
case 0x0150:
atomic_set(&chan->ref, data);
return true;
default:
break;
}
return false;
}
static const struct nvkm_sw_chan_func
nv04_sw_chan_func = {
.mthd = nv04_sw_chan_mthd,
};
static int static int
nv04_sw_context_ctor(struct nvkm_object *parent, struct nvkm_object *engine, nv04_sw_context_ctor(struct nvkm_object *parent, struct nvkm_object *engine,
struct nvkm_oclass *oclass, void *data, u32 size, struct nvkm_oclass *oclass, void *data, u32 size,
@ -116,7 +115,8 @@ nv04_sw_context_ctor(struct nvkm_object *parent, struct nvkm_object *engine,
struct nv04_sw_chan *chan; struct nv04_sw_chan *chan;
int ret; int ret;
ret = nvkm_sw_context_create(parent, engine, oclass, &chan); ret = nvkm_sw_context_create(&nv04_sw_chan_func,
parent, engine, oclass, &chan);
*pobject = nv_object(chan); *pobject = nv_object(chan);
if (ret) if (ret)
return ret; return ret;

View File

@ -31,23 +31,9 @@
* software object classes * software object classes
******************************************************************************/ ******************************************************************************/
static int
nv10_sw_flip(struct nvkm_object *object, u32 mthd, void *args, u32 size)
{
struct nvkm_sw_chan *chan = (void *)nv_engctx(object->parent);
nvkm_event_send(&chan->event, 1, 0, NULL, 0);
return 0;
}
static struct nvkm_omthds
nv10_sw_omthds[] = {
{ 0x0500, 0x0500, nv10_sw_flip },
{}
};
static struct nvkm_oclass static struct nvkm_oclass
nv10_sw_sclass[] = { nv10_sw_sclass[] = {
{ NVIF_IOCTL_NEW_V0_SW_NV10, &nvkm_nvsw_ofuncs, nv10_sw_omthds }, { NVIF_IOCTL_NEW_V0_SW_NV10, &nvkm_nvsw_ofuncs },
{} {}
}; };
@ -55,6 +41,10 @@ nv10_sw_sclass[] = {
* software context * software context
******************************************************************************/ ******************************************************************************/
static const struct nvkm_sw_chan_func
nv10_sw_chan_func = {
};
static int static int
nv10_sw_context_ctor(struct nvkm_object *parent, struct nvkm_object *engine, nv10_sw_context_ctor(struct nvkm_object *parent, struct nvkm_object *engine,
struct nvkm_oclass *oclass, void *data, u32 size, struct nvkm_oclass *oclass, void *data, u32 size,
@ -63,7 +53,8 @@ nv10_sw_context_ctor(struct nvkm_object *parent, struct nvkm_object *engine,
struct nvkm_sw_chan *chan; struct nvkm_sw_chan *chan;
int ret; int ret;
ret = nvkm_sw_context_create(parent, engine, oclass, &chan); ret = nvkm_sw_context_create(&nv10_sw_chan_func,
parent, engine, oclass, &chan);
*pobject = nv_object(chan); *pobject = nv_object(chan);
if (ret) if (ret)
return ret; return ret;

View File

@ -35,80 +35,9 @@
* software object classes * software object classes
******************************************************************************/ ******************************************************************************/
static int
nv50_sw_mthd_dma_vblsem(struct nvkm_object *object, u32 mthd,
void *args, u32 size)
{
struct nv50_sw_chan *chan = (void *)nv_engctx(object->parent);
struct nvkm_fifo_chan *fifo = (void *)nv_object(chan)->parent;
struct nvkm_handle *handle;
int ret = -EINVAL;
handle = nvkm_namedb_get(nv_namedb(fifo), *(u32 *)args);
if (!handle)
return -ENOENT;
if (nv_iclass(handle->object, NV_GPUOBJ_CLASS)) {
struct nvkm_gpuobj *gpuobj = nv_gpuobj(handle->object);
chan->vblank.ctxdma = gpuobj->node->offset >> 4;
ret = 0;
}
nvkm_namedb_put(handle);
return ret;
}
static int
nv50_sw_mthd_vblsem_offset(struct nvkm_object *object, u32 mthd,
void *args, u32 size)
{
struct nv50_sw_chan *chan = (void *)nv_engctx(object->parent);
chan->vblank.offset = *(u32 *)args;
return 0;
}
int
nv50_sw_mthd_vblsem_value(struct nvkm_object *object, u32 mthd,
void *args, u32 size)
{
struct nv50_sw_chan *chan = (void *)nv_engctx(object->parent);
chan->vblank.value = *(u32 *)args;
return 0;
}
int
nv50_sw_mthd_vblsem_release(struct nvkm_object *object, u32 mthd,
void *args, u32 size)
{
struct nv50_sw_chan *chan = (void *)nv_engctx(object->parent);
u32 head = *(u32 *)args;
if (head >= nvkm_disp(chan)->vblank.index_nr)
return -EINVAL;
nvkm_notify_get(&chan->vblank.notify[head]);
return 0;
}
int
nv50_sw_mthd_flip(struct nvkm_object *object, u32 mthd, void *args, u32 size)
{
struct nv50_sw_chan *chan = (void *)nv_engctx(object->parent);
nvkm_event_send(&chan->base.event, 1, 0, NULL, 0);
return 0;
}
static struct nvkm_omthds
nv50_sw_omthds[] = {
{ 0x018c, 0x018c, nv50_sw_mthd_dma_vblsem },
{ 0x0400, 0x0400, nv50_sw_mthd_vblsem_offset },
{ 0x0404, 0x0404, nv50_sw_mthd_vblsem_value },
{ 0x0408, 0x0408, nv50_sw_mthd_vblsem_release },
{ 0x0500, 0x0500, nv50_sw_mthd_flip },
{}
};
static struct nvkm_oclass static struct nvkm_oclass
nv50_sw_sclass[] = { nv50_sw_sclass[] = {
{ NVIF_IOCTL_NEW_V0_SW_NV50, &nvkm_nvsw_ofuncs, nv50_sw_omthds }, { NVIF_IOCTL_NEW_V0_SW_NV50, &nvkm_nvsw_ofuncs },
{} {}
}; };
@ -140,6 +69,31 @@ nv50_sw_vblsem_release(struct nvkm_notify *notify)
return NVKM_NOTIFY_DROP; return NVKM_NOTIFY_DROP;
} }
static bool
nv50_sw_chan_mthd(struct nvkm_sw_chan *base, int subc, u32 mthd, u32 data)
{
struct nv50_sw_chan *chan = nv50_sw_chan(base);
switch (mthd) {
case 0x018c: chan->vblank.ctxdma = data; return true;
case 0x0400: chan->vblank.offset = data; return true;
case 0x0404: chan->vblank.value = data; return true;
case 0x0408:
if (data < nvkm_disp(chan)->vblank.index_nr) {
nvkm_notify_get(&chan->vblank.notify[data]);
return true;
}
break;
default:
break;
}
return false;
}
static const struct nvkm_sw_chan_func
nv50_sw_chan_func = {
.mthd = nv50_sw_chan_mthd,
};
void void
nv50_sw_context_dtor(struct nvkm_object *object) nv50_sw_context_dtor(struct nvkm_object *object)
{ {
@ -162,7 +116,7 @@ nv50_sw_context_ctor(struct nvkm_object *parent, struct nvkm_object *engine,
struct nv50_sw_chan *chan; struct nv50_sw_chan *chan;
int ret, i; int ret, i;
ret = nvkm_sw_context_create(parent, engine, oclass, &chan); ret = nvkm_sw_context_create(pclass->chan, parent, engine, oclass, &chan);
*pobject = nv_object(chan); *pobject = nv_object(chan);
if (ret) if (ret)
return ret; return ret;
@ -194,6 +148,7 @@ nv50_sw_cclass = {
.fini = _nvkm_sw_context_fini, .fini = _nvkm_sw_context_fini,
}, },
.vblank = nv50_sw_vblsem_release, .vblank = nv50_sw_vblsem_release,
.chan = &nv50_sw_chan_func,
}; };
/******************************************************************************* /*******************************************************************************

View File

@ -1,5 +1,6 @@
#ifndef __NVKM_SW_NV50_H__ #ifndef __NVKM_SW_NV50_H__
#define __NVKM_SW_NV50_H__ #define __NVKM_SW_NV50_H__
#define nv50_sw_chan(p) container_of((p), struct nv50_sw_chan, base)
#include "priv.h" #include "priv.h"
#include "chan.h" #include "chan.h"
#include "nvsw.h" #include "nvsw.h"
@ -18,6 +19,7 @@ int nv50_sw_ctor(struct nvkm_object *, struct nvkm_object *,
struct nv50_sw_cclass { struct nv50_sw_cclass {
struct nvkm_oclass base; struct nvkm_oclass base;
int (*vblank)(struct nvkm_notify *); int (*vblank)(struct nvkm_notify *);
const struct nvkm_sw_chan_func *chan;
}; };
struct nv50_sw_chan { struct nv50_sw_chan {