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