radeon_crtc.c revision c503f109
1209ff23fSmrg/*
2209ff23fSmrg * Copyright 2000 ATI Technologies Inc., Markham, Ontario, and
3209ff23fSmrg *                VA Linux Systems Inc., Fremont, California.
4209ff23fSmrg *
5209ff23fSmrg * All Rights Reserved.
6209ff23fSmrg *
7209ff23fSmrg * Permission is hereby granted, free of charge, to any person obtaining
8209ff23fSmrg * a copy of this software and associated documentation files (the
9209ff23fSmrg * "Software"), to deal in the Software without restriction, including
10209ff23fSmrg * without limitation on the rights to use, copy, modify, merge,
11209ff23fSmrg * publish, distribute, sublicense, and/or sell copies of the Software,
12209ff23fSmrg * and to permit persons to whom the Software is furnished to do so,
13209ff23fSmrg * subject to the following conditions:
14209ff23fSmrg *
15209ff23fSmrg * The above copyright notice and this permission notice (including the
16209ff23fSmrg * next paragraph) shall be included in all copies or substantial
17209ff23fSmrg * portions of the Software.
18209ff23fSmrg *
19209ff23fSmrg * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
20209ff23fSmrg * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
21209ff23fSmrg * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
22209ff23fSmrg * NON-INFRINGEMENT.  IN NO EVENT SHALL ATI, VA LINUX SYSTEMS AND/OR
23209ff23fSmrg * THEIR SUPPLIERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
24209ff23fSmrg * WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
25209ff23fSmrg * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
26209ff23fSmrg * DEALINGS IN THE SOFTWARE.
27209ff23fSmrg */
28209ff23fSmrg
29209ff23fSmrg#ifdef HAVE_CONFIG_H
30209ff23fSmrg#include "config.h"
31209ff23fSmrg#endif
32209ff23fSmrg
33209ff23fSmrg#include <string.h>
34209ff23fSmrg#include <stdio.h>
35209ff23fSmrg
36209ff23fSmrg/* X and server generic header files */
37209ff23fSmrg#include "xf86.h"
38209ff23fSmrg#include "xf86_OSproc.h"
39209ff23fSmrg#include "vgaHW.h"
40209ff23fSmrg#include "xf86Modes.h"
41209ff23fSmrg
42209ff23fSmrg/* Driver data structures */
43209ff23fSmrg#include "radeon.h"
44209ff23fSmrg#include "radeon_reg.h"
45209ff23fSmrg#include "radeon_macros.h"
46209ff23fSmrg#include "radeon_probe.h"
47209ff23fSmrg#include "radeon_version.h"
48209ff23fSmrg
49209ff23fSmrg#ifdef XF86DRI
50209ff23fSmrg#define _XF86DRI_SERVER_
51b7e1c893Smrg#include "radeon_drm.h"
52209ff23fSmrg#include "sarea.h"
53209ff23fSmrg#endif
54209ff23fSmrg
55209ff23fSmrgextern void atombios_crtc_mode_set(xf86CrtcPtr crtc,
56209ff23fSmrg				   DisplayModePtr mode,
57209ff23fSmrg				   DisplayModePtr adjusted_mode,
58209ff23fSmrg				   int x, int y);
59209ff23fSmrgextern void atombios_crtc_dpms(xf86CrtcPtr crtc, int mode);
60b7e1c893Smrgextern void
61b7e1c893SmrgRADEONInitDispBandwidthLegacy(ScrnInfoPtr pScrn,
62b7e1c893Smrg			      DisplayModePtr mode1, int pixel_bytes1,
63b7e1c893Smrg			      DisplayModePtr mode2, int pixel_bytes2);
64b7e1c893Smrgextern void
65b7e1c893SmrgRADEONInitDispBandwidthAVIVO(ScrnInfoPtr pScrn,
66b7e1c893Smrg			     DisplayModePtr mode1, int pixel_bytes1,
67b7e1c893Smrg			     DisplayModePtr mode2, int pixel_bytes2);
68209ff23fSmrg
69209ff23fSmrgvoid
70209ff23fSmrgradeon_crtc_dpms(xf86CrtcPtr crtc, int mode)
71209ff23fSmrg{
72209ff23fSmrg    RADEONInfoPtr info = RADEONPTR(crtc->scrn);
73209ff23fSmrg    RADEONEntPtr pRADEONEnt = RADEONEntPriv(crtc->scrn);
74209ff23fSmrg    RADEONCrtcPrivatePtr radeon_crtc = crtc->driver_private;
75209ff23fSmrg    xf86CrtcPtr crtc0 = pRADEONEnt->pCrtc[0];
76209ff23fSmrg
77209ff23fSmrg    if ((mode == DPMSModeOn) && radeon_crtc->enabled)
78209ff23fSmrg	return;
79209ff23fSmrg
80c503f109Smrg    if (mode == DPMSModeOff)
81c503f109Smrg	radeon_crtc_modeset_ioctl(crtc, FALSE);
82c503f109Smrg
83b7e1c893Smrg    if (IS_AVIVO_VARIANT || info->r4xx_atom) {
84209ff23fSmrg	atombios_crtc_dpms(crtc, mode);
85209ff23fSmrg    } else {
86209ff23fSmrg
87209ff23fSmrg	/* need to restore crtc1 before crtc0 or we may get a blank screen
88209ff23fSmrg	 * in some cases
89209ff23fSmrg	 */
90209ff23fSmrg	if ((radeon_crtc->crtc_id == 1) && (mode == DPMSModeOn)) {
91209ff23fSmrg	    if (crtc0->enabled)
92209ff23fSmrg		legacy_crtc_dpms(crtc0,  DPMSModeOff);
93209ff23fSmrg	}
94209ff23fSmrg
95209ff23fSmrg	legacy_crtc_dpms(crtc, mode);
96209ff23fSmrg
97209ff23fSmrg	if ((radeon_crtc->crtc_id == 1) && (mode == DPMSModeOn)) {
98209ff23fSmrg	    if (crtc0->enabled)
99209ff23fSmrg		legacy_crtc_dpms(crtc0, mode);
100209ff23fSmrg	}
101209ff23fSmrg    }
102209ff23fSmrg
103c503f109Smrg    if (mode != DPMSModeOff) {
104c503f109Smrg	radeon_crtc_modeset_ioctl(crtc, TRUE);
105c503f109Smrg	radeon_crtc_load_lut(crtc);
106c503f109Smrg    }
107c503f109Smrg
108209ff23fSmrg    if (mode == DPMSModeOn)
109209ff23fSmrg	radeon_crtc->enabled = TRUE;
110209ff23fSmrg    else
111209ff23fSmrg	radeon_crtc->enabled = FALSE;
112209ff23fSmrg}
113209ff23fSmrg
114209ff23fSmrgstatic Bool
115209ff23fSmrgradeon_crtc_mode_fixup(xf86CrtcPtr crtc, DisplayModePtr mode,
116209ff23fSmrg		     DisplayModePtr adjusted_mode)
117209ff23fSmrg{
118209ff23fSmrg    return TRUE;
119209ff23fSmrg}
120209ff23fSmrg
121209ff23fSmrgstatic void
122209ff23fSmrgradeon_crtc_mode_prepare(xf86CrtcPtr crtc)
123209ff23fSmrg{
124209ff23fSmrg    RADEONCrtcPrivatePtr radeon_crtc = crtc->driver_private;
125209ff23fSmrg
126209ff23fSmrg    if (radeon_crtc->enabled)
127209ff23fSmrg	crtc->funcs->hide_cursor(crtc);
128209ff23fSmrg}
129209ff23fSmrg
130209ff23fSmrgstatic uint32_t RADEONDiv(CARD64 n, uint32_t d)
131209ff23fSmrg{
132209ff23fSmrg    return (n + (d / 2)) / d;
133209ff23fSmrg}
134209ff23fSmrg
135209ff23fSmrgvoid
136209ff23fSmrgRADEONComputePLL(RADEONPLLPtr pll,
137209ff23fSmrg		 unsigned long freq,
138209ff23fSmrg		 uint32_t *chosen_dot_clock_freq,
139209ff23fSmrg		 uint32_t *chosen_feedback_div,
140209ff23fSmrg		 uint32_t *chosen_reference_div,
141209ff23fSmrg		 uint32_t *chosen_post_div,
142209ff23fSmrg		 int flags)
143209ff23fSmrg{
144209ff23fSmrg    uint32_t min_ref_div = pll->min_ref_div;
145209ff23fSmrg    uint32_t max_ref_div = pll->max_ref_div;
146209ff23fSmrg    uint32_t best_vco = pll->best_vco;
147209ff23fSmrg    uint32_t best_post_div = 1;
148209ff23fSmrg    uint32_t best_ref_div = 1;
149209ff23fSmrg    uint32_t best_feedback_div = 1;
150209ff23fSmrg    uint32_t best_freq = -1;
151209ff23fSmrg    uint32_t best_error = 0xffffffff;
152209ff23fSmrg    uint32_t best_vco_diff = 1;
153209ff23fSmrg    uint32_t post_div;
154209ff23fSmrg
155209ff23fSmrg    freq = freq * 1000;
156209ff23fSmrg
157209ff23fSmrg    ErrorF("freq: %lu\n", freq);
158209ff23fSmrg
159209ff23fSmrg    if (flags & RADEON_PLL_USE_REF_DIV)
160209ff23fSmrg	min_ref_div = max_ref_div = pll->reference_div;
161209ff23fSmrg    else {
162209ff23fSmrg	while (min_ref_div < max_ref_div-1) {
163209ff23fSmrg	    uint32_t mid=(min_ref_div+max_ref_div)/2;
164209ff23fSmrg	    uint32_t pll_in = pll->reference_freq / mid;
165209ff23fSmrg	    if (pll_in < pll->pll_in_min)
166209ff23fSmrg		max_ref_div = mid;
167209ff23fSmrg	    else if (pll_in > pll->pll_in_max)
168209ff23fSmrg		min_ref_div = mid;
169209ff23fSmrg	    else break;
170209ff23fSmrg	}
171209ff23fSmrg    }
172209ff23fSmrg
173209ff23fSmrg    for (post_div = pll->min_post_div; post_div <= pll->max_post_div; ++post_div) {
174209ff23fSmrg	uint32_t ref_div;
175209ff23fSmrg
176209ff23fSmrg	if ((flags & RADEON_PLL_NO_ODD_POST_DIV) && (post_div & 1))
177209ff23fSmrg	    continue;
178209ff23fSmrg
179209ff23fSmrg	/* legacy radeons only have a few post_divs */
180209ff23fSmrg	if (flags & RADEON_PLL_LEGACY) {
181209ff23fSmrg	    if ((post_div == 5) ||
182209ff23fSmrg		(post_div == 7) ||
183209ff23fSmrg		(post_div == 9) ||
184209ff23fSmrg		(post_div == 10) ||
185209ff23fSmrg		(post_div == 11))
186209ff23fSmrg		continue;
187209ff23fSmrg	}
188209ff23fSmrg
189209ff23fSmrg	for (ref_div = min_ref_div; ref_div <= max_ref_div; ++ref_div) {
190209ff23fSmrg	    uint32_t feedback_div, current_freq, error, vco_diff;
191209ff23fSmrg	    uint32_t pll_in = pll->reference_freq / ref_div;
192209ff23fSmrg	    uint32_t min_feed_div = pll->min_feedback_div;
193209ff23fSmrg	    uint32_t max_feed_div = pll->max_feedback_div+1;
194209ff23fSmrg
195209ff23fSmrg	    if (pll_in < pll->pll_in_min || pll_in > pll->pll_in_max)
196209ff23fSmrg		continue;
197209ff23fSmrg
198209ff23fSmrg	    while (min_feed_div < max_feed_div) {
199209ff23fSmrg		uint32_t vco;
200209ff23fSmrg
201209ff23fSmrg		feedback_div = (min_feed_div+max_feed_div)/2;
202209ff23fSmrg
203209ff23fSmrg		vco = RADEONDiv((CARD64)pll->reference_freq * feedback_div,
204209ff23fSmrg				ref_div);
205209ff23fSmrg
206209ff23fSmrg		if (vco < pll->pll_out_min) {
207209ff23fSmrg		    min_feed_div = feedback_div+1;
208209ff23fSmrg		    continue;
209209ff23fSmrg		} else if(vco > pll->pll_out_max) {
210209ff23fSmrg		    max_feed_div = feedback_div;
211209ff23fSmrg		    continue;
212209ff23fSmrg		}
213209ff23fSmrg
214209ff23fSmrg		current_freq = RADEONDiv((CARD64)pll->reference_freq * 10000 * feedback_div,
215209ff23fSmrg					 ref_div * post_div);
216209ff23fSmrg
217209ff23fSmrg		error = abs(current_freq - freq);
218209ff23fSmrg		vco_diff = abs(vco - best_vco);
219209ff23fSmrg
220209ff23fSmrg		if ((best_vco == 0 && error < best_error) ||
221209ff23fSmrg		    (best_vco != 0 &&
222209ff23fSmrg		     (error < best_error - 100 ||
223209ff23fSmrg		      (abs(error - best_error) < 100 && vco_diff < best_vco_diff )))) {
224209ff23fSmrg		    best_post_div = post_div;
225209ff23fSmrg		    best_ref_div = ref_div;
226209ff23fSmrg		    best_feedback_div = feedback_div;
227209ff23fSmrg		    best_freq = current_freq;
228209ff23fSmrg		    best_error = error;
229209ff23fSmrg		    best_vco_diff = vco_diff;
230209ff23fSmrg		} else if (current_freq == freq) {
231209ff23fSmrg		    if (best_freq == -1) {
232209ff23fSmrg			best_post_div = post_div;
233209ff23fSmrg			best_ref_div = ref_div;
234209ff23fSmrg			best_feedback_div = feedback_div;
235209ff23fSmrg			best_freq = current_freq;
236209ff23fSmrg			best_error = error;
237209ff23fSmrg			best_vco_diff = vco_diff;
238b7e1c893Smrg		    } else if (((flags & RADEON_PLL_PREFER_LOW_REF_DIV) && (ref_div < best_ref_div)) ||
239b7e1c893Smrg			       ((flags & RADEON_PLL_PREFER_HIGH_REF_DIV) && (ref_div > best_ref_div)) ||
240b7e1c893Smrg			       ((flags & RADEON_PLL_PREFER_LOW_FB_DIV) && (feedback_div < best_feedback_div)) ||
241b7e1c893Smrg			       ((flags & RADEON_PLL_PREFER_HIGH_FB_DIV) && (feedback_div > best_feedback_div)) ||
242b7e1c893Smrg			       ((flags & RADEON_PLL_PREFER_LOW_POST_DIV) && (post_div < best_post_div)) ||
243b7e1c893Smrg			       ((flags & RADEON_PLL_PREFER_HIGH_POST_DIV) && (post_div > best_post_div))) {
244209ff23fSmrg			best_post_div = post_div;
245209ff23fSmrg			best_ref_div = ref_div;
246209ff23fSmrg			best_feedback_div = feedback_div;
247209ff23fSmrg			best_freq = current_freq;
248209ff23fSmrg			best_error = error;
249209ff23fSmrg			best_vco_diff = vco_diff;
250209ff23fSmrg		    }
251209ff23fSmrg		}
252209ff23fSmrg
253209ff23fSmrg		if (current_freq < freq)
254209ff23fSmrg		    min_feed_div = feedback_div+1;
255209ff23fSmrg		else
256209ff23fSmrg		    max_feed_div = feedback_div;
257209ff23fSmrg	    }
258209ff23fSmrg	}
259209ff23fSmrg    }
260209ff23fSmrg
261209ff23fSmrg    ErrorF("best_freq: %u\n", (unsigned int)best_freq);
262209ff23fSmrg    ErrorF("best_feedback_div: %u\n", (unsigned int)best_feedback_div);
263209ff23fSmrg    ErrorF("best_ref_div: %u\n", (unsigned int)best_ref_div);
264209ff23fSmrg    ErrorF("best_post_div: %u\n", (unsigned int)best_post_div);
265209ff23fSmrg
266209ff23fSmrg    if (best_freq == -1)
267209ff23fSmrg	FatalError("Couldn't find valid PLL dividers\n");
268209ff23fSmrg    *chosen_dot_clock_freq = best_freq / 10000;
269209ff23fSmrg    *chosen_feedback_div = best_feedback_div;
270209ff23fSmrg    *chosen_reference_div = best_ref_div;
271209ff23fSmrg    *chosen_post_div = best_post_div;
272209ff23fSmrg
273209ff23fSmrg}
274209ff23fSmrg
275209ff23fSmrgstatic void
276209ff23fSmrgradeon_crtc_mode_set(xf86CrtcPtr crtc, DisplayModePtr mode,
277209ff23fSmrg		     DisplayModePtr adjusted_mode, int x, int y)
278209ff23fSmrg{
279209ff23fSmrg    ScrnInfoPtr pScrn = crtc->scrn;
280209ff23fSmrg    RADEONInfoPtr info = RADEONPTR(pScrn);
281209ff23fSmrg
282b7e1c893Smrg    if (IS_AVIVO_VARIANT || info->r4xx_atom) {
283209ff23fSmrg	atombios_crtc_mode_set(crtc, mode, adjusted_mode, x, y);
284209ff23fSmrg    } else {
285209ff23fSmrg	legacy_crtc_mode_set(crtc, mode, adjusted_mode, x, y);
286209ff23fSmrg    }
287209ff23fSmrg}
288209ff23fSmrg
289209ff23fSmrgstatic void
290209ff23fSmrgradeon_crtc_mode_commit(xf86CrtcPtr crtc)
291209ff23fSmrg{
292209ff23fSmrg    if (crtc->scrn->pScreen != NULL)
293209ff23fSmrg	xf86_reload_cursors(crtc->scrn->pScreen);
294209ff23fSmrg}
295209ff23fSmrg
296209ff23fSmrgvoid
297209ff23fSmrgradeon_crtc_load_lut(xf86CrtcPtr crtc)
298209ff23fSmrg{
299209ff23fSmrg    ScrnInfoPtr pScrn = crtc->scrn;
300209ff23fSmrg    RADEONCrtcPrivatePtr radeon_crtc = crtc->driver_private;
301209ff23fSmrg    RADEONInfoPtr info = RADEONPTR(pScrn);
302209ff23fSmrg    unsigned char *RADEONMMIO = info->MMIO;
303209ff23fSmrg    int i;
304209ff23fSmrg
305209ff23fSmrg    if (!crtc->enabled)
306209ff23fSmrg	return;
307209ff23fSmrg
308209ff23fSmrg    if (IS_AVIVO_VARIANT) {
309209ff23fSmrg	OUTREG(AVIVO_DC_LUTA_CONTROL + radeon_crtc->crtc_offset, 0);
310209ff23fSmrg
311209ff23fSmrg	OUTREG(AVIVO_DC_LUTA_BLACK_OFFSET_BLUE + radeon_crtc->crtc_offset, 0);
312209ff23fSmrg	OUTREG(AVIVO_DC_LUTA_BLACK_OFFSET_GREEN + radeon_crtc->crtc_offset, 0);
313209ff23fSmrg	OUTREG(AVIVO_DC_LUTA_BLACK_OFFSET_RED + radeon_crtc->crtc_offset, 0);
314209ff23fSmrg
315209ff23fSmrg	OUTREG(AVIVO_DC_LUTA_WHITE_OFFSET_BLUE + radeon_crtc->crtc_offset, 0x0000ffff);
316209ff23fSmrg	OUTREG(AVIVO_DC_LUTA_WHITE_OFFSET_GREEN + radeon_crtc->crtc_offset, 0x0000ffff);
317209ff23fSmrg	OUTREG(AVIVO_DC_LUTA_WHITE_OFFSET_RED + radeon_crtc->crtc_offset, 0x0000ffff);
318209ff23fSmrg    }
319209ff23fSmrg
320209ff23fSmrg    PAL_SELECT(radeon_crtc->crtc_id);
321209ff23fSmrg
322209ff23fSmrg    if (IS_AVIVO_VARIANT) {
323209ff23fSmrg	OUTREG(AVIVO_DC_LUT_RW_MODE, 0);
324209ff23fSmrg	OUTREG(AVIVO_DC_LUT_WRITE_EN_MASK, 0x0000003f);
325209ff23fSmrg    }
326209ff23fSmrg
327209ff23fSmrg    for (i = 0; i < 256; i++) {
328209ff23fSmrg	OUTPAL(i, radeon_crtc->lut_r[i], radeon_crtc->lut_g[i], radeon_crtc->lut_b[i]);
329209ff23fSmrg    }
330209ff23fSmrg
331209ff23fSmrg    if (IS_AVIVO_VARIANT) {
332209ff23fSmrg	OUTREG(AVIVO_D1GRPH_LUT_SEL + radeon_crtc->crtc_offset, radeon_crtc->crtc_id);
333209ff23fSmrg    }
334209ff23fSmrg
335209ff23fSmrg}
336209ff23fSmrg
337209ff23fSmrg
338209ff23fSmrgstatic void
339209ff23fSmrgradeon_crtc_gamma_set(xf86CrtcPtr crtc, uint16_t *red, uint16_t *green,
340209ff23fSmrg		      uint16_t *blue, int size)
341209ff23fSmrg{
342209ff23fSmrg    RADEONCrtcPrivatePtr radeon_crtc = crtc->driver_private;
343209ff23fSmrg    ScrnInfoPtr		pScrn = crtc->scrn;
344209ff23fSmrg    int i, j;
345209ff23fSmrg
346209ff23fSmrg    if (pScrn->depth == 16) {
347209ff23fSmrg	for (i = 0; i < 64; i++) {
348209ff23fSmrg	    if (i <= 31) {
349209ff23fSmrg		for (j = 0; j < 8; j++) {
350b7e1c893Smrg		    radeon_crtc->lut_r[i * 8 + j] = red[i] >> 6;
351b7e1c893Smrg		    radeon_crtc->lut_b[i * 8 + j] = blue[i] >> 6;
352209ff23fSmrg		}
353209ff23fSmrg	    }
354209ff23fSmrg
355209ff23fSmrg	    for (j = 0; j < 4; j++) {
356b7e1c893Smrg		radeon_crtc->lut_g[i * 4 + j] = green[i] >> 6;
357209ff23fSmrg	    }
358209ff23fSmrg	}
359209ff23fSmrg    } else {
360209ff23fSmrg	for (i = 0; i < 256; i++) {
361b7e1c893Smrg	    radeon_crtc->lut_r[i] = red[i] >> 6;
362b7e1c893Smrg	    radeon_crtc->lut_g[i] = green[i] >> 6;
363b7e1c893Smrg	    radeon_crtc->lut_b[i] = blue[i] >> 6;
364209ff23fSmrg	}
365209ff23fSmrg    }
366209ff23fSmrg
367209ff23fSmrg    radeon_crtc_load_lut(crtc);
368209ff23fSmrg}
369209ff23fSmrg
370209ff23fSmrgstatic Bool
371209ff23fSmrgradeon_crtc_lock(xf86CrtcPtr crtc)
372209ff23fSmrg{
373209ff23fSmrg    ScrnInfoPtr		pScrn = crtc->scrn;
374209ff23fSmrg    RADEONInfoPtr  info = RADEONPTR(pScrn);
375209ff23fSmrg
376209ff23fSmrg#ifdef XF86DRI
377b7e1c893Smrg    if (info->cp->CPStarted && pScrn->pScreen) {
378209ff23fSmrg	DRILock(pScrn->pScreen, 0);
379209ff23fSmrg	if (info->accelOn)
380209ff23fSmrg	    RADEON_SYNC(info, pScrn);
381209ff23fSmrg	return TRUE;
382209ff23fSmrg    }
383209ff23fSmrg#endif
384209ff23fSmrg    if (info->accelOn)
385209ff23fSmrg        RADEON_SYNC(info, pScrn);
386209ff23fSmrg
387209ff23fSmrg    return FALSE;
388209ff23fSmrg
389209ff23fSmrg}
390209ff23fSmrg
391209ff23fSmrgstatic void
392209ff23fSmrgradeon_crtc_unlock(xf86CrtcPtr crtc)
393209ff23fSmrg{
394209ff23fSmrg    ScrnInfoPtr		pScrn = crtc->scrn;
395209ff23fSmrg    RADEONInfoPtr  info = RADEONPTR(pScrn);
396209ff23fSmrg
397209ff23fSmrg#ifdef XF86DRI
398b7e1c893Smrg	if (info->cp->CPStarted && pScrn->pScreen) DRIUnlock(pScrn->pScreen);
399209ff23fSmrg#endif
400209ff23fSmrg
401209ff23fSmrg    if (info->accelOn)
402209ff23fSmrg        RADEON_SYNC(info, pScrn);
403209ff23fSmrg}
404209ff23fSmrg
405209ff23fSmrg/**
406209ff23fSmrg * Allocates memory for a locked-in-framebuffer shadow of the given
407209ff23fSmrg * width and height for this CRTC's rotated shadow framebuffer.
408209ff23fSmrg */
409209ff23fSmrg
410209ff23fSmrgstatic void *
411209ff23fSmrgradeon_crtc_shadow_allocate (xf86CrtcPtr crtc, int width, int height)
412209ff23fSmrg{
413209ff23fSmrg    ScrnInfoPtr pScrn = crtc->scrn;
414209ff23fSmrg    RADEONInfoPtr  info = RADEONPTR(pScrn);
415209ff23fSmrg    RADEONCrtcPrivatePtr radeon_crtc = crtc->driver_private;
416209ff23fSmrg    unsigned long rotate_pitch;
417209ff23fSmrg    unsigned long rotate_offset;
418209ff23fSmrg    int align = 4096, size;
419209ff23fSmrg    int cpp = pScrn->bitsPerPixel / 8;
420209ff23fSmrg
421b7e1c893Smrg    /* No rotation without accel */
422b7e1c893Smrg    if (((info->ChipFamily >= CHIP_FAMILY_R600) && !info->directRenderingEnabled) ||
423b7e1c893Smrg	xf86ReturnOptValBool(info->Options, OPTION_NOACCEL, FALSE)) {
424b7e1c893Smrg	xf86DrvMsg(pScrn->scrnIndex, X_ERROR,
425b7e1c893Smrg		   "Acceleration required for rotation\n");
426b7e1c893Smrg	return NULL;
427b7e1c893Smrg    }
428b7e1c893Smrg
429209ff23fSmrg    rotate_pitch = pScrn->displayWidth * cpp;
430209ff23fSmrg    size = rotate_pitch * height;
431209ff23fSmrg
432209ff23fSmrg    /* We could get close to what we want here by just creating a pixmap like
433209ff23fSmrg     * normal, but we have to lock it down in framebuffer, and there is no
434209ff23fSmrg     * setter for offscreen area locking in EXA currently.  So, we just
435209ff23fSmrg     * allocate offscreen memory and fake up a pixmap header for it.
436209ff23fSmrg     */
437b7e1c893Smrg    rotate_offset = radeon_legacy_allocate_memory(pScrn, &radeon_crtc->crtc_rotate_mem, size, align);
438b7e1c893Smrg    if (rotate_offset == 0)
439b7e1c893Smrg	return NULL;
440209ff23fSmrg
441209ff23fSmrg    return info->FB + rotate_offset;
442209ff23fSmrg}
443b7e1c893Smrg
444209ff23fSmrg/**
445209ff23fSmrg * Creates a pixmap for this CRTC's rotated shadow framebuffer.
446209ff23fSmrg */
447209ff23fSmrgstatic PixmapPtr
448209ff23fSmrgradeon_crtc_shadow_create(xf86CrtcPtr crtc, void *data, int width, int height)
449209ff23fSmrg{
450209ff23fSmrg    ScrnInfoPtr pScrn = crtc->scrn;
451209ff23fSmrg    unsigned long rotate_pitch;
452209ff23fSmrg    PixmapPtr rotate_pixmap;
453209ff23fSmrg    int cpp = pScrn->bitsPerPixel / 8;
454209ff23fSmrg
455209ff23fSmrg    if (!data)
456209ff23fSmrg	data = radeon_crtc_shadow_allocate(crtc, width, height);
457209ff23fSmrg
458209ff23fSmrg    rotate_pitch = pScrn->displayWidth * cpp;
459209ff23fSmrg
460209ff23fSmrg    rotate_pixmap = GetScratchPixmapHeader(pScrn->pScreen,
461209ff23fSmrg					   width, height,
462209ff23fSmrg					   pScrn->depth,
463209ff23fSmrg					   pScrn->bitsPerPixel,
464209ff23fSmrg					   rotate_pitch,
465209ff23fSmrg					   data);
466209ff23fSmrg
467209ff23fSmrg    if (rotate_pixmap == NULL) {
468209ff23fSmrg	xf86DrvMsg(pScrn->scrnIndex, X_ERROR,
469209ff23fSmrg		   "Couldn't allocate shadow pixmap for rotated CRTC\n");
470209ff23fSmrg    }
471209ff23fSmrg
472209ff23fSmrg    return rotate_pixmap;
473209ff23fSmrg}
474209ff23fSmrg
475209ff23fSmrgstatic void
476209ff23fSmrgradeon_crtc_shadow_destroy(xf86CrtcPtr crtc, PixmapPtr rotate_pixmap, void *data)
477209ff23fSmrg{
478209ff23fSmrg    ScrnInfoPtr pScrn = crtc->scrn;
479209ff23fSmrg    RADEONCrtcPrivatePtr radeon_crtc = crtc->driver_private;
480209ff23fSmrg
481209ff23fSmrg    if (rotate_pixmap)
482209ff23fSmrg	FreeScratchPixmapHeader(rotate_pixmap);
483b7e1c893Smrg
484209ff23fSmrg    if (data) {
485b7e1c893Smrg	radeon_legacy_free_memory(pScrn, radeon_crtc->crtc_rotate_mem);
486b7e1c893Smrg	radeon_crtc->crtc_rotate_mem = NULL;
487b7e1c893Smrg    }
488b7e1c893Smrg
489b7e1c893Smrg}
490b7e1c893Smrg
491b7e1c893Smrg#if XF86_CRTC_VERSION >= 2
492b7e1c893Smrg#include "radeon_atombios.h"
493b7e1c893Smrg
494b7e1c893Smrgextern AtomBiosResult
495b7e1c893Smrgatombios_lock_crtc(atomBiosHandlePtr atomBIOS, int crtc, int lock);
496b7e1c893Smrgextern void
497b7e1c893SmrgRADEONInitCrtcBase(xf86CrtcPtr crtc, RADEONSavePtr save,
498b7e1c893Smrg		   int x, int y);
499b7e1c893Smrgextern void
500b7e1c893SmrgRADEONInitCrtc2Base(xf86CrtcPtr crtc, RADEONSavePtr save,
501b7e1c893Smrg		    int x, int y);
502b7e1c893Smrgextern void
503b7e1c893SmrgRADEONRestoreCrtcBase(ScrnInfoPtr pScrn,
504b7e1c893Smrg		      RADEONSavePtr restore);
505b7e1c893Smrgextern void
506b7e1c893SmrgRADEONRestoreCrtc2Base(ScrnInfoPtr pScrn,
507b7e1c893Smrg		       RADEONSavePtr restore);
508b7e1c893Smrg
509b7e1c893Smrgstatic void
510b7e1c893Smrgradeon_crtc_set_origin(xf86CrtcPtr crtc, int x, int y)
511b7e1c893Smrg{
512b7e1c893Smrg    ScrnInfoPtr pScrn = crtc->scrn;
513b7e1c893Smrg    RADEONCrtcPrivatePtr radeon_crtc = crtc->driver_private;
514b7e1c893Smrg    RADEONInfoPtr info = RADEONPTR(pScrn);
515b7e1c893Smrg    unsigned char *RADEONMMIO = info->MMIO;
516b7e1c893Smrg
517b7e1c893Smrg    if (IS_AVIVO_VARIANT) {
518b7e1c893Smrg	x &= ~3;
519b7e1c893Smrg	y &= ~1;
520b7e1c893Smrg	atombios_lock_crtc(info->atomBIOS, radeon_crtc->crtc_id, 1);
521b7e1c893Smrg	OUTREG(AVIVO_D1MODE_VIEWPORT_START + radeon_crtc->crtc_offset, (x << 16) | y);
522b7e1c893Smrg	atombios_lock_crtc(info->atomBIOS, radeon_crtc->crtc_id, 0);
523b7e1c893Smrg    } else {
524b7e1c893Smrg	switch (radeon_crtc->crtc_id) {
525b7e1c893Smrg	case 0:
526b7e1c893Smrg	    RADEONInitCrtcBase(crtc, info->ModeReg, x, y);
527b7e1c893Smrg	    RADEONRestoreCrtcBase(pScrn, info->ModeReg);
528b7e1c893Smrg	    break;
529b7e1c893Smrg	case 1:
530b7e1c893Smrg	    RADEONInitCrtc2Base(crtc, info->ModeReg, x, y);
531b7e1c893Smrg	    RADEONRestoreCrtc2Base(pScrn, info->ModeReg);
532b7e1c893Smrg	    break;
533b7e1c893Smrg	default:
534b7e1c893Smrg	    break;
535209ff23fSmrg	}
536209ff23fSmrg    }
537209ff23fSmrg}
538b7e1c893Smrg#endif
539209ff23fSmrg
540b7e1c893Smrg
541b7e1c893Smrgstatic xf86CrtcFuncsRec radeon_crtc_funcs = {
542209ff23fSmrg    .dpms = radeon_crtc_dpms,
543209ff23fSmrg    .save = NULL, /* XXX */
544209ff23fSmrg    .restore = NULL, /* XXX */
545209ff23fSmrg    .mode_fixup = radeon_crtc_mode_fixup,
546209ff23fSmrg    .prepare = radeon_crtc_mode_prepare,
547209ff23fSmrg    .mode_set = radeon_crtc_mode_set,
548209ff23fSmrg    .commit = radeon_crtc_mode_commit,
549209ff23fSmrg    .gamma_set = radeon_crtc_gamma_set,
550209ff23fSmrg    .lock = radeon_crtc_lock,
551209ff23fSmrg    .unlock = radeon_crtc_unlock,
552209ff23fSmrg    .shadow_create = radeon_crtc_shadow_create,
553209ff23fSmrg    .shadow_allocate = radeon_crtc_shadow_allocate,
554209ff23fSmrg    .shadow_destroy = radeon_crtc_shadow_destroy,
555209ff23fSmrg    .set_cursor_colors = radeon_crtc_set_cursor_colors,
556209ff23fSmrg    .set_cursor_position = radeon_crtc_set_cursor_position,
557209ff23fSmrg    .show_cursor = radeon_crtc_show_cursor,
558209ff23fSmrg    .hide_cursor = radeon_crtc_hide_cursor,
559209ff23fSmrg    .load_cursor_argb = radeon_crtc_load_cursor_argb,
560209ff23fSmrg    .destroy = NULL, /* XXX */
561b7e1c893Smrg#if XF86_CRTC_VERSION >= 2
562b7e1c893Smrg    .set_origin = radeon_crtc_set_origin,
563b7e1c893Smrg#endif
564209ff23fSmrg};
565209ff23fSmrg
566b7e1c893Smrgvoid
567b7e1c893SmrgRADEONInitDispBandwidth(ScrnInfoPtr pScrn)
568b7e1c893Smrg{
569b7e1c893Smrg    RADEONInfoPtr info = RADEONPTR(pScrn);
570b7e1c893Smrg    xf86CrtcConfigPtr   xf86_config = XF86_CRTC_CONFIG_PTR(pScrn);
571b7e1c893Smrg    DisplayModePtr mode1 = NULL, mode2 = NULL;
572b7e1c893Smrg    int pixel_bytes1 = info->CurrentLayout.pixel_bytes;
573b7e1c893Smrg    int pixel_bytes2 = info->CurrentLayout.pixel_bytes;
574b7e1c893Smrg
575b7e1c893Smrg    if (xf86_config->num_crtc == 2) {
576b7e1c893Smrg	if (xf86_config->crtc[1]->enabled &&
577b7e1c893Smrg	    xf86_config->crtc[0]->enabled) {
578b7e1c893Smrg	    mode1 = &xf86_config->crtc[0]->mode;
579b7e1c893Smrg	    mode2 = &xf86_config->crtc[1]->mode;
580b7e1c893Smrg	} else if (xf86_config->crtc[0]->enabled) {
581b7e1c893Smrg	    mode1 = &xf86_config->crtc[0]->mode;
582b7e1c893Smrg	} else if (xf86_config->crtc[1]->enabled) {
583b7e1c893Smrg	    mode2 = &xf86_config->crtc[1]->mode;
584b7e1c893Smrg	} else
585b7e1c893Smrg	    return;
586b7e1c893Smrg    } else {
587b7e1c893Smrg	if (info->IsPrimary)
588b7e1c893Smrg	    mode1 = &xf86_config->crtc[0]->mode;
589b7e1c893Smrg	else if (info->IsSecondary)
590b7e1c893Smrg	    mode2 = &xf86_config->crtc[0]->mode;
591b7e1c893Smrg	else if (xf86_config->crtc[0]->enabled)
592b7e1c893Smrg	    mode1 = &xf86_config->crtc[0]->mode;
593b7e1c893Smrg	else
594b7e1c893Smrg	    return;
595b7e1c893Smrg    }
596b7e1c893Smrg
597b7e1c893Smrg    if (IS_AVIVO_VARIANT)
598b7e1c893Smrg	RADEONInitDispBandwidthAVIVO(pScrn, mode1, pixel_bytes1, mode2, pixel_bytes2);
599b7e1c893Smrg    else
600b7e1c893Smrg	RADEONInitDispBandwidthLegacy(pScrn, mode1, pixel_bytes1, mode2, pixel_bytes2);
601b7e1c893Smrg}
602b7e1c893Smrg
603209ff23fSmrgBool RADEONAllocateControllers(ScrnInfoPtr pScrn, int mask)
604209ff23fSmrg{
605209ff23fSmrg    RADEONEntPtr pRADEONEnt = RADEONEntPriv(pScrn);
606209ff23fSmrg    RADEONInfoPtr  info = RADEONPTR(pScrn);
607209ff23fSmrg
608b7e1c893Smrg    if (!xf86ReturnOptValBool(info->Options, OPTION_NOACCEL, FALSE)) {
609b7e1c893Smrg	radeon_crtc_funcs.shadow_create = radeon_crtc_shadow_create;
610b7e1c893Smrg	radeon_crtc_funcs.shadow_allocate = radeon_crtc_shadow_allocate;
611b7e1c893Smrg	radeon_crtc_funcs.shadow_destroy = radeon_crtc_shadow_destroy;
612b7e1c893Smrg    }
613b7e1c893Smrg
614209ff23fSmrg    if (mask & 1) {
615209ff23fSmrg	if (pRADEONEnt->Controller[0])
616209ff23fSmrg	    return TRUE;
617209ff23fSmrg
618209ff23fSmrg	pRADEONEnt->pCrtc[0] = xf86CrtcCreate(pScrn, &radeon_crtc_funcs);
619209ff23fSmrg	if (!pRADEONEnt->pCrtc[0])
620209ff23fSmrg	    return FALSE;
621209ff23fSmrg
622209ff23fSmrg	pRADEONEnt->Controller[0] = xnfcalloc(sizeof(RADEONCrtcPrivateRec), 1);
623209ff23fSmrg	if (!pRADEONEnt->Controller[0])
624209ff23fSmrg	    return FALSE;
625209ff23fSmrg
626209ff23fSmrg	pRADEONEnt->pCrtc[0]->driver_private = pRADEONEnt->Controller[0];
627209ff23fSmrg	pRADEONEnt->Controller[0]->crtc_id = 0;
628209ff23fSmrg	pRADEONEnt->Controller[0]->crtc_offset = 0;
629b7e1c893Smrg	pRADEONEnt->Controller[0]->initialized = FALSE;
630209ff23fSmrg	if (info->allowColorTiling)
631209ff23fSmrg	    pRADEONEnt->Controller[0]->can_tile = 1;
632209ff23fSmrg	else
633209ff23fSmrg	    pRADEONEnt->Controller[0]->can_tile = 0;
634209ff23fSmrg    }
635209ff23fSmrg
636209ff23fSmrg    if (mask & 2) {
637209ff23fSmrg	if (!pRADEONEnt->HasCRTC2)
638209ff23fSmrg	    return TRUE;
639209ff23fSmrg
640209ff23fSmrg	pRADEONEnt->pCrtc[1] = xf86CrtcCreate(pScrn, &radeon_crtc_funcs);
641209ff23fSmrg	if (!pRADEONEnt->pCrtc[1])
642209ff23fSmrg	    return FALSE;
643209ff23fSmrg
644209ff23fSmrg	pRADEONEnt->Controller[1] = xnfcalloc(sizeof(RADEONCrtcPrivateRec), 1);
645209ff23fSmrg	if (!pRADEONEnt->Controller[1])
646209ff23fSmrg	    {
647209ff23fSmrg		xfree(pRADEONEnt->Controller[0]);
648209ff23fSmrg		return FALSE;
649209ff23fSmrg	    }
650209ff23fSmrg
651209ff23fSmrg	pRADEONEnt->pCrtc[1]->driver_private = pRADEONEnt->Controller[1];
652209ff23fSmrg	pRADEONEnt->Controller[1]->crtc_id = 1;
653209ff23fSmrg	pRADEONEnt->Controller[1]->crtc_offset = AVIVO_D2CRTC_H_TOTAL - AVIVO_D1CRTC_H_TOTAL;
654b7e1c893Smrg	pRADEONEnt->Controller[1]->initialized = FALSE;
655209ff23fSmrg	if (info->allowColorTiling)
656209ff23fSmrg	    pRADEONEnt->Controller[1]->can_tile = 1;
657209ff23fSmrg	else
658209ff23fSmrg	    pRADEONEnt->Controller[1]->can_tile = 0;
659209ff23fSmrg    }
660209ff23fSmrg
661209ff23fSmrg    return TRUE;
662209ff23fSmrg}
663209ff23fSmrg
664209ff23fSmrg/**
665209ff23fSmrg * In the current world order, there are lists of modes per output, which may
666209ff23fSmrg * or may not include the mode that was asked to be set by XFree86's mode
667209ff23fSmrg * selection.  Find the closest one, in the following preference order:
668209ff23fSmrg *
669209ff23fSmrg * - Equality
670209ff23fSmrg * - Closer in size to the requested mode, but no larger
671209ff23fSmrg * - Closer in refresh rate to the requested mode.
672209ff23fSmrg */
673209ff23fSmrgDisplayModePtr
674209ff23fSmrgRADEONCrtcFindClosestMode(xf86CrtcPtr crtc, DisplayModePtr pMode)
675209ff23fSmrg{
676209ff23fSmrg    ScrnInfoPtr	pScrn = crtc->scrn;
677209ff23fSmrg    xf86CrtcConfigPtr   xf86_config = XF86_CRTC_CONFIG_PTR(pScrn);
678209ff23fSmrg    DisplayModePtr pBest = NULL, pScan = NULL;
679209ff23fSmrg    int i;
680209ff23fSmrg
681209ff23fSmrg    /* Assume that there's only one output connected to the given CRTC. */
682209ff23fSmrg    for (i = 0; i < xf86_config->num_output; i++)
683209ff23fSmrg    {
684209ff23fSmrg	xf86OutputPtr  output = xf86_config->output[i];
685209ff23fSmrg	if (output->crtc == crtc && output->probed_modes != NULL)
686209ff23fSmrg	{
687209ff23fSmrg	    pScan = output->probed_modes;
688209ff23fSmrg	    break;
689209ff23fSmrg	}
690209ff23fSmrg    }
691209ff23fSmrg
692209ff23fSmrg    /* If the pipe doesn't have any detected modes, just let the system try to
693209ff23fSmrg     * spam the desired mode in.
694209ff23fSmrg     */
695209ff23fSmrg    if (pScan == NULL) {
696209ff23fSmrg	RADEONCrtcPrivatePtr  radeon_crtc = crtc->driver_private;
697209ff23fSmrg	xf86DrvMsg(pScrn->scrnIndex, X_WARNING,
698209ff23fSmrg		   "No crtc mode list for crtc %d,"
699209ff23fSmrg		   "continuing with desired mode\n", radeon_crtc->crtc_id);
700209ff23fSmrg	return pMode;
701209ff23fSmrg    }
702209ff23fSmrg
703209ff23fSmrg    for (; pScan != NULL; pScan = pScan->next) {
704209ff23fSmrg	assert(pScan->VRefresh != 0.0);
705209ff23fSmrg
706209ff23fSmrg	/* If there's an exact match, we're done. */
707209ff23fSmrg	if (xf86ModesEqual(pScan, pMode)) {
708209ff23fSmrg	    pBest = pMode;
709209ff23fSmrg	    break;
710209ff23fSmrg	}
711209ff23fSmrg
712209ff23fSmrg	/* Reject if it's larger than the desired mode. */
713209ff23fSmrg	if (pScan->HDisplay > pMode->HDisplay ||
714209ff23fSmrg	    pScan->VDisplay > pMode->VDisplay)
715209ff23fSmrg	{
716209ff23fSmrg	    continue;
717209ff23fSmrg	}
718209ff23fSmrg
719209ff23fSmrg	if (pBest == NULL) {
720209ff23fSmrg	    pBest = pScan;
721209ff23fSmrg	    continue;
722209ff23fSmrg	}
723209ff23fSmrg
724209ff23fSmrg	/* Find if it's closer to the right size than the current best
725209ff23fSmrg	 * option.
726209ff23fSmrg	 */
727209ff23fSmrg	if ((pScan->HDisplay > pBest->HDisplay &&
728209ff23fSmrg	     pScan->VDisplay >= pBest->VDisplay) ||
729209ff23fSmrg	    (pScan->HDisplay >= pBest->HDisplay &&
730209ff23fSmrg	     pScan->VDisplay > pBest->VDisplay))
731209ff23fSmrg	{
732209ff23fSmrg	    pBest = pScan;
733209ff23fSmrg	    continue;
734209ff23fSmrg	}
735209ff23fSmrg
736209ff23fSmrg	/* Find if it's still closer to the right refresh than the current
737209ff23fSmrg	 * best resolution.
738209ff23fSmrg	 */
739209ff23fSmrg	if (pScan->HDisplay == pBest->HDisplay &&
740209ff23fSmrg	    pScan->VDisplay == pBest->VDisplay &&
741209ff23fSmrg	    (fabs(pScan->VRefresh - pMode->VRefresh) <
742209ff23fSmrg	     fabs(pBest->VRefresh - pMode->VRefresh))) {
743209ff23fSmrg	    pBest = pScan;
744209ff23fSmrg	}
745209ff23fSmrg    }
746209ff23fSmrg
747209ff23fSmrg    if (pBest == NULL) {
748209ff23fSmrg	xf86DrvMsg(pScrn->scrnIndex, X_WARNING,
749209ff23fSmrg		   "No suitable mode found to program for the pipe.\n"
750209ff23fSmrg		   "	continuing with desired mode %dx%d@%.1f\n",
751209ff23fSmrg		   pMode->HDisplay, pMode->VDisplay, pMode->VRefresh);
752209ff23fSmrg    } else if (!xf86ModesEqual(pBest, pMode)) {
753209ff23fSmrg      RADEONCrtcPrivatePtr  radeon_crtc = crtc->driver_private;
754209ff23fSmrg      int		    crtc = radeon_crtc->crtc_id;
755209ff23fSmrg      xf86DrvMsg(pScrn->scrnIndex, X_WARNING,
756209ff23fSmrg		   "Choosing pipe %d's mode %dx%d@%.1f instead of xf86 "
757209ff23fSmrg		   "mode %dx%d@%.1f\n", crtc,
758209ff23fSmrg		   pBest->HDisplay, pBest->VDisplay, pBest->VRefresh,
759209ff23fSmrg		   pMode->HDisplay, pMode->VDisplay, pMode->VRefresh);
760209ff23fSmrg	pMode = pBest;
761209ff23fSmrg    }
762209ff23fSmrg    return pMode;
763209ff23fSmrg}
764209ff23fSmrg
765209ff23fSmrgvoid
766209ff23fSmrgRADEONBlank(ScrnInfoPtr pScrn)
767209ff23fSmrg{
768209ff23fSmrg    xf86CrtcConfigPtr   xf86_config = XF86_CRTC_CONFIG_PTR(pScrn);
769209ff23fSmrg    xf86OutputPtr output;
770209ff23fSmrg    xf86CrtcPtr crtc;
771209ff23fSmrg    int o, c;
772209ff23fSmrg
773209ff23fSmrg    for (c = 0; c < xf86_config->num_crtc; c++) {
774209ff23fSmrg	crtc = xf86_config->crtc[c];
775209ff23fSmrg	for (o = 0; o < xf86_config->num_output; o++) {
776209ff23fSmrg	    output = xf86_config->output[o];
777209ff23fSmrg	    if (output->crtc != crtc)
778209ff23fSmrg		continue;
779209ff23fSmrg
780209ff23fSmrg	    output->funcs->dpms(output, DPMSModeOff);
781209ff23fSmrg	}
782209ff23fSmrg	crtc->funcs->dpms(crtc, DPMSModeOff);
783209ff23fSmrg    }
784209ff23fSmrg}
785209ff23fSmrg
786209ff23fSmrgvoid
787209ff23fSmrgRADEONUnblank(ScrnInfoPtr pScrn)
788209ff23fSmrg{
789209ff23fSmrg    xf86CrtcConfigPtr   xf86_config = XF86_CRTC_CONFIG_PTR(pScrn);
790209ff23fSmrg    xf86OutputPtr output;
791209ff23fSmrg    xf86CrtcPtr crtc;
792209ff23fSmrg    int o, c;
793209ff23fSmrg
794209ff23fSmrg    for (c = 0; c < xf86_config->num_crtc; c++) {
795209ff23fSmrg	crtc = xf86_config->crtc[c];
796209ff23fSmrg	if(!crtc->enabled)
797209ff23fSmrg		continue;
798209ff23fSmrg	crtc->funcs->dpms(crtc, DPMSModeOn);
799209ff23fSmrg	for (o = 0; o < xf86_config->num_output; o++) {
800209ff23fSmrg	    output = xf86_config->output[o];
801209ff23fSmrg	    if (output->crtc != crtc)
802209ff23fSmrg		continue;
803209ff23fSmrg
804209ff23fSmrg	    output->funcs->dpms(output, DPMSModeOn);
805209ff23fSmrg	}
806209ff23fSmrg    }
807209ff23fSmrg}
808209ff23fSmrg
809209ff23fSmrgBool
810209ff23fSmrgRADEONSetTiling(ScrnInfoPtr pScrn)
811209ff23fSmrg{
812209ff23fSmrg    xf86CrtcConfigPtr   xf86_config = XF86_CRTC_CONFIG_PTR(pScrn);
813209ff23fSmrg    RADEONInfoPtr info = RADEONPTR(pScrn);
814209ff23fSmrg    RADEONCrtcPrivatePtr radeon_crtc;
815209ff23fSmrg    xf86CrtcPtr crtc;
816209ff23fSmrg    int c;
817209ff23fSmrg    int can_tile = 1;
818209ff23fSmrg    Bool changed = FALSE;
819209ff23fSmrg
820209ff23fSmrg    for (c = 0; c < xf86_config->num_crtc; c++) {
821209ff23fSmrg	crtc = xf86_config->crtc[c];
822209ff23fSmrg	radeon_crtc = crtc->driver_private;
823209ff23fSmrg
824209ff23fSmrg	if (crtc->enabled) {
825209ff23fSmrg	    if (!radeon_crtc->can_tile)
826209ff23fSmrg		can_tile = 0;
827209ff23fSmrg	}
828209ff23fSmrg    }
829209ff23fSmrg
830209ff23fSmrg    if (info->tilingEnabled != can_tile)
831209ff23fSmrg	changed = TRUE;
832209ff23fSmrg
833209ff23fSmrg#ifdef XF86DRI
834209ff23fSmrg    if (info->directRenderingEnabled && (info->tilingEnabled != can_tile)) {
835b7e1c893Smrg	drm_radeon_sarea_t *pSAREAPriv;
836209ff23fSmrg	if (RADEONDRISetParam(pScrn, RADEON_SETPARAM_SWITCH_TILING, (can_tile ? 1 : 0)) < 0)
837209ff23fSmrg	    xf86DrvMsg(pScrn->scrnIndex, X_ERROR,
838209ff23fSmrg		       "[drm] failed changing tiling status\n");
839209ff23fSmrg	/* if this is called during ScreenInit() we don't have pScrn->pScreen yet */
840209ff23fSmrg	pSAREAPriv = DRIGetSAREAPrivate(screenInfo.screens[pScrn->scrnIndex]);
841209ff23fSmrg	info->tilingEnabled = pSAREAPriv->tiling_enabled ? TRUE : FALSE;
842209ff23fSmrg    }
843209ff23fSmrg#endif
844209ff23fSmrg
845209ff23fSmrg    return changed;
846209ff23fSmrg}
847