1 /* 2 * Copyright 2012-15 Advanced Micro Devices, Inc. 3 * 4 * Permission is hereby granted, free of charge, to any person obtaining a 5 * copy of this software and associated documentation files (the "Software"), 6 * to deal in the Software without restriction, including without limitation 7 * the rights to use, copy, modify, merge, publish, distribute, sublicense, 8 * and/or sell copies of the Software, and to permit persons to whom the 9 * Software is furnished to do so, subject to the following conditions: 10 * 11 * The above copyright notice and this permission notice shall be included in 12 * all copies or substantial portions of the Software. 13 * 14 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 15 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 16 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL 17 * THE COPYRIGHT HOLDER(S) OR AUTHOR(S) BE LIABLE FOR ANY CLAIM, DAMAGES OR 18 * OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, 19 * ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR 20 * OTHER DEALINGS IN THE SOFTWARE. 21 * 22 * Authors: AMD 23 * 24 */ 25 26 #include "dm_services.h" 27 28 #include "atom.h" 29 30 #include "dc_bios_types.h" 31 #include "include/gpio_service_interface.h" 32 #include "include/grph_object_ctrl_defs.h" 33 #include "include/bios_parser_interface.h" 34 #include "include/i2caux_interface.h" 35 #include "include/logger_interface.h" 36 37 #include "command_table.h" 38 #include "bios_parser_helper.h" 39 #include "command_table_helper.h" 40 #include "bios_parser.h" 41 #include "bios_parser_types_internal.h" 42 #include "bios_parser_interface.h" 43 44 #include "bios_parser_common.h" 45 /* TODO remove - only needed for default i2c speed */ 46 #include "dc.h" 47 48 #define THREE_PERCENT_OF_10000 300 49 50 #define LAST_RECORD_TYPE 0xff 51 52 #define DC_LOGGER \ 53 bp->base.ctx->logger 54 55 /* GUID to validate external display connection info table (aka OPM module) */ 56 static const uint8_t ext_display_connection_guid[NUMBER_OF_UCHAR_FOR_GUID] = { 57 0x91, 0x6E, 0x57, 0x09, 58 0x3F, 0x6D, 0xD2, 0x11, 59 0x39, 0x8E, 0x00, 0xA0, 60 0xC9, 0x69, 0x72, 0x3B}; 61 62 #define DATA_TABLES(table) (bp->master_data_tbl->ListOfDataTables.table) 63 64 static void get_atom_data_table_revision( 65 ATOM_COMMON_TABLE_HEADER *atom_data_tbl, 66 struct atom_data_revision *tbl_revision); 67 static uint32_t get_dst_number_from_object(struct bios_parser *bp, 68 ATOM_OBJECT *object); 69 static uint32_t get_src_obj_list(struct bios_parser *bp, ATOM_OBJECT *object, 70 uint16_t **id_list); 71 static uint32_t get_dest_obj_list(struct bios_parser *bp, 72 ATOM_OBJECT *object, uint16_t **id_list); 73 static ATOM_OBJECT *get_bios_object(struct bios_parser *bp, 74 struct graphics_object_id id); 75 static enum bp_result get_gpio_i2c_info(struct bios_parser *bp, 76 ATOM_I2C_RECORD *record, 77 struct graphics_object_i2c_info *info); 78 static ATOM_HPD_INT_RECORD *get_hpd_record(struct bios_parser *bp, 79 ATOM_OBJECT *object); 80 static struct device_id device_type_from_device_id(uint16_t device_id); 81 static uint32_t signal_to_ss_id(enum as_signal_type signal); 82 static uint32_t get_support_mask_for_device_id(struct device_id device_id); 83 static ATOM_ENCODER_CAP_RECORD_V2 *get_encoder_cap_record( 84 struct bios_parser *bp, 85 ATOM_OBJECT *object); 86 87 #define BIOS_IMAGE_SIZE_OFFSET 2 88 #define BIOS_IMAGE_SIZE_UNIT 512 89 90 /*****************************************************************************/ 91 static bool bios_parser_construct( 92 struct bios_parser *bp, 93 struct bp_init_data *init, 94 enum dce_version dce_version); 95 96 static uint8_t bios_parser_get_connectors_number( 97 struct dc_bios *dcb); 98 99 static enum bp_result bios_parser_get_embedded_panel_info( 100 struct dc_bios *dcb, 101 struct embedded_panel_info *info); 102 103 /*****************************************************************************/ 104 105 struct dc_bios *bios_parser_create( 106 struct bp_init_data *init, 107 enum dce_version dce_version) 108 { 109 struct bios_parser *bp = NULL; 110 111 bp = kzalloc(sizeof(struct bios_parser), GFP_KERNEL); 112 if (!bp) 113 return NULL; 114 115 if (bios_parser_construct(bp, init, dce_version)) 116 return &bp->base; 117 118 kfree(bp); 119 BREAK_TO_DEBUGGER(); 120 return NULL; 121 } 122 123 static void destruct(struct bios_parser *bp) 124 { 125 kfree(bp->base.bios_local_image); 126 kfree(bp->base.integrated_info); 127 } 128 129 static void bios_parser_destroy(struct dc_bios **dcb) 130 { 131 struct bios_parser *bp = BP_FROM_DCB(*dcb); 132 133 if (!bp) { 134 BREAK_TO_DEBUGGER(); 135 return; 136 } 137 138 destruct(bp); 139 140 kfree(bp); 141 *dcb = NULL; 142 } 143 144 static uint8_t get_number_of_objects(struct bios_parser *bp, uint32_t offset) 145 { 146 ATOM_OBJECT_TABLE *table; 147 148 uint32_t object_table_offset = bp->object_info_tbl_offset + offset; 149 150 table = GET_IMAGE(ATOM_OBJECT_TABLE, object_table_offset); 151 152 if (!table) 153 return 0; 154 else 155 return table->ucNumberOfObjects; 156 } 157 158 static uint8_t bios_parser_get_connectors_number(struct dc_bios *dcb) 159 { 160 struct bios_parser *bp = BP_FROM_DCB(dcb); 161 162 return get_number_of_objects(bp, 163 le16_to_cpu(bp->object_info_tbl.v1_1->usConnectorObjectTableOffset)); 164 } 165 166 static struct graphics_object_id bios_parser_get_encoder_id( 167 struct dc_bios *dcb, 168 uint32_t i) 169 { 170 struct bios_parser *bp = BP_FROM_DCB(dcb); 171 struct graphics_object_id object_id = dal_graphics_object_id_init( 172 0, ENUM_ID_UNKNOWN, OBJECT_TYPE_UNKNOWN); 173 174 uint32_t encoder_table_offset = bp->object_info_tbl_offset 175 + le16_to_cpu(bp->object_info_tbl.v1_1->usEncoderObjectTableOffset); 176 177 ATOM_OBJECT_TABLE *tbl = 178 GET_IMAGE(ATOM_OBJECT_TABLE, encoder_table_offset); 179 180 if (tbl && tbl->ucNumberOfObjects > i) { 181 const uint16_t id = le16_to_cpu(tbl->asObjects[i].usObjectID); 182 183 object_id = object_id_from_bios_object_id(id); 184 } 185 186 return object_id; 187 } 188 189 static struct graphics_object_id bios_parser_get_connector_id( 190 struct dc_bios *dcb, 191 uint8_t i) 192 { 193 struct bios_parser *bp = BP_FROM_DCB(dcb); 194 struct graphics_object_id object_id = dal_graphics_object_id_init( 195 0, ENUM_ID_UNKNOWN, OBJECT_TYPE_UNKNOWN); 196 uint16_t id; 197 198 uint32_t connector_table_offset = bp->object_info_tbl_offset 199 + le16_to_cpu(bp->object_info_tbl.v1_1->usConnectorObjectTableOffset); 200 201 ATOM_OBJECT_TABLE *tbl = 202 GET_IMAGE(ATOM_OBJECT_TABLE, connector_table_offset); 203 204 if (!tbl) { 205 dm_error("Can't get connector table from atom bios.\n"); 206 return object_id; 207 } 208 209 if (tbl->ucNumberOfObjects <= i) { 210 dm_error("Can't find connector id %d in connector table of size %d.\n", 211 i, tbl->ucNumberOfObjects); 212 return object_id; 213 } 214 215 id = le16_to_cpu(tbl->asObjects[i].usObjectID); 216 object_id = object_id_from_bios_object_id(id); 217 return object_id; 218 } 219 220 static uint32_t bios_parser_get_dst_number(struct dc_bios *dcb, 221 struct graphics_object_id id) 222 { 223 struct bios_parser *bp = BP_FROM_DCB(dcb); 224 ATOM_OBJECT *object = get_bios_object(bp, id); 225 226 return get_dst_number_from_object(bp, object); 227 } 228 229 static enum bp_result bios_parser_get_src_obj(struct dc_bios *dcb, 230 struct graphics_object_id object_id, uint32_t index, 231 struct graphics_object_id *src_object_id) 232 { 233 uint32_t number; 234 uint16_t *id; 235 ATOM_OBJECT *object; 236 struct bios_parser *bp = BP_FROM_DCB(dcb); 237 238 if (!src_object_id) 239 return BP_RESULT_BADINPUT; 240 241 object = get_bios_object(bp, object_id); 242 243 if (!object) { 244 BREAK_TO_DEBUGGER(); /* Invalid object id */ 245 return BP_RESULT_BADINPUT; 246 } 247 248 number = get_src_obj_list(bp, object, &id); 249 250 if (number <= index) 251 return BP_RESULT_BADINPUT; 252 253 *src_object_id = object_id_from_bios_object_id(id[index]); 254 255 return BP_RESULT_OK; 256 } 257 258 static enum bp_result bios_parser_get_dst_obj(struct dc_bios *dcb, 259 struct graphics_object_id object_id, uint32_t index, 260 struct graphics_object_id *dest_object_id) 261 { 262 uint32_t number; 263 uint16_t *id = NULL; 264 ATOM_OBJECT *object; 265 struct bios_parser *bp = BP_FROM_DCB(dcb); 266 267 if (!dest_object_id) 268 return BP_RESULT_BADINPUT; 269 270 object = get_bios_object(bp, object_id); 271 272 number = get_dest_obj_list(bp, object, &id); 273 274 if (number <= index || !id) 275 return BP_RESULT_BADINPUT; 276 277 *dest_object_id = object_id_from_bios_object_id(id[index]); 278 279 return BP_RESULT_OK; 280 } 281 282 static enum bp_result bios_parser_get_i2c_info(struct dc_bios *dcb, 283 struct graphics_object_id id, 284 struct graphics_object_i2c_info *info) 285 { 286 uint32_t offset; 287 ATOM_OBJECT *object; 288 ATOM_COMMON_RECORD_HEADER *header; 289 ATOM_I2C_RECORD *record; 290 struct bios_parser *bp = BP_FROM_DCB(dcb); 291 292 if (!info) 293 return BP_RESULT_BADINPUT; 294 295 object = get_bios_object(bp, id); 296 297 if (!object) 298 return BP_RESULT_BADINPUT; 299 300 offset = le16_to_cpu(object->usRecordOffset) 301 + bp->object_info_tbl_offset; 302 303 for (;;) { 304 header = GET_IMAGE(ATOM_COMMON_RECORD_HEADER, offset); 305 306 if (!header) 307 return BP_RESULT_BADBIOSTABLE; 308 309 if (LAST_RECORD_TYPE == header->ucRecordType || 310 !header->ucRecordSize) 311 break; 312 313 if (ATOM_I2C_RECORD_TYPE == header->ucRecordType 314 && sizeof(ATOM_I2C_RECORD) <= header->ucRecordSize) { 315 /* get the I2C info */ 316 record = (ATOM_I2C_RECORD *) header; 317 318 if (get_gpio_i2c_info(bp, record, info) == BP_RESULT_OK) 319 return BP_RESULT_OK; 320 } 321 322 offset += header->ucRecordSize; 323 } 324 325 return BP_RESULT_NORECORD; 326 } 327 328 static enum bp_result get_voltage_ddc_info_v1(uint8_t *i2c_line, 329 ATOM_COMMON_TABLE_HEADER *header, 330 uint8_t *address) 331 { 332 enum bp_result result = BP_RESULT_NORECORD; 333 ATOM_VOLTAGE_OBJECT_INFO *info = 334 (ATOM_VOLTAGE_OBJECT_INFO *) address; 335 336 uint8_t *voltage_current_object = (uint8_t *) &info->asVoltageObj[0]; 337 338 while ((address + le16_to_cpu(header->usStructureSize)) > voltage_current_object) { 339 ATOM_VOLTAGE_OBJECT *object = 340 (ATOM_VOLTAGE_OBJECT *) voltage_current_object; 341 342 if ((object->ucVoltageType == SET_VOLTAGE_INIT_MODE) && 343 (object->ucVoltageType & 344 VOLTAGE_CONTROLLED_BY_I2C_MASK)) { 345 346 *i2c_line = object->asControl.ucVoltageControlI2cLine 347 ^ 0x90; 348 result = BP_RESULT_OK; 349 break; 350 } 351 352 voltage_current_object += object->ucSize; 353 } 354 return result; 355 } 356 357 static enum bp_result get_voltage_ddc_info_v3(uint8_t *i2c_line, 358 uint32_t index, 359 ATOM_COMMON_TABLE_HEADER *header, 360 uint8_t *address) 361 { 362 enum bp_result result = BP_RESULT_NORECORD; 363 ATOM_VOLTAGE_OBJECT_INFO_V3_1 *info = 364 (ATOM_VOLTAGE_OBJECT_INFO_V3_1 *) address; 365 366 uint8_t *voltage_current_object = 367 (uint8_t *) (&(info->asVoltageObj[0])); 368 369 while ((address + le16_to_cpu(header->usStructureSize)) > voltage_current_object) { 370 ATOM_I2C_VOLTAGE_OBJECT_V3 *object = 371 (ATOM_I2C_VOLTAGE_OBJECT_V3 *) voltage_current_object; 372 373 if (object->sHeader.ucVoltageMode == 374 ATOM_INIT_VOLTAGE_REGULATOR) { 375 if (object->sHeader.ucVoltageType == index) { 376 *i2c_line = object->ucVoltageControlI2cLine 377 ^ 0x90; 378 result = BP_RESULT_OK; 379 break; 380 } 381 } 382 383 voltage_current_object += le16_to_cpu(object->sHeader.usSize); 384 } 385 return result; 386 } 387 388 static enum bp_result bios_parser_get_thermal_ddc_info( 389 struct dc_bios *dcb, 390 uint32_t i2c_channel_id, 391 struct graphics_object_i2c_info *info) 392 { 393 struct bios_parser *bp = BP_FROM_DCB(dcb); 394 ATOM_I2C_ID_CONFIG_ACCESS *config; 395 ATOM_I2C_RECORD record; 396 397 if (!info) 398 return BP_RESULT_BADINPUT; 399 400 config = (ATOM_I2C_ID_CONFIG_ACCESS *) &i2c_channel_id; 401 402 record.sucI2cId.bfHW_Capable = config->sbfAccess.bfHW_Capable; 403 record.sucI2cId.bfI2C_LineMux = config->sbfAccess.bfI2C_LineMux; 404 record.sucI2cId.bfHW_EngineID = config->sbfAccess.bfHW_EngineID; 405 406 return get_gpio_i2c_info(bp, &record, info); 407 } 408 409 static enum bp_result bios_parser_get_voltage_ddc_info(struct dc_bios *dcb, 410 uint32_t index, 411 struct graphics_object_i2c_info *info) 412 { 413 uint8_t i2c_line = 0; 414 enum bp_result result = BP_RESULT_NORECORD; 415 uint8_t *voltage_info_address; 416 ATOM_COMMON_TABLE_HEADER *header; 417 struct atom_data_revision revision = {0}; 418 struct bios_parser *bp = BP_FROM_DCB(dcb); 419 420 if (!DATA_TABLES(VoltageObjectInfo)) 421 return result; 422 423 voltage_info_address = bios_get_image(&bp->base, DATA_TABLES(VoltageObjectInfo), sizeof(ATOM_COMMON_TABLE_HEADER)); 424 425 header = (ATOM_COMMON_TABLE_HEADER *) voltage_info_address; 426 427 get_atom_data_table_revision(header, &revision); 428 429 switch (revision.major) { 430 case 1: 431 case 2: 432 result = get_voltage_ddc_info_v1(&i2c_line, header, 433 voltage_info_address); 434 break; 435 case 3: 436 if (revision.minor != 1) 437 break; 438 result = get_voltage_ddc_info_v3(&i2c_line, index, header, 439 voltage_info_address); 440 break; 441 } 442 443 if (result == BP_RESULT_OK) 444 result = bios_parser_get_thermal_ddc_info(dcb, 445 i2c_line, info); 446 447 return result; 448 } 449 450 /* TODO: temporary commented out to suppress 'defined but not used' warning */ 451 #if 0 452 static enum bp_result bios_parser_get_ddc_info_for_i2c_line( 453 struct bios_parser *bp, 454 uint8_t i2c_line, struct graphics_object_i2c_info *info) 455 { 456 uint32_t offset; 457 ATOM_OBJECT *object; 458 ATOM_OBJECT_TABLE *table; 459 uint32_t i; 460 461 if (!info) 462 return BP_RESULT_BADINPUT; 463 464 offset = le16_to_cpu(bp->object_info_tbl.v1_1->usConnectorObjectTableOffset); 465 466 offset += bp->object_info_tbl_offset; 467 468 table = GET_IMAGE(ATOM_OBJECT_TABLE, offset); 469 470 if (!table) 471 return BP_RESULT_BADBIOSTABLE; 472 473 for (i = 0; i < table->ucNumberOfObjects; i++) { 474 object = &table->asObjects[i]; 475 476 if (!object) { 477 BREAK_TO_DEBUGGER(); /* Invalid object id */ 478 return BP_RESULT_BADINPUT; 479 } 480 481 offset = le16_to_cpu(object->usRecordOffset) 482 + bp->object_info_tbl_offset; 483 484 for (;;) { 485 ATOM_COMMON_RECORD_HEADER *header = 486 GET_IMAGE(ATOM_COMMON_RECORD_HEADER, offset); 487 488 if (!header) 489 return BP_RESULT_BADBIOSTABLE; 490 491 offset += header->ucRecordSize; 492 493 if (LAST_RECORD_TYPE == header->ucRecordType || 494 !header->ucRecordSize) 495 break; 496 497 if (ATOM_I2C_RECORD_TYPE == header->ucRecordType 498 && sizeof(ATOM_I2C_RECORD) <= 499 header->ucRecordSize) { 500 ATOM_I2C_RECORD *record = 501 (ATOM_I2C_RECORD *) header; 502 503 if (i2c_line != record->sucI2cId.bfI2C_LineMux) 504 continue; 505 506 /* get the I2C info */ 507 if (get_gpio_i2c_info(bp, record, info) == 508 BP_RESULT_OK) 509 return BP_RESULT_OK; 510 } 511 } 512 } 513 514 return BP_RESULT_NORECORD; 515 } 516 #endif 517 518 static enum bp_result bios_parser_get_hpd_info(struct dc_bios *dcb, 519 struct graphics_object_id id, 520 struct graphics_object_hpd_info *info) 521 { 522 struct bios_parser *bp = BP_FROM_DCB(dcb); 523 ATOM_OBJECT *object; 524 ATOM_HPD_INT_RECORD *record = NULL; 525 526 if (!info) 527 return BP_RESULT_BADINPUT; 528 529 object = get_bios_object(bp, id); 530 531 if (!object) 532 return BP_RESULT_BADINPUT; 533 534 record = get_hpd_record(bp, object); 535 536 if (record != NULL) { 537 info->hpd_int_gpio_uid = record->ucHPDIntGPIOID; 538 info->hpd_active = record->ucPlugged_PinState; 539 return BP_RESULT_OK; 540 } 541 542 return BP_RESULT_NORECORD; 543 } 544 545 static enum bp_result bios_parser_get_device_tag_record( 546 struct bios_parser *bp, 547 ATOM_OBJECT *object, 548 ATOM_CONNECTOR_DEVICE_TAG_RECORD **record) 549 { 550 ATOM_COMMON_RECORD_HEADER *header; 551 uint32_t offset; 552 553 offset = le16_to_cpu(object->usRecordOffset) 554 + bp->object_info_tbl_offset; 555 556 for (;;) { 557 header = GET_IMAGE(ATOM_COMMON_RECORD_HEADER, offset); 558 559 if (!header) 560 return BP_RESULT_BADBIOSTABLE; 561 562 offset += header->ucRecordSize; 563 564 if (LAST_RECORD_TYPE == header->ucRecordType || 565 !header->ucRecordSize) 566 break; 567 568 if (ATOM_CONNECTOR_DEVICE_TAG_RECORD_TYPE != 569 header->ucRecordType) 570 continue; 571 572 if (sizeof(ATOM_CONNECTOR_DEVICE_TAG) > header->ucRecordSize) 573 continue; 574 575 *record = (ATOM_CONNECTOR_DEVICE_TAG_RECORD *) header; 576 return BP_RESULT_OK; 577 } 578 579 return BP_RESULT_NORECORD; 580 } 581 582 static enum bp_result bios_parser_get_device_tag( 583 struct dc_bios *dcb, 584 struct graphics_object_id connector_object_id, 585 uint32_t device_tag_index, 586 struct connector_device_tag_info *info) 587 { 588 struct bios_parser *bp = BP_FROM_DCB(dcb); 589 ATOM_OBJECT *object; 590 ATOM_CONNECTOR_DEVICE_TAG_RECORD *record = NULL; 591 ATOM_CONNECTOR_DEVICE_TAG *device_tag; 592 593 if (!info) 594 return BP_RESULT_BADINPUT; 595 596 /* getBiosObject will return MXM object */ 597 object = get_bios_object(bp, connector_object_id); 598 599 if (!object) { 600 BREAK_TO_DEBUGGER(); /* Invalid object id */ 601 return BP_RESULT_BADINPUT; 602 } 603 604 if (bios_parser_get_device_tag_record(bp, object, &record) 605 != BP_RESULT_OK) 606 return BP_RESULT_NORECORD; 607 608 if (device_tag_index >= record->ucNumberOfDevice) 609 return BP_RESULT_NORECORD; 610 611 device_tag = &record->asDeviceTag[device_tag_index]; 612 613 info->acpi_device = le32_to_cpu(device_tag->ulACPIDeviceEnum); 614 info->dev_id = 615 device_type_from_device_id(le16_to_cpu(device_tag->usDeviceID)); 616 617 return BP_RESULT_OK; 618 } 619 620 static enum bp_result get_firmware_info_v1_4( 621 struct bios_parser *bp, 622 struct dc_firmware_info *info); 623 static enum bp_result get_firmware_info_v2_1( 624 struct bios_parser *bp, 625 struct dc_firmware_info *info); 626 static enum bp_result get_firmware_info_v2_2( 627 struct bios_parser *bp, 628 struct dc_firmware_info *info); 629 630 static enum bp_result bios_parser_get_firmware_info( 631 struct dc_bios *dcb, 632 struct dc_firmware_info *info) 633 { 634 struct bios_parser *bp = BP_FROM_DCB(dcb); 635 enum bp_result result = BP_RESULT_BADBIOSTABLE; 636 ATOM_COMMON_TABLE_HEADER *header; 637 struct atom_data_revision revision; 638 639 if (info && DATA_TABLES(FirmwareInfo)) { 640 header = GET_IMAGE(ATOM_COMMON_TABLE_HEADER, 641 DATA_TABLES(FirmwareInfo)); 642 get_atom_data_table_revision(header, &revision); 643 switch (revision.major) { 644 case 1: 645 switch (revision.minor) { 646 case 4: 647 result = get_firmware_info_v1_4(bp, info); 648 break; 649 default: 650 break; 651 } 652 break; 653 654 case 2: 655 switch (revision.minor) { 656 case 1: 657 result = get_firmware_info_v2_1(bp, info); 658 break; 659 case 2: 660 result = get_firmware_info_v2_2(bp, info); 661 break; 662 default: 663 break; 664 } 665 break; 666 default: 667 break; 668 } 669 } 670 671 return result; 672 } 673 674 static enum bp_result get_firmware_info_v1_4( 675 struct bios_parser *bp, 676 struct dc_firmware_info *info) 677 { 678 ATOM_FIRMWARE_INFO_V1_4 *firmware_info = 679 GET_IMAGE(ATOM_FIRMWARE_INFO_V1_4, 680 DATA_TABLES(FirmwareInfo)); 681 682 if (!info) 683 return BP_RESULT_BADINPUT; 684 685 if (!firmware_info) 686 return BP_RESULT_BADBIOSTABLE; 687 688 memset(info, 0, sizeof(*info)); 689 690 /* Pixel clock pll information. We need to convert from 10KHz units into 691 * KHz units */ 692 info->pll_info.crystal_frequency = 693 le16_to_cpu(firmware_info->usReferenceClock) * 10; 694 info->pll_info.min_input_pxl_clk_pll_frequency = 695 le16_to_cpu(firmware_info->usMinPixelClockPLL_Input) * 10; 696 info->pll_info.max_input_pxl_clk_pll_frequency = 697 le16_to_cpu(firmware_info->usMaxPixelClockPLL_Input) * 10; 698 info->pll_info.min_output_pxl_clk_pll_frequency = 699 le32_to_cpu(firmware_info->ulMinPixelClockPLL_Output) * 10; 700 info->pll_info.max_output_pxl_clk_pll_frequency = 701 le32_to_cpu(firmware_info->ulMaxPixelClockPLL_Output) * 10; 702 703 if (firmware_info->usFirmwareCapability.sbfAccess.MemoryClockSS_Support) 704 /* Since there is no information on the SS, report conservative 705 * value 3% for bandwidth calculation */ 706 /* unit of 0.01% */ 707 info->feature.memory_clk_ss_percentage = THREE_PERCENT_OF_10000; 708 709 if (firmware_info->usFirmwareCapability.sbfAccess.EngineClockSS_Support) 710 /* Since there is no information on the SS,report conservative 711 * value 3% for bandwidth calculation */ 712 /* unit of 0.01% */ 713 info->feature.engine_clk_ss_percentage = THREE_PERCENT_OF_10000; 714 715 return BP_RESULT_OK; 716 } 717 718 static enum bp_result get_ss_info_v3_1( 719 struct bios_parser *bp, 720 uint32_t id, 721 uint32_t index, 722 struct spread_spectrum_info *ss_info); 723 724 static enum bp_result get_firmware_info_v2_1( 725 struct bios_parser *bp, 726 struct dc_firmware_info *info) 727 { 728 ATOM_FIRMWARE_INFO_V2_1 *firmwareInfo = 729 GET_IMAGE(ATOM_FIRMWARE_INFO_V2_1, DATA_TABLES(FirmwareInfo)); 730 struct spread_spectrum_info internalSS; 731 uint32_t index; 732 733 if (!info) 734 return BP_RESULT_BADINPUT; 735 736 if (!firmwareInfo) 737 return BP_RESULT_BADBIOSTABLE; 738 739 memset(info, 0, sizeof(*info)); 740 741 /* Pixel clock pll information. We need to convert from 10KHz units into 742 * KHz units */ 743 info->pll_info.crystal_frequency = 744 le16_to_cpu(firmwareInfo->usCoreReferenceClock) * 10; 745 info->pll_info.min_input_pxl_clk_pll_frequency = 746 le16_to_cpu(firmwareInfo->usMinPixelClockPLL_Input) * 10; 747 info->pll_info.max_input_pxl_clk_pll_frequency = 748 le16_to_cpu(firmwareInfo->usMaxPixelClockPLL_Input) * 10; 749 info->pll_info.min_output_pxl_clk_pll_frequency = 750 le32_to_cpu(firmwareInfo->ulMinPixelClockPLL_Output) * 10; 751 info->pll_info.max_output_pxl_clk_pll_frequency = 752 le32_to_cpu(firmwareInfo->ulMaxPixelClockPLL_Output) * 10; 753 info->default_display_engine_pll_frequency = 754 le32_to_cpu(firmwareInfo->ulDefaultDispEngineClkFreq) * 10; 755 info->external_clock_source_frequency_for_dp = 756 le16_to_cpu(firmwareInfo->usUniphyDPModeExtClkFreq) * 10; 757 info->min_allowed_bl_level = firmwareInfo->ucMinAllowedBL_Level; 758 759 /* There should be only one entry in the SS info table for Memory Clock 760 */ 761 index = 0; 762 if (firmwareInfo->usFirmwareCapability.sbfAccess.MemoryClockSS_Support) 763 /* Since there is no information for external SS, report 764 * conservative value 3% for bandwidth calculation */ 765 /* unit of 0.01% */ 766 info->feature.memory_clk_ss_percentage = THREE_PERCENT_OF_10000; 767 else if (get_ss_info_v3_1(bp, 768 ASIC_INTERNAL_MEMORY_SS, index, &internalSS) == BP_RESULT_OK) { 769 if (internalSS.spread_spectrum_percentage) { 770 info->feature.memory_clk_ss_percentage = 771 internalSS.spread_spectrum_percentage; 772 if (internalSS.type.CENTER_MODE) { 773 /* if it is centermode, the exact SS Percentage 774 * will be round up of half of the percentage 775 * reported in the SS table */ 776 ++info->feature.memory_clk_ss_percentage; 777 info->feature.memory_clk_ss_percentage /= 2; 778 } 779 } 780 } 781 782 /* There should be only one entry in the SS info table for Engine Clock 783 */ 784 index = 1; 785 if (firmwareInfo->usFirmwareCapability.sbfAccess.EngineClockSS_Support) 786 /* Since there is no information for external SS, report 787 * conservative value 3% for bandwidth calculation */ 788 /* unit of 0.01% */ 789 info->feature.engine_clk_ss_percentage = THREE_PERCENT_OF_10000; 790 else if (get_ss_info_v3_1(bp, 791 ASIC_INTERNAL_ENGINE_SS, index, &internalSS) == BP_RESULT_OK) { 792 if (internalSS.spread_spectrum_percentage) { 793 info->feature.engine_clk_ss_percentage = 794 internalSS.spread_spectrum_percentage; 795 if (internalSS.type.CENTER_MODE) { 796 /* if it is centermode, the exact SS Percentage 797 * will be round up of half of the percentage 798 * reported in the SS table */ 799 ++info->feature.engine_clk_ss_percentage; 800 info->feature.engine_clk_ss_percentage /= 2; 801 } 802 } 803 } 804 805 return BP_RESULT_OK; 806 } 807 808 static enum bp_result get_firmware_info_v2_2( 809 struct bios_parser *bp, 810 struct dc_firmware_info *info) 811 { 812 ATOM_FIRMWARE_INFO_V2_2 *firmware_info; 813 struct spread_spectrum_info internal_ss; 814 uint32_t index; 815 816 if (!info) 817 return BP_RESULT_BADINPUT; 818 819 firmware_info = GET_IMAGE(ATOM_FIRMWARE_INFO_V2_2, 820 DATA_TABLES(FirmwareInfo)); 821 822 if (!firmware_info) 823 return BP_RESULT_BADBIOSTABLE; 824 825 memset(info, 0, sizeof(*info)); 826 827 /* Pixel clock pll information. We need to convert from 10KHz units into 828 * KHz units */ 829 info->pll_info.crystal_frequency = 830 le16_to_cpu(firmware_info->usCoreReferenceClock) * 10; 831 info->pll_info.min_input_pxl_clk_pll_frequency = 832 le16_to_cpu(firmware_info->usMinPixelClockPLL_Input) * 10; 833 info->pll_info.max_input_pxl_clk_pll_frequency = 834 le16_to_cpu(firmware_info->usMaxPixelClockPLL_Input) * 10; 835 info->pll_info.min_output_pxl_clk_pll_frequency = 836 le32_to_cpu(firmware_info->ulMinPixelClockPLL_Output) * 10; 837 info->pll_info.max_output_pxl_clk_pll_frequency = 838 le32_to_cpu(firmware_info->ulMaxPixelClockPLL_Output) * 10; 839 info->default_display_engine_pll_frequency = 840 le32_to_cpu(firmware_info->ulDefaultDispEngineClkFreq) * 10; 841 info->external_clock_source_frequency_for_dp = 842 le16_to_cpu(firmware_info->usUniphyDPModeExtClkFreq) * 10; 843 844 /* There should be only one entry in the SS info table for Memory Clock 845 */ 846 index = 0; 847 if (firmware_info->usFirmwareCapability.sbfAccess.MemoryClockSS_Support) 848 /* Since there is no information for external SS, report 849 * conservative value 3% for bandwidth calculation */ 850 /* unit of 0.01% */ 851 info->feature.memory_clk_ss_percentage = THREE_PERCENT_OF_10000; 852 else if (get_ss_info_v3_1(bp, 853 ASIC_INTERNAL_MEMORY_SS, index, &internal_ss) == BP_RESULT_OK) { 854 if (internal_ss.spread_spectrum_percentage) { 855 info->feature.memory_clk_ss_percentage = 856 internal_ss.spread_spectrum_percentage; 857 if (internal_ss.type.CENTER_MODE) { 858 /* if it is centermode, the exact SS Percentage 859 * will be round up of half of the percentage 860 * reported in the SS table */ 861 ++info->feature.memory_clk_ss_percentage; 862 info->feature.memory_clk_ss_percentage /= 2; 863 } 864 } 865 } 866 867 /* There should be only one entry in the SS info table for Engine Clock 868 */ 869 index = 1; 870 if (firmware_info->usFirmwareCapability.sbfAccess.EngineClockSS_Support) 871 /* Since there is no information for external SS, report 872 * conservative value 3% for bandwidth calculation */ 873 /* unit of 0.01% */ 874 info->feature.engine_clk_ss_percentage = THREE_PERCENT_OF_10000; 875 else if (get_ss_info_v3_1(bp, 876 ASIC_INTERNAL_ENGINE_SS, index, &internal_ss) == BP_RESULT_OK) { 877 if (internal_ss.spread_spectrum_percentage) { 878 info->feature.engine_clk_ss_percentage = 879 internal_ss.spread_spectrum_percentage; 880 if (internal_ss.type.CENTER_MODE) { 881 /* if it is centermode, the exact SS Percentage 882 * will be round up of half of the percentage 883 * reported in the SS table */ 884 ++info->feature.engine_clk_ss_percentage; 885 info->feature.engine_clk_ss_percentage /= 2; 886 } 887 } 888 } 889 890 /* Remote Display */ 891 info->remote_display_config = firmware_info->ucRemoteDisplayConfig; 892 893 /* Is allowed minimum BL level */ 894 info->min_allowed_bl_level = firmware_info->ucMinAllowedBL_Level; 895 /* Used starting from CI */ 896 info->smu_gpu_pll_output_freq = 897 (uint32_t) (le32_to_cpu(firmware_info->ulGPUPLL_OutputFreq) * 10); 898 899 return BP_RESULT_OK; 900 } 901 902 static enum bp_result get_ss_info_v3_1( 903 struct bios_parser *bp, 904 uint32_t id, 905 uint32_t index, 906 struct spread_spectrum_info *ss_info) 907 { 908 ATOM_ASIC_INTERNAL_SS_INFO_V3 *ss_table_header_include; 909 ATOM_ASIC_SS_ASSIGNMENT_V3 *tbl; 910 uint32_t table_size; 911 uint32_t i; 912 uint32_t table_index = 0; 913 914 if (!ss_info) 915 return BP_RESULT_BADINPUT; 916 917 if (!DATA_TABLES(ASIC_InternalSS_Info)) 918 return BP_RESULT_UNSUPPORTED; 919 920 ss_table_header_include = GET_IMAGE(ATOM_ASIC_INTERNAL_SS_INFO_V3, 921 DATA_TABLES(ASIC_InternalSS_Info)); 922 table_size = 923 (le16_to_cpu(ss_table_header_include->sHeader.usStructureSize) 924 - sizeof(ATOM_COMMON_TABLE_HEADER)) 925 / sizeof(ATOM_ASIC_SS_ASSIGNMENT_V3); 926 927 tbl = (ATOM_ASIC_SS_ASSIGNMENT_V3 *) 928 &ss_table_header_include->asSpreadSpectrum[0]; 929 930 memset(ss_info, 0, sizeof(struct spread_spectrum_info)); 931 932 for (i = 0; i < table_size; i++) { 933 if (tbl[i].ucClockIndication != (uint8_t) id) 934 continue; 935 936 if (table_index != index) { 937 table_index++; 938 continue; 939 } 940 /* VBIOS introduced new defines for Version 3, same values as 941 * before, so now use these new ones for Version 3. 942 * Shouldn't affect field VBIOS's V3 as define values are still 943 * same. 944 * #define SS_MODE_V3_CENTRE_SPREAD_MASK 0x01 945 * #define SS_MODE_V3_EXTERNAL_SS_MASK 0x02 946 947 * Old VBIOS defines: 948 * #define ATOM_SS_CENTRE_SPREAD_MODE_MASK 0x00000001 949 * #define ATOM_EXTERNAL_SS_MASK 0x00000002 950 */ 951 952 if (SS_MODE_V3_EXTERNAL_SS_MASK & tbl[i].ucSpreadSpectrumMode) 953 ss_info->type.EXTERNAL = true; 954 955 if (SS_MODE_V3_CENTRE_SPREAD_MASK & tbl[i].ucSpreadSpectrumMode) 956 ss_info->type.CENTER_MODE = true; 957 958 /* Older VBIOS (in field) always provides SS percentage in 0.01% 959 * units set Divider to 100 */ 960 ss_info->spread_percentage_divider = 100; 961 962 /* #define SS_MODE_V3_PERCENTAGE_DIV_BY_1000_MASK 0x10 */ 963 if (SS_MODE_V3_PERCENTAGE_DIV_BY_1000_MASK 964 & tbl[i].ucSpreadSpectrumMode) 965 ss_info->spread_percentage_divider = 1000; 966 967 ss_info->type.STEP_AND_DELAY_INFO = false; 968 /* convert [10KHz] into [KHz] */ 969 ss_info->target_clock_range = 970 le32_to_cpu(tbl[i].ulTargetClockRange) * 10; 971 ss_info->spread_spectrum_percentage = 972 (uint32_t)le16_to_cpu(tbl[i].usSpreadSpectrumPercentage); 973 ss_info->spread_spectrum_range = 974 (uint32_t)(le16_to_cpu(tbl[i].usSpreadRateIn10Hz) * 10); 975 976 return BP_RESULT_OK; 977 } 978 return BP_RESULT_NORECORD; 979 } 980 981 static enum bp_result bios_parser_transmitter_control( 982 struct dc_bios *dcb, 983 struct bp_transmitter_control *cntl) 984 { 985 struct bios_parser *bp = BP_FROM_DCB(dcb); 986 987 if (!bp->cmd_tbl.transmitter_control) 988 return BP_RESULT_FAILURE; 989 990 return bp->cmd_tbl.transmitter_control(bp, cntl); 991 } 992 993 static enum bp_result bios_parser_encoder_control( 994 struct dc_bios *dcb, 995 struct bp_encoder_control *cntl) 996 { 997 struct bios_parser *bp = BP_FROM_DCB(dcb); 998 999 if (!bp->cmd_tbl.dig_encoder_control) 1000 return BP_RESULT_FAILURE; 1001 1002 return bp->cmd_tbl.dig_encoder_control(bp, cntl); 1003 } 1004 1005 static enum bp_result bios_parser_adjust_pixel_clock( 1006 struct dc_bios *dcb, 1007 struct bp_adjust_pixel_clock_parameters *bp_params) 1008 { 1009 struct bios_parser *bp = BP_FROM_DCB(dcb); 1010 1011 if (!bp->cmd_tbl.adjust_display_pll) 1012 return BP_RESULT_FAILURE; 1013 1014 return bp->cmd_tbl.adjust_display_pll(bp, bp_params); 1015 } 1016 1017 static enum bp_result bios_parser_set_pixel_clock( 1018 struct dc_bios *dcb, 1019 struct bp_pixel_clock_parameters *bp_params) 1020 { 1021 struct bios_parser *bp = BP_FROM_DCB(dcb); 1022 1023 if (!bp->cmd_tbl.set_pixel_clock) 1024 return BP_RESULT_FAILURE; 1025 1026 return bp->cmd_tbl.set_pixel_clock(bp, bp_params); 1027 } 1028 1029 static enum bp_result bios_parser_set_dce_clock( 1030 struct dc_bios *dcb, 1031 struct bp_set_dce_clock_parameters *bp_params) 1032 { 1033 struct bios_parser *bp = BP_FROM_DCB(dcb); 1034 1035 if (!bp->cmd_tbl.set_dce_clock) 1036 return BP_RESULT_FAILURE; 1037 1038 return bp->cmd_tbl.set_dce_clock(bp, bp_params); 1039 } 1040 1041 static enum bp_result bios_parser_enable_spread_spectrum_on_ppll( 1042 struct dc_bios *dcb, 1043 struct bp_spread_spectrum_parameters *bp_params, 1044 bool enable) 1045 { 1046 struct bios_parser *bp = BP_FROM_DCB(dcb); 1047 1048 if (!bp->cmd_tbl.enable_spread_spectrum_on_ppll) 1049 return BP_RESULT_FAILURE; 1050 1051 return bp->cmd_tbl.enable_spread_spectrum_on_ppll( 1052 bp, bp_params, enable); 1053 1054 } 1055 1056 static enum bp_result bios_parser_program_crtc_timing( 1057 struct dc_bios *dcb, 1058 struct bp_hw_crtc_timing_parameters *bp_params) 1059 { 1060 struct bios_parser *bp = BP_FROM_DCB(dcb); 1061 1062 if (!bp->cmd_tbl.set_crtc_timing) 1063 return BP_RESULT_FAILURE; 1064 1065 return bp->cmd_tbl.set_crtc_timing(bp, bp_params); 1066 } 1067 1068 static enum bp_result bios_parser_program_display_engine_pll( 1069 struct dc_bios *dcb, 1070 struct bp_pixel_clock_parameters *bp_params) 1071 { 1072 struct bios_parser *bp = BP_FROM_DCB(dcb); 1073 1074 if (!bp->cmd_tbl.program_clock) 1075 return BP_RESULT_FAILURE; 1076 1077 return bp->cmd_tbl.program_clock(bp, bp_params); 1078 1079 } 1080 1081 1082 static enum bp_result bios_parser_enable_crtc( 1083 struct dc_bios *dcb, 1084 enum controller_id id, 1085 bool enable) 1086 { 1087 struct bios_parser *bp = BP_FROM_DCB(dcb); 1088 1089 if (!bp->cmd_tbl.enable_crtc) 1090 return BP_RESULT_FAILURE; 1091 1092 return bp->cmd_tbl.enable_crtc(bp, id, enable); 1093 } 1094 1095 static enum bp_result bios_parser_crtc_source_select( 1096 struct dc_bios *dcb, 1097 struct bp_crtc_source_select *bp_params) 1098 { 1099 struct bios_parser *bp = BP_FROM_DCB(dcb); 1100 1101 if (!bp->cmd_tbl.select_crtc_source) 1102 return BP_RESULT_FAILURE; 1103 1104 return bp->cmd_tbl.select_crtc_source(bp, bp_params); 1105 } 1106 1107 static enum bp_result bios_parser_enable_disp_power_gating( 1108 struct dc_bios *dcb, 1109 enum controller_id controller_id, 1110 enum bp_pipe_control_action action) 1111 { 1112 struct bios_parser *bp = BP_FROM_DCB(dcb); 1113 1114 if (!bp->cmd_tbl.enable_disp_power_gating) 1115 return BP_RESULT_FAILURE; 1116 1117 return bp->cmd_tbl.enable_disp_power_gating(bp, controller_id, 1118 action); 1119 } 1120 1121 static bool bios_parser_is_device_id_supported( 1122 struct dc_bios *dcb, 1123 struct device_id id) 1124 { 1125 struct bios_parser *bp = BP_FROM_DCB(dcb); 1126 1127 uint32_t mask = get_support_mask_for_device_id(id); 1128 1129 return (le16_to_cpu(bp->object_info_tbl.v1_1->usDeviceSupport) & mask) != 0; 1130 } 1131 1132 static enum bp_result bios_parser_crt_control( 1133 struct dc_bios *dcb, 1134 enum engine_id engine_id, 1135 bool enable, 1136 uint32_t pixel_clock) 1137 { 1138 struct bios_parser *bp = BP_FROM_DCB(dcb); 1139 uint8_t standard; 1140 1141 if (!bp->cmd_tbl.dac1_encoder_control && 1142 engine_id == ENGINE_ID_DACA) 1143 return BP_RESULT_FAILURE; 1144 if (!bp->cmd_tbl.dac2_encoder_control && 1145 engine_id == ENGINE_ID_DACB) 1146 return BP_RESULT_FAILURE; 1147 /* validate params */ 1148 switch (engine_id) { 1149 case ENGINE_ID_DACA: 1150 case ENGINE_ID_DACB: 1151 break; 1152 default: 1153 /* unsupported engine */ 1154 return BP_RESULT_FAILURE; 1155 } 1156 1157 standard = ATOM_DAC1_PS2; /* == ATOM_DAC2_PS2 */ 1158 1159 if (enable) { 1160 if (engine_id == ENGINE_ID_DACA) { 1161 bp->cmd_tbl.dac1_encoder_control(bp, enable, 1162 pixel_clock, standard); 1163 if (bp->cmd_tbl.dac1_output_control != NULL) 1164 bp->cmd_tbl.dac1_output_control(bp, enable); 1165 } else { 1166 bp->cmd_tbl.dac2_encoder_control(bp, enable, 1167 pixel_clock, standard); 1168 if (bp->cmd_tbl.dac2_output_control != NULL) 1169 bp->cmd_tbl.dac2_output_control(bp, enable); 1170 } 1171 } else { 1172 if (engine_id == ENGINE_ID_DACA) { 1173 if (bp->cmd_tbl.dac1_output_control != NULL) 1174 bp->cmd_tbl.dac1_output_control(bp, enable); 1175 bp->cmd_tbl.dac1_encoder_control(bp, enable, 1176 pixel_clock, standard); 1177 } else { 1178 if (bp->cmd_tbl.dac2_output_control != NULL) 1179 bp->cmd_tbl.dac2_output_control(bp, enable); 1180 bp->cmd_tbl.dac2_encoder_control(bp, enable, 1181 pixel_clock, standard); 1182 } 1183 } 1184 1185 return BP_RESULT_OK; 1186 } 1187 1188 static ATOM_HPD_INT_RECORD *get_hpd_record(struct bios_parser *bp, 1189 ATOM_OBJECT *object) 1190 { 1191 ATOM_COMMON_RECORD_HEADER *header; 1192 uint32_t offset; 1193 1194 if (!object) { 1195 BREAK_TO_DEBUGGER(); /* Invalid object */ 1196 return NULL; 1197 } 1198 1199 offset = le16_to_cpu(object->usRecordOffset) 1200 + bp->object_info_tbl_offset; 1201 1202 for (;;) { 1203 header = GET_IMAGE(ATOM_COMMON_RECORD_HEADER, offset); 1204 1205 if (!header) 1206 return NULL; 1207 1208 if (LAST_RECORD_TYPE == header->ucRecordType || 1209 !header->ucRecordSize) 1210 break; 1211 1212 if (ATOM_HPD_INT_RECORD_TYPE == header->ucRecordType 1213 && sizeof(ATOM_HPD_INT_RECORD) <= header->ucRecordSize) 1214 return (ATOM_HPD_INT_RECORD *) header; 1215 1216 offset += header->ucRecordSize; 1217 } 1218 1219 return NULL; 1220 } 1221 1222 /** 1223 * Get I2C information of input object id 1224 * 1225 * search all records to find the ATOM_I2C_RECORD_TYPE record IR 1226 */ 1227 static ATOM_I2C_RECORD *get_i2c_record( 1228 struct bios_parser *bp, 1229 ATOM_OBJECT *object) 1230 { 1231 uint32_t offset; 1232 ATOM_COMMON_RECORD_HEADER *record_header; 1233 1234 if (!object) { 1235 BREAK_TO_DEBUGGER(); 1236 /* Invalid object */ 1237 return NULL; 1238 } 1239 1240 offset = le16_to_cpu(object->usRecordOffset) 1241 + bp->object_info_tbl_offset; 1242 1243 for (;;) { 1244 record_header = GET_IMAGE(ATOM_COMMON_RECORD_HEADER, offset); 1245 1246 if (!record_header) 1247 return NULL; 1248 1249 if (LAST_RECORD_TYPE == record_header->ucRecordType || 1250 0 == record_header->ucRecordSize) 1251 break; 1252 1253 if (ATOM_I2C_RECORD_TYPE == record_header->ucRecordType && 1254 sizeof(ATOM_I2C_RECORD) <= 1255 record_header->ucRecordSize) { 1256 return (ATOM_I2C_RECORD *)record_header; 1257 } 1258 1259 offset += record_header->ucRecordSize; 1260 } 1261 1262 return NULL; 1263 } 1264 1265 static enum bp_result get_ss_info_from_ss_info_table( 1266 struct bios_parser *bp, 1267 uint32_t id, 1268 struct spread_spectrum_info *ss_info); 1269 static enum bp_result get_ss_info_from_tbl( 1270 struct bios_parser *bp, 1271 uint32_t id, 1272 struct spread_spectrum_info *ss_info); 1273 /** 1274 * bios_parser_get_spread_spectrum_info 1275 * Get spread spectrum information from the ASIC_InternalSS_Info(ver 2.1 or 1276 * ver 3.1) or SS_Info table from the VBIOS. Currently ASIC_InternalSS_Info 1277 * ver 2.1 can co-exist with SS_Info table. Expect ASIC_InternalSS_Info ver 3.1, 1278 * there is only one entry for each signal /ss id. However, there is 1279 * no planning of supporting multiple spread Sprectum entry for EverGreen 1280 * @param [in] this 1281 * @param [in] signal, ASSignalType to be converted to info index 1282 * @param [in] index, number of entries that match the converted info index 1283 * @param [out] ss_info, sprectrum information structure, 1284 * @return Bios parser result code 1285 */ 1286 static enum bp_result bios_parser_get_spread_spectrum_info( 1287 struct dc_bios *dcb, 1288 enum as_signal_type signal, 1289 uint32_t index, 1290 struct spread_spectrum_info *ss_info) 1291 { 1292 struct bios_parser *bp = BP_FROM_DCB(dcb); 1293 enum bp_result result = BP_RESULT_UNSUPPORTED; 1294 uint32_t clk_id_ss = 0; 1295 ATOM_COMMON_TABLE_HEADER *header; 1296 struct atom_data_revision tbl_revision; 1297 1298 if (!ss_info) /* check for bad input */ 1299 return BP_RESULT_BADINPUT; 1300 /* signal translation */ 1301 clk_id_ss = signal_to_ss_id(signal); 1302 1303 if (!DATA_TABLES(ASIC_InternalSS_Info)) 1304 if (!index) 1305 return get_ss_info_from_ss_info_table(bp, clk_id_ss, 1306 ss_info); 1307 1308 header = GET_IMAGE(ATOM_COMMON_TABLE_HEADER, 1309 DATA_TABLES(ASIC_InternalSS_Info)); 1310 get_atom_data_table_revision(header, &tbl_revision); 1311 1312 switch (tbl_revision.major) { 1313 case 2: 1314 switch (tbl_revision.minor) { 1315 case 1: 1316 /* there can not be more then one entry for Internal 1317 * SS Info table version 2.1 */ 1318 if (!index) 1319 return get_ss_info_from_tbl(bp, clk_id_ss, 1320 ss_info); 1321 break; 1322 default: 1323 break; 1324 } 1325 break; 1326 1327 case 3: 1328 switch (tbl_revision.minor) { 1329 case 1: 1330 return get_ss_info_v3_1(bp, clk_id_ss, index, ss_info); 1331 default: 1332 break; 1333 } 1334 break; 1335 default: 1336 break; 1337 } 1338 /* there can not be more then one entry for SS Info table */ 1339 return result; 1340 } 1341 1342 static enum bp_result get_ss_info_from_internal_ss_info_tbl_V2_1( 1343 struct bios_parser *bp, 1344 uint32_t id, 1345 struct spread_spectrum_info *info); 1346 1347 /** 1348 * get_ss_info_from_table 1349 * Get spread sprectrum information from the ASIC_InternalSS_Info Ver 2.1 or 1350 * SS_Info table from the VBIOS 1351 * There can not be more than 1 entry for ASIC_InternalSS_Info Ver 2.1 or 1352 * SS_Info. 1353 * 1354 * @param this 1355 * @param id, spread sprectrum info index 1356 * @param pSSinfo, sprectrum information structure, 1357 * @return Bios parser result code 1358 */ 1359 static enum bp_result get_ss_info_from_tbl( 1360 struct bios_parser *bp, 1361 uint32_t id, 1362 struct spread_spectrum_info *ss_info) 1363 { 1364 if (!ss_info) /* check for bad input, if ss_info is not NULL */ 1365 return BP_RESULT_BADINPUT; 1366 /* for SS_Info table only support DP and LVDS */ 1367 if (id == ASIC_INTERNAL_SS_ON_DP || id == ASIC_INTERNAL_SS_ON_LVDS) 1368 return get_ss_info_from_ss_info_table(bp, id, ss_info); 1369 else 1370 return get_ss_info_from_internal_ss_info_tbl_V2_1(bp, id, 1371 ss_info); 1372 } 1373 1374 /** 1375 * get_ss_info_from_internal_ss_info_tbl_V2_1 1376 * Get spread sprectrum information from the ASIC_InternalSS_Info table Ver 2.1 1377 * from the VBIOS 1378 * There will not be multiple entry for Ver 2.1 1379 * 1380 * @param id, spread sprectrum info index 1381 * @param pSSinfo, sprectrum information structure, 1382 * @return Bios parser result code 1383 */ 1384 static enum bp_result get_ss_info_from_internal_ss_info_tbl_V2_1( 1385 struct bios_parser *bp, 1386 uint32_t id, 1387 struct spread_spectrum_info *info) 1388 { 1389 enum bp_result result = BP_RESULT_UNSUPPORTED; 1390 ATOM_ASIC_INTERNAL_SS_INFO_V2 *header; 1391 ATOM_ASIC_SS_ASSIGNMENT_V2 *tbl; 1392 uint32_t tbl_size, i; 1393 1394 if (!DATA_TABLES(ASIC_InternalSS_Info)) 1395 return result; 1396 1397 header = GET_IMAGE(ATOM_ASIC_INTERNAL_SS_INFO_V2, 1398 DATA_TABLES(ASIC_InternalSS_Info)); 1399 1400 memset(info, 0, sizeof(struct spread_spectrum_info)); 1401 1402 tbl_size = (le16_to_cpu(header->sHeader.usStructureSize) 1403 - sizeof(ATOM_COMMON_TABLE_HEADER)) 1404 / sizeof(ATOM_ASIC_SS_ASSIGNMENT_V2); 1405 1406 tbl = (ATOM_ASIC_SS_ASSIGNMENT_V2 *) 1407 &(header->asSpreadSpectrum[0]); 1408 for (i = 0; i < tbl_size; i++) { 1409 result = BP_RESULT_NORECORD; 1410 1411 if (tbl[i].ucClockIndication != (uint8_t)id) 1412 continue; 1413 1414 if (ATOM_EXTERNAL_SS_MASK 1415 & tbl[i].ucSpreadSpectrumMode) { 1416 info->type.EXTERNAL = true; 1417 } 1418 if (ATOM_SS_CENTRE_SPREAD_MODE_MASK 1419 & tbl[i].ucSpreadSpectrumMode) { 1420 info->type.CENTER_MODE = true; 1421 } 1422 info->type.STEP_AND_DELAY_INFO = false; 1423 /* convert [10KHz] into [KHz] */ 1424 info->target_clock_range = 1425 le32_to_cpu(tbl[i].ulTargetClockRange) * 10; 1426 info->spread_spectrum_percentage = 1427 (uint32_t)le16_to_cpu(tbl[i].usSpreadSpectrumPercentage); 1428 info->spread_spectrum_range = 1429 (uint32_t)(le16_to_cpu(tbl[i].usSpreadRateIn10Hz) * 10); 1430 result = BP_RESULT_OK; 1431 break; 1432 } 1433 1434 return result; 1435 1436 } 1437 1438 /** 1439 * get_ss_info_from_ss_info_table 1440 * Get spread sprectrum information from the SS_Info table from the VBIOS 1441 * if the pointer to info is NULL, indicate the caller what to know the number 1442 * of entries that matches the id 1443 * for, the SS_Info table, there should not be more than 1 entry match. 1444 * 1445 * @param [in] id, spread sprectrum id 1446 * @param [out] pSSinfo, sprectrum information structure, 1447 * @return Bios parser result code 1448 */ 1449 static enum bp_result get_ss_info_from_ss_info_table( 1450 struct bios_parser *bp, 1451 uint32_t id, 1452 struct spread_spectrum_info *ss_info) 1453 { 1454 enum bp_result result = BP_RESULT_UNSUPPORTED; 1455 ATOM_SPREAD_SPECTRUM_INFO *tbl; 1456 ATOM_COMMON_TABLE_HEADER *header; 1457 uint32_t table_size; 1458 uint32_t i; 1459 uint32_t id_local = SS_ID_UNKNOWN; 1460 struct atom_data_revision revision; 1461 1462 /* exist of the SS_Info table */ 1463 /* check for bad input, pSSinfo can not be NULL */ 1464 if (!DATA_TABLES(SS_Info) || !ss_info) 1465 return result; 1466 1467 header = GET_IMAGE(ATOM_COMMON_TABLE_HEADER, DATA_TABLES(SS_Info)); 1468 get_atom_data_table_revision(header, &revision); 1469 1470 tbl = GET_IMAGE(ATOM_SPREAD_SPECTRUM_INFO, DATA_TABLES(SS_Info)); 1471 1472 if (1 != revision.major || 2 > revision.minor) 1473 return result; 1474 1475 /* have to convert from Internal_SS format to SS_Info format */ 1476 switch (id) { 1477 case ASIC_INTERNAL_SS_ON_DP: 1478 id_local = SS_ID_DP1; 1479 break; 1480 case ASIC_INTERNAL_SS_ON_LVDS: 1481 { 1482 struct embedded_panel_info panel_info; 1483 1484 if (bios_parser_get_embedded_panel_info(&bp->base, &panel_info) 1485 == BP_RESULT_OK) 1486 id_local = panel_info.ss_id; 1487 break; 1488 } 1489 default: 1490 break; 1491 } 1492 1493 if (id_local == SS_ID_UNKNOWN) 1494 return result; 1495 1496 table_size = (le16_to_cpu(tbl->sHeader.usStructureSize) - 1497 sizeof(ATOM_COMMON_TABLE_HEADER)) / 1498 sizeof(ATOM_SPREAD_SPECTRUM_ASSIGNMENT); 1499 1500 for (i = 0; i < table_size; i++) { 1501 if (id_local != (uint32_t)tbl->asSS_Info[i].ucSS_Id) 1502 continue; 1503 1504 memset(ss_info, 0, sizeof(struct spread_spectrum_info)); 1505 1506 if (ATOM_EXTERNAL_SS_MASK & 1507 tbl->asSS_Info[i].ucSpreadSpectrumType) 1508 ss_info->type.EXTERNAL = true; 1509 1510 if (ATOM_SS_CENTRE_SPREAD_MODE_MASK & 1511 tbl->asSS_Info[i].ucSpreadSpectrumType) 1512 ss_info->type.CENTER_MODE = true; 1513 1514 ss_info->type.STEP_AND_DELAY_INFO = true; 1515 ss_info->spread_spectrum_percentage = 1516 (uint32_t)le16_to_cpu(tbl->asSS_Info[i].usSpreadSpectrumPercentage); 1517 ss_info->step_and_delay_info.step = tbl->asSS_Info[i].ucSS_Step; 1518 ss_info->step_and_delay_info.delay = 1519 tbl->asSS_Info[i].ucSS_Delay; 1520 ss_info->step_and_delay_info.recommended_ref_div = 1521 tbl->asSS_Info[i].ucRecommendedRef_Div; 1522 ss_info->spread_spectrum_range = 1523 (uint32_t)tbl->asSS_Info[i].ucSS_Range * 10000; 1524 1525 /* there will be only one entry for each display type in SS_info 1526 * table */ 1527 result = BP_RESULT_OK; 1528 break; 1529 } 1530 1531 return result; 1532 } 1533 static enum bp_result get_embedded_panel_info_v1_2( 1534 struct bios_parser *bp, 1535 struct embedded_panel_info *info); 1536 static enum bp_result get_embedded_panel_info_v1_3( 1537 struct bios_parser *bp, 1538 struct embedded_panel_info *info); 1539 1540 static enum bp_result bios_parser_get_embedded_panel_info( 1541 struct dc_bios *dcb, 1542 struct embedded_panel_info *info) 1543 { 1544 struct bios_parser *bp = BP_FROM_DCB(dcb); 1545 ATOM_COMMON_TABLE_HEADER *hdr; 1546 1547 if (!DATA_TABLES(LCD_Info)) 1548 return BP_RESULT_FAILURE; 1549 1550 hdr = GET_IMAGE(ATOM_COMMON_TABLE_HEADER, DATA_TABLES(LCD_Info)); 1551 1552 if (!hdr) 1553 return BP_RESULT_BADBIOSTABLE; 1554 1555 switch (hdr->ucTableFormatRevision) { 1556 case 1: 1557 switch (hdr->ucTableContentRevision) { 1558 case 0: 1559 case 1: 1560 case 2: 1561 return get_embedded_panel_info_v1_2(bp, info); 1562 case 3: 1563 return get_embedded_panel_info_v1_3(bp, info); 1564 default: 1565 break; 1566 } 1567 default: 1568 break; 1569 } 1570 1571 return BP_RESULT_FAILURE; 1572 } 1573 1574 static enum bp_result get_embedded_panel_info_v1_2( 1575 struct bios_parser *bp, 1576 struct embedded_panel_info *info) 1577 { 1578 ATOM_LVDS_INFO_V12 *lvds; 1579 1580 if (!info) 1581 return BP_RESULT_BADINPUT; 1582 1583 if (!DATA_TABLES(LVDS_Info)) 1584 return BP_RESULT_UNSUPPORTED; 1585 1586 lvds = 1587 GET_IMAGE(ATOM_LVDS_INFO_V12, DATA_TABLES(LVDS_Info)); 1588 1589 if (!lvds) 1590 return BP_RESULT_BADBIOSTABLE; 1591 1592 if (1 != lvds->sHeader.ucTableFormatRevision 1593 || 2 > lvds->sHeader.ucTableContentRevision) 1594 return BP_RESULT_UNSUPPORTED; 1595 1596 memset(info, 0, sizeof(struct embedded_panel_info)); 1597 1598 /* We need to convert from 10KHz units into KHz units*/ 1599 info->lcd_timing.pixel_clk = 1600 le16_to_cpu(lvds->sLCDTiming.usPixClk) * 10; 1601 /* usHActive does not include borders, according to VBIOS team*/ 1602 info->lcd_timing.horizontal_addressable = 1603 le16_to_cpu(lvds->sLCDTiming.usHActive); 1604 /* usHBlanking_Time includes borders, so we should really be subtracting 1605 * borders duing this translation, but LVDS generally*/ 1606 /* doesn't have borders, so we should be okay leaving this as is for 1607 * now. May need to revisit if we ever have LVDS with borders*/ 1608 info->lcd_timing.horizontal_blanking_time = 1609 le16_to_cpu(lvds->sLCDTiming.usHBlanking_Time); 1610 /* usVActive does not include borders, according to VBIOS team*/ 1611 info->lcd_timing.vertical_addressable = 1612 le16_to_cpu(lvds->sLCDTiming.usVActive); 1613 /* usVBlanking_Time includes borders, so we should really be subtracting 1614 * borders duing this translation, but LVDS generally*/ 1615 /* doesn't have borders, so we should be okay leaving this as is for 1616 * now. May need to revisit if we ever have LVDS with borders*/ 1617 info->lcd_timing.vertical_blanking_time = 1618 le16_to_cpu(lvds->sLCDTiming.usVBlanking_Time); 1619 info->lcd_timing.horizontal_sync_offset = 1620 le16_to_cpu(lvds->sLCDTiming.usHSyncOffset); 1621 info->lcd_timing.horizontal_sync_width = 1622 le16_to_cpu(lvds->sLCDTiming.usHSyncWidth); 1623 info->lcd_timing.vertical_sync_offset = 1624 le16_to_cpu(lvds->sLCDTiming.usVSyncOffset); 1625 info->lcd_timing.vertical_sync_width = 1626 le16_to_cpu(lvds->sLCDTiming.usVSyncWidth); 1627 info->lcd_timing.horizontal_border = lvds->sLCDTiming.ucHBorder; 1628 info->lcd_timing.vertical_border = lvds->sLCDTiming.ucVBorder; 1629 info->lcd_timing.misc_info.HORIZONTAL_CUT_OFF = 1630 lvds->sLCDTiming.susModeMiscInfo.sbfAccess.HorizontalCutOff; 1631 info->lcd_timing.misc_info.H_SYNC_POLARITY = 1632 ~(uint32_t) 1633 lvds->sLCDTiming.susModeMiscInfo.sbfAccess.HSyncPolarity; 1634 info->lcd_timing.misc_info.V_SYNC_POLARITY = 1635 ~(uint32_t) 1636 lvds->sLCDTiming.susModeMiscInfo.sbfAccess.VSyncPolarity; 1637 info->lcd_timing.misc_info.VERTICAL_CUT_OFF = 1638 lvds->sLCDTiming.susModeMiscInfo.sbfAccess.VerticalCutOff; 1639 info->lcd_timing.misc_info.H_REPLICATION_BY2 = 1640 lvds->sLCDTiming.susModeMiscInfo.sbfAccess.H_ReplicationBy2; 1641 info->lcd_timing.misc_info.V_REPLICATION_BY2 = 1642 lvds->sLCDTiming.susModeMiscInfo.sbfAccess.V_ReplicationBy2; 1643 info->lcd_timing.misc_info.COMPOSITE_SYNC = 1644 lvds->sLCDTiming.susModeMiscInfo.sbfAccess.CompositeSync; 1645 info->lcd_timing.misc_info.INTERLACE = 1646 lvds->sLCDTiming.susModeMiscInfo.sbfAccess.Interlace; 1647 info->lcd_timing.misc_info.DOUBLE_CLOCK = 1648 lvds->sLCDTiming.susModeMiscInfo.sbfAccess.DoubleClock; 1649 info->ss_id = lvds->ucSS_Id; 1650 1651 { 1652 uint8_t rr = le16_to_cpu(lvds->usSupportedRefreshRate); 1653 /* Get minimum supported refresh rate*/ 1654 if (SUPPORTED_LCD_REFRESHRATE_30Hz & rr) 1655 info->supported_rr.REFRESH_RATE_30HZ = 1; 1656 else if (SUPPORTED_LCD_REFRESHRATE_40Hz & rr) 1657 info->supported_rr.REFRESH_RATE_40HZ = 1; 1658 else if (SUPPORTED_LCD_REFRESHRATE_48Hz & rr) 1659 info->supported_rr.REFRESH_RATE_48HZ = 1; 1660 else if (SUPPORTED_LCD_REFRESHRATE_50Hz & rr) 1661 info->supported_rr.REFRESH_RATE_50HZ = 1; 1662 else if (SUPPORTED_LCD_REFRESHRATE_60Hz & rr) 1663 info->supported_rr.REFRESH_RATE_60HZ = 1; 1664 } 1665 1666 /*Drr panel support can be reported by VBIOS*/ 1667 if (LCDPANEL_CAP_DRR_SUPPORTED 1668 & lvds->ucLCDPanel_SpecialHandlingCap) 1669 info->drr_enabled = 1; 1670 1671 if (ATOM_PANEL_MISC_DUAL & lvds->ucLVDS_Misc) 1672 info->lcd_timing.misc_info.DOUBLE_CLOCK = true; 1673 1674 if (ATOM_PANEL_MISC_888RGB & lvds->ucLVDS_Misc) 1675 info->lcd_timing.misc_info.RGB888 = true; 1676 1677 info->lcd_timing.misc_info.GREY_LEVEL = 1678 (uint32_t) (ATOM_PANEL_MISC_GREY_LEVEL & 1679 lvds->ucLVDS_Misc) >> ATOM_PANEL_MISC_GREY_LEVEL_SHIFT; 1680 1681 if (ATOM_PANEL_MISC_SPATIAL & lvds->ucLVDS_Misc) 1682 info->lcd_timing.misc_info.SPATIAL = true; 1683 1684 if (ATOM_PANEL_MISC_TEMPORAL & lvds->ucLVDS_Misc) 1685 info->lcd_timing.misc_info.TEMPORAL = true; 1686 1687 if (ATOM_PANEL_MISC_API_ENABLED & lvds->ucLVDS_Misc) 1688 info->lcd_timing.misc_info.API_ENABLED = true; 1689 1690 return BP_RESULT_OK; 1691 } 1692 1693 static enum bp_result get_embedded_panel_info_v1_3( 1694 struct bios_parser *bp, 1695 struct embedded_panel_info *info) 1696 { 1697 ATOM_LCD_INFO_V13 *lvds; 1698 1699 if (!info) 1700 return BP_RESULT_BADINPUT; 1701 1702 if (!DATA_TABLES(LCD_Info)) 1703 return BP_RESULT_UNSUPPORTED; 1704 1705 lvds = GET_IMAGE(ATOM_LCD_INFO_V13, DATA_TABLES(LCD_Info)); 1706 1707 if (!lvds) 1708 return BP_RESULT_BADBIOSTABLE; 1709 1710 if (!((1 == lvds->sHeader.ucTableFormatRevision) 1711 && (3 <= lvds->sHeader.ucTableContentRevision))) 1712 return BP_RESULT_UNSUPPORTED; 1713 1714 memset(info, 0, sizeof(struct embedded_panel_info)); 1715 1716 /* We need to convert from 10KHz units into KHz units */ 1717 info->lcd_timing.pixel_clk = 1718 le16_to_cpu(lvds->sLCDTiming.usPixClk) * 10; 1719 /* usHActive does not include borders, according to VBIOS team */ 1720 info->lcd_timing.horizontal_addressable = 1721 le16_to_cpu(lvds->sLCDTiming.usHActive); 1722 /* usHBlanking_Time includes borders, so we should really be subtracting 1723 * borders duing this translation, but LVDS generally*/ 1724 /* doesn't have borders, so we should be okay leaving this as is for 1725 * now. May need to revisit if we ever have LVDS with borders*/ 1726 info->lcd_timing.horizontal_blanking_time = 1727 le16_to_cpu(lvds->sLCDTiming.usHBlanking_Time); 1728 /* usVActive does not include borders, according to VBIOS team*/ 1729 info->lcd_timing.vertical_addressable = 1730 le16_to_cpu(lvds->sLCDTiming.usVActive); 1731 /* usVBlanking_Time includes borders, so we should really be subtracting 1732 * borders duing this translation, but LVDS generally*/ 1733 /* doesn't have borders, so we should be okay leaving this as is for 1734 * now. May need to revisit if we ever have LVDS with borders*/ 1735 info->lcd_timing.vertical_blanking_time = 1736 le16_to_cpu(lvds->sLCDTiming.usVBlanking_Time); 1737 info->lcd_timing.horizontal_sync_offset = 1738 le16_to_cpu(lvds->sLCDTiming.usHSyncOffset); 1739 info->lcd_timing.horizontal_sync_width = 1740 le16_to_cpu(lvds->sLCDTiming.usHSyncWidth); 1741 info->lcd_timing.vertical_sync_offset = 1742 le16_to_cpu(lvds->sLCDTiming.usVSyncOffset); 1743 info->lcd_timing.vertical_sync_width = 1744 le16_to_cpu(lvds->sLCDTiming.usVSyncWidth); 1745 info->lcd_timing.horizontal_border = lvds->sLCDTiming.ucHBorder; 1746 info->lcd_timing.vertical_border = lvds->sLCDTiming.ucVBorder; 1747 info->lcd_timing.misc_info.HORIZONTAL_CUT_OFF = 1748 lvds->sLCDTiming.susModeMiscInfo.sbfAccess.HorizontalCutOff; 1749 info->lcd_timing.misc_info.H_SYNC_POLARITY = 1750 ~(uint32_t) 1751 lvds->sLCDTiming.susModeMiscInfo.sbfAccess.HSyncPolarity; 1752 info->lcd_timing.misc_info.V_SYNC_POLARITY = 1753 ~(uint32_t) 1754 lvds->sLCDTiming.susModeMiscInfo.sbfAccess.VSyncPolarity; 1755 info->lcd_timing.misc_info.VERTICAL_CUT_OFF = 1756 lvds->sLCDTiming.susModeMiscInfo.sbfAccess.VerticalCutOff; 1757 info->lcd_timing.misc_info.H_REPLICATION_BY2 = 1758 lvds->sLCDTiming.susModeMiscInfo.sbfAccess.H_ReplicationBy2; 1759 info->lcd_timing.misc_info.V_REPLICATION_BY2 = 1760 lvds->sLCDTiming.susModeMiscInfo.sbfAccess.V_ReplicationBy2; 1761 info->lcd_timing.misc_info.COMPOSITE_SYNC = 1762 lvds->sLCDTiming.susModeMiscInfo.sbfAccess.CompositeSync; 1763 info->lcd_timing.misc_info.INTERLACE = 1764 lvds->sLCDTiming.susModeMiscInfo.sbfAccess.Interlace; 1765 info->lcd_timing.misc_info.DOUBLE_CLOCK = 1766 lvds->sLCDTiming.susModeMiscInfo.sbfAccess.DoubleClock; 1767 info->ss_id = lvds->ucSS_Id; 1768 1769 /* Drr panel support can be reported by VBIOS*/ 1770 if (LCDPANEL_CAP_V13_DRR_SUPPORTED 1771 & lvds->ucLCDPanel_SpecialHandlingCap) 1772 info->drr_enabled = 1; 1773 1774 /* Get supported refresh rate*/ 1775 if (info->drr_enabled == 1) { 1776 uint8_t min_rr = 1777 lvds->sRefreshRateSupport.ucMinRefreshRateForDRR; 1778 uint8_t rr = lvds->sRefreshRateSupport.ucSupportedRefreshRate; 1779 1780 if (min_rr != 0) { 1781 if (SUPPORTED_LCD_REFRESHRATE_30Hz & min_rr) 1782 info->supported_rr.REFRESH_RATE_30HZ = 1; 1783 else if (SUPPORTED_LCD_REFRESHRATE_40Hz & min_rr) 1784 info->supported_rr.REFRESH_RATE_40HZ = 1; 1785 else if (SUPPORTED_LCD_REFRESHRATE_48Hz & min_rr) 1786 info->supported_rr.REFRESH_RATE_48HZ = 1; 1787 else if (SUPPORTED_LCD_REFRESHRATE_50Hz & min_rr) 1788 info->supported_rr.REFRESH_RATE_50HZ = 1; 1789 else if (SUPPORTED_LCD_REFRESHRATE_60Hz & min_rr) 1790 info->supported_rr.REFRESH_RATE_60HZ = 1; 1791 } else { 1792 if (SUPPORTED_LCD_REFRESHRATE_30Hz & rr) 1793 info->supported_rr.REFRESH_RATE_30HZ = 1; 1794 else if (SUPPORTED_LCD_REFRESHRATE_40Hz & rr) 1795 info->supported_rr.REFRESH_RATE_40HZ = 1; 1796 else if (SUPPORTED_LCD_REFRESHRATE_48Hz & rr) 1797 info->supported_rr.REFRESH_RATE_48HZ = 1; 1798 else if (SUPPORTED_LCD_REFRESHRATE_50Hz & rr) 1799 info->supported_rr.REFRESH_RATE_50HZ = 1; 1800 else if (SUPPORTED_LCD_REFRESHRATE_60Hz & rr) 1801 info->supported_rr.REFRESH_RATE_60HZ = 1; 1802 } 1803 } 1804 1805 if (ATOM_PANEL_MISC_V13_DUAL & lvds->ucLCD_Misc) 1806 info->lcd_timing.misc_info.DOUBLE_CLOCK = true; 1807 1808 if (ATOM_PANEL_MISC_V13_8BIT_PER_COLOR & lvds->ucLCD_Misc) 1809 info->lcd_timing.misc_info.RGB888 = true; 1810 1811 info->lcd_timing.misc_info.GREY_LEVEL = 1812 (uint32_t) (ATOM_PANEL_MISC_V13_GREY_LEVEL & 1813 lvds->ucLCD_Misc) >> ATOM_PANEL_MISC_V13_GREY_LEVEL_SHIFT; 1814 1815 return BP_RESULT_OK; 1816 } 1817 1818 /** 1819 * bios_parser_get_encoder_cap_info 1820 * 1821 * @brief 1822 * Get encoder capability information of input object id 1823 * 1824 * @param object_id, Object id 1825 * @param object_id, encoder cap information structure 1826 * 1827 * @return Bios parser result code 1828 * 1829 */ 1830 static enum bp_result bios_parser_get_encoder_cap_info( 1831 struct dc_bios *dcb, 1832 struct graphics_object_id object_id, 1833 struct bp_encoder_cap_info *info) 1834 { 1835 struct bios_parser *bp = BP_FROM_DCB(dcb); 1836 ATOM_OBJECT *object; 1837 ATOM_ENCODER_CAP_RECORD_V2 *record = NULL; 1838 1839 if (!info) 1840 return BP_RESULT_BADINPUT; 1841 1842 object = get_bios_object(bp, object_id); 1843 1844 if (!object) 1845 return BP_RESULT_BADINPUT; 1846 1847 record = get_encoder_cap_record(bp, object); 1848 if (!record) 1849 return BP_RESULT_NORECORD; 1850 1851 info->DP_HBR2_EN = record->usHBR2En; 1852 info->DP_HBR3_EN = record->usHBR3En; 1853 info->HDMI_6GB_EN = record->usHDMI6GEn; 1854 return BP_RESULT_OK; 1855 } 1856 1857 /** 1858 * get_encoder_cap_record 1859 * 1860 * @brief 1861 * Get encoder cap record for the object 1862 * 1863 * @param object, ATOM object 1864 * 1865 * @return atom encoder cap record 1866 * 1867 * @note 1868 * search all records to find the ATOM_ENCODER_CAP_RECORD_V2 record 1869 */ 1870 static ATOM_ENCODER_CAP_RECORD_V2 *get_encoder_cap_record( 1871 struct bios_parser *bp, 1872 ATOM_OBJECT *object) 1873 { 1874 ATOM_COMMON_RECORD_HEADER *header; 1875 uint32_t offset; 1876 1877 if (!object) { 1878 BREAK_TO_DEBUGGER(); /* Invalid object */ 1879 return NULL; 1880 } 1881 1882 offset = le16_to_cpu(object->usRecordOffset) 1883 + bp->object_info_tbl_offset; 1884 1885 for (;;) { 1886 header = GET_IMAGE(ATOM_COMMON_RECORD_HEADER, offset); 1887 1888 if (!header) 1889 return NULL; 1890 1891 offset += header->ucRecordSize; 1892 1893 if (LAST_RECORD_TYPE == header->ucRecordType || 1894 !header->ucRecordSize) 1895 break; 1896 1897 if (ATOM_ENCODER_CAP_RECORD_TYPE != header->ucRecordType) 1898 continue; 1899 1900 if (sizeof(ATOM_ENCODER_CAP_RECORD_V2) <= header->ucRecordSize) 1901 return (ATOM_ENCODER_CAP_RECORD_V2 *)header; 1902 } 1903 1904 return NULL; 1905 } 1906 1907 static uint32_t get_ss_entry_number( 1908 struct bios_parser *bp, 1909 uint32_t id); 1910 static uint32_t get_ss_entry_number_from_internal_ss_info_tbl_v2_1( 1911 struct bios_parser *bp, 1912 uint32_t id); 1913 static uint32_t get_ss_entry_number_from_internal_ss_info_tbl_V3_1( 1914 struct bios_parser *bp, 1915 uint32_t id); 1916 static uint32_t get_ss_entry_number_from_ss_info_tbl( 1917 struct bios_parser *bp, 1918 uint32_t id); 1919 1920 /** 1921 * BiosParserObject::GetNumberofSpreadSpectrumEntry 1922 * Get Number of SpreadSpectrum Entry from the ASIC_InternalSS_Info table from 1923 * the VBIOS that match the SSid (to be converted from signal) 1924 * 1925 * @param[in] signal, ASSignalType to be converted to SSid 1926 * @return number of SS Entry that match the signal 1927 */ 1928 static uint32_t bios_parser_get_ss_entry_number( 1929 struct dc_bios *dcb, 1930 enum as_signal_type signal) 1931 { 1932 struct bios_parser *bp = BP_FROM_DCB(dcb); 1933 uint32_t ss_id = 0; 1934 ATOM_COMMON_TABLE_HEADER *header; 1935 struct atom_data_revision revision; 1936 1937 ss_id = signal_to_ss_id(signal); 1938 1939 if (!DATA_TABLES(ASIC_InternalSS_Info)) 1940 return get_ss_entry_number_from_ss_info_tbl(bp, ss_id); 1941 1942 header = GET_IMAGE(ATOM_COMMON_TABLE_HEADER, 1943 DATA_TABLES(ASIC_InternalSS_Info)); 1944 get_atom_data_table_revision(header, &revision); 1945 1946 switch (revision.major) { 1947 case 2: 1948 switch (revision.minor) { 1949 case 1: 1950 return get_ss_entry_number(bp, ss_id); 1951 default: 1952 break; 1953 } 1954 break; 1955 case 3: 1956 switch (revision.minor) { 1957 case 1: 1958 return 1959 get_ss_entry_number_from_internal_ss_info_tbl_V3_1( 1960 bp, ss_id); 1961 default: 1962 break; 1963 } 1964 break; 1965 default: 1966 break; 1967 } 1968 1969 return 0; 1970 } 1971 1972 /** 1973 * get_ss_entry_number_from_ss_info_tbl 1974 * Get Number of spread spectrum entry from the SS_Info table from the VBIOS. 1975 * 1976 * @note There can only be one entry for each id for SS_Info Table 1977 * 1978 * @param [in] id, spread spectrum id 1979 * @return number of SS Entry that match the id 1980 */ 1981 static uint32_t get_ss_entry_number_from_ss_info_tbl( 1982 struct bios_parser *bp, 1983 uint32_t id) 1984 { 1985 ATOM_SPREAD_SPECTRUM_INFO *tbl; 1986 ATOM_COMMON_TABLE_HEADER *header; 1987 uint32_t table_size; 1988 uint32_t i; 1989 uint32_t number = 0; 1990 uint32_t id_local = SS_ID_UNKNOWN; 1991 struct atom_data_revision revision; 1992 1993 /* SS_Info table exist */ 1994 if (!DATA_TABLES(SS_Info)) 1995 return number; 1996 1997 header = GET_IMAGE(ATOM_COMMON_TABLE_HEADER, 1998 DATA_TABLES(SS_Info)); 1999 get_atom_data_table_revision(header, &revision); 2000 2001 tbl = GET_IMAGE(ATOM_SPREAD_SPECTRUM_INFO, 2002 DATA_TABLES(SS_Info)); 2003 2004 if (1 != revision.major || 2 > revision.minor) 2005 return number; 2006 2007 /* have to convert from Internal_SS format to SS_Info format */ 2008 switch (id) { 2009 case ASIC_INTERNAL_SS_ON_DP: 2010 id_local = SS_ID_DP1; 2011 break; 2012 case ASIC_INTERNAL_SS_ON_LVDS: { 2013 struct embedded_panel_info panel_info; 2014 2015 if (bios_parser_get_embedded_panel_info(&bp->base, &panel_info) 2016 == BP_RESULT_OK) 2017 id_local = panel_info.ss_id; 2018 break; 2019 } 2020 default: 2021 break; 2022 } 2023 2024 if (id_local == SS_ID_UNKNOWN) 2025 return number; 2026 2027 table_size = (le16_to_cpu(tbl->sHeader.usStructureSize) - 2028 sizeof(ATOM_COMMON_TABLE_HEADER)) / 2029 sizeof(ATOM_SPREAD_SPECTRUM_ASSIGNMENT); 2030 2031 for (i = 0; i < table_size; i++) 2032 if (id_local == (uint32_t)tbl->asSS_Info[i].ucSS_Id) { 2033 number = 1; 2034 break; 2035 } 2036 2037 return number; 2038 } 2039 2040 /** 2041 * get_ss_entry_number 2042 * Get spread sprectrum information from the ASIC_InternalSS_Info Ver 2.1 or 2043 * SS_Info table from the VBIOS 2044 * There can not be more than 1 entry for ASIC_InternalSS_Info Ver 2.1 or 2045 * SS_Info. 2046 * 2047 * @param id, spread sprectrum info index 2048 * @return Bios parser result code 2049 */ 2050 static uint32_t get_ss_entry_number(struct bios_parser *bp, uint32_t id) 2051 { 2052 if (id == ASIC_INTERNAL_SS_ON_DP || id == ASIC_INTERNAL_SS_ON_LVDS) 2053 return get_ss_entry_number_from_ss_info_tbl(bp, id); 2054 2055 return get_ss_entry_number_from_internal_ss_info_tbl_v2_1(bp, id); 2056 } 2057 2058 /** 2059 * get_ss_entry_number_from_internal_ss_info_tbl_v2_1 2060 * Get NUmber of spread sprectrum entry from the ASIC_InternalSS_Info table 2061 * Ver 2.1 from the VBIOS 2062 * There will not be multiple entry for Ver 2.1 2063 * 2064 * @param id, spread sprectrum info index 2065 * @return number of SS Entry that match the id 2066 */ 2067 static uint32_t get_ss_entry_number_from_internal_ss_info_tbl_v2_1( 2068 struct bios_parser *bp, 2069 uint32_t id) 2070 { 2071 ATOM_ASIC_INTERNAL_SS_INFO_V2 *header_include; 2072 ATOM_ASIC_SS_ASSIGNMENT_V2 *tbl; 2073 uint32_t size; 2074 uint32_t i; 2075 2076 if (!DATA_TABLES(ASIC_InternalSS_Info)) 2077 return 0; 2078 2079 header_include = GET_IMAGE(ATOM_ASIC_INTERNAL_SS_INFO_V2, 2080 DATA_TABLES(ASIC_InternalSS_Info)); 2081 2082 size = (le16_to_cpu(header_include->sHeader.usStructureSize) 2083 - sizeof(ATOM_COMMON_TABLE_HEADER)) 2084 / sizeof(ATOM_ASIC_SS_ASSIGNMENT_V2); 2085 2086 tbl = (ATOM_ASIC_SS_ASSIGNMENT_V2 *) 2087 &header_include->asSpreadSpectrum[0]; 2088 for (i = 0; i < size; i++) 2089 if (tbl[i].ucClockIndication == (uint8_t)id) 2090 return 1; 2091 2092 return 0; 2093 } 2094 /** 2095 * get_ss_entry_number_from_internal_ss_info_table_V3_1 2096 * Get Number of SpreadSpectrum Entry from the ASIC_InternalSS_Info table of 2097 * the VBIOS that matches id 2098 * 2099 * @param[in] id, spread sprectrum id 2100 * @return number of SS Entry that match the id 2101 */ 2102 static uint32_t get_ss_entry_number_from_internal_ss_info_tbl_V3_1( 2103 struct bios_parser *bp, 2104 uint32_t id) 2105 { 2106 uint32_t number = 0; 2107 ATOM_ASIC_INTERNAL_SS_INFO_V3 *header_include; 2108 ATOM_ASIC_SS_ASSIGNMENT_V3 *tbl; 2109 uint32_t size; 2110 uint32_t i; 2111 2112 if (!DATA_TABLES(ASIC_InternalSS_Info)) 2113 return number; 2114 2115 header_include = GET_IMAGE(ATOM_ASIC_INTERNAL_SS_INFO_V3, 2116 DATA_TABLES(ASIC_InternalSS_Info)); 2117 size = (le16_to_cpu(header_include->sHeader.usStructureSize) - 2118 sizeof(ATOM_COMMON_TABLE_HEADER)) / 2119 sizeof(ATOM_ASIC_SS_ASSIGNMENT_V3); 2120 2121 tbl = (ATOM_ASIC_SS_ASSIGNMENT_V3 *) 2122 &header_include->asSpreadSpectrum[0]; 2123 2124 for (i = 0; i < size; i++) 2125 if (tbl[i].ucClockIndication == (uint8_t)id) 2126 number++; 2127 2128 return number; 2129 } 2130 2131 /** 2132 * bios_parser_get_gpio_pin_info 2133 * Get GpioPin information of input gpio id 2134 * 2135 * @param gpio_id, GPIO ID 2136 * @param info, GpioPin information structure 2137 * @return Bios parser result code 2138 * @note 2139 * to get the GPIO PIN INFO, we need: 2140 * 1. get the GPIO_ID from other object table, see GetHPDInfo() 2141 * 2. in DATA_TABLE.GPIO_Pin_LUT, search all records, to get the registerA 2142 * offset/mask 2143 */ 2144 static enum bp_result bios_parser_get_gpio_pin_info( 2145 struct dc_bios *dcb, 2146 uint32_t gpio_id, 2147 struct gpio_pin_info *info) 2148 { 2149 struct bios_parser *bp = BP_FROM_DCB(dcb); 2150 ATOM_GPIO_PIN_LUT *header; 2151 uint32_t count = 0; 2152 uint32_t i = 0; 2153 2154 if (!DATA_TABLES(GPIO_Pin_LUT)) 2155 return BP_RESULT_BADBIOSTABLE; 2156 2157 header = GET_IMAGE(ATOM_GPIO_PIN_LUT, DATA_TABLES(GPIO_Pin_LUT)); 2158 if (!header) 2159 return BP_RESULT_BADBIOSTABLE; 2160 2161 if (sizeof(ATOM_COMMON_TABLE_HEADER) + sizeof(ATOM_GPIO_PIN_LUT) 2162 > le16_to_cpu(header->sHeader.usStructureSize)) 2163 return BP_RESULT_BADBIOSTABLE; 2164 2165 if (1 != header->sHeader.ucTableContentRevision) 2166 return BP_RESULT_UNSUPPORTED; 2167 2168 count = (le16_to_cpu(header->sHeader.usStructureSize) 2169 - sizeof(ATOM_COMMON_TABLE_HEADER)) 2170 / sizeof(ATOM_GPIO_PIN_ASSIGNMENT); 2171 for (i = 0; i < count; ++i) { 2172 if (header->asGPIO_Pin[i].ucGPIO_ID != gpio_id) 2173 continue; 2174 2175 info->offset = 2176 (uint32_t) le16_to_cpu(header->asGPIO_Pin[i].usGpioPin_AIndex); 2177 info->offset_y = info->offset + 2; 2178 info->offset_en = info->offset + 1; 2179 info->offset_mask = info->offset - 1; 2180 2181 info->mask = (uint32_t) (1 << 2182 header->asGPIO_Pin[i].ucGpioPinBitShift); 2183 info->mask_y = info->mask + 2; 2184 info->mask_en = info->mask + 1; 2185 info->mask_mask = info->mask - 1; 2186 2187 return BP_RESULT_OK; 2188 } 2189 2190 return BP_RESULT_NORECORD; 2191 } 2192 2193 static enum bp_result get_gpio_i2c_info(struct bios_parser *bp, 2194 ATOM_I2C_RECORD *record, 2195 struct graphics_object_i2c_info *info) 2196 { 2197 ATOM_GPIO_I2C_INFO *header; 2198 uint32_t count = 0; 2199 2200 if (!info) 2201 return BP_RESULT_BADINPUT; 2202 2203 /* get the GPIO_I2C info */ 2204 if (!DATA_TABLES(GPIO_I2C_Info)) 2205 return BP_RESULT_BADBIOSTABLE; 2206 2207 header = GET_IMAGE(ATOM_GPIO_I2C_INFO, DATA_TABLES(GPIO_I2C_Info)); 2208 if (!header) 2209 return BP_RESULT_BADBIOSTABLE; 2210 2211 if (sizeof(ATOM_COMMON_TABLE_HEADER) + sizeof(ATOM_GPIO_I2C_ASSIGMENT) 2212 > le16_to_cpu(header->sHeader.usStructureSize)) 2213 return BP_RESULT_BADBIOSTABLE; 2214 2215 if (1 != header->sHeader.ucTableContentRevision) 2216 return BP_RESULT_UNSUPPORTED; 2217 2218 /* get data count */ 2219 count = (le16_to_cpu(header->sHeader.usStructureSize) 2220 - sizeof(ATOM_COMMON_TABLE_HEADER)) 2221 / sizeof(ATOM_GPIO_I2C_ASSIGMENT); 2222 if (count < record->sucI2cId.bfI2C_LineMux) 2223 return BP_RESULT_BADBIOSTABLE; 2224 2225 /* get the GPIO_I2C_INFO */ 2226 info->i2c_hw_assist = record->sucI2cId.bfHW_Capable; 2227 info->i2c_line = record->sucI2cId.bfI2C_LineMux; 2228 info->i2c_engine_id = record->sucI2cId.bfHW_EngineID; 2229 info->i2c_slave_address = record->ucI2CAddr; 2230 2231 info->gpio_info.clk_mask_register_index = 2232 le16_to_cpu(header->asGPIO_Info[info->i2c_line].usClkMaskRegisterIndex); 2233 info->gpio_info.clk_en_register_index = 2234 le16_to_cpu(header->asGPIO_Info[info->i2c_line].usClkEnRegisterIndex); 2235 info->gpio_info.clk_y_register_index = 2236 le16_to_cpu(header->asGPIO_Info[info->i2c_line].usClkY_RegisterIndex); 2237 info->gpio_info.clk_a_register_index = 2238 le16_to_cpu(header->asGPIO_Info[info->i2c_line].usClkA_RegisterIndex); 2239 info->gpio_info.data_mask_register_index = 2240 le16_to_cpu(header->asGPIO_Info[info->i2c_line].usDataMaskRegisterIndex); 2241 info->gpio_info.data_en_register_index = 2242 le16_to_cpu(header->asGPIO_Info[info->i2c_line].usDataEnRegisterIndex); 2243 info->gpio_info.data_y_register_index = 2244 le16_to_cpu(header->asGPIO_Info[info->i2c_line].usDataY_RegisterIndex); 2245 info->gpio_info.data_a_register_index = 2246 le16_to_cpu(header->asGPIO_Info[info->i2c_line].usDataA_RegisterIndex); 2247 2248 info->gpio_info.clk_mask_shift = 2249 header->asGPIO_Info[info->i2c_line].ucClkMaskShift; 2250 info->gpio_info.clk_en_shift = 2251 header->asGPIO_Info[info->i2c_line].ucClkEnShift; 2252 info->gpio_info.clk_y_shift = 2253 header->asGPIO_Info[info->i2c_line].ucClkY_Shift; 2254 info->gpio_info.clk_a_shift = 2255 header->asGPIO_Info[info->i2c_line].ucClkA_Shift; 2256 info->gpio_info.data_mask_shift = 2257 header->asGPIO_Info[info->i2c_line].ucDataMaskShift; 2258 info->gpio_info.data_en_shift = 2259 header->asGPIO_Info[info->i2c_line].ucDataEnShift; 2260 info->gpio_info.data_y_shift = 2261 header->asGPIO_Info[info->i2c_line].ucDataY_Shift; 2262 info->gpio_info.data_a_shift = 2263 header->asGPIO_Info[info->i2c_line].ucDataA_Shift; 2264 2265 return BP_RESULT_OK; 2266 } 2267 2268 static bool dal_graphics_object_id_is_valid(struct graphics_object_id id) 2269 { 2270 bool rc = true; 2271 2272 switch (id.type) { 2273 case OBJECT_TYPE_UNKNOWN: 2274 rc = false; 2275 break; 2276 case OBJECT_TYPE_GPU: 2277 case OBJECT_TYPE_ENGINE: 2278 /* do NOT check for id.id == 0 */ 2279 if (id.enum_id == ENUM_ID_UNKNOWN) 2280 rc = false; 2281 break; 2282 default: 2283 if (id.id == 0 || id.enum_id == ENUM_ID_UNKNOWN) 2284 rc = false; 2285 break; 2286 } 2287 2288 return rc; 2289 } 2290 2291 static bool dal_graphics_object_id_is_equal( 2292 struct graphics_object_id id1, 2293 struct graphics_object_id id2) 2294 { 2295 if (false == dal_graphics_object_id_is_valid(id1)) { 2296 dm_output_to_console( 2297 "%s: Warning: comparing invalid object 'id1'!\n", __func__); 2298 return false; 2299 } 2300 2301 if (false == dal_graphics_object_id_is_valid(id2)) { 2302 dm_output_to_console( 2303 "%s: Warning: comparing invalid object 'id2'!\n", __func__); 2304 return false; 2305 } 2306 2307 if (id1.id == id2.id && id1.enum_id == id2.enum_id 2308 && id1.type == id2.type) 2309 return true; 2310 2311 return false; 2312 } 2313 2314 static ATOM_OBJECT *get_bios_object(struct bios_parser *bp, 2315 struct graphics_object_id id) 2316 { 2317 uint32_t offset; 2318 ATOM_OBJECT_TABLE *tbl; 2319 uint32_t i; 2320 2321 switch (id.type) { 2322 case OBJECT_TYPE_ENCODER: 2323 offset = le16_to_cpu(bp->object_info_tbl.v1_1->usEncoderObjectTableOffset); 2324 break; 2325 2326 case OBJECT_TYPE_CONNECTOR: 2327 offset = le16_to_cpu(bp->object_info_tbl.v1_1->usConnectorObjectTableOffset); 2328 break; 2329 2330 case OBJECT_TYPE_ROUTER: 2331 offset = le16_to_cpu(bp->object_info_tbl.v1_1->usRouterObjectTableOffset); 2332 break; 2333 2334 case OBJECT_TYPE_GENERIC: 2335 if (bp->object_info_tbl.revision.minor < 3) 2336 return NULL; 2337 offset = le16_to_cpu(bp->object_info_tbl.v1_3->usMiscObjectTableOffset); 2338 break; 2339 2340 default: 2341 return NULL; 2342 } 2343 2344 offset += bp->object_info_tbl_offset; 2345 2346 tbl = GET_IMAGE(ATOM_OBJECT_TABLE, offset); 2347 if (!tbl) 2348 return NULL; 2349 2350 for (i = 0; i < tbl->ucNumberOfObjects; i++) 2351 if (dal_graphics_object_id_is_equal(id, 2352 object_id_from_bios_object_id( 2353 le16_to_cpu(tbl->asObjects[i].usObjectID)))) 2354 return &tbl->asObjects[i]; 2355 2356 return NULL; 2357 } 2358 2359 static uint32_t get_dest_obj_list(struct bios_parser *bp, 2360 ATOM_OBJECT *object, uint16_t **id_list) 2361 { 2362 uint32_t offset; 2363 uint8_t *number; 2364 2365 if (!object) { 2366 BREAK_TO_DEBUGGER(); /* Invalid object id */ 2367 return 0; 2368 } 2369 2370 offset = le16_to_cpu(object->usSrcDstTableOffset) 2371 + bp->object_info_tbl_offset; 2372 2373 number = GET_IMAGE(uint8_t, offset); 2374 if (!number) 2375 return 0; 2376 2377 offset += sizeof(uint8_t); 2378 offset += sizeof(uint16_t) * (*number); 2379 2380 number = GET_IMAGE(uint8_t, offset); 2381 if ((!number) || (!*number)) 2382 return 0; 2383 2384 offset += sizeof(uint8_t); 2385 *id_list = (uint16_t *)bios_get_image(&bp->base, offset, *number * sizeof(uint16_t)); 2386 2387 if (!*id_list) 2388 return 0; 2389 2390 return *number; 2391 } 2392 2393 static uint32_t get_src_obj_list(struct bios_parser *bp, ATOM_OBJECT *object, 2394 uint16_t **id_list) 2395 { 2396 uint32_t offset; 2397 uint8_t *number; 2398 2399 if (!object) { 2400 BREAK_TO_DEBUGGER(); /* Invalid object id */ 2401 return 0; 2402 } 2403 2404 offset = le16_to_cpu(object->usSrcDstTableOffset) 2405 + bp->object_info_tbl_offset; 2406 2407 number = GET_IMAGE(uint8_t, offset); 2408 if (!number) 2409 return 0; 2410 2411 offset += sizeof(uint8_t); 2412 *id_list = (uint16_t *)bios_get_image(&bp->base, offset, *number * sizeof(uint16_t)); 2413 2414 if (!*id_list) 2415 return 0; 2416 2417 return *number; 2418 } 2419 2420 static uint32_t get_dst_number_from_object(struct bios_parser *bp, 2421 ATOM_OBJECT *object) 2422 { 2423 uint32_t offset; 2424 uint8_t *number; 2425 2426 if (!object) { 2427 BREAK_TO_DEBUGGER(); /* Invalid encoder object id*/ 2428 return 0; 2429 } 2430 2431 offset = le16_to_cpu(object->usSrcDstTableOffset) 2432 + bp->object_info_tbl_offset; 2433 2434 number = GET_IMAGE(uint8_t, offset); 2435 if (!number) 2436 return 0; 2437 2438 offset += sizeof(uint8_t); 2439 offset += sizeof(uint16_t) * (*number); 2440 2441 number = GET_IMAGE(uint8_t, offset); 2442 2443 if (!number) 2444 return 0; 2445 2446 return *number; 2447 } 2448 2449 static struct device_id device_type_from_device_id(uint16_t device_id) 2450 { 2451 2452 struct device_id result_device_id; 2453 2454 switch (device_id) { 2455 case ATOM_DEVICE_LCD1_SUPPORT: 2456 result_device_id.device_type = DEVICE_TYPE_LCD; 2457 result_device_id.enum_id = 1; 2458 break; 2459 2460 case ATOM_DEVICE_LCD2_SUPPORT: 2461 result_device_id.device_type = DEVICE_TYPE_LCD; 2462 result_device_id.enum_id = 2; 2463 break; 2464 2465 case ATOM_DEVICE_CRT1_SUPPORT: 2466 result_device_id.device_type = DEVICE_TYPE_CRT; 2467 result_device_id.enum_id = 1; 2468 break; 2469 2470 case ATOM_DEVICE_CRT2_SUPPORT: 2471 result_device_id.device_type = DEVICE_TYPE_CRT; 2472 result_device_id.enum_id = 2; 2473 break; 2474 2475 case ATOM_DEVICE_DFP1_SUPPORT: 2476 result_device_id.device_type = DEVICE_TYPE_DFP; 2477 result_device_id.enum_id = 1; 2478 break; 2479 2480 case ATOM_DEVICE_DFP2_SUPPORT: 2481 result_device_id.device_type = DEVICE_TYPE_DFP; 2482 result_device_id.enum_id = 2; 2483 break; 2484 2485 case ATOM_DEVICE_DFP3_SUPPORT: 2486 result_device_id.device_type = DEVICE_TYPE_DFP; 2487 result_device_id.enum_id = 3; 2488 break; 2489 2490 case ATOM_DEVICE_DFP4_SUPPORT: 2491 result_device_id.device_type = DEVICE_TYPE_DFP; 2492 result_device_id.enum_id = 4; 2493 break; 2494 2495 case ATOM_DEVICE_DFP5_SUPPORT: 2496 result_device_id.device_type = DEVICE_TYPE_DFP; 2497 result_device_id.enum_id = 5; 2498 break; 2499 2500 case ATOM_DEVICE_DFP6_SUPPORT: 2501 result_device_id.device_type = DEVICE_TYPE_DFP; 2502 result_device_id.enum_id = 6; 2503 break; 2504 2505 default: 2506 BREAK_TO_DEBUGGER(); /* Invalid device Id */ 2507 result_device_id.device_type = DEVICE_TYPE_UNKNOWN; 2508 result_device_id.enum_id = 0; 2509 } 2510 return result_device_id; 2511 } 2512 2513 static void get_atom_data_table_revision( 2514 ATOM_COMMON_TABLE_HEADER *atom_data_tbl, 2515 struct atom_data_revision *tbl_revision) 2516 { 2517 if (!tbl_revision) 2518 return; 2519 2520 /* initialize the revision to 0 which is invalid revision */ 2521 tbl_revision->major = 0; 2522 tbl_revision->minor = 0; 2523 2524 if (!atom_data_tbl) 2525 return; 2526 2527 tbl_revision->major = 2528 (uint32_t) GET_DATA_TABLE_MAJOR_REVISION(atom_data_tbl); 2529 tbl_revision->minor = 2530 (uint32_t) GET_DATA_TABLE_MINOR_REVISION(atom_data_tbl); 2531 } 2532 2533 static uint32_t signal_to_ss_id(enum as_signal_type signal) 2534 { 2535 uint32_t clk_id_ss = 0; 2536 2537 switch (signal) { 2538 case AS_SIGNAL_TYPE_DVI: 2539 clk_id_ss = ASIC_INTERNAL_SS_ON_TMDS; 2540 break; 2541 case AS_SIGNAL_TYPE_HDMI: 2542 clk_id_ss = ASIC_INTERNAL_SS_ON_HDMI; 2543 break; 2544 case AS_SIGNAL_TYPE_LVDS: 2545 clk_id_ss = ASIC_INTERNAL_SS_ON_LVDS; 2546 break; 2547 case AS_SIGNAL_TYPE_DISPLAY_PORT: 2548 clk_id_ss = ASIC_INTERNAL_SS_ON_DP; 2549 break; 2550 case AS_SIGNAL_TYPE_GPU_PLL: 2551 clk_id_ss = ASIC_INTERNAL_GPUPLL_SS; 2552 break; 2553 default: 2554 break; 2555 } 2556 return clk_id_ss; 2557 } 2558 2559 static uint32_t get_support_mask_for_device_id(struct device_id device_id) 2560 { 2561 enum dal_device_type device_type = device_id.device_type; 2562 uint32_t enum_id = device_id.enum_id; 2563 2564 switch (device_type) { 2565 case DEVICE_TYPE_LCD: 2566 switch (enum_id) { 2567 case 1: 2568 return ATOM_DEVICE_LCD1_SUPPORT; 2569 case 2: 2570 return ATOM_DEVICE_LCD2_SUPPORT; 2571 default: 2572 break; 2573 } 2574 break; 2575 case DEVICE_TYPE_CRT: 2576 switch (enum_id) { 2577 case 1: 2578 return ATOM_DEVICE_CRT1_SUPPORT; 2579 case 2: 2580 return ATOM_DEVICE_CRT2_SUPPORT; 2581 default: 2582 break; 2583 } 2584 break; 2585 case DEVICE_TYPE_DFP: 2586 switch (enum_id) { 2587 case 1: 2588 return ATOM_DEVICE_DFP1_SUPPORT; 2589 case 2: 2590 return ATOM_DEVICE_DFP2_SUPPORT; 2591 case 3: 2592 return ATOM_DEVICE_DFP3_SUPPORT; 2593 case 4: 2594 return ATOM_DEVICE_DFP4_SUPPORT; 2595 case 5: 2596 return ATOM_DEVICE_DFP5_SUPPORT; 2597 case 6: 2598 return ATOM_DEVICE_DFP6_SUPPORT; 2599 default: 2600 break; 2601 } 2602 break; 2603 case DEVICE_TYPE_CV: 2604 switch (enum_id) { 2605 case 1: 2606 return ATOM_DEVICE_CV_SUPPORT; 2607 default: 2608 break; 2609 } 2610 break; 2611 case DEVICE_TYPE_TV: 2612 switch (enum_id) { 2613 case 1: 2614 return ATOM_DEVICE_TV1_SUPPORT; 2615 default: 2616 break; 2617 } 2618 break; 2619 default: 2620 break; 2621 }; 2622 2623 /* Unidentified device ID, return empty support mask. */ 2624 return 0; 2625 } 2626 2627 /** 2628 * HwContext interface for writing MM registers 2629 */ 2630 2631 static bool i2c_read( 2632 struct bios_parser *bp, 2633 struct graphics_object_i2c_info *i2c_info, 2634 uint8_t *buffer, 2635 uint32_t length) 2636 { 2637 struct ddc *ddc; 2638 uint8_t offset[2] = { 0, 0 }; 2639 bool result = false; 2640 struct i2c_command cmd; 2641 struct gpio_ddc_hw_info hw_info = { 2642 i2c_info->i2c_hw_assist, 2643 i2c_info->i2c_line }; 2644 2645 ddc = dal_gpio_create_ddc(bp->base.ctx->gpio_service, 2646 i2c_info->gpio_info.clk_a_register_index, 2647 (1 << i2c_info->gpio_info.clk_a_shift), &hw_info); 2648 2649 if (!ddc) 2650 return result; 2651 2652 /*Using SW engine */ 2653 cmd.engine = I2C_COMMAND_ENGINE_SW; 2654 cmd.speed = ddc->ctx->dc->caps.i2c_speed_in_khz; 2655 2656 { 2657 struct i2c_payload payloads[] = { 2658 { 2659 .address = i2c_info->i2c_slave_address >> 1, 2660 .data = offset, 2661 .length = sizeof(offset), 2662 .write = true 2663 }, 2664 { 2665 .address = i2c_info->i2c_slave_address >> 1, 2666 .data = buffer, 2667 .length = length, 2668 .write = false 2669 } 2670 }; 2671 2672 cmd.payloads = payloads; 2673 cmd.number_of_payloads = ARRAY_SIZE(payloads); 2674 2675 /* TODO route this through drm i2c_adapter */ 2676 result = dal_i2caux_submit_i2c_command( 2677 ddc->ctx->i2caux, 2678 ddc, 2679 &cmd); 2680 } 2681 2682 dal_gpio_destroy_ddc(&ddc); 2683 2684 return result; 2685 } 2686 2687 /** 2688 * Read external display connection info table through i2c. 2689 * validate the GUID and checksum. 2690 * 2691 * @return enum bp_result whether all data was sucessfully read 2692 */ 2693 static enum bp_result get_ext_display_connection_info( 2694 struct bios_parser *bp, 2695 ATOM_OBJECT *opm_object, 2696 ATOM_EXTERNAL_DISPLAY_CONNECTION_INFO *ext_display_connection_info_tbl) 2697 { 2698 bool config_tbl_present = false; 2699 ATOM_I2C_RECORD *i2c_record = NULL; 2700 uint32_t i = 0; 2701 2702 if (opm_object == NULL) 2703 return BP_RESULT_BADINPUT; 2704 2705 i2c_record = get_i2c_record(bp, opm_object); 2706 2707 if (i2c_record != NULL) { 2708 ATOM_GPIO_I2C_INFO *gpio_i2c_header; 2709 struct graphics_object_i2c_info i2c_info; 2710 2711 gpio_i2c_header = GET_IMAGE(ATOM_GPIO_I2C_INFO, 2712 bp->master_data_tbl->ListOfDataTables.GPIO_I2C_Info); 2713 2714 if (NULL == gpio_i2c_header) 2715 return BP_RESULT_BADBIOSTABLE; 2716 2717 if (get_gpio_i2c_info(bp, i2c_record, &i2c_info) != 2718 BP_RESULT_OK) 2719 return BP_RESULT_BADBIOSTABLE; 2720 2721 if (i2c_read(bp, 2722 &i2c_info, 2723 (uint8_t *)ext_display_connection_info_tbl, 2724 sizeof(ATOM_EXTERNAL_DISPLAY_CONNECTION_INFO))) { 2725 config_tbl_present = true; 2726 } 2727 } 2728 2729 /* Validate GUID */ 2730 if (config_tbl_present) 2731 for (i = 0; i < NUMBER_OF_UCHAR_FOR_GUID; i++) { 2732 if (ext_display_connection_info_tbl->ucGuid[i] 2733 != ext_display_connection_guid[i]) { 2734 config_tbl_present = false; 2735 break; 2736 } 2737 } 2738 2739 /* Validate checksum */ 2740 if (config_tbl_present) { 2741 uint8_t check_sum = 0; 2742 uint8_t *buf = 2743 (uint8_t *)ext_display_connection_info_tbl; 2744 2745 for (i = 0; i < sizeof(ATOM_EXTERNAL_DISPLAY_CONNECTION_INFO); 2746 i++) { 2747 check_sum += buf[i]; 2748 } 2749 2750 if (check_sum != 0) 2751 config_tbl_present = false; 2752 } 2753 2754 if (config_tbl_present) 2755 return BP_RESULT_OK; 2756 else 2757 return BP_RESULT_FAILURE; 2758 } 2759 2760 /* 2761 * Gets the first device ID in the same group as the given ID for enumerating. 2762 * For instance, if any DFP device ID is passed, returns the device ID for DFP1. 2763 * 2764 * The first device ID in the same group as the passed device ID, or 0 if no 2765 * matching device group found. 2766 */ 2767 static uint32_t enum_first_device_id(uint32_t dev_id) 2768 { 2769 /* Return the first in the group that this ID belongs to. */ 2770 if (dev_id & ATOM_DEVICE_CRT_SUPPORT) 2771 return ATOM_DEVICE_CRT1_SUPPORT; 2772 else if (dev_id & ATOM_DEVICE_DFP_SUPPORT) 2773 return ATOM_DEVICE_DFP1_SUPPORT; 2774 else if (dev_id & ATOM_DEVICE_LCD_SUPPORT) 2775 return ATOM_DEVICE_LCD1_SUPPORT; 2776 else if (dev_id & ATOM_DEVICE_TV_SUPPORT) 2777 return ATOM_DEVICE_TV1_SUPPORT; 2778 else if (dev_id & ATOM_DEVICE_CV_SUPPORT) 2779 return ATOM_DEVICE_CV_SUPPORT; 2780 2781 /* No group found for this device ID. */ 2782 2783 dm_error("%s: incorrect input %d\n", __func__, dev_id); 2784 /* No matching support flag for given device ID */ 2785 return 0; 2786 } 2787 2788 /* 2789 * Gets the next device ID in the group for a given device ID. 2790 * 2791 * The current device ID being enumerated on. 2792 * 2793 * The next device ID in the group, or 0 if no device exists. 2794 */ 2795 static uint32_t enum_next_dev_id(uint32_t dev_id) 2796 { 2797 /* Get next device ID in the group. */ 2798 switch (dev_id) { 2799 case ATOM_DEVICE_CRT1_SUPPORT: 2800 return ATOM_DEVICE_CRT2_SUPPORT; 2801 case ATOM_DEVICE_LCD1_SUPPORT: 2802 return ATOM_DEVICE_LCD2_SUPPORT; 2803 case ATOM_DEVICE_DFP1_SUPPORT: 2804 return ATOM_DEVICE_DFP2_SUPPORT; 2805 case ATOM_DEVICE_DFP2_SUPPORT: 2806 return ATOM_DEVICE_DFP3_SUPPORT; 2807 case ATOM_DEVICE_DFP3_SUPPORT: 2808 return ATOM_DEVICE_DFP4_SUPPORT; 2809 case ATOM_DEVICE_DFP4_SUPPORT: 2810 return ATOM_DEVICE_DFP5_SUPPORT; 2811 case ATOM_DEVICE_DFP5_SUPPORT: 2812 return ATOM_DEVICE_DFP6_SUPPORT; 2813 } 2814 2815 /* Done enumerating through devices. */ 2816 return 0; 2817 } 2818 2819 /* 2820 * Returns the new device tag record for patched BIOS object. 2821 * 2822 * [IN] pExtDisplayPath - External display path to copy device tag from. 2823 * [IN] deviceSupport - Bit vector for device ID support flags. 2824 * [OUT] pDeviceTag - Device tag structure to fill with patched data. 2825 * 2826 * True if a compatible device ID was found, false otherwise. 2827 */ 2828 static bool get_patched_device_tag( 2829 struct bios_parser *bp, 2830 EXT_DISPLAY_PATH *ext_display_path, 2831 uint32_t device_support, 2832 ATOM_CONNECTOR_DEVICE_TAG *device_tag) 2833 { 2834 uint32_t dev_id; 2835 /* Use fallback behaviour if not supported. */ 2836 if (!bp->remap_device_tags) { 2837 device_tag->ulACPIDeviceEnum = 2838 cpu_to_le32((uint32_t) le16_to_cpu(ext_display_path->usDeviceACPIEnum)); 2839 device_tag->usDeviceID = 2840 cpu_to_le16(le16_to_cpu(ext_display_path->usDeviceTag)); 2841 return true; 2842 } 2843 2844 /* Find the first unused in the same group. */ 2845 dev_id = enum_first_device_id(le16_to_cpu(ext_display_path->usDeviceTag)); 2846 while (dev_id != 0) { 2847 /* Assign this device ID if supported. */ 2848 if ((device_support & dev_id) != 0) { 2849 device_tag->ulACPIDeviceEnum = 2850 cpu_to_le32((uint32_t) le16_to_cpu(ext_display_path->usDeviceACPIEnum)); 2851 device_tag->usDeviceID = cpu_to_le16((USHORT) dev_id); 2852 return true; 2853 } 2854 2855 dev_id = enum_next_dev_id(dev_id); 2856 } 2857 2858 /* No compatible device ID found. */ 2859 return false; 2860 } 2861 2862 /* 2863 * Adds a device tag to a BIOS object's device tag record if there is 2864 * matching device ID supported. 2865 * 2866 * pObject - Pointer to the BIOS object to add the device tag to. 2867 * pExtDisplayPath - Display path to retrieve base device ID from. 2868 * pDeviceSupport - Pointer to bit vector for supported device IDs. 2869 */ 2870 static void add_device_tag_from_ext_display_path( 2871 struct bios_parser *bp, 2872 ATOM_OBJECT *object, 2873 EXT_DISPLAY_PATH *ext_display_path, 2874 uint32_t *device_support) 2875 { 2876 /* Get device tag record for object. */ 2877 ATOM_CONNECTOR_DEVICE_TAG *device_tag = NULL; 2878 ATOM_CONNECTOR_DEVICE_TAG_RECORD *device_tag_record = NULL; 2879 enum bp_result result = 2880 bios_parser_get_device_tag_record( 2881 bp, object, &device_tag_record); 2882 2883 if ((le16_to_cpu(ext_display_path->usDeviceTag) != CONNECTOR_OBJECT_ID_NONE) 2884 && (result == BP_RESULT_OK)) { 2885 uint8_t index; 2886 2887 if ((device_tag_record->ucNumberOfDevice == 1) && 2888 (le16_to_cpu(device_tag_record->asDeviceTag[0].usDeviceID) == 0)) { 2889 /*Workaround bug in current VBIOS releases where 2890 * ucNumberOfDevice = 1 but there is no actual device 2891 * tag data. This w/a is temporary until the updated 2892 * VBIOS is distributed. */ 2893 device_tag_record->ucNumberOfDevice = 2894 device_tag_record->ucNumberOfDevice - 1; 2895 } 2896 2897 /* Attempt to find a matching device ID. */ 2898 index = device_tag_record->ucNumberOfDevice; 2899 device_tag = &device_tag_record->asDeviceTag[index]; 2900 if (get_patched_device_tag( 2901 bp, 2902 ext_display_path, 2903 *device_support, 2904 device_tag)) { 2905 /* Update cached device support to remove assigned ID. 2906 */ 2907 *device_support &= ~le16_to_cpu(device_tag->usDeviceID); 2908 device_tag_record->ucNumberOfDevice++; 2909 } 2910 } 2911 } 2912 2913 /* 2914 * Read out a single EXT_DISPLAY_PATH from the external display connection info 2915 * table. The specific entry in the table is determined by the enum_id passed 2916 * in. 2917 * 2918 * EXT_DISPLAY_PATH describing a single Configuration table entry 2919 */ 2920 2921 #define INVALID_CONNECTOR 0xffff 2922 2923 static EXT_DISPLAY_PATH *get_ext_display_path_entry( 2924 ATOM_EXTERNAL_DISPLAY_CONNECTION_INFO *config_table, 2925 uint32_t bios_object_id) 2926 { 2927 EXT_DISPLAY_PATH *ext_display_path; 2928 uint32_t ext_display_path_index = 2929 ((bios_object_id & ENUM_ID_MASK) >> ENUM_ID_SHIFT) - 1; 2930 2931 if (ext_display_path_index >= MAX_NUMBER_OF_EXT_DISPLAY_PATH) 2932 return NULL; 2933 2934 ext_display_path = &config_table->sPath[ext_display_path_index]; 2935 2936 if (le16_to_cpu(ext_display_path->usDeviceConnector) == INVALID_CONNECTOR) 2937 ext_display_path->usDeviceConnector = cpu_to_le16(0); 2938 2939 return ext_display_path; 2940 } 2941 2942 /* 2943 * Get AUX/DDC information of input object id 2944 * 2945 * search all records to find the ATOM_CONNECTOR_AUXDDC_LUT_RECORD_TYPE record 2946 * IR 2947 */ 2948 static ATOM_CONNECTOR_AUXDDC_LUT_RECORD *get_ext_connector_aux_ddc_lut_record( 2949 struct bios_parser *bp, 2950 ATOM_OBJECT *object) 2951 { 2952 uint32_t offset; 2953 ATOM_COMMON_RECORD_HEADER *header; 2954 2955 if (!object) { 2956 BREAK_TO_DEBUGGER(); 2957 /* Invalid object */ 2958 return NULL; 2959 } 2960 2961 offset = le16_to_cpu(object->usRecordOffset) 2962 + bp->object_info_tbl_offset; 2963 2964 for (;;) { 2965 header = GET_IMAGE(ATOM_COMMON_RECORD_HEADER, offset); 2966 2967 if (!header) 2968 return NULL; 2969 2970 if (LAST_RECORD_TYPE == header->ucRecordType || 2971 0 == header->ucRecordSize) 2972 break; 2973 2974 if (ATOM_CONNECTOR_AUXDDC_LUT_RECORD_TYPE == 2975 header->ucRecordType && 2976 sizeof(ATOM_CONNECTOR_AUXDDC_LUT_RECORD) <= 2977 header->ucRecordSize) 2978 return (ATOM_CONNECTOR_AUXDDC_LUT_RECORD *)(header); 2979 2980 offset += header->ucRecordSize; 2981 } 2982 2983 return NULL; 2984 } 2985 2986 /* 2987 * Get AUX/DDC information of input object id 2988 * 2989 * search all records to find the ATOM_CONNECTOR_AUXDDC_LUT_RECORD_TYPE record 2990 * IR 2991 */ 2992 static ATOM_CONNECTOR_HPDPIN_LUT_RECORD *get_ext_connector_hpd_pin_lut_record( 2993 struct bios_parser *bp, 2994 ATOM_OBJECT *object) 2995 { 2996 uint32_t offset; 2997 ATOM_COMMON_RECORD_HEADER *header; 2998 2999 if (!object) { 3000 BREAK_TO_DEBUGGER(); 3001 /* Invalid object */ 3002 return NULL; 3003 } 3004 3005 offset = le16_to_cpu(object->usRecordOffset) 3006 + bp->object_info_tbl_offset; 3007 3008 for (;;) { 3009 header = GET_IMAGE(ATOM_COMMON_RECORD_HEADER, offset); 3010 3011 if (!header) 3012 return NULL; 3013 3014 if (LAST_RECORD_TYPE == header->ucRecordType || 3015 0 == header->ucRecordSize) 3016 break; 3017 3018 if (ATOM_CONNECTOR_HPDPIN_LUT_RECORD_TYPE == 3019 header->ucRecordType && 3020 sizeof(ATOM_CONNECTOR_HPDPIN_LUT_RECORD) <= 3021 header->ucRecordSize) 3022 return (ATOM_CONNECTOR_HPDPIN_LUT_RECORD *)header; 3023 3024 offset += header->ucRecordSize; 3025 } 3026 3027 return NULL; 3028 } 3029 3030 /* 3031 * Check whether we need to patch the VBIOS connector info table with 3032 * data from an external display connection info table. This is 3033 * necessary to support MXM boards with an OPM (output personality 3034 * module). With these designs, the VBIOS connector info table 3035 * specifies an MXM_CONNECTOR with a unique ID. The driver retrieves 3036 * the external connection info table through i2c and then looks up the 3037 * connector ID to find the real connector type (e.g. DFP1). 3038 * 3039 */ 3040 static enum bp_result patch_bios_image_from_ext_display_connection_info( 3041 struct bios_parser *bp) 3042 { 3043 ATOM_OBJECT_TABLE *connector_tbl; 3044 uint32_t connector_tbl_offset; 3045 struct graphics_object_id object_id; 3046 ATOM_OBJECT *object; 3047 ATOM_EXTERNAL_DISPLAY_CONNECTION_INFO ext_display_connection_info_tbl; 3048 EXT_DISPLAY_PATH *ext_display_path; 3049 ATOM_CONNECTOR_AUXDDC_LUT_RECORD *aux_ddc_lut_record = NULL; 3050 ATOM_I2C_RECORD *i2c_record = NULL; 3051 ATOM_CONNECTOR_HPDPIN_LUT_RECORD *hpd_pin_lut_record = NULL; 3052 ATOM_HPD_INT_RECORD *hpd_record = NULL; 3053 ATOM_OBJECT_TABLE *encoder_table; 3054 uint32_t encoder_table_offset; 3055 ATOM_OBJECT *opm_object = NULL; 3056 uint32_t i = 0; 3057 struct graphics_object_id opm_object_id = 3058 dal_graphics_object_id_init( 3059 GENERIC_ID_MXM_OPM, 3060 ENUM_ID_1, 3061 OBJECT_TYPE_GENERIC); 3062 ATOM_CONNECTOR_DEVICE_TAG_RECORD *dev_tag_record; 3063 uint32_t cached_device_support = 3064 le16_to_cpu(bp->object_info_tbl.v1_1->usDeviceSupport); 3065 3066 uint32_t dst_number; 3067 uint16_t *dst_object_id_list; 3068 3069 opm_object = get_bios_object(bp, opm_object_id); 3070 if (!opm_object) 3071 return BP_RESULT_UNSUPPORTED; 3072 3073 memset(&ext_display_connection_info_tbl, 0, 3074 sizeof(ATOM_EXTERNAL_DISPLAY_CONNECTION_INFO)); 3075 3076 connector_tbl_offset = bp->object_info_tbl_offset 3077 + le16_to_cpu(bp->object_info_tbl.v1_1->usConnectorObjectTableOffset); 3078 connector_tbl = GET_IMAGE(ATOM_OBJECT_TABLE, connector_tbl_offset); 3079 3080 /* Read Connector info table from EEPROM through i2c */ 3081 if (get_ext_display_connection_info(bp, 3082 opm_object, 3083 &ext_display_connection_info_tbl) != BP_RESULT_OK) { 3084 3085 DC_LOG_WARNING("%s: Failed to read Connection Info Table", __func__); 3086 return BP_RESULT_UNSUPPORTED; 3087 } 3088 3089 /* Get pointer to AUX/DDC and HPD LUTs */ 3090 aux_ddc_lut_record = 3091 get_ext_connector_aux_ddc_lut_record(bp, opm_object); 3092 hpd_pin_lut_record = 3093 get_ext_connector_hpd_pin_lut_record(bp, opm_object); 3094 3095 if ((aux_ddc_lut_record == NULL) || (hpd_pin_lut_record == NULL)) 3096 return BP_RESULT_UNSUPPORTED; 3097 3098 /* Cache support bits for currently unmapped device types. */ 3099 if (bp->remap_device_tags) { 3100 for (i = 0; i < connector_tbl->ucNumberOfObjects; ++i) { 3101 uint32_t j; 3102 /* Remove support for all non-MXM connectors. */ 3103 object = &connector_tbl->asObjects[i]; 3104 object_id = object_id_from_bios_object_id( 3105 le16_to_cpu(object->usObjectID)); 3106 if ((OBJECT_TYPE_CONNECTOR != object_id.type) || 3107 (CONNECTOR_ID_MXM == object_id.id)) 3108 continue; 3109 3110 /* Remove support for all device tags. */ 3111 if (bios_parser_get_device_tag_record( 3112 bp, object, &dev_tag_record) != BP_RESULT_OK) 3113 continue; 3114 3115 for (j = 0; j < dev_tag_record->ucNumberOfDevice; ++j) { 3116 ATOM_CONNECTOR_DEVICE_TAG *device_tag = 3117 &dev_tag_record->asDeviceTag[j]; 3118 cached_device_support &= 3119 ~le16_to_cpu(device_tag->usDeviceID); 3120 } 3121 } 3122 } 3123 3124 /* Find all MXM connector objects and patch them with connector info 3125 * from the external display connection info table. */ 3126 for (i = 0; i < connector_tbl->ucNumberOfObjects; i++) { 3127 uint32_t j; 3128 3129 object = &connector_tbl->asObjects[i]; 3130 object_id = object_id_from_bios_object_id(le16_to_cpu(object->usObjectID)); 3131 if ((OBJECT_TYPE_CONNECTOR != object_id.type) || 3132 (CONNECTOR_ID_MXM != object_id.id)) 3133 continue; 3134 3135 /* Get the correct connection info table entry based on the enum 3136 * id. */ 3137 ext_display_path = get_ext_display_path_entry( 3138 &ext_display_connection_info_tbl, 3139 le16_to_cpu(object->usObjectID)); 3140 if (!ext_display_path) 3141 return BP_RESULT_FAILURE; 3142 3143 /* Patch device connector ID */ 3144 object->usObjectID = 3145 cpu_to_le16(le16_to_cpu(ext_display_path->usDeviceConnector)); 3146 3147 /* Patch device tag, ulACPIDeviceEnum. */ 3148 add_device_tag_from_ext_display_path( 3149 bp, 3150 object, 3151 ext_display_path, 3152 &cached_device_support); 3153 3154 /* Patch HPD info */ 3155 if (ext_display_path->ucExtHPDPINLutIndex < 3156 MAX_NUMBER_OF_EXT_HPDPIN_LUT_ENTRIES) { 3157 hpd_record = get_hpd_record(bp, object); 3158 if (hpd_record) { 3159 uint8_t index = 3160 ext_display_path->ucExtHPDPINLutIndex; 3161 hpd_record->ucHPDIntGPIOID = 3162 hpd_pin_lut_record->ucHPDPINMap[index]; 3163 } else { 3164 BREAK_TO_DEBUGGER(); 3165 /* Invalid hpd record */ 3166 return BP_RESULT_FAILURE; 3167 } 3168 } 3169 3170 /* Patch I2C/AUX info */ 3171 if (ext_display_path->ucExtHPDPINLutIndex < 3172 MAX_NUMBER_OF_EXT_AUXDDC_LUT_ENTRIES) { 3173 i2c_record = get_i2c_record(bp, object); 3174 if (i2c_record) { 3175 uint8_t index = 3176 ext_display_path->ucExtAUXDDCLutIndex; 3177 i2c_record->sucI2cId = 3178 aux_ddc_lut_record->ucAUXDDCMap[index]; 3179 } else { 3180 BREAK_TO_DEBUGGER(); 3181 /* Invalid I2C record */ 3182 return BP_RESULT_FAILURE; 3183 } 3184 } 3185 3186 /* Merge with other MXM connectors that map to the same physical 3187 * connector. */ 3188 for (j = i + 1; 3189 j < connector_tbl->ucNumberOfObjects; j++) { 3190 ATOM_OBJECT *next_object; 3191 struct graphics_object_id next_object_id; 3192 EXT_DISPLAY_PATH *next_ext_display_path; 3193 3194 next_object = &connector_tbl->asObjects[j]; 3195 next_object_id = object_id_from_bios_object_id( 3196 le16_to_cpu(next_object->usObjectID)); 3197 3198 if ((OBJECT_TYPE_CONNECTOR != next_object_id.type) && 3199 (CONNECTOR_ID_MXM == next_object_id.id)) 3200 continue; 3201 3202 next_ext_display_path = get_ext_display_path_entry( 3203 &ext_display_connection_info_tbl, 3204 le16_to_cpu(next_object->usObjectID)); 3205 3206 if (next_ext_display_path == NULL) 3207 return BP_RESULT_FAILURE; 3208 3209 /* Merge if using same connector. */ 3210 if ((le16_to_cpu(next_ext_display_path->usDeviceConnector) == 3211 le16_to_cpu(ext_display_path->usDeviceConnector)) && 3212 (le16_to_cpu(ext_display_path->usDeviceConnector) != 0)) { 3213 /* Clear duplicate connector from table. */ 3214 next_object->usObjectID = cpu_to_le16(0); 3215 add_device_tag_from_ext_display_path( 3216 bp, 3217 object, 3218 ext_display_path, 3219 &cached_device_support); 3220 } 3221 } 3222 } 3223 3224 /* Find all encoders which have an MXM object as their destination. 3225 * Replace the MXM object with the real connector Id from the external 3226 * display connection info table */ 3227 3228 encoder_table_offset = bp->object_info_tbl_offset 3229 + le16_to_cpu(bp->object_info_tbl.v1_1->usEncoderObjectTableOffset); 3230 encoder_table = GET_IMAGE(ATOM_OBJECT_TABLE, encoder_table_offset); 3231 3232 for (i = 0; i < encoder_table->ucNumberOfObjects; i++) { 3233 uint32_t j; 3234 3235 object = &encoder_table->asObjects[i]; 3236 3237 dst_number = get_dest_obj_list(bp, object, &dst_object_id_list); 3238 3239 for (j = 0; j < dst_number; j++) { 3240 object_id = object_id_from_bios_object_id( 3241 dst_object_id_list[j]); 3242 3243 if ((OBJECT_TYPE_CONNECTOR != object_id.type) || 3244 (CONNECTOR_ID_MXM != object_id.id)) 3245 continue; 3246 3247 /* Get the correct connection info table entry based on 3248 * the enum id. */ 3249 ext_display_path = 3250 get_ext_display_path_entry( 3251 &ext_display_connection_info_tbl, 3252 dst_object_id_list[j]); 3253 3254 if (ext_display_path == NULL) 3255 return BP_RESULT_FAILURE; 3256 3257 dst_object_id_list[j] = 3258 le16_to_cpu(ext_display_path->usDeviceConnector); 3259 } 3260 } 3261 3262 return BP_RESULT_OK; 3263 } 3264 3265 /* 3266 * Check whether we need to patch the VBIOS connector info table with 3267 * data from an external display connection info table. This is 3268 * necessary to support MXM boards with an OPM (output personality 3269 * module). With these designs, the VBIOS connector info table 3270 * specifies an MXM_CONNECTOR with a unique ID. The driver retrieves 3271 * the external connection info table through i2c and then looks up the 3272 * connector ID to find the real connector type (e.g. DFP1). 3273 * 3274 */ 3275 3276 static void process_ext_display_connection_info(struct bios_parser *bp) 3277 { 3278 ATOM_OBJECT_TABLE *connector_tbl; 3279 uint32_t connector_tbl_offset; 3280 struct graphics_object_id object_id; 3281 ATOM_OBJECT *object; 3282 bool mxm_connector_found = false; 3283 bool null_entry_found = false; 3284 uint32_t i = 0; 3285 3286 connector_tbl_offset = bp->object_info_tbl_offset + 3287 le16_to_cpu(bp->object_info_tbl.v1_1->usConnectorObjectTableOffset); 3288 connector_tbl = GET_IMAGE(ATOM_OBJECT_TABLE, connector_tbl_offset); 3289 3290 /* Look for MXM connectors to determine whether we need patch the VBIOS 3291 * connector info table. Look for null entries to determine whether we 3292 * need to compact connector table. */ 3293 for (i = 0; i < connector_tbl->ucNumberOfObjects; i++) { 3294 object = &connector_tbl->asObjects[i]; 3295 object_id = object_id_from_bios_object_id(le16_to_cpu(object->usObjectID)); 3296 3297 if ((OBJECT_TYPE_CONNECTOR == object_id.type) && 3298 (CONNECTOR_ID_MXM == object_id.id)) { 3299 /* Once we found MXM connector - we can break */ 3300 mxm_connector_found = true; 3301 break; 3302 } else if (OBJECT_TYPE_CONNECTOR != object_id.type) { 3303 /* We need to continue looping - to check if MXM 3304 * connector present */ 3305 null_entry_found = true; 3306 } 3307 } 3308 3309 /* Patch BIOS image */ 3310 if (mxm_connector_found || null_entry_found) { 3311 uint32_t connectors_num = 0; 3312 uint8_t *original_bios; 3313 /* Step 1: Replace bios image with the new copy which will be 3314 * patched */ 3315 bp->base.bios_local_image = kzalloc(bp->base.bios_size, 3316 GFP_KERNEL); 3317 if (bp->base.bios_local_image == NULL) { 3318 BREAK_TO_DEBUGGER(); 3319 /* Failed to alloc bp->base.bios_local_image */ 3320 return; 3321 } 3322 3323 memmove(bp->base.bios_local_image, bp->base.bios, bp->base.bios_size); 3324 original_bios = bp->base.bios; 3325 bp->base.bios = bp->base.bios_local_image; 3326 connector_tbl = 3327 GET_IMAGE(ATOM_OBJECT_TABLE, connector_tbl_offset); 3328 3329 /* Step 2: (only if MXM connector found) Patch BIOS image with 3330 * info from external module */ 3331 if (mxm_connector_found && 3332 patch_bios_image_from_ext_display_connection_info(bp) != 3333 BP_RESULT_OK) { 3334 /* Patching the bios image has failed. We will copy 3335 * again original image provided and afterwards 3336 * only remove null entries */ 3337 memmove( 3338 bp->base.bios_local_image, 3339 original_bios, 3340 bp->base.bios_size); 3341 } 3342 3343 /* Step 3: Compact connector table (remove null entries, valid 3344 * entries moved to beginning) */ 3345 for (i = 0; i < connector_tbl->ucNumberOfObjects; i++) { 3346 object = &connector_tbl->asObjects[i]; 3347 object_id = object_id_from_bios_object_id( 3348 le16_to_cpu(object->usObjectID)); 3349 3350 if (OBJECT_TYPE_CONNECTOR != object_id.type) 3351 continue; 3352 3353 if (i != connectors_num) { 3354 memmove( 3355 &connector_tbl-> 3356 asObjects[connectors_num], 3357 object, 3358 sizeof(ATOM_OBJECT)); 3359 } 3360 ++connectors_num; 3361 } 3362 connector_tbl->ucNumberOfObjects = (uint8_t)connectors_num; 3363 } 3364 } 3365 3366 static void bios_parser_post_init(struct dc_bios *dcb) 3367 { 3368 struct bios_parser *bp = BP_FROM_DCB(dcb); 3369 3370 process_ext_display_connection_info(bp); 3371 } 3372 3373 /** 3374 * bios_parser_set_scratch_critical_state 3375 * 3376 * @brief 3377 * update critical state bit in VBIOS scratch register 3378 * 3379 * @param 3380 * bool - to set or reset state 3381 */ 3382 static void bios_parser_set_scratch_critical_state( 3383 struct dc_bios *dcb, 3384 bool state) 3385 { 3386 bios_set_scratch_critical_state(dcb, state); 3387 } 3388 3389 /* 3390 * get_integrated_info_v8 3391 * 3392 * @brief 3393 * Get V8 integrated BIOS information 3394 * 3395 * @param 3396 * bios_parser *bp - [in]BIOS parser handler to get master data table 3397 * integrated_info *info - [out] store and output integrated info 3398 * 3399 * @return 3400 * enum bp_result - BP_RESULT_OK if information is available, 3401 * BP_RESULT_BADBIOSTABLE otherwise. 3402 */ 3403 static enum bp_result get_integrated_info_v8( 3404 struct bios_parser *bp, 3405 struct integrated_info *info) 3406 { 3407 ATOM_INTEGRATED_SYSTEM_INFO_V1_8 *info_v8; 3408 uint32_t i; 3409 3410 info_v8 = GET_IMAGE(ATOM_INTEGRATED_SYSTEM_INFO_V1_8, 3411 bp->master_data_tbl->ListOfDataTables.IntegratedSystemInfo); 3412 3413 if (info_v8 == NULL) 3414 return BP_RESULT_BADBIOSTABLE; 3415 info->boot_up_engine_clock = le32_to_cpu(info_v8->ulBootUpEngineClock) * 10; 3416 info->dentist_vco_freq = le32_to_cpu(info_v8->ulDentistVCOFreq) * 10; 3417 info->boot_up_uma_clock = le32_to_cpu(info_v8->ulBootUpUMAClock) * 10; 3418 3419 for (i = 0; i < NUMBER_OF_DISP_CLK_VOLTAGE; ++i) { 3420 /* Convert [10KHz] into [KHz] */ 3421 info->disp_clk_voltage[i].max_supported_clk = 3422 le32_to_cpu(info_v8->sDISPCLK_Voltage[i]. 3423 ulMaximumSupportedCLK) * 10; 3424 info->disp_clk_voltage[i].voltage_index = 3425 le32_to_cpu(info_v8->sDISPCLK_Voltage[i].ulVoltageIndex); 3426 } 3427 3428 info->boot_up_req_display_vector = 3429 le32_to_cpu(info_v8->ulBootUpReqDisplayVector); 3430 info->gpu_cap_info = 3431 le32_to_cpu(info_v8->ulGPUCapInfo); 3432 3433 /* 3434 * system_config: Bit[0] = 0 : PCIE power gating disabled 3435 * = 1 : PCIE power gating enabled 3436 * Bit[1] = 0 : DDR-PLL shut down disabled 3437 * = 1 : DDR-PLL shut down enabled 3438 * Bit[2] = 0 : DDR-PLL power down disabled 3439 * = 1 : DDR-PLL power down enabled 3440 */ 3441 info->system_config = le32_to_cpu(info_v8->ulSystemConfig); 3442 info->cpu_cap_info = le32_to_cpu(info_v8->ulCPUCapInfo); 3443 info->boot_up_nb_voltage = 3444 le16_to_cpu(info_v8->usBootUpNBVoltage); 3445 info->ext_disp_conn_info_offset = 3446 le16_to_cpu(info_v8->usExtDispConnInfoOffset); 3447 info->memory_type = info_v8->ucMemoryType; 3448 info->ma_channel_number = info_v8->ucUMAChannelNumber; 3449 info->gmc_restore_reset_time = 3450 le32_to_cpu(info_v8->ulGMCRestoreResetTime); 3451 3452 info->minimum_n_clk = 3453 le32_to_cpu(info_v8->ulNbpStateNClkFreq[0]); 3454 for (i = 1; i < 4; ++i) 3455 info->minimum_n_clk = 3456 info->minimum_n_clk < le32_to_cpu(info_v8->ulNbpStateNClkFreq[i]) ? 3457 info->minimum_n_clk : le32_to_cpu(info_v8->ulNbpStateNClkFreq[i]); 3458 3459 info->idle_n_clk = le32_to_cpu(info_v8->ulIdleNClk); 3460 info->ddr_dll_power_up_time = 3461 le32_to_cpu(info_v8->ulDDR_DLL_PowerUpTime); 3462 info->ddr_pll_power_up_time = 3463 le32_to_cpu(info_v8->ulDDR_PLL_PowerUpTime); 3464 info->pcie_clk_ss_type = le16_to_cpu(info_v8->usPCIEClkSSType); 3465 info->lvds_ss_percentage = 3466 le16_to_cpu(info_v8->usLvdsSSPercentage); 3467 info->lvds_sspread_rate_in_10hz = 3468 le16_to_cpu(info_v8->usLvdsSSpreadRateIn10Hz); 3469 info->hdmi_ss_percentage = 3470 le16_to_cpu(info_v8->usHDMISSPercentage); 3471 info->hdmi_sspread_rate_in_10hz = 3472 le16_to_cpu(info_v8->usHDMISSpreadRateIn10Hz); 3473 info->dvi_ss_percentage = 3474 le16_to_cpu(info_v8->usDVISSPercentage); 3475 info->dvi_sspread_rate_in_10_hz = 3476 le16_to_cpu(info_v8->usDVISSpreadRateIn10Hz); 3477 3478 info->max_lvds_pclk_freq_in_single_link = 3479 le16_to_cpu(info_v8->usMaxLVDSPclkFreqInSingleLink); 3480 info->lvds_misc = info_v8->ucLvdsMisc; 3481 info->lvds_pwr_on_seq_dig_on_to_de_in_4ms = 3482 info_v8->ucLVDSPwrOnSeqDIGONtoDE_in4Ms; 3483 info->lvds_pwr_on_seq_de_to_vary_bl_in_4ms = 3484 info_v8->ucLVDSPwrOnSeqDEtoVARY_BL_in4Ms; 3485 info->lvds_pwr_on_seq_vary_bl_to_blon_in_4ms = 3486 info_v8->ucLVDSPwrOnSeqVARY_BLtoBLON_in4Ms; 3487 info->lvds_pwr_off_seq_vary_bl_to_de_in4ms = 3488 info_v8->ucLVDSPwrOffSeqVARY_BLtoDE_in4Ms; 3489 info->lvds_pwr_off_seq_de_to_dig_on_in4ms = 3490 info_v8->ucLVDSPwrOffSeqDEtoDIGON_in4Ms; 3491 info->lvds_pwr_off_seq_blon_to_vary_bl_in_4ms = 3492 info_v8->ucLVDSPwrOffSeqBLONtoVARY_BL_in4Ms; 3493 info->lvds_off_to_on_delay_in_4ms = 3494 info_v8->ucLVDSOffToOnDelay_in4Ms; 3495 info->lvds_bit_depth_control_val = 3496 le32_to_cpu(info_v8->ulLCDBitDepthControlVal); 3497 3498 for (i = 0; i < NUMBER_OF_AVAILABLE_SCLK; ++i) { 3499 /* Convert [10KHz] into [KHz] */ 3500 info->avail_s_clk[i].supported_s_clk = 3501 le32_to_cpu(info_v8->sAvail_SCLK[i].ulSupportedSCLK) * 10; 3502 info->avail_s_clk[i].voltage_index = 3503 le16_to_cpu(info_v8->sAvail_SCLK[i].usVoltageIndex); 3504 info->avail_s_clk[i].voltage_id = 3505 le16_to_cpu(info_v8->sAvail_SCLK[i].usVoltageID); 3506 } 3507 3508 for (i = 0; i < NUMBER_OF_UCHAR_FOR_GUID; ++i) { 3509 info->ext_disp_conn_info.gu_id[i] = 3510 info_v8->sExtDispConnInfo.ucGuid[i]; 3511 } 3512 3513 for (i = 0; i < MAX_NUMBER_OF_EXT_DISPLAY_PATH; ++i) { 3514 info->ext_disp_conn_info.path[i].device_connector_id = 3515 object_id_from_bios_object_id( 3516 le16_to_cpu(info_v8->sExtDispConnInfo.sPath[i].usDeviceConnector)); 3517 3518 info->ext_disp_conn_info.path[i].ext_encoder_obj_id = 3519 object_id_from_bios_object_id( 3520 le16_to_cpu(info_v8->sExtDispConnInfo.sPath[i].usExtEncoderObjId)); 3521 3522 info->ext_disp_conn_info.path[i].device_tag = 3523 le16_to_cpu(info_v8->sExtDispConnInfo.sPath[i].usDeviceTag); 3524 info->ext_disp_conn_info.path[i].device_acpi_enum = 3525 le16_to_cpu(info_v8->sExtDispConnInfo.sPath[i].usDeviceACPIEnum); 3526 info->ext_disp_conn_info.path[i].ext_aux_ddc_lut_index = 3527 info_v8->sExtDispConnInfo.sPath[i].ucExtAUXDDCLutIndex; 3528 info->ext_disp_conn_info.path[i].ext_hpd_pin_lut_index = 3529 info_v8->sExtDispConnInfo.sPath[i].ucExtHPDPINLutIndex; 3530 info->ext_disp_conn_info.path[i].channel_mapping.raw = 3531 info_v8->sExtDispConnInfo.sPath[i].ucChannelMapping; 3532 } 3533 info->ext_disp_conn_info.checksum = 3534 info_v8->sExtDispConnInfo.ucChecksum; 3535 3536 return BP_RESULT_OK; 3537 } 3538 3539 /* 3540 * get_integrated_info_v8 3541 * 3542 * @brief 3543 * Get V8 integrated BIOS information 3544 * 3545 * @param 3546 * bios_parser *bp - [in]BIOS parser handler to get master data table 3547 * integrated_info *info - [out] store and output integrated info 3548 * 3549 * @return 3550 * enum bp_result - BP_RESULT_OK if information is available, 3551 * BP_RESULT_BADBIOSTABLE otherwise. 3552 */ 3553 static enum bp_result get_integrated_info_v9( 3554 struct bios_parser *bp, 3555 struct integrated_info *info) 3556 { 3557 ATOM_INTEGRATED_SYSTEM_INFO_V1_9 *info_v9; 3558 uint32_t i; 3559 3560 info_v9 = GET_IMAGE(ATOM_INTEGRATED_SYSTEM_INFO_V1_9, 3561 bp->master_data_tbl->ListOfDataTables.IntegratedSystemInfo); 3562 3563 if (!info_v9) 3564 return BP_RESULT_BADBIOSTABLE; 3565 3566 info->boot_up_engine_clock = le32_to_cpu(info_v9->ulBootUpEngineClock) * 10; 3567 info->dentist_vco_freq = le32_to_cpu(info_v9->ulDentistVCOFreq) * 10; 3568 info->boot_up_uma_clock = le32_to_cpu(info_v9->ulBootUpUMAClock) * 10; 3569 3570 for (i = 0; i < NUMBER_OF_DISP_CLK_VOLTAGE; ++i) { 3571 /* Convert [10KHz] into [KHz] */ 3572 info->disp_clk_voltage[i].max_supported_clk = 3573 le32_to_cpu(info_v9->sDISPCLK_Voltage[i].ulMaximumSupportedCLK) * 10; 3574 info->disp_clk_voltage[i].voltage_index = 3575 le32_to_cpu(info_v9->sDISPCLK_Voltage[i].ulVoltageIndex); 3576 } 3577 3578 info->boot_up_req_display_vector = 3579 le32_to_cpu(info_v9->ulBootUpReqDisplayVector); 3580 info->gpu_cap_info = le32_to_cpu(info_v9->ulGPUCapInfo); 3581 3582 /* 3583 * system_config: Bit[0] = 0 : PCIE power gating disabled 3584 * = 1 : PCIE power gating enabled 3585 * Bit[1] = 0 : DDR-PLL shut down disabled 3586 * = 1 : DDR-PLL shut down enabled 3587 * Bit[2] = 0 : DDR-PLL power down disabled 3588 * = 1 : DDR-PLL power down enabled 3589 */ 3590 info->system_config = le32_to_cpu(info_v9->ulSystemConfig); 3591 info->cpu_cap_info = le32_to_cpu(info_v9->ulCPUCapInfo); 3592 info->boot_up_nb_voltage = le16_to_cpu(info_v9->usBootUpNBVoltage); 3593 info->ext_disp_conn_info_offset = le16_to_cpu(info_v9->usExtDispConnInfoOffset); 3594 info->memory_type = info_v9->ucMemoryType; 3595 info->ma_channel_number = info_v9->ucUMAChannelNumber; 3596 info->gmc_restore_reset_time = le32_to_cpu(info_v9->ulGMCRestoreResetTime); 3597 3598 info->minimum_n_clk = le32_to_cpu(info_v9->ulNbpStateNClkFreq[0]); 3599 for (i = 1; i < 4; ++i) 3600 info->minimum_n_clk = 3601 info->minimum_n_clk < le32_to_cpu(info_v9->ulNbpStateNClkFreq[i]) ? 3602 info->minimum_n_clk : le32_to_cpu(info_v9->ulNbpStateNClkFreq[i]); 3603 3604 info->idle_n_clk = le32_to_cpu(info_v9->ulIdleNClk); 3605 info->ddr_dll_power_up_time = le32_to_cpu(info_v9->ulDDR_DLL_PowerUpTime); 3606 info->ddr_pll_power_up_time = le32_to_cpu(info_v9->ulDDR_PLL_PowerUpTime); 3607 info->pcie_clk_ss_type = le16_to_cpu(info_v9->usPCIEClkSSType); 3608 info->lvds_ss_percentage = le16_to_cpu(info_v9->usLvdsSSPercentage); 3609 info->lvds_sspread_rate_in_10hz = le16_to_cpu(info_v9->usLvdsSSpreadRateIn10Hz); 3610 info->hdmi_ss_percentage = le16_to_cpu(info_v9->usHDMISSPercentage); 3611 info->hdmi_sspread_rate_in_10hz = le16_to_cpu(info_v9->usHDMISSpreadRateIn10Hz); 3612 info->dvi_ss_percentage = le16_to_cpu(info_v9->usDVISSPercentage); 3613 info->dvi_sspread_rate_in_10_hz = le16_to_cpu(info_v9->usDVISSpreadRateIn10Hz); 3614 3615 info->max_lvds_pclk_freq_in_single_link = 3616 le16_to_cpu(info_v9->usMaxLVDSPclkFreqInSingleLink); 3617 info->lvds_misc = info_v9->ucLvdsMisc; 3618 info->lvds_pwr_on_seq_dig_on_to_de_in_4ms = 3619 info_v9->ucLVDSPwrOnSeqDIGONtoDE_in4Ms; 3620 info->lvds_pwr_on_seq_de_to_vary_bl_in_4ms = 3621 info_v9->ucLVDSPwrOnSeqDEtoVARY_BL_in4Ms; 3622 info->lvds_pwr_on_seq_vary_bl_to_blon_in_4ms = 3623 info_v9->ucLVDSPwrOnSeqVARY_BLtoBLON_in4Ms; 3624 info->lvds_pwr_off_seq_vary_bl_to_de_in4ms = 3625 info_v9->ucLVDSPwrOffSeqVARY_BLtoDE_in4Ms; 3626 info->lvds_pwr_off_seq_de_to_dig_on_in4ms = 3627 info_v9->ucLVDSPwrOffSeqDEtoDIGON_in4Ms; 3628 info->lvds_pwr_off_seq_blon_to_vary_bl_in_4ms = 3629 info_v9->ucLVDSPwrOffSeqBLONtoVARY_BL_in4Ms; 3630 info->lvds_off_to_on_delay_in_4ms = 3631 info_v9->ucLVDSOffToOnDelay_in4Ms; 3632 info->lvds_bit_depth_control_val = 3633 le32_to_cpu(info_v9->ulLCDBitDepthControlVal); 3634 3635 for (i = 0; i < NUMBER_OF_AVAILABLE_SCLK; ++i) { 3636 /* Convert [10KHz] into [KHz] */ 3637 info->avail_s_clk[i].supported_s_clk = 3638 le32_to_cpu(info_v9->sAvail_SCLK[i].ulSupportedSCLK) * 10; 3639 info->avail_s_clk[i].voltage_index = 3640 le16_to_cpu(info_v9->sAvail_SCLK[i].usVoltageIndex); 3641 info->avail_s_clk[i].voltage_id = 3642 le16_to_cpu(info_v9->sAvail_SCLK[i].usVoltageID); 3643 } 3644 3645 for (i = 0; i < NUMBER_OF_UCHAR_FOR_GUID; ++i) { 3646 info->ext_disp_conn_info.gu_id[i] = 3647 info_v9->sExtDispConnInfo.ucGuid[i]; 3648 } 3649 3650 for (i = 0; i < MAX_NUMBER_OF_EXT_DISPLAY_PATH; ++i) { 3651 info->ext_disp_conn_info.path[i].device_connector_id = 3652 object_id_from_bios_object_id( 3653 le16_to_cpu(info_v9->sExtDispConnInfo.sPath[i].usDeviceConnector)); 3654 3655 info->ext_disp_conn_info.path[i].ext_encoder_obj_id = 3656 object_id_from_bios_object_id( 3657 le16_to_cpu(info_v9->sExtDispConnInfo.sPath[i].usExtEncoderObjId)); 3658 3659 info->ext_disp_conn_info.path[i].device_tag = 3660 le16_to_cpu(info_v9->sExtDispConnInfo.sPath[i].usDeviceTag); 3661 info->ext_disp_conn_info.path[i].device_acpi_enum = 3662 le16_to_cpu(info_v9->sExtDispConnInfo.sPath[i].usDeviceACPIEnum); 3663 info->ext_disp_conn_info.path[i].ext_aux_ddc_lut_index = 3664 info_v9->sExtDispConnInfo.sPath[i].ucExtAUXDDCLutIndex; 3665 info->ext_disp_conn_info.path[i].ext_hpd_pin_lut_index = 3666 info_v9->sExtDispConnInfo.sPath[i].ucExtHPDPINLutIndex; 3667 info->ext_disp_conn_info.path[i].channel_mapping.raw = 3668 info_v9->sExtDispConnInfo.sPath[i].ucChannelMapping; 3669 } 3670 info->ext_disp_conn_info.checksum = 3671 info_v9->sExtDispConnInfo.ucChecksum; 3672 3673 return BP_RESULT_OK; 3674 } 3675 3676 /* 3677 * construct_integrated_info 3678 * 3679 * @brief 3680 * Get integrated BIOS information based on table revision 3681 * 3682 * @param 3683 * bios_parser *bp - [in]BIOS parser handler to get master data table 3684 * integrated_info *info - [out] store and output integrated info 3685 * 3686 * @return 3687 * enum bp_result - BP_RESULT_OK if information is available, 3688 * BP_RESULT_BADBIOSTABLE otherwise. 3689 */ 3690 static enum bp_result construct_integrated_info( 3691 struct bios_parser *bp, 3692 struct integrated_info *info) 3693 { 3694 enum bp_result result = BP_RESULT_BADBIOSTABLE; 3695 3696 ATOM_COMMON_TABLE_HEADER *header; 3697 struct atom_data_revision revision; 3698 3699 if (bp->master_data_tbl->ListOfDataTables.IntegratedSystemInfo) { 3700 header = GET_IMAGE(ATOM_COMMON_TABLE_HEADER, 3701 bp->master_data_tbl->ListOfDataTables.IntegratedSystemInfo); 3702 3703 get_atom_data_table_revision(header, &revision); 3704 3705 /* Don't need to check major revision as they are all 1 */ 3706 switch (revision.minor) { 3707 case 8: 3708 result = get_integrated_info_v8(bp, info); 3709 break; 3710 case 9: 3711 result = get_integrated_info_v9(bp, info); 3712 break; 3713 default: 3714 return result; 3715 3716 } 3717 } 3718 3719 /* Sort voltage table from low to high*/ 3720 if (result == BP_RESULT_OK) { 3721 struct clock_voltage_caps temp = {0, 0}; 3722 uint32_t i; 3723 uint32_t j; 3724 3725 for (i = 1; i < NUMBER_OF_DISP_CLK_VOLTAGE; ++i) { 3726 for (j = i; j > 0; --j) { 3727 if ( 3728 info->disp_clk_voltage[j].max_supported_clk < 3729 info->disp_clk_voltage[j-1].max_supported_clk) { 3730 /* swap j and j - 1*/ 3731 temp = info->disp_clk_voltage[j-1]; 3732 info->disp_clk_voltage[j-1] = 3733 info->disp_clk_voltage[j]; 3734 info->disp_clk_voltage[j] = temp; 3735 } 3736 } 3737 } 3738 3739 } 3740 3741 return result; 3742 } 3743 3744 static struct integrated_info *bios_parser_create_integrated_info( 3745 struct dc_bios *dcb) 3746 { 3747 struct bios_parser *bp = BP_FROM_DCB(dcb); 3748 struct integrated_info *info = NULL; 3749 3750 info = kzalloc(sizeof(struct integrated_info), GFP_KERNEL); 3751 3752 if (info == NULL) { 3753 ASSERT_CRITICAL(0); 3754 return NULL; 3755 } 3756 3757 if (construct_integrated_info(bp, info) == BP_RESULT_OK) 3758 return info; 3759 3760 kfree(info); 3761 3762 return NULL; 3763 } 3764 3765 enum bp_result update_slot_layout_info( 3766 struct dc_bios *dcb, 3767 unsigned int i, 3768 struct slot_layout_info *slot_layout_info, 3769 unsigned int record_offset) 3770 { 3771 unsigned int j; 3772 struct bios_parser *bp; 3773 ATOM_BRACKET_LAYOUT_RECORD *record; 3774 ATOM_COMMON_RECORD_HEADER *record_header; 3775 enum bp_result result = BP_RESULT_NORECORD; 3776 3777 bp = BP_FROM_DCB(dcb); 3778 record = NULL; 3779 record_header = NULL; 3780 3781 for (;;) { 3782 3783 record_header = (ATOM_COMMON_RECORD_HEADER *) 3784 GET_IMAGE(ATOM_COMMON_RECORD_HEADER, record_offset); 3785 if (record_header == NULL) { 3786 result = BP_RESULT_BADBIOSTABLE; 3787 break; 3788 } 3789 3790 /* the end of the list */ 3791 if (record_header->ucRecordType == 0xff || 3792 record_header->ucRecordSize == 0) { 3793 break; 3794 } 3795 3796 if (record_header->ucRecordType == 3797 ATOM_BRACKET_LAYOUT_RECORD_TYPE && 3798 sizeof(ATOM_BRACKET_LAYOUT_RECORD) 3799 <= record_header->ucRecordSize) { 3800 record = (ATOM_BRACKET_LAYOUT_RECORD *) 3801 (record_header); 3802 result = BP_RESULT_OK; 3803 break; 3804 } 3805 3806 record_offset += record_header->ucRecordSize; 3807 } 3808 3809 /* return if the record not found */ 3810 if (result != BP_RESULT_OK) 3811 return result; 3812 3813 /* get slot sizes */ 3814 slot_layout_info->length = record->ucLength; 3815 slot_layout_info->width = record->ucWidth; 3816 3817 /* get info for each connector in the slot */ 3818 slot_layout_info->num_of_connectors = record->ucConnNum; 3819 for (j = 0; j < slot_layout_info->num_of_connectors; ++j) { 3820 slot_layout_info->connectors[j].connector_type = 3821 (enum connector_layout_type) 3822 (record->asConnInfo[j].ucConnectorType); 3823 switch (record->asConnInfo[j].ucConnectorType) { 3824 case CONNECTOR_TYPE_DVI_D: 3825 slot_layout_info->connectors[j].connector_type = 3826 CONNECTOR_LAYOUT_TYPE_DVI_D; 3827 slot_layout_info->connectors[j].length = 3828 CONNECTOR_SIZE_DVI; 3829 break; 3830 3831 case CONNECTOR_TYPE_HDMI: 3832 slot_layout_info->connectors[j].connector_type = 3833 CONNECTOR_LAYOUT_TYPE_HDMI; 3834 slot_layout_info->connectors[j].length = 3835 CONNECTOR_SIZE_HDMI; 3836 break; 3837 3838 case CONNECTOR_TYPE_DISPLAY_PORT: 3839 slot_layout_info->connectors[j].connector_type = 3840 CONNECTOR_LAYOUT_TYPE_DP; 3841 slot_layout_info->connectors[j].length = 3842 CONNECTOR_SIZE_DP; 3843 break; 3844 3845 case CONNECTOR_TYPE_MINI_DISPLAY_PORT: 3846 slot_layout_info->connectors[j].connector_type = 3847 CONNECTOR_LAYOUT_TYPE_MINI_DP; 3848 slot_layout_info->connectors[j].length = 3849 CONNECTOR_SIZE_MINI_DP; 3850 break; 3851 3852 default: 3853 slot_layout_info->connectors[j].connector_type = 3854 CONNECTOR_LAYOUT_TYPE_UNKNOWN; 3855 slot_layout_info->connectors[j].length = 3856 CONNECTOR_SIZE_UNKNOWN; 3857 } 3858 3859 slot_layout_info->connectors[j].position = 3860 record->asConnInfo[j].ucPosition; 3861 slot_layout_info->connectors[j].connector_id = 3862 object_id_from_bios_object_id( 3863 record->asConnInfo[j].usConnectorObjectId); 3864 } 3865 return result; 3866 } 3867 3868 3869 enum bp_result get_bracket_layout_record( 3870 struct dc_bios *dcb, 3871 unsigned int bracket_layout_id, 3872 struct slot_layout_info *slot_layout_info) 3873 { 3874 unsigned int i; 3875 unsigned int record_offset; 3876 struct bios_parser *bp; 3877 enum bp_result result; 3878 ATOM_OBJECT *object; 3879 ATOM_OBJECT_TABLE *object_table; 3880 unsigned int genericTableOffset; 3881 3882 bp = BP_FROM_DCB(dcb); 3883 object = NULL; 3884 if (slot_layout_info == NULL) { 3885 DC_LOG_DETECTION_EDID_PARSER("Invalid slot_layout_info\n"); 3886 return BP_RESULT_BADINPUT; 3887 } 3888 3889 3890 genericTableOffset = bp->object_info_tbl_offset + 3891 bp->object_info_tbl.v1_3->usMiscObjectTableOffset; 3892 object_table = (ATOM_OBJECT_TABLE *) 3893 GET_IMAGE(ATOM_OBJECT_TABLE, genericTableOffset); 3894 if (!object_table) 3895 return BP_RESULT_FAILURE; 3896 3897 result = BP_RESULT_NORECORD; 3898 for (i = 0; i < object_table->ucNumberOfObjects; ++i) { 3899 3900 if (bracket_layout_id == 3901 object_table->asObjects[i].usObjectID) { 3902 3903 object = &object_table->asObjects[i]; 3904 record_offset = object->usRecordOffset + 3905 bp->object_info_tbl_offset; 3906 3907 result = update_slot_layout_info(dcb, i, 3908 slot_layout_info, record_offset); 3909 break; 3910 } 3911 } 3912 return result; 3913 } 3914 3915 static enum bp_result bios_get_board_layout_info( 3916 struct dc_bios *dcb, 3917 struct board_layout_info *board_layout_info) 3918 { 3919 unsigned int i; 3920 struct bios_parser *bp; 3921 enum bp_result record_result; 3922 3923 const unsigned int slot_index_to_vbios_id[MAX_BOARD_SLOTS] = { 3924 GENERICOBJECT_BRACKET_LAYOUT_ENUM_ID1, 3925 GENERICOBJECT_BRACKET_LAYOUT_ENUM_ID2, 3926 0, 0 3927 }; 3928 3929 bp = BP_FROM_DCB(dcb); 3930 if (board_layout_info == NULL) { 3931 DC_LOG_DETECTION_EDID_PARSER("Invalid board_layout_info\n"); 3932 return BP_RESULT_BADINPUT; 3933 } 3934 3935 board_layout_info->num_of_slots = 0; 3936 3937 for (i = 0; i < MAX_BOARD_SLOTS; ++i) { 3938 record_result = get_bracket_layout_record(dcb, 3939 slot_index_to_vbios_id[i], 3940 &board_layout_info->slots[i]); 3941 3942 if (record_result == BP_RESULT_NORECORD && i > 0) 3943 break; /* no more slots present in bios */ 3944 else if (record_result != BP_RESULT_OK) 3945 return record_result; /* fail */ 3946 3947 ++board_layout_info->num_of_slots; 3948 } 3949 3950 /* all data is valid */ 3951 board_layout_info->is_number_of_slots_valid = 1; 3952 board_layout_info->is_slots_size_valid = 1; 3953 board_layout_info->is_connector_offsets_valid = 1; 3954 board_layout_info->is_connector_lengths_valid = 1; 3955 3956 return BP_RESULT_OK; 3957 } 3958 3959 /******************************************************************************/ 3960 3961 static const struct dc_vbios_funcs vbios_funcs = { 3962 .get_connectors_number = bios_parser_get_connectors_number, 3963 3964 .get_encoder_id = bios_parser_get_encoder_id, 3965 3966 .get_connector_id = bios_parser_get_connector_id, 3967 3968 .get_dst_number = bios_parser_get_dst_number, 3969 3970 .get_src_obj = bios_parser_get_src_obj, 3971 3972 .get_dst_obj = bios_parser_get_dst_obj, 3973 3974 .get_i2c_info = bios_parser_get_i2c_info, 3975 3976 .get_voltage_ddc_info = bios_parser_get_voltage_ddc_info, 3977 3978 .get_thermal_ddc_info = bios_parser_get_thermal_ddc_info, 3979 3980 .get_hpd_info = bios_parser_get_hpd_info, 3981 3982 .get_device_tag = bios_parser_get_device_tag, 3983 3984 .get_firmware_info = bios_parser_get_firmware_info, 3985 3986 .get_spread_spectrum_info = bios_parser_get_spread_spectrum_info, 3987 3988 .get_ss_entry_number = bios_parser_get_ss_entry_number, 3989 3990 .get_embedded_panel_info = bios_parser_get_embedded_panel_info, 3991 3992 .get_gpio_pin_info = bios_parser_get_gpio_pin_info, 3993 3994 .get_encoder_cap_info = bios_parser_get_encoder_cap_info, 3995 3996 /* bios scratch register communication */ 3997 .is_accelerated_mode = bios_is_accelerated_mode, 3998 .get_vga_enabled_displays = bios_get_vga_enabled_displays, 3999 4000 .set_scratch_critical_state = bios_parser_set_scratch_critical_state, 4001 4002 .is_device_id_supported = bios_parser_is_device_id_supported, 4003 4004 /* COMMANDS */ 4005 .encoder_control = bios_parser_encoder_control, 4006 4007 .transmitter_control = bios_parser_transmitter_control, 4008 4009 .crt_control = bios_parser_crt_control, /* not used in DAL3. keep for now in case we need to support VGA on Bonaire */ 4010 4011 .enable_crtc = bios_parser_enable_crtc, 4012 4013 .adjust_pixel_clock = bios_parser_adjust_pixel_clock, 4014 4015 .set_pixel_clock = bios_parser_set_pixel_clock, 4016 4017 .set_dce_clock = bios_parser_set_dce_clock, 4018 4019 .enable_spread_spectrum_on_ppll = bios_parser_enable_spread_spectrum_on_ppll, 4020 4021 .program_crtc_timing = bios_parser_program_crtc_timing, /* still use. should probably retire and program directly */ 4022 4023 .crtc_source_select = bios_parser_crtc_source_select, /* still use. should probably retire and program directly */ 4024 4025 .program_display_engine_pll = bios_parser_program_display_engine_pll, 4026 4027 .enable_disp_power_gating = bios_parser_enable_disp_power_gating, 4028 4029 /* SW init and patch */ 4030 .post_init = bios_parser_post_init, /* patch vbios table for mxm module by reading i2c */ 4031 4032 .bios_parser_destroy = bios_parser_destroy, 4033 4034 .get_board_layout_info = bios_get_board_layout_info, 4035 }; 4036 4037 static bool bios_parser_construct( 4038 struct bios_parser *bp, 4039 struct bp_init_data *init, 4040 enum dce_version dce_version) 4041 { 4042 uint16_t *rom_header_offset = NULL; 4043 ATOM_ROM_HEADER *rom_header = NULL; 4044 ATOM_OBJECT_HEADER *object_info_tbl; 4045 struct atom_data_revision tbl_rev = {0}; 4046 4047 if (!init) 4048 return false; 4049 4050 if (!init->bios) 4051 return false; 4052 4053 bp->base.funcs = &vbios_funcs; 4054 bp->base.bios = init->bios; 4055 bp->base.bios_size = bp->base.bios[BIOS_IMAGE_SIZE_OFFSET] * BIOS_IMAGE_SIZE_UNIT; 4056 4057 bp->base.ctx = init->ctx; 4058 bp->base.bios_local_image = NULL; 4059 4060 rom_header_offset = 4061 GET_IMAGE(uint16_t, OFFSET_TO_POINTER_TO_ATOM_ROM_HEADER); 4062 4063 if (!rom_header_offset) 4064 return false; 4065 4066 rom_header = GET_IMAGE(ATOM_ROM_HEADER, *rom_header_offset); 4067 4068 if (!rom_header) 4069 return false; 4070 4071 get_atom_data_table_revision(&rom_header->sHeader, &tbl_rev); 4072 if (tbl_rev.major >= 2 && tbl_rev.minor >= 2) 4073 return false; 4074 4075 bp->master_data_tbl = 4076 GET_IMAGE(ATOM_MASTER_DATA_TABLE, 4077 rom_header->usMasterDataTableOffset); 4078 4079 if (!bp->master_data_tbl) 4080 return false; 4081 4082 bp->object_info_tbl_offset = DATA_TABLES(Object_Header); 4083 4084 if (!bp->object_info_tbl_offset) 4085 return false; 4086 4087 object_info_tbl = 4088 GET_IMAGE(ATOM_OBJECT_HEADER, bp->object_info_tbl_offset); 4089 4090 if (!object_info_tbl) 4091 return false; 4092 4093 get_atom_data_table_revision(&object_info_tbl->sHeader, 4094 &bp->object_info_tbl.revision); 4095 4096 if (bp->object_info_tbl.revision.major == 1 4097 && bp->object_info_tbl.revision.minor >= 3) { 4098 ATOM_OBJECT_HEADER_V3 *tbl_v3; 4099 4100 tbl_v3 = GET_IMAGE(ATOM_OBJECT_HEADER_V3, 4101 bp->object_info_tbl_offset); 4102 if (!tbl_v3) 4103 return false; 4104 4105 bp->object_info_tbl.v1_3 = tbl_v3; 4106 } else if (bp->object_info_tbl.revision.major == 1 4107 && bp->object_info_tbl.revision.minor >= 1) 4108 bp->object_info_tbl.v1_1 = object_info_tbl; 4109 else 4110 return false; 4111 4112 dal_bios_parser_init_cmd_tbl(bp); 4113 dal_bios_parser_init_cmd_tbl_helper(&bp->cmd_helper, dce_version); 4114 4115 bp->base.integrated_info = bios_parser_create_integrated_info(&bp->base); 4116 4117 return true; 4118 } 4119 4120 /******************************************************************************/ 4121