1 /* 2 * Copyright 2006-2011, Haiku, Inc. All Rights Reserved. 3 * Distributed under the terms of the MIT License. 4 * 5 * Authors: 6 * Alexander von Gluck, kallisti5@unixzen.com 7 */ 8 9 10 #include "connector.h" 11 12 #include <assert.h> 13 #include <Debug.h> 14 15 #include "accelerant_protos.h" 16 #include "accelerant.h" 17 #include "bios.h" 18 #include "encoder.h" 19 #include "gpu.h" 20 #include "utility.h" 21 22 23 #undef TRACE 24 25 #define TRACE_CONNECTOR 26 #ifdef TRACE_CONNECTOR 27 # define TRACE(x...) _sPrintf("radeon_hd: " x) 28 #else 29 # define TRACE(x...) ; 30 #endif 31 32 #define ERROR(x...) _sPrintf("radeon_hd: " x) 33 34 35 static void 36 gpio_lock_i2c(void* cookie, bool lock) 37 { 38 gpio_info* info = (gpio_info*)cookie; 39 40 uint32 buffer = 0; 41 42 if (lock == true) { 43 // hwCapable and > DCE3 44 if (info->i2c.hwCapable == true && gInfo->shared_info->dceMajor >= 3) { 45 // Switch GPIO pads to ddc mode 46 buffer = Read32(OUT, info->i2c.sclMaskReg); 47 buffer &= ~(1 << 16); 48 Write32(OUT, info->i2c.sclMaskReg, buffer); 49 } 50 51 // Clear pins 52 buffer = Read32(OUT, info->i2c.sclAReg) & ~info->i2c.sclAMask; 53 Write32(OUT, info->i2c.sclAReg, buffer); 54 buffer = Read32(OUT, info->i2c.sdaAReg) & ~info->i2c.sdaAMask; 55 Write32(OUT, info->i2c.sdaAReg, buffer); 56 } 57 58 // Set pins to input 59 buffer = Read32(OUT, info->i2c.sclEnReg) & ~info->i2c.sclEnMask; 60 Write32(OUT, info->i2c.sclEnReg, buffer); 61 buffer = Read32(OUT, info->i2c.sdaEnReg) & ~info->i2c.sdaEnMask; 62 Write32(OUT, info->i2c.sdaEnReg, buffer); 63 64 // mask clock GPIO pins for software use 65 buffer = Read32(OUT, info->i2c.sclMaskReg); 66 if (lock == true) 67 buffer |= info->i2c.sclMask; 68 else 69 buffer &= ~info->i2c.sclMask; 70 71 Write32(OUT, info->i2c.sclMaskReg, buffer); 72 Read32(OUT, info->i2c.sclMaskReg); 73 74 // mask data GPIO pins for software use 75 buffer = Read32(OUT, info->i2c.sdaMaskReg); 76 if (lock == true) 77 buffer |= info->i2c.sdaMask; 78 else 79 buffer &= ~info->i2c.sdaMask; 80 81 Write32(OUT, info->i2c.sdaMaskReg, buffer); 82 Read32(OUT, info->i2c.sdaMaskReg); 83 } 84 85 86 static status_t 87 gpio_get_i2c_bit(void* cookie, int* _clock, int* _data) 88 { 89 gpio_info* info = (gpio_info*)cookie; 90 91 uint32 scl = Read32(OUT, info->i2c.sclYReg) & info->i2c.sclYMask; 92 uint32 sda = Read32(OUT, info->i2c.sdaYReg) & info->i2c.sdaYMask; 93 94 *_clock = scl != 0; 95 *_data = sda != 0; 96 97 return B_OK; 98 } 99 100 101 static status_t 102 gpio_set_i2c_bit(void* cookie, int clock, int data) 103 { 104 gpio_info* info = (gpio_info*)cookie; 105 106 uint32 scl = Read32(OUT, info->i2c.sclEnReg) & ~info->i2c.sclEnMask; 107 scl |= clock ? 0 : info->i2c.sclEnMask; 108 Write32(OUT, info->i2c.sclEnReg, scl); 109 Read32(OUT, info->i2c.sclEnReg); 110 111 uint32 sda = Read32(OUT, info->i2c.sdaEnReg) & ~info->i2c.sdaEnMask; 112 sda |= data ? 0 : info->i2c.sdaEnMask; 113 Write32(OUT, info->i2c.sdaEnReg, sda); 114 Read32(OUT, info->i2c.sdaEnReg); 115 116 return B_OK; 117 } 118 119 120 uint16 121 connector_pick_atom_hpdid(uint32 connectorIndex) 122 { 123 radeon_shared_info &info = *gInfo->shared_info; 124 125 uint16 atomHPDID = 0xff; 126 uint16 hpdPinIndex = gConnector[connectorIndex]->hpdPinIndex; 127 if (info.dceMajor >= 4 128 && gGPIOInfo[hpdPinIndex]->valid) { 129 130 // See mmDC_GPIO_HPD_A in drm for register value 131 uint32 targetReg = AVIVO_DC_GPIO_HPD_A; 132 if (info.dceMajor >= 13) { 133 ERROR("WARNING: CHECK NEW DCE mmDC_GPIO_HPD_A value!\n"); 134 targetReg = POL_mmDC_GPIO_HPD_A; 135 } else if (info.dceMajor >= 12) 136 targetReg = POL_mmDC_GPIO_HPD_A; 137 else if (info.dceMajor >= 11) 138 targetReg = CAR_mmDC_GPIO_HPD_A; 139 else if (info.dceMajor >= 10) 140 targetReg = VOL_mmDC_GPIO_HPD_A; 141 else if (info.dceMajor >= 8) 142 targetReg = SEA_mmDC_GPIO_HPD_A; 143 else if (info.dceMajor >= 6) 144 targetReg = SI_DC_GPIO_HPD_A; 145 else if (info.dceMajor >= 4) 146 targetReg = EVERGREEN_DC_GPIO_HPD_A; 147 148 // You're drunk AMD, go home. (this makes no sense) 149 if (gGPIOInfo[hpdPinIndex]->hwReg == targetReg) { 150 switch(gGPIOInfo[hpdPinIndex]->hwMask) { 151 case (1 << 0): 152 atomHPDID = 0; 153 break; 154 case (1 << 8): 155 atomHPDID = 1; 156 break; 157 case (1 << 16): 158 atomHPDID = 2; 159 break; 160 case (1 << 24): 161 atomHPDID = 3; 162 break; 163 case (1 << 26): 164 atomHPDID = 4; 165 break; 166 case (1 << 28): 167 atomHPDID = 5; 168 break; 169 } 170 } 171 } 172 return atomHPDID; 173 } 174 175 176 bool 177 connector_read_edid(uint32 connectorIndex, edid1_info* edid) 178 { 179 // ensure things are sane 180 uint32 i2cPinIndex = gConnector[connectorIndex]->i2cPinIndex; 181 if (gGPIOInfo[i2cPinIndex]->valid == false 182 || gGPIOInfo[i2cPinIndex]->i2c.valid == false) { 183 ERROR("%s: invalid gpio %" B_PRIu32 " for connector %" B_PRIu32 "\n", 184 __func__, i2cPinIndex, connectorIndex); 185 return false; 186 } 187 188 i2c_bus bus; 189 190 ddc2_init_timing(&bus); 191 bus.cookie = (void*)gGPIOInfo[i2cPinIndex]; 192 bus.set_signals = &gpio_set_i2c_bit; 193 bus.get_signals = &gpio_get_i2c_bit; 194 195 gpio_lock_i2c(bus.cookie, true); 196 status_t edid_result = ddc2_read_edid1(&bus, edid, NULL, NULL); 197 gpio_lock_i2c(bus.cookie, false); 198 199 if (edid_result != B_OK) 200 return false; 201 202 TRACE("%s: found edid monitor on connector #%" B_PRId32 "\n", 203 __func__, connectorIndex); 204 205 return true; 206 } 207 208 209 bool 210 connector_read_mode_lvds(uint32 connectorIndex, display_mode* mode) 211 { 212 assert(mode); 213 214 uint8 dceMajor; 215 uint8 dceMinor; 216 int index = GetIndexIntoMasterTable(DATA, LVDS_Info); 217 uint16 offset; 218 219 union atomLVDSInfo { 220 struct _ATOM_LVDS_INFO info; 221 struct _ATOM_LVDS_INFO_V12 info_12; 222 }; 223 224 // Wipe out display_mode 225 memset(mode, 0, sizeof(display_mode)); 226 227 if (atom_parse_data_header(gAtomContext, index, NULL, 228 &dceMajor, &dceMinor, &offset) == B_OK) { 229 230 union atomLVDSInfo* lvdsInfo 231 = (union atomLVDSInfo*)(gAtomContext->bios + offset); 232 233 display_timing* timing = &mode->timing; 234 235 // Pixel Clock 236 timing->pixel_clock 237 = B_LENDIAN_TO_HOST_INT16(lvdsInfo->info.sLCDTiming.usPixClk) * 10; 238 // Horizontal 239 timing->h_display 240 = B_LENDIAN_TO_HOST_INT16(lvdsInfo->info.sLCDTiming.usHActive); 241 timing->h_total = timing->h_display + B_LENDIAN_TO_HOST_INT16( 242 lvdsInfo->info.sLCDTiming.usHBlanking_Time); 243 timing->h_sync_start = timing->h_display 244 + B_LENDIAN_TO_HOST_INT16(lvdsInfo->info.sLCDTiming.usHSyncOffset); 245 timing->h_sync_end = timing->h_sync_start 246 + B_LENDIAN_TO_HOST_INT16(lvdsInfo->info.sLCDTiming.usHSyncWidth); 247 // Vertical 248 timing->v_display 249 = B_LENDIAN_TO_HOST_INT16(lvdsInfo->info.sLCDTiming.usVActive); 250 timing->v_total = timing->v_display + B_LENDIAN_TO_HOST_INT16( 251 lvdsInfo->info.sLCDTiming.usVBlanking_Time); 252 timing->v_sync_start = timing->v_display 253 + B_LENDIAN_TO_HOST_INT16(lvdsInfo->info.sLCDTiming.usVSyncOffset); 254 timing->v_sync_end = timing->v_sync_start 255 + B_LENDIAN_TO_HOST_INT16(lvdsInfo->info.sLCDTiming.usVSyncWidth); 256 257 #if 0 258 // Who cares. 259 uint32 powerDelay 260 = B_LENDIAN_TO_HOST_INT16(lvdsInfo->info.usOffDelayInMs); 261 #endif 262 263 // Store special lvds flags the encoder setup needs 264 gConnector[connectorIndex]->lvdsFlags = lvdsInfo->info.ucLVDS_Misc; 265 266 // Spread Spectrum ID (in SS table) 267 gInfo->lvdsSpreadSpectrumID = lvdsInfo->info.ucSS_Id; 268 269 uint16 flags = B_LENDIAN_TO_HOST_INT16( 270 lvdsInfo->info.sLCDTiming.susModeMiscInfo.usAccess); 271 272 if ((flags & ATOM_VSYNC_POLARITY) == 0) 273 timing->flags |= B_POSITIVE_VSYNC; 274 if ((flags & ATOM_HSYNC_POLARITY) == 0) 275 timing->flags |= B_POSITIVE_HSYNC; 276 277 // Extra flags 278 if ((flags & ATOM_INTERLACE) != 0) 279 timing->flags |= B_TIMING_INTERLACED; 280 281 #if 0 282 // We don't use these timing flags at the moment 283 if ((flags & ATOM_COMPOSITESYNC) != 0) 284 timing->flags |= MODE_FLAG_CSYNC; 285 if ((flags & ATOM_DOUBLE_CLOCK_MODE) != 0) 286 timing->flags |= MODE_FLAG_DBLSCAN; 287 #endif 288 289 mode->h_display_start = 0; 290 mode->v_display_start = 0; 291 mode->virtual_width = timing->h_display; 292 mode->virtual_height = timing->v_display; 293 294 // Assume 32-bit color 295 mode->space = B_RGB32_LITTLE; 296 297 TRACE("%s: %" B_PRIu32 " %" B_PRIu16 " %" B_PRIu16 " %" B_PRIu16 298 " %" B_PRIu16 " %" B_PRIu16 " %" B_PRIu16 " %" B_PRIu16 299 " %" B_PRIu16 "\n", __func__, timing->pixel_clock, 300 timing->h_display, timing->h_sync_start, timing->h_sync_end, 301 timing->h_total, timing->v_display, timing->v_sync_start, 302 timing->v_sync_end, timing->v_total); 303 304 return true; 305 } 306 return false; 307 } 308 309 310 static status_t 311 connector_attach_gpio_i2c(uint32 connectorIndex, uint8 hwPin) 312 { 313 gConnector[connectorIndex]->i2cPinIndex = 0; 314 for (uint32 i = 0; i < MAX_GPIO_PINS; i++) { 315 if (gGPIOInfo[i]->hwPin != hwPin) 316 continue; 317 gConnector[connectorIndex]->i2cPinIndex = i; 318 return B_OK; 319 } 320 321 // We couldnt find the GPIO pin in the known GPIO pins. 322 TRACE("%s: can't find GPIO pin 0x%" B_PRIX8 " for connector %" B_PRIu32 "\n", 323 __func__, hwPin, connectorIndex); 324 return B_ERROR; 325 } 326 327 328 static status_t 329 connector_attach_gpio_router(uint32 connectorIndex, uint8 hwPin) 330 { 331 gConnector[connectorIndex]->router.i2cPinIndex = 0; 332 for (uint32 i = 0; i < MAX_GPIO_PINS; i++) { 333 if (gGPIOInfo[i]->hwPin != hwPin) 334 continue; 335 gConnector[connectorIndex]->router.i2cPinIndex = i; 336 return B_OK; 337 } 338 339 // We couldnt find the GPIO pin in the known GPIO pins. 340 TRACE("%s: can't find GPIO pin 0x%" B_PRIX8 " for connector %" B_PRIu32 "\n", 341 __func__, hwPin, connectorIndex); 342 return B_ERROR; 343 } 344 345 346 static status_t 347 connector_attach_gpio_hpd(uint32 connectorIndex, uint8 hwPin) 348 { 349 gConnector[connectorIndex]->hpdPinIndex = 0; 350 351 for (uint32 i = 0; i < MAX_GPIO_PINS; i++) { 352 if (gGPIOInfo[i]->hwPin != hwPin) 353 continue; 354 gConnector[connectorIndex]->hpdPinIndex = i; 355 return B_OK; 356 } 357 358 // We couldnt find the GPIO pin in the known GPIO pins. 359 TRACE("%s: can't find GPIO pin 0x%" B_PRIX8 " for connector %" B_PRIu32 "\n", 360 __func__, hwPin, connectorIndex); 361 return B_ERROR; 362 } 363 364 365 static status_t 366 gpio_general_populate() 367 { 368 int index = GetIndexIntoMasterTable(DATA, GPIO_Pin_LUT); 369 uint16 tableOffset; 370 uint16 tableSize; 371 372 struct _ATOM_GPIO_PIN_LUT* gpioInfo; 373 374 if (atom_parse_data_header(gAtomContext, index, &tableSize, NULL, NULL, 375 &tableOffset)) { 376 ERROR("%s: could't read GPIO_Pin_LUT table from AtomBIOS index %d!\n", 377 __func__, index); 378 } 379 gpioInfo = (struct _ATOM_GPIO_PIN_LUT*)(gAtomContext->bios + tableOffset); 380 381 int numIndices = (tableSize - sizeof(ATOM_COMMON_TABLE_HEADER)) / 382 sizeof(ATOM_GPIO_PIN_ASSIGNMENT); 383 384 // Find the next available GPIO pin index 385 int32 gpioIndex = -1; 386 for(int32 pin = 0; pin < MAX_GPIO_PINS; pin++) { 387 if (!gGPIOInfo[pin]->valid) { 388 gpioIndex = pin; 389 break; 390 } 391 } 392 if (gpioIndex < 0) { 393 ERROR("%s: ERROR: Out of space for additional GPIO pins!\n", __func__); 394 return B_ERROR; 395 } 396 397 ATOM_GPIO_PIN_ASSIGNMENT* pin = gpioInfo->asGPIO_Pin; 398 for (int i = 0; i < numIndices; i++) { 399 if (gGPIOInfo[gpioIndex]->valid) { 400 ERROR("%s: BUG: Attempting to fill already populated gpio pin!\n", 401 __func__); 402 return B_ERROR; 403 } 404 gGPIOInfo[gpioIndex]->valid = true; 405 gGPIOInfo[gpioIndex]->i2c.valid = false; 406 gGPIOInfo[gpioIndex]->hwPin = pin->ucGPIO_ID; 407 gGPIOInfo[gpioIndex]->hwReg 408 = B_LENDIAN_TO_HOST_INT16(pin->usGpioPin_AIndex) * 4; 409 gGPIOInfo[gpioIndex]->hwMask 410 = (1 << pin->ucGpioPinBitShift); 411 pin = (ATOM_GPIO_PIN_ASSIGNMENT*)((uint8*)pin 412 + sizeof(ATOM_GPIO_PIN_ASSIGNMENT)); 413 414 TRACE("%s: general GPIO @ %" B_PRId32 ", valid: %s, " 415 "hwPin: 0x%" B_PRIX32 "\n", __func__, gpioIndex, 416 gGPIOInfo[gpioIndex]->valid ? "true" : "false", 417 gGPIOInfo[gpioIndex]->hwPin); 418 419 gpioIndex++; 420 } 421 return B_OK; 422 } 423 424 425 static status_t 426 gpio_i2c_populate() 427 { 428 radeon_shared_info &info = *gInfo->shared_info; 429 430 int index = GetIndexIntoMasterTable(DATA, GPIO_I2C_Info); 431 uint16 tableOffset; 432 uint16 tableSize; 433 434 if (atom_parse_data_header(gAtomContext, index, &tableSize, 435 NULL, NULL, &tableOffset) != B_OK) { 436 ERROR("%s: could't read GPIO_I2C_Info table from AtomBIOS index %d!\n", 437 __func__, index); 438 return B_ERROR; 439 } 440 441 struct _ATOM_GPIO_I2C_INFO* i2cInfo 442 = (struct _ATOM_GPIO_I2C_INFO*)(gAtomContext->bios + tableOffset); 443 444 uint32 numIndices = (tableSize - sizeof(ATOM_COMMON_TABLE_HEADER)) 445 / sizeof(ATOM_GPIO_I2C_ASSIGMENT); 446 447 if (numIndices > ATOM_MAX_SUPPORTED_DEVICE) { 448 ERROR("%s: ERROR: AtomBIOS contains more GPIO_Info items then I" 449 "was prepared for! (seen: %" B_PRIu32 "; max: %" B_PRIu32 ")\n", 450 __func__, numIndices, (uint32)ATOM_MAX_SUPPORTED_DEVICE); 451 return B_ERROR; 452 } 453 454 // Find the next available GPIO pin index 455 int32 gpioIndex = -1; 456 for(int32 pin = 0; pin < MAX_GPIO_PINS; pin++) { 457 if (!gGPIOInfo[pin]->valid) { 458 gpioIndex = pin; 459 break; 460 } 461 } 462 if (gpioIndex < 0) { 463 ERROR("%s: ERROR: Out of space for additional GPIO pins!\n", __func__); 464 return B_ERROR; 465 } 466 467 for (uint32 i = 0; i < numIndices; i++) { 468 if (gGPIOInfo[gpioIndex]->valid) { 469 ERROR("%s: BUG: Attempting to fill already populated gpio pin!\n", 470 __func__); 471 return B_ERROR; 472 } 473 ATOM_GPIO_I2C_ASSIGMENT* gpio = &i2cInfo->asGPIO_Info[i]; 474 475 if (info.dceMajor >= 3) { 476 if (i == 4 && B_LENDIAN_TO_HOST_INT16(gpio->usClkMaskRegisterIndex) 477 == 0x1fda && gpio->sucI2cId.ucAccess == 0x94) { 478 gpio->sucI2cId.ucAccess = 0x14; 479 TRACE("%s: BUG: GPIO override for DCE 3 occured\n", __func__); 480 } 481 } 482 483 if (info.dceMajor >= 4) { 484 if (i == 7 && B_LENDIAN_TO_HOST_INT16(gpio->usClkMaskRegisterIndex) 485 == 0x1936 && gpio->sucI2cId.ucAccess == 0) { 486 gpio->sucI2cId.ucAccess = 0x97; 487 gpio->ucDataMaskShift = 8; 488 gpio->ucDataEnShift = 8; 489 gpio->ucDataY_Shift = 8; 490 gpio->ucDataA_Shift = 8; 491 TRACE("%s: BUG: GPIO override for DCE 4 occured\n", __func__); 492 } 493 } 494 495 // populate gpio information 496 gGPIOInfo[gpioIndex]->hwPin = gpio->sucI2cId.ucAccess; 497 gGPIOInfo[gpioIndex]->i2c.hwCapable 498 = (gpio->sucI2cId.sbfAccess.bfHW_Capable) ? true : false; 499 500 // GPIO mask (Allows software to control the GPIO pad) 501 // 0 = chip access; 1 = only software; 502 gGPIOInfo[gpioIndex]->i2c.sclMaskReg 503 = B_LENDIAN_TO_HOST_INT16(gpio->usClkMaskRegisterIndex) * 4; 504 gGPIOInfo[gpioIndex]->i2c.sdaMaskReg 505 = B_LENDIAN_TO_HOST_INT16(gpio->usDataMaskRegisterIndex) * 4; 506 gGPIOInfo[gpioIndex]->i2c.sclMask = 1 << gpio->ucClkMaskShift; 507 gGPIOInfo[gpioIndex]->i2c.sdaMask = 1 << gpio->ucDataMaskShift; 508 509 // GPIO output / write (A) enable 510 // 0 = GPIO input (Y); 1 = GPIO output (A); 511 gGPIOInfo[gpioIndex]->i2c.sclEnReg 512 = B_LENDIAN_TO_HOST_INT16(gpio->usClkEnRegisterIndex) * 4; 513 gGPIOInfo[gpioIndex]->i2c.sdaEnReg 514 = B_LENDIAN_TO_HOST_INT16(gpio->usDataEnRegisterIndex) * 4; 515 gGPIOInfo[gpioIndex]->i2c.sclEnMask = 1 << gpio->ucClkEnShift; 516 gGPIOInfo[gpioIndex]->i2c.sdaEnMask = 1 << gpio->ucDataEnShift; 517 518 // GPIO output / write (A) 519 gGPIOInfo[gpioIndex]->i2c.sclAReg 520 = B_LENDIAN_TO_HOST_INT16(gpio->usClkA_RegisterIndex) * 4; 521 gGPIOInfo[gpioIndex]->i2c.sdaAReg 522 = B_LENDIAN_TO_HOST_INT16(gpio->usDataA_RegisterIndex) * 4; 523 gGPIOInfo[gpioIndex]->i2c.sclAMask = 1 << gpio->ucClkA_Shift; 524 gGPIOInfo[gpioIndex]->i2c.sdaAMask = 1 << gpio->ucDataA_Shift; 525 526 // GPIO input / read (Y) 527 gGPIOInfo[gpioIndex]->i2c.sclYReg 528 = B_LENDIAN_TO_HOST_INT16(gpio->usClkY_RegisterIndex) * 4; 529 gGPIOInfo[gpioIndex]->i2c.sdaYReg 530 = B_LENDIAN_TO_HOST_INT16(gpio->usDataY_RegisterIndex) * 4; 531 gGPIOInfo[gpioIndex]->i2c.sclYMask = 1 << gpio->ucClkY_Shift; 532 gGPIOInfo[gpioIndex]->i2c.sdaYMask = 1 << gpio->ucDataY_Shift; 533 534 // ensure data is valid 535 gGPIOInfo[gpioIndex]->i2c.valid 536 = gGPIOInfo[gpioIndex]->i2c.sclMaskReg ? true : false; 537 gGPIOInfo[gpioIndex]->valid = gGPIOInfo[gpioIndex]->i2c.valid; 538 539 TRACE("%s: i2c GPIO @ %" B_PRIu32 ", valid: %s, hwPin: 0x%" B_PRIX32 "\n", 540 __func__, gpioIndex, gGPIOInfo[gpioIndex]->valid ? "true" : "false", 541 gGPIOInfo[gpioIndex]->hwPin); 542 543 gpioIndex++; 544 } 545 546 return B_OK; 547 } 548 549 550 status_t 551 gpio_populate() 552 { 553 status_t result = gpio_general_populate(); 554 if (result != B_OK) 555 return result; 556 557 result = gpio_i2c_populate(); 558 return result; 559 } 560 561 562 status_t 563 connector_probe_legacy() 564 { 565 int index = GetIndexIntoMasterTable(DATA, SupportedDevicesInfo); 566 uint8 tableMajor; 567 uint8 tableMinor; 568 uint16 tableSize; 569 uint16 tableOffset; 570 571 if (atom_parse_data_header(gAtomContext, index, &tableSize, 572 &tableMajor, &tableMinor, &tableOffset) != B_OK) { 573 ERROR("%s: unable to parse data header!\n", __func__); 574 return B_ERROR; 575 } 576 577 union atomSupportedDevices { 578 struct _ATOM_SUPPORTED_DEVICES_INFO info; 579 struct _ATOM_SUPPORTED_DEVICES_INFO_2 info_2; 580 struct _ATOM_SUPPORTED_DEVICES_INFO_2d1 info_2d1; 581 }; 582 union atomSupportedDevices* supportedDevices; 583 supportedDevices = (union atomSupportedDevices*) 584 (gAtomContext->bios + tableOffset); 585 586 uint16 deviceSupport 587 = B_LENDIAN_TO_HOST_INT16(supportedDevices->info.usDeviceSupport); 588 589 uint32 maxDevice; 590 591 if (tableMajor > 1) 592 maxDevice = ATOM_MAX_SUPPORTED_DEVICE; 593 else 594 maxDevice = ATOM_MAX_SUPPORTED_DEVICE_INFO; 595 596 uint32 i; 597 uint32 connectorIndex = 0; 598 for (i = 0; i < maxDevice; i++) { 599 600 gConnector[connectorIndex]->valid = false; 601 602 // check if this connector is used 603 if ((deviceSupport & (1 << i)) == 0) 604 continue; 605 606 if (i == ATOM_DEVICE_CV_INDEX) { 607 TRACE("%s: skipping component video\n", 608 __func__); 609 continue; 610 } 611 612 ATOM_CONNECTOR_INFO_I2C ci 613 = supportedDevices->info.asConnInfo[i]; 614 615 gConnector[connectorIndex]->type = kConnectorConvertLegacy[ 616 ci.sucConnectorInfo.sbfAccess.bfConnectorType]; 617 618 if (gConnector[connectorIndex]->type == VIDEO_CONNECTOR_UNKNOWN) { 619 TRACE("%s: skipping unknown connector at %" B_PRId32 620 " of 0x%" B_PRIX8 "\n", __func__, i, 621 ci.sucConnectorInfo.sbfAccess.bfConnectorType); 622 continue; 623 } 624 625 // TODO: give tv unique connector ids 626 627 // Always set CRT1 and CRT2 as VGA, some cards incorrectly set 628 // VGA ports as DVI 629 if (i == ATOM_DEVICE_CRT1_INDEX || i == ATOM_DEVICE_CRT2_INDEX) 630 gConnector[connectorIndex]->type = VIDEO_CONNECTOR_VGA; 631 632 uint8 dac = ci.sucConnectorInfo.sbfAccess.bfAssociatedDAC; 633 uint32 encoderObject = encoder_object_lookup((1 << i), dac); 634 uint32 encoderID = (encoderObject & OBJECT_ID_MASK) >> OBJECT_ID_SHIFT; 635 636 gConnector[connectorIndex]->valid = true; 637 gConnector[connectorIndex]->flags = (1 << i); 638 gConnector[connectorIndex]->encoder.valid = true; 639 gConnector[connectorIndex]->encoder.objectID = encoderID; 640 gConnector[connectorIndex]->encoder.type 641 = encoder_type_lookup(encoderID, (1 << i)); 642 643 // TODO: Eval external encoders on legacy connector probe 644 gConnector[connectorIndex]->encoderExternal.valid = false; 645 // encoder_is_external(encoderID); 646 647 connector_attach_gpio_i2c(connectorIndex, ci.sucI2cId.ucAccess); 648 649 pll_limit_probe(&gConnector[connectorIndex]->encoder.pll); 650 651 connectorIndex++; 652 } 653 654 // TODO: combine shared connectors 655 656 if (connectorIndex == 0) { 657 TRACE("%s: zero connectors found using legacy detection\n", __func__); 658 return B_ERROR; 659 } 660 661 return B_OK; 662 } 663 664 665 // r600+ 666 status_t 667 connector_probe() 668 { 669 int index = GetIndexIntoMasterTable(DATA, Object_Header); 670 uint8 tableMajor; 671 uint8 tableMinor; 672 uint16 tableSize; 673 uint16 tableOffset; 674 675 if (atom_parse_data_header(gAtomContext, index, &tableSize, 676 &tableMajor, &tableMinor, &tableOffset) != B_OK) { 677 ERROR("%s: ERROR: parsing data header failed!\n", __func__); 678 return B_ERROR; 679 } 680 681 if (tableMinor < 2) { 682 ERROR("%s: ERROR: table minor version unknown! " 683 "(%" B_PRIu8 ".%" B_PRIu8 ")\n", __func__, tableMajor, tableMinor); 684 return B_ERROR; 685 } 686 687 ATOM_CONNECTOR_OBJECT_TABLE* connectorObject; 688 ATOM_ENCODER_OBJECT_TABLE* encoderObject; 689 ATOM_OBJECT_TABLE* routerObject; 690 ATOM_DISPLAY_OBJECT_PATH_TABLE* pathObject; 691 ATOM_OBJECT_HEADER* objectHeader; 692 693 objectHeader = (ATOM_OBJECT_HEADER*)(gAtomContext->bios + tableOffset); 694 pathObject = (ATOM_DISPLAY_OBJECT_PATH_TABLE*) 695 (gAtomContext->bios + tableOffset 696 + B_LENDIAN_TO_HOST_INT16(objectHeader->usDisplayPathTableOffset)); 697 connectorObject = (ATOM_CONNECTOR_OBJECT_TABLE*) 698 (gAtomContext->bios + tableOffset 699 + B_LENDIAN_TO_HOST_INT16(objectHeader->usConnectorObjectTableOffset)); 700 encoderObject = (ATOM_ENCODER_OBJECT_TABLE*) 701 (gAtomContext->bios + tableOffset 702 + B_LENDIAN_TO_HOST_INT16(objectHeader->usEncoderObjectTableOffset)); 703 routerObject = (ATOM_OBJECT_TABLE*) 704 (gAtomContext->bios + tableOffset 705 + B_LENDIAN_TO_HOST_INT16(objectHeader->usRouterObjectTableOffset)); 706 int deviceSupport = B_LENDIAN_TO_HOST_INT16(objectHeader->usDeviceSupport); 707 708 int pathSize = 0; 709 int32 i = 0; 710 711 TRACE("%s: found %" B_PRIu8 " potential display paths.\n", __func__, 712 pathObject->ucNumOfDispPath); 713 714 uint32 connectorIndex = 0; 715 for (i = 0; i < pathObject->ucNumOfDispPath; i++) { 716 717 if (connectorIndex >= ATOM_MAX_SUPPORTED_DEVICE) 718 continue; 719 720 uint8* address = (uint8*)pathObject->asDispPath; 721 ATOM_DISPLAY_OBJECT_PATH* path; 722 address += pathSize; 723 path = (ATOM_DISPLAY_OBJECT_PATH*)address; 724 pathSize += B_LENDIAN_TO_HOST_INT16(path->usSize); 725 726 uint32 connectorType; 727 uint16 connectorFlags = B_LENDIAN_TO_HOST_INT16(path->usDeviceTag); 728 729 if ((deviceSupport & connectorFlags) != 0) { 730 731 uint16 connectorObjectID 732 = (B_LENDIAN_TO_HOST_INT16(path->usConnObjectId) 733 & OBJECT_ID_MASK) >> OBJECT_ID_SHIFT; 734 //uint8 con_obj_num 735 // = (B_LENDIAN_TO_HOST_INT16(path->usConnObjectId) 736 // & ENUM_ID_MASK) >> ENUM_ID_SHIFT; 737 //uint8 con_obj_type 738 // = (B_LENDIAN_TO_HOST_INT16(path->usConnObjectId) 739 // & OBJECT_TYPE_MASK) >> OBJECT_TYPE_SHIFT; 740 741 if (connectorFlags == ATOM_DEVICE_CV_SUPPORT) { 742 TRACE("%s: Path #%" B_PRId32 ": skipping component video.\n", 743 __func__, i); 744 continue; 745 } 746 747 radeon_shared_info &info = *gInfo->shared_info; 748 749 // protect kConnectorConvert 750 if (connectorObjectID >= B_COUNT_OF(kConnectorConvert)) { 751 // This can happen when new atombios revisions introduce 752 // new CONNECTOR_OBJECT_ID_* defines (rare) 753 ERROR("%s: Path #%" B_PRId32 ": Unknown connector object ID!\n", 754 __func__, i); 755 continue; 756 } 757 758 uint16 igpLaneInfo; 759 if ((info.chipsetFlags & CHIP_IGP) != 0) { 760 ERROR("%s: TODO: IGP chip connector detection\n", __func__); 761 // try non-IGP method for now 762 igpLaneInfo = 0; 763 connectorType = kConnectorConvert[connectorObjectID]; 764 } else { 765 igpLaneInfo = 0; 766 connectorType = kConnectorConvert[connectorObjectID]; 767 } 768 769 if (connectorType == VIDEO_CONNECTOR_UNKNOWN) { 770 ERROR("%s: Path #%" B_PRId32 ": skipping unknown connector.\n", 771 __func__, i); 772 continue; 773 } 774 775 connector_info* connector = gConnector[connectorIndex]; 776 777 int32 j; 778 for (j = 0; j < ((B_LENDIAN_TO_HOST_INT16(path->usSize) - 8) / 2); 779 j++) { 780 //uint16 grph_obj_id 781 // = (B_LENDIAN_TO_HOST_INT16(path->usGraphicObjIds[j]) 782 // & OBJECT_ID_MASK) >> OBJECT_ID_SHIFT; 783 //uint8 grph_obj_num 784 // = (B_LENDIAN_TO_HOST_INT16(path->usGraphicObjIds[j]) & 785 // ENUM_ID_MASK) >> ENUM_ID_SHIFT; 786 uint8 graphicObjectType 787 = (B_LENDIAN_TO_HOST_INT16(path->usGraphicObjIds[j]) & 788 OBJECT_TYPE_MASK) >> OBJECT_TYPE_SHIFT; 789 790 if (graphicObjectType == GRAPH_OBJECT_TYPE_ENCODER) { 791 // Found an encoder 792 int32 k; 793 for (k = 0; k < encoderObject->ucNumberOfObjects; k++) { 794 uint16 encoderObjectRaw 795 = B_LENDIAN_TO_HOST_INT16( 796 encoderObject->asObjects[k].usObjectID); 797 if (B_LENDIAN_TO_HOST_INT16(path->usGraphicObjIds[j]) 798 == encoderObjectRaw) { 799 ATOM_COMMON_RECORD_HEADER* record 800 = (ATOM_COMMON_RECORD_HEADER*) 801 ((uint16*)gAtomContext->bios + tableOffset 802 + B_LENDIAN_TO_HOST_INT16( 803 encoderObject->asObjects[k].usRecordOffset)); 804 ATOM_ENCODER_CAP_RECORD* capRecord; 805 uint16 caps = 0; 806 while (record->ucRecordSize > 0 807 && record->ucRecordType > 0 808 && record->ucRecordType 809 <= ATOM_MAX_OBJECT_RECORD_NUMBER) { 810 switch (record->ucRecordType) { 811 case ATOM_ENCODER_CAP_RECORD_TYPE: 812 capRecord = (ATOM_ENCODER_CAP_RECORD*) 813 record; 814 caps = B_LENDIAN_TO_HOST_INT16( 815 capRecord->usEncoderCap); 816 break; 817 } 818 record = (ATOM_COMMON_RECORD_HEADER*) 819 ((char*)record + record->ucRecordSize); 820 } 821 822 uint32 encoderID 823 = (encoderObjectRaw & OBJECT_ID_MASK) 824 >> OBJECT_ID_SHIFT; 825 826 uint32 encoderType = encoder_type_lookup(encoderID, 827 connectorFlags); 828 829 if (encoderType == VIDEO_ENCODER_NONE) { 830 ERROR("%s: Path #%" B_PRId32 ":" 831 "skipping unknown encoder.\n", 832 __func__, i); 833 continue; 834 } 835 836 encoder_info* encoder; 837 838 // External encoders are behind DVO or UNIPHY 839 if (encoder_is_external(encoderID)) { 840 encoder = &connector->encoderExternal; 841 encoder->isExternal = true; 842 encoder->isDPBridge 843 = encoder_is_dp_bridge(encoderID); 844 } else { 845 encoder = &connector->encoder; 846 encoder->isExternal = false; 847 encoder->isDPBridge = false; 848 } 849 850 // Set up found connector encoder generics 851 encoder->valid = true; 852 encoder->capabilities = caps; 853 encoder->objectID = encoderID; 854 encoder->type = encoderType; 855 encoder->linkEnumeration 856 = (encoderObjectRaw & ENUM_ID_MASK) 857 >> ENUM_ID_SHIFT; 858 pll_limit_probe(&encoder->pll); 859 } 860 } 861 // END if object is encoder 862 } else if (graphicObjectType == GRAPH_OBJECT_TYPE_ROUTER) { 863 int32 k; 864 for (k = 0; k < routerObject->ucNumberOfObjects; k++) { 865 uint16 routerObjectID 866 = B_LENDIAN_TO_HOST_INT16(routerObject->asObjects[k].usObjectID); 867 if (B_LENDIAN_TO_HOST_INT16(path->usGraphicObjIds[j]) == routerObjectID) { 868 ATOM_COMMON_RECORD_HEADER* record = (ATOM_COMMON_RECORD_HEADER*) 869 ((uint16*)gAtomContext->bios + tableOffset 870 + B_LENDIAN_TO_HOST_INT16( 871 routerObject->asObjects[k].usRecordOffset)); 872 ATOM_I2C_RECORD* i2cRecord; 873 ATOM_I2C_ID_CONFIG_ACCESS* i2cConfig; 874 ATOM_ROUTER_DDC_PATH_SELECT_RECORD* ddcPath; 875 ATOM_ROUTER_DATA_CLOCK_PATH_SELECT_RECORD* cdPath; 876 ATOM_SRC_DST_TABLE_FOR_ONE_OBJECT* routerConnTable 877 = (ATOM_SRC_DST_TABLE_FOR_ONE_OBJECT *) 878 ((uint16*)gAtomContext->bios + tableOffset 879 + B_LENDIAN_TO_HOST_INT16( 880 routerObject->asObjects[k].usSrcDstTableOffset)); 881 uint8* destObjCount = (uint8*)((uint8*)routerConnTable + 1 882 + (routerConnTable->ucNumberOfSrc * 2)); 883 uint16 *dstObjs = (uint16 *)(destObjCount + 1); 884 885 int enumId; 886 router_info* router = &connector->router; 887 router->objectID = routerObjectID; 888 for (enumId = 0; enumId < (*destObjCount); enumId++) { 889 if (B_LENDIAN_TO_HOST_INT16(path->usConnObjectId) 890 == B_LENDIAN_TO_HOST_INT16(dstObjs[enumId])) 891 break; 892 } 893 while (record->ucRecordSize > 0 && 894 record->ucRecordType > 0 && 895 record->ucRecordType <= ATOM_MAX_OBJECT_RECORD_NUMBER) { 896 switch (record->ucRecordType) { 897 case ATOM_I2C_RECORD_TYPE: 898 i2cRecord = (ATOM_I2C_RECORD*)record; 899 i2cConfig 900 = (ATOM_I2C_ID_CONFIG_ACCESS*)&i2cRecord->sucI2cId; 901 connector_attach_gpio_router(connectorIndex, 902 i2cConfig->ucAccess); 903 router->i2cAddr = i2cRecord->ucI2CAddr >> 1; // ?? 904 break; 905 case ATOM_ROUTER_DDC_PATH_SELECT_RECORD_TYPE: 906 ddcPath = (ATOM_ROUTER_DDC_PATH_SELECT_RECORD*)record; 907 router->ddcValid = true; 908 router->ddcMuxType = ddcPath->ucMuxType; 909 router->ddcMuxControlPin = ddcPath->ucMuxControlPin; 910 router->ddcMuxState = ddcPath->ucMuxState[enumId]; 911 break; 912 case ATOM_ROUTER_DATA_CLOCK_PATH_SELECT_RECORD_TYPE: 913 cdPath 914 = (ATOM_ROUTER_DATA_CLOCK_PATH_SELECT_RECORD *)record; 915 router->cdValid = true; 916 router->cdMuxType = cdPath->ucMuxType; 917 router->cdMuxControlPin = cdPath->ucMuxControlPin; 918 router->cdMuxState = cdPath->ucMuxState[enumId]; 919 break; 920 } 921 922 // move to next record 923 record = (ATOM_COMMON_RECORD_HEADER*) 924 ((char *)record + record->ucRecordSize); 925 } 926 } 927 } 928 } // END if object is router 929 } 930 931 // Set up information buses such as ddc 932 if (((connectorFlags & ATOM_DEVICE_TV_SUPPORT) == 0) 933 && (connectorFlags & ATOM_DEVICE_CV_SUPPORT) == 0) { 934 for (j = 0; j < connectorObject->ucNumberOfObjects; j++) { 935 if (B_LENDIAN_TO_HOST_INT16(path->usConnObjectId) 936 == B_LENDIAN_TO_HOST_INT16( 937 connectorObject->asObjects[j].usObjectID)) { 938 ATOM_COMMON_RECORD_HEADER* record 939 = (ATOM_COMMON_RECORD_HEADER*)(gAtomContext->bios 940 + tableOffset + B_LENDIAN_TO_HOST_INT16( 941 connectorObject->asObjects[j].usRecordOffset)); 942 while (record->ucRecordSize > 0 943 && record->ucRecordType > 0 944 && record->ucRecordType 945 <= ATOM_MAX_OBJECT_RECORD_NUMBER) { 946 ATOM_I2C_RECORD* i2cRecord; 947 ATOM_I2C_ID_CONFIG_ACCESS* i2cConfig; 948 ATOM_HPD_INT_RECORD* hpdRecord; 949 950 switch (record->ucRecordType) { 951 case ATOM_I2C_RECORD_TYPE: 952 i2cRecord 953 = (ATOM_I2C_RECORD*)record; 954 i2cConfig 955 = (ATOM_I2C_ID_CONFIG_ACCESS*) 956 &i2cRecord->sucI2cId; 957 connector_attach_gpio_i2c(connectorIndex, 958 i2cConfig->ucAccess); 959 break; 960 case ATOM_HPD_INT_RECORD_TYPE: 961 hpdRecord = (ATOM_HPD_INT_RECORD*)record; 962 connector_attach_gpio_hpd(connectorIndex, 963 hpdRecord->ucHPDIntGPIOID); 964 break; 965 } 966 967 // move to next record 968 record = (ATOM_COMMON_RECORD_HEADER*) 969 ((char*)record + record->ucRecordSize); 970 } 971 } 972 } 973 } 974 975 connector->valid = true; 976 connector->flags = connectorFlags; 977 connector->type = connectorType; 978 connector->objectID = connectorObjectID; 979 980 connectorIndex++; 981 } // END for each valid connector 982 } // end for each display path 983 984 return B_OK; 985 } 986 987 988 bool 989 connector_is_dp(uint32 connectorIndex) 990 { 991 connector_info* connector = gConnector[connectorIndex]; 992 993 // Traditional DisplayPort connector, or DP over USBC 994 if (connector->type == VIDEO_CONNECTOR_DP 995 || connector->type == VIDEO_CONNECTOR_EDP 996 || connector->type == VIDEO_CONNECTOR_USBC) { 997 return true; 998 } 999 1000 // DisplayPort bridge on external encoder 1001 if (connector->encoderExternal.valid == true 1002 && connector->encoderExternal.isDPBridge == true) { 1003 return true; 1004 } 1005 1006 return false; 1007 } 1008 1009 1010 void 1011 debug_connectors() 1012 { 1013 ERROR("Currently detected connectors=============\n"); 1014 for (uint32 id = 0; id < ATOM_MAX_SUPPORTED_DEVICE; id++) { 1015 if (gConnector[id]->valid == true) { 1016 uint32 connectorType = gConnector[id]->type; 1017 uint16 i2cPinIndex = gConnector[id]->i2cPinIndex; 1018 uint16 hpdPinIndex = gConnector[id]->hpdPinIndex; 1019 1020 ERROR("Connector #%" B_PRIu32 ")\n", id); 1021 ERROR(" + connector: %s\n", 1022 get_connector_name(connectorType)); 1023 ERROR(" + i2c gpio table id: %" B_PRIu16 "\n", i2cPinIndex); 1024 ERROR(" - gpio hw pin: 0x%" B_PRIX32 "\n", 1025 gGPIOInfo[i2cPinIndex]->hwPin); 1026 ERROR(" - gpio valid: %s\n", 1027 gGPIOInfo[i2cPinIndex]->valid ? "true" : "false"); 1028 ERROR(" - i2c valid: %s\n", 1029 gGPIOInfo[i2cPinIndex]->i2c.valid ? "true" : "false"); 1030 1031 // hot plug detection info 1032 ERROR(" + hpd gpio table id: %" B_PRIu16 "\n", hpdPinIndex); 1033 ERROR(" - gpio hw pin: 0x%" B_PRIX32 "\n", 1034 gGPIOInfo[hpdPinIndex]->hwPin); 1035 ERROR(" - gpio valid: %s\n", 1036 gGPIOInfo[hpdPinIndex]->valid ? "true" : "false"); 1037 1038 // router info 1039 router_info* router = &gConnector[id]->router; 1040 ERROR(" + router gpio table id: %" B_PRIu16 "\n", router->i2cPinIndex); 1041 ERROR(" + router (ddc): %s\n", 1042 router->ddcValid ? "true" : "false"); 1043 if (router->ddcValid) { 1044 ERROR(" - mux type: 0x%" B_PRIX8 "\n", router->ddcMuxType); 1045 ERROR(" - mux pin: 0x%" B_PRIX8 "\n", router->ddcMuxControlPin); 1046 ERROR(" - mux state: 0x%" B_PRIX8 "\n", router->ddcMuxState); 1047 } 1048 ERROR(" + router (c/d): %s\n", 1049 router->cdValid ? "true" : "false"); 1050 if (router->cdValid) { 1051 ERROR(" - mux type: 0x%" B_PRIX8 "\n", router->cdMuxType); 1052 ERROR(" - mux pin: 0x%" B_PRIX8 "\n", router->cdMuxControlPin); 1053 ERROR(" - mux state: 0x%" B_PRIX8 "\n", router->cdMuxState); 1054 } 1055 1056 // encoder info 1057 encoder_info* encoder = &gConnector[id]->encoder; 1058 ERROR(" + encoder: %s\n", 1059 get_encoder_name(encoder->type)); 1060 ERROR(" - id: %" B_PRIu16 "\n", encoder->objectID); 1061 ERROR(" - type: %s\n", 1062 encoder_name_lookup(encoder->objectID)); 1063 ERROR(" - capabilities: 0x%" B_PRIX32 "\n", 1064 encoder->capabilities); 1065 ERROR(" - enumeration: %" B_PRIu32 "\n", 1066 encoder->linkEnumeration); 1067 1068 encoder = &gConnector[id]->encoderExternal; 1069 1070 ERROR(" - is bridge: %s\n", 1071 encoder->valid ? "true" : "false"); 1072 1073 if (!encoder->valid) 1074 ERROR(" + external encoder: none\n"); 1075 else { 1076 ERROR(" + external encoder: %s\n", 1077 get_encoder_name(encoder->type)); 1078 ERROR(" - valid: true\n"); 1079 ERROR(" - id: %" B_PRIu16 "\n", 1080 encoder->objectID); 1081 ERROR(" - type: %s\n", 1082 encoder_name_lookup(encoder->objectID)); 1083 ERROR(" - enumeration: %" B_PRIu32 "\n", 1084 encoder->linkEnumeration); 1085 } 1086 1087 uint32 connectorFlags = gConnector[id]->flags; 1088 bool flags = false; 1089 ERROR(" + flags:\n"); 1090 if ((connectorFlags & ATOM_DEVICE_CRT1_SUPPORT) != 0) { 1091 ERROR(" * device CRT1 support\n"); 1092 flags = true; 1093 } 1094 if ((connectorFlags & ATOM_DEVICE_CRT2_SUPPORT) != 0) { 1095 ERROR(" * device CRT2 support\n"); 1096 flags = true; 1097 } 1098 if ((connectorFlags & ATOM_DEVICE_LCD1_SUPPORT) != 0) { 1099 ERROR(" * device LCD1 support\n"); 1100 flags = true; 1101 } 1102 if ((connectorFlags & ATOM_DEVICE_LCD2_SUPPORT) != 0) { 1103 ERROR(" * device LCD2 support\n"); 1104 flags = true; 1105 } 1106 if ((connectorFlags & ATOM_DEVICE_TV1_SUPPORT) != 0) { 1107 ERROR(" * device TV1 support\n"); 1108 flags = true; 1109 } 1110 if ((connectorFlags & ATOM_DEVICE_CV_SUPPORT) != 0) { 1111 ERROR(" * device CV support\n"); 1112 flags = true; 1113 } 1114 if ((connectorFlags & ATOM_DEVICE_DFP1_SUPPORT) != 0) { 1115 ERROR(" * device DFP1 support\n"); 1116 flags = true; 1117 } 1118 if ((connectorFlags & ATOM_DEVICE_DFP2_SUPPORT) != 0) { 1119 ERROR(" * device DFP2 support\n"); 1120 flags = true; 1121 } 1122 if ((connectorFlags & ATOM_DEVICE_DFP3_SUPPORT) != 0) { 1123 ERROR(" * device DFP3 support\n"); 1124 flags = true; 1125 } 1126 if ((connectorFlags & ATOM_DEVICE_DFP4_SUPPORT) != 0) { 1127 ERROR(" * device DFP4 support\n"); 1128 flags = true; 1129 } 1130 if ((connectorFlags & ATOM_DEVICE_DFP5_SUPPORT) != 0) { 1131 ERROR(" * device DFP5 support\n"); 1132 flags = true; 1133 } 1134 if ((connectorFlags & ATOM_DEVICE_DFP6_SUPPORT) != 0) { 1135 ERROR(" * device DFP6 support\n"); 1136 flags = true; 1137 } 1138 if (flags == false) 1139 ERROR(" * no known flags\n"); 1140 } 1141 } 1142 ERROR("==========================================\n"); 1143 } 1144