You've already forked linux-apfs
mirror of
https://github.com/linux-apfs/linux-apfs.git
synced 2026-05-01 15:00:59 -07:00
drm/nouveau: port to nvif client/device/objects
Signed-off-by: Ben Skeggs <bskeggs@redhat.com>
This commit is contained in:
@@ -338,6 +338,7 @@ nouveau-y += nouveau_drm.o nouveau_chan.o nouveau_dma.o nouveau_fence.o
|
||||
nouveau-y += nouveau_vga.o nouveau_agp.o
|
||||
nouveau-y += nouveau_ttm.o nouveau_sgdma.o nouveau_bo.o nouveau_gem.o
|
||||
nouveau-y += nouveau_prime.o nouveau_abi16.o
|
||||
nouveau-y += nouveau_nvif.o
|
||||
nouveau-y += nv04_fence.o nv10_fence.o nv17_fence.o
|
||||
nouveau-y += nv50_fence.o nv84_fence.o nvc0_fence.o
|
||||
|
||||
|
||||
@@ -70,6 +70,8 @@ nv04_display_create(struct drm_device *dev)
|
||||
if (!disp)
|
||||
return -ENOMEM;
|
||||
|
||||
nvif_object_map(nvif_object(&drm->device));
|
||||
|
||||
nouveau_display(dev)->priv = disp;
|
||||
nouveau_display(dev)->dtor = nv04_display_destroy;
|
||||
nouveau_display(dev)->init = nv04_display_init;
|
||||
@@ -144,6 +146,7 @@ void
|
||||
nv04_display_destroy(struct drm_device *dev)
|
||||
{
|
||||
struct nv04_display *disp = nv04_display(dev);
|
||||
struct nouveau_drm *drm = nouveau_drm(dev);
|
||||
struct drm_encoder *encoder;
|
||||
struct drm_crtc *crtc;
|
||||
|
||||
@@ -170,6 +173,8 @@ nv04_display_destroy(struct drm_device *dev)
|
||||
|
||||
nouveau_display(dev)->priv = NULL;
|
||||
kfree(disp);
|
||||
|
||||
nvif_object_unmap(nvif_object(&drm->device));
|
||||
}
|
||||
|
||||
int
|
||||
|
||||
@@ -27,9 +27,6 @@
|
||||
#include "hw.h"
|
||||
|
||||
#include <subdev/bios/pll.h>
|
||||
#include <subdev/fb.h>
|
||||
#include <subdev/clock.h>
|
||||
#include <subdev/timer.h>
|
||||
|
||||
#define CHIPSET_NFORCE 0x01a0
|
||||
#define CHIPSET_NFORCE2 0x01f0
|
||||
|
||||
@@ -35,8 +35,6 @@
|
||||
|
||||
#include <drm/i2c/ch7006.h>
|
||||
|
||||
#include <subdev/i2c.h>
|
||||
|
||||
static struct nouveau_i2c_board_info nv04_tv_encoder_info[] = {
|
||||
{
|
||||
{
|
||||
|
||||
@@ -37,7 +37,6 @@
|
||||
#include <core/device.h>
|
||||
|
||||
#include <subdev/bios/gpio.h>
|
||||
#include <subdev/gpio.h>
|
||||
|
||||
MODULE_PARM_DESC(tv_norm, "Default TV norm.\n"
|
||||
"\t\tSupported: PAL, PAL-M, PAL-N, PAL-Nc, NTSC-M, NTSC-J,\n"
|
||||
|
||||
@@ -27,47 +27,34 @@
|
||||
#include <core/class.h>
|
||||
#include <core/mm.h>
|
||||
|
||||
#include <subdev/fb.h>
|
||||
#include <subdev/timer.h>
|
||||
#include <subdev/instmem.h>
|
||||
#include <engine/graph.h>
|
||||
|
||||
#include "nouveau_drm.h"
|
||||
#include "nouveau_dma.h"
|
||||
#include "nouveau_gem.h"
|
||||
#include "nouveau_chan.h"
|
||||
#include "nouveau_abi16.h"
|
||||
|
||||
void nouveau_drm_hack_device(struct nouveau_drm *, struct nvif_device *);
|
||||
|
||||
struct nouveau_abi16 *
|
||||
nouveau_abi16_get(struct drm_file *file_priv, struct drm_device *dev)
|
||||
{
|
||||
struct nouveau_cli *cli = nouveau_cli(file_priv);
|
||||
struct nouveau_drm *drm = nouveau_drm(dev);
|
||||
mutex_lock(&cli->mutex);
|
||||
if (!cli->abi16) {
|
||||
struct nouveau_abi16 *abi16;
|
||||
cli->abi16 = abi16 = kzalloc(sizeof(*abi16), GFP_KERNEL);
|
||||
if (cli->abi16) {
|
||||
INIT_LIST_HEAD(&abi16->channels);
|
||||
abi16->client = nv_object(cli);
|
||||
|
||||
/* allocate device object targeting client's default
|
||||
* device (ie. the one that belongs to the fd it
|
||||
* opened)
|
||||
*/
|
||||
if (nouveau_object_new(abi16->client, NVDRM_CLIENT,
|
||||
NVDRM_DEVICE, 0x0080,
|
||||
&(struct nv_device_class) {
|
||||
if (nvif_device_init(&cli->base.base, NULL,
|
||||
NVDRM_DEVICE, NV_DEVICE_CLASS,
|
||||
&(struct nv_device_class) {
|
||||
.device = ~0ULL,
|
||||
},
|
||||
sizeof(struct nv_device_class),
|
||||
(struct nouveau_object **)
|
||||
&abi16->device.object) == 0) {
|
||||
nouveau_drm_hack_device(drm, &abi16->device);
|
||||
}, sizeof(struct nv_device_class),
|
||||
&abi16->device) == 0)
|
||||
return cli->abi16;
|
||||
}
|
||||
|
||||
kfree(cli->abi16);
|
||||
cli->abi16 = NULL;
|
||||
@@ -81,7 +68,7 @@ nouveau_abi16_get(struct drm_file *file_priv, struct drm_device *dev)
|
||||
int
|
||||
nouveau_abi16_put(struct nouveau_abi16 *abi16, int ret)
|
||||
{
|
||||
struct nouveau_cli *cli = (void *)abi16->client;
|
||||
struct nouveau_cli *cli = (void *)nvif_client(&abi16->device.base);
|
||||
mutex_unlock(&cli->mutex);
|
||||
return ret;
|
||||
}
|
||||
@@ -144,7 +131,7 @@ nouveau_abi16_chan_fini(struct nouveau_abi16 *abi16,
|
||||
|
||||
/* destroy channel object, all children will be killed too */
|
||||
if (chan->chan) {
|
||||
abi16->handles &= ~(1ULL << (chan->chan->handle & 0xffff));
|
||||
abi16->handles &= ~(1ULL << (chan->chan->object->handle & 0xffff));
|
||||
nouveau_channel_del(&chan->chan);
|
||||
}
|
||||
|
||||
@@ -155,7 +142,7 @@ nouveau_abi16_chan_fini(struct nouveau_abi16 *abi16,
|
||||
void
|
||||
nouveau_abi16_fini(struct nouveau_abi16 *abi16)
|
||||
{
|
||||
struct nouveau_cli *cli = (void *)abi16->client;
|
||||
struct nouveau_cli *cli = (void *)nvif_client(&abi16->device.base);
|
||||
struct nouveau_abi16_chan *chan, *temp;
|
||||
|
||||
/* cleanup channels */
|
||||
@@ -164,7 +151,7 @@ nouveau_abi16_fini(struct nouveau_abi16 *abi16)
|
||||
}
|
||||
|
||||
/* destroy the device object */
|
||||
nouveau_object_del(abi16->client, NVDRM_CLIENT, NVDRM_DEVICE);
|
||||
nvif_device_fini(&abi16->device);
|
||||
|
||||
kfree(cli->abi16);
|
||||
cli->abi16 = NULL;
|
||||
@@ -251,7 +238,6 @@ nouveau_abi16_ioctl_channel_alloc(ABI16_IOCTL_ARGS)
|
||||
struct nouveau_drm *drm = nouveau_drm(dev);
|
||||
struct nouveau_abi16 *abi16 = nouveau_abi16_get(file_priv, dev);
|
||||
struct nouveau_abi16_chan *chan;
|
||||
struct nouveau_client *client;
|
||||
struct nvif_device *device;
|
||||
struct nouveau_instmem *imem;
|
||||
struct nouveau_fb *pfb;
|
||||
@@ -263,7 +249,6 @@ nouveau_abi16_ioctl_channel_alloc(ABI16_IOCTL_ARGS)
|
||||
if (!drm->channel)
|
||||
return nouveau_abi16_put(abi16, -ENODEV);
|
||||
|
||||
client = nv_client(abi16->client);
|
||||
device = &abi16->device;
|
||||
imem = nvkm_instmem(device);
|
||||
pfb = nvkm_fb(device);
|
||||
@@ -298,8 +283,8 @@ nouveau_abi16_ioctl_channel_alloc(ABI16_IOCTL_ARGS)
|
||||
abi16->handles |= (1ULL << init->channel);
|
||||
|
||||
/* create channel object and initialise dma and fence management */
|
||||
ret = nouveau_channel_new(drm, cli, NVDRM_DEVICE, NVDRM_CHAN |
|
||||
init->channel, init->fb_ctxdma_handle,
|
||||
ret = nouveau_channel_new(drm, device, NVDRM_CHAN | init->channel,
|
||||
init->fb_ctxdma_handle,
|
||||
init->tt_ctxdma_handle, &chan->chan);
|
||||
if (ret)
|
||||
goto done;
|
||||
@@ -330,7 +315,7 @@ nouveau_abi16_ioctl_channel_alloc(ABI16_IOCTL_ARGS)
|
||||
goto done;
|
||||
|
||||
if (device->info.family >= NV_DEVICE_INFO_V0_TESLA) {
|
||||
ret = nouveau_bo_vma_add(chan->ntfy, client->vm,
|
||||
ret = nouveau_bo_vma_add(chan->ntfy, cli->vm,
|
||||
&chan->ntfy_vma);
|
||||
if (ret)
|
||||
goto done;
|
||||
@@ -361,7 +346,7 @@ nouveau_abi16_ioctl_channel_free(ABI16_IOCTL_ARGS)
|
||||
return -ENOMEM;
|
||||
|
||||
list_for_each_entry(chan, &abi16->channels, head) {
|
||||
if (chan->chan->handle == (NVDRM_CHAN | req->channel)) {
|
||||
if (chan->chan->object->handle == (NVDRM_CHAN | req->channel)) {
|
||||
nouveau_abi16_chan_fini(abi16, chan);
|
||||
return nouveau_abi16_put(abi16, 0);
|
||||
}
|
||||
@@ -392,8 +377,10 @@ nouveau_abi16_ioctl_grobj_alloc(ABI16_IOCTL_ARGS)
|
||||
return nouveau_abi16_put(abi16, 0);
|
||||
}
|
||||
|
||||
ret = nouveau_object_new(abi16->client, NVDRM_CHAN | init->channel,
|
||||
init->handle, init->class, NULL, 0, &object);
|
||||
/*XXX*/
|
||||
ret = nouveau_object_new(nv_object(nvkm_client(&abi16->device.base)),
|
||||
NVDRM_CHAN | init->channel, init->handle,
|
||||
init->class, NULL, 0, &object);
|
||||
return nouveau_abi16_put(abi16, ret);
|
||||
}
|
||||
|
||||
@@ -418,7 +405,7 @@ nouveau_abi16_ioctl_notifierobj_alloc(ABI16_IOCTL_ARGS)
|
||||
return nouveau_abi16_put(abi16, -EINVAL);
|
||||
|
||||
list_for_each_entry(temp, &abi16->channels, head) {
|
||||
if (temp->chan->handle == (NVDRM_CHAN | info->channel)) {
|
||||
if (temp->chan->object->handle == (NVDRM_CHAN | info->channel)) {
|
||||
chan = temp;
|
||||
break;
|
||||
}
|
||||
@@ -456,9 +443,11 @@ nouveau_abi16_ioctl_notifierobj_alloc(ABI16_IOCTL_ARGS)
|
||||
args.limit += chan->ntfy->bo.offset;
|
||||
}
|
||||
|
||||
ret = nouveau_object_new(abi16->client, chan->chan->handle,
|
||||
ntfy->handle, 0x003d, &args,
|
||||
sizeof(args), &object);
|
||||
/*XXX*/
|
||||
ret = nouveau_object_new(nv_object(nvkm_client(&abi16->device.base)),
|
||||
NVDRM_CHAN | info->channel, ntfy->handle,
|
||||
NV_DMA_IN_MEMORY_CLASS, &args, sizeof(args),
|
||||
&object);
|
||||
if (ret)
|
||||
goto done;
|
||||
|
||||
@@ -483,7 +472,7 @@ nouveau_abi16_ioctl_gpuobj_free(ABI16_IOCTL_ARGS)
|
||||
return -ENOMEM;
|
||||
|
||||
list_for_each_entry(temp, &abi16->channels, head) {
|
||||
if (temp->chan->handle == (NVDRM_CHAN | fini->channel)) {
|
||||
if (temp->chan->object->handle == (NVDRM_CHAN | fini->channel)) {
|
||||
chan = temp;
|
||||
break;
|
||||
}
|
||||
@@ -495,7 +484,9 @@ nouveau_abi16_ioctl_gpuobj_free(ABI16_IOCTL_ARGS)
|
||||
/* synchronize with the user channel and destroy the gpu object */
|
||||
nouveau_channel_idle(chan->chan);
|
||||
|
||||
ret = nouveau_object_del(abi16->client, chan->chan->handle, fini->handle);
|
||||
/*XXX*/
|
||||
ret = nouveau_object_del(nv_object(nvkm_client(&abi16->device.base)),
|
||||
chan->chan->object->handle, fini->handle);
|
||||
if (ret)
|
||||
return nouveau_abi16_put(abi16, ret);
|
||||
|
||||
|
||||
@@ -28,7 +28,6 @@ struct nouveau_abi16_chan {
|
||||
};
|
||||
|
||||
struct nouveau_abi16 {
|
||||
struct nouveau_object *client;
|
||||
struct nvif_device device;
|
||||
struct list_head channels;
|
||||
u64 handles;
|
||||
|
||||
@@ -22,8 +22,6 @@
|
||||
* SOFTWARE.
|
||||
*/
|
||||
|
||||
#include <subdev/bios.h>
|
||||
|
||||
#include <drm/drmP.h>
|
||||
|
||||
#include "nouveau_drm.h"
|
||||
|
||||
@@ -30,10 +30,6 @@
|
||||
#include <core/engine.h>
|
||||
#include <linux/swiotlb.h>
|
||||
|
||||
#include <subdev/fb.h>
|
||||
#include <subdev/vm.h>
|
||||
#include <subdev/bar.h>
|
||||
|
||||
#include "nouveau_drm.h"
|
||||
#include "nouveau_dma.h"
|
||||
#include "nouveau_fence.h"
|
||||
@@ -951,6 +947,7 @@ nouveau_bo_move_m2mf(struct ttm_buffer_object *bo, int evict, bool intr,
|
||||
{
|
||||
struct nouveau_drm *drm = nouveau_bdev(bo->bdev);
|
||||
struct nouveau_channel *chan = drm->ttm.chan;
|
||||
struct nouveau_cli *cli = (void *)nvif_client(&chan->device->base);
|
||||
struct nouveau_fence *fence;
|
||||
int ret;
|
||||
|
||||
@@ -964,7 +961,7 @@ nouveau_bo_move_m2mf(struct ttm_buffer_object *bo, int evict, bool intr,
|
||||
return ret;
|
||||
}
|
||||
|
||||
mutex_lock_nested(&chan->cli->mutex, SINGLE_DEPTH_NESTING);
|
||||
mutex_lock_nested(&cli->mutex, SINGLE_DEPTH_NESTING);
|
||||
ret = nouveau_fence_sync(bo->sync_obj, chan);
|
||||
if (ret == 0) {
|
||||
ret = drm->ttm.move(chan, bo, &bo->mem, new_mem);
|
||||
@@ -979,7 +976,7 @@ nouveau_bo_move_m2mf(struct ttm_buffer_object *bo, int evict, bool intr,
|
||||
}
|
||||
}
|
||||
}
|
||||
mutex_unlock(&chan->cli->mutex);
|
||||
mutex_unlock(&cli->mutex);
|
||||
return ret;
|
||||
}
|
||||
|
||||
@@ -1011,9 +1008,7 @@ nouveau_bo_move_init(struct nouveau_drm *drm)
|
||||
int ret;
|
||||
|
||||
do {
|
||||
struct nouveau_object *object;
|
||||
struct nouveau_channel *chan;
|
||||
u32 handle = (mthd->engine << 16) | mthd->oclass;
|
||||
|
||||
if (mthd->engine)
|
||||
chan = drm->cechan;
|
||||
@@ -1022,13 +1017,14 @@ nouveau_bo_move_init(struct nouveau_drm *drm)
|
||||
if (chan == NULL)
|
||||
continue;
|
||||
|
||||
ret = nouveau_object_new(nv_object(drm), chan->handle, handle,
|
||||
mthd->oclass, NULL, 0, &object);
|
||||
ret = nvif_object_init(chan->object, NULL,
|
||||
mthd->oclass | (mthd->engine << 16),
|
||||
mthd->oclass, NULL, 0,
|
||||
&drm->ttm.copy);
|
||||
if (ret == 0) {
|
||||
ret = mthd->init(chan, handle);
|
||||
ret = mthd->init(chan, drm->ttm.copy.handle);
|
||||
if (ret) {
|
||||
nouveau_object_del(nv_object(drm),
|
||||
chan->handle, handle);
|
||||
nvif_object_fini(&drm->ttm.copy);
|
||||
continue;
|
||||
}
|
||||
|
||||
|
||||
@@ -27,12 +27,6 @@
|
||||
#include <core/device.h>
|
||||
#include <core/class.h>
|
||||
|
||||
#include <subdev/fb.h>
|
||||
#include <subdev/vm.h>
|
||||
#include <subdev/instmem.h>
|
||||
|
||||
#include <engine/software.h>
|
||||
|
||||
#include "nouveau_drm.h"
|
||||
#include "nouveau_dma.h"
|
||||
#include "nouveau_bo.h"
|
||||
@@ -47,7 +41,7 @@ module_param_named(vram_pushbuf, nouveau_vram_pushbuf, int, 0400);
|
||||
int
|
||||
nouveau_channel_idle(struct nouveau_channel *chan)
|
||||
{
|
||||
struct nouveau_cli *cli = chan->cli;
|
||||
struct nouveau_cli *cli = (void *)nvif_client(chan->object);
|
||||
struct nouveau_fence *fence = NULL;
|
||||
int ret;
|
||||
|
||||
@@ -59,7 +53,7 @@ nouveau_channel_idle(struct nouveau_channel *chan)
|
||||
|
||||
if (ret)
|
||||
NV_PRINTK(error, cli, "failed to idle channel 0x%08x [%s]\n",
|
||||
chan->handle, cli->base.name);
|
||||
chan->object->handle, nvkm_client(&cli->base)->name);
|
||||
return ret;
|
||||
}
|
||||
|
||||
@@ -68,36 +62,36 @@ nouveau_channel_del(struct nouveau_channel **pchan)
|
||||
{
|
||||
struct nouveau_channel *chan = *pchan;
|
||||
if (chan) {
|
||||
struct nouveau_object *client = nv_object(chan->cli);
|
||||
if (chan->fence) {
|
||||
nouveau_channel_idle(chan);
|
||||
nouveau_fence(chan->drm)->context_del(chan);
|
||||
}
|
||||
nouveau_object_del(client, NVDRM_DEVICE, chan->handle);
|
||||
nouveau_object_del(client, NVDRM_DEVICE, chan->push.handle);
|
||||
nvif_object_fini(&chan->nvsw);
|
||||
nvif_object_fini(&chan->gart);
|
||||
nvif_object_fini(&chan->vram);
|
||||
nvif_object_ref(NULL, &chan->object);
|
||||
nvif_object_fini(&chan->push.ctxdma);
|
||||
nouveau_bo_vma_del(chan->push.buffer, &chan->push.vma);
|
||||
nouveau_bo_unmap(chan->push.buffer);
|
||||
if (chan->push.buffer && chan->push.buffer->pin_refcnt)
|
||||
nouveau_bo_unpin(chan->push.buffer);
|
||||
nouveau_bo_ref(NULL, &chan->push.buffer);
|
||||
nvif_device_ref(NULL, &chan->device);
|
||||
kfree(chan);
|
||||
}
|
||||
*pchan = NULL;
|
||||
}
|
||||
|
||||
static int
|
||||
nouveau_channel_prep(struct nouveau_drm *drm, struct nouveau_cli *cli,
|
||||
u32 parent, u32 handle, u32 size,
|
||||
struct nouveau_channel **pchan)
|
||||
nouveau_channel_prep(struct nouveau_drm *drm, struct nvif_device *device,
|
||||
u32 handle, u32 size, struct nouveau_channel **pchan)
|
||||
{
|
||||
struct nvif_device *device = &drm->device;
|
||||
struct nouveau_cli *cli = (void *)nvif_client(&device->base);
|
||||
struct nouveau_instmem *imem = nvkm_instmem(device);
|
||||
struct nouveau_vmmgr *vmm = nvkm_vmmgr(device);
|
||||
struct nouveau_fb *pfb = nvkm_fb(device);
|
||||
struct nouveau_client *client = &cli->base;
|
||||
struct nv_dma_class args = {};
|
||||
struct nouveau_channel *chan;
|
||||
struct nouveau_object *push;
|
||||
u32 target;
|
||||
int ret;
|
||||
|
||||
@@ -105,9 +99,8 @@ nouveau_channel_prep(struct nouveau_drm *drm, struct nouveau_cli *cli,
|
||||
if (!chan)
|
||||
return -ENOMEM;
|
||||
|
||||
chan->cli = cli;
|
||||
nvif_device_ref(device, &chan->device);
|
||||
chan->drm = drm;
|
||||
chan->handle = handle;
|
||||
|
||||
/* allocate memory for dma push buffer */
|
||||
target = TTM_PL_FLAG_TT;
|
||||
@@ -132,10 +125,9 @@ nouveau_channel_prep(struct nouveau_drm *drm, struct nouveau_cli *cli,
|
||||
* we be able to call out to other (indirect) push buffers
|
||||
*/
|
||||
chan->push.vma.offset = chan->push.buffer->bo.offset;
|
||||
chan->push.handle = NVDRM_PUSH | (handle & 0xffff);
|
||||
|
||||
if (device->info.family >= NV_DEVICE_INFO_V0_TESLA) {
|
||||
ret = nouveau_bo_vma_add(chan->push.buffer, client->vm,
|
||||
ret = nouveau_bo_vma_add(chan->push.buffer, cli->vm,
|
||||
&chan->push.vma);
|
||||
if (ret) {
|
||||
nouveau_channel_del(pchan);
|
||||
@@ -144,7 +136,7 @@ nouveau_channel_prep(struct nouveau_drm *drm, struct nouveau_cli *cli,
|
||||
|
||||
args.flags = NV_DMA_TARGET_VM | NV_DMA_ACCESS_VM;
|
||||
args.start = 0;
|
||||
args.limit = client->vm->vmm->limit - 1;
|
||||
args.limit = cli->vm->vmm->limit - 1;
|
||||
} else
|
||||
if (chan->push.buffer->bo.mem.mem_type == TTM_PL_VRAM) {
|
||||
u64 limit = pfb->ram->size - imem->reserved - 1;
|
||||
@@ -174,9 +166,9 @@ nouveau_channel_prep(struct nouveau_drm *drm, struct nouveau_cli *cli,
|
||||
}
|
||||
}
|
||||
|
||||
ret = nouveau_object_new(nv_object(chan->cli), parent,
|
||||
chan->push.handle, 0x0002,
|
||||
&args, sizeof(args), &push);
|
||||
ret = nvif_object_init(nvif_object(device), NULL, NVDRM_PUSH |
|
||||
(handle & 0xffff), NV_DMA_FROM_MEMORY_CLASS,
|
||||
&args, sizeof(args), &chan->push.ctxdma);
|
||||
if (ret) {
|
||||
nouveau_channel_del(pchan);
|
||||
return ret;
|
||||
@@ -186,9 +178,8 @@ nouveau_channel_prep(struct nouveau_drm *drm, struct nouveau_cli *cli,
|
||||
}
|
||||
|
||||
static int
|
||||
nouveau_channel_ind(struct nouveau_drm *drm, struct nouveau_cli *cli,
|
||||
u32 parent, u32 handle, u32 engine,
|
||||
struct nouveau_channel **pchan)
|
||||
nouveau_channel_ind(struct nouveau_drm *drm, struct nvif_device *device,
|
||||
u32 handle, u32 engine, struct nouveau_channel **pchan)
|
||||
{
|
||||
static const u16 oclasses[] = { NVE0_CHANNEL_IND_CLASS,
|
||||
NVC0_CHANNEL_IND_CLASS,
|
||||
@@ -201,22 +192,20 @@ nouveau_channel_ind(struct nouveau_drm *drm, struct nouveau_cli *cli,
|
||||
int ret;
|
||||
|
||||
/* allocate dma push buffer */
|
||||
ret = nouveau_channel_prep(drm, cli, parent, handle, 0x12000, &chan);
|
||||
ret = nouveau_channel_prep(drm, device, handle, 0x12000, &chan);
|
||||
*pchan = chan;
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
/* create channel object */
|
||||
args.pushbuf = chan->push.handle;
|
||||
args.pushbuf = chan->push.ctxdma.handle;
|
||||
args.ioffset = 0x10000 + chan->push.vma.offset;
|
||||
args.ilength = 0x02000;
|
||||
args.engine = engine;
|
||||
|
||||
do {
|
||||
ret = nouveau_object_new(nv_object(cli), parent, handle,
|
||||
*oclass++, &args, sizeof(args),
|
||||
(struct nouveau_object **)
|
||||
&chan->object);
|
||||
ret = nvif_object_new(nvif_object(device), handle, *oclass++,
|
||||
&args, sizeof(args), &chan->object);
|
||||
if (ret == 0)
|
||||
return ret;
|
||||
} while (*oclass);
|
||||
@@ -226,8 +215,8 @@ nouveau_channel_ind(struct nouveau_drm *drm, struct nouveau_cli *cli,
|
||||
}
|
||||
|
||||
static int
|
||||
nouveau_channel_dma(struct nouveau_drm *drm, struct nouveau_cli *cli,
|
||||
u32 parent, u32 handle, struct nouveau_channel **pchan)
|
||||
nouveau_channel_dma(struct nouveau_drm *drm, struct nvif_device *device,
|
||||
u32 handle, struct nouveau_channel **pchan)
|
||||
{
|
||||
static const u16 oclasses[] = { NV40_CHANNEL_DMA_CLASS,
|
||||
NV17_CHANNEL_DMA_CLASS,
|
||||
@@ -240,20 +229,18 @@ nouveau_channel_dma(struct nouveau_drm *drm, struct nouveau_cli *cli,
|
||||
int ret;
|
||||
|
||||
/* allocate dma push buffer */
|
||||
ret = nouveau_channel_prep(drm, cli, parent, handle, 0x10000, &chan);
|
||||
ret = nouveau_channel_prep(drm, device, handle, 0x10000, &chan);
|
||||
*pchan = chan;
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
/* create channel object */
|
||||
args.pushbuf = chan->push.handle;
|
||||
args.pushbuf = chan->push.ctxdma.handle;
|
||||
args.offset = chan->push.vma.offset;
|
||||
|
||||
do {
|
||||
ret = nouveau_object_new(nv_object(cli), parent, handle,
|
||||
*oclass++, &args, sizeof(args),
|
||||
(struct nouveau_object **)
|
||||
&chan->object);
|
||||
ret = nvif_object_new(nvif_object(device), handle, *oclass++,
|
||||
&args, sizeof(args), &chan->object);
|
||||
if (ret == 0)
|
||||
return ret;
|
||||
} while (ret && *oclass);
|
||||
@@ -265,13 +252,12 @@ nouveau_channel_dma(struct nouveau_drm *drm, struct nouveau_cli *cli,
|
||||
static int
|
||||
nouveau_channel_init(struct nouveau_channel *chan, u32 vram, u32 gart)
|
||||
{
|
||||
struct nouveau_client *client = nv_client(chan->cli);
|
||||
struct nvif_device *device = &chan->drm->device;
|
||||
struct nvif_device *device = chan->device;
|
||||
struct nouveau_cli *cli = (void *)nvif_client(&device->base);
|
||||
struct nouveau_instmem *imem = nvkm_instmem(device);
|
||||
struct nouveau_vmmgr *vmm = nvkm_vmmgr(device);
|
||||
struct nouveau_fb *pfb = nvkm_fb(device);
|
||||
struct nouveau_software_chan *swch;
|
||||
struct nouveau_object *object;
|
||||
struct nv_dma_class args = {};
|
||||
int ret, i;
|
||||
|
||||
@@ -280,22 +266,23 @@ nouveau_channel_init(struct nouveau_channel *chan, u32 vram, u32 gart)
|
||||
if (device->info.family >= NV_DEVICE_INFO_V0_TESLA) {
|
||||
args.flags = NV_DMA_TARGET_VM | NV_DMA_ACCESS_VM;
|
||||
args.start = 0;
|
||||
args.limit = client->vm->vmm->limit - 1;
|
||||
args.limit = cli->vm->vmm->limit - 1;
|
||||
} else {
|
||||
args.flags = NV_DMA_TARGET_VRAM | NV_DMA_ACCESS_RDWR;
|
||||
args.start = 0;
|
||||
args.limit = pfb->ram->size - imem->reserved - 1;
|
||||
}
|
||||
|
||||
ret = nouveau_object_new(nv_object(client), chan->handle, vram,
|
||||
0x003d, &args, sizeof(args), &object);
|
||||
ret = nvif_object_init(chan->object, NULL, vram,
|
||||
NV_DMA_IN_MEMORY_CLASS, &args,
|
||||
sizeof(args), &chan->vram);
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
if (device->info.family >= NV_DEVICE_INFO_V0_TESLA) {
|
||||
args.flags = NV_DMA_TARGET_VM | NV_DMA_ACCESS_VM;
|
||||
args.start = 0;
|
||||
args.limit = client->vm->vmm->limit - 1;
|
||||
args.limit = cli->vm->vmm->limit - 1;
|
||||
} else
|
||||
if (chan->drm->agp.stat == ENABLED) {
|
||||
args.flags = NV_DMA_TARGET_AGP | NV_DMA_ACCESS_RDWR;
|
||||
@@ -308,17 +295,15 @@ nouveau_channel_init(struct nouveau_channel *chan, u32 vram, u32 gart)
|
||||
args.limit = vmm->limit - 1;
|
||||
}
|
||||
|
||||
ret = nouveau_object_new(nv_object(client), chan->handle, gart,
|
||||
0x003d, &args, sizeof(args), &object);
|
||||
ret = nvif_object_init(chan->object, NULL, gart,
|
||||
NV_DMA_IN_MEMORY_CLASS, &args,
|
||||
sizeof(args), &chan->gart);
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
chan->vram = vram;
|
||||
chan->gart = gart;
|
||||
}
|
||||
|
||||
/* initialise dma tracking parameters */
|
||||
switch (nv_hclass(chan->object) & 0x00ff) {
|
||||
switch (chan->object->oclass & 0x00ff) {
|
||||
case 0x006b:
|
||||
case 0x006e:
|
||||
chan->user_put = 0x40;
|
||||
@@ -350,12 +335,12 @@ nouveau_channel_init(struct nouveau_channel *chan, u32 vram, u32 gart)
|
||||
|
||||
/* allocate software object class (used for fences on <= nv05) */
|
||||
if (device->info.family < NV_DEVICE_INFO_V0_CELSIUS) {
|
||||
ret = nouveau_object_new(nv_object(client), chan->handle,
|
||||
NvSw, 0x006e, NULL, 0, &object);
|
||||
ret = nvif_object_init(chan->object, NULL, NvSw, 0x006e,
|
||||
NULL, 0, &chan->nvsw);
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
swch = (void *)object->parent;
|
||||
swch = (void *)nvkm_object(&chan->nvsw)->parent;
|
||||
swch->flip = nouveau_flip_complete;
|
||||
swch->flip_data = chan;
|
||||
|
||||
@@ -373,16 +358,17 @@ nouveau_channel_init(struct nouveau_channel *chan, u32 vram, u32 gart)
|
||||
}
|
||||
|
||||
int
|
||||
nouveau_channel_new(struct nouveau_drm *drm, struct nouveau_cli *cli,
|
||||
u32 parent, u32 handle, u32 arg0, u32 arg1,
|
||||
nouveau_channel_new(struct nouveau_drm *drm, struct nvif_device *device,
|
||||
u32 handle, u32 arg0, u32 arg1,
|
||||
struct nouveau_channel **pchan)
|
||||
{
|
||||
struct nouveau_cli *cli = (void *)nvif_client(&device->base);
|
||||
int ret;
|
||||
|
||||
ret = nouveau_channel_ind(drm, cli, parent, handle, arg0, pchan);
|
||||
ret = nouveau_channel_ind(drm, device, handle, arg0, pchan);
|
||||
if (ret) {
|
||||
NV_PRINTK(debug, cli, "ib channel create, %d\n", ret);
|
||||
ret = nouveau_channel_dma(drm, cli, parent, handle, pchan);
|
||||
ret = nouveau_channel_dma(drm, device, handle, pchan);
|
||||
if (ret) {
|
||||
NV_PRINTK(debug, cli, "dma channel create, %d\n", ret);
|
||||
return ret;
|
||||
|
||||
@@ -1,20 +1,21 @@
|
||||
#ifndef __NOUVEAU_CHAN_H__
|
||||
#define __NOUVEAU_CHAN_H__
|
||||
|
||||
struct nouveau_cli;
|
||||
#include <nvif/object.h>
|
||||
struct nvif_device;
|
||||
|
||||
struct nouveau_channel {
|
||||
struct nouveau_cli *cli;
|
||||
struct nvif_device *device;
|
||||
struct nouveau_drm *drm;
|
||||
|
||||
u32 handle;
|
||||
u32 vram;
|
||||
u32 gart;
|
||||
struct nvif_object vram;
|
||||
struct nvif_object gart;
|
||||
struct nvif_object nvsw;
|
||||
|
||||
struct {
|
||||
struct nouveau_bo *buffer;
|
||||
struct nouveau_vma vma;
|
||||
u32 handle;
|
||||
struct nvif_object ctxdma;
|
||||
} push;
|
||||
|
||||
/* TODO: this will be reworked in the near future */
|
||||
@@ -38,8 +39,8 @@ struct nouveau_channel {
|
||||
};
|
||||
|
||||
|
||||
int nouveau_channel_new(struct nouveau_drm *, struct nouveau_cli *,
|
||||
u32 parent, u32 handle, u32 arg0, u32 arg1,
|
||||
int nouveau_channel_new(struct nouveau_drm *, struct nvif_device *,
|
||||
u32 handle, u32 arg0, u32 arg1,
|
||||
struct nouveau_channel **);
|
||||
void nouveau_channel_del(struct nouveau_channel **);
|
||||
int nouveau_channel_idle(struct nouveau_channel *);
|
||||
|
||||
@@ -42,10 +42,6 @@
|
||||
#include "nouveau_encoder.h"
|
||||
#include "nouveau_crtc.h"
|
||||
|
||||
#include <subdev/i2c.h>
|
||||
#include <subdev/gpio.h>
|
||||
#include <engine/disp.h>
|
||||
|
||||
#include <nvif/event.h>
|
||||
|
||||
MODULE_PARM_DESC(tv_disable, "Disable TV-out detection");
|
||||
|
||||
@@ -37,8 +37,6 @@
|
||||
|
||||
#include "nouveau_fence.h"
|
||||
|
||||
#include <engine/disp.h>
|
||||
|
||||
#include <core/class.h>
|
||||
#include <nvif/event.h>
|
||||
|
||||
@@ -102,7 +100,7 @@ nouveau_display_scanoutpos_head(struct drm_crtc *crtc, int *vpos, int *hpos,
|
||||
int ret, retry = 1;
|
||||
|
||||
do {
|
||||
ret = nv_exec(disp->core, mthd, &args, sizeof(args));
|
||||
ret = nvif_exec(&disp->disp, mthd, &args, sizeof(args));
|
||||
if (ret != 0)
|
||||
return 0;
|
||||
|
||||
@@ -399,10 +397,10 @@ nouveau_display_create_properties(struct drm_device *dev)
|
||||
struct nouveau_display *disp = nouveau_display(dev);
|
||||
int gen;
|
||||
|
||||
if (nv_mclass(disp->core) < NV50_DISP_CLASS)
|
||||
if (disp->disp.oclass < NV50_DISP_CLASS)
|
||||
gen = 0;
|
||||
else
|
||||
if (nv_mclass(disp->core) < NVD0_DISP_CLASS)
|
||||
if (disp->disp.oclass < NVD0_DISP_CLASS)
|
||||
gen = 1;
|
||||
else
|
||||
gen = 2;
|
||||
@@ -488,14 +486,14 @@ nouveau_display_create(struct drm_device *dev)
|
||||
int i;
|
||||
|
||||
for (i = 0, ret = -ENODEV; ret && i < ARRAY_SIZE(oclass); i++) {
|
||||
ret = nouveau_object_new(nv_object(drm), NVDRM_DEVICE,
|
||||
NVDRM_DISPLAY, oclass[i],
|
||||
NULL, 0, &disp->core);
|
||||
ret = nvif_object_init(nvif_object(&drm->device), NULL,
|
||||
NVDRM_DISPLAY, oclass[i],
|
||||
NULL, 0, &disp->disp);
|
||||
}
|
||||
|
||||
if (ret == 0) {
|
||||
nouveau_display_create_properties(dev);
|
||||
if (nv_mclass(disp->core) < NV50_DISP_CLASS)
|
||||
if (disp->disp.oclass < NV50_DISP_CLASS)
|
||||
ret = nv04_display_create(dev);
|
||||
else
|
||||
ret = nv50_display_create(dev);
|
||||
@@ -528,7 +526,6 @@ void
|
||||
nouveau_display_destroy(struct drm_device *dev)
|
||||
{
|
||||
struct nouveau_display *disp = nouveau_display(dev);
|
||||
struct nouveau_drm *drm = nouveau_drm(dev);
|
||||
|
||||
nouveau_backlight_exit(dev);
|
||||
nouveau_display_vblank_fini(dev);
|
||||
@@ -539,7 +536,7 @@ nouveau_display_destroy(struct drm_device *dev)
|
||||
if (disp->dtor)
|
||||
disp->dtor(dev);
|
||||
|
||||
nouveau_object_del(nv_object(drm), NVDRM_DEVICE, NVDRM_DISPLAY);
|
||||
nvif_object_fini(&disp->disp);
|
||||
|
||||
nouveau_drm(dev)->display = NULL;
|
||||
kfree(disp);
|
||||
@@ -690,12 +687,15 @@ nouveau_crtc_page_flip(struct drm_crtc *crtc, struct drm_framebuffer *fb,
|
||||
struct nouveau_bo *old_bo = nouveau_framebuffer(crtc->primary->fb)->nvbo;
|
||||
struct nouveau_bo *new_bo = nouveau_framebuffer(fb)->nvbo;
|
||||
struct nouveau_page_flip_state *s;
|
||||
struct nouveau_channel *chan = drm->channel;
|
||||
struct nouveau_channel *chan;
|
||||
struct nouveau_cli *cli;
|
||||
struct nouveau_fence *fence;
|
||||
int ret;
|
||||
|
||||
if (!drm->channel)
|
||||
chan = drm->channel;
|
||||
if (!chan)
|
||||
return -ENODEV;
|
||||
cli = (void *)nvif_client(&chan->device->base);
|
||||
|
||||
s = kzalloc(sizeof(*s), GFP_KERNEL);
|
||||
if (!s)
|
||||
@@ -707,7 +707,7 @@ nouveau_crtc_page_flip(struct drm_crtc *crtc, struct drm_framebuffer *fb,
|
||||
goto fail_free;
|
||||
}
|
||||
|
||||
mutex_lock(&chan->cli->mutex);
|
||||
mutex_lock(&cli->mutex);
|
||||
|
||||
/* synchronise rendering channel with the kernel's channel */
|
||||
spin_lock(&new_bo->bo.bdev->fence_lock);
|
||||
@@ -761,7 +761,7 @@ nouveau_crtc_page_flip(struct drm_crtc *crtc, struct drm_framebuffer *fb,
|
||||
ret = nouveau_page_flip_emit(chan, old_bo, new_bo, s, &fence);
|
||||
if (ret)
|
||||
goto fail_unreserve;
|
||||
mutex_unlock(&chan->cli->mutex);
|
||||
mutex_unlock(&cli->mutex);
|
||||
|
||||
/* Update the crtc struct and cleanup */
|
||||
crtc->primary->fb = fb;
|
||||
@@ -777,7 +777,7 @@ fail_unreserve:
|
||||
drm_vblank_put(dev, nouveau_crtc(crtc)->index);
|
||||
ttm_bo_unreserve(&old_bo->bo);
|
||||
fail_unpin:
|
||||
mutex_unlock(&chan->cli->mutex);
|
||||
mutex_unlock(&cli->mutex);
|
||||
if (old_bo != new_bo)
|
||||
nouveau_bo_unpin(new_bo);
|
||||
fail_free:
|
||||
|
||||
@@ -12,6 +12,8 @@ struct nouveau_framebuffer {
|
||||
u32 r_handle;
|
||||
u32 r_format;
|
||||
u32 r_pitch;
|
||||
struct nvif_object h_base[4];
|
||||
struct nvif_object h_core;
|
||||
};
|
||||
|
||||
static inline struct nouveau_framebuffer *
|
||||
@@ -39,7 +41,7 @@ struct nouveau_display {
|
||||
int (*fb_ctor)(struct drm_framebuffer *);
|
||||
void (*fb_dtor)(struct drm_framebuffer *);
|
||||
|
||||
struct nouveau_object *core;
|
||||
struct nvif_object disp;
|
||||
|
||||
struct drm_property *dithering_mode;
|
||||
struct drm_property *dithering_depth;
|
||||
|
||||
@@ -84,12 +84,13 @@ void
|
||||
nv50_dma_push(struct nouveau_channel *chan, struct nouveau_bo *bo,
|
||||
int delta, int length)
|
||||
{
|
||||
struct nouveau_cli *cli = (void *)nvif_client(&chan->device->base);
|
||||
struct nouveau_bo *pb = chan->push.buffer;
|
||||
struct nouveau_vma *vma;
|
||||
int ip = (chan->dma.ib_put * 2) + chan->dma.ib_base;
|
||||
u64 offset;
|
||||
|
||||
vma = nouveau_bo_vma_find(bo, nv_client(chan->cli)->vm);
|
||||
vma = nouveau_bo_vma_find(bo, cli->vm);
|
||||
BUG_ON(!vma);
|
||||
offset = vma->offset + delta;
|
||||
|
||||
|
||||
@@ -32,9 +32,6 @@
|
||||
|
||||
#include <core/class.h>
|
||||
|
||||
#include <subdev/gpio.h>
|
||||
#include <subdev/i2c.h>
|
||||
|
||||
static void
|
||||
nouveau_dp_probe_oui(struct drm_device *dev, struct nouveau_i2c_port *auxch,
|
||||
u8 *dpcd)
|
||||
|
||||
@@ -30,18 +30,10 @@
|
||||
#include "drmP.h"
|
||||
#include "drm_crtc_helper.h"
|
||||
#include <core/device.h>
|
||||
#include <core/client.h>
|
||||
#include <core/gpuobj.h>
|
||||
#include <core/class.h>
|
||||
#include <core/option.h>
|
||||
|
||||
#include <engine/device.h>
|
||||
#include <engine/disp.h>
|
||||
#include <engine/fifo.h>
|
||||
#include <engine/software.h>
|
||||
|
||||
#include <subdev/vm.h>
|
||||
|
||||
#include "nouveau_drm.h"
|
||||
#include "nouveau_dma.h"
|
||||
#include "nouveau_ttm.h"
|
||||
@@ -109,40 +101,34 @@ static int
|
||||
nouveau_cli_create(u64 name, const char *sname,
|
||||
int size, void **pcli)
|
||||
{
|
||||
struct nouveau_cli *cli;
|
||||
int ret;
|
||||
|
||||
*pcli = NULL;
|
||||
ret = nouveau_client_create_(sname, name, nouveau_config,
|
||||
nouveau_debug, size, pcli);
|
||||
cli = *pcli;
|
||||
if (ret) {
|
||||
if (cli)
|
||||
nouveau_client_destroy(&cli->base);
|
||||
*pcli = NULL;
|
||||
struct nouveau_cli *cli = *pcli = kzalloc(size, GFP_KERNEL);
|
||||
if (cli) {
|
||||
int ret = nvif_client_init(NULL, NULL, sname, name,
|
||||
nouveau_config, nouveau_debug,
|
||||
&cli->base);
|
||||
if (ret == 0)
|
||||
mutex_init(&cli->mutex);
|
||||
return ret;
|
||||
}
|
||||
|
||||
mutex_init(&cli->mutex);
|
||||
return 0;
|
||||
return -ENOMEM;
|
||||
}
|
||||
|
||||
static void
|
||||
nouveau_cli_destroy(struct nouveau_cli *cli)
|
||||
{
|
||||
struct nouveau_object *client = nv_object(cli);
|
||||
nouveau_vm_ref(NULL, &cli->base.vm, NULL);
|
||||
nouveau_client_fini(&cli->base, false);
|
||||
atomic_set(&client->refcount, 1);
|
||||
nouveau_object_ref(NULL, &client);
|
||||
nouveau_vm_ref(NULL, &nvkm_client(&cli->base)->vm, NULL);
|
||||
nvif_client_fini(&cli->base);
|
||||
}
|
||||
|
||||
static void
|
||||
nouveau_accel_fini(struct nouveau_drm *drm)
|
||||
{
|
||||
nouveau_gpuobj_ref(NULL, &drm->notify);
|
||||
nouveau_channel_del(&drm->channel);
|
||||
nvif_object_fini(&drm->ntfy);
|
||||
nouveau_gpuobj_ref(NULL, &drm->notify);
|
||||
nvif_object_fini(&drm->nvsw);
|
||||
nouveau_channel_del(&drm->cechan);
|
||||
nvif_object_fini(&drm->ttm.copy);
|
||||
if (drm->fence)
|
||||
nouveau_fence(drm)->dtor(drm);
|
||||
}
|
||||
@@ -151,7 +137,6 @@ static void
|
||||
nouveau_accel_init(struct nouveau_drm *drm)
|
||||
{
|
||||
struct nvif_device *device = &drm->device;
|
||||
struct nouveau_object *object;
|
||||
u32 arg0, arg1;
|
||||
u32 sclass[16];
|
||||
int ret, i;
|
||||
@@ -163,8 +148,7 @@ nouveau_accel_init(struct nouveau_drm *drm)
|
||||
/*XXX: this is crap, but the fence/channel stuff is a little
|
||||
* backwards in some places. this will be fixed.
|
||||
*/
|
||||
ret = nouveau_parent_lclass(nvkm_object(device), sclass,
|
||||
ARRAY_SIZE(sclass));
|
||||
ret = nvif_object_sclass(&device->base, sclass, ARRAY_SIZE(sclass));
|
||||
if (ret < 0)
|
||||
return;
|
||||
|
||||
@@ -202,8 +186,7 @@ nouveau_accel_init(struct nouveau_drm *drm)
|
||||
}
|
||||
|
||||
if (device->info.family >= NV_DEVICE_INFO_V0_KEPLER) {
|
||||
ret = nouveau_channel_new(drm, &drm->client, NVDRM_DEVICE,
|
||||
NVDRM_CHAN + 1,
|
||||
ret = nouveau_channel_new(drm, &drm->device, NVDRM_CHAN + 1,
|
||||
NVE0_CHANNEL_IND_ENGINE_CE0 |
|
||||
NVE0_CHANNEL_IND_ENGINE_CE1, 0,
|
||||
&drm->cechan);
|
||||
@@ -216,9 +199,8 @@ nouveau_accel_init(struct nouveau_drm *drm)
|
||||
if (device->info.chipset >= 0xa3 &&
|
||||
device->info.chipset != 0xaa &&
|
||||
device->info.chipset != 0xac) {
|
||||
ret = nouveau_channel_new(drm, &drm->client, NVDRM_DEVICE,
|
||||
NVDRM_CHAN + 1, NvDmaFB, NvDmaTT,
|
||||
&drm->cechan);
|
||||
ret = nouveau_channel_new(drm, &drm->device, NVDRM_CHAN + 1,
|
||||
NvDmaFB, NvDmaTT, &drm->cechan);
|
||||
if (ret)
|
||||
NV_ERROR(drm, "failed to create ce channel, %d\n", ret);
|
||||
|
||||
@@ -229,18 +211,18 @@ nouveau_accel_init(struct nouveau_drm *drm)
|
||||
arg1 = NvDmaTT;
|
||||
}
|
||||
|
||||
ret = nouveau_channel_new(drm, &drm->client, NVDRM_DEVICE, NVDRM_CHAN,
|
||||
arg0, arg1, &drm->channel);
|
||||
ret = nouveau_channel_new(drm, &drm->device, NVDRM_CHAN, arg0, arg1,
|
||||
&drm->channel);
|
||||
if (ret) {
|
||||
NV_ERROR(drm, "failed to create kernel channel, %d\n", ret);
|
||||
nouveau_accel_fini(drm);
|
||||
return;
|
||||
}
|
||||
|
||||
ret = nouveau_object_new(nv_object(drm), NVDRM_CHAN, NVDRM_NVSW,
|
||||
nouveau_abi16_swclass(drm), NULL, 0, &object);
|
||||
ret = nvif_object_init(drm->channel->object, NULL, NVDRM_NVSW,
|
||||
nouveau_abi16_swclass(drm), NULL, 0, &drm->nvsw);
|
||||
if (ret == 0) {
|
||||
struct nouveau_software_chan *swch = (void *)object->parent;
|
||||
struct nouveau_software_chan *swch;
|
||||
ret = RING_SPACE(drm->channel, 2);
|
||||
if (ret == 0) {
|
||||
if (device->info.family < NV_DEVICE_INFO_V0_FERMI) {
|
||||
@@ -252,7 +234,7 @@ nouveau_accel_init(struct nouveau_drm *drm)
|
||||
OUT_RING (drm->channel, 0x001f0000);
|
||||
}
|
||||
}
|
||||
swch = (void *)object->parent;
|
||||
swch = (void *)nvkm_object(&drm->nvsw)->parent;
|
||||
swch->flip = nouveau_flip_complete;
|
||||
swch->flip_data = drm->channel;
|
||||
}
|
||||
@@ -272,15 +254,15 @@ nouveau_accel_init(struct nouveau_drm *drm)
|
||||
return;
|
||||
}
|
||||
|
||||
ret = nouveau_object_new(nv_object(drm),
|
||||
drm->channel->handle, NvNotify0,
|
||||
0x003d, &(struct nv_dma_class) {
|
||||
ret = nvif_object_init(drm->channel->object, NULL, NvNotify0,
|
||||
NV_DMA_IN_MEMORY_CLASS,
|
||||
&(struct nv_dma_class) {
|
||||
.flags = NV_DMA_TARGET_VRAM |
|
||||
NV_DMA_ACCESS_RDWR,
|
||||
.start = drm->notify->addr,
|
||||
.limit = drm->notify->addr + 31
|
||||
}, sizeof(struct nv_dma_class),
|
||||
&object);
|
||||
}, sizeof(struct nv_dma_class),
|
||||
&drm->ntfy);
|
||||
if (ret) {
|
||||
nouveau_accel_fini(drm);
|
||||
return;
|
||||
@@ -373,27 +355,6 @@ nouveau_get_hdmi_dev(struct nouveau_drm *drm)
|
||||
}
|
||||
}
|
||||
|
||||
void
|
||||
nouveau_drm_hack_device(struct nouveau_drm *drm, struct nvif_device *device)
|
||||
{
|
||||
drm->device.info.chipset = nvkm_device(&drm->device)->chipset;
|
||||
switch (nvkm_device(&drm->device)->card_type) {
|
||||
case NV_04: device->info.family = NV_DEVICE_INFO_V0_TNT; break;
|
||||
case NV_10: device->info.family = NV_DEVICE_INFO_V0_CELSIUS; break;
|
||||
case NV_11: device->info.family = NV_DEVICE_INFO_V0_CELSIUS; break;
|
||||
case NV_20: device->info.family = NV_DEVICE_INFO_V0_KELVIN; break;
|
||||
case NV_30: device->info.family = NV_DEVICE_INFO_V0_RANKINE; break;
|
||||
case NV_40: device->info.family = NV_DEVICE_INFO_V0_CURIE; break;
|
||||
case NV_50: device->info.family = NV_DEVICE_INFO_V0_TESLA; break;
|
||||
case NV_C0: device->info.family = NV_DEVICE_INFO_V0_FERMI; break;
|
||||
case NV_E0: device->info.family = NV_DEVICE_INFO_V0_KEPLER; break;
|
||||
case GM100: device->info.family = NV_DEVICE_INFO_V0_MAXWELL; break;
|
||||
default:
|
||||
BUG_ON(1);
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
static int
|
||||
nouveau_drm_load(struct drm_device *dev, unsigned long flags)
|
||||
{
|
||||
@@ -408,7 +369,8 @@ nouveau_drm_load(struct drm_device *dev, unsigned long flags)
|
||||
|
||||
dev->dev_private = drm;
|
||||
drm->dev = dev;
|
||||
nouveau_client(drm)->debug = nouveau_dbgopt(nouveau_debug, "DRM");
|
||||
nvkm_client(&drm->client.base)->debug =
|
||||
nouveau_dbgopt(nouveau_debug, "DRM");
|
||||
|
||||
INIT_LIST_HEAD(&drm->clients);
|
||||
spin_lock_init(&drm->tile.lock);
|
||||
@@ -422,39 +384,34 @@ nouveau_drm_load(struct drm_device *dev, unsigned long flags)
|
||||
/* dummy device object, doesn't init anything, but allows
|
||||
* agp code access to registers
|
||||
*/
|
||||
ret = nouveau_object_new(nv_object(drm), NVDRM_CLIENT,
|
||||
NVDRM_DEVICE, 0x0080,
|
||||
&(struct nv_device_class) {
|
||||
ret = nvif_device_init(&drm->client.base.base, NULL,
|
||||
NVDRM_DEVICE, NV_DEVICE_CLASS,
|
||||
&(struct nv_device_class) {
|
||||
.device = ~0,
|
||||
.disable =
|
||||
~(NV_DEVICE_DISABLE_MMIO |
|
||||
NV_DEVICE_DISABLE_IDENTIFY),
|
||||
.debug0 = ~0,
|
||||
}, sizeof(struct nv_device_class),
|
||||
(struct nouveau_object **)
|
||||
&drm->device.object);
|
||||
}, sizeof(struct nv_device_class),
|
||||
&drm->device);
|
||||
if (ret)
|
||||
goto fail_device;
|
||||
|
||||
nouveau_drm_hack_device(drm, &drm->device);
|
||||
|
||||
nouveau_agp_reset(drm);
|
||||
nouveau_object_del(nv_object(drm), NVDRM_CLIENT, NVDRM_DEVICE);
|
||||
nvif_device_fini(&drm->device);
|
||||
}
|
||||
|
||||
ret = nouveau_object_new(nv_object(drm), NVDRM_CLIENT, NVDRM_DEVICE,
|
||||
0x0080, &(struct nv_device_class) {
|
||||
ret = nvif_device_init(&drm->client.base.base, NULL, NVDRM_DEVICE,
|
||||
NV_DEVICE_CLASS,
|
||||
&(struct nv_device_class) {
|
||||
.device = ~0,
|
||||
.disable = 0,
|
||||
.debug0 = 0,
|
||||
}, sizeof(struct nv_device_class),
|
||||
(struct nouveau_object **)
|
||||
&drm->device.object);
|
||||
}, sizeof(struct nv_device_class),
|
||||
&drm->device);
|
||||
if (ret)
|
||||
goto fail_device;
|
||||
|
||||
nouveau_drm_hack_device(drm, &drm->device);
|
||||
|
||||
dev->irq_enabled = true;
|
||||
|
||||
/* workaround an odd issue on nvc1 by disabling the device's
|
||||
@@ -473,7 +430,7 @@ nouveau_drm_load(struct drm_device *dev, unsigned long flags)
|
||||
if (ret)
|
||||
goto fail_device;
|
||||
|
||||
drm->client.base.vm = drm->client.vm;
|
||||
nvkm_client(&drm->client.base)->vm = drm->client.vm;
|
||||
}
|
||||
|
||||
ret = nouveau_ttm_init(drm);
|
||||
@@ -519,6 +476,7 @@ fail_ttm:
|
||||
nouveau_agp_fini(drm);
|
||||
nouveau_vga_fini(drm);
|
||||
fail_device:
|
||||
nvif_device_fini(&drm->device);
|
||||
nouveau_cli_destroy(&drm->client);
|
||||
return ret;
|
||||
}
|
||||
@@ -544,6 +502,7 @@ nouveau_drm_unload(struct drm_device *dev)
|
||||
nouveau_agp_fini(drm);
|
||||
nouveau_vga_fini(drm);
|
||||
|
||||
nvif_device_fini(&drm->device);
|
||||
if (drm->hdmi_device)
|
||||
pci_dev_put(drm->hdmi_device);
|
||||
nouveau_cli_destroy(&drm->client);
|
||||
@@ -554,10 +513,12 @@ void
|
||||
nouveau_drm_device_remove(struct drm_device *dev)
|
||||
{
|
||||
struct nouveau_drm *drm = nouveau_drm(dev);
|
||||
struct nouveau_client *client;
|
||||
struct nouveau_object *device;
|
||||
|
||||
dev->irq_enabled = false;
|
||||
device = drm->client.base.device;
|
||||
client = nvkm_client(&drm->client.base);
|
||||
device = client->device;
|
||||
drm_put_dev(dev);
|
||||
|
||||
nouveau_object_ref(NULL, &device);
|
||||
@@ -612,13 +573,13 @@ nouveau_do_suspend(struct drm_device *dev, bool runtime)
|
||||
}
|
||||
|
||||
list_for_each_entry(cli, &drm->clients, head) {
|
||||
ret = nouveau_client_fini(&cli->base, true);
|
||||
ret = nvif_client_suspend(&cli->base);
|
||||
if (ret)
|
||||
goto fail_client;
|
||||
}
|
||||
|
||||
NV_INFO(drm, "suspending kernel object tree...\n");
|
||||
ret = nouveau_client_fini(&drm->client.base, true);
|
||||
ret = nvif_client_suspend(&drm->client.base);
|
||||
if (ret)
|
||||
goto fail_client;
|
||||
|
||||
@@ -627,7 +588,7 @@ nouveau_do_suspend(struct drm_device *dev, bool runtime)
|
||||
|
||||
fail_client:
|
||||
list_for_each_entry_continue_reverse(cli, &drm->clients, head) {
|
||||
nouveau_client_init(&cli->base);
|
||||
nvif_client_resume(&cli->base);
|
||||
}
|
||||
|
||||
if (drm->fence && nouveau_fence(drm)->resume)
|
||||
@@ -675,7 +636,7 @@ nouveau_do_resume(struct drm_device *dev)
|
||||
nouveau_agp_reset(drm);
|
||||
|
||||
NV_INFO(drm, "resuming kernel object tree...\n");
|
||||
nouveau_client_init(&drm->client.base);
|
||||
nvif_client_resume(&drm->client.base);
|
||||
nouveau_agp_init(drm);
|
||||
|
||||
NV_INFO(drm, "resuming client object trees...\n");
|
||||
@@ -683,7 +644,7 @@ nouveau_do_resume(struct drm_device *dev)
|
||||
nouveau_fence(drm)->resume(drm);
|
||||
|
||||
list_for_each_entry(cli, &drm->clients, head) {
|
||||
nouveau_client_init(&cli->base);
|
||||
nvif_client_resume(&cli->base);
|
||||
}
|
||||
|
||||
nouveau_run_vbios_init(dev);
|
||||
@@ -779,6 +740,8 @@ nouveau_drm_open(struct drm_device *dev, struct drm_file *fpriv)
|
||||
if (ret)
|
||||
goto out_suspend;
|
||||
|
||||
cli->base.super = false;
|
||||
|
||||
if (drm->device.info.family >= NV_DEVICE_INFO_V0_TESLA) {
|
||||
ret = nouveau_vm_new(nvkm_device(&drm->device), 0, (1ULL << 40),
|
||||
0x1000, &cli->vm);
|
||||
@@ -787,7 +750,7 @@ nouveau_drm_open(struct drm_device *dev, struct drm_file *fpriv)
|
||||
goto out_suspend;
|
||||
}
|
||||
|
||||
cli->base.vm = cli->vm;
|
||||
nvkm_client(&cli->base)->vm = cli->vm;
|
||||
}
|
||||
|
||||
fpriv->driver_priv = cli;
|
||||
|
||||
@@ -28,6 +28,9 @@
|
||||
#include <core/client.h>
|
||||
#include <core/event.h>
|
||||
|
||||
#include <nvif/client.h>
|
||||
#include <nvif/device.h>
|
||||
|
||||
#include <subdev/vm.h>
|
||||
|
||||
#include <drmP.h>
|
||||
@@ -53,6 +56,17 @@ struct nouveau_drm_tile {
|
||||
bool used;
|
||||
};
|
||||
|
||||
enum nouveau_drm_object_route {
|
||||
NVDRM_OBJECT_NVIF = 0,
|
||||
NVDRM_OBJECT_USIF,
|
||||
NVDRM_OBJECT_ABI16,
|
||||
};
|
||||
|
||||
enum nouveau_drm_notify_route {
|
||||
NVDRM_NOTIFY_NVIF = 0,
|
||||
NVDRM_NOTIFY_USIF
|
||||
};
|
||||
|
||||
enum nouveau_drm_handle {
|
||||
NVDRM_CLIENT = 0xffffffff,
|
||||
NVDRM_DEVICE = 0xdddddddd,
|
||||
@@ -64,7 +78,7 @@ enum nouveau_drm_handle {
|
||||
};
|
||||
|
||||
struct nouveau_cli {
|
||||
struct nouveau_client base;
|
||||
struct nvif_client base;
|
||||
struct nouveau_vm *vm; /*XXX*/
|
||||
struct list_head head;
|
||||
struct mutex mutex;
|
||||
@@ -78,31 +92,6 @@ nouveau_cli(struct drm_file *fpriv)
|
||||
}
|
||||
|
||||
#include <nvif/object.h>
|
||||
#undef nvif_object
|
||||
#undef nvif_rd08
|
||||
#undef nvif_rd16
|
||||
#undef nvif_rd32
|
||||
#undef nvif_wr08
|
||||
#undef nvif_wr16
|
||||
#undef nvif_wr32
|
||||
#undef nvif_mask
|
||||
#undef nvkm_object
|
||||
#undef nvif_exec
|
||||
|
||||
#define nvif_object(a) ({ \
|
||||
struct nvif_object *_object = (a)->object; \
|
||||
(struct nouveau_object *)_object; \
|
||||
})
|
||||
#define nvif_rd08(a,b) nv_ro08(nvif_object(a), (b))
|
||||
#define nvif_rd16(a,b) nv_ro16(nvif_object(a), (b))
|
||||
#define nvif_rd32(a,b) nv_ro32(nvif_object(a), (b))
|
||||
#define nvif_wr08(a,b,c) nv_wo08(nvif_object(a), (b), (c))
|
||||
#define nvif_wr16(a,b,c) nv_wo16(nvif_object(a), (b), (c))
|
||||
#define nvif_wr32(a,b,c) nv_wo32(nvif_object(a), (b), (c))
|
||||
#define nvif_mask(a,b,c,d) nv_mo32(nvif_object(a), (b), (c), (d))
|
||||
#define nvkm_object(a) nvif_object(a)
|
||||
#define nvif_exec(a,b,c,d) nv_exec(nvkm_object(a), (b), (c), (d))
|
||||
|
||||
#include <nvif/device.h>
|
||||
|
||||
extern int nouveau_runtime_pm;
|
||||
@@ -134,6 +123,7 @@ struct nouveau_drm {
|
||||
struct ttm_buffer_object *,
|
||||
struct ttm_mem_reg *, struct ttm_mem_reg *);
|
||||
struct nouveau_channel *chan;
|
||||
struct nvif_object copy;
|
||||
int mtrr;
|
||||
} ttm;
|
||||
|
||||
@@ -151,6 +141,8 @@ struct nouveau_drm {
|
||||
struct nouveau_channel *channel;
|
||||
struct nouveau_gpuobj *notify;
|
||||
struct nouveau_fbdev *fbcon;
|
||||
struct nvif_object nvsw;
|
||||
struct nvif_object ntfy;
|
||||
|
||||
/* nv10-nv40 tiling regions */
|
||||
struct {
|
||||
@@ -192,7 +184,7 @@ void nouveau_drm_device_remove(struct drm_device *dev);
|
||||
|
||||
#define NV_PRINTK(l,c,f,a...) do { \
|
||||
struct nouveau_cli *_cli = (c); \
|
||||
nv_##l(_cli, f, ##a); \
|
||||
nv_##l(_cli->base.base.priv, f, ##a); \
|
||||
} while(0)
|
||||
#define NV_FATAL(drm,f,a...) NV_PRINTK(fatal, &(drm)->client, f, ##a)
|
||||
#define NV_ERROR(drm,f,a...) NV_PRINTK(error, &(drm)->client, f, ##a)
|
||||
|
||||
@@ -54,8 +54,6 @@
|
||||
#include <core/client.h>
|
||||
#include <core/device.h>
|
||||
|
||||
#include <subdev/fb.h>
|
||||
|
||||
MODULE_PARM_DESC(nofbaccel, "Disable fbcon acceleration");
|
||||
static int nouveau_nofbaccel = 0;
|
||||
module_param_named(nofbaccel, nouveau_nofbaccel, int, 0400);
|
||||
@@ -241,6 +239,13 @@ nouveau_fbcon_accel_fini(struct drm_device *dev)
|
||||
fbcon->helper.fbdev->flags |= FBINFO_HWACCEL_DISABLED;
|
||||
console_unlock();
|
||||
nouveau_channel_idle(drm->channel);
|
||||
nvif_object_fini(&fbcon->twod);
|
||||
nvif_object_fini(&fbcon->blit);
|
||||
nvif_object_fini(&fbcon->gdi);
|
||||
nvif_object_fini(&fbcon->patt);
|
||||
nvif_object_fini(&fbcon->rop);
|
||||
nvif_object_fini(&fbcon->clip);
|
||||
nvif_object_fini(&fbcon->surf2d);
|
||||
}
|
||||
}
|
||||
|
||||
@@ -352,7 +357,7 @@ nouveau_fbcon_create(struct drm_fb_helper *helper,
|
||||
|
||||
chan = nouveau_nofbaccel ? NULL : drm->channel;
|
||||
if (chan && device->info.family >= NV_DEVICE_INFO_V0_TESLA) {
|
||||
ret = nouveau_bo_vma_add(nvbo, nv_client(chan->cli)->vm,
|
||||
ret = nouveau_bo_vma_add(nvbo, drm->client.vm,
|
||||
&fbcon->nouveau_fb.vma);
|
||||
if (ret) {
|
||||
NV_ERROR(drm, "failed to map fb into chan: %d\n", ret);
|
||||
|
||||
@@ -37,6 +37,13 @@ struct nouveau_fbdev {
|
||||
struct list_head fbdev_list;
|
||||
struct drm_device *dev;
|
||||
unsigned int saved_flags;
|
||||
struct nvif_object surf2d;
|
||||
struct nvif_object clip;
|
||||
struct nvif_object rop;
|
||||
struct nvif_object patt;
|
||||
struct nvif_object gdi;
|
||||
struct nvif_object blit;
|
||||
struct nvif_object twod;
|
||||
};
|
||||
|
||||
void nouveau_fbcon_restore(void);
|
||||
|
||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user