blob: 20ab2c249b82ede8715c190963740824f23f76e7 [file] [log] [blame]
Ben Skeggs26f6d882011-07-04 16:25:18 +10001/*
2 * Copyright 2011 Red Hat Inc.
3 *
4 * Permission is hereby granted, free of charge, to any person obtaining a
5 * copy of this software and associated documentation files (the "Software"),
6 * to deal in the Software without restriction, including without limitation
7 * the rights to use, copy, modify, merge, publish, distribute, sublicense,
8 * and/or sell copies of the Software, and to permit persons to whom the
9 * Software is furnished to do so, subject to the following conditions:
10 *
11 * The above copyright notice and this permission notice shall be included in
12 * all copies or substantial portions of the Software.
13 *
14 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
15 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
16 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
17 * THE COPYRIGHT HOLDER(S) OR AUTHOR(S) BE LIABLE FOR ANY CLAIM, DAMAGES OR
18 * OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
19 * ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
20 * OTHER DEALINGS IN THE SOFTWARE.
21 *
22 * Authors: Ben Skeggs
23 */
24
Ben Skeggs51beb422011-07-05 10:33:08 +100025#include <linux/dma-mapping.h>
Ben Skeggs83fc0832011-07-05 13:08:40 +100026
Ben Skeggs26f6d882011-07-04 16:25:18 +100027#include "drmP.h"
Ben Skeggs83fc0832011-07-05 13:08:40 +100028#include "drm_crtc_helper.h"
Ben Skeggs26f6d882011-07-04 16:25:18 +100029
30#include "nouveau_drv.h"
31#include "nouveau_connector.h"
32#include "nouveau_encoder.h"
33#include "nouveau_crtc.h"
Ben Skeggs37b034a2011-07-08 14:43:19 +100034#include "nouveau_dma.h"
Ben Skeggs438d99e2011-07-05 16:48:06 +100035#include "nouveau_fb.h"
Ben Skeggs3a89cd02011-07-07 10:47:10 +100036#include "nv50_display.h"
Ben Skeggs26f6d882011-07-04 16:25:18 +100037
38struct nvd0_display {
39 struct nouveau_gpuobj *mem;
Ben Skeggs51beb422011-07-05 10:33:08 +100040 struct {
41 dma_addr_t handle;
42 u32 *ptr;
43 } evo[1];
Ben Skeggsf20ce962011-07-08 13:17:01 +100044
45 struct tasklet_struct tasklet;
Ben Skeggsee417792011-07-08 14:34:45 +100046 u32 modeset;
Ben Skeggs26f6d882011-07-04 16:25:18 +100047};
48
49static struct nvd0_display *
50nvd0_display(struct drm_device *dev)
51{
52 struct drm_nouveau_private *dev_priv = dev->dev_private;
53 return dev_priv->engine.display.priv;
54}
55
Ben Skeggs37b034a2011-07-08 14:43:19 +100056static inline int
Ben Skeggs51beb422011-07-05 10:33:08 +100057evo_icmd(struct drm_device *dev, int id, u32 mthd, u32 data)
58{
59 int ret = 0;
60 nv_mask(dev, 0x610700 + (id * 0x10), 0x00000001, 0x00000001);
61 nv_wr32(dev, 0x610704 + (id * 0x10), data);
62 nv_mask(dev, 0x610704 + (id * 0x10), 0x80000ffc, 0x80000000 | mthd);
63 if (!nv_wait(dev, 0x610704 + (id * 0x10), 0x80000000, 0x00000000))
64 ret = -EBUSY;
65 nv_mask(dev, 0x610700 + (id * 0x10), 0x00000001, 0x00000000);
66 return ret;
67}
68
69static u32 *
70evo_wait(struct drm_device *dev, int id, int nr)
71{
72 struct nvd0_display *disp = nvd0_display(dev);
73 u32 put = nv_rd32(dev, 0x640000 + (id * 0x1000)) / 4;
74
75 if (put + nr >= (PAGE_SIZE / 4)) {
76 disp->evo[id].ptr[put] = 0x20000000;
77
78 nv_wr32(dev, 0x640000 + (id * 0x1000), 0x00000000);
79 if (!nv_wait(dev, 0x640004 + (id * 0x1000), ~0, 0x00000000)) {
80 NV_ERROR(dev, "evo %d dma stalled\n", id);
81 return NULL;
82 }
83
84 put = 0;
85 }
86
87 return disp->evo[id].ptr + put;
88}
89
90static void
91evo_kick(u32 *push, struct drm_device *dev, int id)
92{
93 struct nvd0_display *disp = nvd0_display(dev);
94 nv_wr32(dev, 0x640000 + (id * 0x1000), (push - disp->evo[id].ptr) << 2);
95}
96
97#define evo_mthd(p,m,s) *((p)++) = (((s) << 18) | (m))
98#define evo_data(p,d) *((p)++) = (d)
99
Ben Skeggs83fc0832011-07-05 13:08:40 +1000100static struct drm_crtc *
101nvd0_display_crtc_get(struct drm_encoder *encoder)
102{
103 return nouveau_encoder(encoder)->crtc;
104}
105
Ben Skeggs26f6d882011-07-04 16:25:18 +1000106/******************************************************************************
Ben Skeggs438d99e2011-07-05 16:48:06 +1000107 * CRTC
108 *****************************************************************************/
109static int
110nvd0_crtc_set_dither(struct nouveau_crtc *nv_crtc, bool on, bool update)
111{
112 struct drm_device *dev = nv_crtc->base.dev;
113 u32 *push, mode;
114
115 mode = 0x00000000;
116 if (on) {
117 /* 0x11: 6bpc dynamic 2x2
118 * 0x13: 8bpc dynamic 2x2
119 * 0x19: 6bpc static 2x2
120 * 0x1b: 8bpc static 2x2
121 * 0x21: 6bpc temporal
122 * 0x23: 8bpc temporal
123 */
124 mode = 0x00000011;
125 }
126
127 push = evo_wait(dev, 0, 4);
128 if (push) {
129 evo_mthd(push, 0x0490 + (nv_crtc->index * 0x300), 1);
130 evo_data(push, mode);
131 if (update) {
132 evo_mthd(push, 0x0080, 1);
133 evo_data(push, 0x00000000);
134 }
135 evo_kick(push, dev, 0);
136 }
137
138 return 0;
139}
140
141static int
142nvd0_crtc_set_scale(struct nouveau_crtc *nv_crtc, int type, bool update)
143{
144 struct drm_display_mode *mode = &nv_crtc->base.mode;
145 struct drm_device *dev = nv_crtc->base.dev;
Ben Skeggsf3fdc522011-07-07 16:01:57 +1000146 struct nouveau_connector *nv_connector;
147 u32 *push, outX, outY;
Ben Skeggs438d99e2011-07-05 16:48:06 +1000148
Ben Skeggsf3fdc522011-07-07 16:01:57 +1000149 outX = mode->hdisplay;
150 outY = mode->vdisplay;
151
152 nv_connector = nouveau_crtc_connector_get(nv_crtc);
153 if (nv_connector && nv_connector->native_mode) {
154 struct drm_display_mode *native = nv_connector->native_mode;
155 u32 xratio = (native->hdisplay << 19) / mode->hdisplay;
156 u32 yratio = (native->vdisplay << 19) / mode->vdisplay;
157
158 switch (type) {
159 case DRM_MODE_SCALE_ASPECT:
160 if (xratio > yratio) {
161 outX = (mode->hdisplay * yratio) >> 19;
162 outY = (mode->vdisplay * yratio) >> 19;
163 } else {
164 outX = (mode->hdisplay * xratio) >> 19;
165 outY = (mode->vdisplay * xratio) >> 19;
166 }
167 break;
168 case DRM_MODE_SCALE_FULLSCREEN:
169 outX = native->hdisplay;
170 outY = native->vdisplay;
171 break;
172 default:
173 break;
174 }
175 }
Ben Skeggs438d99e2011-07-05 16:48:06 +1000176
177 push = evo_wait(dev, 0, 16);
178 if (push) {
179 evo_mthd(push, 0x04c0 + (nv_crtc->index * 0x300), 3);
Ben Skeggsf3fdc522011-07-07 16:01:57 +1000180 evo_data(push, (outY << 16) | outX);
181 evo_data(push, (outY << 16) | outX);
182 evo_data(push, (outY << 16) | outX);
Ben Skeggs438d99e2011-07-05 16:48:06 +1000183 evo_mthd(push, 0x0494 + (nv_crtc->index * 0x300), 1);
184 evo_data(push, 0x00000000);
Ben Skeggs438d99e2011-07-05 16:48:06 +1000185 evo_mthd(push, 0x04b8 + (nv_crtc->index * 0x300), 1);
186 evo_data(push, (mode->vdisplay << 16) | mode->hdisplay);
187 if (update) {
188 evo_mthd(push, 0x0080, 1);
189 evo_data(push, 0x00000000);
190 }
191 evo_kick(push, dev, 0);
192 }
193
194 return 0;
195}
196
197static int
198nvd0_crtc_set_image(struct nouveau_crtc *nv_crtc, struct drm_framebuffer *fb,
199 int x, int y, bool update)
200{
201 struct nouveau_framebuffer *nvfb = nouveau_framebuffer(fb);
202 u32 *push;
203
Ben Skeggs438d99e2011-07-05 16:48:06 +1000204 push = evo_wait(fb->dev, 0, 16);
205 if (push) {
206 evo_mthd(push, 0x0460 + (nv_crtc->index * 0x300), 1);
207 evo_data(push, nvfb->nvbo->bo.offset >> 8);
208 evo_mthd(push, 0x0468 + (nv_crtc->index * 0x300), 4);
209 evo_data(push, (fb->height << 16) | fb->width);
210 evo_data(push, nvfb->r_pitch);
211 evo_data(push, nvfb->r_format);
Ben Skeggsc0cc92a2011-07-06 11:40:45 +1000212 evo_data(push, nvfb->r_dma);
Ben Skeggsc6f2f712011-07-08 12:11:58 +1000213 evo_mthd(push, 0x04b0 + (nv_crtc->index * 0x300), 1);
214 evo_data(push, (y << 16) | x);
Ben Skeggsa46232e2011-07-07 15:23:48 +1000215 if (update) {
216 evo_mthd(push, 0x0080, 1);
217 evo_data(push, 0x00000000);
218 }
Ben Skeggs438d99e2011-07-05 16:48:06 +1000219 evo_kick(push, fb->dev, 0);
220 }
221
Ben Skeggsc0cc92a2011-07-06 11:40:45 +1000222 nv_crtc->fb.tile_flags = nvfb->r_dma;
Ben Skeggs438d99e2011-07-05 16:48:06 +1000223 return 0;
224}
225
226static void
227nvd0_crtc_cursor_show(struct nouveau_crtc *nv_crtc, bool show, bool update)
228{
229 struct drm_device *dev = nv_crtc->base.dev;
230 u32 *push = evo_wait(dev, 0, 16);
231 if (push) {
232 if (show) {
233 evo_mthd(push, 0x0480 + (nv_crtc->index * 0x300), 2);
234 evo_data(push, 0x85000000);
235 evo_data(push, nv_crtc->cursor.nvbo->bo.offset >> 8);
236 evo_mthd(push, 0x048c + (nv_crtc->index * 0x300), 1);
Ben Skeggs37b034a2011-07-08 14:43:19 +1000237 evo_data(push, NvEvoVRAM);
Ben Skeggs438d99e2011-07-05 16:48:06 +1000238 } else {
239 evo_mthd(push, 0x0480 + (nv_crtc->index * 0x300), 1);
240 evo_data(push, 0x05000000);
241 evo_mthd(push, 0x048c + (nv_crtc->index * 0x300), 1);
242 evo_data(push, 0x00000000);
243 }
244
245 if (update) {
246 evo_mthd(push, 0x0080, 1);
247 evo_data(push, 0x00000000);
248 }
249
250 evo_kick(push, dev, 0);
251 }
252}
253
254static void
255nvd0_crtc_dpms(struct drm_crtc *crtc, int mode)
256{
257}
258
259static void
260nvd0_crtc_prepare(struct drm_crtc *crtc)
261{
262 struct nouveau_crtc *nv_crtc = nouveau_crtc(crtc);
263 u32 *push;
264
265 push = evo_wait(crtc->dev, 0, 2);
266 if (push) {
267 evo_mthd(push, 0x0474 + (nv_crtc->index * 0x300), 1);
268 evo_data(push, 0x00000000);
269 evo_mthd(push, 0x0440 + (nv_crtc->index * 0x300), 1);
270 evo_data(push, 0x03000000);
271 evo_mthd(push, 0x045c + (nv_crtc->index * 0x300), 1);
272 evo_data(push, 0x00000000);
273 evo_kick(push, crtc->dev, 0);
274 }
275
276 nvd0_crtc_cursor_show(nv_crtc, false, false);
277}
278
279static void
280nvd0_crtc_commit(struct drm_crtc *crtc)
281{
282 struct nouveau_crtc *nv_crtc = nouveau_crtc(crtc);
283 u32 *push;
284
285 push = evo_wait(crtc->dev, 0, 32);
286 if (push) {
287 evo_mthd(push, 0x0474 + (nv_crtc->index * 0x300), 1);
288 evo_data(push, nv_crtc->fb.tile_flags);
289 evo_mthd(push, 0x0440 + (nv_crtc->index * 0x300), 4);
290 evo_data(push, 0x83000000);
291 evo_data(push, nv_crtc->lut.nvbo->bo.offset >> 8);
292 evo_data(push, 0x00000000);
293 evo_data(push, 0x00000000);
294 evo_mthd(push, 0x045c + (nv_crtc->index * 0x300), 1);
Ben Skeggs37b034a2011-07-08 14:43:19 +1000295 evo_data(push, NvEvoVRAM);
Ben Skeggs8ea0d4a2011-07-07 14:49:24 +1000296 evo_mthd(push, 0x0430 + (nv_crtc->index * 0x300), 1);
297 evo_data(push, 0xffffff00);
Ben Skeggs438d99e2011-07-05 16:48:06 +1000298 evo_kick(push, crtc->dev, 0);
299 }
300
301 nvd0_crtc_cursor_show(nv_crtc, nv_crtc->cursor.visible, true);
302}
303
304static bool
305nvd0_crtc_mode_fixup(struct drm_crtc *crtc, struct drm_display_mode *mode,
306 struct drm_display_mode *adjusted_mode)
307{
308 return true;
309}
310
311static int
312nvd0_crtc_swap_fbs(struct drm_crtc *crtc, struct drm_framebuffer *old_fb)
313{
314 struct nouveau_framebuffer *nvfb = nouveau_framebuffer(crtc->fb);
315 int ret;
316
317 ret = nouveau_bo_pin(nvfb->nvbo, TTM_PL_FLAG_VRAM);
318 if (ret)
319 return ret;
320
321 if (old_fb) {
322 nvfb = nouveau_framebuffer(old_fb);
323 nouveau_bo_unpin(nvfb->nvbo);
324 }
325
326 return 0;
327}
328
329static int
330nvd0_crtc_mode_set(struct drm_crtc *crtc, struct drm_display_mode *umode,
331 struct drm_display_mode *mode, int x, int y,
332 struct drm_framebuffer *old_fb)
333{
334 struct nouveau_crtc *nv_crtc = nouveau_crtc(crtc);
335 struct nouveau_connector *nv_connector;
336 u32 htotal = mode->htotal;
337 u32 vtotal = mode->vtotal;
338 u32 hsyncw = mode->hsync_end - mode->hsync_start - 1;
339 u32 vsyncw = mode->vsync_end - mode->vsync_start - 1;
340 u32 hfrntp = mode->hsync_start - mode->hdisplay;
341 u32 vfrntp = mode->vsync_start - mode->vdisplay;
342 u32 hbackp = mode->htotal - mode->hsync_end;
343 u32 vbackp = mode->vtotal - mode->vsync_end;
344 u32 hss2be = hsyncw + hbackp;
345 u32 vss2be = vsyncw + vbackp;
346 u32 hss2de = htotal - hfrntp;
347 u32 vss2de = vtotal - vfrntp;
Ben Skeggs629c1b92011-07-08 09:43:20 +1000348 u32 syncs, *push;
Ben Skeggs438d99e2011-07-05 16:48:06 +1000349 int ret;
350
Ben Skeggs629c1b92011-07-08 09:43:20 +1000351 syncs = 0x00000001;
352 if (mode->flags & DRM_MODE_FLAG_NHSYNC)
353 syncs |= 0x00000008;
354 if (mode->flags & DRM_MODE_FLAG_NVSYNC)
355 syncs |= 0x00000010;
356
Ben Skeggs438d99e2011-07-05 16:48:06 +1000357 ret = nvd0_crtc_swap_fbs(crtc, old_fb);
358 if (ret)
359 return ret;
360
361 push = evo_wait(crtc->dev, 0, 64);
362 if (push) {
363 evo_mthd(push, 0x0410 + (nv_crtc->index * 0x300), 5);
Ben Skeggs629c1b92011-07-08 09:43:20 +1000364 evo_data(push, 0x00000000);
Ben Skeggs438d99e2011-07-05 16:48:06 +1000365 evo_data(push, (vtotal << 16) | htotal);
366 evo_data(push, (vsyncw << 16) | hsyncw);
367 evo_data(push, (vss2be << 16) | hss2be);
368 evo_data(push, (vss2de << 16) | hss2de);
369 evo_mthd(push, 0x042c + (nv_crtc->index * 0x300), 1);
370 evo_data(push, 0x00000000); /* ??? */
371 evo_mthd(push, 0x0450 + (nv_crtc->index * 0x300), 3);
372 evo_data(push, mode->clock * 1000);
373 evo_data(push, 0x00200000); /* ??? */
374 evo_data(push, mode->clock * 1000);
Ben Skeggs629c1b92011-07-08 09:43:20 +1000375 evo_mthd(push, 0x0404 + (nv_crtc->index * 0x300), 1);
376 evo_data(push, syncs);
Ben Skeggs438d99e2011-07-05 16:48:06 +1000377 evo_kick(push, crtc->dev, 0);
378 }
379
380 nv_connector = nouveau_crtc_connector_get(nv_crtc);
381 nvd0_crtc_set_dither(nv_crtc, nv_connector->use_dithering, false);
382 nvd0_crtc_set_scale(nv_crtc, nv_connector->scaling_mode, false);
383 nvd0_crtc_set_image(nv_crtc, crtc->fb, x, y, false);
384 return 0;
385}
386
387static int
388nvd0_crtc_mode_set_base(struct drm_crtc *crtc, int x, int y,
389 struct drm_framebuffer *old_fb)
390{
391 struct nouveau_crtc *nv_crtc = nouveau_crtc(crtc);
392 int ret;
393
394 ret = nvd0_crtc_swap_fbs(crtc, old_fb);
395 if (ret)
396 return ret;
397
398 nvd0_crtc_set_image(nv_crtc, crtc->fb, x, y, true);
399 return 0;
400}
401
402static int
403nvd0_crtc_mode_set_base_atomic(struct drm_crtc *crtc,
404 struct drm_framebuffer *fb, int x, int y,
405 enum mode_set_atomic state)
406{
407 struct nouveau_crtc *nv_crtc = nouveau_crtc(crtc);
408 nvd0_crtc_set_image(nv_crtc, fb, x, y, true);
409 return 0;
410}
411
412static void
413nvd0_crtc_lut_load(struct drm_crtc *crtc)
414{
415 struct nouveau_crtc *nv_crtc = nouveau_crtc(crtc);
416 void __iomem *lut = nvbo_kmap_obj_iovirtual(nv_crtc->lut.nvbo);
417 int i;
418
419 for (i = 0; i < 256; i++) {
Ben Skeggs8ea0d4a2011-07-07 14:49:24 +1000420 writew(0x6000 + (nv_crtc->lut.r[i] >> 2), lut + (i * 0x20) + 0);
421 writew(0x6000 + (nv_crtc->lut.g[i] >> 2), lut + (i * 0x20) + 2);
422 writew(0x6000 + (nv_crtc->lut.b[i] >> 2), lut + (i * 0x20) + 4);
Ben Skeggs438d99e2011-07-05 16:48:06 +1000423 }
424}
425
426static int
427nvd0_crtc_cursor_set(struct drm_crtc *crtc, struct drm_file *file_priv,
428 uint32_t handle, uint32_t width, uint32_t height)
429{
430 struct nouveau_crtc *nv_crtc = nouveau_crtc(crtc);
431 struct drm_device *dev = crtc->dev;
432 struct drm_gem_object *gem;
433 struct nouveau_bo *nvbo;
434 bool visible = (handle != 0);
435 int i, ret = 0;
436
437 if (visible) {
438 if (width != 64 || height != 64)
439 return -EINVAL;
440
441 gem = drm_gem_object_lookup(dev, file_priv, handle);
442 if (unlikely(!gem))
443 return -ENOENT;
444 nvbo = nouveau_gem_object(gem);
445
446 ret = nouveau_bo_map(nvbo);
447 if (ret == 0) {
448 for (i = 0; i < 64 * 64; i++) {
449 u32 v = nouveau_bo_rd32(nvbo, i);
450 nouveau_bo_wr32(nv_crtc->cursor.nvbo, i, v);
451 }
452 nouveau_bo_unmap(nvbo);
453 }
454
455 drm_gem_object_unreference_unlocked(gem);
456 }
457
458 if (visible != nv_crtc->cursor.visible) {
459 nvd0_crtc_cursor_show(nv_crtc, visible, true);
460 nv_crtc->cursor.visible = visible;
461 }
462
463 return ret;
464}
465
466static int
467nvd0_crtc_cursor_move(struct drm_crtc *crtc, int x, int y)
468{
469 struct nouveau_crtc *nv_crtc = nouveau_crtc(crtc);
470 const u32 data = (y << 16) | x;
471
472 nv_wr32(crtc->dev, 0x64d084 + (nv_crtc->index * 0x1000), data);
473 nv_wr32(crtc->dev, 0x64d080 + (nv_crtc->index * 0x1000), 0x00000000);
474 return 0;
475}
476
477static void
478nvd0_crtc_gamma_set(struct drm_crtc *crtc, u16 *r, u16 *g, u16 *b,
479 uint32_t start, uint32_t size)
480{
481 struct nouveau_crtc *nv_crtc = nouveau_crtc(crtc);
482 u32 end = max(start + size, (u32)256);
483 u32 i;
484
485 for (i = start; i < end; i++) {
486 nv_crtc->lut.r[i] = r[i];
487 nv_crtc->lut.g[i] = g[i];
488 nv_crtc->lut.b[i] = b[i];
489 }
490
491 nvd0_crtc_lut_load(crtc);
492}
493
494static void
495nvd0_crtc_destroy(struct drm_crtc *crtc)
496{
497 struct nouveau_crtc *nv_crtc = nouveau_crtc(crtc);
498 nouveau_bo_unmap(nv_crtc->cursor.nvbo);
499 nouveau_bo_ref(NULL, &nv_crtc->cursor.nvbo);
500 nouveau_bo_unmap(nv_crtc->lut.nvbo);
501 nouveau_bo_ref(NULL, &nv_crtc->lut.nvbo);
502 drm_crtc_cleanup(crtc);
503 kfree(crtc);
504}
505
506static const struct drm_crtc_helper_funcs nvd0_crtc_hfunc = {
507 .dpms = nvd0_crtc_dpms,
508 .prepare = nvd0_crtc_prepare,
509 .commit = nvd0_crtc_commit,
510 .mode_fixup = nvd0_crtc_mode_fixup,
511 .mode_set = nvd0_crtc_mode_set,
512 .mode_set_base = nvd0_crtc_mode_set_base,
513 .mode_set_base_atomic = nvd0_crtc_mode_set_base_atomic,
514 .load_lut = nvd0_crtc_lut_load,
515};
516
517static const struct drm_crtc_funcs nvd0_crtc_func = {
518 .cursor_set = nvd0_crtc_cursor_set,
519 .cursor_move = nvd0_crtc_cursor_move,
520 .gamma_set = nvd0_crtc_gamma_set,
521 .set_config = drm_crtc_helper_set_config,
522 .destroy = nvd0_crtc_destroy,
523};
524
525static int
526nvd0_crtc_create(struct drm_device *dev, int index)
527{
528 struct nouveau_crtc *nv_crtc;
529 struct drm_crtc *crtc;
530 int ret, i;
531
532 nv_crtc = kzalloc(sizeof(*nv_crtc), GFP_KERNEL);
533 if (!nv_crtc)
534 return -ENOMEM;
535
536 nv_crtc->index = index;
537 nv_crtc->set_dither = nvd0_crtc_set_dither;
538 nv_crtc->set_scale = nvd0_crtc_set_scale;
539 for (i = 0; i < 256; i++) {
540 nv_crtc->lut.r[i] = i << 8;
541 nv_crtc->lut.g[i] = i << 8;
542 nv_crtc->lut.b[i] = i << 8;
543 }
544
545 crtc = &nv_crtc->base;
546 drm_crtc_init(dev, crtc, &nvd0_crtc_func);
547 drm_crtc_helper_add(crtc, &nvd0_crtc_hfunc);
548 drm_mode_crtc_set_gamma_size(crtc, 256);
549
550 ret = nouveau_bo_new(dev, 64 * 64 * 4, 0x100, TTM_PL_FLAG_VRAM,
551 0, 0x0000, &nv_crtc->cursor.nvbo);
552 if (!ret) {
553 ret = nouveau_bo_pin(nv_crtc->cursor.nvbo, TTM_PL_FLAG_VRAM);
554 if (!ret)
555 ret = nouveau_bo_map(nv_crtc->cursor.nvbo);
556 if (ret)
557 nouveau_bo_ref(NULL, &nv_crtc->cursor.nvbo);
558 }
559
560 if (ret)
561 goto out;
562
Ben Skeggs8ea0d4a2011-07-07 14:49:24 +1000563 ret = nouveau_bo_new(dev, 8192, 0x100, TTM_PL_FLAG_VRAM,
Ben Skeggs438d99e2011-07-05 16:48:06 +1000564 0, 0x0000, &nv_crtc->lut.nvbo);
565 if (!ret) {
566 ret = nouveau_bo_pin(nv_crtc->lut.nvbo, TTM_PL_FLAG_VRAM);
567 if (!ret)
568 ret = nouveau_bo_map(nv_crtc->lut.nvbo);
569 if (ret)
570 nouveau_bo_ref(NULL, &nv_crtc->lut.nvbo);
571 }
572
573 if (ret)
574 goto out;
575
576 nvd0_crtc_lut_load(crtc);
577
578out:
579 if (ret)
580 nvd0_crtc_destroy(crtc);
581 return ret;
582}
583
584/******************************************************************************
Ben Skeggs26f6d882011-07-04 16:25:18 +1000585 * DAC
586 *****************************************************************************/
Ben Skeggs8eaa9662011-07-06 15:25:47 +1000587static void
588nvd0_dac_dpms(struct drm_encoder *encoder, int mode)
589{
590 struct nouveau_encoder *nv_encoder = nouveau_encoder(encoder);
591 struct drm_device *dev = encoder->dev;
592 int or = nv_encoder->or;
593 u32 dpms_ctrl;
594
595 dpms_ctrl = 0x80000000;
596 if (mode == DRM_MODE_DPMS_STANDBY || mode == DRM_MODE_DPMS_OFF)
597 dpms_ctrl |= 0x00000001;
598 if (mode == DRM_MODE_DPMS_SUSPEND || mode == DRM_MODE_DPMS_OFF)
599 dpms_ctrl |= 0x00000004;
600
601 nv_wait(dev, 0x61a004 + (or * 0x0800), 0x80000000, 0x00000000);
602 nv_mask(dev, 0x61a004 + (or * 0x0800), 0xc000007f, dpms_ctrl);
603 nv_wait(dev, 0x61a004 + (or * 0x0800), 0x80000000, 0x00000000);
604}
605
606static bool
607nvd0_dac_mode_fixup(struct drm_encoder *encoder, struct drm_display_mode *mode,
608 struct drm_display_mode *adjusted_mode)
609{
610 struct nouveau_encoder *nv_encoder = nouveau_encoder(encoder);
611 struct nouveau_connector *nv_connector;
612
613 nv_connector = nouveau_encoder_connector_get(nv_encoder);
614 if (nv_connector && nv_connector->native_mode) {
615 if (nv_connector->scaling_mode != DRM_MODE_SCALE_NONE) {
616 int id = adjusted_mode->base.id;
617 *adjusted_mode = *nv_connector->native_mode;
618 adjusted_mode->base.id = id;
619 }
620 }
621
622 return true;
623}
624
625static void
626nvd0_dac_prepare(struct drm_encoder *encoder)
627{
628}
629
630static void
631nvd0_dac_commit(struct drm_encoder *encoder)
632{
633}
634
635static void
636nvd0_dac_mode_set(struct drm_encoder *encoder, struct drm_display_mode *mode,
637 struct drm_display_mode *adjusted_mode)
638{
639 struct nouveau_encoder *nv_encoder = nouveau_encoder(encoder);
640 struct nouveau_crtc *nv_crtc = nouveau_crtc(encoder->crtc);
641 u32 *push;
642
643 nvd0_dac_dpms(encoder, DRM_MODE_DPMS_ON);
644
Ben Skeggsff8ff502011-07-08 11:53:37 +1000645 push = evo_wait(encoder->dev, 0, 4);
Ben Skeggs8eaa9662011-07-06 15:25:47 +1000646 if (push) {
Ben Skeggsff8ff502011-07-08 11:53:37 +1000647 evo_mthd(push, 0x0180 + (nv_encoder->or * 0x20), 2);
Ben Skeggs8eaa9662011-07-06 15:25:47 +1000648 evo_data(push, 1 << nv_crtc->index);
Ben Skeggsff8ff502011-07-08 11:53:37 +1000649 evo_data(push, 0x00ff);
Ben Skeggs8eaa9662011-07-06 15:25:47 +1000650 evo_kick(push, encoder->dev, 0);
651 }
652
653 nv_encoder->crtc = encoder->crtc;
654}
655
656static void
657nvd0_dac_disconnect(struct drm_encoder *encoder)
658{
659 struct nouveau_encoder *nv_encoder = nouveau_encoder(encoder);
660 struct drm_device *dev = encoder->dev;
661 u32 *push;
662
663 if (nv_encoder->crtc) {
664 nvd0_crtc_prepare(nv_encoder->crtc);
665
666 push = evo_wait(dev, 0, 4);
667 if (push) {
668 evo_mthd(push, 0x0180 + (nv_encoder->or * 0x20), 1);
669 evo_data(push, 0x00000000);
670 evo_mthd(push, 0x0080, 1);
671 evo_data(push, 0x00000000);
672 evo_kick(push, dev, 0);
673 }
674
675 nv_encoder->crtc = NULL;
676 }
677}
678
Ben Skeggsb6d8e7e2011-07-07 09:51:29 +1000679static enum drm_connector_status
680nvd0_dac_detect(struct drm_encoder *encoder, struct drm_connector *connector)
681{
Ben Skeggsb6819932011-07-08 11:14:50 +1000682 enum drm_connector_status status = connector_status_disconnected;
683 struct nouveau_encoder *nv_encoder = nouveau_encoder(encoder);
684 struct drm_device *dev = encoder->dev;
685 int or = nv_encoder->or;
686 u32 load;
687
688 nv_wr32(dev, 0x61a00c + (or * 0x800), 0x00100000);
689 udelay(9500);
690 nv_wr32(dev, 0x61a00c + (or * 0x800), 0x80000000);
691
692 load = nv_rd32(dev, 0x61a00c + (or * 0x800));
693 if ((load & 0x38000000) == 0x38000000)
694 status = connector_status_connected;
695
696 nv_wr32(dev, 0x61a00c + (or * 0x800), 0x00000000);
697 return status;
Ben Skeggsb6d8e7e2011-07-07 09:51:29 +1000698}
699
Ben Skeggs8eaa9662011-07-06 15:25:47 +1000700static void
701nvd0_dac_destroy(struct drm_encoder *encoder)
702{
703 drm_encoder_cleanup(encoder);
704 kfree(encoder);
705}
706
707static const struct drm_encoder_helper_funcs nvd0_dac_hfunc = {
708 .dpms = nvd0_dac_dpms,
709 .mode_fixup = nvd0_dac_mode_fixup,
710 .prepare = nvd0_dac_prepare,
711 .commit = nvd0_dac_commit,
712 .mode_set = nvd0_dac_mode_set,
713 .disable = nvd0_dac_disconnect,
714 .get_crtc = nvd0_display_crtc_get,
Ben Skeggsb6d8e7e2011-07-07 09:51:29 +1000715 .detect = nvd0_dac_detect
Ben Skeggs8eaa9662011-07-06 15:25:47 +1000716};
717
718static const struct drm_encoder_funcs nvd0_dac_func = {
719 .destroy = nvd0_dac_destroy,
720};
721
722static int
723nvd0_dac_create(struct drm_connector *connector, struct dcb_entry *dcbe)
724{
725 struct drm_device *dev = connector->dev;
726 struct nouveau_encoder *nv_encoder;
727 struct drm_encoder *encoder;
728
729 nv_encoder = kzalloc(sizeof(*nv_encoder), GFP_KERNEL);
730 if (!nv_encoder)
731 return -ENOMEM;
732 nv_encoder->dcb = dcbe;
733 nv_encoder->or = ffs(dcbe->or) - 1;
734
735 encoder = to_drm_encoder(nv_encoder);
736 encoder->possible_crtcs = dcbe->heads;
737 encoder->possible_clones = 0;
738 drm_encoder_init(dev, encoder, &nvd0_dac_func, DRM_MODE_ENCODER_DAC);
739 drm_encoder_helper_add(encoder, &nvd0_dac_hfunc);
740
741 drm_mode_connector_attach_encoder(connector, encoder);
742 return 0;
743}
Ben Skeggs26f6d882011-07-04 16:25:18 +1000744
745/******************************************************************************
746 * SOR
747 *****************************************************************************/
Ben Skeggs83fc0832011-07-05 13:08:40 +1000748static void
749nvd0_sor_dpms(struct drm_encoder *encoder, int mode)
750{
751 struct nouveau_encoder *nv_encoder = nouveau_encoder(encoder);
752 struct drm_device *dev = encoder->dev;
753 struct drm_encoder *partner;
754 int or = nv_encoder->or;
755 u32 dpms_ctrl;
756
757 nv_encoder->last_dpms = mode;
758
759 list_for_each_entry(partner, &dev->mode_config.encoder_list, head) {
760 struct nouveau_encoder *nv_partner = nouveau_encoder(partner);
761
762 if (partner->encoder_type != DRM_MODE_ENCODER_TMDS)
763 continue;
764
765 if (nv_partner != nv_encoder &&
766 nv_partner->dcb->or == nv_encoder->or) {
767 if (nv_partner->last_dpms == DRM_MODE_DPMS_ON)
768 return;
769 break;
770 }
771 }
772
773 dpms_ctrl = (mode == DRM_MODE_DPMS_ON);
774 dpms_ctrl |= 0x80000000;
775
776 nv_wait(dev, 0x61c004 + (or * 0x0800), 0x80000000, 0x00000000);
777 nv_mask(dev, 0x61c004 + (or * 0x0800), 0x80000001, dpms_ctrl);
778 nv_wait(dev, 0x61c004 + (or * 0x0800), 0x80000000, 0x00000000);
779 nv_wait(dev, 0x61c030 + (or * 0x0800), 0x10000000, 0x00000000);
780}
781
782static bool
783nvd0_sor_mode_fixup(struct drm_encoder *encoder, struct drm_display_mode *mode,
784 struct drm_display_mode *adjusted_mode)
785{
786 struct nouveau_encoder *nv_encoder = nouveau_encoder(encoder);
787 struct nouveau_connector *nv_connector;
788
789 nv_connector = nouveau_encoder_connector_get(nv_encoder);
790 if (nv_connector && nv_connector->native_mode) {
791 if (nv_connector->scaling_mode != DRM_MODE_SCALE_NONE) {
792 int id = adjusted_mode->base.id;
793 *adjusted_mode = *nv_connector->native_mode;
794 adjusted_mode->base.id = id;
795 }
796 }
797
798 return true;
799}
800
801static void
802nvd0_sor_prepare(struct drm_encoder *encoder)
803{
804}
805
806static void
807nvd0_sor_commit(struct drm_encoder *encoder)
808{
809}
810
811static void
Ben Skeggs3b6d83d12011-07-08 12:52:14 +1000812nvd0_sor_mode_set(struct drm_encoder *encoder, struct drm_display_mode *umode,
813 struct drm_display_mode *mode)
Ben Skeggs83fc0832011-07-05 13:08:40 +1000814{
Ben Skeggs3b6d83d12011-07-08 12:52:14 +1000815 struct drm_nouveau_private *dev_priv = encoder->dev->dev_private;
Ben Skeggs83fc0832011-07-05 13:08:40 +1000816 struct nouveau_encoder *nv_encoder = nouveau_encoder(encoder);
817 struct nouveau_crtc *nv_crtc = nouveau_crtc(encoder->crtc);
Ben Skeggs3b6d83d12011-07-08 12:52:14 +1000818 struct nouveau_connector *nv_connector;
819 struct nvbios *bios = &dev_priv->vbios;
Ben Skeggs83fc0832011-07-05 13:08:40 +1000820 u32 mode_ctrl = (1 << nv_crtc->index);
Ben Skeggsff8ff502011-07-08 11:53:37 +1000821 u32 *push, or_config;
Ben Skeggs83fc0832011-07-05 13:08:40 +1000822
Ben Skeggs3b6d83d12011-07-08 12:52:14 +1000823 nv_connector = nouveau_encoder_connector_get(nv_encoder);
824 switch (nv_encoder->dcb->type) {
825 case OUTPUT_TMDS:
826 if (nv_encoder->dcb->sorconf.link & 1) {
827 if (mode->clock < 165000)
828 mode_ctrl |= 0x00000100;
829 else
830 mode_ctrl |= 0x00000500;
831 } else {
832 mode_ctrl |= 0x00000200;
833 }
Ben Skeggs83fc0832011-07-05 13:08:40 +1000834
Ben Skeggs3b6d83d12011-07-08 12:52:14 +1000835 or_config = (mode_ctrl & 0x00000f00) >> 8;
836 if (mode->clock >= 165000)
837 or_config |= 0x0100;
838 break;
839 case OUTPUT_LVDS:
840 or_config = (mode_ctrl & 0x00000f00) >> 8;
841 if (bios->fp_no_ddc) {
842 if (bios->fp.dual_link)
843 or_config |= 0x0100;
844 if (bios->fp.if_is_24bit)
845 or_config |= 0x0200;
846 } else {
847 if (nv_connector->dcb->type == DCB_CONNECTOR_LVDS_SPWG) {
848 if (((u8 *)nv_connector->edid)[121] == 2)
849 or_config |= 0x0100;
850 } else
851 if (mode->clock >= bios->fp.duallink_transition_clk) {
852 or_config |= 0x0100;
853 }
854
855 if (or_config & 0x0100) {
856 if (bios->fp.strapless_is_24bit & 2)
857 or_config |= 0x0200;
858 } else {
859 if (bios->fp.strapless_is_24bit & 1)
860 or_config |= 0x0200;
861 }
862
863 if (nv_connector->base.display_info.bpc == 8)
864 or_config |= 0x0200;
865
866 }
867 break;
868 default:
869 BUG_ON(1);
870 break;
871 }
Ben Skeggsff8ff502011-07-08 11:53:37 +1000872
Ben Skeggs83fc0832011-07-05 13:08:40 +1000873 nvd0_sor_dpms(encoder, DRM_MODE_DPMS_ON);
874
Ben Skeggsff8ff502011-07-08 11:53:37 +1000875 push = evo_wait(encoder->dev, 0, 4);
Ben Skeggs83fc0832011-07-05 13:08:40 +1000876 if (push) {
Ben Skeggsff8ff502011-07-08 11:53:37 +1000877 evo_mthd(push, 0x0200 + (nv_encoder->or * 0x20), 2);
Ben Skeggs83fc0832011-07-05 13:08:40 +1000878 evo_data(push, mode_ctrl);
Ben Skeggsff8ff502011-07-08 11:53:37 +1000879 evo_data(push, or_config);
Ben Skeggs438d99e2011-07-05 16:48:06 +1000880 evo_kick(push, encoder->dev, 0);
Ben Skeggs83fc0832011-07-05 13:08:40 +1000881 }
882
883 nv_encoder->crtc = encoder->crtc;
884}
885
886static void
887nvd0_sor_disconnect(struct drm_encoder *encoder)
888{
889 struct nouveau_encoder *nv_encoder = nouveau_encoder(encoder);
890 struct drm_device *dev = encoder->dev;
Ben Skeggs438d99e2011-07-05 16:48:06 +1000891 u32 *push;
Ben Skeggs83fc0832011-07-05 13:08:40 +1000892
893 if (nv_encoder->crtc) {
Ben Skeggs438d99e2011-07-05 16:48:06 +1000894 nvd0_crtc_prepare(nv_encoder->crtc);
895
896 push = evo_wait(dev, 0, 4);
Ben Skeggs83fc0832011-07-05 13:08:40 +1000897 if (push) {
898 evo_mthd(push, 0x0200 + (nv_encoder->or * 0x20), 1);
899 evo_data(push, 0x00000000);
900 evo_mthd(push, 0x0080, 1);
901 evo_data(push, 0x00000000);
902 evo_kick(push, dev, 0);
903 }
904
905 nv_encoder->crtc = NULL;
906 nv_encoder->last_dpms = DRM_MODE_DPMS_OFF;
907 }
908}
909
910static void
911nvd0_sor_destroy(struct drm_encoder *encoder)
912{
913 drm_encoder_cleanup(encoder);
914 kfree(encoder);
915}
916
917static const struct drm_encoder_helper_funcs nvd0_sor_hfunc = {
918 .dpms = nvd0_sor_dpms,
919 .mode_fixup = nvd0_sor_mode_fixup,
920 .prepare = nvd0_sor_prepare,
921 .commit = nvd0_sor_commit,
922 .mode_set = nvd0_sor_mode_set,
923 .disable = nvd0_sor_disconnect,
924 .get_crtc = nvd0_display_crtc_get,
925};
926
927static const struct drm_encoder_funcs nvd0_sor_func = {
928 .destroy = nvd0_sor_destroy,
929};
930
931static int
932nvd0_sor_create(struct drm_connector *connector, struct dcb_entry *dcbe)
933{
934 struct drm_device *dev = connector->dev;
935 struct nouveau_encoder *nv_encoder;
936 struct drm_encoder *encoder;
937
938 nv_encoder = kzalloc(sizeof(*nv_encoder), GFP_KERNEL);
939 if (!nv_encoder)
940 return -ENOMEM;
941 nv_encoder->dcb = dcbe;
942 nv_encoder->or = ffs(dcbe->or) - 1;
943 nv_encoder->last_dpms = DRM_MODE_DPMS_OFF;
944
945 encoder = to_drm_encoder(nv_encoder);
946 encoder->possible_crtcs = dcbe->heads;
947 encoder->possible_clones = 0;
948 drm_encoder_init(dev, encoder, &nvd0_sor_func, DRM_MODE_ENCODER_TMDS);
949 drm_encoder_helper_add(encoder, &nvd0_sor_hfunc);
950
951 drm_mode_connector_attach_encoder(connector, encoder);
952 return 0;
953}
Ben Skeggs26f6d882011-07-04 16:25:18 +1000954
955/******************************************************************************
956 * IRQ
957 *****************************************************************************/
Ben Skeggs3a89cd02011-07-07 10:47:10 +1000958static struct dcb_entry *
959lookup_dcb(struct drm_device *dev, int id, u32 mc)
960{
961 struct drm_nouveau_private *dev_priv = dev->dev_private;
962 int type, or, i;
963
964 if (id < 4) {
965 type = OUTPUT_ANALOG;
966 or = id;
967 } else {
Ben Skeggs3b6d83d12011-07-08 12:52:14 +1000968 switch (mc & 0x00000f00) {
969 case 0x00000000: type = OUTPUT_LVDS; break;
970 case 0x00000100: type = OUTPUT_TMDS; break;
971 case 0x00000200: type = OUTPUT_TMDS; break;
972 case 0x00000500: type = OUTPUT_TMDS; break;
973 default:
Ben Skeggsee417792011-07-08 14:34:45 +1000974 NV_ERROR(dev, "PDISP: unknown SOR mc 0x%08x\n", mc);
Ben Skeggs3b6d83d12011-07-08 12:52:14 +1000975 return NULL;
976 }
977
978 or = id - 4;
Ben Skeggs3a89cd02011-07-07 10:47:10 +1000979 }
980
981 for (i = 0; i < dev_priv->vbios.dcb.entries; i++) {
982 struct dcb_entry *dcb = &dev_priv->vbios.dcb.entry[i];
983 if (dcb->type == type && (dcb->or & (1 << or)))
984 return dcb;
985 }
986
Ben Skeggsee417792011-07-08 14:34:45 +1000987 NV_ERROR(dev, "PDISP: DCB for %d/0x%08x not found\n", id, mc);
Ben Skeggs3a89cd02011-07-07 10:47:10 +1000988 return NULL;
989}
990
Ben Skeggs46005222011-07-05 11:01:13 +1000991static void
Ben Skeggs37b034a2011-07-08 14:43:19 +1000992nvd0_display_unk1_handler(struct drm_device *dev, u32 crtc, u32 mask)
Ben Skeggs270a5742011-07-05 14:16:05 +1000993{
Ben Skeggs3a89cd02011-07-07 10:47:10 +1000994 struct dcb_entry *dcb;
Ben Skeggs3a89cd02011-07-07 10:47:10 +1000995 int i;
996
Ben Skeggsee417792011-07-08 14:34:45 +1000997 for (i = 0; mask && i < 8; i++) {
Ben Skeggs3a89cd02011-07-07 10:47:10 +1000998 u32 mcc = nv_rd32(dev, 0x640180 + (i * 0x20));
Ben Skeggsee417792011-07-08 14:34:45 +1000999 if (!(mcc & (1 << crtc)))
1000 continue;
Ben Skeggs3a89cd02011-07-07 10:47:10 +10001001
Ben Skeggsee417792011-07-08 14:34:45 +10001002 dcb = lookup_dcb(dev, i, mcc);
1003 if (!dcb)
1004 continue;
Ben Skeggs3a89cd02011-07-07 10:47:10 +10001005
Ben Skeggs3a89cd02011-07-07 10:47:10 +10001006 nouveau_bios_run_display_table(dev, 0x0000, -1, dcb, crtc);
Ben Skeggsee417792011-07-08 14:34:45 +10001007 }
Ben Skeggs3a89cd02011-07-07 10:47:10 +10001008
Ben Skeggs270a5742011-07-05 14:16:05 +10001009 nv_wr32(dev, 0x6101d4, 0x00000000);
1010 nv_wr32(dev, 0x6109d4, 0x00000000);
1011 nv_wr32(dev, 0x6101d0, 0x80000000);
1012}
1013
1014static void
Ben Skeggs37b034a2011-07-08 14:43:19 +10001015nvd0_display_unk2_handler(struct drm_device *dev, u32 crtc, u32 mask)
Ben Skeggs270a5742011-07-05 14:16:05 +10001016{
Ben Skeggs3a89cd02011-07-07 10:47:10 +10001017 struct dcb_entry *dcb;
Ben Skeggs37b034a2011-07-08 14:43:19 +10001018 u32 or, tmp, pclk;
Ben Skeggsee417792011-07-08 14:34:45 +10001019 int i;
Ben Skeggs3a89cd02011-07-07 10:47:10 +10001020
Ben Skeggsee417792011-07-08 14:34:45 +10001021 for (i = 0; mask && i < 8; i++) {
1022 u32 mcc = nv_rd32(dev, 0x640180 + (i * 0x20));
1023 if (!(mcc & (1 << crtc)))
1024 continue;
1025
1026 dcb = lookup_dcb(dev, i, mcc);
1027 if (!dcb)
1028 continue;
1029
Ben Skeggs3a89cd02011-07-07 10:47:10 +10001030 nouveau_bios_run_display_table(dev, 0x0000, -2, dcb, crtc);
Ben Skeggsee417792011-07-08 14:34:45 +10001031 }
Ben Skeggs3a89cd02011-07-07 10:47:10 +10001032
Ben Skeggsee417792011-07-08 14:34:45 +10001033 pclk = nv_rd32(dev, 0x660450 + (crtc * 0x300)) / 1000;
1034 if (mask & 0x00010000) {
1035 nv50_crtc_set_clock(dev, crtc, pclk);
1036 }
Ben Skeggs3a89cd02011-07-07 10:47:10 +10001037
Ben Skeggsee417792011-07-08 14:34:45 +10001038 for (i = 0; mask && i < 8; i++) {
1039 u32 mcp = nv_rd32(dev, 0x660180 + (i * 0x20));
1040 u32 cfg = nv_rd32(dev, 0x660184 + (i * 0x20));
1041 if (!(mcp & (1 << crtc)))
1042 continue;
Ben Skeggs3a89cd02011-07-07 10:47:10 +10001043
Ben Skeggsee417792011-07-08 14:34:45 +10001044 dcb = lookup_dcb(dev, i, mcp);
1045 if (!dcb)
1046 continue;
1047 or = ffs(dcb->or) - 1;
Ben Skeggs3a89cd02011-07-07 10:47:10 +10001048
Ben Skeggsee417792011-07-08 14:34:45 +10001049 nouveau_bios_run_display_table(dev, cfg, pclk, dcb, crtc);
Ben Skeggs3a89cd02011-07-07 10:47:10 +10001050
Ben Skeggsee417792011-07-08 14:34:45 +10001051 nv_wr32(dev, 0x612200 + (crtc * 0x800), 0x00000000);
1052 switch (dcb->type) {
1053 case OUTPUT_ANALOG:
1054 nv_wr32(dev, 0x612280 + (or * 0x800), 0x00000000);
1055 break;
1056 case OUTPUT_TMDS:
1057 case OUTPUT_LVDS:
1058 if (cfg & 0x00000100)
1059 tmp = 0x00000101;
1060 else
1061 tmp = 0x00000000;
1062
1063 nv_mask(dev, 0x612300 + (or * 0x800), 0x00000707, tmp);
1064 break;
1065 default:
1066 break;
1067 }
1068
Ben Skeggs3a89cd02011-07-07 10:47:10 +10001069 break;
1070 }
1071
Ben Skeggs270a5742011-07-05 14:16:05 +10001072 nv_wr32(dev, 0x6101d4, 0x00000000);
1073 nv_wr32(dev, 0x6109d4, 0x00000000);
1074 nv_wr32(dev, 0x6101d0, 0x80000000);
1075}
1076
1077static void
Ben Skeggs37b034a2011-07-08 14:43:19 +10001078nvd0_display_unk4_handler(struct drm_device *dev, u32 crtc, u32 mask)
Ben Skeggs270a5742011-07-05 14:16:05 +10001079{
Ben Skeggs3a89cd02011-07-07 10:47:10 +10001080 struct dcb_entry *dcb;
Ben Skeggsee417792011-07-08 14:34:45 +10001081 int pclk, i;
Ben Skeggs3a89cd02011-07-07 10:47:10 +10001082
Ben Skeggsee417792011-07-08 14:34:45 +10001083 pclk = nv_rd32(dev, 0x660450 + (crtc * 0x300)) / 1000;
Ben Skeggs3a89cd02011-07-07 10:47:10 +10001084
Ben Skeggsee417792011-07-08 14:34:45 +10001085 for (i = 0; mask && i < 8; i++) {
1086 u32 mcp = nv_rd32(dev, 0x660180 + (i * 0x20));
1087 u32 cfg = nv_rd32(dev, 0x660184 + (i * 0x20));
1088 if (!(mcp & (1 << crtc)))
1089 continue;
Ben Skeggs3a89cd02011-07-07 10:47:10 +10001090
Ben Skeggsee417792011-07-08 14:34:45 +10001091 dcb = lookup_dcb(dev, i, mcp);
1092 if (!dcb)
1093 continue;
1094
1095 nouveau_bios_run_display_table(dev, cfg, -pclk, dcb, crtc);
1096 }
1097
Ben Skeggs270a5742011-07-05 14:16:05 +10001098 nv_wr32(dev, 0x6101d4, 0x00000000);
1099 nv_wr32(dev, 0x6109d4, 0x00000000);
1100 nv_wr32(dev, 0x6101d0, 0x80000000);
1101}
1102
1103static void
Ben Skeggsf20ce962011-07-08 13:17:01 +10001104nvd0_display_bh(unsigned long data)
1105{
1106 struct drm_device *dev = (struct drm_device *)data;
1107 struct nvd0_display *disp = nvd0_display(dev);
Ben Skeggs37b034a2011-07-08 14:43:19 +10001108 u32 mask, crtc;
1109 int i;
1110
1111 if (drm_debug & (DRM_UT_DRIVER | DRM_UT_KMS)) {
1112 NV_INFO(dev, "PDISP: modeset req %d\n", disp->modeset);
1113 NV_INFO(dev, " STAT: 0x%08x 0x%08x 0x%08x\n",
1114 nv_rd32(dev, 0x6101d0),
1115 nv_rd32(dev, 0x6101d4), nv_rd32(dev, 0x6109d4));
1116 for (i = 0; i < 8; i++) {
1117 NV_INFO(dev, " %s%d: 0x%08x 0x%08x\n",
1118 i < 4 ? "DAC" : "SOR", i,
1119 nv_rd32(dev, 0x640180 + (i * 0x20)),
1120 nv_rd32(dev, 0x660180 + (i * 0x20)));
1121 }
1122 }
1123
1124 mask = nv_rd32(dev, 0x6101d4);
1125 crtc = 0;
1126 if (!mask) {
1127 mask = nv_rd32(dev, 0x6109d4);
1128 crtc = 1;
1129 }
Ben Skeggsf20ce962011-07-08 13:17:01 +10001130
Ben Skeggsee417792011-07-08 14:34:45 +10001131 if (disp->modeset & 0x00000001)
Ben Skeggs37b034a2011-07-08 14:43:19 +10001132 nvd0_display_unk1_handler(dev, crtc, mask);
Ben Skeggsee417792011-07-08 14:34:45 +10001133 if (disp->modeset & 0x00000002)
Ben Skeggs37b034a2011-07-08 14:43:19 +10001134 nvd0_display_unk2_handler(dev, crtc, mask);
Ben Skeggsee417792011-07-08 14:34:45 +10001135 if (disp->modeset & 0x00000004)
Ben Skeggs37b034a2011-07-08 14:43:19 +10001136 nvd0_display_unk4_handler(dev, crtc, mask);
Ben Skeggsf20ce962011-07-08 13:17:01 +10001137}
1138
1139static void
Ben Skeggs46005222011-07-05 11:01:13 +10001140nvd0_display_intr(struct drm_device *dev)
1141{
Ben Skeggsf20ce962011-07-08 13:17:01 +10001142 struct nvd0_display *disp = nvd0_display(dev);
Ben Skeggs46005222011-07-05 11:01:13 +10001143 u32 intr = nv_rd32(dev, 0x610088);
1144
1145 if (intr & 0x00000002) {
1146 u32 stat = nv_rd32(dev, 0x61009c);
1147 int chid = ffs(stat) - 1;
1148 if (chid >= 0) {
1149 u32 mthd = nv_rd32(dev, 0x6101f0 + (chid * 12));
1150 u32 data = nv_rd32(dev, 0x6101f4 + (chid * 12));
1151 u32 unkn = nv_rd32(dev, 0x6101f8 + (chid * 12));
1152
1153 NV_INFO(dev, "EvoCh: chid %d mthd 0x%04x data 0x%08x "
1154 "0x%08x 0x%08x\n",
1155 chid, (mthd & 0x0000ffc), data, mthd, unkn);
1156 nv_wr32(dev, 0x61009c, (1 << chid));
1157 nv_wr32(dev, 0x6101f0 + (chid * 12), 0x90000000);
1158 }
1159
1160 intr &= ~0x00000002;
1161 }
1162
Ben Skeggs270a5742011-07-05 14:16:05 +10001163 if (intr & 0x00100000) {
1164 u32 stat = nv_rd32(dev, 0x6100ac);
1165
1166 if (stat & 0x00000007) {
Ben Skeggsee417792011-07-08 14:34:45 +10001167 disp->modeset = stat;
Ben Skeggsf20ce962011-07-08 13:17:01 +10001168 tasklet_schedule(&disp->tasklet);
Ben Skeggs270a5742011-07-05 14:16:05 +10001169
Ben Skeggsf20ce962011-07-08 13:17:01 +10001170 nv_wr32(dev, 0x6100ac, (stat & 0x00000007));
Ben Skeggs270a5742011-07-05 14:16:05 +10001171 stat &= ~0x00000007;
1172 }
1173
1174 if (stat) {
1175 NV_INFO(dev, "PDISP: unknown intr24 0x%08x\n", stat);
1176 nv_wr32(dev, 0x6100ac, stat);
1177 }
1178
1179 intr &= ~0x00100000;
1180 }
1181
Ben Skeggs46005222011-07-05 11:01:13 +10001182 if (intr & 0x01000000) {
1183 u32 stat = nv_rd32(dev, 0x6100bc);
1184 nv_wr32(dev, 0x6100bc, stat);
1185 intr &= ~0x01000000;
1186 }
1187
1188 if (intr & 0x02000000) {
1189 u32 stat = nv_rd32(dev, 0x6108bc);
1190 nv_wr32(dev, 0x6108bc, stat);
1191 intr &= ~0x02000000;
1192 }
1193
1194 if (intr)
1195 NV_INFO(dev, "PDISP: unknown intr 0x%08x\n", intr);
1196}
Ben Skeggs26f6d882011-07-04 16:25:18 +10001197
1198/******************************************************************************
1199 * Init
1200 *****************************************************************************/
1201static void
1202nvd0_display_fini(struct drm_device *dev)
1203{
1204 int i;
1205
1206 /* fini cursors */
1207 for (i = 14; i >= 13; i--) {
1208 if (!(nv_rd32(dev, 0x610490 + (i * 0x10)) & 0x00000001))
1209 continue;
1210
1211 nv_mask(dev, 0x610490 + (i * 0x10), 0x00000001, 0x00000000);
1212 nv_wait(dev, 0x610490 + (i * 0x10), 0x00010000, 0x00000000);
1213 nv_mask(dev, 0x610090, 1 << i, 0x00000000);
1214 nv_mask(dev, 0x6100a0, 1 << i, 0x00000000);
1215 }
1216
1217 /* fini master */
1218 if (nv_rd32(dev, 0x610490) & 0x00000010) {
1219 nv_mask(dev, 0x610490, 0x00000010, 0x00000000);
1220 nv_mask(dev, 0x610490, 0x00000003, 0x00000000);
1221 nv_wait(dev, 0x610490, 0x80000000, 0x00000000);
1222 nv_mask(dev, 0x610090, 0x00000001, 0x00000000);
1223 nv_mask(dev, 0x6100a0, 0x00000001, 0x00000000);
1224 }
1225}
1226
1227int
1228nvd0_display_init(struct drm_device *dev)
1229{
1230 struct nvd0_display *disp = nvd0_display(dev);
Ben Skeggsefd272a2011-07-05 11:58:58 +10001231 u32 *push;
Ben Skeggs26f6d882011-07-04 16:25:18 +10001232 int i;
1233
1234 if (nv_rd32(dev, 0x6100ac) & 0x00000100) {
1235 nv_wr32(dev, 0x6100ac, 0x00000100);
1236 nv_mask(dev, 0x6194e8, 0x00000001, 0x00000000);
1237 if (!nv_wait(dev, 0x6194e8, 0x00000002, 0x00000000)) {
1238 NV_ERROR(dev, "PDISP: 0x6194e8 0x%08x\n",
1239 nv_rd32(dev, 0x6194e8));
1240 return -EBUSY;
1241 }
1242 }
1243
Ben Skeggsa36f04c2011-07-06 14:39:23 +10001244 /* nfi what these are exactly, i do know that SOR_MODE_CTRL won't
1245 * work at all unless you do the SOR part below.
1246 */
1247 for (i = 0; i < 3; i++) {
1248 u32 dac = nv_rd32(dev, 0x61a000 + (i * 0x800));
1249 nv_wr32(dev, 0x6101c0 + (i * 0x800), dac);
1250 }
1251
1252 for (i = 0; i < 4; i++) {
1253 u32 sor = nv_rd32(dev, 0x61c000 + (i * 0x800));
1254 nv_wr32(dev, 0x6301c4 + (i * 0x800), sor);
1255 }
1256
1257 for (i = 0; i < 2; i++) {
1258 u32 crtc0 = nv_rd32(dev, 0x616104 + (i * 0x800));
1259 u32 crtc1 = nv_rd32(dev, 0x616108 + (i * 0x800));
1260 u32 crtc2 = nv_rd32(dev, 0x61610c + (i * 0x800));
1261 nv_wr32(dev, 0x6101b4 + (i * 0x800), crtc0);
1262 nv_wr32(dev, 0x6101b8 + (i * 0x800), crtc1);
1263 nv_wr32(dev, 0x6101bc + (i * 0x800), crtc2);
1264 }
1265
1266 /* point at our hash table / objects, enable interrupts */
Ben Skeggs26f6d882011-07-04 16:25:18 +10001267 nv_wr32(dev, 0x610010, (disp->mem->vinst >> 8) | 9);
Ben Skeggs270a5742011-07-05 14:16:05 +10001268 nv_mask(dev, 0x6100b0, 0x00000307, 0x00000307);
Ben Skeggs26f6d882011-07-04 16:25:18 +10001269
1270 /* init master */
Ben Skeggs51beb422011-07-05 10:33:08 +10001271 nv_wr32(dev, 0x610494, (disp->evo[0].handle >> 8) | 3);
Ben Skeggs26f6d882011-07-04 16:25:18 +10001272 nv_wr32(dev, 0x610498, 0x00010000);
Ben Skeggsefd272a2011-07-05 11:58:58 +10001273 nv_wr32(dev, 0x61049c, 0x00000001);
Ben Skeggs26f6d882011-07-04 16:25:18 +10001274 nv_mask(dev, 0x610490, 0x00000010, 0x00000010);
1275 nv_wr32(dev, 0x640000, 0x00000000);
1276 nv_wr32(dev, 0x610490, 0x01000013);
1277 if (!nv_wait(dev, 0x610490, 0x80000000, 0x00000000)) {
1278 NV_ERROR(dev, "PDISP: master 0x%08x\n",
1279 nv_rd32(dev, 0x610490));
1280 return -EBUSY;
1281 }
1282 nv_mask(dev, 0x610090, 0x00000001, 0x00000001);
1283 nv_mask(dev, 0x6100a0, 0x00000001, 0x00000001);
1284
1285 /* init cursors */
1286 for (i = 13; i <= 14; i++) {
1287 nv_wr32(dev, 0x610490 + (i * 0x10), 0x00000001);
1288 if (!nv_wait(dev, 0x610490 + (i * 0x10), 0x00010000, 0x00010000)) {
1289 NV_ERROR(dev, "PDISP: curs%d 0x%08x\n", i,
1290 nv_rd32(dev, 0x610490 + (i * 0x10)));
1291 return -EBUSY;
1292 }
1293
1294 nv_mask(dev, 0x610090, 1 << i, 1 << i);
1295 nv_mask(dev, 0x6100a0, 1 << i, 1 << i);
1296 }
1297
Ben Skeggsefd272a2011-07-05 11:58:58 +10001298 push = evo_wait(dev, 0, 32);
1299 if (!push)
1300 return -EBUSY;
1301 evo_mthd(push, 0x0088, 1);
Ben Skeggs37b034a2011-07-08 14:43:19 +10001302 evo_data(push, NvEvoSync);
Ben Skeggsefd272a2011-07-05 11:58:58 +10001303 evo_mthd(push, 0x0084, 1);
1304 evo_data(push, 0x00000000);
1305 evo_mthd(push, 0x0084, 1);
1306 evo_data(push, 0x80000000);
1307 evo_mthd(push, 0x008c, 1);
1308 evo_data(push, 0x00000000);
1309 evo_kick(push, dev, 0);
1310
Ben Skeggs26f6d882011-07-04 16:25:18 +10001311 return 0;
1312}
1313
1314void
1315nvd0_display_destroy(struct drm_device *dev)
1316{
1317 struct drm_nouveau_private *dev_priv = dev->dev_private;
1318 struct nvd0_display *disp = nvd0_display(dev);
Ben Skeggs51beb422011-07-05 10:33:08 +10001319 struct pci_dev *pdev = dev->pdev;
Ben Skeggs26f6d882011-07-04 16:25:18 +10001320
1321 nvd0_display_fini(dev);
1322
Ben Skeggs51beb422011-07-05 10:33:08 +10001323 pci_free_consistent(pdev, PAGE_SIZE, disp->evo[0].ptr, disp->evo[0].handle);
Ben Skeggs26f6d882011-07-04 16:25:18 +10001324 nouveau_gpuobj_ref(NULL, &disp->mem);
Ben Skeggs46005222011-07-05 11:01:13 +10001325 nouveau_irq_unregister(dev, 26);
Ben Skeggs51beb422011-07-05 10:33:08 +10001326
1327 dev_priv->engine.display.priv = NULL;
Ben Skeggs26f6d882011-07-04 16:25:18 +10001328 kfree(disp);
1329}
1330
1331int
1332nvd0_display_create(struct drm_device *dev)
1333{
1334 struct drm_nouveau_private *dev_priv = dev->dev_private;
Ben Skeggsefd272a2011-07-05 11:58:58 +10001335 struct nouveau_instmem_engine *pinstmem = &dev_priv->engine.instmem;
Ben Skeggs83fc0832011-07-05 13:08:40 +10001336 struct dcb_table *dcb = &dev_priv->vbios.dcb;
1337 struct drm_connector *connector, *tmp;
Ben Skeggs51beb422011-07-05 10:33:08 +10001338 struct pci_dev *pdev = dev->pdev;
Ben Skeggs26f6d882011-07-04 16:25:18 +10001339 struct nvd0_display *disp;
Ben Skeggs83fc0832011-07-05 13:08:40 +10001340 struct dcb_entry *dcbe;
1341 int ret, i;
Ben Skeggs26f6d882011-07-04 16:25:18 +10001342
1343 disp = kzalloc(sizeof(*disp), GFP_KERNEL);
1344 if (!disp)
1345 return -ENOMEM;
1346 dev_priv->engine.display.priv = disp;
1347
Ben Skeggs438d99e2011-07-05 16:48:06 +10001348 /* create crtc objects to represent the hw heads */
1349 for (i = 0; i < 2; i++) {
1350 ret = nvd0_crtc_create(dev, i);
1351 if (ret)
1352 goto out;
1353 }
1354
Ben Skeggs83fc0832011-07-05 13:08:40 +10001355 /* create encoder/connector objects based on VBIOS DCB table */
1356 for (i = 0, dcbe = &dcb->entry[0]; i < dcb->entries; i++, dcbe++) {
1357 connector = nouveau_connector_create(dev, dcbe->connector);
1358 if (IS_ERR(connector))
1359 continue;
1360
1361 if (dcbe->location != DCB_LOC_ON_CHIP) {
1362 NV_WARN(dev, "skipping off-chip encoder %d/%d\n",
1363 dcbe->type, ffs(dcbe->or) - 1);
1364 continue;
1365 }
1366
1367 switch (dcbe->type) {
1368 case OUTPUT_TMDS:
Ben Skeggs3b6d83d12011-07-08 12:52:14 +10001369 case OUTPUT_LVDS:
Ben Skeggs83fc0832011-07-05 13:08:40 +10001370 nvd0_sor_create(connector, dcbe);
1371 break;
Ben Skeggs8eaa9662011-07-06 15:25:47 +10001372 case OUTPUT_ANALOG:
1373 nvd0_dac_create(connector, dcbe);
1374 break;
Ben Skeggs83fc0832011-07-05 13:08:40 +10001375 default:
1376 NV_WARN(dev, "skipping unsupported encoder %d/%d\n",
1377 dcbe->type, ffs(dcbe->or) - 1);
1378 continue;
1379 }
1380 }
1381
1382 /* cull any connectors we created that don't have an encoder */
1383 list_for_each_entry_safe(connector, tmp, &dev->mode_config.connector_list, head) {
1384 if (connector->encoder_ids[0])
1385 continue;
1386
1387 NV_WARN(dev, "%s has no encoders, removing\n",
1388 drm_get_connector_name(connector));
1389 connector->funcs->destroy(connector);
1390 }
1391
Ben Skeggs46005222011-07-05 11:01:13 +10001392 /* setup interrupt handling */
Ben Skeggsf20ce962011-07-08 13:17:01 +10001393 tasklet_init(&disp->tasklet, nvd0_display_bh, (unsigned long)dev);
Ben Skeggs46005222011-07-05 11:01:13 +10001394 nouveau_irq_register(dev, 26, nvd0_display_intr);
1395
Ben Skeggs51beb422011-07-05 10:33:08 +10001396 /* hash table and dma objects for the memory areas we care about */
Ben Skeggsefd272a2011-07-05 11:58:58 +10001397 ret = nouveau_gpuobj_new(dev, NULL, 0x4000, 0x10000,
1398 NVOBJ_FLAG_ZERO_ALLOC, &disp->mem);
Ben Skeggs26f6d882011-07-04 16:25:18 +10001399 if (ret)
1400 goto out;
1401
Ben Skeggsefd272a2011-07-05 11:58:58 +10001402 nv_wo32(disp->mem, 0x1000, 0x00000049);
1403 nv_wo32(disp->mem, 0x1004, (disp->mem->vinst + 0x2000) >> 8);
1404 nv_wo32(disp->mem, 0x1008, (disp->mem->vinst + 0x2fff) >> 8);
1405 nv_wo32(disp->mem, 0x100c, 0x00000000);
1406 nv_wo32(disp->mem, 0x1010, 0x00000000);
1407 nv_wo32(disp->mem, 0x1014, 0x00000000);
Ben Skeggs37b034a2011-07-08 14:43:19 +10001408 nv_wo32(disp->mem, 0x0000, NvEvoSync);
Ben Skeggsefd272a2011-07-05 11:58:58 +10001409 nv_wo32(disp->mem, 0x0004, (0x1000 << 9) | 0x00000001);
1410
Ben Skeggsc0cc92a2011-07-06 11:40:45 +10001411 nv_wo32(disp->mem, 0x1020, 0x00000049);
Ben Skeggsefd272a2011-07-05 11:58:58 +10001412 nv_wo32(disp->mem, 0x1024, 0x00000000);
1413 nv_wo32(disp->mem, 0x1028, (dev_priv->vram_size - 1) >> 8);
1414 nv_wo32(disp->mem, 0x102c, 0x00000000);
1415 nv_wo32(disp->mem, 0x1030, 0x00000000);
1416 nv_wo32(disp->mem, 0x1034, 0x00000000);
Ben Skeggs37b034a2011-07-08 14:43:19 +10001417 nv_wo32(disp->mem, 0x0008, NvEvoVRAM);
Ben Skeggsefd272a2011-07-05 11:58:58 +10001418 nv_wo32(disp->mem, 0x000c, (0x1020 << 9) | 0x00000001);
1419
Ben Skeggsc0cc92a2011-07-06 11:40:45 +10001420 nv_wo32(disp->mem, 0x1040, 0x00000009);
1421 nv_wo32(disp->mem, 0x1044, 0x00000000);
1422 nv_wo32(disp->mem, 0x1048, (dev_priv->vram_size - 1) >> 8);
1423 nv_wo32(disp->mem, 0x104c, 0x00000000);
1424 nv_wo32(disp->mem, 0x1050, 0x00000000);
1425 nv_wo32(disp->mem, 0x1054, 0x00000000);
1426 nv_wo32(disp->mem, 0x0010, NvEvoVRAM_LP);
1427 nv_wo32(disp->mem, 0x0014, (0x1040 << 9) | 0x00000001);
1428
1429 nv_wo32(disp->mem, 0x1060, 0x0fe00009);
1430 nv_wo32(disp->mem, 0x1064, 0x00000000);
1431 nv_wo32(disp->mem, 0x1068, (dev_priv->vram_size - 1) >> 8);
1432 nv_wo32(disp->mem, 0x106c, 0x00000000);
1433 nv_wo32(disp->mem, 0x1070, 0x00000000);
1434 nv_wo32(disp->mem, 0x1074, 0x00000000);
1435 nv_wo32(disp->mem, 0x0018, NvEvoFB32);
1436 nv_wo32(disp->mem, 0x001c, (0x1060 << 9) | 0x00000001);
1437
Ben Skeggsefd272a2011-07-05 11:58:58 +10001438 pinstmem->flush(dev);
1439
Ben Skeggs51beb422011-07-05 10:33:08 +10001440 /* push buffers for evo channels */
1441 disp->evo[0].ptr =
1442 pci_alloc_consistent(pdev, PAGE_SIZE, &disp->evo[0].handle);
1443 if (!disp->evo[0].ptr) {
1444 ret = -ENOMEM;
1445 goto out;
1446 }
1447
Ben Skeggs26f6d882011-07-04 16:25:18 +10001448 ret = nvd0_display_init(dev);
1449 if (ret)
1450 goto out;
1451
1452out:
1453 if (ret)
1454 nvd0_display_destroy(dev);
1455 return ret;
1456}