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 "ObjectID.h" 29 30 #include "atomfirmware.h" 31 #include "atom.h" 32 #include "include/bios_parser_interface.h" 33 34 #include "command_table2.h" 35 #include "command_table_helper2.h" 36 #include "bios_parser_helper.h" 37 #include "bios_parser_types_internal2.h" 38 #include "amdgpu.h" 39 40 41 #define DC_LOGGER \ 42 bp->base.ctx->logger 43 44 #define GET_INDEX_INTO_MASTER_TABLE(MasterOrData, FieldName)\ 45 (((char *)(&((\ 46 struct atom_master_list_of_##MasterOrData##_functions_v2_1 *)0)\ 47 ->FieldName)-(char *)0)/sizeof(uint16_t)) 48 49 #define EXEC_BIOS_CMD_TABLE(fname, params)\ 50 (amdgpu_atom_execute_table(((struct amdgpu_device *)bp->base.ctx->driver_context)->mode_info.atom_context, \ 51 GET_INDEX_INTO_MASTER_TABLE(command, fname), \ 52 (uint32_t *)¶ms) == 0) 53 54 #define BIOS_CMD_TABLE_REVISION(fname, frev, crev)\ 55 amdgpu_atom_parse_cmd_header(((struct amdgpu_device *)bp->base.ctx->driver_context)->mode_info.atom_context, \ 56 GET_INDEX_INTO_MASTER_TABLE(command, fname), &frev, &crev) 57 58 #define BIOS_CMD_TABLE_PARA_REVISION(fname)\ 59 bios_cmd_table_para_revision(bp->base.ctx->driver_context, \ 60 GET_INDEX_INTO_MASTER_TABLE(command, fname)) 61 62 static void init_dig_encoder_control(struct bios_parser *bp); 63 static void init_transmitter_control(struct bios_parser *bp); 64 static void init_set_pixel_clock(struct bios_parser *bp); 65 66 static void init_set_crtc_timing(struct bios_parser *bp); 67 68 static void init_select_crtc_source(struct bios_parser *bp); 69 static void init_enable_crtc(struct bios_parser *bp); 70 71 static void init_external_encoder_control(struct bios_parser *bp); 72 static void init_enable_disp_power_gating(struct bios_parser *bp); 73 static void init_set_dce_clock(struct bios_parser *bp); 74 static void init_get_smu_clock_info(struct bios_parser *bp); 75 76 void dal_firmware_parser_init_cmd_tbl(struct bios_parser *bp) 77 { 78 init_dig_encoder_control(bp); 79 init_transmitter_control(bp); 80 init_set_pixel_clock(bp); 81 82 init_set_crtc_timing(bp); 83 84 init_select_crtc_source(bp); 85 init_enable_crtc(bp); 86 87 init_external_encoder_control(bp); 88 init_enable_disp_power_gating(bp); 89 init_set_dce_clock(bp); 90 init_get_smu_clock_info(bp); 91 } 92 93 static uint32_t bios_cmd_table_para_revision(void *dev, 94 uint32_t index) 95 { 96 struct amdgpu_device *adev = dev; 97 uint8_t frev, crev; 98 99 if (amdgpu_atom_parse_cmd_header(adev->mode_info.atom_context, 100 index, 101 &frev, &crev)) 102 return crev; 103 else 104 return 0; 105 } 106 107 /****************************************************************************** 108 ****************************************************************************** 109 ** 110 ** D I G E N C O D E R C O N T R O L 111 ** 112 ****************************************************************************** 113 *****************************************************************************/ 114 115 static enum bp_result encoder_control_digx_v1_5( 116 struct bios_parser *bp, 117 struct bp_encoder_control *cntl); 118 119 static void init_dig_encoder_control(struct bios_parser *bp) 120 { 121 uint32_t version = 122 BIOS_CMD_TABLE_PARA_REVISION(digxencodercontrol); 123 124 switch (version) { 125 case 5: 126 bp->cmd_tbl.dig_encoder_control = encoder_control_digx_v1_5; 127 break; 128 default: 129 dm_output_to_console("Don't have dig_encoder_control for v%d\n", version); 130 bp->cmd_tbl.dig_encoder_control = NULL; 131 break; 132 } 133 } 134 135 static enum bp_result encoder_control_digx_v1_5( 136 struct bios_parser *bp, 137 struct bp_encoder_control *cntl) 138 { 139 enum bp_result result = BP_RESULT_FAILURE; 140 struct dig_encoder_stream_setup_parameters_v1_5 params = {0}; 141 142 params.digid = (uint8_t)(cntl->engine_id); 143 params.action = bp->cmd_helper->encoder_action_to_atom(cntl->action); 144 145 params.pclk_10khz = cntl->pixel_clock / 10; 146 params.digmode = 147 (uint8_t)(bp->cmd_helper->encoder_mode_bp_to_atom( 148 cntl->signal, 149 cntl->enable_dp_audio)); 150 params.lanenum = (uint8_t)(cntl->lanes_number); 151 152 switch (cntl->color_depth) { 153 case COLOR_DEPTH_888: 154 params.bitpercolor = PANEL_8BIT_PER_COLOR; 155 break; 156 case COLOR_DEPTH_101010: 157 params.bitpercolor = PANEL_10BIT_PER_COLOR; 158 break; 159 case COLOR_DEPTH_121212: 160 params.bitpercolor = PANEL_12BIT_PER_COLOR; 161 break; 162 case COLOR_DEPTH_161616: 163 params.bitpercolor = PANEL_16BIT_PER_COLOR; 164 break; 165 default: 166 break; 167 } 168 169 if (cntl->signal == SIGNAL_TYPE_HDMI_TYPE_A) 170 switch (cntl->color_depth) { 171 case COLOR_DEPTH_101010: 172 params.pclk_10khz = 173 (params.pclk_10khz * 30) / 24; 174 break; 175 case COLOR_DEPTH_121212: 176 params.pclk_10khz = 177 (params.pclk_10khz * 36) / 24; 178 break; 179 case COLOR_DEPTH_161616: 180 params.pclk_10khz = 181 (params.pclk_10khz * 48) / 24; 182 break; 183 default: 184 break; 185 } 186 187 if (EXEC_BIOS_CMD_TABLE(digxencodercontrol, params)) 188 result = BP_RESULT_OK; 189 190 return result; 191 } 192 193 /***************************************************************************** 194 ****************************************************************************** 195 ** 196 ** TRANSMITTER CONTROL 197 ** 198 ****************************************************************************** 199 *****************************************************************************/ 200 201 static enum bp_result transmitter_control_v1_6( 202 struct bios_parser *bp, 203 struct bp_transmitter_control *cntl); 204 205 static void init_transmitter_control(struct bios_parser *bp) 206 { 207 uint8_t frev; 208 uint8_t crev; 209 210 if (BIOS_CMD_TABLE_REVISION(dig1transmittercontrol, frev, crev) == false) 211 BREAK_TO_DEBUGGER(); 212 switch (crev) { 213 case 6: 214 bp->cmd_tbl.transmitter_control = transmitter_control_v1_6; 215 break; 216 default: 217 dm_output_to_console("Don't have transmitter_control for v%d\n", crev); 218 bp->cmd_tbl.transmitter_control = NULL; 219 break; 220 } 221 } 222 223 static enum bp_result transmitter_control_v1_6( 224 struct bios_parser *bp, 225 struct bp_transmitter_control *cntl) 226 { 227 enum bp_result result = BP_RESULT_FAILURE; 228 const struct command_table_helper *cmd = bp->cmd_helper; 229 struct dig_transmitter_control_ps_allocation_v1_6 ps = { { 0 } }; 230 231 ps.param.phyid = cmd->phy_id_to_atom(cntl->transmitter); 232 ps.param.action = (uint8_t)cntl->action; 233 234 if (cntl->action == TRANSMITTER_CONTROL_SET_VOLTAGE_AND_PREEMPASIS) 235 ps.param.mode_laneset.dplaneset = (uint8_t)cntl->lane_settings; 236 else 237 ps.param.mode_laneset.digmode = 238 cmd->signal_type_to_atom_dig_mode(cntl->signal); 239 240 ps.param.lanenum = (uint8_t)cntl->lanes_number; 241 ps.param.hpdsel = cmd->hpd_sel_to_atom(cntl->hpd_sel); 242 ps.param.digfe_sel = cmd->dig_encoder_sel_to_atom(cntl->engine_id); 243 ps.param.connobj_id = (uint8_t)cntl->connector_obj_id.id; 244 ps.param.symclk_10khz = cntl->pixel_clock/10; 245 246 247 if (cntl->action == TRANSMITTER_CONTROL_ENABLE || 248 cntl->action == TRANSMITTER_CONTROL_ACTIAVATE || 249 cntl->action == TRANSMITTER_CONTROL_DEACTIVATE) { 250 DC_LOG_BIOS("%s:ps.param.symclk_10khz = %d\n",\ 251 __func__, ps.param.symclk_10khz); 252 } 253 254 255 /*color_depth not used any more, driver has deep color factor in the Phyclk*/ 256 if (EXEC_BIOS_CMD_TABLE(dig1transmittercontrol, ps)) 257 result = BP_RESULT_OK; 258 return result; 259 } 260 261 /****************************************************************************** 262 ****************************************************************************** 263 ** 264 ** SET PIXEL CLOCK 265 ** 266 ****************************************************************************** 267 *****************************************************************************/ 268 269 static enum bp_result set_pixel_clock_v7( 270 struct bios_parser *bp, 271 struct bp_pixel_clock_parameters *bp_params); 272 273 static void init_set_pixel_clock(struct bios_parser *bp) 274 { 275 switch (BIOS_CMD_TABLE_PARA_REVISION(setpixelclock)) { 276 case 7: 277 bp->cmd_tbl.set_pixel_clock = set_pixel_clock_v7; 278 break; 279 default: 280 dm_output_to_console("Don't have set_pixel_clock for v%d\n", 281 BIOS_CMD_TABLE_PARA_REVISION(setpixelclock)); 282 bp->cmd_tbl.set_pixel_clock = NULL; 283 break; 284 } 285 } 286 287 288 289 static enum bp_result set_pixel_clock_v7( 290 struct bios_parser *bp, 291 struct bp_pixel_clock_parameters *bp_params) 292 { 293 enum bp_result result = BP_RESULT_FAILURE; 294 struct set_pixel_clock_parameter_v1_7 clk; 295 uint8_t controller_id; 296 uint32_t pll_id; 297 298 memset(&clk, 0, sizeof(clk)); 299 300 if (bp->cmd_helper->clock_source_id_to_atom(bp_params->pll_id, &pll_id) 301 && bp->cmd_helper->controller_id_to_atom(bp_params-> 302 controller_id, &controller_id)) { 303 /* Note: VBIOS still wants to use ucCRTC name which is now 304 * 1 byte in ULONG 305 *typedef struct _CRTC_PIXEL_CLOCK_FREQ 306 *{ 307 * target the pixel clock to drive the CRTC timing. 308 * ULONG ulPixelClock:24; 309 * 0 means disable PPLL/DCPLL. Expanded to 24 bits comparing to 310 * previous version. 311 * ATOM_CRTC1~6, indicate the CRTC controller to 312 * ULONG ucCRTC:8; 313 * drive the pixel clock. not used for DCPLL case. 314 *}CRTC_PIXEL_CLOCK_FREQ; 315 *union 316 *{ 317 * pixel clock and CRTC id frequency 318 * CRTC_PIXEL_CLOCK_FREQ ulCrtcPclkFreq; 319 * ULONG ulDispEngClkFreq; dispclk frequency 320 *}; 321 */ 322 clk.crtc_id = controller_id; 323 clk.pll_id = (uint8_t) pll_id; 324 clk.encoderobjid = 325 bp->cmd_helper->encoder_id_to_atom( 326 dal_graphics_object_id_get_encoder_id( 327 bp_params->encoder_object_id)); 328 329 clk.encoder_mode = (uint8_t) bp-> 330 cmd_helper->encoder_mode_bp_to_atom( 331 bp_params->signal_type, false); 332 333 /* We need to convert from KHz units into 10KHz units */ 334 clk.pixclk_100hz = cpu_to_le32(bp_params->target_pixel_clock * 335 10); 336 337 clk.deep_color_ratio = 338 (uint8_t) bp->cmd_helper-> 339 transmitter_color_depth_to_atom( 340 bp_params->color_depth); 341 DC_LOG_BIOS("%s:program display clock = %d"\ 342 "colorDepth = %d\n", __func__,\ 343 bp_params->target_pixel_clock, bp_params->color_depth); 344 345 if (bp_params->flags.FORCE_PROGRAMMING_OF_PLL) 346 clk.miscinfo |= PIXEL_CLOCK_V7_MISC_FORCE_PROG_PPLL; 347 348 if (bp_params->flags.PROGRAM_PHY_PLL_ONLY) 349 clk.miscinfo |= PIXEL_CLOCK_V7_MISC_PROG_PHYPLL; 350 351 if (bp_params->flags.SUPPORT_YUV_420) 352 clk.miscinfo |= PIXEL_CLOCK_V7_MISC_YUV420_MODE; 353 354 if (bp_params->flags.SET_XTALIN_REF_SRC) 355 clk.miscinfo |= PIXEL_CLOCK_V7_MISC_REF_DIV_SRC_XTALIN; 356 357 if (bp_params->flags.SET_GENLOCK_REF_DIV_SRC) 358 clk.miscinfo |= PIXEL_CLOCK_V7_MISC_REF_DIV_SRC_GENLK; 359 360 if (bp_params->signal_type == SIGNAL_TYPE_DVI_DUAL_LINK) 361 clk.miscinfo |= PIXEL_CLOCK_V7_MISC_DVI_DUALLINK_EN; 362 363 if (EXEC_BIOS_CMD_TABLE(setpixelclock, clk)) 364 result = BP_RESULT_OK; 365 } 366 return result; 367 } 368 369 /****************************************************************************** 370 ****************************************************************************** 371 ** 372 ** SET CRTC TIMING 373 ** 374 ****************************************************************************** 375 *****************************************************************************/ 376 377 static enum bp_result set_crtc_using_dtd_timing_v3( 378 struct bios_parser *bp, 379 struct bp_hw_crtc_timing_parameters *bp_params); 380 381 static void init_set_crtc_timing(struct bios_parser *bp) 382 { 383 uint32_t dtd_version = 384 BIOS_CMD_TABLE_PARA_REVISION(setcrtc_usingdtdtiming); 385 386 switch (dtd_version) { 387 case 3: 388 bp->cmd_tbl.set_crtc_timing = 389 set_crtc_using_dtd_timing_v3; 390 break; 391 default: 392 dm_output_to_console("Don't have set_crtc_timing for v%d\n", dtd_version); 393 bp->cmd_tbl.set_crtc_timing = NULL; 394 break; 395 } 396 } 397 398 static enum bp_result set_crtc_using_dtd_timing_v3( 399 struct bios_parser *bp, 400 struct bp_hw_crtc_timing_parameters *bp_params) 401 { 402 enum bp_result result = BP_RESULT_FAILURE; 403 struct set_crtc_using_dtd_timing_parameters params = {0}; 404 uint8_t atom_controller_id; 405 406 if (bp->cmd_helper->controller_id_to_atom( 407 bp_params->controller_id, &atom_controller_id)) 408 params.crtc_id = atom_controller_id; 409 410 /* bios usH_Size wants h addressable size */ 411 params.h_size = cpu_to_le16((uint16_t)bp_params->h_addressable); 412 /* bios usH_Blanking_Time wants borders included in blanking */ 413 params.h_blanking_time = 414 cpu_to_le16((uint16_t)(bp_params->h_total - 415 bp_params->h_addressable)); 416 /* bios usV_Size wants v addressable size */ 417 params.v_size = cpu_to_le16((uint16_t)bp_params->v_addressable); 418 /* bios usV_Blanking_Time wants borders included in blanking */ 419 params.v_blanking_time = 420 cpu_to_le16((uint16_t)(bp_params->v_total - 421 bp_params->v_addressable)); 422 /* bios usHSyncOffset is the offset from the end of h addressable, 423 * our horizontalSyncStart is the offset from the beginning 424 * of h addressable 425 */ 426 params.h_syncoffset = 427 cpu_to_le16((uint16_t)(bp_params->h_sync_start - 428 bp_params->h_addressable)); 429 params.h_syncwidth = cpu_to_le16((uint16_t)bp_params->h_sync_width); 430 /* bios usHSyncOffset is the offset from the end of v addressable, 431 * our verticalSyncStart is the offset from the beginning of 432 * v addressable 433 */ 434 params.v_syncoffset = 435 cpu_to_le16((uint16_t)(bp_params->v_sync_start - 436 bp_params->v_addressable)); 437 params.v_syncwidth = cpu_to_le16((uint16_t)bp_params->v_sync_width); 438 439 /* we assume that overscan from original timing does not get bigger 440 * than 255 441 * we will program all the borders in the Set CRTC Overscan call below 442 */ 443 444 if (bp_params->flags.HSYNC_POSITIVE_POLARITY == 0) 445 params.modemiscinfo = 446 cpu_to_le16(le16_to_cpu(params.modemiscinfo) | 447 ATOM_HSYNC_POLARITY); 448 449 if (bp_params->flags.VSYNC_POSITIVE_POLARITY == 0) 450 params.modemiscinfo = 451 cpu_to_le16(le16_to_cpu(params.modemiscinfo) | 452 ATOM_VSYNC_POLARITY); 453 454 if (bp_params->flags.INTERLACE) { 455 params.modemiscinfo = 456 cpu_to_le16(le16_to_cpu(params.modemiscinfo) | 457 ATOM_INTERLACE); 458 459 /* original DAL code has this condition to apply this 460 * for non-TV/CV only 461 * due to complex MV testing for possible impact 462 * if ( pACParameters->signal != SignalType_YPbPr && 463 * pACParameters->signal != SignalType_Composite && 464 * pACParameters->signal != SignalType_SVideo) 465 */ 466 { 467 /* HW will deduct 0.5 line from 2nd feild. 468 * i.e. for 1080i, it is 2 lines for 1st field, 469 * 2.5 lines for the 2nd feild. we need input as 5 470 * instead of 4. 471 * but it is 4 either from Edid data (spec CEA 861) 472 * or CEA timing table. 473 */ 474 params.v_syncoffset = 475 cpu_to_le16(le16_to_cpu(params.v_syncoffset) + 476 1); 477 478 } 479 } 480 481 if (bp_params->flags.HORZ_COUNT_BY_TWO) 482 params.modemiscinfo = 483 cpu_to_le16(le16_to_cpu(params.modemiscinfo) | 484 0x100); /* ATOM_DOUBLE_CLOCK_MODE */ 485 486 if (EXEC_BIOS_CMD_TABLE(setcrtc_usingdtdtiming, params)) 487 result = BP_RESULT_OK; 488 489 return result; 490 } 491 492 /****************************************************************************** 493 ****************************************************************************** 494 ** 495 ** SELECT CRTC SOURCE 496 ** 497 ****************************************************************************** 498 *****************************************************************************/ 499 500 501 static enum bp_result select_crtc_source_v3( 502 struct bios_parser *bp, 503 struct bp_crtc_source_select *bp_params); 504 505 static void init_select_crtc_source(struct bios_parser *bp) 506 { 507 switch (BIOS_CMD_TABLE_PARA_REVISION(selectcrtc_source)) { 508 case 3: 509 bp->cmd_tbl.select_crtc_source = select_crtc_source_v3; 510 break; 511 default: 512 dm_output_to_console("Don't select_crtc_source enable_crtc for v%d\n", 513 BIOS_CMD_TABLE_PARA_REVISION(selectcrtc_source)); 514 bp->cmd_tbl.select_crtc_source = NULL; 515 break; 516 } 517 } 518 519 520 static enum bp_result select_crtc_source_v3( 521 struct bios_parser *bp, 522 struct bp_crtc_source_select *bp_params) 523 { 524 bool result = BP_RESULT_FAILURE; 525 struct select_crtc_source_parameters_v2_3 params; 526 uint8_t atom_controller_id; 527 uint32_t atom_engine_id; 528 enum signal_type s = bp_params->signal; 529 530 memset(¶ms, 0, sizeof(params)); 531 532 if (bp->cmd_helper->controller_id_to_atom(bp_params->controller_id, 533 &atom_controller_id)) 534 params.crtc_id = atom_controller_id; 535 else 536 return result; 537 538 if (bp->cmd_helper->engine_bp_to_atom(bp_params->engine_id, 539 &atom_engine_id)) 540 params.encoder_id = (uint8_t)atom_engine_id; 541 else 542 return result; 543 544 if (s == SIGNAL_TYPE_EDP || 545 (s == SIGNAL_TYPE_DISPLAY_PORT && bp_params->sink_signal == 546 SIGNAL_TYPE_LVDS)) 547 s = SIGNAL_TYPE_LVDS; 548 549 params.encode_mode = 550 bp->cmd_helper->encoder_mode_bp_to_atom( 551 s, bp_params->enable_dp_audio); 552 /* Needed for VBIOS Random Spatial Dithering feature */ 553 params.dst_bpc = (uint8_t)(bp_params->display_output_bit_depth); 554 555 if (EXEC_BIOS_CMD_TABLE(selectcrtc_source, params)) 556 result = BP_RESULT_OK; 557 558 return result; 559 } 560 561 /****************************************************************************** 562 ****************************************************************************** 563 ** 564 ** ENABLE CRTC 565 ** 566 ****************************************************************************** 567 *****************************************************************************/ 568 569 static enum bp_result enable_crtc_v1( 570 struct bios_parser *bp, 571 enum controller_id controller_id, 572 bool enable); 573 574 static void init_enable_crtc(struct bios_parser *bp) 575 { 576 switch (BIOS_CMD_TABLE_PARA_REVISION(enablecrtc)) { 577 case 1: 578 bp->cmd_tbl.enable_crtc = enable_crtc_v1; 579 break; 580 default: 581 dm_output_to_console("Don't have enable_crtc for v%d\n", 582 BIOS_CMD_TABLE_PARA_REVISION(enablecrtc)); 583 bp->cmd_tbl.enable_crtc = NULL; 584 break; 585 } 586 } 587 588 static enum bp_result enable_crtc_v1( 589 struct bios_parser *bp, 590 enum controller_id controller_id, 591 bool enable) 592 { 593 bool result = BP_RESULT_FAILURE; 594 struct enable_crtc_parameters params = {0}; 595 uint8_t id; 596 597 if (bp->cmd_helper->controller_id_to_atom(controller_id, &id)) 598 params.crtc_id = id; 599 else 600 return BP_RESULT_BADINPUT; 601 602 if (enable) 603 params.enable = ATOM_ENABLE; 604 else 605 params.enable = ATOM_DISABLE; 606 607 if (EXEC_BIOS_CMD_TABLE(enablecrtc, params)) 608 result = BP_RESULT_OK; 609 610 return result; 611 } 612 613 /****************************************************************************** 614 ****************************************************************************** 615 ** 616 ** DISPLAY PLL 617 ** 618 ****************************************************************************** 619 *****************************************************************************/ 620 621 622 623 /****************************************************************************** 624 ****************************************************************************** 625 ** 626 ** EXTERNAL ENCODER CONTROL 627 ** 628 ****************************************************************************** 629 *****************************************************************************/ 630 631 static enum bp_result external_encoder_control_v3( 632 struct bios_parser *bp, 633 struct bp_external_encoder_control *cntl); 634 635 static void init_external_encoder_control( 636 struct bios_parser *bp) 637 { 638 switch (BIOS_CMD_TABLE_PARA_REVISION(externalencodercontrol)) { 639 case 3: 640 bp->cmd_tbl.external_encoder_control = 641 external_encoder_control_v3; 642 break; 643 default: 644 bp->cmd_tbl.external_encoder_control = NULL; 645 break; 646 } 647 } 648 649 static enum bp_result external_encoder_control_v3( 650 struct bios_parser *bp, 651 struct bp_external_encoder_control *cntl) 652 { 653 /* TODO */ 654 return BP_RESULT_OK; 655 } 656 657 /****************************************************************************** 658 ****************************************************************************** 659 ** 660 ** ENABLE DISPLAY POWER GATING 661 ** 662 ****************************************************************************** 663 *****************************************************************************/ 664 665 static enum bp_result enable_disp_power_gating_v2_1( 666 struct bios_parser *bp, 667 enum controller_id crtc_id, 668 enum bp_pipe_control_action action); 669 670 static void init_enable_disp_power_gating( 671 struct bios_parser *bp) 672 { 673 switch (BIOS_CMD_TABLE_PARA_REVISION(enabledisppowergating)) { 674 case 1: 675 bp->cmd_tbl.enable_disp_power_gating = 676 enable_disp_power_gating_v2_1; 677 break; 678 default: 679 dm_output_to_console("Don't enable_disp_power_gating enable_crtc for v%d\n", 680 BIOS_CMD_TABLE_PARA_REVISION(enabledisppowergating)); 681 bp->cmd_tbl.enable_disp_power_gating = NULL; 682 break; 683 } 684 } 685 686 static enum bp_result enable_disp_power_gating_v2_1( 687 struct bios_parser *bp, 688 enum controller_id crtc_id, 689 enum bp_pipe_control_action action) 690 { 691 enum bp_result result = BP_RESULT_FAILURE; 692 693 694 struct enable_disp_power_gating_ps_allocation ps = { { 0 } }; 695 uint8_t atom_crtc_id; 696 697 if (bp->cmd_helper->controller_id_to_atom(crtc_id, &atom_crtc_id)) 698 ps.param.disp_pipe_id = atom_crtc_id; 699 else 700 return BP_RESULT_BADINPUT; 701 702 ps.param.enable = 703 bp->cmd_helper->disp_power_gating_action_to_atom(action); 704 705 if (EXEC_BIOS_CMD_TABLE(enabledisppowergating, ps.param)) 706 result = BP_RESULT_OK; 707 708 return result; 709 } 710 711 /****************************************************************************** 712 ******************************************************************************* 713 ** 714 ** SET DCE CLOCK 715 ** 716 ******************************************************************************* 717 *******************************************************************************/ 718 719 static enum bp_result set_dce_clock_v2_1( 720 struct bios_parser *bp, 721 struct bp_set_dce_clock_parameters *bp_params); 722 723 static void init_set_dce_clock(struct bios_parser *bp) 724 { 725 switch (BIOS_CMD_TABLE_PARA_REVISION(setdceclock)) { 726 case 1: 727 bp->cmd_tbl.set_dce_clock = set_dce_clock_v2_1; 728 break; 729 default: 730 dm_output_to_console("Don't have set_dce_clock for v%d\n", 731 BIOS_CMD_TABLE_PARA_REVISION(setdceclock)); 732 bp->cmd_tbl.set_dce_clock = NULL; 733 break; 734 } 735 } 736 737 static enum bp_result set_dce_clock_v2_1( 738 struct bios_parser *bp, 739 struct bp_set_dce_clock_parameters *bp_params) 740 { 741 enum bp_result result = BP_RESULT_FAILURE; 742 743 struct set_dce_clock_ps_allocation_v2_1 params; 744 uint32_t atom_pll_id; 745 uint32_t atom_clock_type; 746 const struct command_table_helper *cmd = bp->cmd_helper; 747 748 memset(¶ms, 0, sizeof(params)); 749 750 if (!cmd->clock_source_id_to_atom(bp_params->pll_id, &atom_pll_id) || 751 !cmd->dc_clock_type_to_atom(bp_params->clock_type, 752 &atom_clock_type)) 753 return BP_RESULT_BADINPUT; 754 755 params.param.dceclksrc = atom_pll_id; 756 params.param.dceclktype = atom_clock_type; 757 758 if (bp_params->clock_type == DCECLOCK_TYPE_DPREFCLK) { 759 if (bp_params->flags.USE_GENLOCK_AS_SOURCE_FOR_DPREFCLK) 760 params.param.dceclkflag |= 761 DCE_CLOCK_FLAG_PLL_REFCLK_SRC_GENLK; 762 763 if (bp_params->flags.USE_PCIE_AS_SOURCE_FOR_DPREFCLK) 764 params.param.dceclkflag |= 765 DCE_CLOCK_FLAG_PLL_REFCLK_SRC_PCIE; 766 767 if (bp_params->flags.USE_XTALIN_AS_SOURCE_FOR_DPREFCLK) 768 params.param.dceclkflag |= 769 DCE_CLOCK_FLAG_PLL_REFCLK_SRC_XTALIN; 770 771 if (bp_params->flags.USE_GENERICA_AS_SOURCE_FOR_DPREFCLK) 772 params.param.dceclkflag |= 773 DCE_CLOCK_FLAG_PLL_REFCLK_SRC_GENERICA; 774 } else 775 /* only program clock frequency if display clock is used; 776 * VBIOS will program DPREFCLK 777 * We need to convert from KHz units into 10KHz units 778 */ 779 params.param.dceclk_10khz = cpu_to_le32( 780 bp_params->target_clock_frequency / 10); 781 DC_LOG_BIOS("%s:target_clock_frequency = %d"\ 782 "clock_type = %d \n", __func__,\ 783 bp_params->target_clock_frequency,\ 784 bp_params->clock_type); 785 786 if (EXEC_BIOS_CMD_TABLE(setdceclock, params)) { 787 /* Convert from 10KHz units back to KHz */ 788 bp_params->target_clock_frequency = le32_to_cpu( 789 params.param.dceclk_10khz) * 10; 790 result = BP_RESULT_OK; 791 } 792 793 return result; 794 } 795 796 797 /****************************************************************************** 798 ****************************************************************************** 799 ** 800 ** GET SMU CLOCK INFO 801 ** 802 ****************************************************************************** 803 *****************************************************************************/ 804 805 static unsigned int get_smu_clock_info_v3_1(struct bios_parser *bp, uint8_t id); 806 807 static void init_get_smu_clock_info(struct bios_parser *bp) 808 { 809 /* TODO add switch for table vrsion */ 810 bp->cmd_tbl.get_smu_clock_info = get_smu_clock_info_v3_1; 811 812 } 813 814 static unsigned int get_smu_clock_info_v3_1(struct bios_parser *bp, uint8_t id) 815 { 816 struct atom_get_smu_clock_info_parameters_v3_1 smu_input = {0}; 817 struct atom_get_smu_clock_info_output_parameters_v3_1 smu_output; 818 819 smu_input.command = GET_SMU_CLOCK_INFO_V3_1_GET_PLLVCO_FREQ; 820 smu_input.syspll_id = id; 821 822 /* Get Specific Clock */ 823 if (EXEC_BIOS_CMD_TABLE(getsmuclockinfo, smu_input)) { 824 memmove(&smu_output, &smu_input, sizeof( 825 struct atom_get_smu_clock_info_parameters_v3_1)); 826 return smu_output.atom_smu_outputclkfreq.syspllvcofreq_10khz; 827 } 828 829 return 0; 830 } 831 832