blob: c11a2fa43c7f0e2cfe213112189db049e3b49189 [file] [log] [blame]
Ben Skeggs6ee73862009-12-11 19:24:15 +10001/*
2 * Copyright (C) 2008 Maarten Maathuis.
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 "nv50_display.h"
28#include "nouveau_crtc.h"
29#include "nouveau_encoder.h"
30#include "nouveau_connector.h"
31#include "nouveau_fb.h"
Dave Airlie4abe3522010-03-30 05:34:18 +000032#include "nouveau_fbcon.h"
Ben Skeggsa8eaebc2010-09-01 15:24:31 +100033#include "nouveau_ramht.h"
Ben Skeggs6ee73862009-12-11 19:24:15 +100034#include "drm_crtc_helper.h"
35
Ben Skeggs8597a1b2010-09-06 11:39:25 +100036static inline int
37nv50_sor_nr(struct drm_device *dev)
38{
39 struct drm_nouveau_private *dev_priv = dev->dev_private;
40
41 if (dev_priv->chipset < 0x90 ||
42 dev_priv->chipset == 0x92 ||
43 dev_priv->chipset == 0xa0)
44 return 2;
45
46 return 4;
47}
48
Ben Skeggs6ee73862009-12-11 19:24:15 +100049static void
50nv50_evo_channel_del(struct nouveau_channel **pchan)
51{
52 struct nouveau_channel *chan = *pchan;
53
54 if (!chan)
55 return;
56 *pchan = NULL;
57
58 nouveau_gpuobj_channel_takedown(chan);
Ben Skeggs9d59e8a2010-08-27 13:04:41 +100059 nouveau_bo_unmap(chan->pushbuf_bo);
Ben Skeggs6ee73862009-12-11 19:24:15 +100060 nouveau_bo_ref(NULL, &chan->pushbuf_bo);
61
62 if (chan->user)
63 iounmap(chan->user);
64
65 kfree(chan);
66}
67
68static int
69nv50_evo_dmaobj_new(struct nouveau_channel *evo, uint32_t class, uint32_t name,
70 uint32_t tile_flags, uint32_t magic_flags,
71 uint32_t offset, uint32_t limit)
72{
73 struct drm_nouveau_private *dev_priv = evo->dev->dev_private;
74 struct drm_device *dev = evo->dev;
75 struct nouveau_gpuobj *obj = NULL;
76 int ret;
77
78 ret = nouveau_gpuobj_new(dev, evo, 6*4, 32, 0, &obj);
79 if (ret)
80 return ret;
81 obj->engine = NVOBJ_ENGINE_DISPLAY;
82
Ben Skeggsb3beb162010-09-01 15:24:29 +100083 nv_wo32(obj, 0, (tile_flags << 22) | (magic_flags << 16) | class);
84 nv_wo32(obj, 4, limit);
85 nv_wo32(obj, 8, offset);
86 nv_wo32(obj, 12, 0x00000000);
87 nv_wo32(obj, 16, 0x00000000);
Ben Skeggs0165d152010-08-04 17:24:57 +100088 if (dev_priv->card_type < NV_C0)
Ben Skeggsb3beb162010-09-01 15:24:29 +100089 nv_wo32(obj, 20, 0x00010000);
Ben Skeggs0165d152010-08-04 17:24:57 +100090 else
Ben Skeggsb3beb162010-09-01 15:24:29 +100091 nv_wo32(obj, 20, 0x00020000);
Ben Skeggsf56cb862010-07-08 11:29:10 +100092 dev_priv->engine.instmem.flush(dev);
Ben Skeggs6ee73862009-12-11 19:24:15 +100093
Ben Skeggsa8eaebc2010-09-01 15:24:31 +100094 ret = nouveau_ramht_insert(evo, name, obj);
95 nouveau_gpuobj_ref(NULL, &obj);
96 if (ret) {
97 return ret;
98 }
99
Ben Skeggs6ee73862009-12-11 19:24:15 +1000100 return 0;
101}
102
103static int
104nv50_evo_channel_new(struct drm_device *dev, struct nouveau_channel **pchan)
105{
106 struct drm_nouveau_private *dev_priv = dev->dev_private;
Ben Skeggsa8eaebc2010-09-01 15:24:31 +1000107 struct nouveau_gpuobj *ramht = NULL;
Ben Skeggs6ee73862009-12-11 19:24:15 +1000108 struct nouveau_channel *chan;
109 int ret;
110
111 chan = kzalloc(sizeof(struct nouveau_channel), GFP_KERNEL);
112 if (!chan)
113 return -ENOMEM;
114 *pchan = chan;
115
116 chan->id = -1;
117 chan->dev = dev;
118 chan->user_get = 4;
119 chan->user_put = 0;
120
Ben Skeggsa8eaebc2010-09-01 15:24:31 +1000121 ret = nouveau_gpuobj_new(dev, NULL, 32768, 0x1000,
122 NVOBJ_FLAG_ZERO_ALLOC, &chan->ramin);
Ben Skeggs6ee73862009-12-11 19:24:15 +1000123 if (ret) {
124 NV_ERROR(dev, "Error allocating EVO channel memory: %d\n", ret);
125 nv50_evo_channel_del(pchan);
126 return ret;
127 }
128
Ben Skeggsde3a6c02010-09-01 15:24:30 +1000129 ret = drm_mm_init(&chan->ramin_heap, 0, 32768);
Ben Skeggs6ee73862009-12-11 19:24:15 +1000130 if (ret) {
131 NV_ERROR(dev, "Error initialising EVO PRAMIN heap: %d\n", ret);
132 nv50_evo_channel_del(pchan);
133 return ret;
134 }
135
Ben Skeggsa8eaebc2010-09-01 15:24:31 +1000136 ret = nouveau_gpuobj_new(dev, chan, 4096, 16, 0, &ramht);
Ben Skeggs6ee73862009-12-11 19:24:15 +1000137 if (ret) {
138 NV_ERROR(dev, "Unable to allocate EVO RAMHT: %d\n", ret);
139 nv50_evo_channel_del(pchan);
140 return ret;
141 }
142
Ben Skeggsa8eaebc2010-09-01 15:24:31 +1000143 ret = nouveau_ramht_new(dev, ramht, &chan->ramht);
144 nouveau_gpuobj_ref(NULL, &ramht);
145 if (ret) {
146 nv50_evo_channel_del(pchan);
147 return ret;
148 }
149
Ben Skeggs6ee73862009-12-11 19:24:15 +1000150 if (dev_priv->chipset != 0x50) {
151 ret = nv50_evo_dmaobj_new(chan, 0x3d, NvEvoFB16, 0x70, 0x19,
152 0, 0xffffffff);
153 if (ret) {
154 nv50_evo_channel_del(pchan);
155 return ret;
156 }
157
158
159 ret = nv50_evo_dmaobj_new(chan, 0x3d, NvEvoFB32, 0x7a, 0x19,
160 0, 0xffffffff);
161 if (ret) {
162 nv50_evo_channel_del(pchan);
163 return ret;
164 }
165 }
166
167 ret = nv50_evo_dmaobj_new(chan, 0x3d, NvEvoVRAM, 0, 0x19,
Ben Skeggsa76fb4e2010-03-18 09:45:20 +1000168 0, dev_priv->vram_size);
Ben Skeggs6ee73862009-12-11 19:24:15 +1000169 if (ret) {
170 nv50_evo_channel_del(pchan);
171 return ret;
172 }
173
174 ret = nouveau_bo_new(dev, NULL, 4096, 0, TTM_PL_FLAG_VRAM, 0, 0,
175 false, true, &chan->pushbuf_bo);
176 if (ret == 0)
177 ret = nouveau_bo_pin(chan->pushbuf_bo, TTM_PL_FLAG_VRAM);
178 if (ret) {
179 NV_ERROR(dev, "Error creating EVO DMA push buffer: %d\n", ret);
180 nv50_evo_channel_del(pchan);
181 return ret;
182 }
183
184 ret = nouveau_bo_map(chan->pushbuf_bo);
185 if (ret) {
186 NV_ERROR(dev, "Error mapping EVO DMA push buffer: %d\n", ret);
187 nv50_evo_channel_del(pchan);
188 return ret;
189 }
190
191 chan->user = ioremap(pci_resource_start(dev->pdev, 0) +
192 NV50_PDISPLAY_USER(0), PAGE_SIZE);
193 if (!chan->user) {
194 NV_ERROR(dev, "Error mapping EVO control regs.\n");
195 nv50_evo_channel_del(pchan);
196 return -ENOMEM;
197 }
198
199 return 0;
200}
201
202int
Francisco Jerezc88c2e02010-07-24 17:37:33 +0200203nv50_display_early_init(struct drm_device *dev)
204{
205 return 0;
206}
207
208void
209nv50_display_late_takedown(struct drm_device *dev)
210{
211}
212
213int
Ben Skeggs6ee73862009-12-11 19:24:15 +1000214nv50_display_init(struct drm_device *dev)
215{
216 struct drm_nouveau_private *dev_priv = dev->dev_private;
217 struct nouveau_timer_engine *ptimer = &dev_priv->engine.timer;
Ben Skeggsee2e0132010-07-26 09:28:25 +1000218 struct nouveau_gpio_engine *pgpio = &dev_priv->engine.gpio;
Ben Skeggs6ee73862009-12-11 19:24:15 +1000219 struct nouveau_channel *evo = dev_priv->evo;
220 struct drm_connector *connector;
Ben Skeggsd0875ed2010-07-23 11:31:08 +1000221 uint32_t val, ram_amount;
Ben Skeggs6ee73862009-12-11 19:24:15 +1000222 uint64_t start;
223 int ret, i;
224
Maarten Maathuisef2bb502009-12-13 16:53:12 +0100225 NV_DEBUG_KMS(dev, "\n");
Ben Skeggs6ee73862009-12-11 19:24:15 +1000226
227 nv_wr32(dev, 0x00610184, nv_rd32(dev, 0x00614004));
228 /*
229 * I think the 0x006101XX range is some kind of main control area
230 * that enables things.
231 */
232 /* CRTC? */
233 for (i = 0; i < 2; i++) {
234 val = nv_rd32(dev, 0x00616100 + (i * 0x800));
235 nv_wr32(dev, 0x00610190 + (i * 0x10), val);
236 val = nv_rd32(dev, 0x00616104 + (i * 0x800));
237 nv_wr32(dev, 0x00610194 + (i * 0x10), val);
238 val = nv_rd32(dev, 0x00616108 + (i * 0x800));
239 nv_wr32(dev, 0x00610198 + (i * 0x10), val);
240 val = nv_rd32(dev, 0x0061610c + (i * 0x800));
241 nv_wr32(dev, 0x0061019c + (i * 0x10), val);
242 }
243 /* DAC */
244 for (i = 0; i < 3; i++) {
245 val = nv_rd32(dev, 0x0061a000 + (i * 0x800));
246 nv_wr32(dev, 0x006101d0 + (i * 0x04), val);
247 }
248 /* SOR */
Ben Skeggs8597a1b2010-09-06 11:39:25 +1000249 for (i = 0; i < nv50_sor_nr(dev); i++) {
Ben Skeggs6ee73862009-12-11 19:24:15 +1000250 val = nv_rd32(dev, 0x0061c000 + (i * 0x800));
251 nv_wr32(dev, 0x006101e0 + (i * 0x04), val);
252 }
Ben Skeggs8597a1b2010-09-06 11:39:25 +1000253 /* EXT */
Ben Skeggs6ee73862009-12-11 19:24:15 +1000254 for (i = 0; i < 3; i++) {
255 val = nv_rd32(dev, 0x0061e000 + (i * 0x800));
256 nv_wr32(dev, 0x006101f0 + (i * 0x04), val);
257 }
258
259 for (i = 0; i < 3; i++) {
260 nv_wr32(dev, NV50_PDISPLAY_DAC_DPMS_CTRL(i), 0x00550000 |
261 NV50_PDISPLAY_DAC_DPMS_CTRL_PENDING);
262 nv_wr32(dev, NV50_PDISPLAY_DAC_CLK_CTRL1(i), 0x00000001);
263 }
264
265 /* This used to be in crtc unblank, but seems out of place there. */
266 nv_wr32(dev, NV50_PDISPLAY_UNK_380, 0);
267 /* RAM is clamped to 256 MiB. */
Ben Skeggsa76fb4e2010-03-18 09:45:20 +1000268 ram_amount = dev_priv->vram_size;
Maarten Maathuisef2bb502009-12-13 16:53:12 +0100269 NV_DEBUG_KMS(dev, "ram_amount %d\n", ram_amount);
Ben Skeggs6ee73862009-12-11 19:24:15 +1000270 if (ram_amount > 256*1024*1024)
271 ram_amount = 256*1024*1024;
272 nv_wr32(dev, NV50_PDISPLAY_RAM_AMOUNT, ram_amount - 1);
273 nv_wr32(dev, NV50_PDISPLAY_UNK_388, 0x150000);
274 nv_wr32(dev, NV50_PDISPLAY_UNK_38C, 0);
275
276 /* The precise purpose is unknown, i suspect it has something to do
277 * with text mode.
278 */
279 if (nv_rd32(dev, NV50_PDISPLAY_INTR_1) & 0x100) {
280 nv_wr32(dev, NV50_PDISPLAY_INTR_1, 0x100);
281 nv_wr32(dev, 0x006194e8, nv_rd32(dev, 0x006194e8) & ~1);
282 if (!nv_wait(0x006194e8, 2, 0)) {
283 NV_ERROR(dev, "timeout: (0x6194e8 & 2) != 0\n");
284 NV_ERROR(dev, "0x6194e8 = 0x%08x\n",
285 nv_rd32(dev, 0x6194e8));
286 return -EBUSY;
287 }
288 }
289
290 /* taken from nv bug #12637, attempts to un-wedge the hw if it's
291 * stuck in some unspecified state
292 */
293 start = ptimer->read(dev);
294 nv_wr32(dev, NV50_PDISPLAY_CHANNEL_STAT(0), 0x2b00);
295 while ((val = nv_rd32(dev, NV50_PDISPLAY_CHANNEL_STAT(0))) & 0x1e0000) {
296 if ((val & 0x9f0000) == 0x20000)
297 nv_wr32(dev, NV50_PDISPLAY_CHANNEL_STAT(0),
298 val | 0x800000);
299
300 if ((val & 0x3f0000) == 0x30000)
301 nv_wr32(dev, NV50_PDISPLAY_CHANNEL_STAT(0),
302 val | 0x200000);
303
304 if (ptimer->read(dev) - start > 1000000000ULL) {
305 NV_ERROR(dev, "timeout: (0x610200 & 0x1e0000) != 0\n");
306 NV_ERROR(dev, "0x610200 = 0x%08x\n", val);
307 return -EBUSY;
308 }
309 }
310
311 nv_wr32(dev, NV50_PDISPLAY_CTRL_STATE, NV50_PDISPLAY_CTRL_STATE_ENABLE);
312 nv_wr32(dev, NV50_PDISPLAY_CHANNEL_STAT(0), 0x1000b03);
313 if (!nv_wait(NV50_PDISPLAY_CHANNEL_STAT(0), 0x40000000, 0x40000000)) {
314 NV_ERROR(dev, "timeout: (0x610200 & 0x40000000) == 0x40000000\n");
315 NV_ERROR(dev, "0x610200 = 0x%08x\n",
316 nv_rd32(dev, NV50_PDISPLAY_CHANNEL_STAT(0)));
317 return -EBUSY;
318 }
319
320 for (i = 0; i < 2; i++) {
321 nv_wr32(dev, NV50_PDISPLAY_CURSOR_CURSOR_CTRL2(i), 0x2000);
322 if (!nv_wait(NV50_PDISPLAY_CURSOR_CURSOR_CTRL2(i),
323 NV50_PDISPLAY_CURSOR_CURSOR_CTRL2_STATUS, 0)) {
324 NV_ERROR(dev, "timeout: CURSOR_CTRL2_STATUS == 0\n");
325 NV_ERROR(dev, "CURSOR_CTRL2 = 0x%08x\n",
326 nv_rd32(dev, NV50_PDISPLAY_CURSOR_CURSOR_CTRL2(i)));
327 return -EBUSY;
328 }
329
330 nv_wr32(dev, NV50_PDISPLAY_CURSOR_CURSOR_CTRL2(i),
331 NV50_PDISPLAY_CURSOR_CURSOR_CTRL2_ON);
332 if (!nv_wait(NV50_PDISPLAY_CURSOR_CURSOR_CTRL2(i),
333 NV50_PDISPLAY_CURSOR_CURSOR_CTRL2_STATUS,
334 NV50_PDISPLAY_CURSOR_CURSOR_CTRL2_STATUS_ACTIVE)) {
335 NV_ERROR(dev, "timeout: "
336 "CURSOR_CTRL2_STATUS_ACTIVE(%d)\n", i);
337 NV_ERROR(dev, "CURSOR_CTRL2(%d) = 0x%08x\n", i,
338 nv_rd32(dev, NV50_PDISPLAY_CURSOR_CURSOR_CTRL2(i)));
339 return -EBUSY;
340 }
341 }
342
Ben Skeggsa8eaebc2010-09-01 15:24:31 +1000343 nv_wr32(dev, NV50_PDISPLAY_OBJECTS, (evo->ramin->vinst >> 8) | 9);
Ben Skeggs6ee73862009-12-11 19:24:15 +1000344
345 /* initialise fifo */
346 nv_wr32(dev, NV50_PDISPLAY_CHANNEL_DMA_CB(0),
347 ((evo->pushbuf_bo->bo.mem.mm_node->start << PAGE_SHIFT) >> 8) |
348 NV50_PDISPLAY_CHANNEL_DMA_CB_LOCATION_VRAM |
349 NV50_PDISPLAY_CHANNEL_DMA_CB_VALID);
350 nv_wr32(dev, NV50_PDISPLAY_CHANNEL_UNK2(0), 0x00010000);
351 nv_wr32(dev, NV50_PDISPLAY_CHANNEL_UNK3(0), 0x00000002);
352 if (!nv_wait(0x610200, 0x80000000, 0x00000000)) {
353 NV_ERROR(dev, "timeout: (0x610200 & 0x80000000) == 0\n");
354 NV_ERROR(dev, "0x610200 = 0x%08x\n", nv_rd32(dev, 0x610200));
355 return -EBUSY;
356 }
357 nv_wr32(dev, NV50_PDISPLAY_CHANNEL_STAT(0),
358 (nv_rd32(dev, NV50_PDISPLAY_CHANNEL_STAT(0)) & ~0x00000003) |
359 NV50_PDISPLAY_CHANNEL_STAT_DMA_ENABLED);
360 nv_wr32(dev, NV50_PDISPLAY_USER_PUT(0), 0);
361 nv_wr32(dev, NV50_PDISPLAY_CHANNEL_STAT(0), 0x01000003 |
362 NV50_PDISPLAY_CHANNEL_STAT_DMA_ENABLED);
363 nv_wr32(dev, 0x610300, nv_rd32(dev, 0x610300) & ~1);
364
365 evo->dma.max = (4096/4) - 2;
366 evo->dma.put = 0;
367 evo->dma.cur = evo->dma.put;
368 evo->dma.free = evo->dma.max - evo->dma.cur;
369
370 ret = RING_SPACE(evo, NOUVEAU_DMA_SKIPS);
371 if (ret)
372 return ret;
373
374 for (i = 0; i < NOUVEAU_DMA_SKIPS; i++)
375 OUT_RING(evo, 0);
376
377 ret = RING_SPACE(evo, 11);
378 if (ret)
379 return ret;
380 BEGIN_RING(evo, 0, NV50_EVO_UNK84, 2);
381 OUT_RING(evo, NV50_EVO_UNK84_NOTIFY_DISABLED);
382 OUT_RING(evo, NV50_EVO_DMA_NOTIFY_HANDLE_NONE);
383 BEGIN_RING(evo, 0, NV50_EVO_CRTC(0, FB_DMA), 1);
384 OUT_RING(evo, NV50_EVO_CRTC_FB_DMA_HANDLE_NONE);
385 BEGIN_RING(evo, 0, NV50_EVO_CRTC(0, UNK0800), 1);
386 OUT_RING(evo, 0);
387 BEGIN_RING(evo, 0, NV50_EVO_CRTC(0, DISPLAY_START), 1);
388 OUT_RING(evo, 0);
389 BEGIN_RING(evo, 0, NV50_EVO_CRTC(0, UNK082C), 1);
390 OUT_RING(evo, 0);
391 FIRE_RING(evo);
392 if (!nv_wait(0x640004, 0xffffffff, evo->dma.put << 2))
393 NV_ERROR(dev, "evo pushbuf stalled\n");
394
395 /* enable clock change interrupts. */
396 nv_wr32(dev, 0x610028, 0x00010001);
397 nv_wr32(dev, NV50_PDISPLAY_INTR_EN, (NV50_PDISPLAY_INTR_EN_CLK_UNK10 |
398 NV50_PDISPLAY_INTR_EN_CLK_UNK20 |
399 NV50_PDISPLAY_INTR_EN_CLK_UNK40));
400
401 /* enable hotplug interrupts */
Ben Skeggs6ee73862009-12-11 19:24:15 +1000402 list_for_each_entry(connector, &dev->mode_config.connector_list, head) {
403 struct nouveau_connector *conn = nouveau_connector(connector);
Ben Skeggs6ee73862009-12-11 19:24:15 +1000404
Ben Skeggs0ccb3a72010-07-26 11:35:37 +1000405 if (conn->dcb->gpio_tag == 0xff)
406 continue;
407
Ben Skeggsee2e0132010-07-26 09:28:25 +1000408 pgpio->irq_enable(dev, conn->dcb->gpio_tag, true);
Ben Skeggs6ee73862009-12-11 19:24:15 +1000409 }
410
411 return 0;
412}
413
414static int nv50_display_disable(struct drm_device *dev)
415{
416 struct drm_nouveau_private *dev_priv = dev->dev_private;
417 struct drm_crtc *drm_crtc;
418 int ret, i;
419
Maarten Maathuisef2bb502009-12-13 16:53:12 +0100420 NV_DEBUG_KMS(dev, "\n");
Ben Skeggs6ee73862009-12-11 19:24:15 +1000421
422 list_for_each_entry(drm_crtc, &dev->mode_config.crtc_list, head) {
423 struct nouveau_crtc *crtc = nouveau_crtc(drm_crtc);
424
425 nv50_crtc_blank(crtc, true);
426 }
427
428 ret = RING_SPACE(dev_priv->evo, 2);
429 if (ret == 0) {
430 BEGIN_RING(dev_priv->evo, 0, NV50_EVO_UPDATE, 1);
431 OUT_RING(dev_priv->evo, 0);
432 }
433 FIRE_RING(dev_priv->evo);
434
435 /* Almost like ack'ing a vblank interrupt, maybe in the spirit of
436 * cleaning up?
437 */
438 list_for_each_entry(drm_crtc, &dev->mode_config.crtc_list, head) {
439 struct nouveau_crtc *crtc = nouveau_crtc(drm_crtc);
440 uint32_t mask = NV50_PDISPLAY_INTR_1_VBLANK_CRTC_(crtc->index);
441
442 if (!crtc->base.enabled)
443 continue;
444
445 nv_wr32(dev, NV50_PDISPLAY_INTR_1, mask);
446 if (!nv_wait(NV50_PDISPLAY_INTR_1, mask, mask)) {
447 NV_ERROR(dev, "timeout: (0x610024 & 0x%08x) == "
448 "0x%08x\n", mask, mask);
449 NV_ERROR(dev, "0x610024 = 0x%08x\n",
450 nv_rd32(dev, NV50_PDISPLAY_INTR_1));
451 }
452 }
453
454 nv_wr32(dev, NV50_PDISPLAY_CHANNEL_STAT(0), 0);
455 nv_wr32(dev, NV50_PDISPLAY_CTRL_STATE, 0);
456 if (!nv_wait(NV50_PDISPLAY_CHANNEL_STAT(0), 0x1e0000, 0)) {
457 NV_ERROR(dev, "timeout: (0x610200 & 0x1e0000) == 0\n");
458 NV_ERROR(dev, "0x610200 = 0x%08x\n",
459 nv_rd32(dev, NV50_PDISPLAY_CHANNEL_STAT(0)));
460 }
461
462 for (i = 0; i < 3; i++) {
463 if (!nv_wait(NV50_PDISPLAY_SOR_DPMS_STATE(i),
464 NV50_PDISPLAY_SOR_DPMS_STATE_WAIT, 0)) {
465 NV_ERROR(dev, "timeout: SOR_DPMS_STATE_WAIT(%d) == 0\n", i);
466 NV_ERROR(dev, "SOR_DPMS_STATE(%d) = 0x%08x\n", i,
467 nv_rd32(dev, NV50_PDISPLAY_SOR_DPMS_STATE(i)));
468 }
469 }
470
471 /* disable interrupts. */
472 nv_wr32(dev, NV50_PDISPLAY_INTR_EN, 0x00000000);
473
474 /* disable hotplug interrupts */
475 nv_wr32(dev, 0xe054, 0xffffffff);
476 nv_wr32(dev, 0xe050, 0x00000000);
477 if (dev_priv->chipset >= 0x90) {
478 nv_wr32(dev, 0xe074, 0xffffffff);
479 nv_wr32(dev, 0xe070, 0x00000000);
480 }
481 return 0;
482}
483
484int nv50_display_create(struct drm_device *dev)
485{
486 struct drm_nouveau_private *dev_priv = dev->dev_private;
Ben Skeggs04a39c52010-02-24 10:03:05 +1000487 struct dcb_table *dcb = &dev_priv->vbios.dcb;
Ben Skeggs8f1a6082010-06-28 14:35:50 +1000488 struct drm_connector *connector, *ct;
Ben Skeggs6ee73862009-12-11 19:24:15 +1000489 int ret, i;
490
Maarten Maathuisef2bb502009-12-13 16:53:12 +0100491 NV_DEBUG_KMS(dev, "\n");
Ben Skeggs6ee73862009-12-11 19:24:15 +1000492
493 /* init basic kernel modesetting */
494 drm_mode_config_init(dev);
495
496 /* Initialise some optional connector properties. */
497 drm_mode_create_scaling_mode_property(dev);
498 drm_mode_create_dithering_property(dev);
499
500 dev->mode_config.min_width = 0;
501 dev->mode_config.min_height = 0;
502
503 dev->mode_config.funcs = (void *)&nouveau_mode_config_funcs;
504
505 dev->mode_config.max_width = 8192;
506 dev->mode_config.max_height = 8192;
507
508 dev->mode_config.fb_base = dev_priv->fb_phys;
509
510 /* Create EVO channel */
511 ret = nv50_evo_channel_new(dev, &dev_priv->evo);
512 if (ret) {
513 NV_ERROR(dev, "Error creating EVO channel: %d\n", ret);
514 return ret;
515 }
516
517 /* Create CRTC objects */
518 for (i = 0; i < 2; i++)
519 nv50_crtc_create(dev, i);
520
521 /* We setup the encoders from the BIOS table */
522 for (i = 0 ; i < dcb->entries; i++) {
523 struct dcb_entry *entry = &dcb->entry[i];
524
525 if (entry->location != DCB_LOC_ON_CHIP) {
526 NV_WARN(dev, "Off-chip encoder %d/%d unsupported\n",
527 entry->type, ffs(entry->or) - 1);
528 continue;
529 }
530
Ben Skeggs8f1a6082010-06-28 14:35:50 +1000531 connector = nouveau_connector_create(dev, entry->connector);
532 if (IS_ERR(connector))
533 continue;
534
Ben Skeggs6ee73862009-12-11 19:24:15 +1000535 switch (entry->type) {
536 case OUTPUT_TMDS:
537 case OUTPUT_LVDS:
538 case OUTPUT_DP:
Ben Skeggs8f1a6082010-06-28 14:35:50 +1000539 nv50_sor_create(connector, entry);
Ben Skeggs6ee73862009-12-11 19:24:15 +1000540 break;
541 case OUTPUT_ANALOG:
Ben Skeggs8f1a6082010-06-28 14:35:50 +1000542 nv50_dac_create(connector, entry);
Ben Skeggs6ee73862009-12-11 19:24:15 +1000543 break;
544 default:
545 NV_WARN(dev, "DCB encoder %d unknown\n", entry->type);
546 continue;
547 }
Ben Skeggs6ee73862009-12-11 19:24:15 +1000548 }
549
Ben Skeggs8f1a6082010-06-28 14:35:50 +1000550 list_for_each_entry_safe(connector, ct,
551 &dev->mode_config.connector_list, head) {
552 if (!connector->encoder_ids[0]) {
553 NV_WARN(dev, "%s has no encoders, removing\n",
554 drm_get_connector_name(connector));
555 connector->funcs->destroy(connector);
556 }
Ben Skeggs6ee73862009-12-11 19:24:15 +1000557 }
558
559 ret = nv50_display_init(dev);
Ben Skeggsa1663ed2010-03-25 16:01:04 +1000560 if (ret) {
561 nv50_display_destroy(dev);
Ben Skeggs6ee73862009-12-11 19:24:15 +1000562 return ret;
Ben Skeggsa1663ed2010-03-25 16:01:04 +1000563 }
Ben Skeggs6ee73862009-12-11 19:24:15 +1000564
565 return 0;
566}
567
Francisco Jerezc88c2e02010-07-24 17:37:33 +0200568void
569nv50_display_destroy(struct drm_device *dev)
Ben Skeggs6ee73862009-12-11 19:24:15 +1000570{
571 struct drm_nouveau_private *dev_priv = dev->dev_private;
572
Maarten Maathuisef2bb502009-12-13 16:53:12 +0100573 NV_DEBUG_KMS(dev, "\n");
Ben Skeggs6ee73862009-12-11 19:24:15 +1000574
575 drm_mode_config_cleanup(dev);
576
577 nv50_display_disable(dev);
578 nv50_evo_channel_del(&dev_priv->evo);
Ben Skeggs6ee73862009-12-11 19:24:15 +1000579}
580
Ben Skeggs87c0e0e2010-07-06 08:54:34 +1000581static u16
582nv50_display_script_select(struct drm_device *dev, struct dcb_entry *dcb,
583 u32 mc, int pxclk)
Ben Skeggs6ee73862009-12-11 19:24:15 +1000584{
585 struct drm_nouveau_private *dev_priv = dev->dev_private;
Ben Skeggs75c722d2009-12-21 12:16:52 +1000586 struct nouveau_connector *nv_connector = NULL;
587 struct drm_encoder *encoder;
Ben Skeggs04a39c52010-02-24 10:03:05 +1000588 struct nvbios *bios = &dev_priv->vbios;
Ben Skeggs87c0e0e2010-07-06 08:54:34 +1000589 u32 script = 0, or;
Ben Skeggs6ee73862009-12-11 19:24:15 +1000590
Ben Skeggs75c722d2009-12-21 12:16:52 +1000591 list_for_each_entry(encoder, &dev->mode_config.encoder_list, head) {
592 struct nouveau_encoder *nv_encoder = nouveau_encoder(encoder);
593
Ben Skeggs87c0e0e2010-07-06 08:54:34 +1000594 if (nv_encoder->dcb != dcb)
Ben Skeggs75c722d2009-12-21 12:16:52 +1000595 continue;
596
597 nv_connector = nouveau_encoder_connector_get(nv_encoder);
598 break;
599 }
600
Ben Skeggs87c0e0e2010-07-06 08:54:34 +1000601 or = ffs(dcb->or) - 1;
602 switch (dcb->type) {
Ben Skeggs6ee73862009-12-11 19:24:15 +1000603 case OUTPUT_LVDS:
604 script = (mc >> 8) & 0xf;
Ben Skeggs04a39c52010-02-24 10:03:05 +1000605 if (bios->fp_no_ddc) {
Ben Skeggs6ee73862009-12-11 19:24:15 +1000606 if (bios->fp.dual_link)
607 script |= 0x0100;
608 if (bios->fp.if_is_24bit)
609 script |= 0x0200;
610 } else {
611 if (pxclk >= bios->fp.duallink_transition_clk) {
612 script |= 0x0100;
613 if (bios->fp.strapless_is_24bit & 2)
614 script |= 0x0200;
615 } else
616 if (bios->fp.strapless_is_24bit & 1)
617 script |= 0x0200;
Ben Skeggs75c722d2009-12-21 12:16:52 +1000618
619 if (nv_connector && nv_connector->edid &&
620 (nv_connector->edid->revision >= 4) &&
621 (nv_connector->edid->input & 0x70) >= 0x20)
622 script |= 0x0200;
Ben Skeggs6ee73862009-12-11 19:24:15 +1000623 }
624
625 if (nouveau_uscript_lvds >= 0) {
626 NV_INFO(dev, "override script 0x%04x with 0x%04x "
627 "for output LVDS-%d\n", script,
628 nouveau_uscript_lvds, or);
629 script = nouveau_uscript_lvds;
630 }
631 break;
632 case OUTPUT_TMDS:
633 script = (mc >> 8) & 0xf;
634 if (pxclk >= 165000)
635 script |= 0x0100;
636
637 if (nouveau_uscript_tmds >= 0) {
638 NV_INFO(dev, "override script 0x%04x with 0x%04x "
639 "for output TMDS-%d\n", script,
640 nouveau_uscript_tmds, or);
641 script = nouveau_uscript_tmds;
642 }
643 break;
644 case OUTPUT_DP:
645 script = (mc >> 8) & 0xf;
646 break;
647 case OUTPUT_ANALOG:
648 script = 0xff;
649 break;
650 default:
651 NV_ERROR(dev, "modeset on unsupported output type!\n");
652 break;
653 }
654
655 return script;
656}
657
658static void
659nv50_display_vblank_crtc_handler(struct drm_device *dev, int crtc)
660{
661 struct drm_nouveau_private *dev_priv = dev->dev_private;
662 struct nouveau_channel *chan;
663 struct list_head *entry, *tmp;
664
665 list_for_each_safe(entry, tmp, &dev_priv->vbl_waiting) {
666 chan = list_entry(entry, struct nouveau_channel, nvsw.vbl_wait);
667
668 nouveau_bo_wr32(chan->notifier_bo, chan->nvsw.vblsem_offset,
669 chan->nvsw.vblsem_rval);
670 list_del(&chan->nvsw.vbl_wait);
671 }
672}
673
674static void
675nv50_display_vblank_handler(struct drm_device *dev, uint32_t intr)
676{
677 intr &= NV50_PDISPLAY_INTR_1_VBLANK_CRTC;
678
679 if (intr & NV50_PDISPLAY_INTR_1_VBLANK_CRTC_0)
680 nv50_display_vblank_crtc_handler(dev, 0);
681
682 if (intr & NV50_PDISPLAY_INTR_1_VBLANK_CRTC_1)
683 nv50_display_vblank_crtc_handler(dev, 1);
684
685 nv_wr32(dev, NV50_PDISPLAY_INTR_EN, nv_rd32(dev,
686 NV50_PDISPLAY_INTR_EN) & ~intr);
687 nv_wr32(dev, NV50_PDISPLAY_INTR_1, intr);
688}
689
690static void
691nv50_display_unk10_handler(struct drm_device *dev)
692{
Ben Skeggs87c0e0e2010-07-06 08:54:34 +1000693 struct drm_nouveau_private *dev_priv = dev->dev_private;
694 u32 unk30 = nv_rd32(dev, 0x610030), mc;
695 int i, crtc, or, type = OUTPUT_ANY;
Ben Skeggs6ee73862009-12-11 19:24:15 +1000696
Ben Skeggs87c0e0e2010-07-06 08:54:34 +1000697 NV_DEBUG_KMS(dev, "0x610030: 0x%08x\n", unk30);
698 dev_priv->evo_irq.dcb = NULL;
Ben Skeggs6ee73862009-12-11 19:24:15 +1000699
700 nv_wr32(dev, 0x619494, nv_rd32(dev, 0x619494) & ~8);
701
Ben Skeggs87c0e0e2010-07-06 08:54:34 +1000702 /* Determine which CRTC we're dealing with, only 1 ever will be
703 * signalled at the same time with the current nouveau code.
704 */
705 crtc = ffs((unk30 & 0x00000060) >> 5) - 1;
706 if (crtc < 0)
707 goto ack;
Ben Skeggs6ee73862009-12-11 19:24:15 +1000708
Ben Skeggs87c0e0e2010-07-06 08:54:34 +1000709 /* Nothing needs to be done for the encoder */
710 crtc = ffs((unk30 & 0x00000180) >> 7) - 1;
711 if (crtc < 0)
712 goto ack;
713
714 /* Find which encoder was connected to the CRTC */
715 for (i = 0; type == OUTPUT_ANY && i < 3; i++) {
716 mc = nv_rd32(dev, NV50_PDISPLAY_DAC_MODE_CTRL_C(i));
717 NV_DEBUG_KMS(dev, "DAC-%d mc: 0x%08x\n", i, mc);
718 if (!(mc & (1 << crtc)))
719 continue;
720
721 switch ((mc & 0x00000f00) >> 8) {
722 case 0: type = OUTPUT_ANALOG; break;
723 case 1: type = OUTPUT_TV; break;
724 default:
725 NV_ERROR(dev, "invalid mc, DAC-%d: 0x%08x\n", i, mc);
726 goto ack;
727 }
728
729 or = i;
730 }
731
Ben Skeggs8597a1b2010-09-06 11:39:25 +1000732 for (i = 0; type == OUTPUT_ANY && i < nv50_sor_nr(dev); i++) {
Ben Skeggs87c0e0e2010-07-06 08:54:34 +1000733 if (dev_priv->chipset < 0x90 ||
734 dev_priv->chipset == 0x92 ||
735 dev_priv->chipset == 0xa0)
736 mc = nv_rd32(dev, NV50_PDISPLAY_SOR_MODE_CTRL_C(i));
737 else
738 mc = nv_rd32(dev, NV90_PDISPLAY_SOR_MODE_CTRL_C(i));
739
740 NV_DEBUG_KMS(dev, "SOR-%d mc: 0x%08x\n", i, mc);
741 if (!(mc & (1 << crtc)))
742 continue;
743
744 switch ((mc & 0x00000f00) >> 8) {
745 case 0: type = OUTPUT_LVDS; break;
746 case 1: type = OUTPUT_TMDS; break;
747 case 2: type = OUTPUT_TMDS; break;
748 case 5: type = OUTPUT_TMDS; break;
749 case 8: type = OUTPUT_DP; break;
750 case 9: type = OUTPUT_DP; break;
751 default:
752 NV_ERROR(dev, "invalid mc, SOR-%d: 0x%08x\n", i, mc);
753 goto ack;
754 }
755
756 or = i;
757 }
758
759 /* There was no encoder to disable */
760 if (type == OUTPUT_ANY)
761 goto ack;
762
763 /* Disable the encoder */
764 for (i = 0; i < dev_priv->vbios.dcb.entries; i++) {
765 struct dcb_entry *dcb = &dev_priv->vbios.dcb.entry[i];
766
767 if (dcb->type == type && (dcb->or & (1 << or))) {
768 nouveau_bios_run_display_table(dev, dcb, 0, -1);
769 dev_priv->evo_irq.dcb = dcb;
770 goto ack;
771 }
772 }
773
774 NV_ERROR(dev, "no dcb for %d %d 0x%08x\n", or, type, mc);
Ben Skeggs6ee73862009-12-11 19:24:15 +1000775ack:
776 nv_wr32(dev, NV50_PDISPLAY_INTR_1, NV50_PDISPLAY_INTR_1_CLK_UNK10);
777 nv_wr32(dev, 0x610030, 0x80000000);
778}
779
780static void
Ben Skeggsafa3b4c2010-04-23 08:21:48 +1000781nv50_display_unk20_dp_hack(struct drm_device *dev, struct dcb_entry *dcb)
782{
783 int or = ffs(dcb->or) - 1, link = !(dcb->dpconf.sor.link & 1);
784 struct drm_encoder *encoder;
785 uint32_t tmp, unk0 = 0, unk1 = 0;
786
787 if (dcb->type != OUTPUT_DP)
788 return;
789
790 list_for_each_entry(encoder, &dev->mode_config.encoder_list, head) {
791 struct nouveau_encoder *nv_encoder = nouveau_encoder(encoder);
792
793 if (nv_encoder->dcb == dcb) {
794 unk0 = nv_encoder->dp.unk0;
795 unk1 = nv_encoder->dp.unk1;
796 break;
797 }
798 }
799
800 if (unk0 || unk1) {
801 tmp = nv_rd32(dev, NV50_SOR_DP_CTRL(or, link));
802 tmp &= 0xfffffe03;
803 nv_wr32(dev, NV50_SOR_DP_CTRL(or, link), tmp | unk0);
804
805 tmp = nv_rd32(dev, NV50_SOR_DP_UNK128(or, link));
806 tmp &= 0xfef080c0;
807 nv_wr32(dev, NV50_SOR_DP_UNK128(or, link), tmp | unk1);
808 }
809}
810
811static void
Ben Skeggs6ee73862009-12-11 19:24:15 +1000812nv50_display_unk20_handler(struct drm_device *dev)
813{
Ben Skeggs87c0e0e2010-07-06 08:54:34 +1000814 struct drm_nouveau_private *dev_priv = dev->dev_private;
815 u32 unk30 = nv_rd32(dev, 0x610030), tmp, pclk, script, mc;
816 struct dcb_entry *dcb;
817 int i, crtc, or, type = OUTPUT_ANY;
Ben Skeggs6ee73862009-12-11 19:24:15 +1000818
Ben Skeggs87c0e0e2010-07-06 08:54:34 +1000819 NV_DEBUG_KMS(dev, "0x610030: 0x%08x\n", unk30);
820 dcb = dev_priv->evo_irq.dcb;
821 if (dcb) {
822 nouveau_bios_run_display_table(dev, dcb, 0, -2);
823 dev_priv->evo_irq.dcb = NULL;
824 }
825
826 /* CRTC clock change requested? */
827 crtc = ffs((unk30 & 0x00000600) >> 9) - 1;
828 if (crtc >= 0) {
829 pclk = nv_rd32(dev, NV50_PDISPLAY_CRTC_P(crtc, CLOCK));
830 pclk &= 0x003fffff;
831
832 nv50_crtc_set_clock(dev, crtc, pclk);
833
834 tmp = nv_rd32(dev, NV50_PDISPLAY_CRTC_CLK_CTRL2(crtc));
835 tmp &= ~0x000000f;
836 nv_wr32(dev, NV50_PDISPLAY_CRTC_CLK_CTRL2(crtc), tmp);
837 }
838
839 /* Nothing needs to be done for the encoder */
840 crtc = ffs((unk30 & 0x00000180) >> 7) - 1;
841 if (crtc < 0)
Ben Skeggs6ee73862009-12-11 19:24:15 +1000842 goto ack;
Ben Skeggs87c0e0e2010-07-06 08:54:34 +1000843 pclk = nv_rd32(dev, NV50_PDISPLAY_CRTC_P(crtc, CLOCK)) & 0x003fffff;
Ben Skeggs6ee73862009-12-11 19:24:15 +1000844
Ben Skeggs87c0e0e2010-07-06 08:54:34 +1000845 /* Find which encoder is connected to the CRTC */
846 for (i = 0; type == OUTPUT_ANY && i < 3; i++) {
847 mc = nv_rd32(dev, NV50_PDISPLAY_DAC_MODE_CTRL_P(i));
848 NV_DEBUG_KMS(dev, "DAC-%d mc: 0x%08x\n", i, mc);
849 if (!(mc & (1 << crtc)))
850 continue;
Ben Skeggs6ee73862009-12-11 19:24:15 +1000851
Ben Skeggs87c0e0e2010-07-06 08:54:34 +1000852 switch ((mc & 0x00000f00) >> 8) {
853 case 0: type = OUTPUT_ANALOG; break;
854 case 1: type = OUTPUT_TV; break;
855 default:
856 NV_ERROR(dev, "invalid mc, DAC-%d: 0x%08x\n", i, mc);
857 goto ack;
858 }
Ben Skeggs6ee73862009-12-11 19:24:15 +1000859
Ben Skeggs87c0e0e2010-07-06 08:54:34 +1000860 or = i;
861 }
Ben Skeggs6ee73862009-12-11 19:24:15 +1000862
Ben Skeggs8597a1b2010-09-06 11:39:25 +1000863 for (i = 0; type == OUTPUT_ANY && i < nv50_sor_nr(dev); i++) {
Ben Skeggs87c0e0e2010-07-06 08:54:34 +1000864 if (dev_priv->chipset < 0x90 ||
865 dev_priv->chipset == 0x92 ||
866 dev_priv->chipset == 0xa0)
867 mc = nv_rd32(dev, NV50_PDISPLAY_SOR_MODE_CTRL_P(i));
868 else
869 mc = nv_rd32(dev, NV90_PDISPLAY_SOR_MODE_CTRL_P(i));
Ben Skeggs6ee73862009-12-11 19:24:15 +1000870
Ben Skeggs87c0e0e2010-07-06 08:54:34 +1000871 NV_DEBUG_KMS(dev, "SOR-%d mc: 0x%08x\n", i, mc);
872 if (!(mc & (1 << crtc)))
873 continue;
Ben Skeggsafa3b4c2010-04-23 08:21:48 +1000874
Ben Skeggs87c0e0e2010-07-06 08:54:34 +1000875 switch ((mc & 0x00000f00) >> 8) {
876 case 0: type = OUTPUT_LVDS; break;
877 case 1: type = OUTPUT_TMDS; break;
878 case 2: type = OUTPUT_TMDS; break;
879 case 5: type = OUTPUT_TMDS; break;
880 case 8: type = OUTPUT_DP; break;
881 case 9: type = OUTPUT_DP; break;
882 default:
883 NV_ERROR(dev, "invalid mc, SOR-%d: 0x%08x\n", i, mc);
884 goto ack;
885 }
Ben Skeggs6ee73862009-12-11 19:24:15 +1000886
Ben Skeggs87c0e0e2010-07-06 08:54:34 +1000887 or = i;
888 }
889
890 if (type == OUTPUT_ANY)
891 goto ack;
892
893 /* Enable the encoder */
894 for (i = 0; i < dev_priv->vbios.dcb.entries; i++) {
895 dcb = &dev_priv->vbios.dcb.entry[i];
896 if (dcb->type == type && (dcb->or & (1 << or)))
897 break;
898 }
899
900 if (i == dev_priv->vbios.dcb.entries) {
901 NV_ERROR(dev, "no dcb for %d %d 0x%08x\n", or, type, mc);
902 goto ack;
903 }
904
905 script = nv50_display_script_select(dev, dcb, mc, pclk);
906 nouveau_bios_run_display_table(dev, dcb, script, pclk);
907
908 nv50_display_unk20_dp_hack(dev, dcb);
Ben Skeggs87c0e0e2010-07-06 08:54:34 +1000909
910 if (dcb->type != OUTPUT_ANALOG) {
Ben Skeggs6ee73862009-12-11 19:24:15 +1000911 tmp = nv_rd32(dev, NV50_PDISPLAY_SOR_CLK_CTRL2(or));
912 tmp &= ~0x00000f0f;
913 if (script & 0x0100)
914 tmp |= 0x00000101;
915 nv_wr32(dev, NV50_PDISPLAY_SOR_CLK_CTRL2(or), tmp);
916 } else {
917 nv_wr32(dev, NV50_PDISPLAY_DAC_CLK_CTRL2(or), 0);
918 }
919
Ben Skeggs87c0e0e2010-07-06 08:54:34 +1000920 dev_priv->evo_irq.dcb = dcb;
921 dev_priv->evo_irq.pclk = pclk;
922 dev_priv->evo_irq.script = script;
923
Ben Skeggs6ee73862009-12-11 19:24:15 +1000924ack:
925 nv_wr32(dev, NV50_PDISPLAY_INTR_1, NV50_PDISPLAY_INTR_1_CLK_UNK20);
926 nv_wr32(dev, 0x610030, 0x80000000);
927}
928
Ben Skeggs271f29e2010-07-09 10:37:42 +1000929/* If programming a TMDS output on a SOR that can also be configured for
930 * DisplayPort, make sure NV50_SOR_DP_CTRL_ENABLE is forced off.
931 *
932 * It looks like the VBIOS TMDS scripts make an attempt at this, however,
933 * the VBIOS scripts on at least one board I have only switch it off on
934 * link 0, causing a blank display if the output has previously been
935 * programmed for DisplayPort.
936 */
937static void
938nv50_display_unk40_dp_set_tmds(struct drm_device *dev, struct dcb_entry *dcb)
939{
940 int or = ffs(dcb->or) - 1, link = !(dcb->dpconf.sor.link & 1);
941 struct drm_encoder *encoder;
942 u32 tmp;
943
944 if (dcb->type != OUTPUT_TMDS)
945 return;
946
947 list_for_each_entry(encoder, &dev->mode_config.encoder_list, head) {
948 struct nouveau_encoder *nv_encoder = nouveau_encoder(encoder);
949
950 if (nv_encoder->dcb->type == OUTPUT_DP &&
951 nv_encoder->dcb->or & (1 << or)) {
952 tmp = nv_rd32(dev, NV50_SOR_DP_CTRL(or, link));
953 tmp &= ~NV50_SOR_DP_CTRL_ENABLED;
954 nv_wr32(dev, NV50_SOR_DP_CTRL(or, link), tmp);
955 break;
956 }
957 }
958}
959
Ben Skeggs6ee73862009-12-11 19:24:15 +1000960static void
961nv50_display_unk40_handler(struct drm_device *dev)
962{
Ben Skeggs87c0e0e2010-07-06 08:54:34 +1000963 struct drm_nouveau_private *dev_priv = dev->dev_private;
964 struct dcb_entry *dcb = dev_priv->evo_irq.dcb;
965 u16 script = dev_priv->evo_irq.script;
966 u32 unk30 = nv_rd32(dev, 0x610030), pclk = dev_priv->evo_irq.pclk;
Ben Skeggs6ee73862009-12-11 19:24:15 +1000967
Ben Skeggs87c0e0e2010-07-06 08:54:34 +1000968 NV_DEBUG_KMS(dev, "0x610030: 0x%08x\n", unk30);
969 dev_priv->evo_irq.dcb = NULL;
970 if (!dcb)
Ben Skeggs6ee73862009-12-11 19:24:15 +1000971 goto ack;
Ben Skeggs6ee73862009-12-11 19:24:15 +1000972
Ben Skeggs87c0e0e2010-07-06 08:54:34 +1000973 nouveau_bios_run_display_table(dev, dcb, script, -pclk);
Ben Skeggs271f29e2010-07-09 10:37:42 +1000974 nv50_display_unk40_dp_set_tmds(dev, dcb);
975
Ben Skeggs6ee73862009-12-11 19:24:15 +1000976ack:
977 nv_wr32(dev, NV50_PDISPLAY_INTR_1, NV50_PDISPLAY_INTR_1_CLK_UNK40);
978 nv_wr32(dev, 0x610030, 0x80000000);
979 nv_wr32(dev, 0x619494, nv_rd32(dev, 0x619494) | 8);
980}
981
982void
983nv50_display_irq_handler_bh(struct work_struct *work)
984{
985 struct drm_nouveau_private *dev_priv =
986 container_of(work, struct drm_nouveau_private, irq_work);
987 struct drm_device *dev = dev_priv->dev;
988
989 for (;;) {
990 uint32_t intr0 = nv_rd32(dev, NV50_PDISPLAY_INTR_0);
991 uint32_t intr1 = nv_rd32(dev, NV50_PDISPLAY_INTR_1);
992
Maarten Maathuisef2bb502009-12-13 16:53:12 +0100993 NV_DEBUG_KMS(dev, "PDISPLAY_INTR_BH 0x%08x 0x%08x\n", intr0, intr1);
Ben Skeggs6ee73862009-12-11 19:24:15 +1000994
995 if (intr1 & NV50_PDISPLAY_INTR_1_CLK_UNK10)
996 nv50_display_unk10_handler(dev);
997 else
998 if (intr1 & NV50_PDISPLAY_INTR_1_CLK_UNK20)
999 nv50_display_unk20_handler(dev);
1000 else
1001 if (intr1 & NV50_PDISPLAY_INTR_1_CLK_UNK40)
1002 nv50_display_unk40_handler(dev);
1003 else
1004 break;
1005 }
1006
1007 nv_wr32(dev, NV03_PMC_INTR_EN_0, 1);
1008}
1009
1010static void
1011nv50_display_error_handler(struct drm_device *dev)
1012{
1013 uint32_t addr, data;
1014
1015 nv_wr32(dev, NV50_PDISPLAY_INTR_0, 0x00010000);
1016 addr = nv_rd32(dev, NV50_PDISPLAY_TRAPPED_ADDR);
1017 data = nv_rd32(dev, NV50_PDISPLAY_TRAPPED_DATA);
1018
1019 NV_ERROR(dev, "EvoCh %d Mthd 0x%04x Data 0x%08x (0x%04x 0x%02x)\n",
1020 0, addr & 0xffc, data, addr >> 16, (addr >> 12) & 0xf);
1021
1022 nv_wr32(dev, NV50_PDISPLAY_TRAPPED_ADDR, 0x90000000);
1023}
1024
Ben Skeggsa5acac62010-03-30 15:14:41 +10001025void
1026nv50_display_irq_hotplug_bh(struct work_struct *work)
Ben Skeggs6ee73862009-12-11 19:24:15 +10001027{
Ben Skeggsa5acac62010-03-30 15:14:41 +10001028 struct drm_nouveau_private *dev_priv =
1029 container_of(work, struct drm_nouveau_private, hpd_work);
1030 struct drm_device *dev = dev_priv->dev;
Ben Skeggs6ee73862009-12-11 19:24:15 +10001031 struct drm_connector *connector;
1032 const uint32_t gpio_reg[4] = { 0xe104, 0xe108, 0xe280, 0xe284 };
1033 uint32_t unplug_mask, plug_mask, change_mask;
1034 uint32_t hpd0, hpd1 = 0;
1035
1036 hpd0 = nv_rd32(dev, 0xe054) & nv_rd32(dev, 0xe050);
1037 if (dev_priv->chipset >= 0x90)
1038 hpd1 = nv_rd32(dev, 0xe074) & nv_rd32(dev, 0xe070);
1039
1040 plug_mask = (hpd0 & 0x0000ffff) | (hpd1 << 16);
1041 unplug_mask = (hpd0 >> 16) | (hpd1 & 0xffff0000);
1042 change_mask = plug_mask | unplug_mask;
1043
1044 list_for_each_entry(connector, &dev->mode_config.connector_list, head) {
1045 struct drm_encoder_helper_funcs *helper;
1046 struct nouveau_connector *nv_connector =
1047 nouveau_connector(connector);
1048 struct nouveau_encoder *nv_encoder;
1049 struct dcb_gpio_entry *gpio;
1050 uint32_t reg;
1051 bool plugged;
1052
1053 if (!nv_connector->dcb)
1054 continue;
1055
1056 gpio = nouveau_bios_gpio_entry(dev, nv_connector->dcb->gpio_tag);
1057 if (!gpio || !(change_mask & (1 << gpio->line)))
1058 continue;
1059
1060 reg = nv_rd32(dev, gpio_reg[gpio->line >> 3]);
1061 plugged = !!(reg & (4 << ((gpio->line & 7) << 2)));
1062 NV_INFO(dev, "%splugged %s\n", plugged ? "" : "un",
1063 drm_get_connector_name(connector)) ;
1064
1065 if (!connector->encoder || !connector->encoder->crtc ||
1066 !connector->encoder->crtc->enabled)
1067 continue;
1068 nv_encoder = nouveau_encoder(connector->encoder);
1069 helper = connector->encoder->helper_private;
1070
1071 if (nv_encoder->dcb->type != OUTPUT_DP)
1072 continue;
1073
1074 if (plugged)
1075 helper->dpms(connector->encoder, DRM_MODE_DPMS_ON);
1076 else
1077 helper->dpms(connector->encoder, DRM_MODE_DPMS_OFF);
1078 }
1079
1080 nv_wr32(dev, 0xe054, nv_rd32(dev, 0xe054));
1081 if (dev_priv->chipset >= 0x90)
1082 nv_wr32(dev, 0xe074, nv_rd32(dev, 0xe074));
Dave Airlie4abe3522010-03-30 05:34:18 +00001083
Dave Airlieeb1f8e42010-05-07 06:42:51 +00001084 drm_helper_hpd_irq_event(dev);
Ben Skeggs6ee73862009-12-11 19:24:15 +10001085}
1086
1087void
1088nv50_display_irq_handler(struct drm_device *dev)
1089{
1090 struct drm_nouveau_private *dev_priv = dev->dev_private;
1091 uint32_t delayed = 0;
1092
Ben Skeggsa5acac62010-03-30 15:14:41 +10001093 if (nv_rd32(dev, NV50_PMC_INTR_0) & NV50_PMC_INTR_0_HOTPLUG) {
1094 if (!work_pending(&dev_priv->hpd_work))
1095 queue_work(dev_priv->wq, &dev_priv->hpd_work);
1096 }
Ben Skeggs6ee73862009-12-11 19:24:15 +10001097
1098 while (nv_rd32(dev, NV50_PMC_INTR_0) & NV50_PMC_INTR_0_DISPLAY) {
1099 uint32_t intr0 = nv_rd32(dev, NV50_PDISPLAY_INTR_0);
1100 uint32_t intr1 = nv_rd32(dev, NV50_PDISPLAY_INTR_1);
1101 uint32_t clock;
1102
Maarten Maathuisef2bb502009-12-13 16:53:12 +01001103 NV_DEBUG_KMS(dev, "PDISPLAY_INTR 0x%08x 0x%08x\n", intr0, intr1);
Ben Skeggs6ee73862009-12-11 19:24:15 +10001104
1105 if (!intr0 && !(intr1 & ~delayed))
1106 break;
1107
1108 if (intr0 & 0x00010000) {
1109 nv50_display_error_handler(dev);
1110 intr0 &= ~0x00010000;
1111 }
1112
1113 if (intr1 & NV50_PDISPLAY_INTR_1_VBLANK_CRTC) {
1114 nv50_display_vblank_handler(dev, intr1);
1115 intr1 &= ~NV50_PDISPLAY_INTR_1_VBLANK_CRTC;
1116 }
1117
1118 clock = (intr1 & (NV50_PDISPLAY_INTR_1_CLK_UNK10 |
1119 NV50_PDISPLAY_INTR_1_CLK_UNK20 |
1120 NV50_PDISPLAY_INTR_1_CLK_UNK40));
1121 if (clock) {
1122 nv_wr32(dev, NV03_PMC_INTR_EN_0, 0);
1123 if (!work_pending(&dev_priv->irq_work))
1124 queue_work(dev_priv->wq, &dev_priv->irq_work);
1125 delayed |= clock;
1126 intr1 &= ~clock;
1127 }
1128
1129 if (intr0) {
1130 NV_ERROR(dev, "unknown PDISPLAY_INTR_0: 0x%08x\n", intr0);
1131 nv_wr32(dev, NV50_PDISPLAY_INTR_0, intr0);
1132 }
1133
1134 if (intr1) {
1135 NV_ERROR(dev,
1136 "unknown PDISPLAY_INTR_1: 0x%08x\n", intr1);
1137 nv_wr32(dev, NV50_PDISPLAY_INTR_1, intr1);
1138 }
1139 }
1140}
1141