Commit 5de8037a authored by Ben Skeggs's avatar Ben Skeggs

drm/nvc0: enable per-client address spaces

Signed-off-by: default avatarBen Skeggs <bskeggs@redhat.com>
parent ad9ac437
...@@ -690,46 +690,64 @@ nouveau_gpuobj_channel_init_pramin(struct nouveau_channel *chan) ...@@ -690,46 +690,64 @@ nouveau_gpuobj_channel_init_pramin(struct nouveau_channel *chan)
return 0; return 0;
} }
int static int
nouveau_gpuobj_channel_init(struct nouveau_channel *chan, nvc0_gpuobj_channel_init(struct nouveau_channel *chan, struct nouveau_vm *vm)
uint32_t vram_h, uint32_t tt_h)
{ {
struct drm_device *dev = chan->dev; struct drm_device *dev = chan->dev;
struct drm_nouveau_private *dev_priv = dev->dev_private; struct nouveau_gpuobj *pgd = NULL;
struct nouveau_fpriv *fpriv = nouveau_fpriv(chan->file_priv); struct nouveau_vm_pgd *vpgd;
struct nouveau_vm *vm = fpriv ? fpriv->vm : dev_priv->chan_vm;
struct nouveau_gpuobj *vram = NULL, *tt = NULL;
int ret, i; int ret, i;
NV_DEBUG(dev, "ch%d vram=0x%08x tt=0x%08x\n", chan->id, vram_h, tt_h); ret = nouveau_gpuobj_new(dev, NULL, 4096, 0x1000, 0, &chan->ramin);
if (dev_priv->card_type == NV_C0) { if (ret)
struct nouveau_vm_pgd *vpgd; return ret;
ret = nouveau_gpuobj_new(dev, NULL, 4096, 0x1000, 0, /* create page directory for this vm if none currently exists,
&chan->ramin); * will be destroyed automagically when last reference to the
* vm is removed
*/
if (list_empty(&vm->pgd_list)) {
ret = nouveau_gpuobj_new(dev, NULL, 65536, 0x1000, 0, &pgd);
if (ret) if (ret)
return ret; return ret;
}
nouveau_vm_ref(vm, &chan->vm, pgd);
nouveau_gpuobj_ref(NULL, &pgd);
nouveau_vm_ref(vm, &chan->vm, NULL); /* point channel at vm's page directory */
vpgd = list_first_entry(&vm->pgd_list, struct nouveau_vm_pgd, head);
nv_wo32(chan->ramin, 0x0200, lower_32_bits(vpgd->obj->vinst));
nv_wo32(chan->ramin, 0x0204, upper_32_bits(vpgd->obj->vinst));
nv_wo32(chan->ramin, 0x0208, 0xffffffff);
nv_wo32(chan->ramin, 0x020c, 0x000000ff);
vpgd = list_first_entry(&vm->pgd_list, struct nouveau_vm_pgd, head); /* map display semaphore buffers into channel's vm */
nv_wo32(chan->ramin, 0x0200, lower_32_bits(vpgd->obj->vinst)); for (i = 0; i < 2; i++) {
nv_wo32(chan->ramin, 0x0204, upper_32_bits(vpgd->obj->vinst)); struct nv50_display_crtc *dispc = &nv50_display(dev)->crtc[i];
nv_wo32(chan->ramin, 0x0208, 0xffffffff);
nv_wo32(chan->ramin, 0x020c, 0x000000ff);
for (i = 0; i < 2; i++) { ret = nouveau_bo_vma_add(dispc->sem.bo, chan->vm,
struct nv50_display_crtc *dispc = &chan->dispc_vma[i]);
&nv50_display(dev)->crtc[i]; if (ret)
return ret;
}
ret = nouveau_bo_vma_add(dispc->sem.bo, chan->vm, return 0;
&chan->dispc_vma[i]); }
if (ret)
return ret;
}
return 0; int
} nouveau_gpuobj_channel_init(struct nouveau_channel *chan,
uint32_t vram_h, uint32_t tt_h)
{
struct drm_device *dev = chan->dev;
struct drm_nouveau_private *dev_priv = dev->dev_private;
struct nouveau_fpriv *fpriv = nouveau_fpriv(chan->file_priv);
struct nouveau_vm *vm = fpriv ? fpriv->vm : dev_priv->chan_vm;
struct nouveau_gpuobj *vram = NULL, *tt = NULL;
int ret, i;
NV_DEBUG(dev, "ch%d vram=0x%08x tt=0x%08x\n", chan->id, vram_h, tt_h);
if (dev_priv->card_type == NV_C0)
return nvc0_gpuobj_channel_init(chan, vm);
/* Allocate a chunk of memory for per-channel object storage */ /* Allocate a chunk of memory for per-channel object storage */
ret = nouveau_gpuobj_channel_init_pramin(chan); ret = nouveau_gpuobj_channel_init_pramin(chan);
......
...@@ -787,7 +787,12 @@ nouveau_open(struct drm_device *dev, struct drm_file *file_priv) ...@@ -787,7 +787,12 @@ nouveau_open(struct drm_device *dev, struct drm_file *file_priv)
} }
} else } else
if (dev_priv->card_type >= NV_C0) { if (dev_priv->card_type >= NV_C0) {
nouveau_vm_ref(dev_priv->chan_vm, &fpriv->vm, NULL); ret = nouveau_vm_new(dev, 0, (1ULL << 40), 0x0008000000ULL,
&fpriv->vm);
if (ret) {
kfree(fpriv);
return ret;
}
} }
file_priv->driver_priv = fpriv; file_priv->driver_priv = fpriv;
......
...@@ -32,7 +32,6 @@ struct nvc0_instmem_priv { ...@@ -32,7 +32,6 @@ struct nvc0_instmem_priv {
struct nouveau_channel *bar1; struct nouveau_channel *bar1;
struct nouveau_gpuobj *bar3_pgd; struct nouveau_gpuobj *bar3_pgd;
struct nouveau_channel *bar3; struct nouveau_channel *bar3;
struct nouveau_gpuobj *chan_pgd;
}; };
int int
...@@ -181,17 +180,11 @@ nvc0_instmem_init(struct drm_device *dev) ...@@ -181,17 +180,11 @@ nvc0_instmem_init(struct drm_device *dev)
goto error; goto error;
/* channel vm */ /* channel vm */
ret = nouveau_vm_new(dev, 0, (1ULL << 40), 0x0008000000ULL, &vm); ret = nouveau_vm_new(dev, 0, (1ULL << 40), 0x0008000000ULL,
&dev_priv->chan_vm);
if (ret) if (ret)
goto error; goto error;
ret = nouveau_gpuobj_new(dev, NULL, 0x8000, 4096, 0, &priv->chan_pgd);
if (ret)
goto error;
nouveau_vm_ref(vm, &dev_priv->chan_vm, priv->chan_pgd);
nouveau_vm_ref(NULL, &vm, NULL);
nvc0_instmem_resume(dev); nvc0_instmem_resume(dev);
return 0; return 0;
error: error:
...@@ -211,8 +204,7 @@ nvc0_instmem_takedown(struct drm_device *dev) ...@@ -211,8 +204,7 @@ nvc0_instmem_takedown(struct drm_device *dev)
nv_wr32(dev, 0x1704, 0x00000000); nv_wr32(dev, 0x1704, 0x00000000);
nv_wr32(dev, 0x1714, 0x00000000); nv_wr32(dev, 0x1714, 0x00000000);
nouveau_vm_ref(NULL, &dev_priv->chan_vm, priv->chan_pgd); nouveau_vm_ref(NULL, &dev_priv->chan_vm, NULL);
nouveau_gpuobj_ref(NULL, &priv->chan_pgd);
nvc0_channel_del(&priv->bar1); nvc0_channel_del(&priv->bar1);
nouveau_vm_ref(NULL, &dev_priv->bar1_vm, priv->bar1_pgd); nouveau_vm_ref(NULL, &dev_priv->bar1_vm, priv->bar1_pgd);
......
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