aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/gpu/drm
diff options
context:
space:
mode:
authorBen Skeggs <bskeggs@redhat.com>2012-05-01 06:48:08 -0400
committerBen Skeggs <bskeggs@redhat.com>2012-05-24 02:56:11 -0400
commitc420b2dc8dc3cdd507214f4df5c5f96f08812cbe (patch)
tree6dca9f0aba3de22a2bda5fe647d6945d4f4e986e /drivers/gpu/drm
parenta226c32a386bca0426e500954b79e3fd46afc0d9 (diff)
drm/nouveau/fifo: turn all fifo modules into engine modules
Been tested on each major revision that's relevant here, but I'm sure there are still bugs waiting to be ironed out. This is a *very* invasive change. There's a couple of pieces left that I don't like much (eg. other engines using fifo_priv for the channel count), but that's an artefact of there being a master channel list still. This is changing, slowly. Signed-off-by: Ben Skeggs <bskeggs@redhat.com>
Diffstat (limited to 'drivers/gpu/drm')
-rw-r--r--drivers/gpu/drm/nouveau/Makefile4
-rw-r--r--drivers/gpu/drm/nouveau/nouveau_channel.c33
-rw-r--r--drivers/gpu/drm/nouveau/nouveau_drv.c22
-rw-r--r--drivers/gpu/drm/nouveau/nouveau_drv.h73
-rw-r--r--drivers/gpu/drm/nouveau/nouveau_fifo.h32
-rw-r--r--drivers/gpu/drm/nouveau/nouveau_mem.c1
-rw-r--r--drivers/gpu/drm/nouveau/nouveau_object.c4
-rw-r--r--drivers/gpu/drm/nouveau/nouveau_state.c108
-rw-r--r--drivers/gpu/drm/nouveau/nv04_fifo.c367
-rw-r--r--drivers/gpu/drm/nouveau/nv04_graph.c10
-rw-r--r--drivers/gpu/drm/nouveau/nv04_instmem.c15
-rw-r--r--drivers/gpu/drm/nouveau/nv10_fifo.c272
-rw-r--r--drivers/gpu/drm/nouveau/nv10_graph.c9
-rw-r--r--drivers/gpu/drm/nouveau/nv17_fifo.c177
-rw-r--r--drivers/gpu/drm/nouveau/nv20_graph.c4
-rw-r--r--drivers/gpu/drm/nouveau/nv31_mpeg.c4
-rw-r--r--drivers/gpu/drm/nouveau/nv40_fifo.c352
-rw-r--r--drivers/gpu/drm/nouveau/nv40_graph.c4
-rw-r--r--drivers/gpu/drm/nouveau/nv40_pm.c1
-rw-r--r--drivers/gpu/drm/nouveau/nv50_fb.c4
-rw-r--r--drivers/gpu/drm/nouveau/nv50_fifo.c438
-rw-r--r--drivers/gpu/drm/nouveau/nv50_graph.c4
-rw-r--r--drivers/gpu/drm/nouveau/nv50_vm.c2
-rw-r--r--drivers/gpu/drm/nouveau/nv84_fence.c3
-rw-r--r--drivers/gpu/drm/nouveau/nv84_fifo.c241
-rw-r--r--drivers/gpu/drm/nouveau/nvc0_fence.c3
-rw-r--r--drivers/gpu/drm/nouveau/nvc0_fifo.c281
-rw-r--r--drivers/gpu/drm/nouveau/nvc0_graph.c4
-rw-r--r--drivers/gpu/drm/nouveau/nve0_fifo.c284
-rw-r--r--drivers/gpu/drm/nouveau/nve0_graph.c4
30 files changed, 1374 insertions, 1386 deletions
diff --git a/drivers/gpu/drm/nouveau/Makefile b/drivers/gpu/drm/nouveau/Makefile
index ce222eb0a318..fe5267d06ab5 100644
--- a/drivers/gpu/drm/nouveau/Makefile
+++ b/drivers/gpu/drm/nouveau/Makefile
@@ -16,8 +16,8 @@ nouveau-y := nouveau_drv.o nouveau_state.o nouveau_channel.o nouveau_mem.o \
16 nv04_mc.o nv40_mc.o nv50_mc.o \ 16 nv04_mc.o nv40_mc.o nv50_mc.o \
17 nv04_fb.o nv10_fb.o nv20_fb.o nv30_fb.o nv40_fb.o \ 17 nv04_fb.o nv10_fb.o nv20_fb.o nv30_fb.o nv40_fb.o \
18 nv50_fb.o nvc0_fb.o \ 18 nv50_fb.o nvc0_fb.o \
19 nv04_fifo.o nv10_fifo.o nv40_fifo.o nv50_fifo.o nvc0_fifo.o \ 19 nv04_fifo.o nv10_fifo.o nv17_fifo.o nv40_fifo.o nv50_fifo.o \
20 nve0_fifo.o \ 20 nv84_fifo.o nvc0_fifo.o nve0_fifo.o \
21 nv04_fence.o nv10_fence.o nv84_fence.o nvc0_fence.o \ 21 nv04_fence.o nv10_fence.o nv84_fence.o nvc0_fence.o \
22 nv04_software.o nv50_software.o nvc0_software.o \ 22 nv04_software.o nv50_software.o nvc0_software.o \
23 nv04_graph.o nv10_graph.o nv20_graph.o \ 23 nv04_graph.o nv10_graph.o nv20_graph.o \
diff --git a/drivers/gpu/drm/nouveau/nouveau_channel.c b/drivers/gpu/drm/nouveau/nouveau_channel.c
index a1f566758e7b..9420538d2374 100644
--- a/drivers/gpu/drm/nouveau/nouveau_channel.c
+++ b/drivers/gpu/drm/nouveau/nouveau_channel.c
@@ -27,6 +27,7 @@
27#include "nouveau_drv.h" 27#include "nouveau_drv.h"
28#include "nouveau_drm.h" 28#include "nouveau_drm.h"
29#include "nouveau_dma.h" 29#include "nouveau_dma.h"
30#include "nouveau_fifo.h"
30#include "nouveau_ramht.h" 31#include "nouveau_ramht.h"
31#include "nouveau_fence.h" 32#include "nouveau_fence.h"
32#include "nouveau_software.h" 33#include "nouveau_software.h"
@@ -120,8 +121,8 @@ nouveau_channel_alloc(struct drm_device *dev, struct nouveau_channel **chan_ret,
120 uint32_t vram_handle, uint32_t gart_handle) 121 uint32_t vram_handle, uint32_t gart_handle)
121{ 122{
122 struct nouveau_exec_engine *fence = nv_engine(dev, NVOBJ_ENGINE_FENCE); 123 struct nouveau_exec_engine *fence = nv_engine(dev, NVOBJ_ENGINE_FENCE);
124 struct nouveau_fifo_priv *pfifo = nv_engine(dev, NVOBJ_ENGINE_FIFO);
123 struct drm_nouveau_private *dev_priv = dev->dev_private; 125 struct drm_nouveau_private *dev_priv = dev->dev_private;
124 struct nouveau_fifo_engine *pfifo = &dev_priv->engine.fifo;
125 struct nouveau_fpriv *fpriv = nouveau_fpriv(file_priv); 126 struct nouveau_fpriv *fpriv = nouveau_fpriv(file_priv);
126 struct nouveau_channel *chan; 127 struct nouveau_channel *chan;
127 unsigned long flags; 128 unsigned long flags;
@@ -189,20 +190,13 @@ nouveau_channel_alloc(struct drm_device *dev, struct nouveau_channel **chan_ret,
189 if (dev_priv->card_type >= NV_50) 190 if (dev_priv->card_type >= NV_50)
190 chan->user_get_hi = 0x60; 191 chan->user_get_hi = 0x60;
191 192
192 /* disable the fifo caches */ 193 /* create fifo context */
193 if (dev_priv->card_type < NV_50) 194 ret = pfifo->base.context_new(chan, NVOBJ_ENGINE_FIFO);
194 nv_wr32(dev, NV03_PFIFO_CACHES, 0);
195
196 /* Construct initial RAMFC for new channel */
197 ret = pfifo->create_context(chan);
198 if (ret) { 195 if (ret) {
199 nouveau_channel_put(&chan); 196 nouveau_channel_put(&chan);
200 return ret; 197 return ret;
201 } 198 }
202 199
203 if (dev_priv->card_type < NV_50)
204 nv_wr32(dev, NV03_PFIFO_CACHES, 1);
205
206 /* Insert NOPs for NOUVEAU_DMA_SKIPS */ 200 /* Insert NOPs for NOUVEAU_DMA_SKIPS */
207 ret = RING_SPACE(chan, NOUVEAU_DMA_SKIPS); 201 ret = RING_SPACE(chan, NOUVEAU_DMA_SKIPS);
208 if (ret) { 202 if (ret) {
@@ -288,7 +282,6 @@ nouveau_channel_put_unlocked(struct nouveau_channel **pchan)
288 struct nouveau_channel *chan = *pchan; 282 struct nouveau_channel *chan = *pchan;
289 struct drm_device *dev = chan->dev; 283 struct drm_device *dev = chan->dev;
290 struct drm_nouveau_private *dev_priv = dev->dev_private; 284 struct drm_nouveau_private *dev_priv = dev->dev_private;
291 struct nouveau_fifo_engine *pfifo = &dev_priv->engine.fifo;
292 unsigned long flags; 285 unsigned long flags;
293 int i; 286 int i;
294 287
@@ -305,22 +298,12 @@ nouveau_channel_put_unlocked(struct nouveau_channel **pchan)
305 /* give it chance to idle */ 298 /* give it chance to idle */
306 nouveau_channel_idle(chan); 299 nouveau_channel_idle(chan);
307 300
308 /* boot it off the hardware */
309 if (dev_priv->card_type < NV_50)
310 nv_wr32(dev, NV03_PFIFO_CACHES, 0);
311
312 /* destroy the engine specific contexts */ 301 /* destroy the engine specific contexts */
313 for (i = NVOBJ_ENGINE_NR - 1; i >= 0; i--) { 302 for (i = NVOBJ_ENGINE_NR - 1; i >= 0; i--) {
314 if (chan->engctx[i]) 303 if (chan->engctx[i])
315 dev_priv->eng[i]->context_del(chan, i); 304 dev_priv->eng[i]->context_del(chan, i);
316 /*XXX: clean this up later, order is important */
317 if (i == NVOBJ_ENGINE_FENCE)
318 pfifo->destroy_context(chan);
319 } 305 }
320 306
321 if (dev_priv->card_type < NV_50)
322 nv_wr32(dev, NV03_PFIFO_CACHES, 1);
323
324 /* aside from its resources, the channel should now be dead, 307 /* aside from its resources, the channel should now be dead,
325 * remove it from the channel list 308 * remove it from the channel list
326 */ 309 */
@@ -393,13 +376,15 @@ nouveau_channel_idle(struct nouveau_channel *chan)
393void 376void
394nouveau_channel_cleanup(struct drm_device *dev, struct drm_file *file_priv) 377nouveau_channel_cleanup(struct drm_device *dev, struct drm_file *file_priv)
395{ 378{
396 struct drm_nouveau_private *dev_priv = dev->dev_private; 379 struct nouveau_fifo_priv *pfifo = nv_engine(dev, NVOBJ_ENGINE_FIFO);
397 struct nouveau_engine *engine = &dev_priv->engine;
398 struct nouveau_channel *chan; 380 struct nouveau_channel *chan;
399 int i; 381 int i;
400 382
383 if (!pfifo)
384 return;
385
401 NV_DEBUG(dev, "clearing FIFO enables from file_priv\n"); 386 NV_DEBUG(dev, "clearing FIFO enables from file_priv\n");
402 for (i = 0; i < engine->fifo.channels; i++) { 387 for (i = 0; i < pfifo->channels; i++) {
403 chan = nouveau_channel_get(file_priv, i); 388 chan = nouveau_channel_get(file_priv, i);
404 if (IS_ERR(chan)) 389 if (IS_ERR(chan))
405 continue; 390 continue;
diff --git a/drivers/gpu/drm/nouveau/nouveau_drv.c b/drivers/gpu/drm/nouveau/nouveau_drv.c
index 910b97e813fa..cad254c8e387 100644
--- a/drivers/gpu/drm/nouveau/nouveau_drv.c
+++ b/drivers/gpu/drm/nouveau/nouveau_drv.c
@@ -33,6 +33,7 @@
33#include "nouveau_fb.h" 33#include "nouveau_fb.h"
34#include "nouveau_fbcon.h" 34#include "nouveau_fbcon.h"
35#include "nouveau_pm.h" 35#include "nouveau_pm.h"
36#include "nouveau_fifo.h"
36#include "nv50_display.h" 37#include "nv50_display.h"
37 38
38#include "drm_pciids.h" 39#include "drm_pciids.h"
@@ -175,7 +176,7 @@ nouveau_pci_suspend(struct pci_dev *pdev, pm_message_t pm_state)
175 struct drm_device *dev = pci_get_drvdata(pdev); 176 struct drm_device *dev = pci_get_drvdata(pdev);
176 struct drm_nouveau_private *dev_priv = dev->dev_private; 177 struct drm_nouveau_private *dev_priv = dev->dev_private;
177 struct nouveau_instmem_engine *pinstmem = &dev_priv->engine.instmem; 178 struct nouveau_instmem_engine *pinstmem = &dev_priv->engine.instmem;
178 struct nouveau_fifo_engine *pfifo = &dev_priv->engine.fifo; 179 struct nouveau_fifo_priv *pfifo = nv_engine(dev, NVOBJ_ENGINE_FIFO);
179 struct nouveau_channel *chan; 180 struct nouveau_channel *chan;
180 struct drm_crtc *crtc; 181 struct drm_crtc *crtc;
181 int ret, i, e; 182 int ret, i, e;
@@ -214,21 +215,13 @@ nouveau_pci_suspend(struct pci_dev *pdev, pm_message_t pm_state)
214 ttm_bo_evict_mm(&dev_priv->ttm.bdev, TTM_PL_VRAM); 215 ttm_bo_evict_mm(&dev_priv->ttm.bdev, TTM_PL_VRAM);
215 216
216 NV_INFO(dev, "Idling channels...\n"); 217 NV_INFO(dev, "Idling channels...\n");
217 for (i = 0; i < pfifo->channels; i++) { 218 for (i = 0; i < (pfifo ? pfifo->channels : 0); i++) {
218 chan = dev_priv->channels.ptr[i]; 219 chan = dev_priv->channels.ptr[i];
219 220
220 if (chan && chan->pushbuf_bo) 221 if (chan && chan->pushbuf_bo)
221 nouveau_channel_idle(chan); 222 nouveau_channel_idle(chan);
222 } 223 }
223 224
224 if (dev_priv->card_type < NV_50) {
225 nv_wr32(dev, NV03_PFIFO_CACHES, 0);
226 nv_mask(dev, NV04_PFIFO_CACHE1_DMA_PUSH, 0x00000001, 0);
227 nv_wr32(dev, NV03_PFIFO_CACHE1_PUSH0, 0);
228 nv_mask(dev, NV04_PFIFO_CACHE1_PULL0, 0x00000001, 0);
229 }
230 pfifo->unload_context(dev);
231
232 for (e = NVOBJ_ENGINE_NR - 1; e >= 0; e--) { 225 for (e = NVOBJ_ENGINE_NR - 1; e >= 0; e--) {
233 if (!dev_priv->eng[e]) 226 if (!dev_priv->eng[e])
234 continue; 227 continue;
@@ -269,11 +262,6 @@ out_abort:
269 if (dev_priv->eng[e]) 262 if (dev_priv->eng[e])
270 dev_priv->eng[e]->init(dev, e); 263 dev_priv->eng[e]->init(dev, e);
271 } 264 }
272 if (dev_priv->card_type < NV_50) {
273 nv_wr32(dev, NV03_PFIFO_CACHE1_PUSH0, 1);
274 nv_wr32(dev, NV04_PFIFO_CACHE1_PULL0, 1);
275 nv_wr32(dev, NV03_PFIFO_CACHES, 1);
276 }
277 return ret; 265 return ret;
278} 266}
279 267
@@ -281,6 +269,7 @@ int
281nouveau_pci_resume(struct pci_dev *pdev) 269nouveau_pci_resume(struct pci_dev *pdev)
282{ 270{
283 struct drm_device *dev = pci_get_drvdata(pdev); 271 struct drm_device *dev = pci_get_drvdata(pdev);
272 struct nouveau_fifo_priv *pfifo = nv_engine(dev, NVOBJ_ENGINE_FIFO);
284 struct drm_nouveau_private *dev_priv = dev->dev_private; 273 struct drm_nouveau_private *dev_priv = dev->dev_private;
285 struct nouveau_engine *engine = &dev_priv->engine; 274 struct nouveau_engine *engine = &dev_priv->engine;
286 struct drm_crtc *crtc; 275 struct drm_crtc *crtc;
@@ -328,7 +317,6 @@ nouveau_pci_resume(struct pci_dev *pdev)
328 if (dev_priv->eng[i]) 317 if (dev_priv->eng[i])
329 dev_priv->eng[i]->init(dev, i); 318 dev_priv->eng[i]->init(dev, i);
330 } 319 }
331 engine->fifo.init(dev);
332 320
333 nouveau_irq_postinstall(dev); 321 nouveau_irq_postinstall(dev);
334 322
@@ -337,7 +325,7 @@ nouveau_pci_resume(struct pci_dev *pdev)
337 struct nouveau_channel *chan; 325 struct nouveau_channel *chan;
338 int j; 326 int j;
339 327
340 for (i = 0; i < dev_priv->engine.fifo.channels; i++) { 328 for (i = 0; i < (pfifo ? pfifo->channels : 0); i++) {
341 chan = dev_priv->channels.ptr[i]; 329 chan = dev_priv->channels.ptr[i];
342 if (!chan || !chan->pushbuf_bo) 330 if (!chan || !chan->pushbuf_bo)
343 continue; 331 continue;
diff --git a/drivers/gpu/drm/nouveau/nouveau_drv.h b/drivers/gpu/drm/nouveau/nouveau_drv.h
index 9943ccf764ce..1ede35491f54 100644
--- a/drivers/gpu/drm/nouveau/nouveau_drv.h
+++ b/drivers/gpu/drm/nouveau/nouveau_drv.h
@@ -165,9 +165,10 @@ enum nouveau_flags {
165#define NVOBJ_ENGINE_PPP NVOBJ_ENGINE_MPEG 165#define NVOBJ_ENGINE_PPP NVOBJ_ENGINE_MPEG
166#define NVOBJ_ENGINE_BSP 6 166#define NVOBJ_ENGINE_BSP 6
167#define NVOBJ_ENGINE_VP 7 167#define NVOBJ_ENGINE_VP 7
168#define NVOBJ_ENGINE_FENCE 14 168#define NVOBJ_ENGINE_FIFO 14
169#define NVOBJ_ENGINE_DISPLAY 15 169#define NVOBJ_ENGINE_FENCE 15
170#define NVOBJ_ENGINE_NR 16 170#define NVOBJ_ENGINE_NR 16
171#define NVOBJ_ENGINE_DISPLAY (NVOBJ_ENGINE_NR + 0) /*XXX*/
171 172
172#define NVOBJ_FLAG_DONT_MAP (1 << 0) 173#define NVOBJ_FLAG_DONT_MAP (1 << 0)
173#define NVOBJ_FLAG_ZERO_ALLOC (1 << 1) 174#define NVOBJ_FLAG_ZERO_ALLOC (1 << 1)
@@ -248,8 +249,6 @@ struct nouveau_channel {
248 249
249 /* PFIFO context */ 250 /* PFIFO context */
250 struct nouveau_gpuobj *ramfc; 251 struct nouveau_gpuobj *ramfc;
251 struct nouveau_gpuobj *cache;
252 void *fifo_priv;
253 252
254 /* Execution engine contexts */ 253 /* Execution engine contexts */
255 void *engctx[NVOBJ_ENGINE_NR]; 254 void *engctx[NVOBJ_ENGINE_NR];
@@ -283,8 +282,6 @@ struct nouveau_channel {
283 int ib_put; 282 int ib_put;
284 } dma; 283 } dma;
285 284
286 uint32_t sw_subchannel[8];
287
288 struct { 285 struct {
289 bool active; 286 bool active;
290 char name[32]; 287 char name[32];
@@ -347,23 +344,6 @@ struct nouveau_fb_engine {
347 void (*free_tile_region)(struct drm_device *dev, int i); 344 void (*free_tile_region)(struct drm_device *dev, int i);
348}; 345};
349 346
350struct nouveau_fifo_engine {
351 void *priv;
352 int channels;
353
354 struct nouveau_gpuobj *playlist[2];
355 int cur_playlist;
356
357 int (*init)(struct drm_device *);
358 void (*takedown)(struct drm_device *);
359
360 int (*create_context)(struct nouveau_channel *);
361 void (*destroy_context)(struct nouveau_channel *);
362 int (*load_context)(struct nouveau_channel *);
363 int (*unload_context)(struct drm_device *);
364 void (*tlb_flush)(struct drm_device *dev);
365};
366
367struct nouveau_display_engine { 347struct nouveau_display_engine {
368 void *priv; 348 void *priv;
369 int (*early_init)(struct drm_device *); 349 int (*early_init)(struct drm_device *);
@@ -571,7 +551,6 @@ struct nouveau_engine {
571 struct nouveau_mc_engine mc; 551 struct nouveau_mc_engine mc;
572 struct nouveau_timer_engine timer; 552 struct nouveau_timer_engine timer;
573 struct nouveau_fb_engine fb; 553 struct nouveau_fb_engine fb;
574 struct nouveau_fifo_engine fifo;
575 struct nouveau_display_engine display; 554 struct nouveau_display_engine display;
576 struct nouveau_gpio_engine gpio; 555 struct nouveau_gpio_engine gpio;
577 struct nouveau_pm_engine pm; 556 struct nouveau_pm_engine pm;
@@ -1183,52 +1162,6 @@ extern void nv50_fb_vm_trap(struct drm_device *, int display);
1183extern int nvc0_fb_init(struct drm_device *); 1162extern int nvc0_fb_init(struct drm_device *);
1184extern void nvc0_fb_takedown(struct drm_device *); 1163extern void nvc0_fb_takedown(struct drm_device *);
1185 1164
1186/* nv04_fifo.c */
1187extern int nv04_fifo_init(struct drm_device *);
1188extern void nv04_fifo_fini(struct drm_device *);
1189extern int nv04_fifo_create_context(struct nouveau_channel *);
1190extern void nv04_fifo_destroy_context(struct nouveau_channel *);
1191extern int nv04_fifo_load_context(struct nouveau_channel *);
1192extern int nv04_fifo_unload_context(struct drm_device *);
1193extern void nv04_fifo_isr(struct drm_device *);
1194bool nv04_fifo_cache_pull(struct drm_device *, bool enable);
1195
1196/* nv10_fifo.c */
1197extern int nv10_fifo_init(struct drm_device *);
1198extern int nv10_fifo_create_context(struct nouveau_channel *);
1199extern int nv10_fifo_load_context(struct nouveau_channel *);
1200extern int nv10_fifo_unload_context(struct drm_device *);
1201
1202/* nv40_fifo.c */
1203extern int nv40_fifo_init(struct drm_device *);
1204extern int nv40_fifo_create_context(struct nouveau_channel *);
1205extern int nv40_fifo_load_context(struct nouveau_channel *);
1206extern int nv40_fifo_unload_context(struct drm_device *);
1207
1208/* nv50_fifo.c */
1209extern int nv50_fifo_init(struct drm_device *);
1210extern void nv50_fifo_takedown(struct drm_device *);
1211extern int nv50_fifo_create_context(struct nouveau_channel *);
1212extern void nv50_fifo_destroy_context(struct nouveau_channel *);
1213extern int nv50_fifo_load_context(struct nouveau_channel *);
1214extern int nv50_fifo_unload_context(struct drm_device *);
1215extern void nv50_fifo_tlb_flush(struct drm_device *dev);
1216
1217/* nvc0_fifo.c */
1218extern int nvc0_fifo_init(struct drm_device *);
1219extern void nvc0_fifo_takedown(struct drm_device *);
1220extern int nvc0_fifo_create_context(struct nouveau_channel *);
1221extern void nvc0_fifo_destroy_context(struct nouveau_channel *);
1222extern int nvc0_fifo_load_context(struct nouveau_channel *);
1223extern int nvc0_fifo_unload_context(struct drm_device *);
1224
1225/* nve0_fifo.c */
1226extern int nve0_fifo_init(struct drm_device *);
1227extern void nve0_fifo_takedown(struct drm_device *);
1228extern int nve0_fifo_create_context(struct nouveau_channel *);
1229extern void nve0_fifo_destroy_context(struct nouveau_channel *);
1230extern int nve0_fifo_unload_context(struct drm_device *);
1231
1232/* nv04_graph.c */ 1165/* nv04_graph.c */
1233extern int nv04_graph_create(struct drm_device *); 1166extern int nv04_graph_create(struct drm_device *);
1234extern int nv04_graph_object_new(struct nouveau_channel *, int, u32, u16); 1167extern int nv04_graph_object_new(struct nouveau_channel *, int, u32, u16);
diff --git a/drivers/gpu/drm/nouveau/nouveau_fifo.h b/drivers/gpu/drm/nouveau/nouveau_fifo.h
new file mode 100644
index 000000000000..ce99cab2f257
--- /dev/null
+++ b/drivers/gpu/drm/nouveau/nouveau_fifo.h
@@ -0,0 +1,32 @@
1#ifndef __NOUVEAU_FIFO_H__
2#define __NOUVEAU_FIFO_H__
3
4struct nouveau_fifo_priv {
5 struct nouveau_exec_engine base;
6 u32 channels;
7};
8
9struct nouveau_fifo_chan {
10};
11
12bool nv04_fifo_cache_pull(struct drm_device *, bool);
13void nv04_fifo_context_del(struct nouveau_channel *, int);
14int nv04_fifo_fini(struct drm_device *, int, bool);
15int nv04_fifo_init(struct drm_device *, int);
16void nv04_fifo_isr(struct drm_device *);
17void nv04_fifo_destroy(struct drm_device *, int);
18
19void nv50_fifo_playlist_update(struct drm_device *);
20void nv50_fifo_destroy(struct drm_device *, int);
21void nv50_fifo_tlb_flush(struct drm_device *, int);
22
23int nv04_fifo_create(struct drm_device *);
24int nv10_fifo_create(struct drm_device *);
25int nv17_fifo_create(struct drm_device *);
26int nv40_fifo_create(struct drm_device *);
27int nv50_fifo_create(struct drm_device *);
28int nv84_fifo_create(struct drm_device *);
29int nvc0_fifo_create(struct drm_device *);
30int nve0_fifo_create(struct drm_device *);
31
32#endif
diff --git a/drivers/gpu/drm/nouveau/nouveau_mem.c b/drivers/gpu/drm/nouveau/nouveau_mem.c
index fd7273459ad6..5b498ea32e14 100644
--- a/drivers/gpu/drm/nouveau/nouveau_mem.c
+++ b/drivers/gpu/drm/nouveau/nouveau_mem.c
@@ -39,6 +39,7 @@
39#include "nouveau_pm.h" 39#include "nouveau_pm.h"
40#include "nouveau_mm.h" 40#include "nouveau_mm.h"
41#include "nouveau_vm.h" 41#include "nouveau_vm.h"
42#include "nouveau_fifo.h"
42#include "nouveau_fence.h" 43#include "nouveau_fence.h"
43 44
44/* 45/*
diff --git a/drivers/gpu/drm/nouveau/nouveau_object.c b/drivers/gpu/drm/nouveau/nouveau_object.c
index 8e0b38f35975..d7e56ce410b0 100644
--- a/drivers/gpu/drm/nouveau/nouveau_object.c
+++ b/drivers/gpu/drm/nouveau/nouveau_object.c
@@ -34,6 +34,7 @@
34#include "drm.h" 34#include "drm.h"
35#include "nouveau_drv.h" 35#include "nouveau_drv.h"
36#include "nouveau_drm.h" 36#include "nouveau_drm.h"
37#include "nouveau_fifo.h"
37#include "nouveau_ramht.h" 38#include "nouveau_ramht.h"
38#include "nouveau_software.h" 39#include "nouveau_software.h"
39#include "nouveau_vm.h" 40#include "nouveau_vm.h"
@@ -120,12 +121,13 @@ nouveau_gpuobj_mthd_call2(struct drm_device *dev, int chid,
120 u32 class, u32 mthd, u32 data) 121 u32 class, u32 mthd, u32 data)
121{ 122{
122 struct drm_nouveau_private *dev_priv = dev->dev_private; 123 struct drm_nouveau_private *dev_priv = dev->dev_private;
124 struct nouveau_fifo_priv *pfifo = nv_engine(dev, NVOBJ_ENGINE_FIFO);
123 struct nouveau_channel *chan = NULL; 125 struct nouveau_channel *chan = NULL;
124 unsigned long flags; 126 unsigned long flags;
125 int ret = -EINVAL; 127 int ret = -EINVAL;
126 128
127 spin_lock_irqsave(&dev_priv->channels.lock, flags); 129 spin_lock_irqsave(&dev_priv->channels.lock, flags);
128 if (chid >= 0 && chid < dev_priv->engine.fifo.channels) 130 if (chid >= 0 && chid < pfifo->channels)
129 chan = dev_priv->channels.ptr[chid]; 131 chan = dev_priv->channels.ptr[chid];
130 if (chan) 132 if (chan)
131 ret = nouveau_gpuobj_mthd_call(chan, class, mthd, data); 133 ret = nouveau_gpuobj_mthd_call(chan, class, mthd, data);
diff --git a/drivers/gpu/drm/nouveau/nouveau_state.c b/drivers/gpu/drm/nouveau/nouveau_state.c
index 660a033b6ddf..e4e73a13a2b2 100644
--- a/drivers/gpu/drm/nouveau/nouveau_state.c
+++ b/drivers/gpu/drm/nouveau/nouveau_state.c
@@ -39,6 +39,7 @@
39#include "nouveau_gpio.h" 39#include "nouveau_gpio.h"
40#include "nouveau_pm.h" 40#include "nouveau_pm.h"
41#include "nv50_display.h" 41#include "nv50_display.h"
42#include "nouveau_fifo.h"
42#include "nouveau_fence.h" 43#include "nouveau_fence.h"
43#include "nouveau_software.h" 44#include "nouveau_software.h"
44 45
@@ -68,13 +69,6 @@ static int nouveau_init_engine_ptrs(struct drm_device *dev)
68 engine->timer.takedown = nv04_timer_takedown; 69 engine->timer.takedown = nv04_timer_takedown;
69 engine->fb.init = nv04_fb_init; 70 engine->fb.init = nv04_fb_init;
70 engine->fb.takedown = nv04_fb_takedown; 71 engine->fb.takedown = nv04_fb_takedown;
71 engine->fifo.channels = 16;
72 engine->fifo.init = nv04_fifo_init;
73 engine->fifo.takedown = nv04_fifo_fini;
74 engine->fifo.create_context = nv04_fifo_create_context;
75 engine->fifo.destroy_context = nv04_fifo_destroy_context;
76 engine->fifo.load_context = nv04_fifo_load_context;
77 engine->fifo.unload_context = nv04_fifo_unload_context;
78 engine->display.early_init = nv04_display_early_init; 72 engine->display.early_init = nv04_display_early_init;
79 engine->display.late_takedown = nv04_display_late_takedown; 73 engine->display.late_takedown = nv04_display_late_takedown;
80 engine->display.create = nv04_display_create; 74 engine->display.create = nv04_display_create;
@@ -108,13 +102,6 @@ static int nouveau_init_engine_ptrs(struct drm_device *dev)
108 engine->fb.init_tile_region = nv10_fb_init_tile_region; 102 engine->fb.init_tile_region = nv10_fb_init_tile_region;
109 engine->fb.set_tile_region = nv10_fb_set_tile_region; 103 engine->fb.set_tile_region = nv10_fb_set_tile_region;
110 engine->fb.free_tile_region = nv10_fb_free_tile_region; 104 engine->fb.free_tile_region = nv10_fb_free_tile_region;
111 engine->fifo.channels = 32;
112 engine->fifo.init = nv10_fifo_init;
113 engine->fifo.takedown = nv04_fifo_fini;
114 engine->fifo.create_context = nv10_fifo_create_context;
115 engine->fifo.destroy_context = nv04_fifo_destroy_context;
116 engine->fifo.load_context = nv10_fifo_load_context;
117 engine->fifo.unload_context = nv10_fifo_unload_context;
118 engine->display.early_init = nv04_display_early_init; 105 engine->display.early_init = nv04_display_early_init;
119 engine->display.late_takedown = nv04_display_late_takedown; 106 engine->display.late_takedown = nv04_display_late_takedown;
120 engine->display.create = nv04_display_create; 107 engine->display.create = nv04_display_create;
@@ -154,13 +141,6 @@ static int nouveau_init_engine_ptrs(struct drm_device *dev)
154 engine->fb.init_tile_region = nv20_fb_init_tile_region; 141 engine->fb.init_tile_region = nv20_fb_init_tile_region;
155 engine->fb.set_tile_region = nv20_fb_set_tile_region; 142 engine->fb.set_tile_region = nv20_fb_set_tile_region;
156 engine->fb.free_tile_region = nv20_fb_free_tile_region; 143 engine->fb.free_tile_region = nv20_fb_free_tile_region;
157 engine->fifo.channels = 32;
158 engine->fifo.init = nv10_fifo_init;
159 engine->fifo.takedown = nv04_fifo_fini;
160 engine->fifo.create_context = nv10_fifo_create_context;
161 engine->fifo.destroy_context = nv04_fifo_destroy_context;
162 engine->fifo.load_context = nv10_fifo_load_context;
163 engine->fifo.unload_context = nv10_fifo_unload_context;
164 engine->display.early_init = nv04_display_early_init; 144 engine->display.early_init = nv04_display_early_init;
165 engine->display.late_takedown = nv04_display_late_takedown; 145 engine->display.late_takedown = nv04_display_late_takedown;
166 engine->display.create = nv04_display_create; 146 engine->display.create = nv04_display_create;
@@ -196,13 +176,6 @@ static int nouveau_init_engine_ptrs(struct drm_device *dev)
196 engine->fb.init_tile_region = nv30_fb_init_tile_region; 176 engine->fb.init_tile_region = nv30_fb_init_tile_region;
197 engine->fb.set_tile_region = nv10_fb_set_tile_region; 177 engine->fb.set_tile_region = nv10_fb_set_tile_region;
198 engine->fb.free_tile_region = nv30_fb_free_tile_region; 178 engine->fb.free_tile_region = nv30_fb_free_tile_region;
199 engine->fifo.channels = 32;
200 engine->fifo.init = nv10_fifo_init;
201 engine->fifo.takedown = nv04_fifo_fini;
202 engine->fifo.create_context = nv10_fifo_create_context;
203 engine->fifo.destroy_context = nv04_fifo_destroy_context;
204 engine->fifo.load_context = nv10_fifo_load_context;
205 engine->fifo.unload_context = nv10_fifo_unload_context;
206 engine->display.early_init = nv04_display_early_init; 179 engine->display.early_init = nv04_display_early_init;
207 engine->display.late_takedown = nv04_display_late_takedown; 180 engine->display.late_takedown = nv04_display_late_takedown;
208 engine->display.create = nv04_display_create; 181 engine->display.create = nv04_display_create;
@@ -241,13 +214,6 @@ static int nouveau_init_engine_ptrs(struct drm_device *dev)
241 engine->fb.init_tile_region = nv30_fb_init_tile_region; 214 engine->fb.init_tile_region = nv30_fb_init_tile_region;
242 engine->fb.set_tile_region = nv40_fb_set_tile_region; 215 engine->fb.set_tile_region = nv40_fb_set_tile_region;
243 engine->fb.free_tile_region = nv30_fb_free_tile_region; 216 engine->fb.free_tile_region = nv30_fb_free_tile_region;
244 engine->fifo.channels = 32;
245 engine->fifo.init = nv40_fifo_init;
246 engine->fifo.takedown = nv04_fifo_fini;
247 engine->fifo.create_context = nv40_fifo_create_context;
248 engine->fifo.destroy_context = nv04_fifo_destroy_context;
249 engine->fifo.load_context = nv40_fifo_load_context;
250 engine->fifo.unload_context = nv40_fifo_unload_context;
251 engine->display.early_init = nv04_display_early_init; 217 engine->display.early_init = nv04_display_early_init;
252 engine->display.late_takedown = nv04_display_late_takedown; 218 engine->display.late_takedown = nv04_display_late_takedown;
253 engine->display.create = nv04_display_create; 219 engine->display.create = nv04_display_create;
@@ -294,14 +260,6 @@ static int nouveau_init_engine_ptrs(struct drm_device *dev)
294 engine->timer.takedown = nv04_timer_takedown; 260 engine->timer.takedown = nv04_timer_takedown;
295 engine->fb.init = nv50_fb_init; 261 engine->fb.init = nv50_fb_init;
296 engine->fb.takedown = nv50_fb_takedown; 262 engine->fb.takedown = nv50_fb_takedown;
297 engine->fifo.channels = 128;
298 engine->fifo.init = nv50_fifo_init;
299 engine->fifo.takedown = nv50_fifo_takedown;
300 engine->fifo.create_context = nv50_fifo_create_context;
301 engine->fifo.destroy_context = nv50_fifo_destroy_context;
302 engine->fifo.load_context = nv50_fifo_load_context;
303 engine->fifo.unload_context = nv50_fifo_unload_context;
304 engine->fifo.tlb_flush = nv50_fifo_tlb_flush;
305 engine->display.early_init = nv50_display_early_init; 263 engine->display.early_init = nv50_display_early_init;
306 engine->display.late_takedown = nv50_display_late_takedown; 264 engine->display.late_takedown = nv50_display_late_takedown;
307 engine->display.create = nv50_display_create; 265 engine->display.create = nv50_display_create;
@@ -365,13 +323,6 @@ static int nouveau_init_engine_ptrs(struct drm_device *dev)
365 engine->timer.takedown = nv04_timer_takedown; 323 engine->timer.takedown = nv04_timer_takedown;
366 engine->fb.init = nvc0_fb_init; 324 engine->fb.init = nvc0_fb_init;
367 engine->fb.takedown = nvc0_fb_takedown; 325 engine->fb.takedown = nvc0_fb_takedown;
368 engine->fifo.channels = 128;
369 engine->fifo.init = nvc0_fifo_init;
370 engine->fifo.takedown = nvc0_fifo_takedown;
371 engine->fifo.create_context = nvc0_fifo_create_context;
372 engine->fifo.destroy_context = nvc0_fifo_destroy_context;
373 engine->fifo.load_context = nvc0_fifo_load_context;
374 engine->fifo.unload_context = nvc0_fifo_unload_context;
375 engine->display.early_init = nv50_display_early_init; 326 engine->display.early_init = nv50_display_early_init;
376 engine->display.late_takedown = nv50_display_late_takedown; 327 engine->display.late_takedown = nv50_display_late_takedown;
377 engine->display.create = nv50_display_create; 328 engine->display.create = nv50_display_create;
@@ -414,13 +365,6 @@ static int nouveau_init_engine_ptrs(struct drm_device *dev)
414 engine->timer.takedown = nv04_timer_takedown; 365 engine->timer.takedown = nv04_timer_takedown;
415 engine->fb.init = nvc0_fb_init; 366 engine->fb.init = nvc0_fb_init;
416 engine->fb.takedown = nvc0_fb_takedown; 367 engine->fb.takedown = nvc0_fb_takedown;
417 engine->fifo.channels = 128;
418 engine->fifo.init = nvc0_fifo_init;
419 engine->fifo.takedown = nvc0_fifo_takedown;
420 engine->fifo.create_context = nvc0_fifo_create_context;
421 engine->fifo.destroy_context = nvc0_fifo_destroy_context;
422 engine->fifo.load_context = nvc0_fifo_load_context;
423 engine->fifo.unload_context = nvc0_fifo_unload_context;
424 engine->display.early_init = nouveau_stub_init; 368 engine->display.early_init = nouveau_stub_init;
425 engine->display.late_takedown = nouveau_stub_takedown; 369 engine->display.late_takedown = nouveau_stub_takedown;
426 engine->display.create = nvd0_display_create; 370 engine->display.create = nvd0_display_create;
@@ -461,13 +405,6 @@ static int nouveau_init_engine_ptrs(struct drm_device *dev)
461 engine->timer.takedown = nv04_timer_takedown; 405 engine->timer.takedown = nv04_timer_takedown;
462 engine->fb.init = nvc0_fb_init; 406 engine->fb.init = nvc0_fb_init;
463 engine->fb.takedown = nvc0_fb_takedown; 407 engine->fb.takedown = nvc0_fb_takedown;
464 engine->fifo.channels = 4096;
465 engine->fifo.init = nve0_fifo_init;
466 engine->fifo.takedown = nve0_fifo_takedown;
467 engine->fifo.create_context = nve0_fifo_create_context;
468 engine->fifo.destroy_context = nve0_fifo_destroy_context;
469 engine->fifo.load_context = nvc0_fifo_load_context;
470 engine->fifo.unload_context = nve0_fifo_unload_context;
471 engine->display.early_init = nouveau_stub_init; 408 engine->display.early_init = nouveau_stub_init;
472 engine->display.late_takedown = nouveau_stub_takedown; 409 engine->display.late_takedown = nouveau_stub_takedown;
473 engine->display.create = nvd0_display_create; 410 engine->display.create = nvd0_display_create;
@@ -728,6 +665,38 @@ nouveau_card_init(struct drm_device *dev)
728 if (!dev_priv->noaccel) { 665 if (!dev_priv->noaccel) {
729 switch (dev_priv->card_type) { 666 switch (dev_priv->card_type) {
730 case NV_04: 667 case NV_04:
668 nv04_fifo_create(dev);
669 break;
670 case NV_10:
671 case NV_20:
672 case NV_30:
673 if (dev_priv->chipset < 0x17)
674 nv10_fifo_create(dev);
675 else
676 nv17_fifo_create(dev);
677 break;
678 case NV_40:
679 nv40_fifo_create(dev);
680 break;
681 case NV_50:
682 if (dev_priv->chipset == 0x50)
683 nv50_fifo_create(dev);
684 else
685 nv84_fifo_create(dev);
686 break;
687 case NV_C0:
688 case NV_D0:
689 nvc0_fifo_create(dev);
690 break;
691 case NV_E0:
692 nve0_fifo_create(dev);
693 break;
694 default:
695 break;
696 }
697
698 switch (dev_priv->card_type) {
699 case NV_04:
731 nv04_fence_create(dev); 700 nv04_fence_create(dev);
732 break; 701 break;
733 case NV_10: 702 case NV_10:
@@ -859,16 +828,11 @@ nouveau_card_init(struct drm_device *dev)
859 goto out_engine; 828 goto out_engine;
860 } 829 }
861 } 830 }
862
863 /* PFIFO */
864 ret = engine->fifo.init(dev);
865 if (ret)
866 goto out_engine;
867 } 831 }
868 832
869 ret = nouveau_irq_init(dev); 833 ret = nouveau_irq_init(dev);
870 if (ret) 834 if (ret)
871 goto out_fifo; 835 goto out_engine;
872 836
873 ret = nouveau_display_create(dev); 837 ret = nouveau_display_create(dev);
874 if (ret) 838 if (ret)
@@ -901,9 +865,6 @@ out_pm:
901 nouveau_display_destroy(dev); 865 nouveau_display_destroy(dev);
902out_irq: 866out_irq:
903 nouveau_irq_fini(dev); 867 nouveau_irq_fini(dev);
904out_fifo:
905 if (!dev_priv->noaccel)
906 engine->fifo.takedown(dev);
907out_engine: 868out_engine:
908 if (!dev_priv->noaccel) { 869 if (!dev_priv->noaccel) {
909 for (e = e - 1; e >= 0; e--) { 870 for (e = e - 1; e >= 0; e--) {
@@ -956,7 +917,6 @@ static void nouveau_card_takedown(struct drm_device *dev)
956 nouveau_display_destroy(dev); 917 nouveau_display_destroy(dev);
957 918
958 if (!dev_priv->noaccel) { 919 if (!dev_priv->noaccel) {
959 engine->fifo.takedown(dev);
960 for (e = NVOBJ_ENGINE_NR - 1; e >= 0; e--) { 920 for (e = NVOBJ_ENGINE_NR - 1; e >= 0; e--) {
961 if (dev_priv->eng[e]) { 921 if (dev_priv->eng[e]) {
962 dev_priv->eng[e]->fini(dev, e, false); 922 dev_priv->eng[e]->fini(dev, e, false);
diff --git a/drivers/gpu/drm/nouveau/nv04_fifo.c b/drivers/gpu/drm/nouveau/nv04_fifo.c
index 584c24d457f0..a6295cd00ec7 100644
--- a/drivers/gpu/drm/nouveau/nv04_fifo.c
+++ b/drivers/gpu/drm/nouveau/nv04_fifo.c
@@ -1,5 +1,5 @@
1/* 1/*
2 * Copyright (C) 2007 Ben Skeggs. 2 * Copyright (C) 2012 Ben Skeggs.
3 * All Rights Reserved. 3 * All Rights Reserved.
4 * 4 *
5 * Permission is hereby granted, free of charge, to any person obtaining 5 * Permission is hereby granted, free of charge, to any person obtaining
@@ -27,21 +27,38 @@
27#include "drmP.h" 27#include "drmP.h"
28#include "drm.h" 28#include "drm.h"
29#include "nouveau_drv.h" 29#include "nouveau_drv.h"
30#include "nouveau_ramht.h" 30#include "nouveau_fifo.h"
31#include "nouveau_util.h" 31#include "nouveau_util.h"
32 32#include "nouveau_ramht.h"
33#define NV04_RAMFC(c) (dev_priv->ramfc->pinst + ((c) * NV04_RAMFC__SIZE)) 33#include "nouveau_software.h"
34#define NV04_RAMFC__SIZE 32 34
35#define NV04_RAMFC_DMA_PUT 0x00 35static struct ramfc_desc {
36#define NV04_RAMFC_DMA_GET 0x04 36 unsigned bits:6;
37#define NV04_RAMFC_DMA_INSTANCE 0x08 37 unsigned ctxs:5;
38#define NV04_RAMFC_DMA_STATE 0x0C 38 unsigned ctxp:8;
39#define NV04_RAMFC_DMA_FETCH 0x10 39 unsigned regs:5;
40#define NV04_RAMFC_ENGINE 0x14 40 unsigned regp;
41#define NV04_RAMFC_PULL1_ENGINE 0x18 41} nv04_ramfc[] = {
42 42 { 32, 0, 0x00, 0, NV04_PFIFO_CACHE1_DMA_PUT },
43#define RAMFC_WR(offset, val) nv_wo32(chan->ramfc, NV04_RAMFC_##offset, (val)) 43 { 32, 0, 0x04, 0, NV04_PFIFO_CACHE1_DMA_GET },
44#define RAMFC_RD(offset) nv_ro32(chan->ramfc, NV04_RAMFC_##offset) 44 { 16, 0, 0x08, 0, NV04_PFIFO_CACHE1_DMA_INSTANCE },
45 { 16, 16, 0x08, 0, NV04_PFIFO_CACHE1_DMA_DCOUNT },
46 { 32, 0, 0x0c, 0, NV04_PFIFO_CACHE1_DMA_STATE },
47 { 32, 0, 0x10, 0, NV04_PFIFO_CACHE1_DMA_FETCH },
48 { 32, 0, 0x14, 0, NV04_PFIFO_CACHE1_ENGINE },
49 { 32, 0, 0x18, 0, NV04_PFIFO_CACHE1_PULL1 },
50 {}
51};
52
53struct nv04_fifo_priv {
54 struct nouveau_fifo_priv base;
55 struct ramfc_desc *ramfc_desc;
56};
57
58struct nv04_fifo_chan {
59 struct nouveau_fifo_chan base;
60 struct nouveau_gpuobj *ramfc;
61};
45 62
46bool 63bool
47nv04_fifo_cache_pull(struct drm_device *dev, bool enable) 64nv04_fifo_cache_pull(struct drm_device *dev, bool enable)
@@ -58,13 +75,13 @@ nv04_fifo_cache_pull(struct drm_device *dev, bool enable)
58 * invalidate the most recently calculated instance. 75 * invalidate the most recently calculated instance.
59 */ 76 */
60 if (!nv_wait(dev, NV04_PFIFO_CACHE1_PULL0, 77 if (!nv_wait(dev, NV04_PFIFO_CACHE1_PULL0,
61 NV04_PFIFO_CACHE1_PULL0_HASH_BUSY, 0)) 78 NV04_PFIFO_CACHE1_PULL0_HASH_BUSY, 0))
62 NV_ERROR(dev, "Timeout idling the PFIFO puller.\n"); 79 NV_ERROR(dev, "Timeout idling the PFIFO puller.\n");
63 80
64 if (nv_rd32(dev, NV04_PFIFO_CACHE1_PULL0) & 81 if (nv_rd32(dev, NV04_PFIFO_CACHE1_PULL0) &
65 NV04_PFIFO_CACHE1_PULL0_HASH_FAILED) 82 NV04_PFIFO_CACHE1_PULL0_HASH_FAILED)
66 nv_wr32(dev, NV03_PFIFO_INTR_0, 83 nv_wr32(dev, NV03_PFIFO_INTR_0,
67 NV_PFIFO_INTR_CACHE_ERROR); 84 NV_PFIFO_INTR_CACHE_ERROR);
68 85
69 nv_wr32(dev, NV04_PFIFO_CACHE1_HASH, 0); 86 nv_wr32(dev, NV04_PFIFO_CACHE1_HASH, 0);
70 } 87 }
@@ -72,238 +89,182 @@ nv04_fifo_cache_pull(struct drm_device *dev, bool enable)
72 return pull & 1; 89 return pull & 1;
73} 90}
74 91
75#ifdef __BIG_ENDIAN 92static int
76#define DMA_FETCH_ENDIANNESS NV_PFIFO_CACHE1_BIG_ENDIAN 93nv04_fifo_context_new(struct nouveau_channel *chan, int engine)
77#else
78#define DMA_FETCH_ENDIANNESS 0
79#endif
80
81int
82nv04_fifo_create_context(struct nouveau_channel *chan)
83{ 94{
84 struct drm_device *dev = chan->dev; 95 struct drm_device *dev = chan->dev;
85 struct drm_nouveau_private *dev_priv = dev->dev_private; 96 struct drm_nouveau_private *dev_priv = dev->dev_private;
97 struct nv04_fifo_priv *priv = nv_engine(dev, engine);
98 struct nv04_fifo_chan *fctx;
86 unsigned long flags; 99 unsigned long flags;
87 int ret; 100 int ret;
88 101
89 ret = nouveau_gpuobj_new_fake(dev, NV04_RAMFC(chan->id), ~0, 102 fctx = chan->engctx[engine] = kzalloc(sizeof(*fctx), GFP_KERNEL);
90 NV04_RAMFC__SIZE, 103 if (!fctx)
91 NVOBJ_FLAG_ZERO_ALLOC | 104 return -ENOMEM;
92 NVOBJ_FLAG_ZERO_FREE,
93 &chan->ramfc);
94 if (ret)
95 return ret;
96 105
106 /* map channel control registers */
97 chan->user = ioremap(pci_resource_start(dev->pdev, 0) + 107 chan->user = ioremap(pci_resource_start(dev->pdev, 0) +
98 NV03_USER(chan->id), PAGE_SIZE); 108 NV03_USER(chan->id), PAGE_SIZE);
99 if (!chan->user) 109 if (!chan->user) {
100 return -ENOMEM; 110 ret = -ENOMEM;
101 111 goto error;
102 spin_lock_irqsave(&dev_priv->context_switch_lock, flags); 112 }
103
104 /* Setup initial state */
105 RAMFC_WR(DMA_PUT, chan->pushbuf_base);
106 RAMFC_WR(DMA_GET, chan->pushbuf_base);
107 RAMFC_WR(DMA_INSTANCE, chan->pushbuf->pinst >> 4);
108 RAMFC_WR(DMA_FETCH, (NV_PFIFO_CACHE1_DMA_FETCH_TRIG_128_BYTES |
109 NV_PFIFO_CACHE1_DMA_FETCH_SIZE_128_BYTES |
110 NV_PFIFO_CACHE1_DMA_FETCH_MAX_REQS_8 |
111 DMA_FETCH_ENDIANNESS));
112 113
113 /* enable the fifo dma operation */ 114 /* initialise default fifo context */
114 nv_wr32(dev, NV04_PFIFO_MODE, 115 ret = nouveau_gpuobj_new_fake(dev, dev_priv->ramfc->pinst +
115 nv_rd32(dev, NV04_PFIFO_MODE) | (1 << chan->id)); 116 chan->id * 32, ~0, 32,
117 NVOBJ_FLAG_ZERO_FREE, &fctx->ramfc);
118 if (ret)
119 goto error;
120
121 nv_wo32(fctx->ramfc, 0x00, chan->pushbuf_base);
122 nv_wo32(fctx->ramfc, 0x04, chan->pushbuf_base);
123 nv_wo32(fctx->ramfc, 0x08, chan->pushbuf->pinst >> 4);
124 nv_wo32(fctx->ramfc, 0x0c, 0x00000000);
125 nv_wo32(fctx->ramfc, 0x10, NV_PFIFO_CACHE1_DMA_FETCH_TRIG_128_BYTES |
126 NV_PFIFO_CACHE1_DMA_FETCH_SIZE_128_BYTES |
127#ifdef __BIG_ENDIAN
128 NV_PFIFO_CACHE1_BIG_ENDIAN |
129#endif
130 NV_PFIFO_CACHE1_DMA_FETCH_MAX_REQS_8);
131 nv_wo32(fctx->ramfc, 0x14, 0x00000000);
132 nv_wo32(fctx->ramfc, 0x18, 0x00000000);
133 nv_wo32(fctx->ramfc, 0x1c, 0x00000000);
116 134
135 /* enable dma mode on the channel */
136 spin_lock_irqsave(&dev_priv->context_switch_lock, flags);
137 nv_mask(dev, NV04_PFIFO_MODE, (1 << chan->id), (1 << chan->id));
117 spin_unlock_irqrestore(&dev_priv->context_switch_lock, flags); 138 spin_unlock_irqrestore(&dev_priv->context_switch_lock, flags);
118 return 0; 139
140error:
141 if (ret)
142 priv->base.base.context_del(chan, engine);
143 return ret;
119} 144}
120 145
121void 146void
122nv04_fifo_destroy_context(struct nouveau_channel *chan) 147nv04_fifo_context_del(struct nouveau_channel *chan, int engine)
123{ 148{
124 struct drm_device *dev = chan->dev; 149 struct drm_device *dev = chan->dev;
125 struct drm_nouveau_private *dev_priv = dev->dev_private; 150 struct drm_nouveau_private *dev_priv = dev->dev_private;
126 struct nouveau_fifo_engine *pfifo = &dev_priv->engine.fifo; 151 struct nv04_fifo_priv *priv = nv_engine(chan->dev, engine);
152 struct nv04_fifo_chan *fctx = chan->engctx[engine];
153 struct ramfc_desc *c = priv->ramfc_desc;
127 unsigned long flags; 154 unsigned long flags;
155 int chid;
128 156
157 /* prevent fifo context switches */
129 spin_lock_irqsave(&dev_priv->context_switch_lock, flags); 158 spin_lock_irqsave(&dev_priv->context_switch_lock, flags);
130 nv_wr32(dev, NV03_PFIFO_CACHES, 0); 159 nv_wr32(dev, NV03_PFIFO_CACHES, 0);
131 160
132 /* Unload the context if it's the currently active one */ 161 /* if this channel is active, replace it with a null context */
133 if ((nv_rd32(dev, NV03_PFIFO_CACHE1_PUSH1) & 0xf) == chan->id) { 162 chid = nv_rd32(dev, NV03_PFIFO_CACHE1_PUSH1) & priv->base.channels;
163 if (chid == chan->id) {
134 nv_mask(dev, NV04_PFIFO_CACHE1_DMA_PUSH, 0x00000001, 0); 164 nv_mask(dev, NV04_PFIFO_CACHE1_DMA_PUSH, 0x00000001, 0);
135 nv_wr32(dev, NV03_PFIFO_CACHE1_PUSH0, 0); 165 nv_wr32(dev, NV03_PFIFO_CACHE1_PUSH0, 0);
136 nv_mask(dev, NV04_PFIFO_CACHE1_PULL0, 0x00000001, 0); 166 nv_mask(dev, NV04_PFIFO_CACHE1_PULL0, 0x00000001, 0);
137 pfifo->unload_context(dev); 167
168 do {
169 u32 mask = ((1ULL << c->bits) - 1) << c->regs;
170 nv_mask(dev, c->regp, mask, 0x00000000);
171 } while ((++c)->bits);
172
173 nv_wr32(dev, NV03_PFIFO_CACHE1_GET, 0);
174 nv_wr32(dev, NV03_PFIFO_CACHE1_PUT, 0);
175 nv_wr32(dev, NV03_PFIFO_CACHE1_PUSH1, priv->base.channels);
138 nv_wr32(dev, NV03_PFIFO_CACHE1_PUSH0, 1); 176 nv_wr32(dev, NV03_PFIFO_CACHE1_PUSH0, 1);
139 nv_wr32(dev, NV04_PFIFO_CACHE1_PULL0, 1); 177 nv_wr32(dev, NV04_PFIFO_CACHE1_PULL0, 1);
140 } 178 }
141 179
142 /* Keep it from being rescheduled */ 180 /* restore normal operation, after disabling dma mode */
143 nv_mask(dev, NV04_PFIFO_MODE, 1 << chan->id, 0); 181 nv_mask(dev, NV04_PFIFO_MODE, 1 << chan->id, 0);
144 nv_wr32(dev, NV03_PFIFO_CACHES, 1); 182 nv_wr32(dev, NV03_PFIFO_CACHES, 1);
145 spin_unlock_irqrestore(&dev_priv->context_switch_lock, flags); 183 spin_unlock_irqrestore(&dev_priv->context_switch_lock, flags);
146 184
147 /* Free the channel resources */ 185 /* clean up */
186 nouveau_gpuobj_ref(NULL, &fctx->ramfc);
187 nouveau_gpuobj_ref(NULL, &chan->ramfc); /*XXX: nv40 */
148 if (chan->user) { 188 if (chan->user) {
149 iounmap(chan->user); 189 iounmap(chan->user);
150 chan->user = NULL; 190 chan->user = NULL;
151 } 191 }
152 nouveau_gpuobj_ref(NULL, &chan->ramfc);
153}
154
155static void
156nv04_fifo_do_load_context(struct drm_device *dev, int chid)
157{
158 struct drm_nouveau_private *dev_priv = dev->dev_private;
159 uint32_t fc = NV04_RAMFC(chid), tmp;
160
161 nv_wr32(dev, NV04_PFIFO_CACHE1_DMA_PUT, nv_ri32(dev, fc + 0));
162 nv_wr32(dev, NV04_PFIFO_CACHE1_DMA_GET, nv_ri32(dev, fc + 4));
163 tmp = nv_ri32(dev, fc + 8);
164 nv_wr32(dev, NV04_PFIFO_CACHE1_DMA_INSTANCE, tmp & 0xFFFF);
165 nv_wr32(dev, NV04_PFIFO_CACHE1_DMA_DCOUNT, tmp >> 16);
166 nv_wr32(dev, NV04_PFIFO_CACHE1_DMA_STATE, nv_ri32(dev, fc + 12));
167 nv_wr32(dev, NV04_PFIFO_CACHE1_DMA_FETCH, nv_ri32(dev, fc + 16));
168 nv_wr32(dev, NV04_PFIFO_CACHE1_ENGINE, nv_ri32(dev, fc + 20));
169 nv_wr32(dev, NV04_PFIFO_CACHE1_PULL1, nv_ri32(dev, fc + 24));
170
171 nv_wr32(dev, NV03_PFIFO_CACHE1_GET, 0);
172 nv_wr32(dev, NV03_PFIFO_CACHE1_PUT, 0);
173} 192}
174 193
175int 194int
176nv04_fifo_load_context(struct nouveau_channel *chan) 195nv04_fifo_init(struct drm_device *dev, int engine)
177{
178 uint32_t tmp;
179
180 nv_wr32(chan->dev, NV03_PFIFO_CACHE1_PUSH1,
181 NV03_PFIFO_CACHE1_PUSH1_DMA | chan->id);
182 nv04_fifo_do_load_context(chan->dev, chan->id);
183 nv_wr32(chan->dev, NV04_PFIFO_CACHE1_DMA_PUSH, 1);
184
185 /* Reset NV04_PFIFO_CACHE1_DMA_CTL_AT_INFO to INVALID */
186 tmp = nv_rd32(chan->dev, NV04_PFIFO_CACHE1_DMA_CTL) & ~(1 << 31);
187 nv_wr32(chan->dev, NV04_PFIFO_CACHE1_DMA_CTL, tmp);
188
189 return 0;
190}
191
192int
193nv04_fifo_unload_context(struct drm_device *dev)
194{ 196{
195 struct drm_nouveau_private *dev_priv = dev->dev_private; 197 struct drm_nouveau_private *dev_priv = dev->dev_private;
196 struct nouveau_fifo_engine *pfifo = &dev_priv->engine.fifo; 198 struct nv04_fifo_priv *priv = nv_engine(dev, engine);
197 struct nouveau_channel *chan = NULL; 199 int i;
198 uint32_t tmp;
199 int chid;
200 200
201 chid = nv_rd32(dev, NV03_PFIFO_CACHE1_PUSH1) & 0xf; 201 nv_mask(dev, NV03_PMC_ENABLE, NV_PMC_ENABLE_PFIFO, 0);
202 if (chid < 0 || chid >= dev_priv->engine.fifo.channels) 202 nv_mask(dev, NV03_PMC_ENABLE, NV_PMC_ENABLE_PFIFO, NV_PMC_ENABLE_PFIFO);
203 return 0;
204 203
205 chan = dev_priv->channels.ptr[chid]; 204 nv_wr32(dev, NV04_PFIFO_DELAY_0, 0x000000ff);
206 if (!chan) { 205 nv_wr32(dev, NV04_PFIFO_DMA_TIMESLICE, 0x0101ffff);
207 NV_ERROR(dev, "Inactive channel on PFIFO: %d\n", chid);
208 return -EINVAL;
209 }
210
211 RAMFC_WR(DMA_PUT, nv_rd32(dev, NV04_PFIFO_CACHE1_DMA_PUT));
212 RAMFC_WR(DMA_GET, nv_rd32(dev, NV04_PFIFO_CACHE1_DMA_GET));
213 tmp = nv_rd32(dev, NV04_PFIFO_CACHE1_DMA_DCOUNT) << 16;
214 tmp |= nv_rd32(dev, NV04_PFIFO_CACHE1_DMA_INSTANCE);
215 RAMFC_WR(DMA_INSTANCE, tmp);
216 RAMFC_WR(DMA_STATE, nv_rd32(dev, NV04_PFIFO_CACHE1_DMA_STATE));
217 RAMFC_WR(DMA_FETCH, nv_rd32(dev, NV04_PFIFO_CACHE1_DMA_FETCH));
218 RAMFC_WR(ENGINE, nv_rd32(dev, NV04_PFIFO_CACHE1_ENGINE));
219 RAMFC_WR(PULL1_ENGINE, nv_rd32(dev, NV04_PFIFO_CACHE1_PULL1));
220
221 nv04_fifo_do_load_context(dev, pfifo->channels - 1);
222 nv_wr32(dev, NV03_PFIFO_CACHE1_PUSH1, pfifo->channels - 1);
223 return 0;
224}
225
226static void
227nv04_fifo_init_reset(struct drm_device *dev)
228{
229 nv_wr32(dev, NV03_PMC_ENABLE,
230 nv_rd32(dev, NV03_PMC_ENABLE) & ~NV_PMC_ENABLE_PFIFO);
231 nv_wr32(dev, NV03_PMC_ENABLE,
232 nv_rd32(dev, NV03_PMC_ENABLE) | NV_PMC_ENABLE_PFIFO);
233
234 nv_wr32(dev, 0x003224, 0x000f0078);
235 nv_wr32(dev, 0x002044, 0x0101ffff);
236 nv_wr32(dev, 0x002040, 0x000000ff);
237 nv_wr32(dev, 0x002500, 0x00000000);
238 nv_wr32(dev, 0x003000, 0x00000000);
239 nv_wr32(dev, 0x003050, 0x00000000);
240 nv_wr32(dev, 0x003200, 0x00000000);
241 nv_wr32(dev, 0x003250, 0x00000000);
242 nv_wr32(dev, 0x003220, 0x00000000);
243
244 nv_wr32(dev, 0x003250, 0x00000000);
245 nv_wr32(dev, 0x003270, 0x00000000);
246 nv_wr32(dev, 0x003210, 0x00000000);
247}
248
249static void
250nv04_fifo_init_ramxx(struct drm_device *dev)
251{
252 struct drm_nouveau_private *dev_priv = dev->dev_private;
253 206
254 nv_wr32(dev, NV03_PFIFO_RAMHT, (0x03 << 24) /* search 128 */ | 207 nv_wr32(dev, NV03_PFIFO_RAMHT, (0x03 << 24) /* search 128 */ |
255 ((dev_priv->ramht->bits - 9) << 16) | 208 ((dev_priv->ramht->bits - 9) << 16) |
256 (dev_priv->ramht->gpuobj->pinst >> 8)); 209 (dev_priv->ramht->gpuobj->pinst >> 8));
257 nv_wr32(dev, NV03_PFIFO_RAMRO, dev_priv->ramro->pinst >> 8); 210 nv_wr32(dev, NV03_PFIFO_RAMRO, dev_priv->ramro->pinst >> 8);
258 nv_wr32(dev, NV03_PFIFO_RAMFC, dev_priv->ramfc->pinst >> 8); 211 nv_wr32(dev, NV03_PFIFO_RAMFC, dev_priv->ramfc->pinst >> 8);
259}
260 212
261static void 213 nv_wr32(dev, NV03_PFIFO_CACHE1_PUSH1, priv->base.channels);
262nv04_fifo_init_intr(struct drm_device *dev)
263{
264 nouveau_irq_register(dev, 8, nv04_fifo_isr);
265 nv_wr32(dev, 0x002100, 0xffffffff);
266 nv_wr32(dev, 0x002140, 0xffffffff);
267}
268 214
269int 215 nv_wr32(dev, NV03_PFIFO_INTR_0, 0xffffffff);
270nv04_fifo_init(struct drm_device *dev) 216 nv_wr32(dev, NV03_PFIFO_INTR_EN_0, 0xffffffff);
271{
272 struct drm_nouveau_private *dev_priv = dev->dev_private;
273 struct nouveau_fifo_engine *pfifo = &dev_priv->engine.fifo;
274 int i;
275 217
276 nv04_fifo_init_reset(dev);
277 nv04_fifo_init_ramxx(dev);
278
279 nv04_fifo_do_load_context(dev, pfifo->channels - 1);
280 nv_wr32(dev, NV03_PFIFO_CACHE1_PUSH1, pfifo->channels - 1);
281
282 nv04_fifo_init_intr(dev);
283 nv_wr32(dev, NV03_PFIFO_CACHE1_PUSH0, 1); 218 nv_wr32(dev, NV03_PFIFO_CACHE1_PUSH0, 1);
284 nv_wr32(dev, NV04_PFIFO_CACHE1_PULL0, 1); 219 nv_wr32(dev, NV04_PFIFO_CACHE1_PULL0, 1);
285 nv_wr32(dev, NV03_PFIFO_CACHES, 1); 220 nv_wr32(dev, NV03_PFIFO_CACHES, 1);
286 221
287 for (i = 0; i < dev_priv->engine.fifo.channels; i++) { 222 for (i = 0; i < priv->base.channels; i++) {
288 if (dev_priv->channels.ptr[i]) { 223 if (dev_priv->channels.ptr[i])
289 uint32_t mode = nv_rd32(dev, NV04_PFIFO_MODE); 224 nv_mask(dev, NV04_PFIFO_MODE, (1 << i), (1 << i));
290 nv_wr32(dev, NV04_PFIFO_MODE, mode | (1 << i));
291 }
292 } 225 }
293 226
294 return 0; 227 return 0;
295} 228}
296 229
297void 230int
298nv04_fifo_fini(struct drm_device *dev) 231nv04_fifo_fini(struct drm_device *dev, int engine, bool suspend)
299{ 232{
300 nv_wr32(dev, 0x2140, 0x00000000); 233 struct drm_nouveau_private *dev_priv = dev->dev_private;
301 nouveau_irq_unregister(dev, 8); 234 struct nv04_fifo_priv *priv = nv_engine(dev, engine);
235 struct nouveau_channel *chan;
236 int chid;
237
238 /* prevent context switches and halt fifo operation */
239 nv_wr32(dev, NV03_PFIFO_CACHES, 0);
240 nv_wr32(dev, NV04_PFIFO_CACHE1_DMA_PUSH, 0);
241 nv_wr32(dev, NV03_PFIFO_CACHE1_PUSH0, 0);
242 nv_wr32(dev, NV04_PFIFO_CACHE1_PULL0, 0);
243
244 /* store current fifo context in ramfc */
245 chid = nv_rd32(dev, NV03_PFIFO_CACHE1_PUSH1) & priv->base.channels;
246 chan = dev_priv->channels.ptr[chid];
247 if (suspend && chid != priv->base.channels && chan) {
248 struct nv04_fifo_chan *fctx = chan->engctx[engine];
249 struct nouveau_gpuobj *ctx = fctx->ramfc;
250 struct ramfc_desc *c = priv->ramfc_desc;
251 do {
252 u32 rm = ((1ULL << c->bits) - 1) << c->regs;
253 u32 cm = ((1ULL << c->bits) - 1) << c->ctxs;
254 u32 rv = (nv_rd32(dev, c->regp) & rm) >> c->regs;
255 u32 cv = (nv_ro32(ctx, c->ctxp) & ~cm);
256 nv_wo32(ctx, c->ctxp, cv | (rv << c->ctxs));
257 } while ((++c)->bits);
258 }
259
260 nv_wr32(dev, NV03_PFIFO_INTR_EN_0, 0x00000000);
261 return 0;
302} 262}
303 263
304static bool 264static bool
305nouveau_fifo_swmthd(struct drm_device *dev, u32 chid, u32 addr, u32 data) 265nouveau_fifo_swmthd(struct drm_device *dev, u32 chid, u32 addr, u32 data)
306{ 266{
267 struct nouveau_fifo_priv *pfifo = nv_engine(dev, NVOBJ_ENGINE_FIFO);
307 struct drm_nouveau_private *dev_priv = dev->dev_private; 268 struct drm_nouveau_private *dev_priv = dev->dev_private;
308 struct nouveau_channel *chan = NULL; 269 struct nouveau_channel *chan = NULL;
309 struct nouveau_gpuobj *obj; 270 struct nouveau_gpuobj *obj;
@@ -314,7 +275,7 @@ nouveau_fifo_swmthd(struct drm_device *dev, u32 chid, u32 addr, u32 data)
314 u32 engine; 275 u32 engine;
315 276
316 spin_lock_irqsave(&dev_priv->channels.lock, flags); 277 spin_lock_irqsave(&dev_priv->channels.lock, flags);
317 if (likely(chid >= 0 && chid < dev_priv->engine.fifo.channels)) 278 if (likely(chid >= 0 && chid < pfifo->channels))
318 chan = dev_priv->channels.ptr[chid]; 279 chan = dev_priv->channels.ptr[chid];
319 if (unlikely(!chan)) 280 if (unlikely(!chan))
320 goto out; 281 goto out;
@@ -325,7 +286,6 @@ nouveau_fifo_swmthd(struct drm_device *dev, u32 chid, u32 addr, u32 data)
325 if (unlikely(!obj || obj->engine != NVOBJ_ENGINE_SW)) 286 if (unlikely(!obj || obj->engine != NVOBJ_ENGINE_SW))
326 break; 287 break;
327 288
328 chan->sw_subchannel[subc] = obj->class;
329 engine = 0x0000000f << (subc * 4); 289 engine = 0x0000000f << (subc * 4);
330 290
331 nv_mask(dev, NV04_PFIFO_CACHE1_ENGINE, engine, 0x00000000); 291 nv_mask(dev, NV04_PFIFO_CACHE1_ENGINE, engine, 0x00000000);
@@ -336,7 +296,7 @@ nouveau_fifo_swmthd(struct drm_device *dev, u32 chid, u32 addr, u32 data)
336 if (unlikely(((engine >> (subc * 4)) & 0xf) != 0)) 296 if (unlikely(((engine >> (subc * 4)) & 0xf) != 0))
337 break; 297 break;
338 298
339 if (!nouveau_gpuobj_mthd_call(chan, chan->sw_subchannel[subc], 299 if (!nouveau_gpuobj_mthd_call(chan, nouveau_software_class(dev),
340 mthd, data)) 300 mthd, data))
341 handled = true; 301 handled = true;
342 break; 302 break;
@@ -359,6 +319,7 @@ static const char *nv_dma_state_err(u32 state)
359void 319void
360nv04_fifo_isr(struct drm_device *dev) 320nv04_fifo_isr(struct drm_device *dev)
361{ 321{
322 struct nouveau_fifo_priv *pfifo = nv_engine(dev, NVOBJ_ENGINE_FIFO);
362 struct drm_nouveau_private *dev_priv = dev->dev_private; 323 struct drm_nouveau_private *dev_priv = dev->dev_private;
363 uint32_t status, reassign; 324 uint32_t status, reassign;
364 int cnt = 0; 325 int cnt = 0;
@@ -369,8 +330,7 @@ nv04_fifo_isr(struct drm_device *dev)
369 330
370 nv_wr32(dev, NV03_PFIFO_CACHES, 0); 331 nv_wr32(dev, NV03_PFIFO_CACHES, 0);
371 332
372 chid = nv_rd32(dev, NV03_PFIFO_CACHE1_PUSH1); 333 chid = nv_rd32(dev, NV03_PFIFO_CACHE1_PUSH1) & pfifo->channels;
373 chid &= dev_priv->engine.fifo.channels - 1;
374 get = nv_rd32(dev, NV03_PFIFO_CACHE1_GET); 334 get = nv_rd32(dev, NV03_PFIFO_CACHE1_GET);
375 335
376 if (status & NV_PFIFO_INTR_CACHE_ERROR) { 336 if (status & NV_PFIFO_INTR_CACHE_ERROR) {
@@ -509,3 +469,38 @@ nv04_fifo_isr(struct drm_device *dev)
509 469
510 nv_wr32(dev, NV03_PMC_INTR_0, NV_PMC_INTR_0_PFIFO_PENDING); 470 nv_wr32(dev, NV03_PMC_INTR_0, NV_PMC_INTR_0_PFIFO_PENDING);
511} 471}
472
473void
474nv04_fifo_destroy(struct drm_device *dev, int engine)
475{
476 struct drm_nouveau_private *dev_priv = dev->dev_private;
477 struct nv04_fifo_priv *priv = nv_engine(dev, engine);
478
479 nouveau_irq_unregister(dev, 8);
480
481 dev_priv->eng[engine] = NULL;
482 kfree(priv);
483}
484
485int
486nv04_fifo_create(struct drm_device *dev)
487{
488 struct drm_nouveau_private *dev_priv = dev->dev_private;
489 struct nv04_fifo_priv *priv;
490
491 priv = kzalloc(sizeof(*priv), GFP_KERNEL);
492 if (!priv)
493 return -ENOMEM;
494
495 priv->base.base.destroy = nv04_fifo_destroy;
496 priv->base.base.init = nv04_fifo_init;
497 priv->base.base.fini = nv04_fifo_fini;
498 priv->base.base.context_new = nv04_fifo_context_new;
499 priv->base.base.context_del = nv04_fifo_context_del;
500 priv->base.channels = 15;
501 priv->ramfc_desc = nv04_ramfc;
502 dev_priv->eng[NVOBJ_ENGINE_FIFO] = &priv->base.base;
503
504 nouveau_irq_register(dev, 8, nv04_fifo_isr);
505 return 0;
506}
diff --git a/drivers/gpu/drm/nouveau/nv04_graph.c b/drivers/gpu/drm/nouveau/nv04_graph.c
index 5b5f3ba85f5c..72f1a62903b3 100644
--- a/drivers/gpu/drm/nouveau/nv04_graph.c
+++ b/drivers/gpu/drm/nouveau/nv04_graph.c
@@ -356,12 +356,12 @@ static struct nouveau_channel *
356nv04_graph_channel(struct drm_device *dev) 356nv04_graph_channel(struct drm_device *dev)
357{ 357{
358 struct drm_nouveau_private *dev_priv = dev->dev_private; 358 struct drm_nouveau_private *dev_priv = dev->dev_private;
359 int chid = dev_priv->engine.fifo.channels; 359 int chid = 15;
360 360
361 if (nv_rd32(dev, NV04_PGRAPH_CTX_CONTROL) & 0x00010000) 361 if (nv_rd32(dev, NV04_PGRAPH_CTX_CONTROL) & 0x00010000)
362 chid = nv_rd32(dev, NV04_PGRAPH_CTX_USER) >> 24; 362 chid = nv_rd32(dev, NV04_PGRAPH_CTX_USER) >> 24;
363 363
364 if (chid >= dev_priv->engine.fifo.channels) 364 if (chid > 15)
365 return NULL; 365 return NULL;
366 366
367 return dev_priv->channels.ptr[chid]; 367 return dev_priv->channels.ptr[chid];
@@ -404,7 +404,6 @@ nv04_graph_load_context(struct nouveau_channel *chan)
404static int 404static int
405nv04_graph_unload_context(struct drm_device *dev) 405nv04_graph_unload_context(struct drm_device *dev)
406{ 406{
407 struct drm_nouveau_private *dev_priv = dev->dev_private;
408 struct nouveau_channel *chan = NULL; 407 struct nouveau_channel *chan = NULL;
409 struct graph_state *ctx; 408 struct graph_state *ctx;
410 uint32_t tmp; 409 uint32_t tmp;
@@ -420,7 +419,7 @@ nv04_graph_unload_context(struct drm_device *dev)
420 419
421 nv_wr32(dev, NV04_PGRAPH_CTX_CONTROL, 0x10000000); 420 nv_wr32(dev, NV04_PGRAPH_CTX_CONTROL, 0x10000000);
422 tmp = nv_rd32(dev, NV04_PGRAPH_CTX_USER) & 0x00ffffff; 421 tmp = nv_rd32(dev, NV04_PGRAPH_CTX_USER) & 0x00ffffff;
423 tmp |= (dev_priv->engine.fifo.channels - 1) << 24; 422 tmp |= 15 << 24;
424 nv_wr32(dev, NV04_PGRAPH_CTX_USER, tmp); 423 nv_wr32(dev, NV04_PGRAPH_CTX_USER, tmp);
425 return 0; 424 return 0;
426} 425}
@@ -495,7 +494,6 @@ nv04_graph_object_new(struct nouveau_channel *chan, int engine,
495static int 494static int
496nv04_graph_init(struct drm_device *dev, int engine) 495nv04_graph_init(struct drm_device *dev, int engine)
497{ 496{
498 struct drm_nouveau_private *dev_priv = dev->dev_private;
499 uint32_t tmp; 497 uint32_t tmp;
500 498
501 nv_wr32(dev, NV03_PMC_ENABLE, nv_rd32(dev, NV03_PMC_ENABLE) & 499 nv_wr32(dev, NV03_PMC_ENABLE, nv_rd32(dev, NV03_PMC_ENABLE) &
@@ -527,7 +525,7 @@ nv04_graph_init(struct drm_device *dev, int engine)
527 nv_wr32(dev, NV04_PGRAPH_STATE , 0xFFFFFFFF); 525 nv_wr32(dev, NV04_PGRAPH_STATE , 0xFFFFFFFF);
528 nv_wr32(dev, NV04_PGRAPH_CTX_CONTROL , 0x10000100); 526 nv_wr32(dev, NV04_PGRAPH_CTX_CONTROL , 0x10000100);
529 tmp = nv_rd32(dev, NV04_PGRAPH_CTX_USER) & 0x00ffffff; 527 tmp = nv_rd32(dev, NV04_PGRAPH_CTX_USER) & 0x00ffffff;
530 tmp |= (dev_priv->engine.fifo.channels - 1) << 24; 528 tmp |= 15 << 24;
531 nv_wr32(dev, NV04_PGRAPH_CTX_USER, tmp); 529 nv_wr32(dev, NV04_PGRAPH_CTX_USER, tmp);
532 530
533 /* These don't belong here, they're part of a per-channel context */ 531 /* These don't belong here, they're part of a per-channel context */
diff --git a/drivers/gpu/drm/nouveau/nv04_instmem.c b/drivers/gpu/drm/nouveau/nv04_instmem.c
index 1acc626f74b0..ef7a934a499a 100644
--- a/drivers/gpu/drm/nouveau/nv04_instmem.c
+++ b/drivers/gpu/drm/nouveau/nv04_instmem.c
@@ -1,6 +1,8 @@
1#include "drmP.h" 1#include "drmP.h"
2#include "drm.h" 2#include "drm.h"
3
3#include "nouveau_drv.h" 4#include "nouveau_drv.h"
5#include "nouveau_fifo.h"
4#include "nouveau_ramht.h" 6#include "nouveau_ramht.h"
5 7
6/* returns the size of fifo context */ 8/* returns the size of fifo context */
@@ -10,12 +12,15 @@ nouveau_fifo_ctx_size(struct drm_device *dev)
10 struct drm_nouveau_private *dev_priv = dev->dev_private; 12 struct drm_nouveau_private *dev_priv = dev->dev_private;
11 13
12 if (dev_priv->chipset >= 0x40) 14 if (dev_priv->chipset >= 0x40)
13 return 128; 15 return 128 * 32;
14 else 16 else
15 if (dev_priv->chipset >= 0x17) 17 if (dev_priv->chipset >= 0x17)
16 return 64; 18 return 64 * 32;
19 else
20 if (dev_priv->chipset >= 0x10)
21 return 32 * 32;
17 22
18 return 32; 23 return 32 * 16;
19} 24}
20 25
21int nv04_instmem_init(struct drm_device *dev) 26int nv04_instmem_init(struct drm_device *dev)
@@ -39,7 +44,7 @@ int nv04_instmem_init(struct drm_device *dev)
39 else if (nv44_graph_class(dev)) rsvd = 0x4980 * vs; 44 else if (nv44_graph_class(dev)) rsvd = 0x4980 * vs;
40 else rsvd = 0x4a40 * vs; 45 else rsvd = 0x4a40 * vs;
41 rsvd += 16 * 1024; 46 rsvd += 16 * 1024;
42 rsvd *= dev_priv->engine.fifo.channels; 47 rsvd *= 32; /* per-channel */
43 48
44 rsvd += 512 * 1024; /* pci(e)gart table */ 49 rsvd += 512 * 1024; /* pci(e)gart table */
45 rsvd += 512 * 1024; /* object storage */ 50 rsvd += 512 * 1024; /* object storage */
@@ -67,7 +72,7 @@ int nv04_instmem_init(struct drm_device *dev)
67 return ret; 72 return ret;
68 73
69 /* And RAMFC */ 74 /* And RAMFC */
70 length = dev_priv->engine.fifo.channels * nouveau_fifo_ctx_size(dev); 75 length = nouveau_fifo_ctx_size(dev);
71 switch (dev_priv->card_type) { 76 switch (dev_priv->card_type) {
72 case NV_40: 77 case NV_40:
73 offset = 0x20000; 78 offset = 0x20000;
diff --git a/drivers/gpu/drm/nouveau/nv10_fifo.c b/drivers/gpu/drm/nouveau/nv10_fifo.c
index 476451c6f961..f1fe7d758241 100644
--- a/drivers/gpu/drm/nouveau/nv10_fifo.c
+++ b/drivers/gpu/drm/nouveau/nv10_fifo.c
@@ -1,5 +1,5 @@
1/* 1/*
2 * Copyright (C) 2007 Ben Skeggs. 2 * Copyright (C) 2012 Ben Skeggs.
3 * All Rights Reserved. 3 * All Rights Reserved.
4 * 4 *
5 * Permission is hereby granted, free of charge, to any person obtaining 5 * Permission is hereby granted, free of charge, to any person obtaining
@@ -27,214 +27,112 @@
27#include "drmP.h" 27#include "drmP.h"
28#include "drm.h" 28#include "drm.h"
29#include "nouveau_drv.h" 29#include "nouveau_drv.h"
30#include "nouveau_fifo.h"
31#include "nouveau_util.h"
30#include "nouveau_ramht.h" 32#include "nouveau_ramht.h"
31 33
32#define NV10_RAMFC(c) (dev_priv->ramfc->pinst + ((c) * NV10_RAMFC__SIZE)) 34static struct ramfc_desc {
33#define NV10_RAMFC__SIZE ((dev_priv->chipset) >= 0x17 ? 64 : 32) 35 unsigned bits:6;
34 36 unsigned ctxs:5;
35int 37 unsigned ctxp:8;
36nv10_fifo_create_context(struct nouveau_channel *chan) 38 unsigned regs:5;
39 unsigned regp;
40} nv10_ramfc[] = {
41 { 32, 0, 0x00, 0, NV04_PFIFO_CACHE1_DMA_PUT },
42 { 32, 0, 0x04, 0, NV04_PFIFO_CACHE1_DMA_GET },
43 { 32, 0, 0x08, 0, NV10_PFIFO_CACHE1_REF_CNT },
44 { 16, 0, 0x0c, 0, NV04_PFIFO_CACHE1_DMA_INSTANCE },
45 { 16, 16, 0x0c, 0, NV04_PFIFO_CACHE1_DMA_DCOUNT },
46 { 32, 0, 0x10, 0, NV04_PFIFO_CACHE1_DMA_STATE },
47 { 32, 0, 0x14, 0, NV04_PFIFO_CACHE1_DMA_FETCH },
48 { 32, 0, 0x18, 0, NV04_PFIFO_CACHE1_ENGINE },
49 { 32, 0, 0x1c, 0, NV04_PFIFO_CACHE1_PULL1 },
50 {}
51};
52
53struct nv10_fifo_priv {
54 struct nouveau_fifo_priv base;
55 struct ramfc_desc *ramfc_desc;
56};
57
58struct nv10_fifo_chan {
59 struct nouveau_fifo_chan base;
60 struct nouveau_gpuobj *ramfc;
61};
62
63static int
64nv10_fifo_context_new(struct nouveau_channel *chan, int engine)
37{ 65{
38 struct drm_nouveau_private *dev_priv = chan->dev->dev_private;
39 struct drm_device *dev = chan->dev; 66 struct drm_device *dev = chan->dev;
40 uint32_t fc = NV10_RAMFC(chan->id); 67 struct drm_nouveau_private *dev_priv = dev->dev_private;
68 struct nv10_fifo_priv *priv = nv_engine(dev, engine);
69 struct nv10_fifo_chan *fctx;
70 unsigned long flags;
41 int ret; 71 int ret;
42 72
43 ret = nouveau_gpuobj_new_fake(dev, NV10_RAMFC(chan->id), ~0, 73 fctx = chan->engctx[engine] = kzalloc(sizeof(*fctx), GFP_KERNEL);
44 NV10_RAMFC__SIZE, NVOBJ_FLAG_ZERO_ALLOC | 74 if (!fctx)
45 NVOBJ_FLAG_ZERO_FREE, &chan->ramfc); 75 return -ENOMEM;
46 if (ret)
47 return ret;
48 76
77 /* map channel control registers */
49 chan->user = ioremap(pci_resource_start(dev->pdev, 0) + 78 chan->user = ioremap(pci_resource_start(dev->pdev, 0) +
50 NV03_USER(chan->id), PAGE_SIZE); 79 NV03_USER(chan->id), PAGE_SIZE);
51 if (!chan->user) 80 if (!chan->user) {
52 return -ENOMEM; 81 ret = -ENOMEM;
82 goto error;
83 }
53 84
54 /* Fill entries that are seen filled in dumps of nvidia driver just 85 /* initialise default fifo context */
55 * after channel's is put into DMA mode 86 ret = nouveau_gpuobj_new_fake(dev, dev_priv->ramfc->pinst +
56 */ 87 chan->id * 32, ~0, 32,
57 nv_wi32(dev, fc + 0, chan->pushbuf_base); 88 NVOBJ_FLAG_ZERO_FREE, &fctx->ramfc);
58 nv_wi32(dev, fc + 4, chan->pushbuf_base); 89 if (ret)
59 nv_wi32(dev, fc + 12, chan->pushbuf->pinst >> 4); 90 goto error;
60 nv_wi32(dev, fc + 20, NV_PFIFO_CACHE1_DMA_FETCH_TRIG_128_BYTES | 91
61 NV_PFIFO_CACHE1_DMA_FETCH_SIZE_128_BYTES | 92 nv_wo32(fctx->ramfc, 0x00, chan->pushbuf_base);
62 NV_PFIFO_CACHE1_DMA_FETCH_MAX_REQS_8 | 93 nv_wo32(fctx->ramfc, 0x04, chan->pushbuf_base);
94 nv_wo32(fctx->ramfc, 0x08, 0x00000000);
95 nv_wo32(fctx->ramfc, 0x0c, chan->pushbuf->pinst >> 4);
96 nv_wo32(fctx->ramfc, 0x10, 0x00000000);
97 nv_wo32(fctx->ramfc, 0x14, NV_PFIFO_CACHE1_DMA_FETCH_TRIG_128_BYTES |
98 NV_PFIFO_CACHE1_DMA_FETCH_SIZE_128_BYTES |
63#ifdef __BIG_ENDIAN 99#ifdef __BIG_ENDIAN
64 NV_PFIFO_CACHE1_BIG_ENDIAN | 100 NV_PFIFO_CACHE1_BIG_ENDIAN |
65#endif 101#endif
66 0); 102 NV_PFIFO_CACHE1_DMA_FETCH_MAX_REQS_8);
67 103 nv_wo32(fctx->ramfc, 0x18, 0x00000000);
68 /* enable the fifo dma operation */ 104 nv_wo32(fctx->ramfc, 0x1c, 0x00000000);
69 nv_wr32(dev, NV04_PFIFO_MODE,
70 nv_rd32(dev, NV04_PFIFO_MODE) | (1 << chan->id));
71 return 0;
72}
73
74static void
75nv10_fifo_do_load_context(struct drm_device *dev, int chid)
76{
77 struct drm_nouveau_private *dev_priv = dev->dev_private;
78 uint32_t fc = NV10_RAMFC(chid), tmp;
79
80 nv_wr32(dev, NV04_PFIFO_CACHE1_DMA_PUT, nv_ri32(dev, fc + 0));
81 nv_wr32(dev, NV04_PFIFO_CACHE1_DMA_GET, nv_ri32(dev, fc + 4));
82 nv_wr32(dev, NV10_PFIFO_CACHE1_REF_CNT, nv_ri32(dev, fc + 8));
83 105
84 tmp = nv_ri32(dev, fc + 12); 106 /* enable dma mode on the channel */
85 nv_wr32(dev, NV04_PFIFO_CACHE1_DMA_INSTANCE, tmp & 0xFFFF); 107 spin_lock_irqsave(&dev_priv->context_switch_lock, flags);
86 nv_wr32(dev, NV04_PFIFO_CACHE1_DMA_DCOUNT, tmp >> 16); 108 nv_mask(dev, NV04_PFIFO_MODE, (1 << chan->id), (1 << chan->id));
109 spin_unlock_irqrestore(&dev_priv->context_switch_lock, flags);
87 110
88 nv_wr32(dev, NV04_PFIFO_CACHE1_DMA_STATE, nv_ri32(dev, fc + 16)); 111error:
89 nv_wr32(dev, NV04_PFIFO_CACHE1_DMA_FETCH, nv_ri32(dev, fc + 20)); 112 if (ret)
90 nv_wr32(dev, NV04_PFIFO_CACHE1_ENGINE, nv_ri32(dev, fc + 24)); 113 priv->base.base.context_del(chan, engine);
91 nv_wr32(dev, NV04_PFIFO_CACHE1_PULL1, nv_ri32(dev, fc + 28)); 114 return ret;
92
93 if (dev_priv->chipset < 0x17)
94 goto out;
95
96 nv_wr32(dev, NV10_PFIFO_CACHE1_ACQUIRE_VALUE, nv_ri32(dev, fc + 32));
97 tmp = nv_ri32(dev, fc + 36);
98 nv_wr32(dev, NV10_PFIFO_CACHE1_ACQUIRE_TIMESTAMP, tmp);
99 nv_wr32(dev, NV10_PFIFO_CACHE1_ACQUIRE_TIMEOUT, nv_ri32(dev, fc + 40));
100 nv_wr32(dev, NV10_PFIFO_CACHE1_SEMAPHORE, nv_ri32(dev, fc + 44));
101 nv_wr32(dev, NV10_PFIFO_CACHE1_DMA_SUBROUTINE, nv_ri32(dev, fc + 48));
102
103out:
104 nv_wr32(dev, NV03_PFIFO_CACHE1_GET, 0);
105 nv_wr32(dev, NV03_PFIFO_CACHE1_PUT, 0);
106}
107
108int
109nv10_fifo_load_context(struct nouveau_channel *chan)
110{
111 struct drm_device *dev = chan->dev;
112 uint32_t tmp;
113
114 nv10_fifo_do_load_context(dev, chan->id);
115
116 nv_wr32(dev, NV03_PFIFO_CACHE1_PUSH1,
117 NV03_PFIFO_CACHE1_PUSH1_DMA | chan->id);
118 nv_wr32(dev, NV04_PFIFO_CACHE1_DMA_PUSH, 1);
119
120 /* Reset NV04_PFIFO_CACHE1_DMA_CTL_AT_INFO to INVALID */
121 tmp = nv_rd32(dev, NV04_PFIFO_CACHE1_DMA_CTL) & ~(1 << 31);
122 nv_wr32(dev, NV04_PFIFO_CACHE1_DMA_CTL, tmp);
123
124 return 0;
125} 115}
126 116
127int 117int
128nv10_fifo_unload_context(struct drm_device *dev) 118nv10_fifo_create(struct drm_device *dev)
129{ 119{
130 struct drm_nouveau_private *dev_priv = dev->dev_private; 120 struct drm_nouveau_private *dev_priv = dev->dev_private;
131 struct nouveau_fifo_engine *pfifo = &dev_priv->engine.fifo; 121 struct nv10_fifo_priv *priv;
132 uint32_t fc, tmp;
133 int chid;
134
135 chid = nv_rd32(dev, NV03_PFIFO_CACHE1_PUSH1) & 0x1f;
136 if (chid < 0 || chid >= dev_priv->engine.fifo.channels)
137 return 0;
138 fc = NV10_RAMFC(chid);
139 122
140 nv_wi32(dev, fc + 0, nv_rd32(dev, NV04_PFIFO_CACHE1_DMA_PUT)); 123 priv = kzalloc(sizeof(*priv), GFP_KERNEL);
141 nv_wi32(dev, fc + 4, nv_rd32(dev, NV04_PFIFO_CACHE1_DMA_GET)); 124 if (!priv)
142 nv_wi32(dev, fc + 8, nv_rd32(dev, NV10_PFIFO_CACHE1_REF_CNT)); 125 return -ENOMEM;
143 tmp = nv_rd32(dev, NV04_PFIFO_CACHE1_DMA_INSTANCE) & 0xFFFF;
144 tmp |= (nv_rd32(dev, NV04_PFIFO_CACHE1_DMA_DCOUNT) << 16);
145 nv_wi32(dev, fc + 12, tmp);
146 nv_wi32(dev, fc + 16, nv_rd32(dev, NV04_PFIFO_CACHE1_DMA_STATE));
147 nv_wi32(dev, fc + 20, nv_rd32(dev, NV04_PFIFO_CACHE1_DMA_FETCH));
148 nv_wi32(dev, fc + 24, nv_rd32(dev, NV04_PFIFO_CACHE1_ENGINE));
149 nv_wi32(dev, fc + 28, nv_rd32(dev, NV04_PFIFO_CACHE1_PULL1));
150
151 if (dev_priv->chipset < 0x17)
152 goto out;
153
154 nv_wi32(dev, fc + 32, nv_rd32(dev, NV10_PFIFO_CACHE1_ACQUIRE_VALUE));
155 tmp = nv_rd32(dev, NV10_PFIFO_CACHE1_ACQUIRE_TIMESTAMP);
156 nv_wi32(dev, fc + 36, tmp);
157 nv_wi32(dev, fc + 40, nv_rd32(dev, NV10_PFIFO_CACHE1_ACQUIRE_TIMEOUT));
158 nv_wi32(dev, fc + 44, nv_rd32(dev, NV10_PFIFO_CACHE1_SEMAPHORE));
159 nv_wi32(dev, fc + 48, nv_rd32(dev, NV04_PFIFO_CACHE1_DMA_GET));
160
161out:
162 nv10_fifo_do_load_context(dev, pfifo->channels - 1);
163 nv_wr32(dev, NV03_PFIFO_CACHE1_PUSH1, pfifo->channels - 1);
164 return 0;
165}
166
167static void
168nv10_fifo_init_reset(struct drm_device *dev)
169{
170 nv_wr32(dev, NV03_PMC_ENABLE,
171 nv_rd32(dev, NV03_PMC_ENABLE) & ~NV_PMC_ENABLE_PFIFO);
172 nv_wr32(dev, NV03_PMC_ENABLE,
173 nv_rd32(dev, NV03_PMC_ENABLE) | NV_PMC_ENABLE_PFIFO);
174
175 nv_wr32(dev, 0x003224, 0x000f0078);
176 nv_wr32(dev, 0x002044, 0x0101ffff);
177 nv_wr32(dev, 0x002040, 0x000000ff);
178 nv_wr32(dev, 0x002500, 0x00000000);
179 nv_wr32(dev, 0x003000, 0x00000000);
180 nv_wr32(dev, 0x003050, 0x00000000);
181
182 nv_wr32(dev, 0x003258, 0x00000000);
183 nv_wr32(dev, 0x003210, 0x00000000);
184 nv_wr32(dev, 0x003270, 0x00000000);
185}
186
187static void
188nv10_fifo_init_ramxx(struct drm_device *dev)
189{
190 struct drm_nouveau_private *dev_priv = dev->dev_private;
191 126
192 nv_wr32(dev, NV03_PFIFO_RAMHT, (0x03 << 24) /* search 128 */ | 127 priv->base.base.destroy = nv04_fifo_destroy;
193 ((dev_priv->ramht->bits - 9) << 16) | 128 priv->base.base.init = nv04_fifo_init;
194 (dev_priv->ramht->gpuobj->pinst >> 8)); 129 priv->base.base.fini = nv04_fifo_fini;
195 nv_wr32(dev, NV03_PFIFO_RAMRO, dev_priv->ramro->pinst >> 8); 130 priv->base.base.context_new = nv10_fifo_context_new;
131 priv->base.base.context_del = nv04_fifo_context_del;
132 priv->base.channels = 31;
133 priv->ramfc_desc = nv10_ramfc;
134 dev_priv->eng[NVOBJ_ENGINE_FIFO] = &priv->base.base;
196 135
197 if (dev_priv->chipset < 0x17) {
198 nv_wr32(dev, NV03_PFIFO_RAMFC, dev_priv->ramfc->pinst >> 8);
199 } else {
200 nv_wr32(dev, NV03_PFIFO_RAMFC, (dev_priv->ramfc->pinst >> 8) |
201 (1 << 16) /* 64 Bytes entry*/);
202 /* XXX nvidia blob set bit 18, 21,23 for nv20 & nv30 */
203 }
204}
205
206static void
207nv10_fifo_init_intr(struct drm_device *dev)
208{
209 nouveau_irq_register(dev, 8, nv04_fifo_isr); 136 nouveau_irq_register(dev, 8, nv04_fifo_isr);
210 nv_wr32(dev, 0x002100, 0xffffffff);
211 nv_wr32(dev, 0x002140, 0xffffffff);
212}
213
214int
215nv10_fifo_init(struct drm_device *dev)
216{
217 struct drm_nouveau_private *dev_priv = dev->dev_private;
218 struct nouveau_fifo_engine *pfifo = &dev_priv->engine.fifo;
219 int i;
220
221 nv10_fifo_init_reset(dev);
222 nv10_fifo_init_ramxx(dev);
223
224 nv10_fifo_do_load_context(dev, pfifo->channels - 1);
225 nv_wr32(dev, NV03_PFIFO_CACHE1_PUSH1, pfifo->channels - 1);
226
227 nv10_fifo_init_intr(dev);
228 nv_wr32(dev, NV03_PFIFO_CACHE1_PUSH0, 1);
229 nv_wr32(dev, NV04_PFIFO_CACHE1_PULL0, 1);
230 nv_wr32(dev, NV03_PFIFO_CACHES, 1);
231
232 for (i = 0; i < dev_priv->engine.fifo.channels; i++) {
233 if (dev_priv->channels.ptr[i]) {
234 uint32_t mode = nv_rd32(dev, NV04_PFIFO_MODE);
235 nv_wr32(dev, NV04_PFIFO_MODE, mode | (1 << i));
236 }
237 }
238
239 return 0; 137 return 0;
240} 138}
diff --git a/drivers/gpu/drm/nouveau/nv10_graph.c b/drivers/gpu/drm/nouveau/nv10_graph.c
index 10c0eb5d4233..fb1d88a951de 100644
--- a/drivers/gpu/drm/nouveau/nv10_graph.c
+++ b/drivers/gpu/drm/nouveau/nv10_graph.c
@@ -759,7 +759,6 @@ static int
759nv10_graph_unload_context(struct drm_device *dev) 759nv10_graph_unload_context(struct drm_device *dev)
760{ 760{
761 struct drm_nouveau_private *dev_priv = dev->dev_private; 761 struct drm_nouveau_private *dev_priv = dev->dev_private;
762 struct nouveau_fifo_engine *pfifo = &dev_priv->engine.fifo;
763 struct nouveau_channel *chan; 762 struct nouveau_channel *chan;
764 struct graph_state *ctx; 763 struct graph_state *ctx;
765 uint32_t tmp; 764 uint32_t tmp;
@@ -782,7 +781,7 @@ nv10_graph_unload_context(struct drm_device *dev)
782 781
783 nv_wr32(dev, NV10_PGRAPH_CTX_CONTROL, 0x10000000); 782 nv_wr32(dev, NV10_PGRAPH_CTX_CONTROL, 0x10000000);
784 tmp = nv_rd32(dev, NV10_PGRAPH_CTX_USER) & 0x00ffffff; 783 tmp = nv_rd32(dev, NV10_PGRAPH_CTX_USER) & 0x00ffffff;
785 tmp |= (pfifo->channels - 1) << 24; 784 tmp |= 31 << 24;
786 nv_wr32(dev, NV10_PGRAPH_CTX_USER, tmp); 785 nv_wr32(dev, NV10_PGRAPH_CTX_USER, tmp);
787 return 0; 786 return 0;
788} 787}
@@ -822,12 +821,12 @@ struct nouveau_channel *
822nv10_graph_channel(struct drm_device *dev) 821nv10_graph_channel(struct drm_device *dev)
823{ 822{
824 struct drm_nouveau_private *dev_priv = dev->dev_private; 823 struct drm_nouveau_private *dev_priv = dev->dev_private;
825 int chid = dev_priv->engine.fifo.channels; 824 int chid = 31;
826 825
827 if (nv_rd32(dev, NV10_PGRAPH_CTX_CONTROL) & 0x00010000) 826 if (nv_rd32(dev, NV10_PGRAPH_CTX_CONTROL) & 0x00010000)
828 chid = nv_rd32(dev, NV10_PGRAPH_CTX_USER) >> 24; 827 chid = nv_rd32(dev, NV10_PGRAPH_CTX_USER) >> 24;
829 828
830 if (chid >= dev_priv->engine.fifo.channels) 829 if (chid >= 31)
831 return NULL; 830 return NULL;
832 831
833 return dev_priv->channels.ptr[chid]; 832 return dev_priv->channels.ptr[chid];
@@ -948,7 +947,7 @@ nv10_graph_init(struct drm_device *dev, int engine)
948 nv_wr32(dev, NV10_PGRAPH_STATE, 0xFFFFFFFF); 947 nv_wr32(dev, NV10_PGRAPH_STATE, 0xFFFFFFFF);
949 948
950 tmp = nv_rd32(dev, NV10_PGRAPH_CTX_USER) & 0x00ffffff; 949 tmp = nv_rd32(dev, NV10_PGRAPH_CTX_USER) & 0x00ffffff;
951 tmp |= (dev_priv->engine.fifo.channels - 1) << 24; 950 tmp |= 31 << 24;
952 nv_wr32(dev, NV10_PGRAPH_CTX_USER, tmp); 951 nv_wr32(dev, NV10_PGRAPH_CTX_USER, tmp);
953 nv_wr32(dev, NV10_PGRAPH_CTX_CONTROL, 0x10000100); 952 nv_wr32(dev, NV10_PGRAPH_CTX_CONTROL, 0x10000100);
954 nv_wr32(dev, NV10_PGRAPH_FFINTFC_ST2, 0x08000000); 953 nv_wr32(dev, NV10_PGRAPH_FFINTFC_ST2, 0x08000000);
diff --git a/drivers/gpu/drm/nouveau/nv17_fifo.c b/drivers/gpu/drm/nouveau/nv17_fifo.c
new file mode 100644
index 000000000000..d9e482e4abee
--- /dev/null
+++ b/drivers/gpu/drm/nouveau/nv17_fifo.c
@@ -0,0 +1,177 @@
1/*
2 * Copyright (C) 2012 Ben Skeggs.
3 * All Rights Reserved.
4 *
5 * Permission is hereby granted, free of charge, to any person obtaining
6 * a copy of this software and associated documentation files (the
7 * "Software"), to deal in the Software without restriction, including
8 * without limitation the rights to use, copy, modify, merge, publish,
9 * distribute, sublicense, and/or sell copies of the Software, and to
10 * permit persons to whom the Software is furnished to do so, subject to
11 * the following conditions:
12 *
13 * The above copyright notice and this permission notice (including the
14 * next paragraph) shall be included in all copies or substantial
15 * portions of the Software.
16 *
17 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
18 * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
19 * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
20 * IN NO EVENT SHALL THE COPYRIGHT OWNER(S) AND/OR ITS SUPPLIERS BE
21 * LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
22 * OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
23 * WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
24 *
25 */
26
27#include "drmP.h"
28#include "drm.h"
29#include "nouveau_drv.h"
30#include "nouveau_fifo.h"
31#include "nouveau_util.h"
32#include "nouveau_ramht.h"
33
34static struct ramfc_desc {
35 unsigned bits:6;
36 unsigned ctxs:5;
37 unsigned ctxp:8;
38 unsigned regs:5;
39 unsigned regp;
40} nv17_ramfc[] = {
41 { 32, 0, 0x00, 0, NV04_PFIFO_CACHE1_DMA_PUT },
42 { 32, 0, 0x04, 0, NV04_PFIFO_CACHE1_DMA_GET },
43 { 32, 0, 0x08, 0, NV10_PFIFO_CACHE1_REF_CNT },
44 { 16, 0, 0x0c, 0, NV04_PFIFO_CACHE1_DMA_INSTANCE },
45 { 16, 16, 0x0c, 0, NV04_PFIFO_CACHE1_DMA_DCOUNT },
46 { 32, 0, 0x10, 0, NV04_PFIFO_CACHE1_DMA_STATE },
47 { 32, 0, 0x14, 0, NV04_PFIFO_CACHE1_DMA_FETCH },
48 { 32, 0, 0x18, 0, NV04_PFIFO_CACHE1_ENGINE },
49 { 32, 0, 0x1c, 0, NV04_PFIFO_CACHE1_PULL1 },
50 { 32, 0, 0x20, 0, NV10_PFIFO_CACHE1_ACQUIRE_VALUE },
51 { 32, 0, 0x24, 0, NV10_PFIFO_CACHE1_ACQUIRE_TIMESTAMP },
52 { 32, 0, 0x28, 0, NV10_PFIFO_CACHE1_ACQUIRE_TIMEOUT },
53 { 32, 0, 0x2c, 0, NV10_PFIFO_CACHE1_SEMAPHORE },
54 { 32, 0, 0x30, 0, NV10_PFIFO_CACHE1_DMA_SUBROUTINE },
55 {}
56};
57
58struct nv17_fifo_priv {
59 struct nouveau_fifo_priv base;
60 struct ramfc_desc *ramfc_desc;
61};
62
63struct nv17_fifo_chan {
64 struct nouveau_fifo_chan base;
65 struct nouveau_gpuobj *ramfc;
66};
67
68static int
69nv17_fifo_context_new(struct nouveau_channel *chan, int engine)
70{
71 struct drm_device *dev = chan->dev;
72 struct drm_nouveau_private *dev_priv = dev->dev_private;
73 struct nv17_fifo_priv *priv = nv_engine(dev, engine);
74 struct nv17_fifo_chan *fctx;
75 unsigned long flags;
76 int ret;
77
78 fctx = chan->engctx[engine] = kzalloc(sizeof(*fctx), GFP_KERNEL);
79 if (!fctx)
80 return -ENOMEM;
81
82 /* map channel control registers */
83 chan->user = ioremap(pci_resource_start(dev->pdev, 0) +
84 NV03_USER(chan->id), PAGE_SIZE);
85 if (!chan->user) {
86 ret = -ENOMEM;
87 goto error;
88 }
89
90 /* initialise default fifo context */
91 ret = nouveau_gpuobj_new_fake(dev, dev_priv->ramfc->pinst +
92 chan->id * 64, ~0, 64,
93 NVOBJ_FLAG_ZERO_ALLOC |
94 NVOBJ_FLAG_ZERO_FREE, &fctx->ramfc);
95 if (ret)
96 goto error;
97
98 nv_wo32(fctx->ramfc, 0x00, chan->pushbuf_base);
99 nv_wo32(fctx->ramfc, 0x04, chan->pushbuf_base);
100 nv_wo32(fctx->ramfc, 0x0c, chan->pushbuf->pinst >> 4);
101 nv_wo32(fctx->ramfc, 0x14, NV_PFIFO_CACHE1_DMA_FETCH_TRIG_128_BYTES |
102 NV_PFIFO_CACHE1_DMA_FETCH_SIZE_128_BYTES |
103#ifdef __BIG_ENDIAN
104 NV_PFIFO_CACHE1_BIG_ENDIAN |
105#endif
106 NV_PFIFO_CACHE1_DMA_FETCH_MAX_REQS_8);
107
108 /* enable dma mode on the channel */
109 spin_lock_irqsave(&dev_priv->context_switch_lock, flags);
110 nv_mask(dev, NV04_PFIFO_MODE, (1 << chan->id), (1 << chan->id));
111 spin_unlock_irqrestore(&dev_priv->context_switch_lock, flags);
112
113error:
114 if (ret)
115 priv->base.base.context_del(chan, engine);
116 return ret;
117}
118
119static int
120nv17_fifo_init(struct drm_device *dev, int engine)
121{
122 struct drm_nouveau_private *dev_priv = dev->dev_private;
123 struct nv17_fifo_priv *priv = nv_engine(dev, engine);
124 int i;
125
126 nv_mask(dev, NV03_PMC_ENABLE, NV_PMC_ENABLE_PFIFO, 0);
127 nv_mask(dev, NV03_PMC_ENABLE, NV_PMC_ENABLE_PFIFO, NV_PMC_ENABLE_PFIFO);
128
129 nv_wr32(dev, NV04_PFIFO_DELAY_0, 0x000000ff);
130 nv_wr32(dev, NV04_PFIFO_DMA_TIMESLICE, 0x0101ffff);
131
132 nv_wr32(dev, NV03_PFIFO_RAMHT, (0x03 << 24) /* search 128 */ |
133 ((dev_priv->ramht->bits - 9) << 16) |
134 (dev_priv->ramht->gpuobj->pinst >> 8));
135 nv_wr32(dev, NV03_PFIFO_RAMRO, dev_priv->ramro->pinst >> 8);
136 nv_wr32(dev, NV03_PFIFO_RAMFC, 0x00010000 |
137 dev_priv->ramfc->pinst >> 8);
138
139 nv_wr32(dev, NV03_PFIFO_CACHE1_PUSH1, priv->base.channels);
140
141 nv_wr32(dev, NV03_PFIFO_INTR_0, 0xffffffff);
142 nv_wr32(dev, NV03_PFIFO_INTR_EN_0, 0xffffffff);
143
144 nv_wr32(dev, NV03_PFIFO_CACHE1_PUSH0, 1);
145 nv_wr32(dev, NV04_PFIFO_CACHE1_PULL0, 1);
146 nv_wr32(dev, NV03_PFIFO_CACHES, 1);
147
148 for (i = 0; i < priv->base.channels; i++) {
149 if (dev_priv->channels.ptr[i])
150 nv_mask(dev, NV04_PFIFO_MODE, (1 << i), (1 << i));
151 }
152
153 return 0;
154}
155
156int
157nv17_fifo_create(struct drm_device *dev)
158{
159 struct drm_nouveau_private *dev_priv = dev->dev_private;
160 struct nv17_fifo_priv *priv;
161
162 priv = kzalloc(sizeof(*priv), GFP_KERNEL);
163 if (!priv)
164 return -ENOMEM;
165
166 priv->base.base.destroy = nv04_fifo_destroy;
167 priv->base.base.init = nv17_fifo_init;
168 priv->base.base.fini = nv04_fifo_fini;
169 priv->base.base.context_new = nv17_fifo_context_new;
170 priv->base.base.context_del = nv04_fifo_context_del;
171 priv->base.channels = 31;
172 priv->ramfc_desc = nv17_ramfc;
173 dev_priv->eng[NVOBJ_ENGINE_FIFO] = &priv->base.base;
174
175 nouveau_irq_register(dev, 8, nv04_fifo_isr);
176 return 0;
177}
diff --git a/drivers/gpu/drm/nouveau/nv20_graph.c b/drivers/gpu/drm/nouveau/nv20_graph.c
index 385e2b49a554..e34ea30758f6 100644
--- a/drivers/gpu/drm/nouveau/nv20_graph.c
+++ b/drivers/gpu/drm/nouveau/nv20_graph.c
@@ -43,8 +43,6 @@ struct nv20_graph_engine {
43int 43int
44nv20_graph_unload_context(struct drm_device *dev) 44nv20_graph_unload_context(struct drm_device *dev)
45{ 45{
46 struct drm_nouveau_private *dev_priv = dev->dev_private;
47 struct nouveau_fifo_engine *pfifo = &dev_priv->engine.fifo;
48 struct nouveau_channel *chan; 46 struct nouveau_channel *chan;
49 struct nouveau_gpuobj *grctx; 47 struct nouveau_gpuobj *grctx;
50 u32 tmp; 48 u32 tmp;
@@ -62,7 +60,7 @@ nv20_graph_unload_context(struct drm_device *dev)
62 60
63 nv_wr32(dev, NV10_PGRAPH_CTX_CONTROL, 0x10000000); 61 nv_wr32(dev, NV10_PGRAPH_CTX_CONTROL, 0x10000000);
64 tmp = nv_rd32(dev, NV10_PGRAPH_CTX_USER) & 0x00ffffff; 62 tmp = nv_rd32(dev, NV10_PGRAPH_CTX_USER) & 0x00ffffff;
65 tmp |= (pfifo->channels - 1) << 24; 63 tmp |= 31 << 24;
66 nv_wr32(dev, NV10_PGRAPH_CTX_USER, tmp); 64 nv_wr32(dev, NV10_PGRAPH_CTX_USER, tmp);
67 return 0; 65 return 0;
68} 66}
diff --git a/drivers/gpu/drm/nouveau/nv31_mpeg.c b/drivers/gpu/drm/nouveau/nv31_mpeg.c
index 6f06a0713f00..5f239bf658c4 100644
--- a/drivers/gpu/drm/nouveau/nv31_mpeg.c
+++ b/drivers/gpu/drm/nouveau/nv31_mpeg.c
@@ -24,6 +24,7 @@
24 24
25#include "drmP.h" 25#include "drmP.h"
26#include "nouveau_drv.h" 26#include "nouveau_drv.h"
27#include "nouveau_fifo.h"
27#include "nouveau_ramht.h" 28#include "nouveau_ramht.h"
28 29
29struct nv31_mpeg_engine { 30struct nv31_mpeg_engine {
@@ -208,6 +209,7 @@ nv31_mpeg_mthd_dma(struct nouveau_channel *chan, u32 class, u32 mthd, u32 data)
208static int 209static int
209nv31_mpeg_isr_chid(struct drm_device *dev, u32 inst) 210nv31_mpeg_isr_chid(struct drm_device *dev, u32 inst)
210{ 211{
212 struct nouveau_fifo_priv *pfifo = nv_engine(dev, NVOBJ_ENGINE_FIFO);
211 struct drm_nouveau_private *dev_priv = dev->dev_private; 213 struct drm_nouveau_private *dev_priv = dev->dev_private;
212 struct nouveau_gpuobj *ctx; 214 struct nouveau_gpuobj *ctx;
213 unsigned long flags; 215 unsigned long flags;
@@ -218,7 +220,7 @@ nv31_mpeg_isr_chid(struct drm_device *dev, u32 inst)
218 return 0; 220 return 0;
219 221
220 spin_lock_irqsave(&dev_priv->channels.lock, flags); 222 spin_lock_irqsave(&dev_priv->channels.lock, flags);
221 for (i = 0; i < dev_priv->engine.fifo.channels; i++) { 223 for (i = 0; i < pfifo->channels; i++) {
222 if (!dev_priv->channels.ptr[i]) 224 if (!dev_priv->channels.ptr[i])
223 continue; 225 continue;
224 226
diff --git a/drivers/gpu/drm/nouveau/nv40_fifo.c b/drivers/gpu/drm/nouveau/nv40_fifo.c
index 8d346617f55f..cdc818479b0a 100644
--- a/drivers/gpu/drm/nouveau/nv40_fifo.c
+++ b/drivers/gpu/drm/nouveau/nv40_fifo.c
@@ -1,5 +1,5 @@
1/* 1/*
2 * Copyright (C) 2007 Ben Skeggs. 2 * Copyright (C) 2012 Ben Skeggs.
3 * All Rights Reserved. 3 * All Rights Reserved.
4 * 4 *
5 * Permission is hereby granted, free of charge, to any person obtaining 5 * Permission is hereby granted, free of charge, to any person obtaining
@@ -25,215 +25,123 @@
25 */ 25 */
26 26
27#include "drmP.h" 27#include "drmP.h"
28#include "drm.h"
28#include "nouveau_drv.h" 29#include "nouveau_drv.h"
29#include "nouveau_drm.h" 30#include "nouveau_fifo.h"
31#include "nouveau_util.h"
30#include "nouveau_ramht.h" 32#include "nouveau_ramht.h"
31 33
32#define NV40_RAMFC(c) (dev_priv->ramfc->pinst + ((c) * NV40_RAMFC__SIZE)) 34static struct ramfc_desc {
33#define NV40_RAMFC__SIZE 128 35 unsigned bits:6;
34 36 unsigned ctxs:5;
35int 37 unsigned ctxp:8;
36nv40_fifo_create_context(struct nouveau_channel *chan) 38 unsigned regs:5;
39 unsigned regp;
40} nv40_ramfc[] = {
41 { 32, 0, 0x00, 0, NV04_PFIFO_CACHE1_DMA_PUT },
42 { 32, 0, 0x04, 0, NV04_PFIFO_CACHE1_DMA_GET },
43 { 32, 0, 0x08, 0, NV10_PFIFO_CACHE1_REF_CNT },
44 { 32, 0, 0x0c, 0, NV04_PFIFO_CACHE1_DMA_INSTANCE },
45 { 32, 0, 0x10, 0, NV04_PFIFO_CACHE1_DMA_DCOUNT },
46 { 32, 0, 0x14, 0, NV04_PFIFO_CACHE1_DMA_STATE },
47 { 28, 0, 0x18, 0, NV04_PFIFO_CACHE1_DMA_FETCH },
48 { 2, 28, 0x18, 28, 0x002058 },
49 { 32, 0, 0x1c, 0, NV04_PFIFO_CACHE1_ENGINE },
50 { 32, 0, 0x20, 0, NV04_PFIFO_CACHE1_PULL1 },
51 { 32, 0, 0x24, 0, NV10_PFIFO_CACHE1_ACQUIRE_VALUE },
52 { 32, 0, 0x28, 0, NV10_PFIFO_CACHE1_ACQUIRE_TIMESTAMP },
53 { 32, 0, 0x2c, 0, NV10_PFIFO_CACHE1_ACQUIRE_TIMEOUT },
54 { 32, 0, 0x30, 0, NV10_PFIFO_CACHE1_SEMAPHORE },
55 { 32, 0, 0x34, 0, NV10_PFIFO_CACHE1_DMA_SUBROUTINE },
56 { 32, 0, 0x38, 0, NV40_PFIFO_GRCTX_INSTANCE },
57 { 17, 0, 0x3c, 0, NV04_PFIFO_DMA_TIMESLICE },
58 { 32, 0, 0x40, 0, 0x0032e4 },
59 { 32, 0, 0x44, 0, 0x0032e8 },
60 { 32, 0, 0x4c, 0, 0x002088 },
61 { 32, 0, 0x50, 0, 0x003300 },
62 { 32, 0, 0x54, 0, 0x00330c },
63 {}
64};
65
66struct nv40_fifo_priv {
67 struct nouveau_fifo_priv base;
68 struct ramfc_desc *ramfc_desc;
69};
70
71struct nv40_fifo_chan {
72 struct nouveau_fifo_chan base;
73 struct nouveau_gpuobj *ramfc;
74};
75
76static int
77nv40_fifo_context_new(struct nouveau_channel *chan, int engine)
37{ 78{
38 struct drm_device *dev = chan->dev; 79 struct drm_device *dev = chan->dev;
39 struct drm_nouveau_private *dev_priv = dev->dev_private; 80 struct drm_nouveau_private *dev_priv = dev->dev_private;
40 uint32_t fc = NV40_RAMFC(chan->id); 81 struct nv40_fifo_priv *priv = nv_engine(dev, engine);
82 struct nv40_fifo_chan *fctx;
41 unsigned long flags; 83 unsigned long flags;
42 int ret; 84 int ret;
43 85
44 ret = nouveau_gpuobj_new_fake(dev, NV40_RAMFC(chan->id), ~0, 86 fctx = chan->engctx[engine] = kzalloc(sizeof(*fctx), GFP_KERNEL);
45 NV40_RAMFC__SIZE, NVOBJ_FLAG_ZERO_ALLOC | 87 if (!fctx)
46 NVOBJ_FLAG_ZERO_FREE, &chan->ramfc);
47 if (ret)
48 return ret;
49
50 chan->user = ioremap(pci_resource_start(dev->pdev, 0) +
51 NV40_USER(chan->id), PAGE_SIZE);
52 if (!chan->user)
53 return -ENOMEM; 88 return -ENOMEM;
54 89
55 spin_lock_irqsave(&dev_priv->context_switch_lock, flags); 90 /* map channel control registers */
91 chan->user = ioremap(pci_resource_start(dev->pdev, 0) +
92 NV03_USER(chan->id), PAGE_SIZE);
93 if (!chan->user) {
94 ret = -ENOMEM;
95 goto error;
96 }
56 97
57 nv_wi32(dev, fc + 0, chan->pushbuf_base); 98 /* initialise default fifo context */
58 nv_wi32(dev, fc + 4, chan->pushbuf_base); 99 ret = nouveau_gpuobj_new_fake(dev, dev_priv->ramfc->pinst +
59 nv_wi32(dev, fc + 12, chan->pushbuf->pinst >> 4); 100 chan->id * 128, ~0, 128,
60 nv_wi32(dev, fc + 24, NV_PFIFO_CACHE1_DMA_FETCH_TRIG_128_BYTES | 101 NVOBJ_FLAG_ZERO_ALLOC |
61 NV_PFIFO_CACHE1_DMA_FETCH_SIZE_128_BYTES | 102 NVOBJ_FLAG_ZERO_FREE, &fctx->ramfc);
62 NV_PFIFO_CACHE1_DMA_FETCH_MAX_REQS_8 | 103 if (ret)
104 goto error;
105
106 nv_wo32(fctx->ramfc, 0x00, chan->pushbuf_base);
107 nv_wo32(fctx->ramfc, 0x04, chan->pushbuf_base);
108 nv_wo32(fctx->ramfc, 0x0c, chan->pushbuf->pinst >> 4);
109 nv_wo32(fctx->ramfc, 0x18, 0x30000000 |
110 NV_PFIFO_CACHE1_DMA_FETCH_TRIG_128_BYTES |
111 NV_PFIFO_CACHE1_DMA_FETCH_SIZE_128_BYTES |
63#ifdef __BIG_ENDIAN 112#ifdef __BIG_ENDIAN
64 NV_PFIFO_CACHE1_BIG_ENDIAN | 113 NV_PFIFO_CACHE1_BIG_ENDIAN |
65#endif 114#endif
66 0x30000000 /* no idea.. */); 115 NV_PFIFO_CACHE1_DMA_FETCH_MAX_REQS_8);
67 nv_wi32(dev, fc + 60, 0x0001FFFF); 116 nv_wo32(fctx->ramfc, 0x3c, 0x0001ffff);
68
69 /* enable the fifo dma operation */
70 nv_wr32(dev, NV04_PFIFO_MODE,
71 nv_rd32(dev, NV04_PFIFO_MODE) | (1 << chan->id));
72 117
118 /* enable dma mode on the channel */
119 spin_lock_irqsave(&dev_priv->context_switch_lock, flags);
120 nv_mask(dev, NV04_PFIFO_MODE, (1 << chan->id), (1 << chan->id));
73 spin_unlock_irqrestore(&dev_priv->context_switch_lock, flags); 121 spin_unlock_irqrestore(&dev_priv->context_switch_lock, flags);
74 return 0;
75}
76
77static void
78nv40_fifo_do_load_context(struct drm_device *dev, int chid)
79{
80 struct drm_nouveau_private *dev_priv = dev->dev_private;
81 uint32_t fc = NV40_RAMFC(chid), tmp, tmp2;
82
83 nv_wr32(dev, NV04_PFIFO_CACHE1_DMA_PUT, nv_ri32(dev, fc + 0));
84 nv_wr32(dev, NV04_PFIFO_CACHE1_DMA_GET, nv_ri32(dev, fc + 4));
85 nv_wr32(dev, NV10_PFIFO_CACHE1_REF_CNT, nv_ri32(dev, fc + 8));
86 nv_wr32(dev, NV04_PFIFO_CACHE1_DMA_INSTANCE, nv_ri32(dev, fc + 12));
87 nv_wr32(dev, NV04_PFIFO_CACHE1_DMA_DCOUNT, nv_ri32(dev, fc + 16));
88 nv_wr32(dev, NV04_PFIFO_CACHE1_DMA_STATE, nv_ri32(dev, fc + 20));
89
90 /* No idea what 0x2058 is.. */
91 tmp = nv_ri32(dev, fc + 24);
92 tmp2 = nv_rd32(dev, 0x2058) & 0xFFF;
93 tmp2 |= (tmp & 0x30000000);
94 nv_wr32(dev, 0x2058, tmp2);
95 tmp &= ~0x30000000;
96 nv_wr32(dev, NV04_PFIFO_CACHE1_DMA_FETCH, tmp);
97
98 nv_wr32(dev, NV04_PFIFO_CACHE1_ENGINE, nv_ri32(dev, fc + 28));
99 nv_wr32(dev, NV04_PFIFO_CACHE1_PULL1, nv_ri32(dev, fc + 32));
100 nv_wr32(dev, NV10_PFIFO_CACHE1_ACQUIRE_VALUE, nv_ri32(dev, fc + 36));
101 tmp = nv_ri32(dev, fc + 40);
102 nv_wr32(dev, NV10_PFIFO_CACHE1_ACQUIRE_TIMESTAMP, tmp);
103 nv_wr32(dev, NV10_PFIFO_CACHE1_ACQUIRE_TIMEOUT, nv_ri32(dev, fc + 44));
104 nv_wr32(dev, NV10_PFIFO_CACHE1_SEMAPHORE, nv_ri32(dev, fc + 48));
105 nv_wr32(dev, NV10_PFIFO_CACHE1_DMA_SUBROUTINE, nv_ri32(dev, fc + 52));
106 nv_wr32(dev, NV40_PFIFO_GRCTX_INSTANCE, nv_ri32(dev, fc + 56));
107
108 /* Don't clobber the TIMEOUT_ENABLED flag when restoring from RAMFC */
109 tmp = nv_rd32(dev, NV04_PFIFO_DMA_TIMESLICE) & ~0x1FFFF;
110 tmp |= nv_ri32(dev, fc + 60) & 0x1FFFF;
111 nv_wr32(dev, NV04_PFIFO_DMA_TIMESLICE, tmp);
112 122
113 nv_wr32(dev, 0x32e4, nv_ri32(dev, fc + 64)); 123 /*XXX: remove this later, need fifo engine context commit hook */
114 /* NVIDIA does this next line twice... */ 124 nouveau_gpuobj_ref(fctx->ramfc, &chan->ramfc);
115 nv_wr32(dev, 0x32e8, nv_ri32(dev, fc + 68));
116 nv_wr32(dev, 0x2088, nv_ri32(dev, fc + 76));
117 nv_wr32(dev, 0x3300, nv_ri32(dev, fc + 80));
118 nv_wr32(dev, 0x330c, nv_ri32(dev, fc + 84));
119 125
120 nv_wr32(dev, NV03_PFIFO_CACHE1_GET, 0); 126error:
121 nv_wr32(dev, NV03_PFIFO_CACHE1_PUT, 0); 127 if (ret)
122} 128 priv->base.base.context_del(chan, engine);
123 129 return ret;
124int
125nv40_fifo_load_context(struct nouveau_channel *chan)
126{
127 struct drm_device *dev = chan->dev;
128 uint32_t tmp;
129
130 nv40_fifo_do_load_context(dev, chan->id);
131
132 /* Set channel active, and in DMA mode */
133 nv_wr32(dev, NV03_PFIFO_CACHE1_PUSH1,
134 NV40_PFIFO_CACHE1_PUSH1_DMA | chan->id);
135 nv_wr32(dev, NV04_PFIFO_CACHE1_DMA_PUSH, 1);
136
137 /* Reset DMA_CTL_AT_INFO to INVALID */
138 tmp = nv_rd32(dev, NV04_PFIFO_CACHE1_DMA_CTL) & ~(1 << 31);
139 nv_wr32(dev, NV04_PFIFO_CACHE1_DMA_CTL, tmp);
140
141 return 0;
142} 130}
143 131
144int 132static int
145nv40_fifo_unload_context(struct drm_device *dev) 133nv40_fifo_init(struct drm_device *dev, int engine)
146{ 134{
147 struct drm_nouveau_private *dev_priv = dev->dev_private; 135 struct drm_nouveau_private *dev_priv = dev->dev_private;
148 struct nouveau_fifo_engine *pfifo = &dev_priv->engine.fifo; 136 struct nv40_fifo_priv *priv = nv_engine(dev, engine);
149 uint32_t fc, tmp;
150 int chid;
151
152 chid = nv_rd32(dev, NV03_PFIFO_CACHE1_PUSH1) & 0x1f;
153 if (chid < 0 || chid >= dev_priv->engine.fifo.channels)
154 return 0;
155 fc = NV40_RAMFC(chid);
156
157 nv_wi32(dev, fc + 0, nv_rd32(dev, NV04_PFIFO_CACHE1_DMA_PUT));
158 nv_wi32(dev, fc + 4, nv_rd32(dev, NV04_PFIFO_CACHE1_DMA_GET));
159 nv_wi32(dev, fc + 8, nv_rd32(dev, NV10_PFIFO_CACHE1_REF_CNT));
160 nv_wi32(dev, fc + 12, nv_rd32(dev, NV04_PFIFO_CACHE1_DMA_INSTANCE));
161 nv_wi32(dev, fc + 16, nv_rd32(dev, NV04_PFIFO_CACHE1_DMA_DCOUNT));
162 nv_wi32(dev, fc + 20, nv_rd32(dev, NV04_PFIFO_CACHE1_DMA_STATE));
163 tmp = nv_rd32(dev, NV04_PFIFO_CACHE1_DMA_FETCH);
164 tmp |= nv_rd32(dev, 0x2058) & 0x30000000;
165 nv_wi32(dev, fc + 24, tmp);
166 nv_wi32(dev, fc + 28, nv_rd32(dev, NV04_PFIFO_CACHE1_ENGINE));
167 nv_wi32(dev, fc + 32, nv_rd32(dev, NV04_PFIFO_CACHE1_PULL1));
168 nv_wi32(dev, fc + 36, nv_rd32(dev, NV10_PFIFO_CACHE1_ACQUIRE_VALUE));
169 tmp = nv_rd32(dev, NV10_PFIFO_CACHE1_ACQUIRE_TIMESTAMP);
170 nv_wi32(dev, fc + 40, tmp);
171 nv_wi32(dev, fc + 44, nv_rd32(dev, NV10_PFIFO_CACHE1_ACQUIRE_TIMEOUT));
172 nv_wi32(dev, fc + 48, nv_rd32(dev, NV10_PFIFO_CACHE1_SEMAPHORE));
173 /* NVIDIA read 0x3228 first, then write DMA_GET here.. maybe something
174 * more involved depending on the value of 0x3228?
175 */
176 nv_wi32(dev, fc + 52, nv_rd32(dev, NV04_PFIFO_CACHE1_DMA_GET));
177 nv_wi32(dev, fc + 56, nv_rd32(dev, NV40_PFIFO_GRCTX_INSTANCE));
178 nv_wi32(dev, fc + 60, nv_rd32(dev, NV04_PFIFO_DMA_TIMESLICE) & 0x1ffff);
179 /* No idea what the below is for exactly, ripped from a mmio-trace */
180 nv_wi32(dev, fc + 64, nv_rd32(dev, NV40_PFIFO_UNK32E4));
181 /* NVIDIA do this next line twice.. bug? */
182 nv_wi32(dev, fc + 68, nv_rd32(dev, 0x32e8));
183 nv_wi32(dev, fc + 76, nv_rd32(dev, 0x2088));
184 nv_wi32(dev, fc + 80, nv_rd32(dev, 0x3300));
185#if 0 /* no real idea which is PUT/GET in UNK_48.. */
186 tmp = nv_rd32(dev, NV04_PFIFO_CACHE1_GET);
187 tmp |= (nv_rd32(dev, NV04_PFIFO_CACHE1_PUT) << 16);
188 nv_wi32(dev, fc + 72, tmp);
189#endif
190 nv_wi32(dev, fc + 84, nv_rd32(dev, 0x330c));
191
192 nv40_fifo_do_load_context(dev, pfifo->channels - 1);
193 nv_wr32(dev, NV03_PFIFO_CACHE1_PUSH1,
194 NV40_PFIFO_CACHE1_PUSH1_DMA | (pfifo->channels - 1));
195 return 0;
196}
197
198static void
199nv40_fifo_init_reset(struct drm_device *dev)
200{
201 int i; 137 int i;
202 138
203 nv_wr32(dev, NV03_PMC_ENABLE, 139 nv_mask(dev, NV03_PMC_ENABLE, NV_PMC_ENABLE_PFIFO, 0);
204 nv_rd32(dev, NV03_PMC_ENABLE) & ~NV_PMC_ENABLE_PFIFO); 140 nv_mask(dev, NV03_PMC_ENABLE, NV_PMC_ENABLE_PFIFO, NV_PMC_ENABLE_PFIFO);
205 nv_wr32(dev, NV03_PMC_ENABLE,
206 nv_rd32(dev, NV03_PMC_ENABLE) | NV_PMC_ENABLE_PFIFO);
207 141
208 nv_wr32(dev, 0x003224, 0x000f0078);
209 nv_wr32(dev, 0x003210, 0x00000000);
210 nv_wr32(dev, 0x003270, 0x00000000);
211 nv_wr32(dev, 0x003240, 0x00000000);
212 nv_wr32(dev, 0x003244, 0x00000000);
213 nv_wr32(dev, 0x003258, 0x00000000);
214 nv_wr32(dev, 0x002504, 0x00000000);
215 for (i = 0; i < 16; i++)
216 nv_wr32(dev, 0x002510 + (i * 4), 0x00000000);
217 nv_wr32(dev, 0x00250c, 0x0000ffff);
218 nv_wr32(dev, 0x002048, 0x00000000);
219 nv_wr32(dev, 0x003228, 0x00000000);
220 nv_wr32(dev, 0x0032e8, 0x00000000);
221 nv_wr32(dev, 0x002410, 0x00000000);
222 nv_wr32(dev, 0x002420, 0x00000000);
223 nv_wr32(dev, 0x002058, 0x00000001);
224 nv_wr32(dev, 0x00221c, 0x00000000);
225 /* something with 0x2084, read/modify/write, no change */
226 nv_wr32(dev, 0x002040, 0x000000ff); 142 nv_wr32(dev, 0x002040, 0x000000ff);
227 nv_wr32(dev, 0x002500, 0x00000000); 143 nv_wr32(dev, 0x002044, 0x2101ffff);
228 nv_wr32(dev, 0x003200, 0x00000000); 144 nv_wr32(dev, 0x002058, 0x00000001);
229
230 nv_wr32(dev, NV04_PFIFO_DMA_TIMESLICE, 0x2101ffff);
231}
232
233static void
234nv40_fifo_init_ramxx(struct drm_device *dev)
235{
236 struct drm_nouveau_private *dev_priv = dev->dev_private;
237 145
238 nv_wr32(dev, NV03_PFIFO_RAMHT, (0x03 << 24) /* search 128 */ | 146 nv_wr32(dev, NV03_PFIFO_RAMHT, (0x03 << 24) /* search 128 */ |
239 ((dev_priv->ramht->bits - 9) << 16) | 147 ((dev_priv->ramht->bits - 9) << 16) |
@@ -244,65 +152,59 @@ nv40_fifo_init_ramxx(struct drm_device *dev)
244 case 0x47: 152 case 0x47:
245 case 0x49: 153 case 0x49:
246 case 0x4b: 154 case 0x4b:
247 nv_wr32(dev, 0x2230, 1); 155 nv_wr32(dev, 0x002230, 0x00000001);
248 break;
249 default:
250 break;
251 }
252
253 switch (dev_priv->chipset) {
254 case 0x40: 156 case 0x40:
255 case 0x41: 157 case 0x41:
256 case 0x42: 158 case 0x42:
257 case 0x43: 159 case 0x43:
258 case 0x45: 160 case 0x45:
259 case 0x47:
260 case 0x48: 161 case 0x48:
261 case 0x49: 162 nv_wr32(dev, 0x002220, 0x00030002);
262 case 0x4b:
263 nv_wr32(dev, NV40_PFIFO_RAMFC, 0x30002);
264 break; 163 break;
265 default: 164 default:
266 nv_wr32(dev, 0x2230, 0); 165 nv_wr32(dev, 0x002230, 0x00000000);
267 nv_wr32(dev, NV40_PFIFO_RAMFC, 166 nv_wr32(dev, 0x002220, ((dev_priv->vram_size - 512 * 1024 +
268 ((dev_priv->vram_size - 512 * 1024 + 167 dev_priv->ramfc->pinst) >> 16) |
269 dev_priv->ramfc->pinst) >> 16) | (3 << 16)); 168 0x00030000);
270 break; 169 break;
271 } 170 }
272}
273
274static void
275nv40_fifo_init_intr(struct drm_device *dev)
276{
277 nouveau_irq_register(dev, 8, nv04_fifo_isr);
278 nv_wr32(dev, 0x002100, 0xffffffff);
279 nv_wr32(dev, 0x002140, 0xffffffff);
280}
281
282int
283nv40_fifo_init(struct drm_device *dev)
284{
285 struct drm_nouveau_private *dev_priv = dev->dev_private;
286 struct nouveau_fifo_engine *pfifo = &dev_priv->engine.fifo;
287 int i;
288 171
289 nv40_fifo_init_reset(dev); 172 nv_wr32(dev, NV03_PFIFO_CACHE1_PUSH1, priv->base.channels);
290 nv40_fifo_init_ramxx(dev);
291 173
292 nv40_fifo_do_load_context(dev, pfifo->channels - 1); 174 nv_wr32(dev, NV03_PFIFO_INTR_0, 0xffffffff);
293 nv_wr32(dev, NV03_PFIFO_CACHE1_PUSH1, pfifo->channels - 1); 175 nv_wr32(dev, NV03_PFIFO_INTR_EN_0, 0xffffffff);
294 176
295 nv40_fifo_init_intr(dev);
296 nv_wr32(dev, NV03_PFIFO_CACHE1_PUSH0, 1); 177 nv_wr32(dev, NV03_PFIFO_CACHE1_PUSH0, 1);
297 nv_wr32(dev, NV04_PFIFO_CACHE1_PULL0, 1); 178 nv_wr32(dev, NV04_PFIFO_CACHE1_PULL0, 1);
298 nv_wr32(dev, NV03_PFIFO_CACHES, 1); 179 nv_wr32(dev, NV03_PFIFO_CACHES, 1);
299 180
300 for (i = 0; i < dev_priv->engine.fifo.channels; i++) { 181 for (i = 0; i < priv->base.channels; i++) {
301 if (dev_priv->channels.ptr[i]) { 182 if (dev_priv->channels.ptr[i])
302 uint32_t mode = nv_rd32(dev, NV04_PFIFO_MODE); 183 nv_mask(dev, NV04_PFIFO_MODE, (1 << i), (1 << i));
303 nv_wr32(dev, NV04_PFIFO_MODE, mode | (1 << i));
304 }
305 } 184 }
306 185
307 return 0; 186 return 0;
308} 187}
188
189int
190nv40_fifo_create(struct drm_device *dev)
191{
192 struct drm_nouveau_private *dev_priv = dev->dev_private;
193 struct nv40_fifo_priv *priv;
194
195 priv = kzalloc(sizeof(*priv), GFP_KERNEL);
196 if (!priv)
197 return -ENOMEM;
198
199 priv->base.base.destroy = nv04_fifo_destroy;
200 priv->base.base.init = nv40_fifo_init;
201 priv->base.base.fini = nv04_fifo_fini;
202 priv->base.base.context_new = nv40_fifo_context_new;
203 priv->base.base.context_del = nv04_fifo_context_del;
204 priv->base.channels = 31;
205 priv->ramfc_desc = nv40_ramfc;
206 dev_priv->eng[NVOBJ_ENGINE_FIFO] = &priv->base.base;
207
208 nouveau_irq_register(dev, 8, nv04_fifo_isr);
209 return 0;
210}
diff --git a/drivers/gpu/drm/nouveau/nv40_graph.c b/drivers/gpu/drm/nouveau/nv40_graph.c
index 03837200762d..aa9e2df64a26 100644
--- a/drivers/gpu/drm/nouveau/nv40_graph.c
+++ b/drivers/gpu/drm/nouveau/nv40_graph.c
@@ -27,6 +27,7 @@
27#include "drmP.h" 27#include "drmP.h"
28#include "drm.h" 28#include "drm.h"
29#include "nouveau_drv.h" 29#include "nouveau_drv.h"
30#include "nouveau_fifo.h"
30#include "nouveau_ramht.h" 31#include "nouveau_ramht.h"
31 32
32struct nv40_graph_engine { 33struct nv40_graph_engine {
@@ -345,13 +346,14 @@ nv40_graph_fini(struct drm_device *dev, int engine, bool suspend)
345static int 346static int
346nv40_graph_isr_chid(struct drm_device *dev, u32 inst) 347nv40_graph_isr_chid(struct drm_device *dev, u32 inst)
347{ 348{
349 struct nouveau_fifo_priv *pfifo = nv_engine(dev, NVOBJ_ENGINE_FIFO);
348 struct drm_nouveau_private *dev_priv = dev->dev_private; 350 struct drm_nouveau_private *dev_priv = dev->dev_private;
349 struct nouveau_gpuobj *grctx; 351 struct nouveau_gpuobj *grctx;
350 unsigned long flags; 352 unsigned long flags;
351 int i; 353 int i;
352 354
353 spin_lock_irqsave(&dev_priv->channels.lock, flags); 355 spin_lock_irqsave(&dev_priv->channels.lock, flags);
354 for (i = 0; i < dev_priv->engine.fifo.channels; i++) { 356 for (i = 0; i < pfifo->channels; i++) {
355 if (!dev_priv->channels.ptr[i]) 357 if (!dev_priv->channels.ptr[i])
356 continue; 358 continue;
357 grctx = dev_priv->channels.ptr[i]->engctx[NVOBJ_ENGINE_GR]; 359 grctx = dev_priv->channels.ptr[i]->engctx[NVOBJ_ENGINE_GR];
diff --git a/drivers/gpu/drm/nouveau/nv40_pm.c b/drivers/gpu/drm/nouveau/nv40_pm.c
index c7615381c5d9..e66273aff493 100644
--- a/drivers/gpu/drm/nouveau/nv40_pm.c
+++ b/drivers/gpu/drm/nouveau/nv40_pm.c
@@ -27,6 +27,7 @@
27#include "nouveau_bios.h" 27#include "nouveau_bios.h"
28#include "nouveau_pm.h" 28#include "nouveau_pm.h"
29#include "nouveau_hw.h" 29#include "nouveau_hw.h"
30#include "nouveau_fifo.h"
30 31
31#define min2(a,b) ((a) < (b) ? (a) : (b)) 32#define min2(a,b) ((a) < (b) ? (a) : (b))
32 33
diff --git a/drivers/gpu/drm/nouveau/nv50_fb.c b/drivers/gpu/drm/nouveau/nv50_fb.c
index bdd2afe29205..f1e4b9e07d14 100644
--- a/drivers/gpu/drm/nouveau/nv50_fb.c
+++ b/drivers/gpu/drm/nouveau/nv50_fb.c
@@ -2,6 +2,7 @@
2#include "drm.h" 2#include "drm.h"
3#include "nouveau_drv.h" 3#include "nouveau_drv.h"
4#include "nouveau_drm.h" 4#include "nouveau_drm.h"
5#include "nouveau_fifo.h"
5 6
6struct nv50_fb_priv { 7struct nv50_fb_priv {
7 struct page *r100c08_page; 8 struct page *r100c08_page;
@@ -212,6 +213,7 @@ static struct nouveau_enum vm_fault[] = {
212void 213void
213nv50_fb_vm_trap(struct drm_device *dev, int display) 214nv50_fb_vm_trap(struct drm_device *dev, int display)
214{ 215{
216 struct nouveau_fifo_priv *pfifo = nv_engine(dev, NVOBJ_ENGINE_FIFO);
215 struct drm_nouveau_private *dev_priv = dev->dev_private; 217 struct drm_nouveau_private *dev_priv = dev->dev_private;
216 const struct nouveau_enum *en, *cl; 218 const struct nouveau_enum *en, *cl;
217 unsigned long flags; 219 unsigned long flags;
@@ -236,7 +238,7 @@ nv50_fb_vm_trap(struct drm_device *dev, int display)
236 /* lookup channel id */ 238 /* lookup channel id */
237 chinst = (trap[2] << 16) | trap[1]; 239 chinst = (trap[2] << 16) | trap[1];
238 spin_lock_irqsave(&dev_priv->channels.lock, flags); 240 spin_lock_irqsave(&dev_priv->channels.lock, flags);
239 for (ch = 0; ch < dev_priv->engine.fifo.channels; ch++) { 241 for (ch = 0; ch < pfifo->channels; ch++) {
240 struct nouveau_channel *chan = dev_priv->channels.ptr[ch]; 242 struct nouveau_channel *chan = dev_priv->channels.ptr[ch];
241 243
242 if (!chan || !chan->ramin) 244 if (!chan || !chan->ramin)
diff --git a/drivers/gpu/drm/nouveau/nv50_fifo.c b/drivers/gpu/drm/nouveau/nv50_fifo.c
index 12e9e8270f45..55383b85db0b 100644
--- a/drivers/gpu/drm/nouveau/nv50_fifo.c
+++ b/drivers/gpu/drm/nouveau/nv50_fifo.c
@@ -1,5 +1,5 @@
1/* 1/*
2 * Copyright (C) 2007 Ben Skeggs. 2 * Copyright (C) 2012 Ben Skeggs.
3 * All Rights Reserved. 3 * All Rights Reserved.
4 * 4 *
5 * Permission is hereby granted, free of charge, to any person obtaining 5 * Permission is hereby granted, free of charge, to any person obtaining
@@ -27,288 +27,135 @@
27#include "drmP.h" 27#include "drmP.h"
28#include "drm.h" 28#include "drm.h"
29#include "nouveau_drv.h" 29#include "nouveau_drv.h"
30#include "nouveau_fifo.h"
30#include "nouveau_ramht.h" 31#include "nouveau_ramht.h"
31#include "nouveau_vm.h" 32#include "nouveau_vm.h"
32 33
33static void 34struct nv50_fifo_priv {
35 struct nouveau_fifo_priv base;
36 struct nouveau_gpuobj *playlist[2];
37 int cur_playlist;
38};
39
40struct nv50_fifo_chan {
41 struct nouveau_fifo_chan base;
42};
43
44void
34nv50_fifo_playlist_update(struct drm_device *dev) 45nv50_fifo_playlist_update(struct drm_device *dev)
35{ 46{
47 struct nv50_fifo_priv *priv = nv_engine(dev, NVOBJ_ENGINE_FIFO);
36 struct drm_nouveau_private *dev_priv = dev->dev_private; 48 struct drm_nouveau_private *dev_priv = dev->dev_private;
37 struct nouveau_fifo_engine *pfifo = &dev_priv->engine.fifo;
38 struct nouveau_gpuobj *cur; 49 struct nouveau_gpuobj *cur;
39 int i, p; 50 int i, p;
40 51
41 NV_DEBUG(dev, "\n"); 52 cur = priv->playlist[priv->cur_playlist];
42 53 priv->cur_playlist = !priv->cur_playlist;
43 cur = pfifo->playlist[pfifo->cur_playlist];
44 pfifo->cur_playlist = !pfifo->cur_playlist;
45 54
46 for (i = 0, p = 0; i < pfifo->channels; i++) { 55 for (i = 0, p = 0; i < priv->base.channels; i++) {
47 if (nv_rd32(dev, 0x002600 + (i * 4)) & 0x80000000) 56 if (nv_rd32(dev, 0x002600 + (i * 4)) & 0x80000000)
48 nv_wo32(cur, p++ * 4, i); 57 nv_wo32(cur, p++ * 4, i);
49 } 58 }
50 59
51 dev_priv->engine.instmem.flush(dev); 60 dev_priv->engine.instmem.flush(dev);
52 61
53 nv_wr32(dev, 0x32f4, cur->vinst >> 12); 62 nv_wr32(dev, 0x0032f4, cur->vinst >> 12);
54 nv_wr32(dev, 0x32ec, p); 63 nv_wr32(dev, 0x0032ec, p);
55 nv_wr32(dev, 0x2500, 0x101); 64 nv_wr32(dev, 0x002500, 0x00000101);
56}
57
58static void
59nv50_fifo_channel_enable(struct drm_device *dev, int channel)
60{
61 struct drm_nouveau_private *dev_priv = dev->dev_private;
62 struct nouveau_channel *chan = dev_priv->channels.ptr[channel];
63 uint32_t inst;
64
65 NV_DEBUG(dev, "ch%d\n", channel);
66
67 if (dev_priv->chipset == 0x50)
68 inst = chan->ramfc->vinst >> 12;
69 else
70 inst = chan->ramfc->vinst >> 8;
71
72 nv_wr32(dev, NV50_PFIFO_CTX_TABLE(channel), inst |
73 NV50_PFIFO_CTX_TABLE_CHANNEL_ENABLED);
74}
75
76static void
77nv50_fifo_channel_disable(struct drm_device *dev, int channel)
78{
79 struct drm_nouveau_private *dev_priv = dev->dev_private;
80 uint32_t inst;
81
82 NV_DEBUG(dev, "ch%d\n", channel);
83
84 if (dev_priv->chipset == 0x50)
85 inst = NV50_PFIFO_CTX_TABLE_INSTANCE_MASK_G80;
86 else
87 inst = NV50_PFIFO_CTX_TABLE_INSTANCE_MASK_G84;
88 nv_wr32(dev, NV50_PFIFO_CTX_TABLE(channel), inst);
89}
90
91static void
92nv50_fifo_init_reset(struct drm_device *dev)
93{
94 uint32_t pmc_e = NV_PMC_ENABLE_PFIFO;
95
96 NV_DEBUG(dev, "\n");
97
98 nv_wr32(dev, NV03_PMC_ENABLE, nv_rd32(dev, NV03_PMC_ENABLE) & ~pmc_e);
99 nv_wr32(dev, NV03_PMC_ENABLE, nv_rd32(dev, NV03_PMC_ENABLE) | pmc_e);
100}
101
102static void
103nv50_fifo_init_intr(struct drm_device *dev)
104{
105 NV_DEBUG(dev, "\n");
106
107 nouveau_irq_register(dev, 8, nv04_fifo_isr);
108 nv_wr32(dev, NV03_PFIFO_INTR_0, 0xFFFFFFFF);
109 nv_wr32(dev, NV03_PFIFO_INTR_EN_0, 0xFFFFFFFF);
110}
111
112static void
113nv50_fifo_init_context_table(struct drm_device *dev)
114{
115 struct drm_nouveau_private *dev_priv = dev->dev_private;
116 int i;
117
118 NV_DEBUG(dev, "\n");
119
120 for (i = 0; i < NV50_PFIFO_CTX_TABLE__SIZE; i++) {
121 if (dev_priv->channels.ptr[i])
122 nv50_fifo_channel_enable(dev, i);
123 else
124 nv50_fifo_channel_disable(dev, i);
125 }
126
127 nv50_fifo_playlist_update(dev);
128}
129
130static void
131nv50_fifo_init_regs__nv(struct drm_device *dev)
132{
133 NV_DEBUG(dev, "\n");
134
135 nv_wr32(dev, 0x250c, 0x6f3cfc34);
136}
137
138static void
139nv50_fifo_init_regs(struct drm_device *dev)
140{
141 NV_DEBUG(dev, "\n");
142
143 nv_wr32(dev, 0x2500, 0);
144 nv_wr32(dev, 0x3250, 0);
145 nv_wr32(dev, 0x3220, 0);
146 nv_wr32(dev, 0x3204, 0);
147 nv_wr32(dev, 0x3210, 0);
148 nv_wr32(dev, 0x3270, 0);
149 nv_wr32(dev, 0x2044, 0x01003fff);
150
151 /* Enable dummy channels setup by nv50_instmem.c */
152 nv50_fifo_channel_enable(dev, 0);
153 nv50_fifo_channel_enable(dev, 127);
154}
155
156int
157nv50_fifo_init(struct drm_device *dev)
158{
159 struct drm_nouveau_private *dev_priv = dev->dev_private;
160 struct nouveau_fifo_engine *pfifo = &dev_priv->engine.fifo;
161 int ret;
162
163 NV_DEBUG(dev, "\n");
164
165 if (pfifo->playlist[0]) {
166 pfifo->cur_playlist = !pfifo->cur_playlist;
167 goto just_reset;
168 }
169
170 ret = nouveau_gpuobj_new(dev, NULL, 128*4, 0x1000,
171 NVOBJ_FLAG_ZERO_ALLOC,
172 &pfifo->playlist[0]);
173 if (ret) {
174 NV_ERROR(dev, "error creating playlist 0: %d\n", ret);
175 return ret;
176 }
177
178 ret = nouveau_gpuobj_new(dev, NULL, 128*4, 0x1000,
179 NVOBJ_FLAG_ZERO_ALLOC,
180 &pfifo->playlist[1]);
181 if (ret) {
182 nouveau_gpuobj_ref(NULL, &pfifo->playlist[0]);
183 NV_ERROR(dev, "error creating playlist 1: %d\n", ret);
184 return ret;
185 }
186
187just_reset:
188 nv50_fifo_init_reset(dev);
189 nv50_fifo_init_intr(dev);
190 nv50_fifo_init_context_table(dev);
191 nv50_fifo_init_regs__nv(dev);
192 nv50_fifo_init_regs(dev);
193 nv_wr32(dev, NV03_PFIFO_CACHE1_PUSH0, 1);
194 nv_wr32(dev, NV04_PFIFO_CACHE1_PULL0, 1);
195 nv_wr32(dev, NV03_PFIFO_CACHES, 1);
196
197 return 0;
198} 65}
199 66
200void 67static int
201nv50_fifo_takedown(struct drm_device *dev) 68nv50_fifo_context_new(struct nouveau_channel *chan, int engine)
202{
203 struct drm_nouveau_private *dev_priv = dev->dev_private;
204 struct nouveau_fifo_engine *pfifo = &dev_priv->engine.fifo;
205
206 NV_DEBUG(dev, "\n");
207
208 if (!pfifo->playlist[0])
209 return;
210
211 nv_wr32(dev, 0x2140, 0x00000000);
212 nouveau_irq_unregister(dev, 8);
213
214 nouveau_gpuobj_ref(NULL, &pfifo->playlist[0]);
215 nouveau_gpuobj_ref(NULL, &pfifo->playlist[1]);
216}
217
218int
219nv50_fifo_create_context(struct nouveau_channel *chan)
220{ 69{
70 struct nv50_fifo_priv *priv = nv_engine(chan->dev, engine);
71 struct nv50_fifo_chan *fctx;
221 struct drm_device *dev = chan->dev; 72 struct drm_device *dev = chan->dev;
222 struct drm_nouveau_private *dev_priv = dev->dev_private; 73 struct drm_nouveau_private *dev_priv = dev->dev_private;
223 struct nouveau_gpuobj *ramfc = NULL; 74 u64 ib_offset = chan->pushbuf_base + chan->dma.ib_base * 4;
224 uint64_t ib_offset = chan->pushbuf_base + chan->dma.ib_base * 4; 75 u64 instance = chan->ramin->vinst >> 12;
225 unsigned long flags; 76 unsigned long flags;
226 int ret; 77 int ret = 0, i;
227 78
228 NV_DEBUG(dev, "ch%d\n", chan->id); 79 fctx = chan->engctx[engine] = kzalloc(sizeof(*fctx), GFP_KERNEL);
229 80 if (!fctx)
230 if (dev_priv->chipset == 0x50) { 81 return -ENOMEM;
231 ret = nouveau_gpuobj_new_fake(dev, chan->ramin->pinst, 82 atomic_inc(&chan->vm->engref[engine]);
232 chan->ramin->vinst, 0x100,
233 NVOBJ_FLAG_ZERO_ALLOC |
234 NVOBJ_FLAG_ZERO_FREE,
235 &chan->ramfc);
236 if (ret)
237 return ret;
238
239 ret = nouveau_gpuobj_new_fake(dev, chan->ramin->pinst + 0x0400,
240 chan->ramin->vinst + 0x0400,
241 4096, 0, &chan->cache);
242 if (ret)
243 return ret;
244 } else {
245 ret = nouveau_gpuobj_new(dev, chan, 0x100, 256,
246 NVOBJ_FLAG_ZERO_ALLOC |
247 NVOBJ_FLAG_ZERO_FREE, &chan->ramfc);
248 if (ret)
249 return ret;
250
251 ret = nouveau_gpuobj_new(dev, chan, 4096, 1024,
252 0, &chan->cache);
253 if (ret)
254 return ret;
255 }
256 ramfc = chan->ramfc;
257 83
258 chan->user = ioremap(pci_resource_start(dev->pdev, 0) + 84 chan->user = ioremap(pci_resource_start(dev->pdev, 0) +
259 NV50_USER(chan->id), PAGE_SIZE); 85 NV50_USER(chan->id), PAGE_SIZE);
260 if (!chan->user) 86 if (!chan->user) {
261 return -ENOMEM; 87 ret = -ENOMEM;
262 88 goto error;
263 spin_lock_irqsave(&dev_priv->context_switch_lock, flags);
264
265 nv_wo32(ramfc, 0x48, chan->pushbuf->cinst >> 4);
266 nv_wo32(ramfc, 0x80, ((chan->ramht->bits - 9) << 27) |
267 (4 << 24) /* SEARCH_FULL */ |
268 (chan->ramht->gpuobj->cinst >> 4));
269 nv_wo32(ramfc, 0x44, 0x01003fff);
270 nv_wo32(ramfc, 0x60, 0x7fffffff);
271 nv_wo32(ramfc, 0x40, 0x00000000);
272 nv_wo32(ramfc, 0x7c, 0x30000001);
273 nv_wo32(ramfc, 0x78, 0x00000000);
274 nv_wo32(ramfc, 0x3c, 0x403f6078);
275 nv_wo32(ramfc, 0x50, lower_32_bits(ib_offset));
276 nv_wo32(ramfc, 0x54, upper_32_bits(ib_offset) |
277 drm_order(chan->dma.ib_max + 1) << 16);
278
279 if (dev_priv->chipset != 0x50) {
280 nv_wo32(chan->ramin, 0, chan->id);
281 nv_wo32(chan->ramin, 4, chan->ramfc->vinst >> 8);
282
283 nv_wo32(ramfc, 0x88, chan->cache->vinst >> 10);
284 nv_wo32(ramfc, 0x98, chan->ramin->vinst >> 12);
285 } 89 }
286 90
91 for (i = 0; i < 0x100; i += 4)
92 nv_wo32(chan->ramin, i, 0x00000000);
93 nv_wo32(chan->ramin, 0x3c, 0x403f6078);
94 nv_wo32(chan->ramin, 0x40, 0x00000000);
95 nv_wo32(chan->ramin, 0x44, 0x01003fff);
96 nv_wo32(chan->ramin, 0x48, chan->pushbuf->cinst >> 4);
97 nv_wo32(chan->ramin, 0x50, lower_32_bits(ib_offset));
98 nv_wo32(chan->ramin, 0x54, upper_32_bits(ib_offset) |
99 drm_order(chan->dma.ib_max + 1) << 16);
100 nv_wo32(chan->ramin, 0x60, 0x7fffffff);
101 nv_wo32(chan->ramin, 0x78, 0x00000000);
102 nv_wo32(chan->ramin, 0x7c, 0x30000001);
103 nv_wo32(chan->ramin, 0x80, ((chan->ramht->bits - 9) << 27) |
104 (4 << 24) /* SEARCH_FULL */ |
105 (chan->ramht->gpuobj->cinst >> 4));
106
287 dev_priv->engine.instmem.flush(dev); 107 dev_priv->engine.instmem.flush(dev);
288 108
289 nv50_fifo_channel_enable(dev, chan->id); 109 spin_lock_irqsave(&dev_priv->context_switch_lock, flags);
110 nv_wr32(dev, 0x002600 + (chan->id * 4), 0x80000000 | instance);
290 nv50_fifo_playlist_update(dev); 111 nv50_fifo_playlist_update(dev);
291 spin_unlock_irqrestore(&dev_priv->context_switch_lock, flags); 112 spin_unlock_irqrestore(&dev_priv->context_switch_lock, flags);
292 return 0; 113
114error:
115 if (ret)
116 priv->base.base.context_del(chan, engine);
117 return ret;
293} 118}
294 119
295static bool 120static bool
296nv50_fifo_wait_kickoff(void *data) 121nv50_fifo_kickoff(struct nouveau_channel *chan)
297{ 122{
298 struct drm_nouveau_private *dev_priv = data; 123 struct drm_device *dev = chan->dev;
299 struct drm_device *dev = dev_priv->dev; 124 bool done = true;
300 125 u32 me;
301 if (dev_priv->chipset == 0x50) { 126
302 u32 me_enable = nv_mask(dev, 0x00b860, 0x00000001, 0x00000001); 127 /* HW bug workaround:
303 nv_wr32(dev, 0x00b860, me_enable); 128 *
129 * PFIFO will hang forever if the connected engines don't report
130 * that they've processed the context switch request.
131 *
132 * In order for the kickoff to work, we need to ensure all the
133 * connected engines are in a state where they can answer.
134 *
135 * Newer chipsets don't seem to suffer from this issue, and well,
136 * there's also a "ignore these engines" bitmask reg we can use
137 * if we hit the issue there..
138 */
139
140 /* PME: make sure engine is enabled */
141 me = nv_mask(dev, 0x00b860, 0x00000001, 0x00000001);
142
143 /* do the kickoff... */
144 nv_wr32(dev, 0x0032fc, chan->ramin->vinst >> 12);
145 if (!nv_wait_ne(dev, 0x0032fc, 0xffffffff, 0xffffffff)) {
146 NV_INFO(dev, "PFIFO: channel %d unload timeout\n", chan->id);
147 done = false;
304 } 148 }
305 149
306 return nv_rd32(dev, 0x0032fc) != 0xffffffff; 150 /* restore any engine states we changed, and exit */
151 nv_wr32(dev, 0x00b860, me);
152 return done;
307} 153}
308 154
309void 155static void
310nv50_fifo_destroy_context(struct nouveau_channel *chan) 156nv50_fifo_context_del(struct nouveau_channel *chan, int engine)
311{ 157{
158 struct nv50_fifo_chan *fctx = chan->engctx[engine];
312 struct drm_device *dev = chan->dev; 159 struct drm_device *dev = chan->dev;
313 struct drm_nouveau_private *dev_priv = dev->dev_private; 160 struct drm_nouveau_private *dev_priv = dev->dev_private;
314 unsigned long flags; 161 unsigned long flags;
@@ -319,9 +166,7 @@ nv50_fifo_destroy_context(struct nouveau_channel *chan)
319 nv50_fifo_playlist_update(dev); 166 nv50_fifo_playlist_update(dev);
320 167
321 /* tell any engines on this channel to unload their contexts */ 168 /* tell any engines on this channel to unload their contexts */
322 nv_wr32(dev, 0x0032fc, chan->ramin->vinst >> 12); 169 nv50_fifo_kickoff(chan);
323 if (!nv_wait_cb(dev, nv50_fifo_wait_kickoff, dev_priv))
324 NV_INFO(dev, "PFIFO: channel %d unload timeout\n", chan->id);
325 170
326 nv_wr32(dev, 0x002600 + (chan->id * 4), 0x00000000); 171 nv_wr32(dev, 0x002600 + (chan->id * 4), 0x00000000);
327 spin_unlock_irqrestore(&dev_priv->context_switch_lock, flags); 172 spin_unlock_irqrestore(&dev_priv->context_switch_lock, flags);
@@ -332,41 +177,118 @@ nv50_fifo_destroy_context(struct nouveau_channel *chan)
332 chan->user = NULL; 177 chan->user = NULL;
333 } 178 }
334 179
335 nouveau_gpuobj_ref(NULL, &chan->ramfc); 180 atomic_dec(&chan->vm->engref[engine]);
336 nouveau_gpuobj_ref(NULL, &chan->cache); 181 chan->engctx[engine] = NULL;
182 kfree(fctx);
337} 183}
338 184
339int 185static int
340nv50_fifo_load_context(struct nouveau_channel *chan) 186nv50_fifo_init(struct drm_device *dev, int engine)
341{ 187{
188 struct drm_nouveau_private *dev_priv = dev->dev_private;
189 u32 instance;
190 int i;
191
192 nv_mask(dev, 0x000200, 0x00000100, 0x00000000);
193 nv_mask(dev, 0x000200, 0x00000100, 0x00000100);
194 nv_wr32(dev, 0x00250c, 0x6f3cfc34);
195 nv_wr32(dev, 0x002044, 0x01003fff);
196
197 nv_wr32(dev, 0x002100, 0xffffffff);
198 nv_wr32(dev, 0x002140, 0xffffffff);
199
200 for (i = 0; i < 128; i++) {
201 struct nouveau_channel *chan = dev_priv->channels.ptr[i];
202 if (chan && chan->engctx[engine])
203 instance = 0x80000000 | chan->ramin->vinst >> 12;
204 else
205 instance = 0x00000000;
206 nv_wr32(dev, 0x002600 + (i * 4), instance);
207 }
208
209 nv50_fifo_playlist_update(dev);
210
211 nv_wr32(dev, 0x003200, 1);
212 nv_wr32(dev, 0x003250, 1);
213 nv_wr32(dev, 0x002500, 1);
342 return 0; 214 return 0;
343} 215}
344 216
345int 217static int
346nv50_fifo_unload_context(struct drm_device *dev) 218nv50_fifo_fini(struct drm_device *dev, int engine, bool suspend)
347{ 219{
348 struct drm_nouveau_private *dev_priv = dev->dev_private; 220 struct drm_nouveau_private *dev_priv = dev->dev_private;
221 struct nv50_fifo_priv *priv = nv_engine(dev, engine);
349 int i; 222 int i;
350 223
351 /* set playlist length to zero, fifo will unload context */ 224 /* set playlist length to zero, fifo will unload context */
352 nv_wr32(dev, 0x0032ec, 0); 225 nv_wr32(dev, 0x0032ec, 0);
353 226
354 /* tell all connected engines to unload their contexts */ 227 /* tell all connected engines to unload their contexts */
355 for (i = 0; i < dev_priv->engine.fifo.channels; i++) { 228 for (i = 0; i < priv->base.channels; i++) {
356 struct nouveau_channel *chan = dev_priv->channels.ptr[i]; 229 struct nouveau_channel *chan = dev_priv->channels.ptr[i];
357 if (chan) 230 if (chan && !nv50_fifo_kickoff(chan))
358 nv_wr32(dev, 0x0032fc, chan->ramin->vinst >> 12);
359 if (!nv_wait_cb(dev, nv50_fifo_wait_kickoff, dev_priv)) {
360 NV_INFO(dev, "PFIFO: channel %d unload timeout\n", i);
361 return -EBUSY; 231 return -EBUSY;
362 }
363 } 232 }
364 233
234 nv_wr32(dev, 0x002140, 0);
365 return 0; 235 return 0;
366} 236}
367 237
368void 238void
369nv50_fifo_tlb_flush(struct drm_device *dev) 239nv50_fifo_tlb_flush(struct drm_device *dev, int engine)
370{ 240{
371 nv50_vm_flush_engine(dev, 5); 241 nv50_vm_flush_engine(dev, 5);
372} 242}
243
244void
245nv50_fifo_destroy(struct drm_device *dev, int engine)
246{
247 struct drm_nouveau_private *dev_priv = dev->dev_private;
248 struct nv50_fifo_priv *priv = nv_engine(dev, engine);
249
250 nouveau_irq_unregister(dev, 8);
251
252 nouveau_gpuobj_ref(NULL, &priv->playlist[0]);
253 nouveau_gpuobj_ref(NULL, &priv->playlist[1]);
254
255 dev_priv->eng[engine] = NULL;
256 kfree(priv);
257}
258
259int
260nv50_fifo_create(struct drm_device *dev)
261{
262 struct drm_nouveau_private *dev_priv = dev->dev_private;
263 struct nv50_fifo_priv *priv;
264 int ret;
265
266 priv = kzalloc(sizeof(*priv), GFP_KERNEL);
267 if (!priv)
268 return -ENOMEM;
269
270 priv->base.base.destroy = nv50_fifo_destroy;
271 priv->base.base.init = nv50_fifo_init;
272 priv->base.base.fini = nv50_fifo_fini;
273 priv->base.base.context_new = nv50_fifo_context_new;
274 priv->base.base.context_del = nv50_fifo_context_del;
275 priv->base.base.tlb_flush = nv50_fifo_tlb_flush;
276 priv->base.channels = 127;
277 dev_priv->eng[NVOBJ_ENGINE_FIFO] = &priv->base.base;
278
279 ret = nouveau_gpuobj_new(dev, NULL, priv->base.channels * 4, 0x1000,
280 NVOBJ_FLAG_ZERO_ALLOC, &priv->playlist[0]);
281 if (ret)
282 goto error;
283
284 ret = nouveau_gpuobj_new(dev, NULL, priv->base.channels * 4, 0x1000,
285 NVOBJ_FLAG_ZERO_ALLOC, &priv->playlist[1]);
286 if (ret)
287 goto error;
288
289 nouveau_irq_register(dev, 8, nv04_fifo_isr);
290error:
291 if (ret)
292 priv->base.base.destroy(dev, NVOBJ_ENGINE_FIFO);
293 return ret;
294}
diff --git a/drivers/gpu/drm/nouveau/nv50_graph.c b/drivers/gpu/drm/nouveau/nv50_graph.c
index 46b2c95de4d0..636e22b1530c 100644
--- a/drivers/gpu/drm/nouveau/nv50_graph.c
+++ b/drivers/gpu/drm/nouveau/nv50_graph.c
@@ -27,6 +27,7 @@
27#include "drmP.h" 27#include "drmP.h"
28#include "drm.h" 28#include "drm.h"
29#include "nouveau_drv.h" 29#include "nouveau_drv.h"
30#include "nouveau_fifo.h"
30#include "nouveau_ramht.h" 31#include "nouveau_ramht.h"
31#include "nouveau_dma.h" 32#include "nouveau_dma.h"
32#include "nouveau_vm.h" 33#include "nouveau_vm.h"
@@ -710,13 +711,14 @@ nv50_pgraph_trap_handler(struct drm_device *dev, u32 display, u64 inst, u32 chid
710int 711int
711nv50_graph_isr_chid(struct drm_device *dev, u64 inst) 712nv50_graph_isr_chid(struct drm_device *dev, u64 inst)
712{ 713{
714 struct nouveau_fifo_priv *pfifo = nv_engine(dev, NVOBJ_ENGINE_FIFO);
713 struct drm_nouveau_private *dev_priv = dev->dev_private; 715 struct drm_nouveau_private *dev_priv = dev->dev_private;
714 struct nouveau_channel *chan; 716 struct nouveau_channel *chan;
715 unsigned long flags; 717 unsigned long flags;
716 int i; 718 int i;
717 719
718 spin_lock_irqsave(&dev_priv->channels.lock, flags); 720 spin_lock_irqsave(&dev_priv->channels.lock, flags);
719 for (i = 0; i < dev_priv->engine.fifo.channels; i++) { 721 for (i = 0; i < pfifo->channels; i++) {
720 chan = dev_priv->channels.ptr[i]; 722 chan = dev_priv->channels.ptr[i];
721 if (!chan || !chan->ramin) 723 if (!chan || !chan->ramin)
722 continue; 724 continue;
diff --git a/drivers/gpu/drm/nouveau/nv50_vm.c b/drivers/gpu/drm/nouveau/nv50_vm.c
index 44fbac9c7d93..179bb42a635c 100644
--- a/drivers/gpu/drm/nouveau/nv50_vm.c
+++ b/drivers/gpu/drm/nouveau/nv50_vm.c
@@ -147,7 +147,6 @@ nv50_vm_flush(struct nouveau_vm *vm)
147{ 147{
148 struct drm_nouveau_private *dev_priv = vm->dev->dev_private; 148 struct drm_nouveau_private *dev_priv = vm->dev->dev_private;
149 struct nouveau_instmem_engine *pinstmem = &dev_priv->engine.instmem; 149 struct nouveau_instmem_engine *pinstmem = &dev_priv->engine.instmem;
150 struct nouveau_fifo_engine *pfifo = &dev_priv->engine.fifo;
151 int i; 150 int i;
152 151
153 pinstmem->flush(vm->dev); 152 pinstmem->flush(vm->dev);
@@ -158,7 +157,6 @@ nv50_vm_flush(struct nouveau_vm *vm)
158 return; 157 return;
159 } 158 }
160 159
161 pfifo->tlb_flush(vm->dev);
162 for (i = 0; i < NVOBJ_ENGINE_NR; i++) { 160 for (i = 0; i < NVOBJ_ENGINE_NR; i++) {
163 if (atomic_read(&vm->engref[i])) 161 if (atomic_read(&vm->engref[i]))
164 dev_priv->eng[i]->tlb_flush(vm->dev, i); 162 dev_priv->eng[i]->tlb_flush(vm->dev, i);
diff --git a/drivers/gpu/drm/nouveau/nv84_fence.c b/drivers/gpu/drm/nouveau/nv84_fence.c
index 0ac98c0efc71..c2f889b0d340 100644
--- a/drivers/gpu/drm/nouveau/nv84_fence.c
+++ b/drivers/gpu/drm/nouveau/nv84_fence.c
@@ -25,6 +25,7 @@
25#include "drmP.h" 25#include "drmP.h"
26#include "nouveau_drv.h" 26#include "nouveau_drv.h"
27#include "nouveau_dma.h" 27#include "nouveau_dma.h"
28#include "nouveau_fifo.h"
28#include "nouveau_ramht.h" 29#include "nouveau_ramht.h"
29#include "nouveau_fence.h" 30#include "nouveau_fence.h"
30 31
@@ -145,8 +146,8 @@ nv84_fence_destroy(struct drm_device *dev, int engine)
145int 146int
146nv84_fence_create(struct drm_device *dev) 147nv84_fence_create(struct drm_device *dev)
147{ 148{
149 struct nouveau_fifo_priv *pfifo = nv_engine(dev, NVOBJ_ENGINE_FIFO);
148 struct drm_nouveau_private *dev_priv = dev->dev_private; 150 struct drm_nouveau_private *dev_priv = dev->dev_private;
149 struct nouveau_fifo_engine *pfifo = &dev_priv->engine.fifo;
150 struct nv84_fence_priv *priv; 151 struct nv84_fence_priv *priv;
151 int ret; 152 int ret;
152 153
diff --git a/drivers/gpu/drm/nouveau/nv84_fifo.c b/drivers/gpu/drm/nouveau/nv84_fifo.c
new file mode 100644
index 000000000000..cc82d799fc3b
--- /dev/null
+++ b/drivers/gpu/drm/nouveau/nv84_fifo.c
@@ -0,0 +1,241 @@
1/*
2 * Copyright (C) 2012 Ben Skeggs.
3 * All Rights Reserved.
4 *
5 * Permission is hereby granted, free of charge, to any person obtaining
6 * a copy of this software and associated documentation files (the
7 * "Software"), to deal in the Software without restriction, including
8 * without limitation the rights to use, copy, modify, merge, publish,
9 * distribute, sublicense, and/or sell copies of the Software, and to
10 * permit persons to whom the Software is furnished to do so, subject to
11 * the following conditions:
12 *
13 * The above copyright notice and this permission notice (including the
14 * next paragraph) shall be included in all copies or substantial
15 * portions of the Software.
16 *
17 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
18 * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
19 * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
20 * IN NO EVENT SHALL THE COPYRIGHT OWNER(S) AND/OR ITS SUPPLIERS BE
21 * LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
22 * OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
23 * WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
24 *
25 */
26
27#include "drmP.h"
28#include "drm.h"
29#include "nouveau_drv.h"
30#include "nouveau_fifo.h"
31#include "nouveau_ramht.h"
32#include "nouveau_vm.h"
33
34struct nv84_fifo_priv {
35 struct nouveau_fifo_priv base;
36 struct nouveau_gpuobj *playlist[2];
37 int cur_playlist;
38};
39
40struct nv84_fifo_chan {
41 struct nouveau_fifo_chan base;
42 struct nouveau_gpuobj *ramfc;
43 struct nouveau_gpuobj *cache;
44};
45
46static int
47nv84_fifo_context_new(struct nouveau_channel *chan, int engine)
48{
49 struct nv84_fifo_priv *priv = nv_engine(chan->dev, engine);
50 struct nv84_fifo_chan *fctx;
51 struct drm_device *dev = chan->dev;
52 struct drm_nouveau_private *dev_priv = dev->dev_private;
53 u64 ib_offset = chan->pushbuf_base + chan->dma.ib_base * 4;
54 u64 instance;
55 unsigned long flags;
56 int ret;
57
58 fctx = chan->engctx[engine] = kzalloc(sizeof(*fctx), GFP_KERNEL);
59 if (!fctx)
60 return -ENOMEM;
61 atomic_inc(&chan->vm->engref[engine]);
62
63 chan->user = ioremap(pci_resource_start(dev->pdev, 0) +
64 NV50_USER(chan->id), PAGE_SIZE);
65 if (!chan->user) {
66 ret = -ENOMEM;
67 goto error;
68 }
69
70 ret = nouveau_gpuobj_new(dev, chan, 256, 256, NVOBJ_FLAG_ZERO_ALLOC |
71 NVOBJ_FLAG_ZERO_FREE, &fctx->ramfc);
72 if (ret)
73 goto error;
74
75 instance = fctx->ramfc->vinst >> 8;
76
77 ret = nouveau_gpuobj_new(dev, chan, 4096, 1024, 0, &fctx->cache);
78 if (ret)
79 goto error;
80
81 nv_wo32(fctx->ramfc, 0x3c, 0x403f6078);
82 nv_wo32(fctx->ramfc, 0x40, 0x00000000);
83 nv_wo32(fctx->ramfc, 0x44, 0x01003fff);
84 nv_wo32(fctx->ramfc, 0x48, chan->pushbuf->cinst >> 4);
85 nv_wo32(fctx->ramfc, 0x50, lower_32_bits(ib_offset));
86 nv_wo32(fctx->ramfc, 0x54, upper_32_bits(ib_offset) |
87 drm_order(chan->dma.ib_max + 1) << 16);
88 nv_wo32(fctx->ramfc, 0x60, 0x7fffffff);
89 nv_wo32(fctx->ramfc, 0x78, 0x00000000);
90 nv_wo32(fctx->ramfc, 0x7c, 0x30000001);
91 nv_wo32(fctx->ramfc, 0x80, ((chan->ramht->bits - 9) << 27) |
92 (4 << 24) /* SEARCH_FULL */ |
93 (chan->ramht->gpuobj->cinst >> 4));
94 nv_wo32(fctx->ramfc, 0x88, fctx->cache->vinst >> 10);
95 nv_wo32(fctx->ramfc, 0x98, chan->ramin->vinst >> 12);
96
97 nv_wo32(chan->ramin, 0x00, chan->id);
98 nv_wo32(chan->ramin, 0x04, fctx->ramfc->vinst >> 8);
99
100 dev_priv->engine.instmem.flush(dev);
101
102 spin_lock_irqsave(&dev_priv->context_switch_lock, flags);
103 nv_wr32(dev, 0x002600 + (chan->id * 4), 0x80000000 | instance);
104 nv50_fifo_playlist_update(dev);
105 spin_unlock_irqrestore(&dev_priv->context_switch_lock, flags);
106
107error:
108 if (ret)
109 priv->base.base.context_del(chan, engine);
110 return ret;
111}
112
113static void
114nv84_fifo_context_del(struct nouveau_channel *chan, int engine)
115{
116 struct nv84_fifo_chan *fctx = chan->engctx[engine];
117 struct drm_device *dev = chan->dev;
118 struct drm_nouveau_private *dev_priv = dev->dev_private;
119 unsigned long flags;
120
121 /* remove channel from playlist, will context switch if active */
122 spin_lock_irqsave(&dev_priv->context_switch_lock, flags);
123 nv_mask(dev, 0x002600 + (chan->id * 4), 0x80000000, 0x00000000);
124 nv50_fifo_playlist_update(dev);
125
126 /* tell any engines on this channel to unload their contexts */
127 nv_wr32(dev, 0x0032fc, chan->ramin->vinst >> 12);
128 if (!nv_wait_ne(dev, 0x0032fc, 0xffffffff, 0xffffffff))
129 NV_INFO(dev, "PFIFO: channel %d unload timeout\n", chan->id);
130
131 nv_wr32(dev, 0x002600 + (chan->id * 4), 0x00000000);
132 spin_unlock_irqrestore(&dev_priv->context_switch_lock, flags);
133
134 /* clean up */
135 if (chan->user) {
136 iounmap(chan->user);
137 chan->user = NULL;
138 }
139
140 nouveau_gpuobj_ref(NULL, &fctx->ramfc);
141 nouveau_gpuobj_ref(NULL, &fctx->cache);
142
143 atomic_dec(&chan->vm->engref[engine]);
144 chan->engctx[engine] = NULL;
145 kfree(fctx);
146}
147
148static int
149nv84_fifo_init(struct drm_device *dev, int engine)
150{
151 struct drm_nouveau_private *dev_priv = dev->dev_private;
152 struct nv84_fifo_chan *fctx;
153 u32 instance;
154 int i;
155
156 nv_mask(dev, 0x000200, 0x00000100, 0x00000000);
157 nv_mask(dev, 0x000200, 0x00000100, 0x00000100);
158 nv_wr32(dev, 0x00250c, 0x6f3cfc34);
159 nv_wr32(dev, 0x002044, 0x01003fff);
160
161 nv_wr32(dev, 0x002100, 0xffffffff);
162 nv_wr32(dev, 0x002140, 0xffffffff);
163
164 for (i = 0; i < 128; i++) {
165 struct nouveau_channel *chan = dev_priv->channels.ptr[i];
166 if (chan && (fctx = chan->engctx[engine]))
167 instance = 0x80000000 | fctx->ramfc->vinst >> 8;
168 else
169 instance = 0x00000000;
170 nv_wr32(dev, 0x002600 + (i * 4), instance);
171 }
172
173 nv50_fifo_playlist_update(dev);
174
175 nv_wr32(dev, 0x003200, 1);
176 nv_wr32(dev, 0x003250, 1);
177 nv_wr32(dev, 0x002500, 1);
178 return 0;
179}
180
181static int
182nv84_fifo_fini(struct drm_device *dev, int engine, bool suspend)
183{
184 struct drm_nouveau_private *dev_priv = dev->dev_private;
185 struct nv84_fifo_priv *priv = nv_engine(dev, engine);
186 int i;
187
188 /* set playlist length to zero, fifo will unload context */
189 nv_wr32(dev, 0x0032ec, 0);
190
191 /* tell all connected engines to unload their contexts */
192 for (i = 0; i < priv->base.channels; i++) {
193 struct nouveau_channel *chan = dev_priv->channels.ptr[i];
194 if (chan)
195 nv_wr32(dev, 0x0032fc, chan->ramin->vinst >> 12);
196 if (!nv_wait_ne(dev, 0x0032fc, 0xffffffff, 0xffffffff)) {
197 NV_INFO(dev, "PFIFO: channel %d unload timeout\n", i);
198 return -EBUSY;
199 }
200 }
201
202 nv_wr32(dev, 0x002140, 0);
203 return 0;
204}
205
206int
207nv84_fifo_create(struct drm_device *dev)
208{
209 struct drm_nouveau_private *dev_priv = dev->dev_private;
210 struct nv84_fifo_priv *priv;
211 int ret;
212
213 priv = kzalloc(sizeof(*priv), GFP_KERNEL);
214 if (!priv)
215 return -ENOMEM;
216
217 priv->base.base.destroy = nv50_fifo_destroy;
218 priv->base.base.init = nv84_fifo_init;
219 priv->base.base.fini = nv84_fifo_fini;
220 priv->base.base.context_new = nv84_fifo_context_new;
221 priv->base.base.context_del = nv84_fifo_context_del;
222 priv->base.base.tlb_flush = nv50_fifo_tlb_flush;
223 priv->base.channels = 127;
224 dev_priv->eng[NVOBJ_ENGINE_FIFO] = &priv->base.base;
225
226 ret = nouveau_gpuobj_new(dev, NULL, priv->base.channels * 4, 0x1000,
227 NVOBJ_FLAG_ZERO_ALLOC, &priv->playlist[0]);
228 if (ret)
229 goto error;
230
231 ret = nouveau_gpuobj_new(dev, NULL, priv->base.channels * 4, 0x1000,
232 NVOBJ_FLAG_ZERO_ALLOC, &priv->playlist[1]);
233 if (ret)
234 goto error;
235
236 nouveau_irq_register(dev, 8, nv04_fifo_isr);
237error:
238 if (ret)
239 priv->base.base.destroy(dev, NVOBJ_ENGINE_FIFO);
240 return ret;
241}
diff --git a/drivers/gpu/drm/nouveau/nvc0_fence.c b/drivers/gpu/drm/nouveau/nvc0_fence.c
index 817228cd1a95..47ab388a606e 100644
--- a/drivers/gpu/drm/nouveau/nvc0_fence.c
+++ b/drivers/gpu/drm/nouveau/nvc0_fence.c
@@ -25,6 +25,7 @@
25#include "drmP.h" 25#include "drmP.h"
26#include "nouveau_drv.h" 26#include "nouveau_drv.h"
27#include "nouveau_dma.h" 27#include "nouveau_dma.h"
28#include "nouveau_fifo.h"
28#include "nouveau_ramht.h" 29#include "nouveau_ramht.h"
29#include "nouveau_fence.h" 30#include "nouveau_fence.h"
30 31
@@ -148,8 +149,8 @@ nvc0_fence_destroy(struct drm_device *dev, int engine)
148int 149int
149nvc0_fence_create(struct drm_device *dev) 150nvc0_fence_create(struct drm_device *dev)
150{ 151{
152 struct nouveau_fifo_priv *pfifo = nv_engine(dev, NVOBJ_ENGINE_FIFO);
151 struct drm_nouveau_private *dev_priv = dev->dev_private; 153 struct drm_nouveau_private *dev_priv = dev->dev_private;
152 struct nouveau_fifo_engine *pfifo = &dev_priv->engine.fifo;
153 struct nvc0_fence_priv *priv; 154 struct nvc0_fence_priv *priv;
154 int ret; 155 int ret;
155 156
diff --git a/drivers/gpu/drm/nouveau/nvc0_fifo.c b/drivers/gpu/drm/nouveau/nvc0_fifo.c
index 471723eaf8ad..7d85553d518c 100644
--- a/drivers/gpu/drm/nouveau/nvc0_fifo.c
+++ b/drivers/gpu/drm/nouveau/nvc0_fifo.c
@@ -26,10 +26,12 @@
26 26
27#include "nouveau_drv.h" 27#include "nouveau_drv.h"
28#include "nouveau_mm.h" 28#include "nouveau_mm.h"
29#include "nouveau_fifo.h"
29 30
30static void nvc0_fifo_isr(struct drm_device *); 31static void nvc0_fifo_isr(struct drm_device *);
31 32
32struct nvc0_fifo_priv { 33struct nvc0_fifo_priv {
34 struct nouveau_fifo_priv base;
33 struct nouveau_gpuobj *playlist[2]; 35 struct nouveau_gpuobj *playlist[2];
34 int cur_playlist; 36 int cur_playlist;
35 struct nouveau_vma user_vma; 37 struct nouveau_vma user_vma;
@@ -37,8 +39,8 @@ struct nvc0_fifo_priv {
37}; 39};
38 40
39struct nvc0_fifo_chan { 41struct nvc0_fifo_chan {
42 struct nouveau_fifo_chan base;
40 struct nouveau_gpuobj *user; 43 struct nouveau_gpuobj *user;
41 struct nouveau_gpuobj *ramfc;
42}; 44};
43 45
44static void 46static void
@@ -46,8 +48,7 @@ nvc0_fifo_playlist_update(struct drm_device *dev)
46{ 48{
47 struct drm_nouveau_private *dev_priv = dev->dev_private; 49 struct drm_nouveau_private *dev_priv = dev->dev_private;
48 struct nouveau_instmem_engine *pinstmem = &dev_priv->engine.instmem; 50 struct nouveau_instmem_engine *pinstmem = &dev_priv->engine.instmem;
49 struct nouveau_fifo_engine *pfifo = &dev_priv->engine.fifo; 51 struct nvc0_fifo_priv *priv = nv_engine(dev, NVOBJ_ENGINE_FIFO);
50 struct nvc0_fifo_priv *priv = pfifo->priv;
51 struct nouveau_gpuobj *cur; 52 struct nouveau_gpuobj *cur;
52 int i, p; 53 int i, p;
53 54
@@ -69,31 +70,20 @@ nvc0_fifo_playlist_update(struct drm_device *dev)
69 NV_ERROR(dev, "PFIFO - playlist update failed\n"); 70 NV_ERROR(dev, "PFIFO - playlist update failed\n");
70} 71}
71 72
72int 73static int
73nvc0_fifo_create_context(struct nouveau_channel *chan) 74nvc0_fifo_context_new(struct nouveau_channel *chan, int engine)
74{ 75{
75 struct drm_device *dev = chan->dev; 76 struct drm_device *dev = chan->dev;
76 struct drm_nouveau_private *dev_priv = dev->dev_private; 77 struct drm_nouveau_private *dev_priv = dev->dev_private;
77 struct nouveau_instmem_engine *pinstmem = &dev_priv->engine.instmem; 78 struct nouveau_instmem_engine *pinstmem = &dev_priv->engine.instmem;
78 struct nouveau_fifo_engine *pfifo = &dev_priv->engine.fifo; 79 struct nvc0_fifo_priv *priv = nv_engine(dev, engine);
79 struct nvc0_fifo_priv *priv = pfifo->priv; 80 struct nvc0_fifo_chan *fctx;
80 struct nvc0_fifo_chan *fifoch;
81 u64 ib_virt = chan->pushbuf_base + chan->dma.ib_base * 4; 81 u64 ib_virt = chan->pushbuf_base + chan->dma.ib_base * 4;
82 int ret; 82 int ret, i;
83 83
84 chan->fifo_priv = kzalloc(sizeof(*fifoch), GFP_KERNEL); 84 fctx = chan->engctx[engine] = kzalloc(sizeof(*fctx), GFP_KERNEL);
85 if (!chan->fifo_priv) 85 if (!fctx)
86 return -ENOMEM; 86 return -ENOMEM;
87 fifoch = chan->fifo_priv;
88
89 /* allocate vram for control regs, map into polling area */
90 ret = nouveau_gpuobj_new(dev, NULL, 0x1000, 0x1000,
91 NVOBJ_FLAG_ZERO_ALLOC, &fifoch->user);
92 if (ret)
93 goto error;
94
95 nouveau_vm_map_at(&priv->user_vma, chan->id * 0x1000,
96 *(struct nouveau_mem **)fifoch->user->node);
97 87
98 chan->user = ioremap_wc(pci_resource_start(dev->pdev, 1) + 88 chan->user = ioremap_wc(pci_resource_start(dev->pdev, 1) +
99 priv->user_vma.offset + (chan->id * 0x1000), 89 priv->user_vma.offset + (chan->id * 0x1000),
@@ -103,175 +93,77 @@ nvc0_fifo_create_context(struct nouveau_channel *chan)
103 goto error; 93 goto error;
104 } 94 }
105 95
106 /* ramfc */ 96 /* allocate vram for control regs, map into polling area */
107 ret = nouveau_gpuobj_new_fake(dev, chan->ramin->pinst, 97 ret = nouveau_gpuobj_new(dev, NULL, 0x1000, 0x1000,
108 chan->ramin->vinst, 0x100, 98 NVOBJ_FLAG_ZERO_ALLOC, &fctx->user);
109 NVOBJ_FLAG_ZERO_ALLOC, &fifoch->ramfc);
110 if (ret) 99 if (ret)
111 goto error; 100 goto error;
112 101
113 nv_wo32(fifoch->ramfc, 0x08, lower_32_bits(fifoch->user->vinst)); 102 nouveau_vm_map_at(&priv->user_vma, chan->id * 0x1000,
114 nv_wo32(fifoch->ramfc, 0x0c, upper_32_bits(fifoch->user->vinst)); 103 *(struct nouveau_mem **)fctx->user->node);
115 nv_wo32(fifoch->ramfc, 0x10, 0x0000face); 104
116 nv_wo32(fifoch->ramfc, 0x30, 0xfffff902); 105 for (i = 0; i < 0x100; i += 4)
117 nv_wo32(fifoch->ramfc, 0x48, lower_32_bits(ib_virt)); 106 nv_wo32(chan->ramin, i, 0x00000000);
118 nv_wo32(fifoch->ramfc, 0x4c, drm_order(chan->dma.ib_max + 1) << 16 | 107 nv_wo32(chan->ramin, 0x08, lower_32_bits(fctx->user->vinst));
108 nv_wo32(chan->ramin, 0x0c, upper_32_bits(fctx->user->vinst));
109 nv_wo32(chan->ramin, 0x10, 0x0000face);
110 nv_wo32(chan->ramin, 0x30, 0xfffff902);
111 nv_wo32(chan->ramin, 0x48, lower_32_bits(ib_virt));
112 nv_wo32(chan->ramin, 0x4c, drm_order(chan->dma.ib_max + 1) << 16 |
119 upper_32_bits(ib_virt)); 113 upper_32_bits(ib_virt));
120 nv_wo32(fifoch->ramfc, 0x54, 0x00000002); 114 nv_wo32(chan->ramin, 0x54, 0x00000002);
121 nv_wo32(fifoch->ramfc, 0x84, 0x20400000); 115 nv_wo32(chan->ramin, 0x84, 0x20400000);
122 nv_wo32(fifoch->ramfc, 0x94, 0x30000001); 116 nv_wo32(chan->ramin, 0x94, 0x30000001);
123 nv_wo32(fifoch->ramfc, 0x9c, 0x00000100); 117 nv_wo32(chan->ramin, 0x9c, 0x00000100);
124 nv_wo32(fifoch->ramfc, 0xa4, 0x1f1f1f1f); 118 nv_wo32(chan->ramin, 0xa4, 0x1f1f1f1f);
125 nv_wo32(fifoch->ramfc, 0xa8, 0x1f1f1f1f); 119 nv_wo32(chan->ramin, 0xa8, 0x1f1f1f1f);
126 nv_wo32(fifoch->ramfc, 0xac, 0x0000001f); 120 nv_wo32(chan->ramin, 0xac, 0x0000001f);
127 nv_wo32(fifoch->ramfc, 0xb8, 0xf8000000); 121 nv_wo32(chan->ramin, 0xb8, 0xf8000000);
128 nv_wo32(fifoch->ramfc, 0xf8, 0x10003080); /* 0x002310 */ 122 nv_wo32(chan->ramin, 0xf8, 0x10003080); /* 0x002310 */
129 nv_wo32(fifoch->ramfc, 0xfc, 0x10000010); /* 0x002350 */ 123 nv_wo32(chan->ramin, 0xfc, 0x10000010); /* 0x002350 */
130 pinstmem->flush(dev); 124 pinstmem->flush(dev);
131 125
132 nv_wr32(dev, 0x003000 + (chan->id * 8), 0xc0000000 | 126 nv_wr32(dev, 0x003000 + (chan->id * 8), 0xc0000000 |
133 (chan->ramin->vinst >> 12)); 127 (chan->ramin->vinst >> 12));
134 nv_wr32(dev, 0x003004 + (chan->id * 8), 0x001f0001); 128 nv_wr32(dev, 0x003004 + (chan->id * 8), 0x001f0001);
135 nvc0_fifo_playlist_update(dev); 129 nvc0_fifo_playlist_update(dev);
136 return 0;
137 130
138error: 131error:
139 pfifo->destroy_context(chan); 132 if (ret)
133 priv->base.base.context_del(chan, engine);
140 return ret; 134 return ret;
141} 135}
142 136
143void 137static void
144nvc0_fifo_destroy_context(struct nouveau_channel *chan) 138nvc0_fifo_context_del(struct nouveau_channel *chan, int engine)
145{ 139{
140 struct nvc0_fifo_chan *fctx = chan->engctx[engine];
146 struct drm_device *dev = chan->dev; 141 struct drm_device *dev = chan->dev;
147 struct nvc0_fifo_chan *fifoch;
148 142
149 nv_mask(dev, 0x003004 + (chan->id * 8), 0x00000001, 0x00000000); 143 nv_mask(dev, 0x003004 + (chan->id * 8), 0x00000001, 0x00000000);
150 nv_wr32(dev, 0x002634, chan->id); 144 nv_wr32(dev, 0x002634, chan->id);
151 if (!nv_wait(dev, 0x0002634, 0xffffffff, chan->id)) 145 if (!nv_wait(dev, 0x0002634, 0xffffffff, chan->id))
152 NV_WARN(dev, "0x2634 != chid: 0x%08x\n", nv_rd32(dev, 0x2634)); 146 NV_WARN(dev, "0x2634 != chid: 0x%08x\n", nv_rd32(dev, 0x2634));
153
154 nvc0_fifo_playlist_update(dev); 147 nvc0_fifo_playlist_update(dev);
155
156 nv_wr32(dev, 0x003000 + (chan->id * 8), 0x00000000); 148 nv_wr32(dev, 0x003000 + (chan->id * 8), 0x00000000);
157 149
150 nouveau_gpuobj_ref(NULL, &fctx->user);
158 if (chan->user) { 151 if (chan->user) {
159 iounmap(chan->user); 152 iounmap(chan->user);
160 chan->user = NULL; 153 chan->user = NULL;
161 } 154 }
162 155
163 fifoch = chan->fifo_priv; 156 chan->engctx[engine] = NULL;
164 chan->fifo_priv = NULL; 157 kfree(fctx);
165 if (!fifoch)
166 return;
167
168 nouveau_gpuobj_ref(NULL, &fifoch->ramfc);
169 nouveau_gpuobj_ref(NULL, &fifoch->user);
170 kfree(fifoch);
171}
172
173int
174nvc0_fifo_load_context(struct nouveau_channel *chan)
175{
176 return 0;
177}
178
179int
180nvc0_fifo_unload_context(struct drm_device *dev)
181{
182 int i;
183
184 for (i = 0; i < 128; i++) {
185 if (!(nv_rd32(dev, 0x003004 + (i * 8)) & 1))
186 continue;
187
188 nv_mask(dev, 0x003004 + (i * 8), 0x00000001, 0x00000000);
189 nv_wr32(dev, 0x002634, i);
190 if (!nv_wait(dev, 0x002634, 0xffffffff, i)) {
191 NV_INFO(dev, "PFIFO: kick ch %d failed: 0x%08x\n",
192 i, nv_rd32(dev, 0x002634));
193 return -EBUSY;
194 }
195 }
196
197 return 0;
198}
199
200static void
201nvc0_fifo_destroy(struct drm_device *dev)
202{
203 struct drm_nouveau_private *dev_priv = dev->dev_private;
204 struct nouveau_fifo_engine *pfifo = &dev_priv->engine.fifo;
205 struct nvc0_fifo_priv *priv;
206
207 priv = pfifo->priv;
208 if (!priv)
209 return;
210
211 nouveau_vm_put(&priv->user_vma);
212 nouveau_gpuobj_ref(NULL, &priv->playlist[1]);
213 nouveau_gpuobj_ref(NULL, &priv->playlist[0]);
214 kfree(priv);
215}
216
217void
218nvc0_fifo_takedown(struct drm_device *dev)
219{
220 nv_wr32(dev, 0x002140, 0x00000000);
221 nvc0_fifo_destroy(dev);
222} 158}
223 159
224static int 160static int
225nvc0_fifo_create(struct drm_device *dev) 161nvc0_fifo_init(struct drm_device *dev, int engine)
226{
227 struct drm_nouveau_private *dev_priv = dev->dev_private;
228 struct nouveau_fifo_engine *pfifo = &dev_priv->engine.fifo;
229 struct nvc0_fifo_priv *priv;
230 int ret;
231
232 priv = kzalloc(sizeof(*priv), GFP_KERNEL);
233 if (!priv)
234 return -ENOMEM;
235 pfifo->priv = priv;
236
237 ret = nouveau_gpuobj_new(dev, NULL, 0x1000, 0x1000, 0,
238 &priv->playlist[0]);
239 if (ret)
240 goto error;
241
242 ret = nouveau_gpuobj_new(dev, NULL, 0x1000, 0x1000, 0,
243 &priv->playlist[1]);
244 if (ret)
245 goto error;
246
247 ret = nouveau_vm_get(dev_priv->bar1_vm, pfifo->channels * 0x1000,
248 12, NV_MEM_ACCESS_RW, &priv->user_vma);
249 if (ret)
250 goto error;
251
252 nouveau_irq_register(dev, 8, nvc0_fifo_isr);
253 return 0;
254
255error:
256 nvc0_fifo_destroy(dev);
257 return ret;
258}
259
260int
261nvc0_fifo_init(struct drm_device *dev)
262{ 162{
263 struct drm_nouveau_private *dev_priv = dev->dev_private; 163 struct drm_nouveau_private *dev_priv = dev->dev_private;
264 struct nouveau_fifo_engine *pfifo = &dev_priv->engine.fifo; 164 struct nvc0_fifo_priv *priv = nv_engine(dev, engine);
265 struct nouveau_channel *chan; 165 struct nouveau_channel *chan;
266 struct nvc0_fifo_priv *priv; 166 int i;
267 int ret, i;
268
269 if (!pfifo->priv) {
270 ret = nvc0_fifo_create(dev);
271 if (ret)
272 return ret;
273 }
274 priv = pfifo->priv;
275 167
276 /* reset PFIFO, enable all available PSUBFIFO areas */ 168 /* reset PFIFO, enable all available PSUBFIFO areas */
277 nv_mask(dev, 0x000200, 0x00000100, 0x00000000); 169 nv_mask(dev, 0x000200, 0x00000100, 0x00000000);
@@ -309,7 +201,7 @@ nvc0_fifo_init(struct drm_device *dev)
309 /* restore PFIFO context table */ 201 /* restore PFIFO context table */
310 for (i = 0; i < 128; i++) { 202 for (i = 0; i < 128; i++) {
311 chan = dev_priv->channels.ptr[i]; 203 chan = dev_priv->channels.ptr[i];
312 if (!chan || !chan->fifo_priv) 204 if (!chan || !chan->engctx[engine])
313 continue; 205 continue;
314 206
315 nv_wr32(dev, 0x003000 + (i * 8), 0xc0000000 | 207 nv_wr32(dev, 0x003000 + (i * 8), 0xc0000000 |
@@ -321,6 +213,29 @@ nvc0_fifo_init(struct drm_device *dev)
321 return 0; 213 return 0;
322} 214}
323 215
216static int
217nvc0_fifo_fini(struct drm_device *dev, int engine, bool suspend)
218{
219 int i;
220
221 for (i = 0; i < 128; i++) {
222 if (!(nv_rd32(dev, 0x003004 + (i * 8)) & 1))
223 continue;
224
225 nv_mask(dev, 0x003004 + (i * 8), 0x00000001, 0x00000000);
226 nv_wr32(dev, 0x002634, i);
227 if (!nv_wait(dev, 0x002634, 0xffffffff, i)) {
228 NV_INFO(dev, "PFIFO: kick ch %d failed: 0x%08x\n",
229 i, nv_rd32(dev, 0x002634));
230 return -EBUSY;
231 }
232 }
233
234 nv_wr32(dev, 0x002140, 0x00000000);
235 return 0;
236}
237
238
324struct nouveau_enum nvc0_fifo_fault_unit[] = { 239struct nouveau_enum nvc0_fifo_fault_unit[] = {
325 { 0x00, "PGRAPH" }, 240 { 0x00, "PGRAPH" },
326 { 0x03, "PEEPHOLE" }, 241 { 0x03, "PEEPHOLE" },
@@ -410,13 +325,14 @@ nvc0_fifo_isr_vm_fault(struct drm_device *dev, int unit)
410static int 325static int
411nvc0_fifo_page_flip(struct drm_device *dev, u32 chid) 326nvc0_fifo_page_flip(struct drm_device *dev, u32 chid)
412{ 327{
328 struct nvc0_fifo_priv *priv = nv_engine(dev, NVOBJ_ENGINE_FIFO);
413 struct drm_nouveau_private *dev_priv = dev->dev_private; 329 struct drm_nouveau_private *dev_priv = dev->dev_private;
414 struct nouveau_channel *chan = NULL; 330 struct nouveau_channel *chan = NULL;
415 unsigned long flags; 331 unsigned long flags;
416 int ret = -EINVAL; 332 int ret = -EINVAL;
417 333
418 spin_lock_irqsave(&dev_priv->channels.lock, flags); 334 spin_lock_irqsave(&dev_priv->channels.lock, flags);
419 if (likely(chid >= 0 && chid < dev_priv->engine.fifo.channels)) { 335 if (likely(chid >= 0 && chid < priv->base.channels)) {
420 chan = dev_priv->channels.ptr[chid]; 336 chan = dev_priv->channels.ptr[chid];
421 if (likely(chan)) 337 if (likely(chan))
422 ret = nouveau_finish_page_flip(chan, NULL); 338 ret = nouveau_finish_page_flip(chan, NULL);
@@ -505,3 +421,56 @@ nvc0_fifo_isr(struct drm_device *dev)
505 nv_wr32(dev, 0x002140, 0); 421 nv_wr32(dev, 0x002140, 0);
506 } 422 }
507} 423}
424
425static void
426nvc0_fifo_destroy(struct drm_device *dev, int engine)
427{
428 struct nvc0_fifo_priv *priv = nv_engine(dev, NVOBJ_ENGINE_FIFO);
429 struct drm_nouveau_private *dev_priv = dev->dev_private;
430
431 nouveau_vm_put(&priv->user_vma);
432 nouveau_gpuobj_ref(NULL, &priv->playlist[1]);
433 nouveau_gpuobj_ref(NULL, &priv->playlist[0]);
434
435 dev_priv->eng[engine] = NULL;
436 kfree(priv);
437}
438
439int
440nvc0_fifo_create(struct drm_device *dev)
441{
442 struct drm_nouveau_private *dev_priv = dev->dev_private;
443 struct nvc0_fifo_priv *priv;
444 int ret;
445
446 priv = kzalloc(sizeof(*priv), GFP_KERNEL);
447 if (!priv)
448 return -ENOMEM;
449
450 priv->base.base.destroy = nvc0_fifo_destroy;
451 priv->base.base.init = nvc0_fifo_init;
452 priv->base.base.fini = nvc0_fifo_fini;
453 priv->base.base.context_new = nvc0_fifo_context_new;
454 priv->base.base.context_del = nvc0_fifo_context_del;
455 priv->base.channels = 128;
456 dev_priv->eng[NVOBJ_ENGINE_FIFO] = &priv->base.base;
457
458 ret = nouveau_gpuobj_new(dev, NULL, 4096, 4096, 0, &priv->playlist[0]);
459 if (ret)
460 goto error;
461
462 ret = nouveau_gpuobj_new(dev, NULL, 4096, 4096, 0, &priv->playlist[1]);
463 if (ret)
464 goto error;
465
466 ret = nouveau_vm_get(dev_priv->bar1_vm, priv->base.channels * 0x1000,
467 12, NV_MEM_ACCESS_RW, &priv->user_vma);
468 if (ret)
469 goto error;
470
471 nouveau_irq_register(dev, 8, nvc0_fifo_isr);
472error:
473 if (ret)
474 priv->base.base.destroy(dev, NVOBJ_ENGINE_FIFO);
475 return ret;
476}
diff --git a/drivers/gpu/drm/nouveau/nvc0_graph.c b/drivers/gpu/drm/nouveau/nvc0_graph.c
index 9066102d1159..2a01e6e47724 100644
--- a/drivers/gpu/drm/nouveau/nvc0_graph.c
+++ b/drivers/gpu/drm/nouveau/nvc0_graph.c
@@ -29,6 +29,7 @@
29 29
30#include "nouveau_drv.h" 30#include "nouveau_drv.h"
31#include "nouveau_mm.h" 31#include "nouveau_mm.h"
32#include "nouveau_fifo.h"
32 33
33#include "nvc0_graph.h" 34#include "nvc0_graph.h"
34#include "nvc0_grhub.fuc.h" 35#include "nvc0_grhub.fuc.h"
@@ -620,13 +621,14 @@ nvc0_graph_init(struct drm_device *dev, int engine)
620int 621int
621nvc0_graph_isr_chid(struct drm_device *dev, u64 inst) 622nvc0_graph_isr_chid(struct drm_device *dev, u64 inst)
622{ 623{
624 struct nouveau_fifo_priv *pfifo = nv_engine(dev, NVOBJ_ENGINE_FIFO);
623 struct drm_nouveau_private *dev_priv = dev->dev_private; 625 struct drm_nouveau_private *dev_priv = dev->dev_private;
624 struct nouveau_channel *chan; 626 struct nouveau_channel *chan;
625 unsigned long flags; 627 unsigned long flags;
626 int i; 628 int i;
627 629
628 spin_lock_irqsave(&dev_priv->channels.lock, flags); 630 spin_lock_irqsave(&dev_priv->channels.lock, flags);
629 for (i = 0; i < dev_priv->engine.fifo.channels; i++) { 631 for (i = 0; i < pfifo->channels; i++) {
630 chan = dev_priv->channels.ptr[i]; 632 chan = dev_priv->channels.ptr[i];
631 if (!chan || !chan->ramin) 633 if (!chan || !chan->ramin)
632 continue; 634 continue;
diff --git a/drivers/gpu/drm/nouveau/nve0_fifo.c b/drivers/gpu/drm/nouveau/nve0_fifo.c
index 52c54e0fdcee..1855ecbd843b 100644
--- a/drivers/gpu/drm/nouveau/nve0_fifo.c
+++ b/drivers/gpu/drm/nouveau/nve0_fifo.c
@@ -26,6 +26,7 @@
26 26
27#include "nouveau_drv.h" 27#include "nouveau_drv.h"
28#include "nouveau_mm.h" 28#include "nouveau_mm.h"
29#include "nouveau_fifo.h"
29 30
30#define NVE0_FIFO_ENGINE_NUM 32 31#define NVE0_FIFO_ENGINE_NUM 32
31 32
@@ -37,6 +38,7 @@ struct nve0_fifo_engine {
37}; 38};
38 39
39struct nve0_fifo_priv { 40struct nve0_fifo_priv {
41 struct nouveau_fifo_priv base;
40 struct nve0_fifo_engine engine[NVE0_FIFO_ENGINE_NUM]; 42 struct nve0_fifo_engine engine[NVE0_FIFO_ENGINE_NUM];
41 struct { 43 struct {
42 struct nouveau_gpuobj *mem; 44 struct nouveau_gpuobj *mem;
@@ -46,7 +48,7 @@ struct nve0_fifo_priv {
46}; 48};
47 49
48struct nve0_fifo_chan { 50struct nve0_fifo_chan {
49 struct nouveau_gpuobj *ramfc; 51 struct nouveau_fifo_chan base;
50 u32 engine; 52 u32 engine;
51}; 53};
52 54
@@ -55,8 +57,7 @@ nve0_fifo_playlist_update(struct drm_device *dev, u32 engine)
55{ 57{
56 struct drm_nouveau_private *dev_priv = dev->dev_private; 58 struct drm_nouveau_private *dev_priv = dev->dev_private;
57 struct nouveau_instmem_engine *pinstmem = &dev_priv->engine.instmem; 59 struct nouveau_instmem_engine *pinstmem = &dev_priv->engine.instmem;
58 struct nouveau_fifo_engine *pfifo = &dev_priv->engine.fifo; 60 struct nve0_fifo_priv *priv = nv_engine(dev, NVOBJ_ENGINE_FIFO);
59 struct nve0_fifo_priv *priv = pfifo->priv;
60 struct nve0_fifo_engine *peng = &priv->engine[engine]; 61 struct nve0_fifo_engine *peng = &priv->engine[engine];
61 struct nouveau_gpuobj *cur; 62 struct nouveau_gpuobj *cur;
62 u32 match = (engine << 16) | 0x00000001; 63 u32 match = (engine << 16) | 0x00000001;
@@ -75,7 +76,7 @@ nve0_fifo_playlist_update(struct drm_device *dev, u32 engine)
75 76
76 peng->cur_playlist = !peng->cur_playlist; 77 peng->cur_playlist = !peng->cur_playlist;
77 78
78 for (i = 0, p = 0; i < pfifo->channels; i++) { 79 for (i = 0, p = 0; i < priv->base.channels; i++) {
79 u32 ctrl = nv_rd32(dev, 0x800004 + (i * 8)) & 0x001f0001; 80 u32 ctrl = nv_rd32(dev, 0x800004 + (i * 8)) & 0x001f0001;
80 if (ctrl != match) 81 if (ctrl != match)
81 continue; 82 continue;
@@ -91,24 +92,23 @@ nve0_fifo_playlist_update(struct drm_device *dev, u32 engine)
91 NV_ERROR(dev, "PFIFO: playlist %d update timeout\n", engine); 92 NV_ERROR(dev, "PFIFO: playlist %d update timeout\n", engine);
92} 93}
93 94
94int 95static int
95nve0_fifo_create_context(struct nouveau_channel *chan) 96nve0_fifo_context_new(struct nouveau_channel *chan, int engine)
96{ 97{
97 struct drm_device *dev = chan->dev; 98 struct drm_device *dev = chan->dev;
98 struct drm_nouveau_private *dev_priv = dev->dev_private; 99 struct drm_nouveau_private *dev_priv = dev->dev_private;
99 struct nouveau_instmem_engine *pinstmem = &dev_priv->engine.instmem; 100 struct nouveau_instmem_engine *pinstmem = &dev_priv->engine.instmem;
100 struct nouveau_fifo_engine *pfifo = &dev_priv->engine.fifo; 101 struct nve0_fifo_priv *priv = nv_engine(dev, engine);
101 struct nve0_fifo_priv *priv = pfifo->priv; 102 struct nve0_fifo_chan *fctx;
102 struct nve0_fifo_chan *fifoch;
103 u64 usermem = priv->user.mem->vinst + chan->id * 512; 103 u64 usermem = priv->user.mem->vinst + chan->id * 512;
104 u64 ib_virt = chan->pushbuf_base + chan->dma.ib_base * 4; 104 u64 ib_virt = chan->pushbuf_base + chan->dma.ib_base * 4;
105 int ret; 105 int ret = 0, i;
106 106
107 chan->fifo_priv = kzalloc(sizeof(*fifoch), GFP_KERNEL); 107 fctx = chan->engctx[engine] = kzalloc(sizeof(*fctx), GFP_KERNEL);
108 if (!chan->fifo_priv) 108 if (!fctx)
109 return -ENOMEM; 109 return -ENOMEM;
110 fifoch = chan->fifo_priv; 110
111 fifoch->engine = 0; /* PGRAPH */ 111 fctx->engine = 0; /* PGRAPH */
112 112
113 /* allocate vram for control regs, map into polling area */ 113 /* allocate vram for control regs, map into polling area */
114 chan->user = ioremap_wc(pci_resource_start(dev->pdev, 1) + 114 chan->user = ioremap_wc(pci_resource_start(dev->pdev, 1) +
@@ -118,56 +118,48 @@ nve0_fifo_create_context(struct nouveau_channel *chan)
118 goto error; 118 goto error;
119 } 119 }
120 120
121 /* ramfc */ 121 for (i = 0; i < 0x100; i += 4)
122 ret = nouveau_gpuobj_new_fake(dev, chan->ramin->pinst, 122 nv_wo32(chan->ramin, i, 0x00000000);
123 chan->ramin->vinst, 0x100, 123 nv_wo32(chan->ramin, 0x08, lower_32_bits(usermem));
124 NVOBJ_FLAG_ZERO_ALLOC, &fifoch->ramfc); 124 nv_wo32(chan->ramin, 0x0c, upper_32_bits(usermem));
125 if (ret) 125 nv_wo32(chan->ramin, 0x10, 0x0000face);
126 goto error; 126 nv_wo32(chan->ramin, 0x30, 0xfffff902);
127 127 nv_wo32(chan->ramin, 0x48, lower_32_bits(ib_virt));
128 nv_wo32(fifoch->ramfc, 0x08, lower_32_bits(usermem)); 128 nv_wo32(chan->ramin, 0x4c, drm_order(chan->dma.ib_max + 1) << 16 |
129 nv_wo32(fifoch->ramfc, 0x0c, upper_32_bits(usermem));
130 nv_wo32(fifoch->ramfc, 0x10, 0x0000face);
131 nv_wo32(fifoch->ramfc, 0x30, 0xfffff902);
132 nv_wo32(fifoch->ramfc, 0x48, lower_32_bits(ib_virt));
133 nv_wo32(fifoch->ramfc, 0x4c, drm_order(chan->dma.ib_max + 1) << 16 |
134 upper_32_bits(ib_virt)); 129 upper_32_bits(ib_virt));
135 nv_wo32(fifoch->ramfc, 0x84, 0x20400000); 130 nv_wo32(chan->ramin, 0x84, 0x20400000);
136 nv_wo32(fifoch->ramfc, 0x94, 0x30000001); 131 nv_wo32(chan->ramin, 0x94, 0x30000001);
137 nv_wo32(fifoch->ramfc, 0x9c, 0x00000100); 132 nv_wo32(chan->ramin, 0x9c, 0x00000100);
138 nv_wo32(fifoch->ramfc, 0xac, 0x0000001f); 133 nv_wo32(chan->ramin, 0xac, 0x0000001f);
139 nv_wo32(fifoch->ramfc, 0xe4, 0x00000000); 134 nv_wo32(chan->ramin, 0xe4, 0x00000000);
140 nv_wo32(fifoch->ramfc, 0xe8, chan->id); 135 nv_wo32(chan->ramin, 0xe8, chan->id);
141 nv_wo32(fifoch->ramfc, 0xf8, 0x10003080); /* 0x002310 */ 136 nv_wo32(chan->ramin, 0xf8, 0x10003080); /* 0x002310 */
142 nv_wo32(fifoch->ramfc, 0xfc, 0x10000010); /* 0x002350 */ 137 nv_wo32(chan->ramin, 0xfc, 0x10000010); /* 0x002350 */
143 pinstmem->flush(dev); 138 pinstmem->flush(dev);
144 139
145 nv_wr32(dev, 0x800000 + (chan->id * 8), 0x80000000 | 140 nv_wr32(dev, 0x800000 + (chan->id * 8), 0x80000000 |
146 (chan->ramin->vinst >> 12)); 141 (chan->ramin->vinst >> 12));
147 nv_mask(dev, 0x800004 + (chan->id * 8), 0x00000400, 0x00000400); 142 nv_mask(dev, 0x800004 + (chan->id * 8), 0x00000400, 0x00000400);
148 nve0_fifo_playlist_update(dev, fifoch->engine); 143 nve0_fifo_playlist_update(dev, fctx->engine);
149 nv_mask(dev, 0x800004 + (chan->id * 8), 0x00000400, 0x00000400); 144 nv_mask(dev, 0x800004 + (chan->id * 8), 0x00000400, 0x00000400);
150 return 0;
151 145
152error: 146error:
153 pfifo->destroy_context(chan); 147 if (ret)
148 priv->base.base.context_del(chan, engine);
154 return ret; 149 return ret;
155} 150}
156 151
157void 152static void
158nve0_fifo_destroy_context(struct nouveau_channel *chan) 153nve0_fifo_context_del(struct nouveau_channel *chan, int engine)
159{ 154{
160 struct nve0_fifo_chan *fifoch = chan->fifo_priv; 155 struct nve0_fifo_chan *fctx = chan->engctx[engine];
161 struct drm_device *dev = chan->dev; 156 struct drm_device *dev = chan->dev;
162 157
163 if (!fifoch)
164 return;
165
166 nv_mask(dev, 0x800004 + (chan->id * 8), 0x00000800, 0x00000800); 158 nv_mask(dev, 0x800004 + (chan->id * 8), 0x00000800, 0x00000800);
167 nv_wr32(dev, 0x002634, chan->id); 159 nv_wr32(dev, 0x002634, chan->id);
168 if (!nv_wait(dev, 0x0002634, 0xffffffff, chan->id)) 160 if (!nv_wait(dev, 0x0002634, 0xffffffff, chan->id))
169 NV_WARN(dev, "0x2634 != chid: 0x%08x\n", nv_rd32(dev, 0x2634)); 161 NV_WARN(dev, "0x2634 != chid: 0x%08x\n", nv_rd32(dev, 0x2634));
170 nve0_fifo_playlist_update(dev, fifoch->engine); 162 nve0_fifo_playlist_update(dev, fctx->engine);
171 nv_wr32(dev, 0x800000 + (chan->id * 8), 0x00000000); 163 nv_wr32(dev, 0x800000 + (chan->id * 8), 0x00000000);
172 164
173 if (chan->user) { 165 if (chan->user) {
@@ -175,118 +167,17 @@ nve0_fifo_destroy_context(struct nouveau_channel *chan)
175 chan->user = NULL; 167 chan->user = NULL;
176 } 168 }
177 169
178 nouveau_gpuobj_ref(NULL, &fifoch->ramfc); 170 chan->engctx[NVOBJ_ENGINE_FIFO] = NULL;
179 chan->fifo_priv = NULL; 171 kfree(fctx);
180 kfree(fifoch);
181}
182
183int
184nve0_fifo_load_context(struct nouveau_channel *chan)
185{
186 return 0;
187}
188
189int
190nve0_fifo_unload_context(struct drm_device *dev)
191{
192 struct drm_nouveau_private *dev_priv = dev->dev_private;
193 struct nouveau_fifo_engine *pfifo = &dev_priv->engine.fifo;
194 int i;
195
196 for (i = 0; i < pfifo->channels; i++) {
197 if (!(nv_rd32(dev, 0x800004 + (i * 8)) & 1))
198 continue;
199
200 nv_mask(dev, 0x800004 + (i * 8), 0x00000800, 0x00000800);
201 nv_wr32(dev, 0x002634, i);
202 if (!nv_wait(dev, 0x002634, 0xffffffff, i)) {
203 NV_INFO(dev, "PFIFO: kick ch %d failed: 0x%08x\n",
204 i, nv_rd32(dev, 0x002634));
205 return -EBUSY;
206 }
207 }
208
209 return 0;
210}
211
212static void
213nve0_fifo_destroy(struct drm_device *dev)
214{
215 struct drm_nouveau_private *dev_priv = dev->dev_private;
216 struct nouveau_fifo_engine *pfifo = &dev_priv->engine.fifo;
217 struct nve0_fifo_priv *priv;
218 int i;
219
220 priv = pfifo->priv;
221 if (!priv)
222 return;
223
224 nouveau_vm_put(&priv->user.bar);
225 nouveau_gpuobj_ref(NULL, &priv->user.mem);
226
227 for (i = 0; i < NVE0_FIFO_ENGINE_NUM; i++) {
228 nouveau_gpuobj_ref(NULL, &priv->engine[i].playlist[0]);
229 nouveau_gpuobj_ref(NULL, &priv->engine[i].playlist[1]);
230 }
231 kfree(priv);
232}
233
234void
235nve0_fifo_takedown(struct drm_device *dev)
236{
237 nv_wr32(dev, 0x002140, 0x00000000);
238 nve0_fifo_destroy(dev);
239} 172}
240 173
241static int 174static int
242nve0_fifo_create(struct drm_device *dev) 175nve0_fifo_init(struct drm_device *dev, int engine)
243{
244 struct drm_nouveau_private *dev_priv = dev->dev_private;
245 struct nouveau_fifo_engine *pfifo = &dev_priv->engine.fifo;
246 struct nve0_fifo_priv *priv;
247 int ret;
248
249 priv = kzalloc(sizeof(*priv), GFP_KERNEL);
250 if (!priv)
251 return -ENOMEM;
252 pfifo->priv = priv;
253
254 ret = nouveau_gpuobj_new(dev, NULL, pfifo->channels * 512, 0x1000,
255 NVOBJ_FLAG_ZERO_ALLOC, &priv->user.mem);
256 if (ret)
257 goto error;
258
259 ret = nouveau_vm_get(dev_priv->bar1_vm, priv->user.mem->size,
260 12, NV_MEM_ACCESS_RW, &priv->user.bar);
261 if (ret)
262 goto error;
263
264 nouveau_vm_map(&priv->user.bar, *(struct nouveau_mem **)priv->user.mem->node);
265
266 nouveau_irq_register(dev, 8, nve0_fifo_isr);
267 return 0;
268
269error:
270 nve0_fifo_destroy(dev);
271 return ret;
272}
273
274int
275nve0_fifo_init(struct drm_device *dev)
276{ 176{
277 struct drm_nouveau_private *dev_priv = dev->dev_private; 177 struct drm_nouveau_private *dev_priv = dev->dev_private;
278 struct nouveau_fifo_engine *pfifo = &dev_priv->engine.fifo; 178 struct nve0_fifo_priv *priv = nv_engine(dev, engine);
279 struct nouveau_channel *chan; 179 struct nve0_fifo_chan *fctx;
280 struct nve0_fifo_chan *fifoch; 180 int i;
281 struct nve0_fifo_priv *priv;
282 int ret, i;
283
284 if (!pfifo->priv) {
285 ret = nve0_fifo_create(dev);
286 if (ret)
287 return ret;
288 }
289 priv = pfifo->priv;
290 181
291 /* reset PFIFO, enable all available PSUBFIFO areas */ 182 /* reset PFIFO, enable all available PSUBFIFO areas */
292 nv_mask(dev, 0x000200, 0x00000100, 0x00000000); 183 nv_mask(dev, 0x000200, 0x00000100, 0x00000000);
@@ -310,22 +201,44 @@ nve0_fifo_init(struct drm_device *dev)
310 nv_wr32(dev, 0x002140, 0xbfffffff); 201 nv_wr32(dev, 0x002140, 0xbfffffff);
311 202
312 /* restore PFIFO context table */ 203 /* restore PFIFO context table */
313 for (i = 0; i < pfifo->channels; i++) { 204 for (i = 0; i < priv->base.channels; i++) {
314 chan = dev_priv->channels.ptr[i]; 205 struct nouveau_channel *chan = dev_priv->channels.ptr[i];
315 if (!chan || !chan->fifo_priv) 206 if (!chan || !(fctx = chan->engctx[engine]))
316 continue; 207 continue;
317 fifoch = chan->fifo_priv;
318 208
319 nv_wr32(dev, 0x800000 + (i * 8), 0x80000000 | 209 nv_wr32(dev, 0x800000 + (i * 8), 0x80000000 |
320 (chan->ramin->vinst >> 12)); 210 (chan->ramin->vinst >> 12));
321 nv_mask(dev, 0x800004 + (i * 8), 0x00000400, 0x00000400); 211 nv_mask(dev, 0x800004 + (i * 8), 0x00000400, 0x00000400);
322 nve0_fifo_playlist_update(dev, fifoch->engine); 212 nve0_fifo_playlist_update(dev, fctx->engine);
323 nv_mask(dev, 0x800004 + (i * 8), 0x00000400, 0x00000400); 213 nv_mask(dev, 0x800004 + (i * 8), 0x00000400, 0x00000400);
324 } 214 }
325 215
326 return 0; 216 return 0;
327} 217}
328 218
219static int
220nve0_fifo_fini(struct drm_device *dev, int engine, bool suspend)
221{
222 struct nve0_fifo_priv *priv = nv_engine(dev, engine);
223 int i;
224
225 for (i = 0; i < priv->base.channels; i++) {
226 if (!(nv_rd32(dev, 0x800004 + (i * 8)) & 1))
227 continue;
228
229 nv_mask(dev, 0x800004 + (i * 8), 0x00000800, 0x00000800);
230 nv_wr32(dev, 0x002634, i);
231 if (!nv_wait(dev, 0x002634, 0xffffffff, i)) {
232 NV_INFO(dev, "PFIFO: kick ch %d failed: 0x%08x\n",
233 i, nv_rd32(dev, 0x002634));
234 return -EBUSY;
235 }
236 }
237
238 nv_wr32(dev, 0x002140, 0x00000000);
239 return 0;
240}
241
329struct nouveau_enum nve0_fifo_fault_unit[] = { 242struct nouveau_enum nve0_fifo_fault_unit[] = {
330 {} 243 {}
331}; 244};
@@ -451,3 +364,60 @@ nve0_fifo_isr(struct drm_device *dev)
451 nv_wr32(dev, 0x002140, 0); 364 nv_wr32(dev, 0x002140, 0);
452 } 365 }
453} 366}
367
368static void
369nve0_fifo_destroy(struct drm_device *dev, int engine)
370{
371 struct drm_nouveau_private *dev_priv = dev->dev_private;
372 struct nve0_fifo_priv *priv = nv_engine(dev, engine);
373 int i;
374
375 nouveau_vm_put(&priv->user.bar);
376 nouveau_gpuobj_ref(NULL, &priv->user.mem);
377
378 for (i = 0; i < NVE0_FIFO_ENGINE_NUM; i++) {
379 nouveau_gpuobj_ref(NULL, &priv->engine[i].playlist[0]);
380 nouveau_gpuobj_ref(NULL, &priv->engine[i].playlist[1]);
381 }
382
383 dev_priv->eng[engine] = NULL;
384 kfree(priv);
385}
386
387int
388nve0_fifo_create(struct drm_device *dev)
389{
390 struct drm_nouveau_private *dev_priv = dev->dev_private;
391 struct nve0_fifo_priv *priv;
392 int ret;
393
394 priv = kzalloc(sizeof(*priv), GFP_KERNEL);
395 if (!priv)
396 return -ENOMEM;
397
398 priv->base.base.destroy = nve0_fifo_destroy;
399 priv->base.base.init = nve0_fifo_init;
400 priv->base.base.fini = nve0_fifo_fini;
401 priv->base.base.context_new = nve0_fifo_context_new;
402 priv->base.base.context_del = nve0_fifo_context_del;
403 priv->base.channels = 4096;
404 dev_priv->eng[NVOBJ_ENGINE_FIFO] = &priv->base.base;
405
406 ret = nouveau_gpuobj_new(dev, NULL, priv->base.channels * 512, 0x1000,
407 NVOBJ_FLAG_ZERO_ALLOC, &priv->user.mem);
408 if (ret)
409 goto error;
410
411 ret = nouveau_vm_get(dev_priv->bar1_vm, priv->user.mem->size,
412 12, NV_MEM_ACCESS_RW, &priv->user.bar);
413 if (ret)
414 goto error;
415
416 nouveau_vm_map(&priv->user.bar, *(struct nouveau_mem **)priv->user.mem->node);
417
418 nouveau_irq_register(dev, 8, nve0_fifo_isr);
419error:
420 if (ret)
421 priv->base.base.destroy(dev, NVOBJ_ENGINE_FIFO);
422 return ret;
423}
diff --git a/drivers/gpu/drm/nouveau/nve0_graph.c b/drivers/gpu/drm/nouveau/nve0_graph.c
index 9cc3cf246e09..41356274ca61 100644
--- a/drivers/gpu/drm/nouveau/nve0_graph.c
+++ b/drivers/gpu/drm/nouveau/nve0_graph.c
@@ -29,6 +29,7 @@
29 29
30#include "nouveau_drv.h" 30#include "nouveau_drv.h"
31#include "nouveau_mm.h" 31#include "nouveau_mm.h"
32#include "nouveau_fifo.h"
32 33
33#include "nve0_graph.h" 34#include "nve0_graph.h"
34 35
@@ -548,13 +549,14 @@ nve0_graph_init(struct drm_device *dev, int engine)
548int 549int
549nve0_graph_isr_chid(struct drm_device *dev, u64 inst) 550nve0_graph_isr_chid(struct drm_device *dev, u64 inst)
550{ 551{
552 struct nouveau_fifo_priv *pfifo = nv_engine(dev, NVOBJ_ENGINE_FIFO);
551 struct drm_nouveau_private *dev_priv = dev->dev_private; 553 struct drm_nouveau_private *dev_priv = dev->dev_private;
552 struct nouveau_channel *chan; 554 struct nouveau_channel *chan;
553 unsigned long flags; 555 unsigned long flags;
554 int i; 556 int i;
555 557
556 spin_lock_irqsave(&dev_priv->channels.lock, flags); 558 spin_lock_irqsave(&dev_priv->channels.lock, flags);
557 for (i = 0; i < dev_priv->engine.fifo.channels; i++) { 559 for (i = 0; i < pfifo->channels; i++) {
558 chan = dev_priv->channels.ptr[i]; 560 chan = dev_priv->channels.ptr[i];
559 if (!chan || !chan->ramin) 561 if (!chan || !chan->ramin)
560 continue; 562 continue;