1 /* 2 * Copyright 2002-2004, Thomas Kurschel. All Rights Reserved. 3 * Distributed under the terms of the MIT License. 4 */ 5 6 /*! 7 Radeon monitor detection 8 */ 9 10 #include <stdlib.h> 11 #include <string.h> 12 13 #include "radeon_accelerant.h" 14 #include "mmio.h" 15 #include "crtc_regs.h" 16 #include "dac_regs.h" 17 #include "pll_regs.h" 18 #include "tv_out_regs.h" 19 #include "config_regs.h" 20 #include "ddc_regs.h" 21 #include "gpiopad_regs.h" 22 #include "fp_regs.h" 23 #include "pll_access.h" 24 #include "theatre_regs.h" 25 #include "set_mode.h" 26 #include "ddc.h" 27 28 29 typedef struct { 30 accelerator_info *ai; 31 uint32 port; 32 } ddc_port_info; 33 34 35 //! Get I2C signals 36 static status_t 37 get_signals(void *cookie, int *clk, int *data) 38 { 39 ddc_port_info *info = (ddc_port_info *)cookie; 40 vuint8 *regs = info->ai->regs; 41 uint32 value; 42 43 value = INREG(regs, info->port); 44 *clk = (value >> RADEON_GPIO_Y_SHIFT_1) & 1; 45 *data = (value >> RADEON_GPIO_Y_SHIFT_0) & 1; 46 47 return B_OK; 48 } 49 50 51 //! Set I2C signals 52 static status_t 53 set_signals(void *cookie, int clk, int data) 54 { 55 ddc_port_info *info = (ddc_port_info *)cookie; 56 vuint8 *regs = info->ai->regs; 57 uint32 value; 58 59 value = INREG(regs, info->port); 60 value &= ~(RADEON_GPIO_A_1 | RADEON_GPIO_A_0); 61 value &= ~(RADEON_GPIO_EN_0 | RADEON_GPIO_EN_1); 62 value |= ((1-clk) << RADEON_GPIO_EN_SHIFT_1) 63 | ((1-data) << RADEON_GPIO_EN_SHIFT_0); 64 65 OUTREG(regs, info->port, value); 66 return B_OK; 67 } 68 69 70 /*! Read EDID information from monitor 71 ddc_port - register to use for DDC2 communication 72 */ 73 bool 74 Radeon_ReadEDID(accelerator_info *ai, uint32 ddcPort, edid1_info *edid) 75 { 76 i2c_bus bus; 77 ddc_port_info info; 78 void *vdif; 79 size_t vdifLength; 80 81 info.ai = ai; 82 info.port = ddcPort; 83 84 ddc2_init_timing(&bus); 85 bus.cookie = &info; 86 bus.set_signals = &set_signals; 87 bus.get_signals = &get_signals; 88 89 if (ddc2_read_edid1(&bus, edid, &vdif, &vdifLength) != B_OK) 90 return false; 91 92 SHOW_FLOW(2, "Found DDC-capable monitor @0x%04x", ddcPort); 93 94 free(vdif); 95 return true; 96 } 97 98 99 // search for display connect to CRT DAC 100 // colour - true, if only a colour monitor is to be accepted 101 static bool 102 Radeon_DetectCRTInt(accelerator_info *ai, bool colour) 103 { 104 vuint8 *regs = ai->regs; 105 uint32 old_crtc_ext_cntl, old_dac_ext_cntl, old_dac_cntl, value; 106 bool found; 107 108 // makes sure there is a signal 109 old_crtc_ext_cntl = INREG(regs, RADEON_CRTC_EXT_CNTL); 110 111 value = old_crtc_ext_cntl | RADEON_CRTC_CRT_ON; 112 OUTREG(regs, RADEON_CRTC_EXT_CNTL, value); 113 114 // force DAC to output constant voltage 115 // for colour monitors, RGB is tested, for B/W only G 116 old_dac_ext_cntl = INREG(regs, RADEON_DAC_EXT_CNTL); 117 118 value = RADEON_DAC_FORCE_BLANK_OFF_EN | RADEON_DAC_FORCE_DATA_EN 119 | (colour ? RADEON_DAC_FORCE_DATA_SEL_RGB : RADEON_DAC_FORCE_DATA_SEL_G) 120 | (0x1b6 << RADEON_DAC_FORCE_DATA_SHIFT); 121 OUTREG(regs, RADEON_DAC_EXT_CNTL, value); 122 123 // enable DAC and tell it to use VGA signals 124 old_dac_cntl = INREG(regs, RADEON_DAC_CNTL); 125 126 value = old_dac_cntl & ~(RADEON_DAC_RANGE_CNTL_MASK | RADEON_DAC_PDWN); 127 value |= RADEON_DAC_RANGE_CNTL_PS2 | RADEON_DAC_CMP_EN; 128 OUTREG(regs, RADEON_DAC_CNTL, value); 129 130 // specs says that we should wait 1µs before checking but sample 131 // code uses 2 ms; we use long delay to be on safe side 132 // (though we don't want to make it too long as the monitor 133 // gets no sync signal now) 134 snooze(2000); 135 136 // let's see whether there is some 137 found = (INREG(regs, RADEON_DAC_CNTL) & RADEON_DAC_CMP_OUTPUT) != 0; 138 if (found) { 139 SHOW_INFO(2, "Found %s CRT connected to CRT-DAC", 140 colour ? "colour" : "b/w"); 141 } 142 143 OUTREG(regs, RADEON_DAC_CNTL, old_dac_cntl); 144 OUTREG(regs, RADEON_DAC_EXT_CNTL, old_dac_ext_cntl); 145 OUTREG(regs, RADEON_CRTC_EXT_CNTL, old_crtc_ext_cntl); 146 147 return found; 148 } 149 150 151 //! Check whethere there is a CRT connected to CRT DAC 152 static bool 153 Radeon_DetectCRT(accelerator_info *ai) 154 { 155 vuint32 old_vclk_ecp_cntl, value; 156 bool found; 157 158 // enforce clock so the DAC gets activated 159 old_vclk_ecp_cntl = Radeon_INPLL(ai->regs, ai->si->asic, 160 RADEON_VCLK_ECP_CNTL); 161 162 value = old_vclk_ecp_cntl 163 & ~(RADEON_PIXCLK_ALWAYS_ONb | RADEON_PIXCLK_DAC_ALWAYS_ONb); 164 Radeon_OUTPLL(ai->regs, ai->si->asic, RADEON_VCLK_ECP_CNTL, value); 165 166 // search first for colour, then for B/W monitor 167 found = Radeon_DetectCRTInt(ai, true) || Radeon_DetectCRTInt(ai, false); 168 169 Radeon_OUTPLL(ai->regs, ai->si->asic, RADEON_VCLK_ECP_CNTL, 170 old_vclk_ecp_cntl); 171 172 return found; 173 } 174 175 176 //! CRT on TV-DAC detection for rv200 and below checked for rv200 177 static bool 178 Radeon_DetectTVCRT_RV200(accelerator_info *ai) 179 { 180 vuint8 *regs = ai->regs; 181 uint32 old_crtc2_gen_cntl, old_tv_dac_cntl, old_dac_cntl2, value; 182 bool found; 183 184 // enable CRTC2, setting 8 bpp (we just pick any valid value) 185 old_crtc2_gen_cntl = INREG(regs, RADEON_CRTC2_GEN_CNTL); 186 187 value = old_crtc2_gen_cntl & ~RADEON_CRTC2_PIX_WIDTH_MASK; 188 value |= RADEON_CRTC2_CRT2_ON | (2 << RADEON_CRTC2_PIX_WIDTH_SHIFT); 189 OUTREG(regs, RADEON_CRTC2_GEN_CNTL, value); 190 191 // enable TV-DAC, choosing VGA signal level 192 old_tv_dac_cntl = INREG(regs, RADEON_TV_DAC_CNTL); 193 194 value = RADEON_TV_DAC_CNTL_NBLANK | RADEON_TV_DAC_CNTL_NHOLD 195 | RADEON_TV_DAC_CNTL_DETECT | RADEON_TV_DAC_CNTL_STD_PS2; 196 OUTREG(regs, RADEON_TV_DAC_CNTL, value); 197 198 // enforce constant DAC output voltage on RGB 199 value = RADEON_DAC2_FORCE_BLANK_OFF_EN | RADEON_DAC2_FORCE_DATA_EN 200 | RADEON_DAC_FORCE_DATA_SEL_RGB 201 | (0x180 << RADEON_DAC_FORCE_DATA_SHIFT); 202 OUTREG(regs, RADEON_DAC_EXT_CNTL, value); 203 204 old_dac_cntl2 = INREG(regs, RADEON_DAC_CNTL2); 205 206 // set DAC in CRT mode and enable detection 207 // TODO: make sure we really use CRTC2 - this is ASIC dependant 208 value = old_dac_cntl2 | RADEON_DAC2_CLK_SEL_CRT | RADEON_DAC2_CMP_EN; 209 OUTREG(regs, RADEON_DAC_CNTL2, value); 210 211 snooze(10000); 212 213 // let's see what we've got! 214 found = (INREG(regs, RADEON_DAC_CNTL2) & RADEON_DAC2_CMP_OUTPUT) != 0; 215 if (found) 216 SHOW_INFO0(2, "Found CRT connected to TV-DAC, i.e. DVI port"); 217 218 OUTREG(regs, RADEON_DAC_CNTL2, old_dac_cntl2); 219 OUTREG(regs, RADEON_DAC_EXT_CNTL, 0); 220 OUTREG(regs, RADEON_TV_DAC_CNTL, old_tv_dac_cntl); 221 OUTREG(regs, RADEON_CRTC2_GEN_CNTL, old_crtc2_gen_cntl); 222 223 return found; 224 } 225 226 227 //! CRT on TV-DAC detection for r300 checked for r300 228 static bool 229 Radeon_DetectTVCRT_R300(accelerator_info *ai) 230 { 231 vuint8 *regs = ai->regs; 232 uint32 old_crtc2_gen_cntl, old_tv_dac_cntl, old_dac_cntl2, value; 233 uint32 old_radeon_gpiopad_a; 234 bool found; 235 236 old_radeon_gpiopad_a = INREG(regs, RADEON_GPIOPAD_A); 237 238 // whatever these flags mean - let's pray they won't get changed 239 OUTREGP(regs, RADEON_GPIOPAD_EN, 1, ~1); 240 OUTREGP(regs, RADEON_GPIOPAD_MASK, 1, ~1); 241 OUTREGP(regs, RADEON_GPIOPAD_A, 1, ~1); 242 243 old_crtc2_gen_cntl = INREG(regs, RADEON_CRTC2_GEN_CNTL); 244 245 // enable DAC, choose valid pixel format and enable DPMS 246 // as usual, the code doesn't take into account whether the TV-DAC 247 // does really use CRTC2 248 value = old_crtc2_gen_cntl; 249 value &= ~RADEON_CRTC2_PIX_WIDTH_MASK; 250 value |= (2 << RADEON_CRTC2_PIX_WIDTH_SHIFT) | RADEON_CRTC2_CRT2_ON 251 | RADEON_CRTC2_VSYNC_TRISTAT; 252 OUTREG(regs, RADEON_CRTC2_GEN_CNTL, value); 253 254 old_tv_dac_cntl = INREG(regs, RADEON_TV_DAC_CNTL); 255 256 // enable TV-DAC 257 OUTREG(regs, RADEON_TV_DAC_CNTL, RADEON_TV_DAC_CNTL_NBLANK 258 | RADEON_TV_DAC_CNTL_NHOLD | RADEON_TV_DAC_CNTL_DETECT 259 | RADEON_TV_DAC_CNTL_STD_PS2); 260 261 // force constant voltage output of DAC for impedance test 262 OUTREG(regs, RADEON_DAC_EXT_CNTL, RADEON_DAC2_FORCE_BLANK_OFF_EN 263 | RADEON_DAC2_FORCE_DATA_EN | RADEON_DAC_FORCE_DATA_SEL_RGB 264 | (0x1b6 << RADEON_DAC_FORCE_DATA_SHIFT)); 265 266 old_dac_cntl2 = INREG(regs, RADEON_DAC_CNTL2); 267 268 // enable CRT mode of TV-DAC and enable comparator 269 OUTREG(regs, RADEON_DAC_CNTL2, old_dac_cntl2 | RADEON_DAC2_CLK_SEL_CRT 270 | RADEON_DAC2_CMP_EN); 271 272 snooze(10000); 273 274 // check connection of blue data signal to see whether there is a CRT 275 found = (INREG(regs, RADEON_DAC_CNTL2) & RADEON_DAC2_CMP_OUT_B) != 0; 276 277 // clean up the mess 278 OUTREG(regs, RADEON_DAC_CNTL2, old_dac_cntl2); 279 OUTREG(regs, RADEON_DAC_EXT_CNTL, 0); 280 OUTREG(regs, RADEON_TV_DAC_CNTL, old_tv_dac_cntl); 281 OUTREG(regs, RADEON_CRTC2_GEN_CNTL, old_crtc2_gen_cntl); 282 283 OUTREGP(regs, RADEON_GPIOPAD_A, old_radeon_gpiopad_a, ~1); 284 285 return found; 286 } 287 288 289 //! Check whether there is a CRT connected to TV-DAC 290 static bool 291 Radeon_DetectTVCRT(accelerator_info *ai) 292 { 293 if (ai->si->is_mobility) 294 return dd_none; 295 296 switch (ai->si->asic) { 297 case rt_r100: 298 // original Radeons have pure DVI only and mobility chips 299 // have no DVI connector 300 // TBD: can they have a docking station for CRT on TV-DAC? 301 return dd_none; 302 303 case rt_rv100: 304 case rt_rv200: 305 case rt_rv250: 306 case rt_rv280: 307 // IGP is guessed 308 case rt_rs100: 309 case rt_rs200: 310 case rt_rs300: 311 return Radeon_DetectTVCRT_RV200(ai); 312 313 case rt_r300: 314 case rt_r350: 315 case rt_rv350: 316 case rt_rv380: 317 case rt_r420: 318 return Radeon_DetectTVCRT_R300(ai); 319 320 case rt_r200: 321 // r200 has no built-in TV-out and thus no TV-DAC to use for 322 // second CRT 323 return dd_none; 324 } 325 326 return dd_none; 327 } 328 329 330 //! TV detection for rv200 and below should work for M6 and RV200 331 static display_device_e 332 Radeon_DetectTV_RV200(accelerator_info *ai, bool tv_crt_found) 333 { 334 vuint8 *regs = ai->regs; 335 uint32 value, old_dac_cntl2, old_crtc_ext_cntl, old_crtc2_gen_cntl; 336 uint32 old_tv_master_cntl, old_tv_dac_cntl, old_pre_dac_mux_cntl; 337 uint32 config_cntl; 338 display_device_e displays = dd_none; 339 340 // give up if there is a CRT connected to TV-DAC 341 if (tv_crt_found) 342 return dd_none; 343 344 // enable TV mode 345 old_dac_cntl2 = INREG(regs, RADEON_DAC_CNTL2); 346 value = old_dac_cntl2 & ~RADEON_DAC2_CLK_SEL_CRT; 347 OUTREG(regs, RADEON_DAC_CNTL2, value); 348 349 old_crtc_ext_cntl = INREG(regs, RADEON_CRTC_EXT_CNTL); 350 old_crtc2_gen_cntl = INREG(regs, RADEON_CRTC2_GEN_CNTL); 351 old_tv_master_cntl = INREG(regs, RADEON_TV_MASTER_CNTL); 352 353 // enable TV output 354 value = old_tv_master_cntl | RADEON_TV_MASTER_CNTL_TV_ON; 355 value &= ~( 356 RADEON_TV_MASTER_CNTL_TV_ASYNC_RST | 357 RADEON_TV_MASTER_CNTL_RESTART_PHASE_FIX | 358 RADEON_TV_MASTER_CNTL_CRT_FIFO_CE_EN | 359 RADEON_TV_MASTER_CNTL_TV_FIFO_CE_EN | 360 RADEON_TV_MASTER_CNTL_RE_SYNC_NOW_SEL_MASK); 361 value |= 362 RADEON_TV_MASTER_CNTL_TV_FIFO_ASYNC_RST | 363 RADEON_TV_MASTER_CNTL_CRT_ASYNC_RST; 364 OUTREG(regs, RADEON_TV_MASTER_CNTL, value); 365 366 old_tv_dac_cntl = INREG(regs, RADEON_TV_DAC_CNTL); 367 368 config_cntl = INREG(regs, RADEON_CONFIG_CNTL); 369 370 // unlock TV DAC 371 value = 372 RADEON_TV_DAC_CNTL_NBLANK | RADEON_TV_DAC_CNTL_NHOLD | 373 RADEON_TV_DAC_CNTL_DETECT | RADEON_TV_DAC_CNTL_STD_NTSC | 374 (8 << RADEON_TV_DAC_CNTL_BGADJ_SHIFT) | 375 ((((config_cntl & RADEON_CFG_ATI_REV_ID_MASK) == 0) ? 8 : 4) << RADEON_TV_DAC_CNTL_DACADJ_SHIFT); 376 OUTREG(regs, RADEON_TV_DAC_CNTL, value); 377 378 old_pre_dac_mux_cntl = INREG(regs, RADEON_TV_PRE_DAC_MUX_CNTL); 379 380 // force constant DAC output voltage 381 value = 382 RADEON_TV_PRE_DAC_MUX_CNTL_C_GRN_EN | RADEON_TV_PRE_DAC_MUX_CNTL_CMP_BLU_EN | 383 (RADEON_TV_MUX_FORCE_DAC_DATA << RADEON_TV_PRE_DAC_MUX_CNTL_RED_MX_SHIFT) | 384 (RADEON_TV_MUX_FORCE_DAC_DATA << RADEON_TV_PRE_DAC_MUX_CNTL_GRN_MX_SHIFT) | 385 (RADEON_TV_MUX_FORCE_DAC_DATA << RADEON_TV_PRE_DAC_MUX_CNTL_BLU_MX_SHIFT) | 386 (0x109 << RADEON_TV_PRE_DAC_MUX_CNTL_FORCE_DAC_DATA_SHIFT); 387 OUTREG(regs, RADEON_TV_PRE_DAC_MUX_CNTL, value); 388 389 // let things settle a bit 390 snooze(3000); 391 392 // now see which wires are connected 393 value = INREG(regs, RADEON_TV_DAC_CNTL); 394 if ((value & RADEON_TV_DAC_CNTL_GDACDET) != 0) { 395 displays |= dd_stv; 396 SHOW_INFO0(2, "S-Video TV-Out is connected"); 397 } 398 399 if ((value & RADEON_TV_DAC_CNTL_BDACDET) != 0) { 400 displays |= dd_ctv; 401 SHOW_INFO0(2, "Composite TV-Out is connected"); 402 } 403 404 OUTREG(regs, RADEON_TV_PRE_DAC_MUX_CNTL, old_pre_dac_mux_cntl); 405 OUTREG(regs, RADEON_TV_DAC_CNTL, old_tv_dac_cntl); 406 OUTREG(regs, RADEON_TV_MASTER_CNTL, old_tv_master_cntl); 407 OUTREG(regs, RADEON_CRTC2_GEN_CNTL, old_crtc2_gen_cntl); 408 OUTREG(regs, RADEON_CRTC_EXT_CNTL, old_crtc_ext_cntl); 409 OUTREG(regs, RADEON_DAC_CNTL2, old_dac_cntl2); 410 411 return displays; 412 } 413 414 415 // TV detection for r300 series 416 // should work for R300 417 static display_device_e Radeon_DetectTV_R300(accelerator_info *ai) 418 { 419 vuint8 *regs = ai->regs; 420 display_device_e displays = dd_none; 421 uint32 tmp, old_dac_cntl2, old_crtc2_gen_cntl, old_dac_ext_cntl, old_tv_dac_cntl; 422 uint32 old_radeon_gpiopad_a; 423 424 old_radeon_gpiopad_a = INREG(regs, RADEON_GPIOPAD_A); 425 426 // whatever these flags mean - let's pray they won't get changed 427 OUTREGP(regs, RADEON_GPIOPAD_EN, 1, ~1); 428 OUTREGP(regs, RADEON_GPIOPAD_MASK, 1, ~1); 429 OUTREGP(regs, RADEON_GPIOPAD_A, 0, ~1); 430 431 old_dac_cntl2 = INREG(regs, RADEON_DAC_CNTL2); 432 433 // set CRT mode (!) of TV-DAC 434 OUTREG(regs, RADEON_DAC_CNTL2, RADEON_DAC2_CLK_SEL_CRT); 435 436 old_crtc2_gen_cntl = INREG(regs, RADEON_CRTC2_GEN_CNTL); 437 438 // enable TV-Out output, but set DPMS mode 439 // (this seems to be not correct if TV-Out is connected to CRTC1, 440 // but it doesn't really hurt having wrong DPMS mode) 441 OUTREG(regs, RADEON_CRTC2_GEN_CNTL, 442 RADEON_CRTC2_CRT2_ON | RADEON_CRTC2_VSYNC_TRISTAT); 443 444 old_dac_ext_cntl = INREG(regs, RADEON_DAC_EXT_CNTL); 445 446 // force constant voltage output of DAC for impedance test 447 OUTREG(regs, RADEON_DAC_EXT_CNTL, 448 RADEON_DAC2_FORCE_BLANK_OFF_EN | RADEON_DAC2_FORCE_DATA_EN | 449 RADEON_DAC_FORCE_DATA_SEL_RGB | 450 (0xec << RADEON_DAC_FORCE_DATA_SHIFT)); 451 452 old_tv_dac_cntl = INREG(regs, RADEON_TV_DAC_CNTL); 453 454 // get TV-DAC running (or something...) 455 OUTREG(regs, RADEON_TV_DAC_CNTL, 456 RADEON_TV_DAC_CNTL_STD_NTSC | 457 (8 << RADEON_TV_DAC_CNTL_BGADJ_SHIFT) | 458 (6 << RADEON_TV_DAC_CNTL_DACADJ_SHIFT)); 459 460 (void)INREG(regs, RADEON_TV_DAC_CNTL); 461 462 snooze(4000); 463 464 OUTREG(regs, RADEON_TV_DAC_CNTL, 465 RADEON_TV_DAC_CNTL_NBLANK | RADEON_TV_DAC_CNTL_NHOLD | 466 RADEON_TV_DAC_CNTL_DETECT | 467 RADEON_TV_DAC_CNTL_STD_NTSC | 468 (8 << RADEON_TV_DAC_CNTL_BGADJ_SHIFT) | 469 (6 << RADEON_TV_DAC_CNTL_DACADJ_SHIFT)); 470 471 (void)INREG(regs, RADEON_TV_DAC_CNTL); 472 473 snooze(6000); 474 475 // now see which wires are connected 476 tmp = INREG(regs, RADEON_TV_DAC_CNTL); 477 if ((tmp & RADEON_TV_DAC_CNTL_GDACDET) != 0) { 478 displays |= dd_stv; 479 SHOW_INFO0(2, "S-Video TV-Out is connected"); 480 } 481 482 if ((tmp & RADEON_TV_DAC_CNTL_BDACDET) != 0) { 483 displays |= dd_ctv; 484 SHOW_INFO0(2, "Composite TV-Out is connected"); 485 } 486 487 // clean up the mess we did 488 OUTREG(regs, RADEON_TV_DAC_CNTL, old_tv_dac_cntl); 489 OUTREG(regs, RADEON_DAC_EXT_CNTL, old_dac_ext_cntl); 490 OUTREG(regs, RADEON_CRTC2_GEN_CNTL, old_crtc2_gen_cntl); 491 OUTREG(regs, RADEON_DAC_CNTL2, old_dac_cntl2); 492 493 OUTREGP(regs, RADEON_GPIOPAD_A, old_radeon_gpiopad_a, ~1); 494 495 return displays; 496 } 497 498 499 // save readout of TV detection comparators 500 static bool readTVDetect(accelerator_info *ai) 501 { 502 uint32 tmp; 503 int i; 504 bigtime_t start_time; 505 bool detect; 506 507 // make output constant 508 Radeon_VIPWrite(ai, ai->si->theatre_channel, THEATRE_VIP_TV_DAC_CNTL, 509 RADEON_TV_DAC_CNTL_STD_NTSC | RADEON_TV_DAC_CNTL_DETECT | RADEON_TV_DAC_CNTL_NBLANK); 510 511 // check detection result 512 Radeon_VIPRead(ai, ai->si->theatre_channel, THEATRE_VIP_TV_DAC_CNTL, &tmp); 513 detect = (tmp & RADEON_TV_DAC_CNTL_CMPOUT) != 0; 514 515 //SHOW_FLOW(2, "detect=%d", detect); 516 517 start_time = system_time(); 518 519 do { 520 // wait for stable detect signal 521 for (i = 0; i < 5; ++i) { 522 bool cur_detect; 523 524 Radeon_VIPWrite(ai, ai->si->theatre_channel, THEATRE_VIP_TV_DAC_CNTL, 525 RADEON_TV_DAC_CNTL_STD_NTSC | RADEON_TV_DAC_CNTL_DETECT | RADEON_TV_DAC_CNTL_NBLANK | 526 RADEON_TV_DAC_CNTL_NHOLD); 527 Radeon_VIPWrite(ai, ai->si->theatre_channel, THEATRE_VIP_TV_DAC_CNTL, 528 RADEON_TV_DAC_CNTL_STD_NTSC | RADEON_TV_DAC_CNTL_DETECT | RADEON_TV_DAC_CNTL_NBLANK); 529 530 Radeon_VIPRead(ai, ai->si->theatre_channel, THEATRE_VIP_TV_DAC_CNTL, &tmp); 531 cur_detect = (tmp & RADEON_TV_DAC_CNTL_CMPOUT) != 0; 532 533 //SHOW_FLOW(2, "cur_detect=%d", cur_detect); 534 535 if (cur_detect != detect) 536 break; 537 538 detect = cur_detect; 539 } 540 541 if (i == 5) { 542 //SHOW_FLOW(2, "return %d", detect); 543 return detect; 544 } 545 546 // don't wait forever - give up after 1 second 547 } while (system_time() - start_time < 1000000); 548 549 SHOW_FLOW0(2, "timeout"); 550 return false; 551 } 552 553 554 //! Detect TV connected to external Theatre-Out 555 static display_device_e 556 Radeon_DetectTV_Theatre(accelerator_info *ai) 557 { 558 uint32 old_tv_dac_cntl, old_pre_dac_mux_cntl, old_modulator_cntl1; 559 uint32 old_master_cntl; 560 uint32 uv_adr, old_last_fifo_entry, old_mid_fifo_entry, last_fifo_addr; 561 display_device_e displays = dd_none; 562 563 if (ai->si->tv_chip != tc_external_rt1) 564 return dd_none; 565 566 // save previous values (TV-Out may be running) 567 Radeon_VIPRead(ai, ai->si->theatre_channel, THEATRE_VIP_TV_DAC_CNTL, 568 &old_tv_dac_cntl); 569 570 // enable DAC and comparators 571 Radeon_VIPWrite(ai, ai->si->theatre_channel, THEATRE_VIP_TV_DAC_CNTL, 572 RADEON_TV_DAC_CNTL_STD_NTSC | RADEON_TV_DAC_CNTL_DETECT 573 | RADEON_TV_DAC_CNTL_NHOLD | RADEON_TV_DAC_CNTL_NBLANK); 574 575 Radeon_VIPRead(ai, ai->si->theatre_channel, THEATRE_VIP_PRE_DAC_MUX_CNTL, 576 &old_pre_dac_mux_cntl); 577 Radeon_VIPRead(ai, ai->si->theatre_channel, THEATRE_VIP_MODULATOR_CNTL1, 578 &old_modulator_cntl1); 579 Radeon_VIPRead(ai, ai->si->theatre_channel, THEATRE_VIP_MASTER_CNTL, 580 &old_master_cntl); 581 582 // save output timing 583 Radeon_VIPRead(ai, ai->si->theatre_channel, THEATRE_VIP_UV_ADR, &uv_adr); 584 585 last_fifo_addr = (uv_adr & RADEON_TV_UV_ADR_MAX_UV_ADR_MASK) * 2 + 1; 586 587 old_last_fifo_entry = Radeon_TheatreReadFIFO(ai, last_fifo_addr); 588 old_mid_fifo_entry = Radeon_TheatreReadFIFO(ai, 0x18f); 589 590 Radeon_TheatreWriteFIFO(ai, last_fifo_addr, 0x20208); 591 Radeon_TheatreWriteFIFO(ai, 0x18f, 0x3ff2608); 592 593 // stop TV-Out to savely program it 594 Radeon_VIPWrite(ai, ai->si->theatre_channel, THEATRE_VIP_MASTER_CNTL, 595 RADEON_TV_MASTER_CNTL_TV_FIFO_ASYNC_RST 596 | RADEON_TV_MASTER_CNTL_TV_ASYNC_RST); 597 598 // set constant base level 599 Radeon_VIPWrite(ai, ai->si->theatre_channel, THEATRE_VIP_MODULATOR_CNTL1, 600 (0x2c << RADEON_TV_MODULATOR_CNTL1_SET_UP_LEVEL_SHIFT) 601 | (0x2c << RADEON_TV_MODULATOR_CNTL1_BLANK_LEVEL_SHIFT)); 602 603 // enable output 604 Radeon_VIPWrite(ai, ai->si->theatre_channel, THEATRE_VIP_MASTER_CNTL, 605 RADEON_TV_MASTER_CNTL_TV_ASYNC_RST); 606 607 Radeon_VIPWrite(ai, ai->si->theatre_channel, THEATRE_VIP_MASTER_CNTL, 0); 608 609 // set constant Composite output 610 Radeon_VIPWrite(ai, ai->si->theatre_channel, THEATRE_VIP_PRE_DAC_MUX_CNTL, 611 RADEON_TV_PRE_DAC_MUX_CNTL_CMP_BLU_EN 612 | RADEON_TV_PRE_DAC_MUX_CNTL_DAC_DITHER_EN 613 | (9 << RADEON_TV_PRE_DAC_MUX_CNTL_BLU_MX_SHIFT) 614 | (0xa8 << RADEON_TV_PRE_DAC_MUX_CNTL_FORCE_DAC_DATA_SHIFT)); 615 616 // check for S-Video connection 617 if (readTVDetect(ai)) { 618 SHOW_FLOW0(2, "Composite-Out of Rage Theatre is connected"); 619 displays |= dd_ctv; 620 } 621 622 // enable output changes 623 Radeon_VIPWrite(ai, ai->si->theatre_channel, THEATRE_VIP_TV_DAC_CNTL, 624 RADEON_TV_DAC_CNTL_STD_NTSC | RADEON_TV_DAC_CNTL_DETECT 625 | RADEON_TV_DAC_CNTL_NBLANK | RADEON_TV_DAC_CNTL_NHOLD); 626 627 // set constant Y-output of S-Video adapter 628 Radeon_VIPWrite(ai, ai->si->theatre_channel, THEATRE_VIP_PRE_DAC_MUX_CNTL, 629 RADEON_TV_PRE_DAC_MUX_CNTL_Y_RED_EN 630 | RADEON_TV_PRE_DAC_MUX_CNTL_DAC_DITHER_EN 631 | (9 << RADEON_TV_PRE_DAC_MUX_CNTL_RED_MX_SHIFT) 632 | (0xa8 << RADEON_TV_PRE_DAC_MUX_CNTL_FORCE_DAC_DATA_SHIFT)); 633 634 // check for composite connection 635 if (readTVDetect(ai)) { 636 SHOW_FLOW0(2, "S-Video-Out of Rage Theatre is connected"); 637 displays |= dd_stv; 638 } 639 640 // restore everything 641 Radeon_TheatreWriteFIFO(ai, last_fifo_addr, old_last_fifo_entry); 642 Radeon_TheatreWriteFIFO(ai, 0x18f, old_mid_fifo_entry); 643 644 Radeon_VIPWrite(ai, ai->si->theatre_channel, THEATRE_VIP_MASTER_CNTL, 645 old_master_cntl); 646 Radeon_VIPWrite(ai, ai->si->theatre_channel, THEATRE_VIP_MODULATOR_CNTL1, 647 old_modulator_cntl1); 648 Radeon_VIPWrite(ai, ai->si->theatre_channel, THEATRE_VIP_PRE_DAC_MUX_CNTL, 649 old_pre_dac_mux_cntl); 650 Radeon_VIPWrite(ai, ai->si->theatre_channel, THEATRE_VIP_TV_DAC_CNTL, 651 old_tv_dac_cntl); 652 653 return displays; 654 } 655 656 657 /*! 658 Check whether there is a TV connected to TV-DAC 659 returns bit set, i.e. there can be S-Video or composite or both 660 */ 661 static display_device_e 662 Radeon_DetectTV(accelerator_info *ai, bool tv_crt_found) 663 { 664 switch (ai->si->asic) { 665 case rt_r100: 666 case rt_r200: 667 return Radeon_DetectTV_Theatre(ai); 668 669 case rt_rv100: 670 case rt_rv200: 671 case rt_rv250: 672 case rt_rv280: 673 // IGP method is guessed 674 case rt_rs100: 675 case rt_rs200: 676 case rt_rs300: 677 return Radeon_DetectTV_RV200(ai, tv_crt_found); 678 679 case rt_r300: 680 case rt_r350: 681 case rt_rv350: 682 case rt_rv380: 683 case rt_r420: 684 return Radeon_DetectTV_R300(ai); 685 } 686 687 return dd_none; 688 } 689 690 691 //! Get native monitor timing, using Detailed Monitor Description 692 static void 693 Radeon_FindFPTiming_DetailedMonitorDesc(const edid1_info *edid, fp_info *fp, 694 uint32 *max_hsize, uint32 *max_vsize) 695 { 696 int i; 697 698 for (i = 0; i < EDID1_NUM_DETAILED_MONITOR_DESC; ++i) { 699 if (edid->detailed_monitor[i].monitor_desc_type == EDID1_IS_DETAILED_TIMING) { 700 const edid1_detailed_timing *timing = &edid->detailed_monitor[i].data.detailed_timing; 701 702 SHOW_FLOW(2, "Found detailed timing for mode %dx%d in DDC data", 703 (int)timing->h_active, (int)timing->v_active); 704 705 if (timing->h_active > *max_hsize && timing->v_active > *max_vsize) { 706 *max_hsize = timing->h_active; 707 *max_vsize = timing->v_active; 708 709 // copy it to timing specification 710 fp->panel_xres = timing->h_active; 711 fp->h_blank = timing->h_blank; 712 fp->h_over_plus = timing->h_sync_off; 713 fp->h_sync_width = timing->h_sync_width; 714 715 fp->panel_yres = timing->v_active; 716 fp->v_blank = timing->v_blank; 717 fp->v_over_plus = timing->v_sync_off; 718 fp->v_sync_width = timing->v_sync_width; 719 720 // BeOS uses kHz, but the timing is in 10 kHz 721 fp->dot_clock = timing->pixel_clock * 10; 722 } 723 } 724 } 725 } 726 727 728 /*! Get native monitor timing, using Standard Timing table; 729 this table doesn't contain the actual timing, so we try to find a 730 appropriate VESA modes for the resolutions given in the table 731 */ 732 static void 733 Radeon_FindFPTiming_StandardTiming(const edid1_info *edid, fp_info *fp, 734 uint32 *max_hsize, uint32 *max_vsize) 735 { 736 int i; 737 738 for (i = 0; i < EDID1_NUM_STD_TIMING; ++i) { 739 const edid1_std_timing *std_timing = &edid->std_timing[i]; 740 int best_fit = -1; 741 int best_refresh_deviation = 10000; 742 int j; 743 744 if (std_timing->h_size <= 256) 745 continue; 746 747 for (j = 0; j < (int)vesa_mode_list_count; ++j) { 748 int refresh_rate, cur_refresh_deviation; 749 750 if (vesa_mode_list[j].h_display != std_timing->h_size 751 || vesa_mode_list[j].v_display != std_timing->v_size) 752 continue; 753 754 // take pixel_clock times 1000 because is is in kHz 755 // further, take it times 1000 again, to get 1/1000 frames 756 // as refresh rate 757 refresh_rate = (int64)vesa_mode_list[j].pixel_clock * 1000*1000 758 / (vesa_mode_list[j].h_total * vesa_mode_list[j].v_total); 759 760 // standard timing is in frames, so multiple by it to get 1/1000 frames 761 // result is scaled by 100 to get difference in percentage; 762 cur_refresh_deviation = (100 * (refresh_rate - std_timing->refresh 763 * 1000)) / refresh_rate; 764 765 if (cur_refresh_deviation < 0) 766 cur_refresh_deviation = -cur_refresh_deviation; 767 768 // less then 1 percent difference is (hopefully) OK, 769 // if there are multiple, we take best one 770 // (if the screen is that picky, it should have defined an enhanced timing) 771 if (cur_refresh_deviation < 1 772 && cur_refresh_deviation < best_refresh_deviation) { 773 best_fit = j; 774 best_refresh_deviation = cur_refresh_deviation; 775 } 776 } 777 778 if (best_fit < 0) { 779 SHOW_FLOW(2, "Unsupported standard mode %dx%d@%dHz (not VESA)", 780 std_timing->h_size, std_timing->v_size, std_timing->refresh); 781 continue; 782 } 783 784 if (std_timing->h_size > *max_hsize && std_timing->h_size > *max_vsize) { 785 const display_timing *timing = &vesa_mode_list[best_fit]; 786 787 SHOW_FLOW(2, "Found DDC data for standard mode %dx%d", 788 (int)timing->h_display, (int)timing->v_display); 789 790 *max_hsize = timing->h_display; 791 *max_vsize = timing->h_display; 792 793 // copy it to timing specification 794 fp->panel_xres = timing->h_display; 795 fp->h_blank = timing->h_total - timing->h_display; 796 fp->h_over_plus = timing->h_sync_start - timing->h_display; 797 fp->h_sync_width = timing->h_sync_end - timing->h_sync_start; 798 799 fp->panel_yres = timing->v_display; 800 fp->v_blank = timing->v_total - timing->v_display; 801 fp->v_over_plus = timing->v_sync_start - timing->v_display; 802 fp->v_sync_width = timing->v_sync_end - timing->v_sync_start; 803 804 fp->dot_clock = timing->pixel_clock; 805 } 806 } 807 } 808 809 810 //! Read edid data of flat panel and setup its timing accordingly 811 static status_t 812 Radeon_StoreFPEDID(accelerator_info *ai, int port, const edid1_info *edid) 813 { 814 fp_info *fp = &ai->si->flatpanels[port]; 815 uint32 max_hsize, max_vsize; 816 817 //SHOW_FLOW0(2, "EDID data read from DVI port via DDC2:"); 818 //edid_dump(edid); 819 820 // find detailed timing with maximum resolution 821 max_hsize = max_vsize = 0; 822 Radeon_FindFPTiming_DetailedMonitorDesc(edid, fp, &max_hsize, &max_vsize); 823 824 if (max_hsize == 0) { 825 SHOW_FLOW0(2, "Timing is not explicitely defined in DDC - checking standard modes"); 826 827 Radeon_FindFPTiming_StandardTiming(edid, fp, &max_hsize, &max_vsize); 828 if (max_hsize == 0) { 829 SHOW_FLOW0(2, "Still found no valid native mode, disabling DVI"); 830 return B_ERROR; 831 } 832 } 833 834 SHOW_INFO(2, "h_disp=%d, h_blank=%d, h_over_plus=%d, h_sync_width=%d", 835 fp->panel_xres, fp->h_blank, fp->h_over_plus, fp->h_sync_width); 836 SHOW_INFO(2, "v_disp=%d, v_blank=%d, v_over_plus=%d, v_sync_width=%d", 837 fp->panel_yres, fp->v_blank, fp->v_over_plus, fp->v_sync_width); 838 SHOW_INFO(2, "pixel_clock=%d kHz", fp->dot_clock); 839 840 return B_OK; 841 } 842 843 844 static void 845 Radeon_ConnectorInfo(accelerator_info *ai, int port, disp_entity* ptr_entity) 846 { 847 const char* mon; 848 const char* ddc = ptr_entity->port_info[port].ddc_type == ddc_none_detected 849 ? "None" : ptr_entity->port_info[port].ddc_type == ddc_monid 850 ? "Mon ID" : ptr_entity->port_info[port].ddc_type == ddc_dvi 851 ? "DVI DDC" : ptr_entity->port_info[port].ddc_type == ddc_vga 852 ? "VGA DDC" : ptr_entity->port_info[port].ddc_type == ddc_crt2 853 ? "CRT2 DDC" : "Error"; 854 const char* tmds = ptr_entity->port_info[port].tmds_type == tmds_unknown 855 ? "None" : ptr_entity->port_info[port].tmds_type == tmds_int 856 ? "Internal" : ptr_entity->port_info[port].tmds_type == tmds_ext 857 ? "External" : "??? "; 858 const char* dac = ptr_entity->port_info[port].dac_type == dac_unknown 859 ? "Unknown" : ptr_entity->port_info[port].dac_type == dac_primary 860 ? "Primary" : ptr_entity->port_info[port].dac_type == dac_tvdac 861 ? "TV / External" : "Error"; 862 const char* con; 863 864 if (ai->si->is_atombios) { 865 con = ptr_entity->port_info[port].connector_type == connector_none_atom 866 ? "None" : ptr_entity->port_info[port].connector_type == connector_vga_atom 867 ? "VGA" : ptr_entity->port_info[port].connector_type == connector_dvi_i_atom 868 ? "DVI-I" : ptr_entity->port_info[port].connector_type == connector_dvi_d_atom 869 ? "DVI-D" : ptr_entity->port_info[port].connector_type == connector_dvi_a_atom 870 ? "DVI-A" : ptr_entity->port_info[port].connector_type == connector_stv_atom 871 ? "S-Video TV" : ptr_entity->port_info[port].connector_type == connector_ctv_atom 872 ? "Composite TV" : ptr_entity->port_info[port].connector_type == connector_lvds_atom 873 ? "LVDS" : ptr_entity->port_info[port].connector_type == connector_digital_atom 874 ? "Digital" : ptr_entity->port_info[port].connector_type == connector_unsupported_atom 875 ? "N/A " : "Err "; 876 } else { 877 con = ptr_entity->port_info[port].connector_type == connector_none 878 ? "None" : ptr_entity->port_info[port].connector_type == connector_crt 879 ? "VGA" : ptr_entity->port_info[port].connector_type == connector_dvi_i 880 ? "DVI-I" : ptr_entity->port_info[port].connector_type == connector_dvi_d 881 ? "DVI-D" : ptr_entity->port_info[port].connector_type == connector_proprietary 882 ? "Proprietary" : ptr_entity->port_info[port].connector_type == connector_stv 883 ? "S-Video TV" : ptr_entity->port_info[port].connector_type == connector_ctv 884 ? "Composite TV" : ptr_entity->port_info[port].connector_type == connector_unsupported 885 ? "N/A" : "Err"; 886 } 887 888 mon = ptr_entity->port_info[port].mon_type == mt_unknown ? "???" 889 : ptr_entity->port_info[port].mon_type == mt_none ? "None" 890 : ptr_entity->port_info[port].mon_type == mt_crt ? "CRT " 891 : ptr_entity->port_info[port].mon_type == mt_lcd ? "LCD " 892 : ptr_entity->port_info[port].mon_type == mt_dfp ? "DVI " 893 : ptr_entity->port_info[port].mon_type == mt_ctv ? "Composite TV" 894 : ptr_entity->port_info[port].mon_type == mt_stv ? "S-Video TV" 895 : "Err ?"; 896 897 SHOW_INFO(2, "Port %d:- \nMonitor: %s\nConn Type: %s\nDDC Port: %s\nTMDS Type: %s\nDAC Type: %s", 898 port, mon, con, ddc, tmds, dac); 899 } 900 901 902 /*! 903 Detect connected displays devices 904 whished_num_heads - how many heads the requested display mode needs 905 */ 906 void 907 Radeon_DetectDisplays(accelerator_info *ai) 908 { 909 shared_info *si = ai->si; 910 911 disp_entity* routes = &si->routing; 912 display_device_e displays = 0; 913 display_device_e controlled_displays = ai->vc->controlled_displays; 914 int i; 915 916 uint32 edid_regs[] = { 917 0, 918 RADEON_GPIO_MONID, 919 RADEON_GPIO_DVI_DDC, 920 RADEON_GPIO_VGA_DDC, 921 RADEON_GPIO_CRT2_DDC 922 }; 923 924 // lock hardware so noone bothers us 925 Radeon_WaitForIdle(ai, true); 926 927 // alwats make TMDS_INT port first 928 if (routes->port_info[1].tmds_type == tmds_int) { 929 radeon_connector swap_entity; 930 swap_entity = routes->port_info[0]; 931 routes->port_info[0] = routes->port_info[1]; 932 routes->port_info[1] = swap_entity; 933 SHOW_FLOW0(2, "Swapping TMDS_INT to first port"); 934 } else if (routes->port_info[0].tmds_type != tmds_int 935 && routes->port_info[1].tmds_type != tmds_int) { 936 // no TMDS_INT port, make primary DAC port first 937 // On my Inspiron 8600 both internal and external ports are 938 // marked DAC_PRIMARY in BIOS. So be extra careful - only 939 // swap when the first port is not DAC_PRIMARY 940 if (routes->port_info[1].dac_type == dac_primary 941 && routes->port_info[0].dac_type != dac_primary) { 942 radeon_connector swap_entity; 943 swap_entity = routes->port_info[0]; 944 routes->port_info[0] = routes->port_info[1]; 945 routes->port_info[1] = swap_entity; 946 SHOW_FLOW0(2, "Swapping Primary Dac to front"); 947 } 948 } 949 950 if (si->asic == rt_rs300) { 951 // RS300 only has single Dac of TV type 952 // For RS300/RS350/RS400 chips, there is no primary DAC. 953 // Force VGA port to use TVDAC 954 if (routes->port_info[0].connector_type == connector_crt) { 955 routes->port_info[0].dac_type = dac_tvdac; 956 routes->port_info[1].dac_type = dac_primary; 957 } else { 958 routes->port_info[1].dac_type = dac_primary; 959 routes->port_info[0].dac_type = dac_tvdac; 960 } 961 } else if (si->num_crtc == 1) { 962 routes->port_info[0].dac_type = dac_primary; 963 } 964 965 // use DDC to detect monitors - if we can read DDC, there must be 966 // a monitor 967 for (i = 0; i < 2; i++) { 968 //TODO could skip edid reading instead if we already have it, but what 969 //if monitors have been hot swapped? Also rely on edid for DVI-D detection 970 //if (routes->port_info[i].mon_type != mt_unknown) { 971 // SHOW_FLOW0(2, "known type, skpping detection"); 972 // continue; 973 //} 974 memset(&routes->port_info[i].edid , 0, sizeof(edid1_info)); 975 switch (routes->port_info[i].ddc_type) { 976 case ddc_monid: 977 case ddc_dvi: 978 case ddc_vga: 979 case ddc_crt2: 980 if (Radeon_ReadEDID(ai, 981 edid_regs[routes->port_info[i].ddc_type], 982 &routes->port_info[i].edid)) { 983 routes->port_info[i].edid_valid = true; 984 SHOW_FLOW(2, "Edid Data for CRTC %d on line %d", i, routes->port_info[i].ddc_type); 985 edid_dump(&routes->port_info[i].edid); 986 } else { 987 routes->port_info[i].mon_type = mt_none; 988 } 989 break; 990 991 default: 992 SHOW_FLOW(2, "No Edid Pin Assigned to CRTC %d ", i); 993 routes->port_info[i].mon_type = mt_none; 994 } 995 996 if (routes->port_info[i].edid_valid) { 997 if (routes->port_info[i].edid.display.input_type == 1) { 998 SHOW_FLOW0(2, "Must be a DVI monitor"); 999 1000 // Note some laptops have a DVI output that uses internal TMDS, 1001 // when its DVI is enabled by hotkey, LVDS panel is not used. 1002 // In this case, the laptop is configured as DVI+VGA as a normal 1003 // desktop card. 1004 // Also for laptop, when X starts with lid closed (no DVI connection) 1005 // both LDVS and TMDS are disable, we still need to treat it as a LVDS panel. 1006 if (routes->port_info[i].tmds_type == tmds_ext){ 1007 // store info about DVI-connected flat-panel 1008 if (Radeon_StoreFPEDID(ai, i, &routes->port_info[i].edid) == B_OK) { 1009 SHOW_INFO0(2, "Found Ext Laptop DVI"); 1010 routes->port_info[i].mon_type = mt_dfp; 1011 ai->si->flatpanels[i].is_fp2 = true; 1012 displays |= dd_dvi_ext; 1013 } else { 1014 SHOW_ERROR0(2, "Disabled Ext DVI - invalid EDID"); 1015 } 1016 } else { 1017 if (INREG(ai->regs, RADEON_FP_GEN_CNTL) & (1 << 7) || (!si->is_mobility)) { 1018 // store info about DVI-connected flat-panel 1019 if (Radeon_StoreFPEDID(ai, i, &routes->port_info[i].edid) == B_OK) { 1020 SHOW_INFO0(2, "Found DVI"); 1021 routes->port_info[i].mon_type = mt_dfp; 1022 displays |= dd_dvi; 1023 } else { 1024 SHOW_ERROR0(2, "Disabled DVI - invalid EDID"); 1025 } 1026 } else { 1027 SHOW_INFO0(2, "Laptop Panel Found"); 1028 routes->port_info[i].mon_type = mt_lcd; 1029 displays |= dd_lvds; 1030 } 1031 ai->si->flatpanels[1].is_fp2 = FALSE; 1032 } 1033 } else { 1034 // must be the analog portion of DVI 1035 // I'm not sure about Radeons with one CRTC - do they have DVI-I or DVI-D? 1036 // anyway - if there are two CRTC, analog portion must be connected 1037 // to TV-DAC; if there is one CRTC, it must be the normal VGA-DAC 1038 if (si->num_crtc > 1) { 1039 SHOW_FLOW0(2, "Must be an analog monitor on DVI port"); 1040 routes->port_info[i].mon_type = mt_crt; 1041 displays |= dd_tv_crt; 1042 } else { 1043 SHOW_FLOW0(2, "Seems to be a CRT on VGA port!?"); 1044 routes->port_info[i].mon_type = mt_crt; 1045 displays |= dd_crt; 1046 } 1047 } 1048 } 1049 } 1050 1051 if (!routes->port_info[0].edid_valid) { 1052 SHOW_INFO0(2, "Searching port 0"); 1053 if (si->is_mobility && (INREG(ai->regs, RADEON_BIOS_4_SCRATCH) & 4)) { 1054 SHOW_INFO0(2, "Found Laptop Panel"); 1055 routes->port_info[0].mon_type = mt_lcd; 1056 displays |= dd_lvds; 1057 } 1058 } 1059 1060 if (!routes->port_info[1].edid_valid) { 1061 if (si->is_mobility && (INREG(ai->regs, RADEON_FP2_GEN_CNTL) & RADEON_FP2_FPON)) { 1062 SHOW_INFO0(2, "Found Ext Laptop DVI"); 1063 routes->port_info[1].mon_type = mt_dfp; 1064 displays |= dd_dvi_ext; 1065 } 1066 } 1067 1068 if (routes->port_info[0].mon_type == mt_none) { 1069 if (routes->port_info[1].mon_type == mt_none) { 1070 routes->port_info[0].mon_type = mt_crt; 1071 } else { 1072 radeon_connector portSwapEntity; 1073 fp_info panelInfoSwapEntity; 1074 1075 portSwapEntity = routes->port_info[0]; 1076 routes->port_info[0] = routes->port_info[1]; 1077 routes->port_info[1] = portSwapEntity; 1078 1079 panelInfoSwapEntity = ai->si->flatpanels[0]; 1080 ai->si->flatpanels[0] = ai->si->flatpanels[1]; 1081 ai->si->flatpanels[1] = panelInfoSwapEntity; 1082 1083 SHOW_ERROR0(2, "swapping active port 2 to free port 1"); 1084 } 1085 1086 } 1087 1088 routes->reversed_DAC = false; 1089 if (routes->port_info[1].dac_type == dac_tvdac) { 1090 SHOW_ERROR0(2, "Reversed dac detected (not impl. yet)"); 1091 routes->reversed_DAC = true; 1092 } 1093 1094 // we may have overseen monitors if they don't support DDC or 1095 // have broken DDC data (like mine); 1096 // time to do a physical wire test; this test is more reliable, but it 1097 // leads to distortions on screen, which is not very nice to look at 1098 1099 // for DVI, there is no mercy if no DDC data is there - we wouldn't 1100 // even know the native resolution of the panel! 1101 1102 // all versions have a standard VGA port 1103 if ((displays & dd_crt) == 0 && (controlled_displays & dd_crt) != 0 1104 && Radeon_DetectCRT(ai)) 1105 displays |= dd_crt; 1106 1107 // check VGA signal routed to DVI port 1108 // (the detection code checks whether there is hardware for that) 1109 if ((displays & dd_tv_crt) == 0 1110 && (controlled_displays & dd_tv_crt) != 0 1111 && Radeon_DetectTVCRT(ai)) 1112 displays |= dd_tv_crt; 1113 1114 // check TV-out connector 1115 if ((controlled_displays & (dd_ctv | dd_stv)) != 0) 1116 displays |= Radeon_DetectTV(ai, (displays & dd_tv_crt) != 0); 1117 1118 SHOW_INFO(0, "Detected monitors: 0x%x", displays); 1119 1120 displays &= controlled_displays; 1121 1122 // if no monitor found, we define to have a CRT connected to CRT-DAC 1123 if (displays == 0) 1124 displays = dd_crt; 1125 1126 Radeon_ConnectorInfo(ai, 0, routes); 1127 Radeon_ConnectorInfo(ai, 1, routes); 1128 1129 ai->vc->connected_displays = displays; 1130 1131 RELEASE_BEN(si->cp.lock); 1132 } 1133