radeon_crtc.c revision ad43ddac
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 135ad43ddacSmrgstatic void 136ad43ddacSmrgRADEONComputePLL_old(RADEONPLLPtr pll, 137ad43ddacSmrg unsigned long freq, 138ad43ddacSmrg uint32_t *chosen_dot_clock_freq, 139ad43ddacSmrg uint32_t *chosen_feedback_div, 140ad43ddacSmrg uint32_t *chosen_frac_feedback_div, 141ad43ddacSmrg uint32_t *chosen_reference_div, 142ad43ddacSmrg uint32_t *chosen_post_div, 143ad43ddacSmrg int flags) 144209ff23fSmrg{ 145209ff23fSmrg uint32_t min_ref_div = pll->min_ref_div; 146209ff23fSmrg uint32_t max_ref_div = pll->max_ref_div; 147ad43ddacSmrg uint32_t min_post_div = pll->min_post_div; 148ad43ddacSmrg uint32_t max_post_div = pll->max_post_div; 149ad43ddacSmrg uint32_t min_fractional_feed_div = 0; 150ad43ddacSmrg uint32_t max_fractional_feed_div = 0; 151209ff23fSmrg uint32_t best_vco = pll->best_vco; 152209ff23fSmrg uint32_t best_post_div = 1; 153209ff23fSmrg uint32_t best_ref_div = 1; 154209ff23fSmrg uint32_t best_feedback_div = 1; 155ad43ddacSmrg uint32_t best_frac_feedback_div = 0; 156209ff23fSmrg uint32_t best_freq = -1; 157209ff23fSmrg uint32_t best_error = 0xffffffff; 158209ff23fSmrg uint32_t best_vco_diff = 1; 159209ff23fSmrg uint32_t post_div; 160209ff23fSmrg 161209ff23fSmrg freq = freq * 1000; 162209ff23fSmrg 163209ff23fSmrg ErrorF("freq: %lu\n", freq); 164209ff23fSmrg 165209ff23fSmrg if (flags & RADEON_PLL_USE_REF_DIV) 166209ff23fSmrg min_ref_div = max_ref_div = pll->reference_div; 167209ff23fSmrg else { 168209ff23fSmrg while (min_ref_div < max_ref_div-1) { 169209ff23fSmrg uint32_t mid=(min_ref_div+max_ref_div)/2; 170209ff23fSmrg uint32_t pll_in = pll->reference_freq / mid; 171209ff23fSmrg if (pll_in < pll->pll_in_min) 172209ff23fSmrg max_ref_div = mid; 173209ff23fSmrg else if (pll_in > pll->pll_in_max) 174209ff23fSmrg min_ref_div = mid; 175209ff23fSmrg else break; 176209ff23fSmrg } 177209ff23fSmrg } 178209ff23fSmrg 179ad43ddacSmrg if (flags & RADEON_PLL_USE_POST_DIV) 180ad43ddacSmrg min_post_div = max_post_div = pll->post_div; 181ad43ddacSmrg 182ad43ddacSmrg if (flags & RADEON_PLL_USE_FRAC_FB_DIV) { 183ad43ddacSmrg min_fractional_feed_div = pll->min_frac_feedback_div; 184ad43ddacSmrg max_fractional_feed_div = pll->max_frac_feedback_div; 185ad43ddacSmrg } 186ad43ddacSmrg 187ad43ddacSmrg for (post_div = min_post_div; post_div <= max_post_div; ++post_div) { 188209ff23fSmrg uint32_t ref_div; 189209ff23fSmrg 190209ff23fSmrg if ((flags & RADEON_PLL_NO_ODD_POST_DIV) && (post_div & 1)) 191209ff23fSmrg continue; 192209ff23fSmrg 193209ff23fSmrg /* legacy radeons only have a few post_divs */ 194209ff23fSmrg if (flags & RADEON_PLL_LEGACY) { 195209ff23fSmrg if ((post_div == 5) || 196209ff23fSmrg (post_div == 7) || 197209ff23fSmrg (post_div == 9) || 198209ff23fSmrg (post_div == 10) || 199209ff23fSmrg (post_div == 11)) 200209ff23fSmrg continue; 201209ff23fSmrg } 202209ff23fSmrg 203209ff23fSmrg for (ref_div = min_ref_div; ref_div <= max_ref_div; ++ref_div) { 204ad43ddacSmrg uint32_t feedback_div, current_freq = 0, error, vco_diff; 205209ff23fSmrg uint32_t pll_in = pll->reference_freq / ref_div; 206209ff23fSmrg uint32_t min_feed_div = pll->min_feedback_div; 207209ff23fSmrg uint32_t max_feed_div = pll->max_feedback_div+1; 208209ff23fSmrg 209209ff23fSmrg if (pll_in < pll->pll_in_min || pll_in > pll->pll_in_max) 210209ff23fSmrg continue; 211209ff23fSmrg 212209ff23fSmrg while (min_feed_div < max_feed_div) { 213209ff23fSmrg uint32_t vco; 214ad43ddacSmrg uint32_t min_frac_feed_div = min_fractional_feed_div; 215ad43ddacSmrg uint32_t max_frac_feed_div = max_fractional_feed_div+1; 216ad43ddacSmrg uint32_t frac_feedback_div; 217ad43ddacSmrg CARD64 tmp; 218209ff23fSmrg 219209ff23fSmrg feedback_div = (min_feed_div+max_feed_div)/2; 220209ff23fSmrg 221ad43ddacSmrg tmp = (CARD64)pll->reference_freq * feedback_div; 222ad43ddacSmrg vco = RADEONDiv(tmp, ref_div); 223209ff23fSmrg 224209ff23fSmrg if (vco < pll->pll_out_min) { 225209ff23fSmrg min_feed_div = feedback_div+1; 226209ff23fSmrg continue; 227209ff23fSmrg } else if(vco > pll->pll_out_max) { 228209ff23fSmrg max_feed_div = feedback_div; 229209ff23fSmrg continue; 230209ff23fSmrg } 231209ff23fSmrg 232ad43ddacSmrg while (min_frac_feed_div < max_frac_feed_div) { 233ad43ddacSmrg frac_feedback_div = (min_frac_feed_div+max_frac_feed_div)/2; 234ad43ddacSmrg tmp = (CARD64)pll->reference_freq * 10000 * feedback_div; 235ad43ddacSmrg tmp += (CARD64)pll->reference_freq * 1000 * frac_feedback_div; 236ad43ddacSmrg current_freq = RADEONDiv(tmp, ref_div * post_div); 237ad43ddacSmrg 238ad43ddacSmrg if (flags & RADEON_PLL_PREFER_CLOSEST_LOWER) { 239ad43ddacSmrg error = freq - current_freq; 240ad43ddacSmrg error = error < 0 ? 0xffffffff : error; 241ad43ddacSmrg } else 242ad43ddacSmrg error = abs(current_freq - freq); 243ad43ddacSmrg vco_diff = abs(vco - best_vco); 244ad43ddacSmrg 245ad43ddacSmrg if ((best_vco == 0 && error < best_error) || 246ad43ddacSmrg (best_vco != 0 && 247ad43ddacSmrg (error < best_error - 100 || 248ad43ddacSmrg (abs(error - best_error) < 100 && vco_diff < best_vco_diff )))) { 249209ff23fSmrg best_post_div = post_div; 250209ff23fSmrg best_ref_div = ref_div; 251209ff23fSmrg best_feedback_div = feedback_div; 252ad43ddacSmrg best_frac_feedback_div = frac_feedback_div; 253209ff23fSmrg best_freq = current_freq; 254209ff23fSmrg best_error = error; 255209ff23fSmrg best_vco_diff = vco_diff; 256ad43ddacSmrg } else if (current_freq == freq) { 257ad43ddacSmrg if (best_freq == -1) { 258ad43ddacSmrg best_post_div = post_div; 259ad43ddacSmrg best_ref_div = ref_div; 260ad43ddacSmrg best_feedback_div = feedback_div; 261ad43ddacSmrg best_frac_feedback_div = frac_feedback_div; 262ad43ddacSmrg best_freq = current_freq; 263ad43ddacSmrg best_error = error; 264ad43ddacSmrg best_vco_diff = vco_diff; 265ad43ddacSmrg } else if (((flags & RADEON_PLL_PREFER_LOW_REF_DIV) && (ref_div < best_ref_div)) || 266ad43ddacSmrg ((flags & RADEON_PLL_PREFER_HIGH_REF_DIV) && (ref_div > best_ref_div)) || 267ad43ddacSmrg ((flags & RADEON_PLL_PREFER_LOW_FB_DIV) && (feedback_div < best_feedback_div)) || 268ad43ddacSmrg ((flags & RADEON_PLL_PREFER_HIGH_FB_DIV) && (feedback_div > best_feedback_div)) || 269ad43ddacSmrg ((flags & RADEON_PLL_PREFER_LOW_POST_DIV) && (post_div < best_post_div)) || 270ad43ddacSmrg ((flags & RADEON_PLL_PREFER_HIGH_POST_DIV) && (post_div > best_post_div))) { 271ad43ddacSmrg best_post_div = post_div; 272ad43ddacSmrg best_ref_div = ref_div; 273ad43ddacSmrg best_feedback_div = feedback_div; 274ad43ddacSmrg best_frac_feedback_div = frac_feedback_div; 275ad43ddacSmrg best_freq = current_freq; 276ad43ddacSmrg best_error = error; 277ad43ddacSmrg best_vco_diff = vco_diff; 278ad43ddacSmrg } 279209ff23fSmrg } 280ad43ddacSmrg if (current_freq < freq) 281ad43ddacSmrg min_frac_feed_div = frac_feedback_div+1; 282ad43ddacSmrg else 283ad43ddacSmrg max_frac_feed_div = frac_feedback_div; 284209ff23fSmrg } 285209ff23fSmrg if (current_freq < freq) 286209ff23fSmrg min_feed_div = feedback_div+1; 287209ff23fSmrg else 288209ff23fSmrg max_feed_div = feedback_div; 289209ff23fSmrg } 290209ff23fSmrg } 291209ff23fSmrg } 292209ff23fSmrg 293209ff23fSmrg ErrorF("best_freq: %u\n", (unsigned int)best_freq); 294209ff23fSmrg ErrorF("best_feedback_div: %u\n", (unsigned int)best_feedback_div); 295ad43ddacSmrg ErrorF("best_frac_feedback_div: %u\n", (unsigned int)best_frac_feedback_div); 296209ff23fSmrg ErrorF("best_ref_div: %u\n", (unsigned int)best_ref_div); 297209ff23fSmrg ErrorF("best_post_div: %u\n", (unsigned int)best_post_div); 298209ff23fSmrg 299209ff23fSmrg if (best_freq == -1) 300209ff23fSmrg FatalError("Couldn't find valid PLL dividers\n"); 301209ff23fSmrg *chosen_dot_clock_freq = best_freq / 10000; 302209ff23fSmrg *chosen_feedback_div = best_feedback_div; 303ad43ddacSmrg *chosen_frac_feedback_div = best_frac_feedback_div; 304209ff23fSmrg *chosen_reference_div = best_ref_div; 305209ff23fSmrg *chosen_post_div = best_post_div; 306209ff23fSmrg 307209ff23fSmrg} 308209ff23fSmrg 309ad43ddacSmrgstatic Bool 310ad43ddacSmrgcalc_fb_div(RADEONPLLPtr pll, 311ad43ddacSmrg unsigned long freq, 312ad43ddacSmrg int flags, 313ad43ddacSmrg int post_div, 314ad43ddacSmrg int ref_div, 315ad43ddacSmrg int *fb_div, 316ad43ddacSmrg int *fb_div_frac) 317ad43ddacSmrg{ 318ad43ddacSmrg float ffreq = freq / 10; 319ad43ddacSmrg float vco_freq = ffreq * post_div; 320ad43ddacSmrg float feedback_divider = vco_freq * ref_div / pll->reference_freq; 321ad43ddacSmrg 322ad43ddacSmrg if (flags & RADEON_PLL_USE_FRAC_FB_DIV) { 323ad43ddacSmrg feedback_divider = floor((feedback_divider * 10.0) + 0.5) * 0.1; 324ad43ddacSmrg 325ad43ddacSmrg *fb_div = floor(feedback_divider); 326ad43ddacSmrg *fb_div_frac = fmod(feedback_divider, 1.0) * 10.0; 327ad43ddacSmrg 328ad43ddacSmrg } else { 329ad43ddacSmrg *fb_div = floor(feedback_divider + 0.5); 330ad43ddacSmrg *fb_div_frac = 0; 331ad43ddacSmrg } 332ad43ddacSmrg if ((*fb_div < pll->min_feedback_div) || (*fb_div > pll->max_feedback_div)) 333ad43ddacSmrg return FALSE; 334ad43ddacSmrg else 335ad43ddacSmrg return TRUE; 336ad43ddacSmrg} 337ad43ddacSmrg 338ad43ddacSmrgstatic Bool 339ad43ddacSmrgcalc_fb_ref_div(RADEONPLLPtr pll, 340ad43ddacSmrg unsigned long freq, 341ad43ddacSmrg int flags, 342ad43ddacSmrg int post_div, 343ad43ddacSmrg int *fb_div, 344ad43ddacSmrg int *fb_div_frac, 345ad43ddacSmrg int *ref_div) 346ad43ddacSmrg{ 347ad43ddacSmrg float ffreq = freq / 10; 348ad43ddacSmrg float max_error = ffreq * 0.0025; 349ad43ddacSmrg float vco, error, pll_out; 350ad43ddacSmrg 351ad43ddacSmrg for ((*ref_div) = pll->min_ref_div; (*ref_div) < pll->max_ref_div; ++(*ref_div)) { 352ad43ddacSmrg if (calc_fb_div(pll, freq, flags, post_div, (*ref_div), fb_div, fb_div_frac)) { 353ad43ddacSmrg vco = pll->reference_freq * ((*fb_div) + ((*fb_div_frac) * 0.1)) / (*ref_div); 354ad43ddacSmrg 355ad43ddacSmrg if ((vco < pll->pll_out_min) || (vco > pll->pll_out_max)) 356ad43ddacSmrg continue; 357ad43ddacSmrg 358ad43ddacSmrg pll_out = vco / post_div; 359ad43ddacSmrg 360ad43ddacSmrg error = pll_out - ffreq; 361ad43ddacSmrg if ((fabs(error) <= max_error) && (error >= 0)) 362ad43ddacSmrg return TRUE; 363ad43ddacSmrg } 364ad43ddacSmrg } 365ad43ddacSmrg return FALSE; 366ad43ddacSmrg} 367ad43ddacSmrg 368ad43ddacSmrgstatic void 369ad43ddacSmrgRADEONComputePLL_new(RADEONPLLPtr pll, 370ad43ddacSmrg unsigned long freq, 371ad43ddacSmrg uint32_t *chosen_dot_clock_freq, 372ad43ddacSmrg uint32_t *chosen_feedback_div, 373ad43ddacSmrg uint32_t *chosen_frac_feedback_div, 374ad43ddacSmrg uint32_t *chosen_reference_div, 375ad43ddacSmrg uint32_t *chosen_post_div, 376ad43ddacSmrg int flags) 377ad43ddacSmrg{ 378ad43ddacSmrg float ffreq = freq / 10; 379ad43ddacSmrg float vco_frequency; 380ad43ddacSmrg int fb_div = 0, fb_div_frac = 0, post_div = 0, ref_div = 0; 381ad43ddacSmrg uint32_t best_freq = 0; 382ad43ddacSmrg 383ad43ddacSmrg if (flags & RADEON_PLL_USE_POST_DIV) { 384ad43ddacSmrg post_div = pll->post_div; 385ad43ddacSmrg if ((post_div < pll->min_post_div) || (post_div > pll->max_post_div)) 386ad43ddacSmrg goto done; 387ad43ddacSmrg vco_frequency = ffreq * post_div; 388ad43ddacSmrg if ((vco_frequency < pll->pll_out_min) || (vco_frequency > pll->pll_out_max)) 389ad43ddacSmrg goto done; 390ad43ddacSmrg 391ad43ddacSmrg if (flags & RADEON_PLL_USE_REF_DIV) { 392ad43ddacSmrg ref_div = pll->reference_div; 393ad43ddacSmrg if ((ref_div < pll->min_ref_div) || (ref_div > pll->max_ref_div)) 394ad43ddacSmrg goto done; 395ad43ddacSmrg if (!calc_fb_div(pll, freq, flags, post_div, ref_div, &fb_div, &fb_div_frac)) 396ad43ddacSmrg goto done; 397ad43ddacSmrg } 398ad43ddacSmrg } else { 399ad43ddacSmrg for (post_div = pll->max_post_div; post_div >= pll->min_post_div; --post_div) { 400ad43ddacSmrg if (flags & RADEON_PLL_LEGACY) { 401ad43ddacSmrg if ((post_div == 5) || 402ad43ddacSmrg (post_div == 7) || 403ad43ddacSmrg (post_div == 9) || 404ad43ddacSmrg (post_div == 10) || 405ad43ddacSmrg (post_div == 11)) 406ad43ddacSmrg continue; 407ad43ddacSmrg } 408ad43ddacSmrg if ((flags & RADEON_PLL_NO_ODD_POST_DIV) && (post_div & 1)) 409ad43ddacSmrg continue; 410ad43ddacSmrg 411ad43ddacSmrg vco_frequency = ffreq * post_div; 412ad43ddacSmrg if ((vco_frequency < pll->pll_out_min) || (vco_frequency > pll->pll_out_max)) 413ad43ddacSmrg continue; 414ad43ddacSmrg if (flags & RADEON_PLL_USE_REF_DIV) { 415ad43ddacSmrg ref_div = pll->reference_div; 416ad43ddacSmrg if ((ref_div < pll->min_ref_div) || (ref_div > pll->max_ref_div)) 417ad43ddacSmrg goto done; 418ad43ddacSmrg if (calc_fb_div(pll, freq, flags, post_div, ref_div, &fb_div, &fb_div_frac)) 419ad43ddacSmrg break; 420ad43ddacSmrg } else { 421ad43ddacSmrg if (calc_fb_ref_div(pll, freq, flags, post_div, &fb_div, &fb_div_frac, &ref_div)) 422ad43ddacSmrg break; 423ad43ddacSmrg } 424ad43ddacSmrg } 425ad43ddacSmrg } 426ad43ddacSmrg 427ad43ddacSmrg best_freq = pll->reference_freq * 10 * fb_div; 428ad43ddacSmrg best_freq += pll->reference_freq * fb_div_frac; 429ad43ddacSmrg best_freq = best_freq / (ref_div * post_div); 430ad43ddacSmrg 431ad43ddacSmrg ErrorF("best_freq: %u\n", (unsigned int)best_freq); 432ad43ddacSmrg ErrorF("best_feedback_div: %u\n", (unsigned int)fb_div); 433ad43ddacSmrg ErrorF("best_frac_feedback_div: %u\n", (unsigned int)fb_div_frac); 434ad43ddacSmrg ErrorF("best_ref_div: %u\n", (unsigned int)ref_div); 435ad43ddacSmrg ErrorF("best_post_div: %u\n", (unsigned int)post_div); 436ad43ddacSmrg 437ad43ddacSmrgdone: 438ad43ddacSmrg if (best_freq == 0) 439ad43ddacSmrg FatalError("Couldn't find valid PLL dividers\n"); 440ad43ddacSmrg 441ad43ddacSmrg *chosen_dot_clock_freq = best_freq; 442ad43ddacSmrg *chosen_feedback_div = fb_div; 443ad43ddacSmrg *chosen_frac_feedback_div = fb_div_frac; 444ad43ddacSmrg *chosen_reference_div = ref_div; 445ad43ddacSmrg *chosen_post_div = post_div; 446ad43ddacSmrg 447ad43ddacSmrg} 448ad43ddacSmrg 449ad43ddacSmrgvoid 450ad43ddacSmrgRADEONComputePLL(ScrnInfoPtr pScrn, 451ad43ddacSmrg RADEONPLLPtr pll, 452ad43ddacSmrg unsigned long freq, 453ad43ddacSmrg uint32_t *chosen_dot_clock_freq, 454ad43ddacSmrg uint32_t *chosen_feedback_div, 455ad43ddacSmrg uint32_t *chosen_frac_feedback_div, 456ad43ddacSmrg uint32_t *chosen_reference_div, 457ad43ddacSmrg uint32_t *chosen_post_div, 458ad43ddacSmrg int flags) 459ad43ddacSmrg{ 460ad43ddacSmrg RADEONInfoPtr info = RADEONPTR(pScrn); 461ad43ddacSmrg 462ad43ddacSmrg if (IS_AVIVO_VARIANT) { 463ad43ddacSmrg if (xf86ReturnOptValBool(info->Options, OPTION_NEW_PLL, TRUE)) { 464ad43ddacSmrg /* disable frac fb dividers */ 465ad43ddacSmrg flags &= ~RADEON_PLL_USE_FRAC_FB_DIV; 466ad43ddacSmrg RADEONComputePLL_new(pll, freq, chosen_dot_clock_freq, 467ad43ddacSmrg chosen_feedback_div, chosen_frac_feedback_div, 468ad43ddacSmrg chosen_reference_div, chosen_post_div, flags); 469ad43ddacSmrg } else 470ad43ddacSmrg RADEONComputePLL_old(pll, freq, chosen_dot_clock_freq, 471ad43ddacSmrg chosen_feedback_div, chosen_frac_feedback_div, 472ad43ddacSmrg chosen_reference_div, chosen_post_div, flags); 473ad43ddacSmrg } else { 474ad43ddacSmrg if (xf86ReturnOptValBool(info->Options, OPTION_NEW_PLL, FALSE)) 475ad43ddacSmrg RADEONComputePLL_new(pll, freq, chosen_dot_clock_freq, 476ad43ddacSmrg chosen_feedback_div, chosen_frac_feedback_div, 477ad43ddacSmrg chosen_reference_div, chosen_post_div, flags); 478ad43ddacSmrg else 479ad43ddacSmrg RADEONComputePLL_old(pll, freq, chosen_dot_clock_freq, 480ad43ddacSmrg chosen_feedback_div, chosen_frac_feedback_div, 481ad43ddacSmrg chosen_reference_div, chosen_post_div, flags); 482ad43ddacSmrg } 483ad43ddacSmrg} 484ad43ddacSmrg 485209ff23fSmrgstatic void 486209ff23fSmrgradeon_crtc_mode_set(xf86CrtcPtr crtc, DisplayModePtr mode, 487209ff23fSmrg DisplayModePtr adjusted_mode, int x, int y) 488209ff23fSmrg{ 489209ff23fSmrg ScrnInfoPtr pScrn = crtc->scrn; 490209ff23fSmrg RADEONInfoPtr info = RADEONPTR(pScrn); 491209ff23fSmrg 492b7e1c893Smrg if (IS_AVIVO_VARIANT || info->r4xx_atom) { 493209ff23fSmrg atombios_crtc_mode_set(crtc, mode, adjusted_mode, x, y); 494209ff23fSmrg } else { 495209ff23fSmrg legacy_crtc_mode_set(crtc, mode, adjusted_mode, x, y); 496209ff23fSmrg } 497209ff23fSmrg} 498209ff23fSmrg 499209ff23fSmrgstatic void 500209ff23fSmrgradeon_crtc_mode_commit(xf86CrtcPtr crtc) 501209ff23fSmrg{ 502209ff23fSmrg if (crtc->scrn->pScreen != NULL) 503209ff23fSmrg xf86_reload_cursors(crtc->scrn->pScreen); 504209ff23fSmrg} 505209ff23fSmrg 506209ff23fSmrgvoid 507209ff23fSmrgradeon_crtc_load_lut(xf86CrtcPtr crtc) 508209ff23fSmrg{ 509209ff23fSmrg ScrnInfoPtr pScrn = crtc->scrn; 510209ff23fSmrg RADEONCrtcPrivatePtr radeon_crtc = crtc->driver_private; 511209ff23fSmrg RADEONInfoPtr info = RADEONPTR(pScrn); 512209ff23fSmrg unsigned char *RADEONMMIO = info->MMIO; 513209ff23fSmrg int i; 514209ff23fSmrg 515209ff23fSmrg if (!crtc->enabled) 516209ff23fSmrg return; 517209ff23fSmrg 518ad43ddacSmrg if (IS_DCE4_VARIANT) { 519ad43ddacSmrg OUTREG(EVERGREEN_DC_LUT_CONTROL + radeon_crtc->crtc_offset, 0); 520209ff23fSmrg 521ad43ddacSmrg OUTREG(EVERGREEN_DC_LUT_BLACK_OFFSET_BLUE + radeon_crtc->crtc_offset, 0); 522ad43ddacSmrg OUTREG(EVERGREEN_DC_LUT_BLACK_OFFSET_GREEN + radeon_crtc->crtc_offset, 0); 523ad43ddacSmrg OUTREG(EVERGREEN_DC_LUT_BLACK_OFFSET_RED + radeon_crtc->crtc_offset, 0); 524209ff23fSmrg 525ad43ddacSmrg OUTREG(EVERGREEN_DC_LUT_WHITE_OFFSET_BLUE + radeon_crtc->crtc_offset, 0x0000ffff); 526ad43ddacSmrg OUTREG(EVERGREEN_DC_LUT_WHITE_OFFSET_GREEN + radeon_crtc->crtc_offset, 0x0000ffff); 527ad43ddacSmrg OUTREG(EVERGREEN_DC_LUT_WHITE_OFFSET_RED + radeon_crtc->crtc_offset, 0x0000ffff); 528209ff23fSmrg 529ad43ddacSmrg OUTREG(EVERGREEN_DC_LUT_RW_MODE + radeon_crtc->crtc_offset, 0); 530ad43ddacSmrg OUTREG(EVERGREEN_DC_LUT_WRITE_EN_MASK + radeon_crtc->crtc_offset, 0x00000007); 531209ff23fSmrg 532ad43ddacSmrg for (i = 0; i < 256; i++) { 533ad43ddacSmrg OUTREG(EVERGREEN_DC_LUT_RW_INDEX + radeon_crtc->crtc_offset, i); 534ad43ddacSmrg OUTREG(EVERGREEN_DC_LUT_30_COLOR + radeon_crtc->crtc_offset, 535ad43ddacSmrg (((radeon_crtc->lut_r[i]) << 20) | 536ad43ddacSmrg ((radeon_crtc->lut_g[i]) << 10) | 537ad43ddacSmrg (radeon_crtc->lut_b[i]))); 538ad43ddacSmrg } 539ad43ddacSmrg } else { 540ad43ddacSmrg if (IS_AVIVO_VARIANT) { 541ad43ddacSmrg OUTREG(AVIVO_DC_LUTA_CONTROL + radeon_crtc->crtc_offset, 0); 542209ff23fSmrg 543ad43ddacSmrg OUTREG(AVIVO_DC_LUTA_BLACK_OFFSET_BLUE + radeon_crtc->crtc_offset, 0); 544ad43ddacSmrg OUTREG(AVIVO_DC_LUTA_BLACK_OFFSET_GREEN + radeon_crtc->crtc_offset, 0); 545ad43ddacSmrg OUTREG(AVIVO_DC_LUTA_BLACK_OFFSET_RED + radeon_crtc->crtc_offset, 0); 546209ff23fSmrg 547ad43ddacSmrg OUTREG(AVIVO_DC_LUTA_WHITE_OFFSET_BLUE + radeon_crtc->crtc_offset, 0x0000ffff); 548ad43ddacSmrg OUTREG(AVIVO_DC_LUTA_WHITE_OFFSET_GREEN + radeon_crtc->crtc_offset, 0x0000ffff); 549ad43ddacSmrg OUTREG(AVIVO_DC_LUTA_WHITE_OFFSET_RED + radeon_crtc->crtc_offset, 0x0000ffff); 550ad43ddacSmrg } 551ad43ddacSmrg 552ad43ddacSmrg PAL_SELECT(radeon_crtc->crtc_id); 553ad43ddacSmrg 554ad43ddacSmrg if (IS_AVIVO_VARIANT) { 555ad43ddacSmrg OUTREG(AVIVO_DC_LUT_RW_MODE, 0); 556ad43ddacSmrg OUTREG(AVIVO_DC_LUT_WRITE_EN_MASK, 0x0000003f); 557ad43ddacSmrg } 558ad43ddacSmrg 559ad43ddacSmrg for (i = 0; i < 256; i++) { 560ad43ddacSmrg OUTPAL(i, radeon_crtc->lut_r[i], radeon_crtc->lut_g[i], radeon_crtc->lut_b[i]); 561ad43ddacSmrg } 562ad43ddacSmrg 563ad43ddacSmrg if (IS_AVIVO_VARIANT) 564ad43ddacSmrg OUTREG(AVIVO_D1GRPH_LUT_SEL + radeon_crtc->crtc_offset, radeon_crtc->crtc_id); 565209ff23fSmrg } 566209ff23fSmrg 567209ff23fSmrg} 568209ff23fSmrg 569209ff23fSmrgstatic void 570209ff23fSmrgradeon_crtc_gamma_set(xf86CrtcPtr crtc, uint16_t *red, uint16_t *green, 571209ff23fSmrg uint16_t *blue, int size) 572209ff23fSmrg{ 573209ff23fSmrg RADEONCrtcPrivatePtr radeon_crtc = crtc->driver_private; 574209ff23fSmrg ScrnInfoPtr pScrn = crtc->scrn; 575209ff23fSmrg int i, j; 576209ff23fSmrg 577209ff23fSmrg if (pScrn->depth == 16) { 578209ff23fSmrg for (i = 0; i < 64; i++) { 579209ff23fSmrg if (i <= 31) { 580209ff23fSmrg for (j = 0; j < 8; j++) { 581b7e1c893Smrg radeon_crtc->lut_r[i * 8 + j] = red[i] >> 6; 582b7e1c893Smrg radeon_crtc->lut_b[i * 8 + j] = blue[i] >> 6; 583209ff23fSmrg } 584209ff23fSmrg } 585209ff23fSmrg 586209ff23fSmrg for (j = 0; j < 4; j++) { 587b7e1c893Smrg radeon_crtc->lut_g[i * 4 + j] = green[i] >> 6; 588209ff23fSmrg } 589209ff23fSmrg } 590209ff23fSmrg } else { 591209ff23fSmrg for (i = 0; i < 256; i++) { 592b7e1c893Smrg radeon_crtc->lut_r[i] = red[i] >> 6; 593b7e1c893Smrg radeon_crtc->lut_g[i] = green[i] >> 6; 594b7e1c893Smrg radeon_crtc->lut_b[i] = blue[i] >> 6; 595209ff23fSmrg } 596209ff23fSmrg } 597209ff23fSmrg 598209ff23fSmrg radeon_crtc_load_lut(crtc); 599209ff23fSmrg} 600209ff23fSmrg 601209ff23fSmrgstatic Bool 602209ff23fSmrgradeon_crtc_lock(xf86CrtcPtr crtc) 603209ff23fSmrg{ 604209ff23fSmrg ScrnInfoPtr pScrn = crtc->scrn; 605209ff23fSmrg RADEONInfoPtr info = RADEONPTR(pScrn); 606209ff23fSmrg 607209ff23fSmrg#ifdef XF86DRI 608b7e1c893Smrg if (info->cp->CPStarted && pScrn->pScreen) { 609209ff23fSmrg DRILock(pScrn->pScreen, 0); 610209ff23fSmrg if (info->accelOn) 611209ff23fSmrg RADEON_SYNC(info, pScrn); 612209ff23fSmrg return TRUE; 613209ff23fSmrg } 614209ff23fSmrg#endif 615209ff23fSmrg if (info->accelOn) 616209ff23fSmrg RADEON_SYNC(info, pScrn); 617209ff23fSmrg 618209ff23fSmrg return FALSE; 619209ff23fSmrg 620209ff23fSmrg} 621209ff23fSmrg 622209ff23fSmrgstatic void 623209ff23fSmrgradeon_crtc_unlock(xf86CrtcPtr crtc) 624209ff23fSmrg{ 625209ff23fSmrg ScrnInfoPtr pScrn = crtc->scrn; 626209ff23fSmrg RADEONInfoPtr info = RADEONPTR(pScrn); 627209ff23fSmrg 628209ff23fSmrg#ifdef XF86DRI 629b7e1c893Smrg if (info->cp->CPStarted && pScrn->pScreen) DRIUnlock(pScrn->pScreen); 630209ff23fSmrg#endif 631209ff23fSmrg 632209ff23fSmrg if (info->accelOn) 633209ff23fSmrg RADEON_SYNC(info, pScrn); 634209ff23fSmrg} 635209ff23fSmrg 636209ff23fSmrg/** 637209ff23fSmrg * Allocates memory for a locked-in-framebuffer shadow of the given 638209ff23fSmrg * width and height for this CRTC's rotated shadow framebuffer. 639209ff23fSmrg */ 640209ff23fSmrg 641209ff23fSmrgstatic void * 642209ff23fSmrgradeon_crtc_shadow_allocate (xf86CrtcPtr crtc, int width, int height) 643209ff23fSmrg{ 644209ff23fSmrg ScrnInfoPtr pScrn = crtc->scrn; 645209ff23fSmrg RADEONInfoPtr info = RADEONPTR(pScrn); 646209ff23fSmrg RADEONCrtcPrivatePtr radeon_crtc = crtc->driver_private; 647209ff23fSmrg unsigned long rotate_pitch; 648209ff23fSmrg unsigned long rotate_offset; 649ad43ddacSmrg int size; 650209ff23fSmrg int cpp = pScrn->bitsPerPixel / 8; 651209ff23fSmrg 652b7e1c893Smrg /* No rotation without accel */ 653b7e1c893Smrg if (((info->ChipFamily >= CHIP_FAMILY_R600) && !info->directRenderingEnabled) || 654b7e1c893Smrg xf86ReturnOptValBool(info->Options, OPTION_NOACCEL, FALSE)) { 655b7e1c893Smrg xf86DrvMsg(pScrn->scrnIndex, X_ERROR, 656b7e1c893Smrg "Acceleration required for rotation\n"); 657b7e1c893Smrg return NULL; 658b7e1c893Smrg } 659b7e1c893Smrg 660209ff23fSmrg rotate_pitch = pScrn->displayWidth * cpp; 661209ff23fSmrg size = rotate_pitch * height; 662209ff23fSmrg 663209ff23fSmrg /* We could get close to what we want here by just creating a pixmap like 664209ff23fSmrg * normal, but we have to lock it down in framebuffer, and there is no 665209ff23fSmrg * setter for offscreen area locking in EXA currently. So, we just 666209ff23fSmrg * allocate offscreen memory and fake up a pixmap header for it. 667209ff23fSmrg */ 668ad43ddacSmrg rotate_offset = radeon_legacy_allocate_memory(pScrn, &radeon_crtc->crtc_rotate_mem, 669ad43ddacSmrg size, RADEON_GPU_PAGE_SIZE, RADEON_GEM_DOMAIN_VRAM); 670b7e1c893Smrg if (rotate_offset == 0) 671b7e1c893Smrg return NULL; 672209ff23fSmrg 673209ff23fSmrg return info->FB + rotate_offset; 674209ff23fSmrg} 675b7e1c893Smrg 676209ff23fSmrg/** 677209ff23fSmrg * Creates a pixmap for this CRTC's rotated shadow framebuffer. 678209ff23fSmrg */ 679209ff23fSmrgstatic PixmapPtr 680209ff23fSmrgradeon_crtc_shadow_create(xf86CrtcPtr crtc, void *data, int width, int height) 681209ff23fSmrg{ 682209ff23fSmrg ScrnInfoPtr pScrn = crtc->scrn; 683209ff23fSmrg unsigned long rotate_pitch; 684209ff23fSmrg PixmapPtr rotate_pixmap; 685209ff23fSmrg int cpp = pScrn->bitsPerPixel / 8; 686209ff23fSmrg 687209ff23fSmrg if (!data) 688209ff23fSmrg data = radeon_crtc_shadow_allocate(crtc, width, height); 689209ff23fSmrg 690209ff23fSmrg rotate_pitch = pScrn->displayWidth * cpp; 691209ff23fSmrg 692209ff23fSmrg rotate_pixmap = GetScratchPixmapHeader(pScrn->pScreen, 693209ff23fSmrg width, height, 694209ff23fSmrg pScrn->depth, 695209ff23fSmrg pScrn->bitsPerPixel, 696209ff23fSmrg rotate_pitch, 697209ff23fSmrg data); 698209ff23fSmrg 699209ff23fSmrg if (rotate_pixmap == NULL) { 700209ff23fSmrg xf86DrvMsg(pScrn->scrnIndex, X_ERROR, 701209ff23fSmrg "Couldn't allocate shadow pixmap for rotated CRTC\n"); 702209ff23fSmrg } 703209ff23fSmrg 704209ff23fSmrg return rotate_pixmap; 705209ff23fSmrg} 706209ff23fSmrg 707209ff23fSmrgstatic void 708209ff23fSmrgradeon_crtc_shadow_destroy(xf86CrtcPtr crtc, PixmapPtr rotate_pixmap, void *data) 709209ff23fSmrg{ 710209ff23fSmrg ScrnInfoPtr pScrn = crtc->scrn; 711209ff23fSmrg RADEONCrtcPrivatePtr radeon_crtc = crtc->driver_private; 712209ff23fSmrg 713209ff23fSmrg if (rotate_pixmap) 714209ff23fSmrg FreeScratchPixmapHeader(rotate_pixmap); 715b7e1c893Smrg 716209ff23fSmrg if (data) { 717b7e1c893Smrg radeon_legacy_free_memory(pScrn, radeon_crtc->crtc_rotate_mem); 718b7e1c893Smrg radeon_crtc->crtc_rotate_mem = NULL; 719b7e1c893Smrg } 720b7e1c893Smrg 721b7e1c893Smrg} 722b7e1c893Smrg 723b7e1c893Smrg#if XF86_CRTC_VERSION >= 2 724b7e1c893Smrg#include "radeon_atombios.h" 725b7e1c893Smrg 726b7e1c893Smrgextern AtomBiosResult 727b7e1c893Smrgatombios_lock_crtc(atomBiosHandlePtr atomBIOS, int crtc, int lock); 728b7e1c893Smrgextern void 729b7e1c893SmrgRADEONInitCrtcBase(xf86CrtcPtr crtc, RADEONSavePtr save, 730b7e1c893Smrg int x, int y); 731b7e1c893Smrgextern void 732b7e1c893SmrgRADEONInitCrtc2Base(xf86CrtcPtr crtc, RADEONSavePtr save, 733b7e1c893Smrg int x, int y); 734b7e1c893Smrgextern void 735b7e1c893SmrgRADEONRestoreCrtcBase(ScrnInfoPtr pScrn, 736b7e1c893Smrg RADEONSavePtr restore); 737b7e1c893Smrgextern void 738b7e1c893SmrgRADEONRestoreCrtc2Base(ScrnInfoPtr pScrn, 739b7e1c893Smrg RADEONSavePtr restore); 740b7e1c893Smrg 741b7e1c893Smrgstatic void 742b7e1c893Smrgradeon_crtc_set_origin(xf86CrtcPtr crtc, int x, int y) 743b7e1c893Smrg{ 744b7e1c893Smrg ScrnInfoPtr pScrn = crtc->scrn; 745b7e1c893Smrg RADEONCrtcPrivatePtr radeon_crtc = crtc->driver_private; 746b7e1c893Smrg RADEONInfoPtr info = RADEONPTR(pScrn); 747b7e1c893Smrg unsigned char *RADEONMMIO = info->MMIO; 748b7e1c893Smrg 749ad43ddacSmrg 750ad43ddacSmrg if (IS_DCE4_VARIANT) { 751ad43ddacSmrg x &= ~3; 752ad43ddacSmrg y &= ~1; 753ad43ddacSmrg atombios_lock_crtc(info->atomBIOS, radeon_crtc->crtc_id, 1); 754ad43ddacSmrg OUTREG(EVERGREEN_VIEWPORT_START + radeon_crtc->crtc_offset, (x << 16) | y); 755ad43ddacSmrg atombios_lock_crtc(info->atomBIOS, radeon_crtc->crtc_id, 0); 756ad43ddacSmrg } else if (IS_AVIVO_VARIANT) { 757b7e1c893Smrg x &= ~3; 758b7e1c893Smrg y &= ~1; 759b7e1c893Smrg atombios_lock_crtc(info->atomBIOS, radeon_crtc->crtc_id, 1); 760b7e1c893Smrg OUTREG(AVIVO_D1MODE_VIEWPORT_START + radeon_crtc->crtc_offset, (x << 16) | y); 761b7e1c893Smrg atombios_lock_crtc(info->atomBIOS, radeon_crtc->crtc_id, 0); 762b7e1c893Smrg } else { 763b7e1c893Smrg switch (radeon_crtc->crtc_id) { 764b7e1c893Smrg case 0: 765b7e1c893Smrg RADEONInitCrtcBase(crtc, info->ModeReg, x, y); 766b7e1c893Smrg RADEONRestoreCrtcBase(pScrn, info->ModeReg); 767b7e1c893Smrg break; 768b7e1c893Smrg case 1: 769b7e1c893Smrg RADEONInitCrtc2Base(crtc, info->ModeReg, x, y); 770b7e1c893Smrg RADEONRestoreCrtc2Base(pScrn, info->ModeReg); 771b7e1c893Smrg break; 772b7e1c893Smrg default: 773b7e1c893Smrg break; 774209ff23fSmrg } 775209ff23fSmrg } 776209ff23fSmrg} 777b7e1c893Smrg#endif 778209ff23fSmrg 779b7e1c893Smrg 780b7e1c893Smrgstatic xf86CrtcFuncsRec radeon_crtc_funcs = { 781209ff23fSmrg .dpms = radeon_crtc_dpms, 782209ff23fSmrg .save = NULL, /* XXX */ 783209ff23fSmrg .restore = NULL, /* XXX */ 784209ff23fSmrg .mode_fixup = radeon_crtc_mode_fixup, 785209ff23fSmrg .prepare = radeon_crtc_mode_prepare, 786209ff23fSmrg .mode_set = radeon_crtc_mode_set, 787209ff23fSmrg .commit = radeon_crtc_mode_commit, 788209ff23fSmrg .gamma_set = radeon_crtc_gamma_set, 789209ff23fSmrg .lock = radeon_crtc_lock, 790209ff23fSmrg .unlock = radeon_crtc_unlock, 791209ff23fSmrg .shadow_create = radeon_crtc_shadow_create, 792209ff23fSmrg .shadow_allocate = radeon_crtc_shadow_allocate, 793209ff23fSmrg .shadow_destroy = radeon_crtc_shadow_destroy, 794209ff23fSmrg .set_cursor_colors = radeon_crtc_set_cursor_colors, 795209ff23fSmrg .set_cursor_position = radeon_crtc_set_cursor_position, 796209ff23fSmrg .show_cursor = radeon_crtc_show_cursor, 797209ff23fSmrg .hide_cursor = radeon_crtc_hide_cursor, 798209ff23fSmrg .load_cursor_argb = radeon_crtc_load_cursor_argb, 799209ff23fSmrg .destroy = NULL, /* XXX */ 800b7e1c893Smrg#if XF86_CRTC_VERSION >= 2 801b7e1c893Smrg .set_origin = radeon_crtc_set_origin, 802b7e1c893Smrg#endif 803209ff23fSmrg}; 804209ff23fSmrg 805b7e1c893Smrgvoid 806b7e1c893SmrgRADEONInitDispBandwidth(ScrnInfoPtr pScrn) 807b7e1c893Smrg{ 808b7e1c893Smrg RADEONInfoPtr info = RADEONPTR(pScrn); 809b7e1c893Smrg xf86CrtcConfigPtr xf86_config = XF86_CRTC_CONFIG_PTR(pScrn); 810b7e1c893Smrg DisplayModePtr mode1 = NULL, mode2 = NULL; 811b7e1c893Smrg int pixel_bytes1 = info->CurrentLayout.pixel_bytes; 812b7e1c893Smrg int pixel_bytes2 = info->CurrentLayout.pixel_bytes; 813b7e1c893Smrg 814ad43ddacSmrg /* XXX fix me */ 815ad43ddacSmrg if (IS_DCE4_VARIANT) 816ad43ddacSmrg return; 817ad43ddacSmrg 818b7e1c893Smrg if (xf86_config->num_crtc == 2) { 819b7e1c893Smrg if (xf86_config->crtc[1]->enabled && 820b7e1c893Smrg xf86_config->crtc[0]->enabled) { 821b7e1c893Smrg mode1 = &xf86_config->crtc[0]->mode; 822b7e1c893Smrg mode2 = &xf86_config->crtc[1]->mode; 823b7e1c893Smrg } else if (xf86_config->crtc[0]->enabled) { 824b7e1c893Smrg mode1 = &xf86_config->crtc[0]->mode; 825b7e1c893Smrg } else if (xf86_config->crtc[1]->enabled) { 826b7e1c893Smrg mode2 = &xf86_config->crtc[1]->mode; 827b7e1c893Smrg } else 828b7e1c893Smrg return; 829b7e1c893Smrg } else { 830b7e1c893Smrg if (info->IsPrimary) 831b7e1c893Smrg mode1 = &xf86_config->crtc[0]->mode; 832b7e1c893Smrg else if (info->IsSecondary) 833b7e1c893Smrg mode2 = &xf86_config->crtc[0]->mode; 834b7e1c893Smrg else if (xf86_config->crtc[0]->enabled) 835b7e1c893Smrg mode1 = &xf86_config->crtc[0]->mode; 836b7e1c893Smrg else 837b7e1c893Smrg return; 838b7e1c893Smrg } 839b7e1c893Smrg 840b7e1c893Smrg if (IS_AVIVO_VARIANT) 841b7e1c893Smrg RADEONInitDispBandwidthAVIVO(pScrn, mode1, pixel_bytes1, mode2, pixel_bytes2); 842b7e1c893Smrg else 843b7e1c893Smrg RADEONInitDispBandwidthLegacy(pScrn, mode1, pixel_bytes1, mode2, pixel_bytes2); 844b7e1c893Smrg} 845b7e1c893Smrg 846209ff23fSmrgBool RADEONAllocateControllers(ScrnInfoPtr pScrn, int mask) 847209ff23fSmrg{ 848209ff23fSmrg RADEONEntPtr pRADEONEnt = RADEONEntPriv(pScrn); 849209ff23fSmrg RADEONInfoPtr info = RADEONPTR(pScrn); 850ad43ddacSmrg int i; 851209ff23fSmrg 852b7e1c893Smrg if (!xf86ReturnOptValBool(info->Options, OPTION_NOACCEL, FALSE)) { 853b7e1c893Smrg radeon_crtc_funcs.shadow_create = radeon_crtc_shadow_create; 854b7e1c893Smrg radeon_crtc_funcs.shadow_allocate = radeon_crtc_shadow_allocate; 855b7e1c893Smrg radeon_crtc_funcs.shadow_destroy = radeon_crtc_shadow_destroy; 856b7e1c893Smrg } 857b7e1c893Smrg 858209ff23fSmrg if (mask & 1) { 859209ff23fSmrg if (pRADEONEnt->Controller[0]) 860209ff23fSmrg return TRUE; 861209ff23fSmrg 862209ff23fSmrg pRADEONEnt->pCrtc[0] = xf86CrtcCreate(pScrn, &radeon_crtc_funcs); 863209ff23fSmrg if (!pRADEONEnt->pCrtc[0]) 864209ff23fSmrg return FALSE; 865209ff23fSmrg 866209ff23fSmrg pRADEONEnt->Controller[0] = xnfcalloc(sizeof(RADEONCrtcPrivateRec), 1); 867209ff23fSmrg if (!pRADEONEnt->Controller[0]) 868209ff23fSmrg return FALSE; 869209ff23fSmrg 870209ff23fSmrg pRADEONEnt->pCrtc[0]->driver_private = pRADEONEnt->Controller[0]; 871209ff23fSmrg pRADEONEnt->Controller[0]->crtc_id = 0; 872209ff23fSmrg pRADEONEnt->Controller[0]->crtc_offset = 0; 873b7e1c893Smrg pRADEONEnt->Controller[0]->initialized = FALSE; 874209ff23fSmrg if (info->allowColorTiling) 875209ff23fSmrg pRADEONEnt->Controller[0]->can_tile = 1; 876209ff23fSmrg else 877209ff23fSmrg pRADEONEnt->Controller[0]->can_tile = 0; 878ad43ddacSmrg pRADEONEnt->Controller[0]->pll_id = -1; 879209ff23fSmrg } 880209ff23fSmrg 881209ff23fSmrg if (mask & 2) { 882209ff23fSmrg if (!pRADEONEnt->HasCRTC2) 883209ff23fSmrg return TRUE; 884209ff23fSmrg 885209ff23fSmrg pRADEONEnt->pCrtc[1] = xf86CrtcCreate(pScrn, &radeon_crtc_funcs); 886209ff23fSmrg if (!pRADEONEnt->pCrtc[1]) 887209ff23fSmrg return FALSE; 888209ff23fSmrg 889209ff23fSmrg pRADEONEnt->Controller[1] = xnfcalloc(sizeof(RADEONCrtcPrivateRec), 1); 890209ff23fSmrg if (!pRADEONEnt->Controller[1]) 891209ff23fSmrg { 892209ff23fSmrg xfree(pRADEONEnt->Controller[0]); 893209ff23fSmrg return FALSE; 894209ff23fSmrg } 895209ff23fSmrg 896209ff23fSmrg pRADEONEnt->pCrtc[1]->driver_private = pRADEONEnt->Controller[1]; 897209ff23fSmrg pRADEONEnt->Controller[1]->crtc_id = 1; 898ad43ddacSmrg if (IS_DCE4_VARIANT) 899ad43ddacSmrg pRADEONEnt->Controller[1]->crtc_offset = EVERGREEN_CRTC1_REGISTER_OFFSET; 900ad43ddacSmrg else 901ad43ddacSmrg pRADEONEnt->Controller[1]->crtc_offset = AVIVO_D2CRTC_H_TOTAL - AVIVO_D1CRTC_H_TOTAL; 902b7e1c893Smrg pRADEONEnt->Controller[1]->initialized = FALSE; 903209ff23fSmrg if (info->allowColorTiling) 904209ff23fSmrg pRADEONEnt->Controller[1]->can_tile = 1; 905209ff23fSmrg else 906209ff23fSmrg pRADEONEnt->Controller[1]->can_tile = 0; 907ad43ddacSmrg pRADEONEnt->Controller[1]->pll_id = -1; 908ad43ddacSmrg } 909ad43ddacSmrg 910ad43ddacSmrg /* 6 crtcs on DCE4 chips */ 911ad43ddacSmrg if (IS_DCE4_VARIANT && ((mask & 3) == 3)) { 912ad43ddacSmrg for (i = 2; i < RADEON_MAX_CRTC; i++) { 913ad43ddacSmrg pRADEONEnt->pCrtc[i] = xf86CrtcCreate(pScrn, &radeon_crtc_funcs); 914ad43ddacSmrg if (!pRADEONEnt->pCrtc[i]) 915ad43ddacSmrg return FALSE; 916ad43ddacSmrg 917ad43ddacSmrg pRADEONEnt->Controller[i] = xnfcalloc(sizeof(RADEONCrtcPrivateRec), 1); 918ad43ddacSmrg if (!pRADEONEnt->Controller[i]) 919ad43ddacSmrg { 920ad43ddacSmrg xfree(pRADEONEnt->Controller[i]); 921ad43ddacSmrg return FALSE; 922ad43ddacSmrg } 923ad43ddacSmrg 924ad43ddacSmrg pRADEONEnt->pCrtc[i]->driver_private = pRADEONEnt->Controller[i]; 925ad43ddacSmrg pRADEONEnt->Controller[i]->crtc_id = i; 926ad43ddacSmrg switch (i) { 927ad43ddacSmrg case 0: 928ad43ddacSmrg pRADEONEnt->Controller[i]->crtc_offset = EVERGREEN_CRTC0_REGISTER_OFFSET; 929ad43ddacSmrg break; 930ad43ddacSmrg case 1: 931ad43ddacSmrg pRADEONEnt->Controller[i]->crtc_offset = EVERGREEN_CRTC1_REGISTER_OFFSET; 932ad43ddacSmrg break; 933ad43ddacSmrg case 2: 934ad43ddacSmrg pRADEONEnt->Controller[i]->crtc_offset = EVERGREEN_CRTC2_REGISTER_OFFSET; 935ad43ddacSmrg break; 936ad43ddacSmrg case 3: 937ad43ddacSmrg pRADEONEnt->Controller[i]->crtc_offset = EVERGREEN_CRTC3_REGISTER_OFFSET; 938ad43ddacSmrg break; 939ad43ddacSmrg case 4: 940ad43ddacSmrg pRADEONEnt->Controller[i]->crtc_offset = EVERGREEN_CRTC4_REGISTER_OFFSET; 941ad43ddacSmrg break; 942ad43ddacSmrg case 5: 943ad43ddacSmrg pRADEONEnt->Controller[i]->crtc_offset = EVERGREEN_CRTC5_REGISTER_OFFSET; 944ad43ddacSmrg break; 945ad43ddacSmrg } 946ad43ddacSmrg pRADEONEnt->Controller[i]->initialized = FALSE; 947ad43ddacSmrg if (info->allowColorTiling) 948ad43ddacSmrg pRADEONEnt->Controller[i]->can_tile = 1; 949ad43ddacSmrg else 950ad43ddacSmrg pRADEONEnt->Controller[i]->can_tile = 0; 951ad43ddacSmrg pRADEONEnt->Controller[i]->pll_id = -1; 952ad43ddacSmrg } 953209ff23fSmrg } 954209ff23fSmrg 955209ff23fSmrg return TRUE; 956209ff23fSmrg} 957209ff23fSmrg 958209ff23fSmrg/** 959209ff23fSmrg * In the current world order, there are lists of modes per output, which may 960209ff23fSmrg * or may not include the mode that was asked to be set by XFree86's mode 961209ff23fSmrg * selection. Find the closest one, in the following preference order: 962209ff23fSmrg * 963209ff23fSmrg * - Equality 964209ff23fSmrg * - Closer in size to the requested mode, but no larger 965209ff23fSmrg * - Closer in refresh rate to the requested mode. 966209ff23fSmrg */ 967209ff23fSmrgDisplayModePtr 968209ff23fSmrgRADEONCrtcFindClosestMode(xf86CrtcPtr crtc, DisplayModePtr pMode) 969209ff23fSmrg{ 970209ff23fSmrg ScrnInfoPtr pScrn = crtc->scrn; 971209ff23fSmrg xf86CrtcConfigPtr xf86_config = XF86_CRTC_CONFIG_PTR(pScrn); 972209ff23fSmrg DisplayModePtr pBest = NULL, pScan = NULL; 973209ff23fSmrg int i; 974209ff23fSmrg 975209ff23fSmrg /* Assume that there's only one output connected to the given CRTC. */ 976209ff23fSmrg for (i = 0; i < xf86_config->num_output; i++) 977209ff23fSmrg { 978209ff23fSmrg xf86OutputPtr output = xf86_config->output[i]; 979209ff23fSmrg if (output->crtc == crtc && output->probed_modes != NULL) 980209ff23fSmrg { 981209ff23fSmrg pScan = output->probed_modes; 982209ff23fSmrg break; 983209ff23fSmrg } 984209ff23fSmrg } 985209ff23fSmrg 986209ff23fSmrg /* If the pipe doesn't have any detected modes, just let the system try to 987209ff23fSmrg * spam the desired mode in. 988209ff23fSmrg */ 989209ff23fSmrg if (pScan == NULL) { 990209ff23fSmrg RADEONCrtcPrivatePtr radeon_crtc = crtc->driver_private; 991209ff23fSmrg xf86DrvMsg(pScrn->scrnIndex, X_WARNING, 992209ff23fSmrg "No crtc mode list for crtc %d," 993209ff23fSmrg "continuing with desired mode\n", radeon_crtc->crtc_id); 994209ff23fSmrg return pMode; 995209ff23fSmrg } 996209ff23fSmrg 997209ff23fSmrg for (; pScan != NULL; pScan = pScan->next) { 998209ff23fSmrg assert(pScan->VRefresh != 0.0); 999209ff23fSmrg 1000209ff23fSmrg /* If there's an exact match, we're done. */ 1001209ff23fSmrg if (xf86ModesEqual(pScan, pMode)) { 1002209ff23fSmrg pBest = pMode; 1003209ff23fSmrg break; 1004209ff23fSmrg } 1005209ff23fSmrg 1006209ff23fSmrg /* Reject if it's larger than the desired mode. */ 1007209ff23fSmrg if (pScan->HDisplay > pMode->HDisplay || 1008209ff23fSmrg pScan->VDisplay > pMode->VDisplay) 1009209ff23fSmrg { 1010209ff23fSmrg continue; 1011209ff23fSmrg } 1012209ff23fSmrg 1013209ff23fSmrg if (pBest == NULL) { 1014209ff23fSmrg pBest = pScan; 1015209ff23fSmrg continue; 1016209ff23fSmrg } 1017209ff23fSmrg 1018209ff23fSmrg /* Find if it's closer to the right size than the current best 1019209ff23fSmrg * option. 1020209ff23fSmrg */ 1021209ff23fSmrg if ((pScan->HDisplay > pBest->HDisplay && 1022209ff23fSmrg pScan->VDisplay >= pBest->VDisplay) || 1023209ff23fSmrg (pScan->HDisplay >= pBest->HDisplay && 1024209ff23fSmrg pScan->VDisplay > pBest->VDisplay)) 1025209ff23fSmrg { 1026209ff23fSmrg pBest = pScan; 1027209ff23fSmrg continue; 1028209ff23fSmrg } 1029209ff23fSmrg 1030209ff23fSmrg /* Find if it's still closer to the right refresh than the current 1031209ff23fSmrg * best resolution. 1032209ff23fSmrg */ 1033209ff23fSmrg if (pScan->HDisplay == pBest->HDisplay && 1034209ff23fSmrg pScan->VDisplay == pBest->VDisplay && 1035209ff23fSmrg (fabs(pScan->VRefresh - pMode->VRefresh) < 1036209ff23fSmrg fabs(pBest->VRefresh - pMode->VRefresh))) { 1037209ff23fSmrg pBest = pScan; 1038209ff23fSmrg } 1039209ff23fSmrg } 1040209ff23fSmrg 1041209ff23fSmrg if (pBest == NULL) { 1042209ff23fSmrg xf86DrvMsg(pScrn->scrnIndex, X_WARNING, 1043209ff23fSmrg "No suitable mode found to program for the pipe.\n" 1044209ff23fSmrg " continuing with desired mode %dx%d@%.1f\n", 1045209ff23fSmrg pMode->HDisplay, pMode->VDisplay, pMode->VRefresh); 1046209ff23fSmrg } else if (!xf86ModesEqual(pBest, pMode)) { 1047209ff23fSmrg RADEONCrtcPrivatePtr radeon_crtc = crtc->driver_private; 1048209ff23fSmrg int crtc = radeon_crtc->crtc_id; 1049209ff23fSmrg xf86DrvMsg(pScrn->scrnIndex, X_WARNING, 1050209ff23fSmrg "Choosing pipe %d's mode %dx%d@%.1f instead of xf86 " 1051209ff23fSmrg "mode %dx%d@%.1f\n", crtc, 1052209ff23fSmrg pBest->HDisplay, pBest->VDisplay, pBest->VRefresh, 1053209ff23fSmrg pMode->HDisplay, pMode->VDisplay, pMode->VRefresh); 1054209ff23fSmrg pMode = pBest; 1055209ff23fSmrg } 1056209ff23fSmrg return pMode; 1057209ff23fSmrg} 1058209ff23fSmrg 1059209ff23fSmrgvoid 1060209ff23fSmrgRADEONBlank(ScrnInfoPtr pScrn) 1061209ff23fSmrg{ 1062209ff23fSmrg xf86CrtcConfigPtr xf86_config = XF86_CRTC_CONFIG_PTR(pScrn); 1063209ff23fSmrg xf86OutputPtr output; 1064209ff23fSmrg xf86CrtcPtr crtc; 1065209ff23fSmrg int o, c; 1066209ff23fSmrg 1067209ff23fSmrg for (c = 0; c < xf86_config->num_crtc; c++) { 1068209ff23fSmrg crtc = xf86_config->crtc[c]; 1069209ff23fSmrg for (o = 0; o < xf86_config->num_output; o++) { 1070209ff23fSmrg output = xf86_config->output[o]; 1071209ff23fSmrg if (output->crtc != crtc) 1072209ff23fSmrg continue; 1073209ff23fSmrg 1074209ff23fSmrg output->funcs->dpms(output, DPMSModeOff); 1075209ff23fSmrg } 1076209ff23fSmrg crtc->funcs->dpms(crtc, DPMSModeOff); 1077209ff23fSmrg } 1078209ff23fSmrg} 1079209ff23fSmrg 1080209ff23fSmrgvoid 1081209ff23fSmrgRADEONUnblank(ScrnInfoPtr pScrn) 1082209ff23fSmrg{ 1083209ff23fSmrg xf86CrtcConfigPtr xf86_config = XF86_CRTC_CONFIG_PTR(pScrn); 1084209ff23fSmrg xf86OutputPtr output; 1085209ff23fSmrg xf86CrtcPtr crtc; 1086209ff23fSmrg int o, c; 1087209ff23fSmrg 1088209ff23fSmrg for (c = 0; c < xf86_config->num_crtc; c++) { 1089209ff23fSmrg crtc = xf86_config->crtc[c]; 1090209ff23fSmrg if(!crtc->enabled) 1091209ff23fSmrg continue; 1092209ff23fSmrg crtc->funcs->dpms(crtc, DPMSModeOn); 1093209ff23fSmrg for (o = 0; o < xf86_config->num_output; o++) { 1094209ff23fSmrg output = xf86_config->output[o]; 1095209ff23fSmrg if (output->crtc != crtc) 1096209ff23fSmrg continue; 1097209ff23fSmrg 1098209ff23fSmrg output->funcs->dpms(output, DPMSModeOn); 1099209ff23fSmrg } 1100209ff23fSmrg } 1101209ff23fSmrg} 1102209ff23fSmrg 1103209ff23fSmrgBool 1104209ff23fSmrgRADEONSetTiling(ScrnInfoPtr pScrn) 1105209ff23fSmrg{ 1106209ff23fSmrg xf86CrtcConfigPtr xf86_config = XF86_CRTC_CONFIG_PTR(pScrn); 1107209ff23fSmrg RADEONInfoPtr info = RADEONPTR(pScrn); 1108209ff23fSmrg RADEONCrtcPrivatePtr radeon_crtc; 1109209ff23fSmrg xf86CrtcPtr crtc; 1110209ff23fSmrg int c; 1111209ff23fSmrg int can_tile = 1; 1112209ff23fSmrg Bool changed = FALSE; 1113209ff23fSmrg 1114209ff23fSmrg for (c = 0; c < xf86_config->num_crtc; c++) { 1115209ff23fSmrg crtc = xf86_config->crtc[c]; 1116209ff23fSmrg radeon_crtc = crtc->driver_private; 1117209ff23fSmrg 1118209ff23fSmrg if (crtc->enabled) { 1119209ff23fSmrg if (!radeon_crtc->can_tile) 1120209ff23fSmrg can_tile = 0; 1121209ff23fSmrg } 1122209ff23fSmrg } 1123209ff23fSmrg 1124209ff23fSmrg if (info->tilingEnabled != can_tile) 1125209ff23fSmrg changed = TRUE; 1126209ff23fSmrg 1127209ff23fSmrg#ifdef XF86DRI 1128209ff23fSmrg if (info->directRenderingEnabled && (info->tilingEnabled != can_tile)) { 1129b7e1c893Smrg drm_radeon_sarea_t *pSAREAPriv; 1130209ff23fSmrg if (RADEONDRISetParam(pScrn, RADEON_SETPARAM_SWITCH_TILING, (can_tile ? 1 : 0)) < 0) 1131209ff23fSmrg xf86DrvMsg(pScrn->scrnIndex, X_ERROR, 1132209ff23fSmrg "[drm] failed changing tiling status\n"); 1133209ff23fSmrg /* if this is called during ScreenInit() we don't have pScrn->pScreen yet */ 1134209ff23fSmrg pSAREAPriv = DRIGetSAREAPrivate(screenInfo.screens[pScrn->scrnIndex]); 1135209ff23fSmrg info->tilingEnabled = pSAREAPriv->tiling_enabled ? TRUE : FALSE; 1136209ff23fSmrg } 1137209ff23fSmrg#endif 1138209ff23fSmrg 1139209ff23fSmrg return changed; 1140209ff23fSmrg} 1141