• Home
  • History
  • Annotate
  • Line#
  • Navigate
  • Raw
  • Download
  • only in /netgear-R7000-V1.0.7.12_1.2.5/components/opensource/linux/linux-2.6.36/drivers/gpu/drm/nouveau/
1/*
2 * Copyright (C) 2007 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 "nouveau_drv.h"
29#include "nouveau_drm.h"
30
31#define NV40_RAMFC(c) (dev_priv->ramfc_offset + ((c) * NV40_RAMFC__SIZE))
32#define NV40_RAMFC__SIZE 128
33
34int
35nv40_fifo_create_context(struct nouveau_channel *chan)
36{
37	struct drm_device *dev = chan->dev;
38	struct drm_nouveau_private *dev_priv = dev->dev_private;
39	uint32_t fc = NV40_RAMFC(chan->id);
40	unsigned long flags;
41	int ret;
42
43	ret = nouveau_gpuobj_new_fake(dev, NV40_RAMFC(chan->id), ~0,
44				      NV40_RAMFC__SIZE, NVOBJ_FLAG_ZERO_ALLOC |
45				      NVOBJ_FLAG_ZERO_FREE, NULL, &chan->ramfc);
46	if (ret)
47		return ret;
48
49	spin_lock_irqsave(&dev_priv->context_switch_lock, flags);
50
51	nv_wi32(dev, fc +  0, chan->pushbuf_base);
52	nv_wi32(dev, fc +  4, chan->pushbuf_base);
53	nv_wi32(dev, fc + 12, chan->pushbuf->instance >> 4);
54	nv_wi32(dev, fc + 24, NV_PFIFO_CACHE1_DMA_FETCH_TRIG_128_BYTES |
55			      NV_PFIFO_CACHE1_DMA_FETCH_SIZE_128_BYTES |
56			      NV_PFIFO_CACHE1_DMA_FETCH_MAX_REQS_8 |
57#ifdef __BIG_ENDIAN
58			      NV_PFIFO_CACHE1_BIG_ENDIAN |
59#endif
60			      0x30000000 /* no idea.. */);
61	nv_wi32(dev, fc + 56, chan->ramin_grctx->instance >> 4);
62	nv_wi32(dev, fc + 60, 0x0001FFFF);
63
64	/* enable the fifo dma operation */
65	nv_wr32(dev, NV04_PFIFO_MODE,
66		nv_rd32(dev, NV04_PFIFO_MODE) | (1 << chan->id));
67
68	spin_unlock_irqrestore(&dev_priv->context_switch_lock, flags);
69	return 0;
70}
71
72void
73nv40_fifo_destroy_context(struct nouveau_channel *chan)
74{
75	struct drm_device *dev = chan->dev;
76
77	nv_wr32(dev, NV04_PFIFO_MODE,
78		nv_rd32(dev, NV04_PFIFO_MODE) & ~(1 << chan->id));
79
80	if (chan->ramfc)
81		nouveau_gpuobj_ref_del(dev, &chan->ramfc);
82}
83
84static void
85nv40_fifo_do_load_context(struct drm_device *dev, int chid)
86{
87	struct drm_nouveau_private *dev_priv = dev->dev_private;
88	uint32_t fc = NV40_RAMFC(chid), tmp, tmp2;
89
90	nv_wr32(dev, NV04_PFIFO_CACHE1_DMA_PUT, nv_ri32(dev, fc + 0));
91	nv_wr32(dev, NV04_PFIFO_CACHE1_DMA_GET, nv_ri32(dev, fc + 4));
92	nv_wr32(dev, NV10_PFIFO_CACHE1_REF_CNT, nv_ri32(dev, fc + 8));
93	nv_wr32(dev, NV04_PFIFO_CACHE1_DMA_INSTANCE, nv_ri32(dev, fc + 12));
94	nv_wr32(dev, NV04_PFIFO_CACHE1_DMA_DCOUNT, nv_ri32(dev, fc + 16));
95	nv_wr32(dev, NV04_PFIFO_CACHE1_DMA_STATE, nv_ri32(dev, fc + 20));
96
97	/* No idea what 0x2058 is.. */
98	tmp   = nv_ri32(dev, fc + 24);
99	tmp2  = nv_rd32(dev, 0x2058) & 0xFFF;
100	tmp2 |= (tmp & 0x30000000);
101	nv_wr32(dev, 0x2058, tmp2);
102	tmp  &= ~0x30000000;
103	nv_wr32(dev, NV04_PFIFO_CACHE1_DMA_FETCH, tmp);
104
105	nv_wr32(dev, NV04_PFIFO_CACHE1_ENGINE, nv_ri32(dev, fc + 28));
106	nv_wr32(dev, NV04_PFIFO_CACHE1_PULL1, nv_ri32(dev, fc + 32));
107	nv_wr32(dev, NV10_PFIFO_CACHE1_ACQUIRE_VALUE, nv_ri32(dev, fc + 36));
108	tmp = nv_ri32(dev, fc + 40);
109	nv_wr32(dev, NV10_PFIFO_CACHE1_ACQUIRE_TIMESTAMP, tmp);
110	nv_wr32(dev, NV10_PFIFO_CACHE1_ACQUIRE_TIMEOUT, nv_ri32(dev, fc + 44));
111	nv_wr32(dev, NV10_PFIFO_CACHE1_SEMAPHORE, nv_ri32(dev, fc + 48));
112	nv_wr32(dev, NV10_PFIFO_CACHE1_DMA_SUBROUTINE, nv_ri32(dev, fc + 52));
113	nv_wr32(dev, NV40_PFIFO_GRCTX_INSTANCE, nv_ri32(dev, fc + 56));
114
115	/* Don't clobber the TIMEOUT_ENABLED flag when restoring from RAMFC */
116	tmp  = nv_rd32(dev, NV04_PFIFO_DMA_TIMESLICE) & ~0x1FFFF;
117	tmp |= nv_ri32(dev, fc + 60) & 0x1FFFF;
118	nv_wr32(dev, NV04_PFIFO_DMA_TIMESLICE, tmp);
119
120	nv_wr32(dev, 0x32e4, nv_ri32(dev, fc + 64));
121	/* NVIDIA does this next line twice... */
122	nv_wr32(dev, 0x32e8, nv_ri32(dev, fc + 68));
123	nv_wr32(dev, 0x2088, nv_ri32(dev, fc + 76));
124	nv_wr32(dev, 0x3300, nv_ri32(dev, fc + 80));
125
126	nv_wr32(dev, NV03_PFIFO_CACHE1_GET, 0);
127	nv_wr32(dev, NV03_PFIFO_CACHE1_PUT, 0);
128}
129
130int
131nv40_fifo_load_context(struct nouveau_channel *chan)
132{
133	struct drm_device *dev = chan->dev;
134	uint32_t tmp;
135
136	nv40_fifo_do_load_context(dev, chan->id);
137
138	/* Set channel active, and in DMA mode */
139	nv_wr32(dev, NV03_PFIFO_CACHE1_PUSH1,
140		     NV40_PFIFO_CACHE1_PUSH1_DMA | chan->id);
141	nv_wr32(dev, NV04_PFIFO_CACHE1_DMA_PUSH, 1);
142
143	/* Reset DMA_CTL_AT_INFO to INVALID */
144	tmp = nv_rd32(dev, NV04_PFIFO_CACHE1_DMA_CTL) & ~(1 << 31);
145	nv_wr32(dev, NV04_PFIFO_CACHE1_DMA_CTL, tmp);
146
147	return 0;
148}
149
150int
151nv40_fifo_unload_context(struct drm_device *dev)
152{
153	struct drm_nouveau_private *dev_priv = dev->dev_private;
154	struct nouveau_fifo_engine *pfifo = &dev_priv->engine.fifo;
155	uint32_t fc, tmp;
156	int chid;
157
158	chid = pfifo->channel_id(dev);
159	if (chid < 0 || chid >= dev_priv->engine.fifo.channels)
160		return 0;
161	fc = NV40_RAMFC(chid);
162
163	nv_wi32(dev, fc + 0, nv_rd32(dev, NV04_PFIFO_CACHE1_DMA_PUT));
164	nv_wi32(dev, fc + 4, nv_rd32(dev, NV04_PFIFO_CACHE1_DMA_GET));
165	nv_wi32(dev, fc + 8, nv_rd32(dev, NV10_PFIFO_CACHE1_REF_CNT));
166	nv_wi32(dev, fc + 12, nv_rd32(dev, NV04_PFIFO_CACHE1_DMA_INSTANCE));
167	nv_wi32(dev, fc + 16, nv_rd32(dev, NV04_PFIFO_CACHE1_DMA_DCOUNT));
168	nv_wi32(dev, fc + 20, nv_rd32(dev, NV04_PFIFO_CACHE1_DMA_STATE));
169	tmp  = nv_rd32(dev, NV04_PFIFO_CACHE1_DMA_FETCH);
170	tmp |= nv_rd32(dev, 0x2058) & 0x30000000;
171	nv_wi32(dev, fc + 24, tmp);
172	nv_wi32(dev, fc + 28, nv_rd32(dev, NV04_PFIFO_CACHE1_ENGINE));
173	nv_wi32(dev, fc + 32, nv_rd32(dev, NV04_PFIFO_CACHE1_PULL1));
174	nv_wi32(dev, fc + 36, nv_rd32(dev, NV10_PFIFO_CACHE1_ACQUIRE_VALUE));
175	tmp = nv_rd32(dev, NV10_PFIFO_CACHE1_ACQUIRE_TIMESTAMP);
176	nv_wi32(dev, fc + 40, tmp);
177	nv_wi32(dev, fc + 44, nv_rd32(dev, NV10_PFIFO_CACHE1_ACQUIRE_TIMEOUT));
178	nv_wi32(dev, fc + 48, nv_rd32(dev, NV10_PFIFO_CACHE1_SEMAPHORE));
179	/* NVIDIA read 0x3228 first, then write DMA_GET here.. maybe something
180	 * more involved depending on the value of 0x3228?
181	 */
182	nv_wi32(dev, fc + 52, nv_rd32(dev, NV04_PFIFO_CACHE1_DMA_GET));
183	nv_wi32(dev, fc + 56, nv_rd32(dev, NV40_PFIFO_GRCTX_INSTANCE));
184	nv_wi32(dev, fc + 60, nv_rd32(dev, NV04_PFIFO_DMA_TIMESLICE) & 0x1ffff);
185	/* No idea what the below is for exactly, ripped from a mmio-trace */
186	nv_wi32(dev, fc + 64, nv_rd32(dev, NV40_PFIFO_UNK32E4));
187	/* NVIDIA do this next line twice.. bug? */
188	nv_wi32(dev, fc + 68, nv_rd32(dev, 0x32e8));
189	nv_wi32(dev, fc + 76, nv_rd32(dev, 0x2088));
190	nv_wi32(dev, fc + 80, nv_rd32(dev, 0x3300));
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;
202
203	nv_wr32(dev, NV03_PMC_ENABLE,
204		nv_rd32(dev, NV03_PMC_ENABLE) & ~NV_PMC_ENABLE_PFIFO);
205	nv_wr32(dev, NV03_PMC_ENABLE,
206		nv_rd32(dev, NV03_PMC_ENABLE) |  NV_PMC_ENABLE_PFIFO);
207
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);
227	nv_wr32(dev, 0x002500, 0x00000000);
228	nv_wr32(dev, 0x003200, 0x00000000);
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
238	nv_wr32(dev, NV03_PFIFO_RAMHT, (0x03 << 24) /* search 128 */ |
239				       ((dev_priv->ramht_bits - 9) << 16) |
240				       (dev_priv->ramht_offset >> 8));
241	nv_wr32(dev, NV03_PFIFO_RAMRO, dev_priv->ramro_offset>>8);
242
243	switch (dev_priv->chipset) {
244	case 0x47:
245	case 0x49:
246	case 0x4b:
247		nv_wr32(dev, 0x2230, 1);
248		break;
249	default:
250		break;
251	}
252
253	switch (dev_priv->chipset) {
254	case 0x40:
255	case 0x41:
256	case 0x42:
257	case 0x43:
258	case 0x45:
259	case 0x47:
260	case 0x48:
261	case 0x49:
262	case 0x4b:
263		nv_wr32(dev, NV40_PFIFO_RAMFC, 0x30002);
264		break;
265	default:
266		nv_wr32(dev, 0x2230, 0);
267		nv_wr32(dev, NV40_PFIFO_RAMFC,
268			((dev_priv->vram_size - 512 * 1024 +
269			  dev_priv->ramfc_offset) >> 16) | (3 << 16));
270		break;
271	}
272}
273
274static void
275nv40_fifo_init_intr(struct drm_device *dev)
276{
277	nv_wr32(dev, 0x002100, 0xffffffff);
278	nv_wr32(dev, 0x002140, 0xffffffff);
279}
280
281int
282nv40_fifo_init(struct drm_device *dev)
283{
284	struct drm_nouveau_private *dev_priv = dev->dev_private;
285	struct nouveau_fifo_engine *pfifo = &dev_priv->engine.fifo;
286	int i;
287
288	nv40_fifo_init_reset(dev);
289	nv40_fifo_init_ramxx(dev);
290
291	nv40_fifo_do_load_context(dev, pfifo->channels - 1);
292	nv_wr32(dev, NV03_PFIFO_CACHE1_PUSH1, pfifo->channels - 1);
293
294	nv40_fifo_init_intr(dev);
295	pfifo->enable(dev);
296	pfifo->reassign(dev, true);
297
298	for (i = 0; i < dev_priv->engine.fifo.channels; i++) {
299		if (dev_priv->fifos[i]) {
300			uint32_t mode = nv_rd32(dev, NV04_PFIFO_MODE);
301			nv_wr32(dev, NV04_PFIFO_MODE, mode | (1 << i));
302		}
303	}
304
305	return 0;
306}
307