Linux Audio

Check our new training course

Loading...
v6.13.7
   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#include "amdgpu.h"
  28#include "atom.h"
  29
  30#include "include/bios_parser_interface.h"
  31
  32#include "command_table.h"
  33#include "command_table_helper.h"
  34#include "bios_parser_helper.h"
  35#include "bios_parser_types_internal.h"
  36
  37#define EXEC_BIOS_CMD_TABLE(command, params)\
  38	(amdgpu_atom_execute_table(((struct amdgpu_device *)bp->base.ctx->driver_context)->mode_info.atom_context, \
  39		GetIndexIntoMasterTable(COMMAND, command), \
  40		(uint32_t *)&params, sizeof(params)) == 0)
  41
  42#define BIOS_CMD_TABLE_REVISION(command, frev, crev)\
  43	amdgpu_atom_parse_cmd_header(((struct amdgpu_device *)bp->base.ctx->driver_context)->mode_info.atom_context, \
  44		GetIndexIntoMasterTable(COMMAND, command), &frev, &crev)
  45
  46#define BIOS_CMD_TABLE_PARA_REVISION(command)\
  47	bios_cmd_table_para_revision(bp->base.ctx->driver_context, \
  48		GetIndexIntoMasterTable(COMMAND, command))
  49
  50static void init_dig_encoder_control(struct bios_parser *bp);
  51static void init_transmitter_control(struct bios_parser *bp);
  52static void init_set_pixel_clock(struct bios_parser *bp);
  53static void init_enable_spread_spectrum_on_ppll(struct bios_parser *bp);
  54static void init_adjust_display_pll(struct bios_parser *bp);
  55static void init_dac_encoder_control(struct bios_parser *bp);
  56static void init_dac_output_control(struct bios_parser *bp);
  57static void init_set_crtc_timing(struct bios_parser *bp);
 
  58static void init_enable_crtc(struct bios_parser *bp);
  59static void init_enable_crtc_mem_req(struct bios_parser *bp);
  60static void init_external_encoder_control(struct bios_parser *bp);
  61static void init_enable_disp_power_gating(struct bios_parser *bp);
  62static void init_program_clock(struct bios_parser *bp);
  63static void init_set_dce_clock(struct bios_parser *bp);
  64
  65void dal_bios_parser_init_cmd_tbl(struct bios_parser *bp)
  66{
  67	init_dig_encoder_control(bp);
  68	init_transmitter_control(bp);
  69	init_set_pixel_clock(bp);
  70	init_enable_spread_spectrum_on_ppll(bp);
  71	init_adjust_display_pll(bp);
  72	init_dac_encoder_control(bp);
  73	init_dac_output_control(bp);
  74	init_set_crtc_timing(bp);
 
  75	init_enable_crtc(bp);
  76	init_enable_crtc_mem_req(bp);
  77	init_program_clock(bp);
  78	init_external_encoder_control(bp);
  79	init_enable_disp_power_gating(bp);
  80	init_set_dce_clock(bp);
  81}
  82
  83static uint32_t bios_cmd_table_para_revision(void *dev,
  84					     uint32_t index)
  85{
  86	struct amdgpu_device *adev = dev;
  87	uint8_t frev, crev;
  88
  89	if (amdgpu_atom_parse_cmd_header(adev->mode_info.atom_context,
  90					index,
  91					&frev, &crev))
  92		return crev;
  93	else
  94		return 0;
 
  95}
  96
  97/*******************************************************************************
  98 ********************************************************************************
  99 **
 100 **                  D I G E N C O D E R C O N T R O L
 101 **
 102 ********************************************************************************
 103 *******************************************************************************/
 104static enum bp_result encoder_control_digx_v3(
 105	struct bios_parser *bp,
 106	struct bp_encoder_control *cntl);
 107
 108static enum bp_result encoder_control_digx_v4(
 109	struct bios_parser *bp,
 110	struct bp_encoder_control *cntl);
 111
 112static enum bp_result encoder_control_digx_v5(
 113	struct bios_parser *bp,
 114	struct bp_encoder_control *cntl);
 115
 116static void init_encoder_control_dig_v1(struct bios_parser *bp);
 117
 118static void init_dig_encoder_control(struct bios_parser *bp)
 119{
 120	uint32_t version =
 121		BIOS_CMD_TABLE_PARA_REVISION(DIGxEncoderControl);
 122
 123	switch (version) {
 124	case 2:
 125		bp->cmd_tbl.dig_encoder_control = encoder_control_digx_v3;
 126		break;
 127	case 4:
 128		bp->cmd_tbl.dig_encoder_control = encoder_control_digx_v4;
 129		break;
 130
 131	case 5:
 132		bp->cmd_tbl.dig_encoder_control = encoder_control_digx_v5;
 133		break;
 134
 135	default:
 136		init_encoder_control_dig_v1(bp);
 137		break;
 138	}
 139}
 140
 141static enum bp_result encoder_control_dig_v1(
 142	struct bios_parser *bp,
 143	struct bp_encoder_control *cntl);
 144static enum bp_result encoder_control_dig1_v1(
 145	struct bios_parser *bp,
 146	struct bp_encoder_control *cntl);
 147static enum bp_result encoder_control_dig2_v1(
 148	struct bios_parser *bp,
 149	struct bp_encoder_control *cntl);
 150
 151static void init_encoder_control_dig_v1(struct bios_parser *bp)
 152{
 153	struct cmd_tbl *cmd_tbl = &bp->cmd_tbl;
 154
 155	if (1 == BIOS_CMD_TABLE_PARA_REVISION(DIG1EncoderControl))
 156		cmd_tbl->encoder_control_dig1 = encoder_control_dig1_v1;
 157	else
 158		cmd_tbl->encoder_control_dig1 = NULL;
 159
 160	if (1 == BIOS_CMD_TABLE_PARA_REVISION(DIG2EncoderControl))
 161		cmd_tbl->encoder_control_dig2 = encoder_control_dig2_v1;
 162	else
 163		cmd_tbl->encoder_control_dig2 = NULL;
 164
 165	cmd_tbl->dig_encoder_control = encoder_control_dig_v1;
 166}
 167
 168static enum bp_result encoder_control_dig_v1(
 169	struct bios_parser *bp,
 170	struct bp_encoder_control *cntl)
 171{
 172	enum bp_result result = BP_RESULT_FAILURE;
 173	struct cmd_tbl *cmd_tbl = &bp->cmd_tbl;
 174
 175	if (cntl != NULL)
 176		switch (cntl->engine_id) {
 177		case ENGINE_ID_DIGA:
 178			if (cmd_tbl->encoder_control_dig1 != NULL)
 179				result =
 180					cmd_tbl->encoder_control_dig1(bp, cntl);
 181			break;
 182		case ENGINE_ID_DIGB:
 183			if (cmd_tbl->encoder_control_dig2 != NULL)
 184				result =
 185					cmd_tbl->encoder_control_dig2(bp, cntl);
 186			break;
 187
 188		default:
 189			break;
 190		}
 191
 192	return result;
 193}
 194
 195static enum bp_result encoder_control_dig1_v1(
 196	struct bios_parser *bp,
 197	struct bp_encoder_control *cntl)
 198{
 199	enum bp_result result = BP_RESULT_FAILURE;
 200	DIG_ENCODER_CONTROL_PARAMETERS_V2 params = {0};
 201
 202	bp->cmd_helper->assign_control_parameter(bp->cmd_helper, cntl, &params);
 203
 204	if (EXEC_BIOS_CMD_TABLE(DIG1EncoderControl, params))
 205		result = BP_RESULT_OK;
 206
 207	return result;
 208}
 209
 210static enum bp_result encoder_control_dig2_v1(
 211	struct bios_parser *bp,
 212	struct bp_encoder_control *cntl)
 213{
 214	enum bp_result result = BP_RESULT_FAILURE;
 215	DIG_ENCODER_CONTROL_PARAMETERS_V2 params = {0};
 216
 217	bp->cmd_helper->assign_control_parameter(bp->cmd_helper, cntl, &params);
 218
 219	if (EXEC_BIOS_CMD_TABLE(DIG2EncoderControl, params))
 220		result = BP_RESULT_OK;
 221
 222	return result;
 223}
 224
 225static enum bp_result encoder_control_digx_v3(
 226	struct bios_parser *bp,
 227	struct bp_encoder_control *cntl)
 228{
 229	enum bp_result result = BP_RESULT_FAILURE;
 230	DIG_ENCODER_CONTROL_PARAMETERS_V3 params = {0};
 231
 232	if (LANE_COUNT_FOUR < cntl->lanes_number)
 233		params.acConfig.ucDPLinkRate = 1; /* dual link 2.7GHz */
 234	else
 235		params.acConfig.ucDPLinkRate = 0; /* single link 1.62GHz */
 236
 237	params.acConfig.ucDigSel = (uint8_t)(cntl->engine_id);
 238
 239	/* We need to convert from KHz units into 10KHz units */
 240	params.ucAction = bp->cmd_helper->encoder_action_to_atom(cntl->action);
 241	params.usPixelClock = cpu_to_le16((uint16_t)(cntl->pixel_clock / 10));
 242	params.ucEncoderMode =
 243			(uint8_t)bp->cmd_helper->encoder_mode_bp_to_atom(
 244					cntl->signal,
 245					cntl->enable_dp_audio);
 246	params.ucLaneNum = (uint8_t)(cntl->lanes_number);
 247
 248	switch (cntl->color_depth) {
 249	case COLOR_DEPTH_888:
 250		params.ucBitPerColor = PANEL_8BIT_PER_COLOR;
 251		break;
 252	case COLOR_DEPTH_101010:
 253		params.ucBitPerColor = PANEL_10BIT_PER_COLOR;
 254		break;
 255	case COLOR_DEPTH_121212:
 256		params.ucBitPerColor = PANEL_12BIT_PER_COLOR;
 257		break;
 258	case COLOR_DEPTH_161616:
 259		params.ucBitPerColor = PANEL_16BIT_PER_COLOR;
 260		break;
 261	default:
 262		break;
 263	}
 264
 265	if (EXEC_BIOS_CMD_TABLE(DIGxEncoderControl, params))
 266		result = BP_RESULT_OK;
 267
 268	return result;
 269}
 270
 271static enum bp_result encoder_control_digx_v4(
 272	struct bios_parser *bp,
 273	struct bp_encoder_control *cntl)
 274{
 275	enum bp_result result = BP_RESULT_FAILURE;
 276	DIG_ENCODER_CONTROL_PARAMETERS_V4 params = {0};
 277
 278	if (LANE_COUNT_FOUR < cntl->lanes_number)
 279		params.acConfig.ucDPLinkRate = 1; /* dual link 2.7GHz */
 280	else
 281		params.acConfig.ucDPLinkRate = 0; /* single link 1.62GHz */
 282
 283	params.acConfig.ucDigSel = (uint8_t)(cntl->engine_id);
 284
 285	/* We need to convert from KHz units into 10KHz units */
 286	params.ucAction = bp->cmd_helper->encoder_action_to_atom(cntl->action);
 287	params.usPixelClock = cpu_to_le16((uint16_t)(cntl->pixel_clock / 10));
 288	params.ucEncoderMode =
 289			(uint8_t)(bp->cmd_helper->encoder_mode_bp_to_atom(
 290					cntl->signal,
 291					cntl->enable_dp_audio));
 292	params.ucLaneNum = (uint8_t)(cntl->lanes_number);
 293
 294	switch (cntl->color_depth) {
 295	case COLOR_DEPTH_888:
 296		params.ucBitPerColor = PANEL_8BIT_PER_COLOR;
 297		break;
 298	case COLOR_DEPTH_101010:
 299		params.ucBitPerColor = PANEL_10BIT_PER_COLOR;
 300		break;
 301	case COLOR_DEPTH_121212:
 302		params.ucBitPerColor = PANEL_12BIT_PER_COLOR;
 303		break;
 304	case COLOR_DEPTH_161616:
 305		params.ucBitPerColor = PANEL_16BIT_PER_COLOR;
 306		break;
 307	default:
 308		break;
 309	}
 310
 311	if (EXEC_BIOS_CMD_TABLE(DIGxEncoderControl, params))
 312		result = BP_RESULT_OK;
 313
 314	return result;
 315}
 316
 317static enum bp_result encoder_control_digx_v5(
 318	struct bios_parser *bp,
 319	struct bp_encoder_control *cntl)
 320{
 321	enum bp_result result = BP_RESULT_FAILURE;
 322	ENCODER_STREAM_SETUP_PARAMETERS_V5 params = {0};
 323
 324	params.ucDigId = (uint8_t)(cntl->engine_id);
 325	params.ucAction = bp->cmd_helper->encoder_action_to_atom(cntl->action);
 326
 327	params.ulPixelClock = cntl->pixel_clock / 10;
 328	params.ucDigMode =
 329			(uint8_t)(bp->cmd_helper->encoder_mode_bp_to_atom(
 330					cntl->signal,
 331					cntl->enable_dp_audio));
 332	params.ucLaneNum = (uint8_t)(cntl->lanes_number);
 333
 334	switch (cntl->color_depth) {
 335	case COLOR_DEPTH_888:
 336		params.ucBitPerColor = PANEL_8BIT_PER_COLOR;
 337		break;
 338	case COLOR_DEPTH_101010:
 339		params.ucBitPerColor = PANEL_10BIT_PER_COLOR;
 340		break;
 341	case COLOR_DEPTH_121212:
 342		params.ucBitPerColor = PANEL_12BIT_PER_COLOR;
 343		break;
 344	case COLOR_DEPTH_161616:
 345		params.ucBitPerColor = PANEL_16BIT_PER_COLOR;
 346		break;
 347	default:
 348		break;
 349	}
 350
 351	if (cntl->signal == SIGNAL_TYPE_HDMI_TYPE_A)
 352		switch (cntl->color_depth) {
 353		case COLOR_DEPTH_101010:
 354			params.ulPixelClock =
 355				(params.ulPixelClock * 30) / 24;
 356			break;
 357		case COLOR_DEPTH_121212:
 358			params.ulPixelClock =
 359				(params.ulPixelClock * 36) / 24;
 360			break;
 361		case COLOR_DEPTH_161616:
 362			params.ulPixelClock =
 363				(params.ulPixelClock * 48) / 24;
 364			break;
 365		default:
 366			break;
 367		}
 368
 369	if (EXEC_BIOS_CMD_TABLE(DIGxEncoderControl, params))
 370		result = BP_RESULT_OK;
 371
 372	return result;
 373}
 374
 375/*******************************************************************************
 376 ********************************************************************************
 377 **
 378 **                  TRANSMITTER CONTROL
 379 **
 380 ********************************************************************************
 381 *******************************************************************************/
 382
 383static enum bp_result transmitter_control_v2(
 384	struct bios_parser *bp,
 385	struct bp_transmitter_control *cntl);
 386static enum bp_result transmitter_control_v3(
 387	struct bios_parser *bp,
 388	struct bp_transmitter_control *cntl);
 389static enum bp_result transmitter_control_v4(
 390	struct bios_parser *bp,
 391	struct bp_transmitter_control *cntl);
 392static enum bp_result transmitter_control_v1_5(
 393	struct bios_parser *bp,
 394	struct bp_transmitter_control *cntl);
 395static enum bp_result transmitter_control_v1_6(
 396	struct bios_parser *bp,
 397	struct bp_transmitter_control *cntl);
 398
 399static void init_transmitter_control(struct bios_parser *bp)
 400{
 401	uint8_t frev;
 402	uint8_t crev = 0;
 403
 404	if (BIOS_CMD_TABLE_REVISION(UNIPHYTransmitterControl,
 405			frev, crev) == false)
 406		BREAK_TO_DEBUGGER();
 407	switch (crev) {
 408	case 2:
 409		bp->cmd_tbl.transmitter_control = transmitter_control_v2;
 410		break;
 411	case 3:
 412		bp->cmd_tbl.transmitter_control = transmitter_control_v3;
 413		break;
 414	case 4:
 415		bp->cmd_tbl.transmitter_control = transmitter_control_v4;
 416		break;
 417	case 5:
 418		bp->cmd_tbl.transmitter_control = transmitter_control_v1_5;
 419		break;
 420	case 6:
 421		bp->cmd_tbl.transmitter_control = transmitter_control_v1_6;
 422		break;
 423	default:
 424		dm_output_to_console("Don't have transmitter_control for v%d\n", crev);
 425		bp->cmd_tbl.transmitter_control = NULL;
 426		break;
 427	}
 428}
 429
 430static enum bp_result transmitter_control_v2(
 431	struct bios_parser *bp,
 432	struct bp_transmitter_control *cntl)
 433{
 434	enum bp_result result = BP_RESULT_FAILURE;
 435	DIG_TRANSMITTER_CONTROL_PARAMETERS_V2 params;
 436	enum connector_id connector_id =
 437		dal_graphics_object_id_get_connector_id(cntl->connector_obj_id);
 438
 439	memset(&params, 0, sizeof(params));
 440
 441	switch (cntl->transmitter) {
 442	case TRANSMITTER_UNIPHY_A:
 443	case TRANSMITTER_UNIPHY_B:
 444	case TRANSMITTER_UNIPHY_C:
 445	case TRANSMITTER_UNIPHY_D:
 446	case TRANSMITTER_UNIPHY_E:
 447	case TRANSMITTER_UNIPHY_F:
 448	case TRANSMITTER_TRAVIS_LCD:
 449		break;
 450	default:
 451		return BP_RESULT_BADINPUT;
 452	}
 453
 454	switch (cntl->action) {
 455	case TRANSMITTER_CONTROL_INIT:
 456		if ((CONNECTOR_ID_DUAL_LINK_DVII == connector_id) ||
 457				(CONNECTOR_ID_DUAL_LINK_DVID == connector_id))
 458			/* on INIT this bit should be set according to the
 459			 * physical connector
 460			 * Bit0: dual link connector flag
 461			 * =0 connector is single link connector
 462			 * =1 connector is dual link connector
 463			 */
 464			params.acConfig.fDualLinkConnector = 1;
 465
 466		/* connector object id */
 467		params.usInitInfo =
 468				cpu_to_le16((uint8_t)cntl->connector_obj_id.id);
 469		break;
 470	case TRANSMITTER_CONTROL_SET_VOLTAGE_AND_PREEMPASIS:
 471		/* voltage swing and pre-emphsis */
 472		params.asMode.ucLaneSel = (uint8_t)cntl->lane_select;
 473		params.asMode.ucLaneSet = (uint8_t)cntl->lane_settings;
 474		break;
 475	default:
 476		/* if dual-link */
 477		if (LANE_COUNT_FOUR < cntl->lanes_number) {
 478			/* on ENABLE/DISABLE this bit should be set according to
 479			 * actual timing (number of lanes)
 480			 * Bit0: dual link connector flag
 481			 * =0 connector is single link connector
 482			 * =1 connector is dual link connector
 483			 */
 484			params.acConfig.fDualLinkConnector = 1;
 485
 486			/* link rate, half for dual link
 487			 * We need to convert from KHz units into 20KHz units
 488			 */
 489			params.usPixelClock =
 490					cpu_to_le16((uint16_t)(cntl->pixel_clock / 20));
 491		} else
 492			/* link rate, half for dual link
 493			 * We need to convert from KHz units into 10KHz units
 494			 */
 495			params.usPixelClock =
 496					cpu_to_le16((uint16_t)(cntl->pixel_clock / 10));
 497		break;
 498	}
 499
 500	/* 00 - coherent mode
 501	 * 01 - incoherent mode
 502	 */
 503
 504	params.acConfig.fCoherentMode = cntl->coherent;
 505
 506	if ((TRANSMITTER_UNIPHY_B == cntl->transmitter)
 507			|| (TRANSMITTER_UNIPHY_D == cntl->transmitter)
 508			|| (TRANSMITTER_UNIPHY_F == cntl->transmitter))
 509		/* Bit2: Transmitter Link selection
 510		 * =0 when bit0=0, single link A/C/E, when bit0=1,
 511		 * master link A/C/E
 512		 * =1 when bit0=0, single link B/D/F, when bit0=1,
 513		 * master link B/D/F
 514		 */
 515		params.acConfig.ucLinkSel = 1;
 516
 517	if (ENGINE_ID_DIGB == cntl->engine_id)
 518		/* Bit3: Transmitter data source selection
 519		 * =0 DIGA is data source.
 520		 * =1 DIGB is data source.
 521		 * This bit is only useful when ucAction= ATOM_ENABLE
 522		 */
 523		params.acConfig.ucEncoderSel = 1;
 524
 525	if (CONNECTOR_ID_DISPLAY_PORT == connector_id ||
 526	    CONNECTOR_ID_USBC == connector_id)
 527		/* Bit4: DP connector flag
 528		 * =0 connector is none-DP connector
 529		 * =1 connector is DP connector
 530		 */
 531		params.acConfig.fDPConnector = 1;
 532
 533	/* Bit[7:6]: Transmitter selection
 534	 * =0 UNIPHY_ENCODER: UNIPHYA/B
 535	 * =1 UNIPHY1_ENCODER: UNIPHYC/D
 536	 * =2 UNIPHY2_ENCODER: UNIPHYE/F
 537	 * =3 reserved
 538	 */
 539	params.acConfig.ucTransmitterSel =
 540			(uint8_t)bp->cmd_helper->transmitter_bp_to_atom(
 541					cntl->transmitter);
 542
 543	params.ucAction = (uint8_t)cntl->action;
 544
 545	if (EXEC_BIOS_CMD_TABLE(UNIPHYTransmitterControl, params))
 546		result = BP_RESULT_OK;
 547
 548	return result;
 549}
 550
 551static enum bp_result transmitter_control_v3(
 552	struct bios_parser *bp,
 553	struct bp_transmitter_control *cntl)
 554{
 555	enum bp_result result = BP_RESULT_FAILURE;
 556	DIG_TRANSMITTER_CONTROL_PARAMETERS_V3 params;
 557	uint32_t pll_id;
 558	enum connector_id conn_id =
 559			dal_graphics_object_id_get_connector_id(cntl->connector_obj_id);
 560	const struct command_table_helper *cmd = bp->cmd_helper;
 561	bool dual_link_conn = (CONNECTOR_ID_DUAL_LINK_DVII == conn_id)
 562					|| (CONNECTOR_ID_DUAL_LINK_DVID == conn_id);
 563
 564	memset(&params, 0, sizeof(params));
 565
 566	switch (cntl->transmitter) {
 567	case TRANSMITTER_UNIPHY_A:
 568	case TRANSMITTER_UNIPHY_B:
 569	case TRANSMITTER_UNIPHY_C:
 570	case TRANSMITTER_UNIPHY_D:
 571	case TRANSMITTER_UNIPHY_E:
 572	case TRANSMITTER_UNIPHY_F:
 573	case TRANSMITTER_TRAVIS_LCD:
 574		break;
 575	default:
 576		return BP_RESULT_BADINPUT;
 577	}
 578
 579	if (!cmd->clock_source_id_to_atom(cntl->pll_id, &pll_id))
 580		return BP_RESULT_BADINPUT;
 581
 582	/* fill information based on the action */
 583	switch (cntl->action) {
 584	case TRANSMITTER_CONTROL_INIT:
 585		if (dual_link_conn) {
 586			/* on INIT this bit should be set according to the
 587			 * phisycal connector
 588			 * Bit0: dual link connector flag
 589			 * =0 connector is single link connector
 590			 * =1 connector is dual link connector
 591			 */
 592			params.acConfig.fDualLinkConnector = 1;
 593		}
 594
 595		/* connector object id */
 596		params.usInitInfo =
 597				cpu_to_le16((uint8_t)(cntl->connector_obj_id.id));
 598		break;
 599	case TRANSMITTER_CONTROL_SET_VOLTAGE_AND_PREEMPASIS:
 600		/* votage swing and pre-emphsis */
 601		params.asMode.ucLaneSel = (uint8_t)cntl->lane_select;
 602		params.asMode.ucLaneSet = (uint8_t)cntl->lane_settings;
 603		break;
 604	default:
 605		if (dual_link_conn && cntl->multi_path)
 606			/* on ENABLE/DISABLE this bit should be set according to
 607			 * actual timing (number of lanes)
 608			 * Bit0: dual link connector flag
 609			 * =0 connector is single link connector
 610			 * =1 connector is dual link connector
 611			 */
 612			params.acConfig.fDualLinkConnector = 1;
 613
 614		/* if dual-link */
 615		if (LANE_COUNT_FOUR < cntl->lanes_number) {
 616			/* on ENABLE/DISABLE this bit should be set according to
 617			 * actual timing (number of lanes)
 618			 * Bit0: dual link connector flag
 619			 * =0 connector is single link connector
 620			 * =1 connector is dual link connector
 621			 */
 622			params.acConfig.fDualLinkConnector = 1;
 623
 624			/* link rate, half for dual link
 625			 * We need to convert from KHz units into 20KHz units
 626			 */
 627			params.usPixelClock =
 628					cpu_to_le16((uint16_t)(cntl->pixel_clock / 20));
 629		} else {
 630			/* link rate, half for dual link
 631			 * We need to convert from KHz units into 10KHz units
 632			 */
 633			params.usPixelClock =
 634					cpu_to_le16((uint16_t)(cntl->pixel_clock / 10));
 635		}
 636		break;
 637	}
 638
 639	/* 00 - coherent mode
 640	 * 01 - incoherent mode
 641	 */
 642
 643	params.acConfig.fCoherentMode = cntl->coherent;
 644
 645	if ((TRANSMITTER_UNIPHY_B == cntl->transmitter)
 646		|| (TRANSMITTER_UNIPHY_D == cntl->transmitter)
 647		|| (TRANSMITTER_UNIPHY_F == cntl->transmitter))
 648		/* Bit2: Transmitter Link selection
 649		 * =0 when bit0=0, single link A/C/E, when bit0=1,
 650		 * master link A/C/E
 651		 * =1 when bit0=0, single link B/D/F, when bit0=1,
 652		 * master link B/D/F
 653		 */
 654		params.acConfig.ucLinkSel = 1;
 655
 656	if (ENGINE_ID_DIGB == cntl->engine_id)
 657		/* Bit3: Transmitter data source selection
 658		 * =0 DIGA is data source.
 659		 * =1 DIGB is data source.
 660		 * This bit is only useful when ucAction= ATOM_ENABLE
 661		 */
 662		params.acConfig.ucEncoderSel = 1;
 663
 664	/* Bit[7:6]: Transmitter selection
 665	 * =0 UNIPHY_ENCODER: UNIPHYA/B
 666	 * =1 UNIPHY1_ENCODER: UNIPHYC/D
 667	 * =2 UNIPHY2_ENCODER: UNIPHYE/F
 668	 * =3 reserved
 669	 */
 670	params.acConfig.ucTransmitterSel =
 671			(uint8_t)cmd->transmitter_bp_to_atom(cntl->transmitter);
 672
 673	params.ucLaneNum = (uint8_t)cntl->lanes_number;
 674
 675	params.acConfig.ucRefClkSource = (uint8_t)pll_id;
 676
 677	params.ucAction = (uint8_t)cntl->action;
 678
 679	if (EXEC_BIOS_CMD_TABLE(UNIPHYTransmitterControl, params))
 680		result = BP_RESULT_OK;
 681
 682	return result;
 683}
 684
 685static enum bp_result transmitter_control_v4(
 686	struct bios_parser *bp,
 687	struct bp_transmitter_control *cntl)
 688{
 689	enum bp_result result = BP_RESULT_FAILURE;
 690	DIG_TRANSMITTER_CONTROL_PARAMETERS_V4 params;
 691	uint32_t ref_clk_src_id;
 692	enum connector_id conn_id =
 693			dal_graphics_object_id_get_connector_id(cntl->connector_obj_id);
 694	const struct command_table_helper *cmd = bp->cmd_helper;
 695
 696	memset(&params, 0, sizeof(params));
 697
 698	switch (cntl->transmitter) {
 699	case TRANSMITTER_UNIPHY_A:
 700	case TRANSMITTER_UNIPHY_B:
 701	case TRANSMITTER_UNIPHY_C:
 702	case TRANSMITTER_UNIPHY_D:
 703	case TRANSMITTER_UNIPHY_E:
 704	case TRANSMITTER_UNIPHY_F:
 705	case TRANSMITTER_TRAVIS_LCD:
 706		break;
 707	default:
 708		return BP_RESULT_BADINPUT;
 709	}
 710
 711	if (!cmd->clock_source_id_to_ref_clk_src(cntl->pll_id, &ref_clk_src_id))
 712		return BP_RESULT_BADINPUT;
 713
 714	switch (cntl->action) {
 715	case TRANSMITTER_CONTROL_INIT:
 716	{
 717		if ((CONNECTOR_ID_DUAL_LINK_DVII == conn_id) ||
 718				(CONNECTOR_ID_DUAL_LINK_DVID == conn_id))
 719			/* on INIT this bit should be set according to the
 720			 * phisycal connector
 721			 * Bit0: dual link connector flag
 722			 * =0 connector is single link connector
 723			 * =1 connector is dual link connector
 724			 */
 725			params.acConfig.fDualLinkConnector = 1;
 726
 727		/* connector object id */
 728		params.usInitInfo =
 729				cpu_to_le16((uint8_t)(cntl->connector_obj_id.id));
 730	}
 731	break;
 732	case TRANSMITTER_CONTROL_SET_VOLTAGE_AND_PREEMPASIS:
 733		/* votage swing and pre-emphsis */
 734		params.asMode.ucLaneSel = (uint8_t)(cntl->lane_select);
 735		params.asMode.ucLaneSet = (uint8_t)(cntl->lane_settings);
 736		break;
 737	default:
 738		if ((CONNECTOR_ID_DUAL_LINK_DVII == conn_id) ||
 739				(CONNECTOR_ID_DUAL_LINK_DVID == conn_id))
 740			/* on ENABLE/DISABLE this bit should be set according to
 741			 * actual timing (number of lanes)
 742			 * Bit0: dual link connector flag
 743			 * =0 connector is single link connector
 744			 * =1 connector is dual link connector
 745			 */
 746			params.acConfig.fDualLinkConnector = 1;
 747
 748		/* if dual-link */
 749		if (LANE_COUNT_FOUR < cntl->lanes_number)
 750			/* link rate, half for dual link
 751			 * We need to convert from KHz units into 20KHz units
 752			 */
 753			params.usPixelClock =
 754					cpu_to_le16((uint16_t)(cntl->pixel_clock / 20));
 755		else {
 756			/* link rate, half for dual link
 757			 * We need to convert from KHz units into 10KHz units
 758			 */
 759			params.usPixelClock =
 760					cpu_to_le16((uint16_t)(cntl->pixel_clock / 10));
 761		}
 762		break;
 763	}
 764
 765	/* 00 - coherent mode
 766	 * 01 - incoherent mode
 767	 */
 768
 769	params.acConfig.fCoherentMode = cntl->coherent;
 770
 771	if ((TRANSMITTER_UNIPHY_B == cntl->transmitter)
 772		|| (TRANSMITTER_UNIPHY_D == cntl->transmitter)
 773		|| (TRANSMITTER_UNIPHY_F == cntl->transmitter))
 774		/* Bit2: Transmitter Link selection
 775		 * =0 when bit0=0, single link A/C/E, when bit0=1,
 776		 * master link A/C/E
 777		 * =1 when bit0=0, single link B/D/F, when bit0=1,
 778		 * master link B/D/F
 779		 */
 780		params.acConfig.ucLinkSel = 1;
 781
 782	if (ENGINE_ID_DIGB == cntl->engine_id)
 783		/* Bit3: Transmitter data source selection
 784		 * =0 DIGA is data source.
 785		 * =1 DIGB is data source.
 786		 * This bit is only useful when ucAction= ATOM_ENABLE
 787		 */
 788		params.acConfig.ucEncoderSel = 1;
 789
 790	/* Bit[7:6]: Transmitter selection
 791	 * =0 UNIPHY_ENCODER: UNIPHYA/B
 792	 * =1 UNIPHY1_ENCODER: UNIPHYC/D
 793	 * =2 UNIPHY2_ENCODER: UNIPHYE/F
 794	 * =3 reserved
 795	 */
 796	params.acConfig.ucTransmitterSel =
 797		(uint8_t)(cmd->transmitter_bp_to_atom(cntl->transmitter));
 798	params.ucLaneNum = (uint8_t)(cntl->lanes_number);
 799	params.acConfig.ucRefClkSource = (uint8_t)(ref_clk_src_id);
 800	params.ucAction = (uint8_t)(cntl->action);
 801
 802	if (EXEC_BIOS_CMD_TABLE(UNIPHYTransmitterControl, params))
 803		result = BP_RESULT_OK;
 804
 805	return result;
 806}
 807
 808static enum bp_result transmitter_control_v1_5(
 809	struct bios_parser *bp,
 810	struct bp_transmitter_control *cntl)
 811{
 812	enum bp_result result = BP_RESULT_FAILURE;
 813	const struct command_table_helper *cmd = bp->cmd_helper;
 814	DIG_TRANSMITTER_CONTROL_PARAMETERS_V1_5 params;
 815
 816	memset(&params, 0, sizeof(params));
 817	params.ucPhyId = cmd->phy_id_to_atom(cntl->transmitter);
 818	params.ucAction = (uint8_t)cntl->action;
 819	params.ucLaneNum = (uint8_t)cntl->lanes_number;
 820	params.ucConnObjId = (uint8_t)cntl->connector_obj_id.id;
 821
 822	params.ucDigMode =
 823		cmd->signal_type_to_atom_dig_mode(cntl->signal);
 824	params.asConfig.ucPhyClkSrcId =
 825		cmd->clock_source_id_to_atom_phy_clk_src_id(cntl->pll_id);
 826	/* 00 - coherent mode */
 827	params.asConfig.ucCoherentMode = cntl->coherent;
 828	params.asConfig.ucHPDSel =
 829		cmd->hpd_sel_to_atom(cntl->hpd_sel);
 830	params.ucDigEncoderSel =
 831		cmd->dig_encoder_sel_to_atom(cntl->engine_id);
 832	params.ucDPLaneSet = (uint8_t) cntl->lane_settings;
 833	params.usSymClock = cpu_to_le16((uint16_t) (cntl->pixel_clock / 10));
 834	/*
 835	 * In SI/TN case, caller have to set usPixelClock as following:
 836	 * DP mode: usPixelClock = DP_LINK_CLOCK/10
 837	 * (DP_LINK_CLOCK = 1.62GHz, 2.7GHz, 5.4GHz)
 838	 * DVI single link mode: usPixelClock = pixel clock
 839	 * DVI dual link mode: usPixelClock = pixel clock
 840	 * HDMI mode: usPixelClock = pixel clock * deep_color_ratio
 841	 * (=1: 8bpp, =1.25: 10bpp, =1.5:12bpp, =2: 16bpp)
 842	 * LVDS mode: usPixelClock = pixel clock
 843	 */
 844	if  (cntl->signal == SIGNAL_TYPE_HDMI_TYPE_A) {
 845		switch (cntl->color_depth) {
 846		case COLOR_DEPTH_101010:
 847			params.usSymClock =
 848				cpu_to_le16((le16_to_cpu(params.usSymClock) * 30) / 24);
 849			break;
 850		case COLOR_DEPTH_121212:
 851			params.usSymClock =
 852				cpu_to_le16((le16_to_cpu(params.usSymClock) * 36) / 24);
 853			break;
 854		case COLOR_DEPTH_161616:
 855			params.usSymClock =
 856				cpu_to_le16((le16_to_cpu(params.usSymClock) * 48) / 24);
 857			break;
 858		default:
 859			break;
 860		}
 861	}
 862
 863	if (EXEC_BIOS_CMD_TABLE(UNIPHYTransmitterControl, params))
 864		result = BP_RESULT_OK;
 865
 866	return result;
 867}
 868
 869static enum bp_result transmitter_control_v1_6(
 870	struct bios_parser *bp,
 871	struct bp_transmitter_control *cntl)
 872{
 873	enum bp_result result = BP_RESULT_FAILURE;
 874	const struct command_table_helper *cmd = bp->cmd_helper;
 875	DIG_TRANSMITTER_CONTROL_PARAMETERS_V1_6 params;
 876
 877	memset(&params, 0, sizeof(params));
 878	params.ucPhyId = cmd->phy_id_to_atom(cntl->transmitter);
 879	params.ucAction = (uint8_t)cntl->action;
 880
 881	if (cntl->action == TRANSMITTER_CONTROL_SET_VOLTAGE_AND_PREEMPASIS)
 882		params.ucDPLaneSet = (uint8_t)cntl->lane_settings;
 883	else
 884		params.ucDigMode = cmd->signal_type_to_atom_dig_mode(cntl->signal);
 885
 886	params.ucLaneNum = (uint8_t)cntl->lanes_number;
 887	params.ucHPDSel = cmd->hpd_sel_to_atom(cntl->hpd_sel);
 888	params.ucDigEncoderSel = cmd->dig_encoder_sel_to_atom(cntl->engine_id);
 889	params.ucConnObjId = (uint8_t)cntl->connector_obj_id.id;
 890	params.ulSymClock = cntl->pixel_clock/10;
 891
 892	/*
 893	 * In SI/TN case, caller have to set usPixelClock as following:
 894	 * DP mode: usPixelClock = DP_LINK_CLOCK/10
 895	 * (DP_LINK_CLOCK = 1.62GHz, 2.7GHz, 5.4GHz)
 896	 * DVI single link mode: usPixelClock = pixel clock
 897	 * DVI dual link mode: usPixelClock = pixel clock
 898	 * HDMI mode: usPixelClock = pixel clock * deep_color_ratio
 899	 * (=1: 8bpp, =1.25: 10bpp, =1.5:12bpp, =2: 16bpp)
 900	 * LVDS mode: usPixelClock = pixel clock
 901	 */
 902	switch (cntl->signal) {
 903	case SIGNAL_TYPE_HDMI_TYPE_A:
 904		switch (cntl->color_depth) {
 905		case COLOR_DEPTH_101010:
 906			params.ulSymClock =
 907				cpu_to_le16((le16_to_cpu(params.ulSymClock) * 30) / 24);
 908			break;
 909		case COLOR_DEPTH_121212:
 910			params.ulSymClock =
 911				cpu_to_le16((le16_to_cpu(params.ulSymClock) * 36) / 24);
 912			break;
 913		case COLOR_DEPTH_161616:
 914			params.ulSymClock =
 915				cpu_to_le16((le16_to_cpu(params.ulSymClock) * 48) / 24);
 916			break;
 917		default:
 918			break;
 919		}
 920		break;
 921		default:
 922			break;
 923	}
 924
 925	if (EXEC_BIOS_CMD_TABLE(UNIPHYTransmitterControl, params))
 926		result = BP_RESULT_OK;
 927	return result;
 928}
 929
 930/*******************************************************************************
 931 ********************************************************************************
 932 **
 933 **                  SET PIXEL CLOCK
 934 **
 935 ********************************************************************************
 936 *******************************************************************************/
 937
 938static enum bp_result set_pixel_clock_v3(
 939	struct bios_parser *bp,
 940	struct bp_pixel_clock_parameters *bp_params);
 941static enum bp_result set_pixel_clock_v5(
 942	struct bios_parser *bp,
 943	struct bp_pixel_clock_parameters *bp_params);
 944static enum bp_result set_pixel_clock_v6(
 945	struct bios_parser *bp,
 946	struct bp_pixel_clock_parameters *bp_params);
 947static enum bp_result set_pixel_clock_v7(
 948	struct bios_parser *bp,
 949	struct bp_pixel_clock_parameters *bp_params);
 950
 951static void init_set_pixel_clock(struct bios_parser *bp)
 952{
 953	switch (BIOS_CMD_TABLE_PARA_REVISION(SetPixelClock)) {
 954	case 3:
 955		bp->cmd_tbl.set_pixel_clock = set_pixel_clock_v3;
 956		break;
 957	case 5:
 958		bp->cmd_tbl.set_pixel_clock = set_pixel_clock_v5;
 959		break;
 960	case 6:
 961		bp->cmd_tbl.set_pixel_clock = set_pixel_clock_v6;
 962		break;
 963	case 7:
 964		bp->cmd_tbl.set_pixel_clock = set_pixel_clock_v7;
 965		break;
 966	default:
 967		dm_output_to_console("Don't have set_pixel_clock for v%d\n",
 968			 BIOS_CMD_TABLE_PARA_REVISION(SetPixelClock));
 969		bp->cmd_tbl.set_pixel_clock = NULL;
 970		break;
 971	}
 972}
 973
 974static enum bp_result set_pixel_clock_v3(
 975	struct bios_parser *bp,
 976	struct bp_pixel_clock_parameters *bp_params)
 977{
 978	enum bp_result result = BP_RESULT_FAILURE;
 979	PIXEL_CLOCK_PARAMETERS_V3 *params;
 980	SET_PIXEL_CLOCK_PS_ALLOCATION allocation;
 981
 982	memset(&allocation, 0, sizeof(allocation));
 983
 984	if (CLOCK_SOURCE_ID_PLL1 == bp_params->pll_id)
 985		allocation.sPCLKInput.ucPpll = ATOM_PPLL1;
 986	else if (CLOCK_SOURCE_ID_PLL2 == bp_params->pll_id)
 987		allocation.sPCLKInput.ucPpll = ATOM_PPLL2;
 988	else
 989		return BP_RESULT_BADINPUT;
 990
 991	allocation.sPCLKInput.usRefDiv =
 992			cpu_to_le16((uint16_t)bp_params->reference_divider);
 993	allocation.sPCLKInput.usFbDiv =
 994			cpu_to_le16((uint16_t)bp_params->feedback_divider);
 995	allocation.sPCLKInput.ucFracFbDiv =
 996			(uint8_t)bp_params->fractional_feedback_divider;
 997	allocation.sPCLKInput.ucPostDiv =
 998			(uint8_t)bp_params->pixel_clock_post_divider;
 999
1000	/* We need to convert from 100Hz units into 10KHz units */
1001	allocation.sPCLKInput.usPixelClock =
1002			cpu_to_le16((uint16_t)(bp_params->target_pixel_clock_100hz / 100));
1003
1004	params = (PIXEL_CLOCK_PARAMETERS_V3 *)&allocation.sPCLKInput;
1005	params->ucTransmitterId =
1006			bp->cmd_helper->encoder_id_to_atom(
1007					dal_graphics_object_id_get_encoder_id(
1008							bp_params->encoder_object_id));
1009	params->ucEncoderMode =
1010			(uint8_t)(bp->cmd_helper->encoder_mode_bp_to_atom(
1011					bp_params->signal_type, false));
1012
1013	if (bp_params->flags.FORCE_PROGRAMMING_OF_PLL)
1014		params->ucMiscInfo |= PIXEL_CLOCK_MISC_FORCE_PROG_PPLL;
1015
1016	if (bp_params->flags.USE_E_CLOCK_AS_SOURCE_FOR_D_CLOCK)
1017		params->ucMiscInfo |= PIXEL_CLOCK_MISC_USE_ENGINE_FOR_DISPCLK;
1018
1019	if (CONTROLLER_ID_D1 != bp_params->controller_id)
1020		params->ucMiscInfo |= PIXEL_CLOCK_MISC_CRTC_SEL_CRTC2;
1021
1022	if (EXEC_BIOS_CMD_TABLE(SetPixelClock, allocation))
1023		result = BP_RESULT_OK;
1024
1025	return result;
1026}
1027
1028#ifndef SET_PIXEL_CLOCK_PS_ALLOCATION_V5
1029/* video bios did not define this: */
1030typedef struct _SET_PIXEL_CLOCK_PS_ALLOCATION_V5 {
1031	PIXEL_CLOCK_PARAMETERS_V5 sPCLKInput;
1032	/* Caller doesn't need to init this portion */
1033	ENABLE_SPREAD_SPECTRUM_ON_PPLL sReserved;
1034} SET_PIXEL_CLOCK_PS_ALLOCATION_V5;
1035#endif
1036
1037#ifndef SET_PIXEL_CLOCK_PS_ALLOCATION_V6
1038/* video bios did not define this: */
1039typedef struct _SET_PIXEL_CLOCK_PS_ALLOCATION_V6 {
1040	PIXEL_CLOCK_PARAMETERS_V6 sPCLKInput;
1041	/* Caller doesn't need to init this portion */
1042	ENABLE_SPREAD_SPECTRUM_ON_PPLL sReserved;
1043} SET_PIXEL_CLOCK_PS_ALLOCATION_V6;
1044#endif
1045
1046static enum bp_result set_pixel_clock_v5(
1047	struct bios_parser *bp,
1048	struct bp_pixel_clock_parameters *bp_params)
1049{
1050	enum bp_result result = BP_RESULT_FAILURE;
1051	SET_PIXEL_CLOCK_PS_ALLOCATION_V5 clk;
1052	uint8_t controller_id;
1053	uint32_t pll_id;
1054
1055	memset(&clk, 0, sizeof(clk));
1056
1057	if (bp->cmd_helper->clock_source_id_to_atom(bp_params->pll_id, &pll_id)
1058			&& bp->cmd_helper->controller_id_to_atom(
1059					bp_params->controller_id, &controller_id)) {
1060		clk.sPCLKInput.ucCRTC = controller_id;
1061		clk.sPCLKInput.ucPpll = (uint8_t)pll_id;
1062		clk.sPCLKInput.ucRefDiv =
1063				(uint8_t)(bp_params->reference_divider);
1064		clk.sPCLKInput.usFbDiv =
1065				cpu_to_le16((uint16_t)(bp_params->feedback_divider));
1066		clk.sPCLKInput.ulFbDivDecFrac =
1067				cpu_to_le32(bp_params->fractional_feedback_divider);
1068		clk.sPCLKInput.ucPostDiv =
1069				(uint8_t)(bp_params->pixel_clock_post_divider);
1070		clk.sPCLKInput.ucTransmitterID =
1071				bp->cmd_helper->encoder_id_to_atom(
1072						dal_graphics_object_id_get_encoder_id(
1073								bp_params->encoder_object_id));
1074		clk.sPCLKInput.ucEncoderMode =
1075				(uint8_t)bp->cmd_helper->encoder_mode_bp_to_atom(
1076						bp_params->signal_type, false);
1077
1078		/* We need to convert from 100Hz units into 10KHz units */
1079		clk.sPCLKInput.usPixelClock =
1080				cpu_to_le16((uint16_t)(bp_params->target_pixel_clock_100hz / 100));
1081
1082		if (bp_params->flags.FORCE_PROGRAMMING_OF_PLL)
1083			clk.sPCLKInput.ucMiscInfo |=
1084					PIXEL_CLOCK_MISC_FORCE_PROG_PPLL;
1085
1086		if (bp_params->flags.SET_EXTERNAL_REF_DIV_SRC)
1087			clk.sPCLKInput.ucMiscInfo |=
1088					PIXEL_CLOCK_MISC_REF_DIV_SRC;
1089
1090		/* clkV5.ucMiscInfo bit[3:2]= HDMI panel bit depth: =0: 24bpp
1091		 * =1:30bpp, =2:32bpp
1092		 * driver choose program it itself, i.e. here we program it
1093		 * to 888 by default.
1094		 */
1095		if (bp_params->signal_type == SIGNAL_TYPE_HDMI_TYPE_A)
1096			switch (bp_params->color_depth) {
1097			case TRANSMITTER_COLOR_DEPTH_30:
1098				/* yes this is correct, the atom define is wrong */
1099				clk.sPCLKInput.ucMiscInfo |= PIXEL_CLOCK_V5_MISC_HDMI_32BPP;
1100				break;
1101			case TRANSMITTER_COLOR_DEPTH_36:
1102				/* yes this is correct, the atom define is wrong */
1103				clk.sPCLKInput.ucMiscInfo |= PIXEL_CLOCK_V5_MISC_HDMI_30BPP;
1104				break;
1105			default:
1106				break;
1107			}
1108
1109		if (EXEC_BIOS_CMD_TABLE(SetPixelClock, clk))
1110			result = BP_RESULT_OK;
1111	}
1112
1113	return result;
1114}
1115
1116static enum bp_result set_pixel_clock_v6(
1117	struct bios_parser *bp,
1118	struct bp_pixel_clock_parameters *bp_params)
1119{
1120	enum bp_result result = BP_RESULT_FAILURE;
1121	SET_PIXEL_CLOCK_PS_ALLOCATION_V6 clk;
1122	uint8_t controller_id;
1123	uint32_t pll_id;
1124
1125	memset(&clk, 0, sizeof(clk));
1126
1127	if (bp->cmd_helper->clock_source_id_to_atom(bp_params->pll_id, &pll_id)
1128			&& bp->cmd_helper->controller_id_to_atom(
1129					bp_params->controller_id, &controller_id)) {
1130		/* Note: VBIOS still wants to use ucCRTC name which is now
1131		 * 1 byte in ULONG
1132		 *typedef struct _CRTC_PIXEL_CLOCK_FREQ
1133		 *{
1134		 * target the pixel clock to drive the CRTC timing.
1135		 * ULONG ulPixelClock:24;
1136		 * 0 means disable PPLL/DCPLL. Expanded to 24 bits comparing to
1137		 * previous version.
1138		 * ATOM_CRTC1~6, indicate the CRTC controller to
1139		 * ULONG ucCRTC:8;
1140		 * drive the pixel clock. not used for DCPLL case.
1141		 *}CRTC_PIXEL_CLOCK_FREQ;
1142		 *union
1143		 *{
1144		 * pixel clock and CRTC id frequency
1145		 * CRTC_PIXEL_CLOCK_FREQ ulCrtcPclkFreq;
1146		 * ULONG ulDispEngClkFreq; dispclk frequency
1147		 *};
1148		 */
1149		clk.sPCLKInput.ulCrtcPclkFreq.ucCRTC = controller_id;
1150		clk.sPCLKInput.ucPpll = (uint8_t) pll_id;
1151		clk.sPCLKInput.ucRefDiv =
1152				(uint8_t) bp_params->reference_divider;
1153		clk.sPCLKInput.usFbDiv =
1154				cpu_to_le16((uint16_t) bp_params->feedback_divider);
1155		clk.sPCLKInput.ulFbDivDecFrac =
1156				cpu_to_le32(bp_params->fractional_feedback_divider);
1157		clk.sPCLKInput.ucPostDiv =
1158				(uint8_t) bp_params->pixel_clock_post_divider;
1159		clk.sPCLKInput.ucTransmitterID =
1160				bp->cmd_helper->encoder_id_to_atom(
1161						dal_graphics_object_id_get_encoder_id(
1162								bp_params->encoder_object_id));
1163		clk.sPCLKInput.ucEncoderMode =
1164				(uint8_t) bp->cmd_helper->encoder_mode_bp_to_atom(
1165						bp_params->signal_type, false);
1166
1167		/* We need to convert from 100 Hz units into 10KHz units */
1168		clk.sPCLKInput.ulCrtcPclkFreq.ulPixelClock =
1169				cpu_to_le32(bp_params->target_pixel_clock_100hz / 100);
1170
1171		if (bp_params->flags.FORCE_PROGRAMMING_OF_PLL) {
1172			clk.sPCLKInput.ucMiscInfo |=
1173					PIXEL_CLOCK_V6_MISC_FORCE_PROG_PPLL;
1174		}
1175
1176		if (bp_params->flags.SET_EXTERNAL_REF_DIV_SRC) {
1177			clk.sPCLKInput.ucMiscInfo |=
1178					PIXEL_CLOCK_V6_MISC_REF_DIV_SRC;
1179		}
1180
1181		/* clkV6.ucMiscInfo bit[3:2]= HDMI panel bit depth: =0:
1182		 * 24bpp =1:30bpp, =2:32bpp
1183		 * driver choose program it itself, i.e. here we pass required
1184		 * target rate that includes deep color.
1185		 */
1186		if (bp_params->signal_type == SIGNAL_TYPE_HDMI_TYPE_A)
1187			switch (bp_params->color_depth) {
1188			case TRANSMITTER_COLOR_DEPTH_30:
1189				clk.sPCLKInput.ucMiscInfo |= PIXEL_CLOCK_V6_MISC_HDMI_30BPP_V6;
1190				break;
1191			case TRANSMITTER_COLOR_DEPTH_36:
1192				clk.sPCLKInput.ucMiscInfo |= PIXEL_CLOCK_V6_MISC_HDMI_36BPP_V6;
1193				break;
1194			case TRANSMITTER_COLOR_DEPTH_48:
1195				clk.sPCLKInput.ucMiscInfo |= PIXEL_CLOCK_V6_MISC_HDMI_48BPP;
1196				break;
1197			default:
1198				break;
1199			}
1200
1201		if (EXEC_BIOS_CMD_TABLE(SetPixelClock, clk))
1202			result = BP_RESULT_OK;
1203	}
1204
1205	return result;
1206}
1207
1208static enum bp_result set_pixel_clock_v7(
1209	struct bios_parser *bp,
1210	struct bp_pixel_clock_parameters *bp_params)
1211{
1212	enum bp_result result = BP_RESULT_FAILURE;
1213	PIXEL_CLOCK_PARAMETERS_V7 clk;
1214	uint8_t controller_id;
1215	uint32_t pll_id;
1216
1217	memset(&clk, 0, sizeof(clk));
1218
1219	if (bp->cmd_helper->clock_source_id_to_atom(bp_params->pll_id, &pll_id)
1220			&& bp->cmd_helper->controller_id_to_atom(bp_params->controller_id, &controller_id)) {
1221		/* Note: VBIOS still wants to use ucCRTC name which is now
1222		 * 1 byte in ULONG
1223		 *typedef struct _CRTC_PIXEL_CLOCK_FREQ
1224		 *{
1225		 * target the pixel clock to drive the CRTC timing.
1226		 * ULONG ulPixelClock:24;
1227		 * 0 means disable PPLL/DCPLL. Expanded to 24 bits comparing to
1228		 * previous version.
1229		 * ATOM_CRTC1~6, indicate the CRTC controller to
1230		 * ULONG ucCRTC:8;
1231		 * drive the pixel clock. not used for DCPLL case.
1232		 *}CRTC_PIXEL_CLOCK_FREQ;
1233		 *union
1234		 *{
1235		 * pixel clock and CRTC id frequency
1236		 * CRTC_PIXEL_CLOCK_FREQ ulCrtcPclkFreq;
1237		 * ULONG ulDispEngClkFreq; dispclk frequency
1238		 *};
1239		 */
1240		clk.ucCRTC = controller_id;
1241		clk.ucPpll = (uint8_t) pll_id;
1242		clk.ucTransmitterID = bp->cmd_helper->encoder_id_to_atom(dal_graphics_object_id_get_encoder_id(bp_params->encoder_object_id));
1243		clk.ucEncoderMode = (uint8_t) bp->cmd_helper->encoder_mode_bp_to_atom(bp_params->signal_type, false);
1244
1245		clk.ulPixelClock = cpu_to_le32(bp_params->target_pixel_clock_100hz);
 
1246
1247		clk.ucDeepColorRatio = (uint8_t) bp->cmd_helper->transmitter_color_depth_to_atom(bp_params->color_depth);
1248
1249		if (bp_params->flags.FORCE_PROGRAMMING_OF_PLL)
1250			clk.ucMiscInfo |= PIXEL_CLOCK_V7_MISC_FORCE_PROG_PPLL;
1251
1252		if (bp_params->flags.SET_EXTERNAL_REF_DIV_SRC)
1253			clk.ucMiscInfo |= PIXEL_CLOCK_V7_MISC_REF_DIV_SRC;
1254
1255		if (bp_params->flags.PROGRAM_PHY_PLL_ONLY)
1256			clk.ucMiscInfo |= PIXEL_CLOCK_V7_MISC_PROG_PHYPLL;
1257
1258		if (bp_params->flags.SUPPORT_YUV_420)
1259			clk.ucMiscInfo |= PIXEL_CLOCK_V7_MISC_YUV420_MODE;
1260
1261		if (bp_params->flags.SET_XTALIN_REF_SRC)
1262			clk.ucMiscInfo |= PIXEL_CLOCK_V7_MISC_REF_DIV_SRC_XTALIN;
1263
1264		if (bp_params->flags.SET_GENLOCK_REF_DIV_SRC)
1265			clk.ucMiscInfo |= PIXEL_CLOCK_V7_MISC_REF_DIV_SRC_GENLK;
1266
1267		if (bp_params->signal_type == SIGNAL_TYPE_DVI_DUAL_LINK)
1268			clk.ucMiscInfo |= PIXEL_CLOCK_V7_MISC_DVI_DUALLINK_EN;
1269
1270		if (EXEC_BIOS_CMD_TABLE(SetPixelClock, clk))
1271			result = BP_RESULT_OK;
1272	}
1273	return result;
1274}
1275
1276/*******************************************************************************
1277 ********************************************************************************
1278 **
1279 **                  ENABLE PIXEL CLOCK SS
1280 **
1281 ********************************************************************************
1282 *******************************************************************************/
1283static enum bp_result enable_spread_spectrum_on_ppll_v1(
1284	struct bios_parser *bp,
1285	struct bp_spread_spectrum_parameters *bp_params,
1286	bool enable);
1287static enum bp_result enable_spread_spectrum_on_ppll_v2(
1288	struct bios_parser *bp,
1289	struct bp_spread_spectrum_parameters *bp_params,
1290	bool enable);
1291static enum bp_result enable_spread_spectrum_on_ppll_v3(
1292	struct bios_parser *bp,
1293	struct bp_spread_spectrum_parameters *bp_params,
1294	bool enable);
1295
1296static void init_enable_spread_spectrum_on_ppll(struct bios_parser *bp)
1297{
1298	switch (BIOS_CMD_TABLE_PARA_REVISION(EnableSpreadSpectrumOnPPLL)) {
1299	case 1:
1300		bp->cmd_tbl.enable_spread_spectrum_on_ppll =
1301				enable_spread_spectrum_on_ppll_v1;
1302		break;
1303	case 2:
1304		bp->cmd_tbl.enable_spread_spectrum_on_ppll =
1305				enable_spread_spectrum_on_ppll_v2;
1306		break;
1307	case 3:
1308		bp->cmd_tbl.enable_spread_spectrum_on_ppll =
1309				enable_spread_spectrum_on_ppll_v3;
1310		break;
1311	default:
1312		dm_output_to_console("Don't have enable_spread_spectrum_on_ppll for v%d\n",
1313			 BIOS_CMD_TABLE_PARA_REVISION(EnableSpreadSpectrumOnPPLL));
1314		bp->cmd_tbl.enable_spread_spectrum_on_ppll = NULL;
1315		break;
1316	}
1317}
1318
1319static enum bp_result enable_spread_spectrum_on_ppll_v1(
1320	struct bios_parser *bp,
1321	struct bp_spread_spectrum_parameters *bp_params,
1322	bool enable)
1323{
1324	enum bp_result result = BP_RESULT_FAILURE;
1325	ENABLE_SPREAD_SPECTRUM_ON_PPLL params;
1326
1327	memset(&params, 0, sizeof(params));
1328
1329	if ((enable == true) && (bp_params->percentage > 0))
1330		params.ucEnable = ATOM_ENABLE;
1331	else
1332		params.ucEnable = ATOM_DISABLE;
1333
1334	params.usSpreadSpectrumPercentage =
1335			cpu_to_le16((uint16_t)bp_params->percentage);
1336	params.ucSpreadSpectrumStep =
1337			(uint8_t)bp_params->ver1.step;
1338	params.ucSpreadSpectrumDelay =
1339			(uint8_t)bp_params->ver1.delay;
1340	/* convert back to unit of 10KHz */
1341	params.ucSpreadSpectrumRange =
1342			(uint8_t)(bp_params->ver1.range / 10000);
1343
1344	if (bp_params->flags.EXTERNAL_SS)
1345		params.ucSpreadSpectrumType |= ATOM_EXTERNAL_SS_MASK;
1346
1347	if (bp_params->flags.CENTER_SPREAD)
1348		params.ucSpreadSpectrumType |= ATOM_SS_CENTRE_SPREAD_MODE;
1349
1350	if (bp_params->pll_id == CLOCK_SOURCE_ID_PLL1)
1351		params.ucPpll = ATOM_PPLL1;
1352	else if (bp_params->pll_id == CLOCK_SOURCE_ID_PLL2)
1353		params.ucPpll = ATOM_PPLL2;
1354	else
1355		BREAK_TO_DEBUGGER(); /* Unexpected PLL value!! */
1356
1357	if (EXEC_BIOS_CMD_TABLE(EnableSpreadSpectrumOnPPLL, params))
1358		result = BP_RESULT_OK;
1359
1360	return result;
1361}
1362
1363static enum bp_result enable_spread_spectrum_on_ppll_v2(
1364	struct bios_parser *bp,
1365	struct bp_spread_spectrum_parameters *bp_params,
1366	bool enable)
1367{
1368	enum bp_result result = BP_RESULT_FAILURE;
1369	ENABLE_SPREAD_SPECTRUM_ON_PPLL_V2 params;
1370
1371	memset(&params, 0, sizeof(params));
1372
1373	if (bp_params->pll_id == CLOCK_SOURCE_ID_PLL1)
1374		params.ucSpreadSpectrumType = ATOM_PPLL_SS_TYPE_V2_P1PLL;
1375	else if (bp_params->pll_id == CLOCK_SOURCE_ID_PLL2)
1376		params.ucSpreadSpectrumType = ATOM_PPLL_SS_TYPE_V2_P2PLL;
1377	else
1378		BREAK_TO_DEBUGGER(); /* Unexpected PLL value!! */
1379
1380	if ((enable == true) && (bp_params->percentage > 0)) {
1381		params.ucEnable = ATOM_ENABLE;
1382
1383		params.usSpreadSpectrumPercentage =
1384				cpu_to_le16((uint16_t)(bp_params->percentage));
1385		params.usSpreadSpectrumStep =
1386				cpu_to_le16((uint16_t)(bp_params->ds.ds_frac_size));
1387
1388		if (bp_params->flags.EXTERNAL_SS)
1389			params.ucSpreadSpectrumType |=
1390					ATOM_PPLL_SS_TYPE_V2_EXT_SPREAD;
1391
1392		if (bp_params->flags.CENTER_SPREAD)
1393			params.ucSpreadSpectrumType |=
1394					ATOM_PPLL_SS_TYPE_V2_CENTRE_SPREAD;
1395
1396		/* Both amounts need to be left shifted first before bit
1397		 * comparison. Otherwise, the result will always be zero here
1398		 */
1399		params.usSpreadSpectrumAmount = cpu_to_le16((uint16_t)(
1400				((bp_params->ds.feedback_amount <<
1401						ATOM_PPLL_SS_AMOUNT_V2_FBDIV_SHIFT) &
1402						ATOM_PPLL_SS_AMOUNT_V2_FBDIV_MASK) |
1403						((bp_params->ds.nfrac_amount <<
1404								ATOM_PPLL_SS_AMOUNT_V2_NFRAC_SHIFT) &
1405								ATOM_PPLL_SS_AMOUNT_V2_NFRAC_MASK)));
1406	} else
1407		params.ucEnable = ATOM_DISABLE;
1408
1409	if (EXEC_BIOS_CMD_TABLE(EnableSpreadSpectrumOnPPLL, params))
1410		result = BP_RESULT_OK;
1411
1412	return result;
1413}
1414
1415static enum bp_result enable_spread_spectrum_on_ppll_v3(
1416	struct bios_parser *bp,
1417	struct bp_spread_spectrum_parameters *bp_params,
1418	bool enable)
1419{
1420	enum bp_result result = BP_RESULT_FAILURE;
1421	ENABLE_SPREAD_SPECTRUM_ON_PPLL_V3 params;
1422
1423	memset(&params, 0, sizeof(params));
1424
1425	switch (bp_params->pll_id) {
1426	case CLOCK_SOURCE_ID_PLL0:
1427		/* ATOM_PPLL_SS_TYPE_V3_P0PLL; this is pixel clock only,
1428		 * not for SI display clock.
1429		 */
1430		params.ucSpreadSpectrumType = ATOM_PPLL_SS_TYPE_V3_DCPLL;
1431		break;
1432	case CLOCK_SOURCE_ID_PLL1:
1433		params.ucSpreadSpectrumType = ATOM_PPLL_SS_TYPE_V3_P1PLL;
1434		break;
1435
1436	case CLOCK_SOURCE_ID_PLL2:
1437		params.ucSpreadSpectrumType = ATOM_PPLL_SS_TYPE_V3_P2PLL;
1438		break;
1439
1440	case CLOCK_SOURCE_ID_DCPLL:
1441		params.ucSpreadSpectrumType = ATOM_PPLL_SS_TYPE_V3_DCPLL;
1442		break;
1443
1444	default:
1445		BREAK_TO_DEBUGGER();
1446		/* Unexpected PLL value!! */
1447		return result;
1448	}
1449
1450	if (enable == true) {
1451		params.ucEnable = ATOM_ENABLE;
1452
1453		params.usSpreadSpectrumAmountFrac =
1454				cpu_to_le16((uint16_t)(bp_params->ds_frac_amount));
1455		params.usSpreadSpectrumStep =
1456				cpu_to_le16((uint16_t)(bp_params->ds.ds_frac_size));
1457
1458		if (bp_params->flags.EXTERNAL_SS)
1459			params.ucSpreadSpectrumType |=
1460					ATOM_PPLL_SS_TYPE_V3_EXT_SPREAD;
1461		if (bp_params->flags.CENTER_SPREAD)
1462			params.ucSpreadSpectrumType |=
1463					ATOM_PPLL_SS_TYPE_V3_CENTRE_SPREAD;
1464
1465		/* Both amounts need to be left shifted first before bit
1466		 * comparison. Otherwise, the result will always be zero here
1467		 */
1468		params.usSpreadSpectrumAmount = cpu_to_le16((uint16_t)(
1469				((bp_params->ds.feedback_amount <<
1470						ATOM_PPLL_SS_AMOUNT_V3_FBDIV_SHIFT) &
1471						ATOM_PPLL_SS_AMOUNT_V3_FBDIV_MASK) |
1472						((bp_params->ds.nfrac_amount <<
1473								ATOM_PPLL_SS_AMOUNT_V3_NFRAC_SHIFT) &
1474								ATOM_PPLL_SS_AMOUNT_V3_NFRAC_MASK)));
1475	} else
1476		params.ucEnable = ATOM_DISABLE;
1477
1478	if (EXEC_BIOS_CMD_TABLE(EnableSpreadSpectrumOnPPLL, params))
1479		result = BP_RESULT_OK;
1480
1481	return result;
1482}
1483
1484/*******************************************************************************
1485 ********************************************************************************
1486 **
1487 **                  ADJUST DISPLAY PLL
1488 **
1489 ********************************************************************************
1490 *******************************************************************************/
1491
1492static enum bp_result adjust_display_pll_v2(
1493	struct bios_parser *bp,
1494	struct bp_adjust_pixel_clock_parameters *bp_params);
1495static enum bp_result adjust_display_pll_v3(
1496	struct bios_parser *bp,
1497	struct bp_adjust_pixel_clock_parameters *bp_params);
1498
1499static void init_adjust_display_pll(struct bios_parser *bp)
1500{
1501	switch (BIOS_CMD_TABLE_PARA_REVISION(AdjustDisplayPll)) {
1502	case 2:
1503		bp->cmd_tbl.adjust_display_pll = adjust_display_pll_v2;
1504		break;
1505	case 3:
1506		bp->cmd_tbl.adjust_display_pll = adjust_display_pll_v3;
1507		break;
1508	default:
1509		dm_output_to_console("Don't have adjust_display_pll for v%d\n",
1510			 BIOS_CMD_TABLE_PARA_REVISION(AdjustDisplayPll));
1511		bp->cmd_tbl.adjust_display_pll = NULL;
1512		break;
1513	}
1514}
1515
1516static enum bp_result adjust_display_pll_v2(
1517	struct bios_parser *bp,
1518	struct bp_adjust_pixel_clock_parameters *bp_params)
1519{
1520	enum bp_result result = BP_RESULT_FAILURE;
1521	ADJUST_DISPLAY_PLL_PS_ALLOCATION params = { 0 };
1522
1523	/* We need to convert from KHz units into 10KHz units and then convert
1524	 * output pixel clock back 10KHz-->KHz */
1525	uint32_t pixel_clock_10KHz_in = bp_params->pixel_clock / 10;
1526
1527	params.usPixelClock = cpu_to_le16((uint16_t)(pixel_clock_10KHz_in));
1528	params.ucTransmitterID =
1529			bp->cmd_helper->encoder_id_to_atom(
1530					dal_graphics_object_id_get_encoder_id(
1531							bp_params->encoder_object_id));
1532	params.ucEncodeMode =
1533			(uint8_t)bp->cmd_helper->encoder_mode_bp_to_atom(
1534					bp_params->signal_type, false);
1535
1536	if (EXEC_BIOS_CMD_TABLE(AdjustDisplayPll, params)) {
1537		/* Convert output pixel clock back 10KHz-->KHz: multiply
1538		 * original pixel clock in KHz by ratio
1539		 * [output pxlClk/input pxlClk] */
1540		uint64_t pixel_clk_10_khz_out =
1541				(uint64_t)le16_to_cpu(params.usPixelClock);
1542		uint64_t pixel_clk = (uint64_t)bp_params->pixel_clock;
1543
1544		if (pixel_clock_10KHz_in != 0) {
1545			bp_params->adjusted_pixel_clock =
1546					div_u64(pixel_clk * pixel_clk_10_khz_out,
1547							pixel_clock_10KHz_in);
1548		} else {
1549			bp_params->adjusted_pixel_clock = 0;
1550			BREAK_TO_DEBUGGER();
1551		}
1552
1553		result = BP_RESULT_OK;
1554	}
1555
1556	return result;
1557}
1558
1559static enum bp_result adjust_display_pll_v3(
1560	struct bios_parser *bp,
1561	struct bp_adjust_pixel_clock_parameters *bp_params)
1562{
1563	enum bp_result result = BP_RESULT_FAILURE;
1564	ADJUST_DISPLAY_PLL_PS_ALLOCATION_V3 params;
1565	uint32_t pixel_clk_10_kHz_in = bp_params->pixel_clock / 10;
1566
1567	memset(&params, 0, sizeof(params));
1568
1569	/* We need to convert from KHz units into 10KHz units and then convert
1570	 * output pixel clock back 10KHz-->KHz */
1571	params.sInput.usPixelClock = cpu_to_le16((uint16_t)pixel_clk_10_kHz_in);
1572	params.sInput.ucTransmitterID =
1573			bp->cmd_helper->encoder_id_to_atom(
1574					dal_graphics_object_id_get_encoder_id(
1575							bp_params->encoder_object_id));
1576	params.sInput.ucEncodeMode =
1577			(uint8_t)bp->cmd_helper->encoder_mode_bp_to_atom(
1578					bp_params->signal_type, false);
1579
1580	if (bp_params->ss_enable == true)
1581		params.sInput.ucDispPllConfig |= DISPPLL_CONFIG_SS_ENABLE;
1582
1583	if (bp_params->signal_type == SIGNAL_TYPE_DVI_DUAL_LINK)
1584		params.sInput.ucDispPllConfig |= DISPPLL_CONFIG_DUAL_LINK;
1585
1586	if (EXEC_BIOS_CMD_TABLE(AdjustDisplayPll, params)) {
1587		/* Convert output pixel clock back 10KHz-->KHz: multiply
1588		 * original pixel clock in KHz by ratio
1589		 * [output pxlClk/input pxlClk] */
1590		uint64_t pixel_clk_10_khz_out =
1591				(uint64_t)le32_to_cpu(params.sOutput.ulDispPllFreq);
1592		uint64_t pixel_clk = (uint64_t)bp_params->pixel_clock;
1593
1594		if (pixel_clk_10_kHz_in != 0) {
1595			bp_params->adjusted_pixel_clock =
1596					div_u64(pixel_clk * pixel_clk_10_khz_out,
1597							pixel_clk_10_kHz_in);
1598		} else {
1599			bp_params->adjusted_pixel_clock = 0;
1600			BREAK_TO_DEBUGGER();
1601		}
1602
1603		bp_params->reference_divider = params.sOutput.ucRefDiv;
1604		bp_params->pixel_clock_post_divider = params.sOutput.ucPostDiv;
1605
1606		result = BP_RESULT_OK;
1607	}
1608
1609	return result;
1610}
1611
1612/*******************************************************************************
1613 ********************************************************************************
1614 **
1615 **                  DAC ENCODER CONTROL
1616 **
1617 ********************************************************************************
1618 *******************************************************************************/
1619
1620static enum bp_result dac1_encoder_control_v1(
1621	struct bios_parser *bp,
1622	bool enable,
1623	uint32_t pixel_clock,
1624	uint8_t dac_standard);
1625static enum bp_result dac2_encoder_control_v1(
1626	struct bios_parser *bp,
1627	bool enable,
1628	uint32_t pixel_clock,
1629	uint8_t dac_standard);
1630
1631static void init_dac_encoder_control(struct bios_parser *bp)
1632{
1633	switch (BIOS_CMD_TABLE_PARA_REVISION(DAC1EncoderControl)) {
1634	case 1:
1635		bp->cmd_tbl.dac1_encoder_control = dac1_encoder_control_v1;
1636		break;
1637	default:
1638		bp->cmd_tbl.dac1_encoder_control = NULL;
1639		break;
1640	}
1641	switch (BIOS_CMD_TABLE_PARA_REVISION(DAC2EncoderControl)) {
1642	case 1:
1643		bp->cmd_tbl.dac2_encoder_control = dac2_encoder_control_v1;
1644		break;
1645	default:
1646		bp->cmd_tbl.dac2_encoder_control = NULL;
1647		break;
1648	}
1649}
1650
1651static void dac_encoder_control_prepare_params(
1652	DAC_ENCODER_CONTROL_PS_ALLOCATION *params,
1653	bool enable,
1654	uint32_t pixel_clock,
1655	uint8_t dac_standard)
1656{
1657	params->ucDacStandard = dac_standard;
1658	if (enable)
1659		params->ucAction = ATOM_ENABLE;
1660	else
1661		params->ucAction = ATOM_DISABLE;
1662
1663	/* We need to convert from KHz units into 10KHz units
1664	 * it looks as if the TvControl do not care about pixel clock
1665	 */
1666	params->usPixelClock = cpu_to_le16((uint16_t)(pixel_clock / 10));
1667}
1668
1669static enum bp_result dac1_encoder_control_v1(
1670	struct bios_parser *bp,
1671	bool enable,
1672	uint32_t pixel_clock,
1673	uint8_t dac_standard)
1674{
1675	enum bp_result result = BP_RESULT_FAILURE;
1676	DAC_ENCODER_CONTROL_PS_ALLOCATION params;
1677
1678	dac_encoder_control_prepare_params(
1679		&params,
1680		enable,
1681		pixel_clock,
1682		dac_standard);
1683
1684	if (EXEC_BIOS_CMD_TABLE(DAC1EncoderControl, params))
1685		result = BP_RESULT_OK;
1686
1687	return result;
1688}
1689
1690static enum bp_result dac2_encoder_control_v1(
1691	struct bios_parser *bp,
1692	bool enable,
1693	uint32_t pixel_clock,
1694	uint8_t dac_standard)
1695{
1696	enum bp_result result = BP_RESULT_FAILURE;
1697	DAC_ENCODER_CONTROL_PS_ALLOCATION params;
1698
1699	dac_encoder_control_prepare_params(
1700		&params,
1701		enable,
1702		pixel_clock,
1703		dac_standard);
1704
1705	if (EXEC_BIOS_CMD_TABLE(DAC2EncoderControl, params))
1706		result = BP_RESULT_OK;
1707
1708	return result;
1709}
1710
1711/*******************************************************************************
1712 ********************************************************************************
1713 **
1714 **                  DAC OUTPUT CONTROL
1715 **
1716 ********************************************************************************
1717 *******************************************************************************/
1718static enum bp_result dac1_output_control_v1(
1719	struct bios_parser *bp,
1720	bool enable);
1721static enum bp_result dac2_output_control_v1(
1722	struct bios_parser *bp,
1723	bool enable);
1724
1725static void init_dac_output_control(struct bios_parser *bp)
1726{
1727	switch (BIOS_CMD_TABLE_PARA_REVISION(DAC1OutputControl)) {
1728	case 1:
1729		bp->cmd_tbl.dac1_output_control = dac1_output_control_v1;
1730		break;
1731	default:
1732		bp->cmd_tbl.dac1_output_control = NULL;
1733		break;
1734	}
1735	switch (BIOS_CMD_TABLE_PARA_REVISION(DAC2OutputControl)) {
1736	case 1:
1737		bp->cmd_tbl.dac2_output_control = dac2_output_control_v1;
1738		break;
1739	default:
1740		bp->cmd_tbl.dac2_output_control = NULL;
1741		break;
1742	}
1743}
1744
1745static enum bp_result dac1_output_control_v1(
1746	struct bios_parser *bp, bool enable)
1747{
1748	enum bp_result result = BP_RESULT_FAILURE;
1749	DISPLAY_DEVICE_OUTPUT_CONTROL_PS_ALLOCATION params;
1750
1751	if (enable)
1752		params.ucAction = ATOM_ENABLE;
1753	else
1754		params.ucAction = ATOM_DISABLE;
1755
1756	if (EXEC_BIOS_CMD_TABLE(DAC1OutputControl, params))
1757		result = BP_RESULT_OK;
1758
1759	return result;
1760}
1761
1762static enum bp_result dac2_output_control_v1(
1763	struct bios_parser *bp, bool enable)
1764{
1765	enum bp_result result = BP_RESULT_FAILURE;
1766	DISPLAY_DEVICE_OUTPUT_CONTROL_PS_ALLOCATION params;
1767
1768	if (enable)
1769		params.ucAction = ATOM_ENABLE;
1770	else
1771		params.ucAction = ATOM_DISABLE;
1772
1773	if (EXEC_BIOS_CMD_TABLE(DAC2OutputControl, params))
1774		result = BP_RESULT_OK;
1775
1776	return result;
1777}
1778
1779/*******************************************************************************
1780 ********************************************************************************
1781 **
1782 **                  SET CRTC TIMING
1783 **
1784 ********************************************************************************
1785 *******************************************************************************/
1786
1787static enum bp_result set_crtc_using_dtd_timing_v3(
1788	struct bios_parser *bp,
1789	struct bp_hw_crtc_timing_parameters *bp_params);
1790static enum bp_result set_crtc_timing_v1(
1791	struct bios_parser *bp,
1792	struct bp_hw_crtc_timing_parameters *bp_params);
1793
1794static void init_set_crtc_timing(struct bios_parser *bp)
1795{
1796	uint32_t dtd_version =
1797			BIOS_CMD_TABLE_PARA_REVISION(SetCRTC_UsingDTDTiming);
1798	if (dtd_version > 2)
1799		switch (dtd_version) {
1800		case 3:
1801			bp->cmd_tbl.set_crtc_timing =
1802					set_crtc_using_dtd_timing_v3;
1803			break;
1804		default:
1805			dm_output_to_console("Don't have set_crtc_timing for dtd v%d\n",
1806				 dtd_version);
1807			bp->cmd_tbl.set_crtc_timing = NULL;
1808			break;
1809		}
1810	else
1811		switch (BIOS_CMD_TABLE_PARA_REVISION(SetCRTC_Timing)) {
1812		case 1:
1813			bp->cmd_tbl.set_crtc_timing = set_crtc_timing_v1;
1814			break;
1815		default:
1816			dm_output_to_console("Don't have set_crtc_timing for v%d\n",
1817				 BIOS_CMD_TABLE_PARA_REVISION(SetCRTC_Timing));
1818			bp->cmd_tbl.set_crtc_timing = NULL;
1819			break;
1820		}
1821}
1822
1823static enum bp_result set_crtc_timing_v1(
1824	struct bios_parser *bp,
1825	struct bp_hw_crtc_timing_parameters *bp_params)
1826{
1827	enum bp_result result = BP_RESULT_FAILURE;
1828	SET_CRTC_TIMING_PARAMETERS_PS_ALLOCATION params = {0};
1829	uint8_t atom_controller_id;
1830
1831	if (bp->cmd_helper->controller_id_to_atom(
1832			bp_params->controller_id, &atom_controller_id))
1833		params.ucCRTC = atom_controller_id;
1834
1835	params.usH_Total = cpu_to_le16((uint16_t)(bp_params->h_total));
1836	params.usH_Disp = cpu_to_le16((uint16_t)(bp_params->h_addressable));
1837	params.usH_SyncStart = cpu_to_le16((uint16_t)(bp_params->h_sync_start));
1838	params.usH_SyncWidth = cpu_to_le16((uint16_t)(bp_params->h_sync_width));
1839	params.usV_Total = cpu_to_le16((uint16_t)(bp_params->v_total));
1840	params.usV_Disp = cpu_to_le16((uint16_t)(bp_params->v_addressable));
1841	params.usV_SyncStart =
1842			cpu_to_le16((uint16_t)(bp_params->v_sync_start));
1843	params.usV_SyncWidth =
1844			cpu_to_le16((uint16_t)(bp_params->v_sync_width));
1845
1846	/* VBIOS does not expect any value except zero into this call, for
1847	 * underscan use another entry ProgramOverscan call but when mode
1848	 * 1776x1000 with the overscan 72x44 .e.i. 1920x1080 @30 DAL2 is ok,
1849	 * but when same ,but 60 Hz there is corruption
1850	 * DAL1 does not allow the mode 1776x1000@60
1851	 */
1852	params.ucOverscanRight = (uint8_t)bp_params->h_overscan_right;
1853	params.ucOverscanLeft = (uint8_t)bp_params->h_overscan_left;
1854	params.ucOverscanBottom = (uint8_t)bp_params->v_overscan_bottom;
1855	params.ucOverscanTop = (uint8_t)bp_params->v_overscan_top;
1856
1857	if (0 == bp_params->flags.HSYNC_POSITIVE_POLARITY)
1858		params.susModeMiscInfo.usAccess =
1859				cpu_to_le16(le16_to_cpu(params.susModeMiscInfo.usAccess) | ATOM_HSYNC_POLARITY);
1860
1861	if (0 == bp_params->flags.VSYNC_POSITIVE_POLARITY)
1862		params.susModeMiscInfo.usAccess =
1863				cpu_to_le16(le16_to_cpu(params.susModeMiscInfo.usAccess) | ATOM_VSYNC_POLARITY);
1864
1865	if (bp_params->flags.INTERLACE) {
1866		params.susModeMiscInfo.usAccess =
1867				cpu_to_le16(le16_to_cpu(params.susModeMiscInfo.usAccess) | ATOM_INTERLACE);
1868
1869		/* original DAL code has this condition to apply tis for
1870		 * non-TV/CV only due to complex MV testing for possible
1871		 * impact
1872		 * if (pACParameters->signal != SignalType_YPbPr &&
1873		 *  pACParameters->signal != SignalType_Composite &&
1874		 *  pACParameters->signal != SignalType_SVideo)
1875		 */
1876		/* HW will deduct 0.5 line from 2nd feild.
1877		 * i.e. for 1080i, it is 2 lines for 1st field, 2.5
1878		 * lines for the 2nd feild. we need input as 5 instead
1879		 * of 4, but it is 4 either from Edid data
1880		 * (spec CEA 861) or CEA timing table.
1881		 */
1882		params.usV_SyncStart =
1883				cpu_to_le16((uint16_t)(bp_params->v_sync_start + 1));
1884	}
1885
1886	if (bp_params->flags.HORZ_COUNT_BY_TWO)
1887		params.susModeMiscInfo.usAccess =
1888				cpu_to_le16(le16_to_cpu(params.susModeMiscInfo.usAccess) | ATOM_DOUBLE_CLOCK_MODE);
1889
1890	if (EXEC_BIOS_CMD_TABLE(SetCRTC_Timing, params))
1891		result = BP_RESULT_OK;
1892
1893	return result;
1894}
1895
1896static enum bp_result set_crtc_using_dtd_timing_v3(
1897	struct bios_parser *bp,
1898	struct bp_hw_crtc_timing_parameters *bp_params)
1899{
1900	enum bp_result result = BP_RESULT_FAILURE;
1901	SET_CRTC_USING_DTD_TIMING_PARAMETERS params = {0};
1902	uint8_t atom_controller_id;
1903
1904	if (bp->cmd_helper->controller_id_to_atom(
1905			bp_params->controller_id, &atom_controller_id))
1906		params.ucCRTC = atom_controller_id;
1907
1908	/* bios usH_Size wants h addressable size */
1909	params.usH_Size = cpu_to_le16((uint16_t)bp_params->h_addressable);
1910	/* bios usH_Blanking_Time wants borders included in blanking */
1911	params.usH_Blanking_Time =
1912			cpu_to_le16((uint16_t)(bp_params->h_total - bp_params->h_addressable));
1913	/* bios usV_Size wants v addressable size */
1914	params.usV_Size = cpu_to_le16((uint16_t)bp_params->v_addressable);
1915	/* bios usV_Blanking_Time wants borders included in blanking */
1916	params.usV_Blanking_Time =
1917			cpu_to_le16((uint16_t)(bp_params->v_total - bp_params->v_addressable));
1918	/* bios usHSyncOffset is the offset from the end of h addressable,
1919	 * our horizontalSyncStart is the offset from the beginning
1920	 * of h addressable */
1921	params.usH_SyncOffset =
1922			cpu_to_le16((uint16_t)(bp_params->h_sync_start - bp_params->h_addressable));
1923	params.usH_SyncWidth = cpu_to_le16((uint16_t)bp_params->h_sync_width);
1924	/* bios usHSyncOffset is the offset from the end of v addressable,
1925	 * our verticalSyncStart is the offset from the beginning of
1926	 * v addressable */
1927	params.usV_SyncOffset =
1928			cpu_to_le16((uint16_t)(bp_params->v_sync_start - bp_params->v_addressable));
1929	params.usV_SyncWidth = cpu_to_le16((uint16_t)bp_params->v_sync_width);
1930
1931	/* we assume that overscan from original timing does not get bigger
1932	 * than 255
1933	 * we will program all the borders in the Set CRTC Overscan call below
1934	 */
1935
1936	if (0 == bp_params->flags.HSYNC_POSITIVE_POLARITY)
1937		params.susModeMiscInfo.usAccess =
1938				cpu_to_le16(le16_to_cpu(params.susModeMiscInfo.usAccess) | ATOM_HSYNC_POLARITY);
1939
1940	if (0 == bp_params->flags.VSYNC_POSITIVE_POLARITY)
1941		params.susModeMiscInfo.usAccess =
1942				cpu_to_le16(le16_to_cpu(params.susModeMiscInfo.usAccess) | ATOM_VSYNC_POLARITY);
1943
1944	if (bp_params->flags.INTERLACE)	{
1945		params.susModeMiscInfo.usAccess =
1946				cpu_to_le16(le16_to_cpu(params.susModeMiscInfo.usAccess) | ATOM_INTERLACE);
1947
1948		/* original DAL code has this condition to apply this
1949		 * for non-TV/CV only
1950		 * due to complex MV testing for possible impact
1951		 * if ( pACParameters->signal != SignalType_YPbPr &&
1952		 *  pACParameters->signal != SignalType_Composite &&
1953		 *  pACParameters->signal != SignalType_SVideo)
1954		 */
1955		{
1956			/* HW will deduct 0.5 line from 2nd feild.
1957			 * i.e. for 1080i, it is 2 lines for 1st field,
1958			 * 2.5 lines for the 2nd feild. we need input as 5
1959			 * instead of 4.
1960			 * but it is 4 either from Edid data (spec CEA 861)
1961			 * or CEA timing table.
1962			 */
1963			le16_add_cpu(&params.usV_SyncOffset, 1);
 
 
1964		}
1965	}
1966
1967	if (bp_params->flags.HORZ_COUNT_BY_TWO)
1968		params.susModeMiscInfo.usAccess =
1969				cpu_to_le16(le16_to_cpu(params.susModeMiscInfo.usAccess) | ATOM_DOUBLE_CLOCK_MODE);
1970
1971	if (EXEC_BIOS_CMD_TABLE(SetCRTC_UsingDTDTiming, params))
1972		result = BP_RESULT_OK;
1973
1974	return result;
1975}
1976
1977/*******************************************************************************
1978 ********************************************************************************
1979 **
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1980 **                  ENABLE CRTC
1981 **
1982 ********************************************************************************
1983 *******************************************************************************/
1984
1985static enum bp_result enable_crtc_v1(
1986	struct bios_parser *bp,
1987	enum controller_id controller_id,
1988	bool enable);
1989
1990static void init_enable_crtc(struct bios_parser *bp)
1991{
1992	switch (BIOS_CMD_TABLE_PARA_REVISION(EnableCRTC)) {
1993	case 1:
1994		bp->cmd_tbl.enable_crtc = enable_crtc_v1;
1995		break;
1996	default:
1997		dm_output_to_console("Don't have enable_crtc for v%d\n",
1998			 BIOS_CMD_TABLE_PARA_REVISION(EnableCRTC));
1999		bp->cmd_tbl.enable_crtc = NULL;
2000		break;
2001	}
2002}
2003
2004static enum bp_result enable_crtc_v1(
2005	struct bios_parser *bp,
2006	enum controller_id controller_id,
2007	bool enable)
2008{
2009	bool result = BP_RESULT_FAILURE;
2010	ENABLE_CRTC_PARAMETERS params = {0};
2011	uint8_t id;
2012
2013	if (bp->cmd_helper->controller_id_to_atom(controller_id, &id))
2014		params.ucCRTC = id;
2015	else
2016		return BP_RESULT_BADINPUT;
2017
2018	if (enable)
2019		params.ucEnable = ATOM_ENABLE;
2020	else
2021		params.ucEnable = ATOM_DISABLE;
2022
2023	if (EXEC_BIOS_CMD_TABLE(EnableCRTC, params))
2024		result = BP_RESULT_OK;
2025
2026	return result;
2027}
2028
2029/*******************************************************************************
2030 ********************************************************************************
2031 **
2032 **                  ENABLE CRTC MEM REQ
2033 **
2034 ********************************************************************************
2035 *******************************************************************************/
2036
2037static enum bp_result enable_crtc_mem_req_v1(
2038	struct bios_parser *bp,
2039	enum controller_id controller_id,
2040	bool enable);
2041
2042static void init_enable_crtc_mem_req(struct bios_parser *bp)
2043{
2044	switch (BIOS_CMD_TABLE_PARA_REVISION(EnableCRTCMemReq)) {
2045	case 1:
2046		bp->cmd_tbl.enable_crtc_mem_req = enable_crtc_mem_req_v1;
2047		break;
2048	default:
2049		bp->cmd_tbl.enable_crtc_mem_req = NULL;
2050		break;
2051	}
2052}
2053
2054static enum bp_result enable_crtc_mem_req_v1(
2055	struct bios_parser *bp,
2056	enum controller_id controller_id,
2057	bool enable)
2058{
2059	bool result = BP_RESULT_BADINPUT;
2060	ENABLE_CRTC_PARAMETERS params = {0};
2061	uint8_t id;
2062
2063	if (bp->cmd_helper->controller_id_to_atom(controller_id, &id)) {
2064		params.ucCRTC = id;
2065
2066		if (enable)
2067			params.ucEnable = ATOM_ENABLE;
2068		else
2069			params.ucEnable = ATOM_DISABLE;
2070
2071		if (EXEC_BIOS_CMD_TABLE(EnableCRTCMemReq, params))
2072			result = BP_RESULT_OK;
2073		else
2074			result = BP_RESULT_FAILURE;
2075	}
2076
2077	return result;
2078}
2079
2080/*******************************************************************************
2081 ********************************************************************************
2082 **
2083 **                  DISPLAY PLL
2084 **
2085 ********************************************************************************
2086 *******************************************************************************/
2087
2088static enum bp_result program_clock_v5(
2089	struct bios_parser *bp,
2090	struct bp_pixel_clock_parameters *bp_params);
2091static enum bp_result program_clock_v6(
2092	struct bios_parser *bp,
2093	struct bp_pixel_clock_parameters *bp_params);
2094
2095static void init_program_clock(struct bios_parser *bp)
2096{
2097	switch (BIOS_CMD_TABLE_PARA_REVISION(SetPixelClock)) {
2098	case 5:
2099		bp->cmd_tbl.program_clock = program_clock_v5;
2100		break;
2101	case 6:
2102		bp->cmd_tbl.program_clock = program_clock_v6;
2103		break;
2104	default:
2105		dm_output_to_console("Don't have program_clock for v%d\n",
2106			 BIOS_CMD_TABLE_PARA_REVISION(SetPixelClock));
2107		bp->cmd_tbl.program_clock = NULL;
2108		break;
2109	}
2110}
2111
2112static enum bp_result program_clock_v5(
2113	struct bios_parser *bp,
2114	struct bp_pixel_clock_parameters *bp_params)
2115{
2116	enum bp_result result = BP_RESULT_FAILURE;
2117
2118	SET_PIXEL_CLOCK_PS_ALLOCATION_V5 params;
2119	uint32_t atom_pll_id;
2120
2121	memset(&params, 0, sizeof(params));
2122	if (!bp->cmd_helper->clock_source_id_to_atom(
2123			bp_params->pll_id, &atom_pll_id)) {
2124		BREAK_TO_DEBUGGER(); /* Invalid Input!! */
2125		return BP_RESULT_BADINPUT;
2126	}
2127
2128	/* We need to convert from KHz units into 10KHz units */
2129	params.sPCLKInput.ucPpll = (uint8_t) atom_pll_id;
2130	params.sPCLKInput.usPixelClock =
2131			cpu_to_le16((uint16_t) (bp_params->target_pixel_clock_100hz / 100));
2132	params.sPCLKInput.ucCRTC = (uint8_t) ATOM_CRTC_INVALID;
2133
2134	if (bp_params->flags.SET_EXTERNAL_REF_DIV_SRC)
2135		params.sPCLKInput.ucMiscInfo |= PIXEL_CLOCK_MISC_REF_DIV_SRC;
2136
2137	if (EXEC_BIOS_CMD_TABLE(SetPixelClock, params))
2138		result = BP_RESULT_OK;
2139
2140	return result;
2141}
2142
2143static enum bp_result program_clock_v6(
2144	struct bios_parser *bp,
2145	struct bp_pixel_clock_parameters *bp_params)
2146{
2147	enum bp_result result = BP_RESULT_FAILURE;
2148
2149	SET_PIXEL_CLOCK_PS_ALLOCATION_V6 params;
2150	uint32_t atom_pll_id;
2151
2152	memset(&params, 0, sizeof(params));
2153
2154	if (!bp->cmd_helper->clock_source_id_to_atom(
2155			bp_params->pll_id, &atom_pll_id)) {
2156		BREAK_TO_DEBUGGER(); /*Invalid Input!!*/
2157		return BP_RESULT_BADINPUT;
2158	}
2159
2160	/* We need to convert from KHz units into 10KHz units */
2161	params.sPCLKInput.ucPpll = (uint8_t)atom_pll_id;
2162	params.sPCLKInput.ulDispEngClkFreq =
2163			cpu_to_le32(bp_params->target_pixel_clock_100hz / 100);
2164
2165	if (bp_params->flags.SET_EXTERNAL_REF_DIV_SRC)
2166		params.sPCLKInput.ucMiscInfo |= PIXEL_CLOCK_MISC_REF_DIV_SRC;
2167
2168	if (bp_params->flags.SET_DISPCLK_DFS_BYPASS)
2169		params.sPCLKInput.ucMiscInfo |= PIXEL_CLOCK_V6_MISC_DPREFCLK_BYPASS;
2170
2171	if (EXEC_BIOS_CMD_TABLE(SetPixelClock, params)) {
2172		/* True display clock is returned by VBIOS if DFS bypass
2173		 * is enabled. */
2174		bp_params->dfs_bypass_display_clock =
2175				(uint32_t)(le32_to_cpu(params.sPCLKInput.ulDispEngClkFreq) * 10);
2176		result = BP_RESULT_OK;
2177	}
2178
2179	return result;
2180}
2181
2182/*******************************************************************************
2183 ********************************************************************************
2184 **
2185 **                  EXTERNAL ENCODER CONTROL
2186 **
2187 ********************************************************************************
2188 *******************************************************************************/
2189
2190static enum bp_result external_encoder_control_v3(
2191	struct bios_parser *bp,
2192	struct bp_external_encoder_control *cntl);
2193
2194static void init_external_encoder_control(
2195	struct bios_parser *bp)
2196{
2197	switch (BIOS_CMD_TABLE_PARA_REVISION(ExternalEncoderControl)) {
2198	case 3:
2199		bp->cmd_tbl.external_encoder_control =
2200				external_encoder_control_v3;
2201		break;
2202	default:
2203		bp->cmd_tbl.external_encoder_control = NULL;
2204		break;
2205	}
2206}
2207
2208static enum bp_result external_encoder_control_v3(
2209	struct bios_parser *bp,
2210	struct bp_external_encoder_control *cntl)
2211{
2212	enum bp_result result = BP_RESULT_FAILURE;
2213
2214	/* we need use _PS_Alloc struct */
2215	EXTERNAL_ENCODER_CONTROL_PS_ALLOCATION_V3 params;
2216	EXTERNAL_ENCODER_CONTROL_PARAMETERS_V3 *cntl_params;
2217	struct graphics_object_id encoder;
2218	bool is_input_signal_dp = false;
2219
2220	memset(&params, 0, sizeof(params));
2221
2222	cntl_params = &params.sExtEncoder;
2223
2224	encoder = cntl->encoder_id;
2225
2226	/* check if encoder supports external encoder control table */
2227	switch (dal_graphics_object_id_get_encoder_id(encoder)) {
2228	case ENCODER_ID_EXTERNAL_NUTMEG:
2229	case ENCODER_ID_EXTERNAL_TRAVIS:
2230		is_input_signal_dp = true;
2231		break;
2232
2233	default:
2234		BREAK_TO_DEBUGGER();
2235		return BP_RESULT_BADINPUT;
2236	}
2237
2238	/* Fill information based on the action
2239	 *
2240	 * Bit[6:4]: indicate external encoder, applied to all functions.
2241	 * =0: external encoder1, mapped to external encoder enum id1
2242	 * =1: external encoder2, mapped to external encoder enum id2
2243	 *
2244	 * enum ObjectEnumId
2245	 * {
2246	 *  EnumId_Unknown = 0,
2247	 *  EnumId_1,
2248	 *  EnumId_2,
2249	 * };
2250	 */
2251	cntl_params->ucConfig = (uint8_t)((encoder.enum_id - 1) << 4);
2252
2253	switch (cntl->action) {
2254	case EXTERNAL_ENCODER_CONTROL_INIT:
2255		/* output display connector type. Only valid in encoder
2256		 * initialization */
2257		cntl_params->usConnectorId =
2258				cpu_to_le16((uint16_t)cntl->connector_obj_id.id);
2259		break;
2260	case EXTERNAL_ENCODER_CONTROL_SETUP:
2261		/* EXTERNAL_ENCODER_CONTROL_PARAMETERS_V3 pixel clock unit in
2262		 * 10KHz
2263		 * output display device pixel clock frequency in unit of 10KHz.
2264		 * Only valid in setup and enableoutput
2265		 */
2266		cntl_params->usPixelClock =
2267				cpu_to_le16((uint16_t)(cntl->pixel_clock / 10));
2268		/* Indicate display output signal type drive by external
2269		 * encoder, only valid in setup and enableoutput */
2270		cntl_params->ucEncoderMode =
2271				(uint8_t)bp->cmd_helper->encoder_mode_bp_to_atom(
2272						cntl->signal, false);
2273
2274		if (is_input_signal_dp) {
2275			/* Bit[0]: indicate link rate, =1: 2.7Ghz, =0: 1.62Ghz,
2276			 * only valid in encoder setup with DP mode. */
2277			if (LINK_RATE_HIGH == cntl->link_rate)
2278				cntl_params->ucConfig |= 1;
2279			/* output color depth Indicate encoder data bpc format
2280			 * in DP mode, only valid in encoder setup in DP mode.
2281			 */
2282			cntl_params->ucBitPerColor =
2283					(uint8_t)(cntl->color_depth);
2284		}
2285		/* Indicate how many lanes used by external encoder, only valid
2286		 * in encoder setup and enableoutput. */
2287		cntl_params->ucLaneNum = (uint8_t)(cntl->lanes_number);
2288		break;
2289	case EXTERNAL_ENCODER_CONTROL_ENABLE:
2290		cntl_params->usPixelClock =
2291				cpu_to_le16((uint16_t)(cntl->pixel_clock / 10));
2292		cntl_params->ucEncoderMode =
2293				(uint8_t)bp->cmd_helper->encoder_mode_bp_to_atom(
2294						cntl->signal, false);
2295		cntl_params->ucLaneNum = (uint8_t)cntl->lanes_number;
2296		break;
2297	default:
2298		break;
2299	}
2300
2301	cntl_params->ucAction = (uint8_t)cntl->action;
2302
2303	if (EXEC_BIOS_CMD_TABLE(ExternalEncoderControl, params))
2304		result = BP_RESULT_OK;
2305
2306	return result;
2307}
2308
2309/*******************************************************************************
2310 ********************************************************************************
2311 **
2312 **                  ENABLE DISPLAY POWER GATING
2313 **
2314 ********************************************************************************
2315 *******************************************************************************/
2316
2317static enum bp_result enable_disp_power_gating_v2_1(
2318	struct bios_parser *bp,
2319	enum controller_id crtc_id,
2320	enum bp_pipe_control_action action);
2321
2322static void init_enable_disp_power_gating(
2323	struct bios_parser *bp)
2324{
2325	switch (BIOS_CMD_TABLE_PARA_REVISION(EnableDispPowerGating)) {
2326	case 1:
2327		bp->cmd_tbl.enable_disp_power_gating =
2328				enable_disp_power_gating_v2_1;
2329		break;
2330	default:
2331		dm_output_to_console("Don't enable_disp_power_gating enable_crtc for v%d\n",
2332			 BIOS_CMD_TABLE_PARA_REVISION(EnableDispPowerGating));
2333		bp->cmd_tbl.enable_disp_power_gating = NULL;
2334		break;
2335	}
2336}
2337
2338static enum bp_result enable_disp_power_gating_v2_1(
2339	struct bios_parser *bp,
2340	enum controller_id crtc_id,
2341	enum bp_pipe_control_action action)
2342{
2343	enum bp_result result = BP_RESULT_FAILURE;
2344
2345	ENABLE_DISP_POWER_GATING_PS_ALLOCATION params = {0};
2346	uint8_t atom_crtc_id;
2347
2348	if (bp->cmd_helper->controller_id_to_atom(crtc_id, &atom_crtc_id))
2349		params.ucDispPipeId = atom_crtc_id;
2350	else
2351		return BP_RESULT_BADINPUT;
2352
2353	params.ucEnable =
2354			bp->cmd_helper->disp_power_gating_action_to_atom(action);
2355
2356	if (EXEC_BIOS_CMD_TABLE(EnableDispPowerGating, params))
2357		result = BP_RESULT_OK;
2358
2359	return result;
2360}
2361
2362/*******************************************************************************
2363 ********************************************************************************
2364 **
2365 **                  SET DCE CLOCK
2366 **
2367 ********************************************************************************
2368 *******************************************************************************/
2369static enum bp_result set_dce_clock_v2_1(
2370	struct bios_parser *bp,
2371	struct bp_set_dce_clock_parameters *bp_params);
2372
2373static void init_set_dce_clock(struct bios_parser *bp)
2374{
2375	switch (BIOS_CMD_TABLE_PARA_REVISION(SetDCEClock)) {
2376	case 1:
2377		bp->cmd_tbl.set_dce_clock = set_dce_clock_v2_1;
2378		break;
2379	default:
2380		dm_output_to_console("Don't have set_dce_clock for v%d\n",
2381			 BIOS_CMD_TABLE_PARA_REVISION(SetDCEClock));
2382		bp->cmd_tbl.set_dce_clock = NULL;
2383		break;
2384	}
2385}
2386
2387static enum bp_result set_dce_clock_v2_1(
2388	struct bios_parser *bp,
2389	struct bp_set_dce_clock_parameters *bp_params)
2390{
2391	enum bp_result result = BP_RESULT_FAILURE;
2392
2393	SET_DCE_CLOCK_PS_ALLOCATION_V2_1 params;
2394	uint32_t atom_pll_id;
2395	uint32_t atom_clock_type;
2396	const struct command_table_helper *cmd = bp->cmd_helper;
2397
2398	memset(&params, 0, sizeof(params));
2399
2400	if (!cmd->clock_source_id_to_atom(bp_params->pll_id, &atom_pll_id) ||
2401			!cmd->dc_clock_type_to_atom(bp_params->clock_type, &atom_clock_type))
2402		return BP_RESULT_BADINPUT;
2403
2404	params.asParam.ucDCEClkSrc  = atom_pll_id;
2405	params.asParam.ucDCEClkType = atom_clock_type;
2406
2407	if (bp_params->clock_type == DCECLOCK_TYPE_DPREFCLK) {
2408		if (bp_params->flags.USE_GENLOCK_AS_SOURCE_FOR_DPREFCLK)
2409			params.asParam.ucDCEClkFlag |= DCE_CLOCK_FLAG_PLL_REFCLK_SRC_GENLK;
2410
2411		if (bp_params->flags.USE_PCIE_AS_SOURCE_FOR_DPREFCLK)
2412			params.asParam.ucDCEClkFlag |= DCE_CLOCK_FLAG_PLL_REFCLK_SRC_PCIE;
2413
2414		if (bp_params->flags.USE_XTALIN_AS_SOURCE_FOR_DPREFCLK)
2415			params.asParam.ucDCEClkFlag |= DCE_CLOCK_FLAG_PLL_REFCLK_SRC_XTALIN;
2416
2417		if (bp_params->flags.USE_GENERICA_AS_SOURCE_FOR_DPREFCLK)
2418			params.asParam.ucDCEClkFlag |= DCE_CLOCK_FLAG_PLL_REFCLK_SRC_GENERICA;
2419	}
2420	else
2421		/* only program clock frequency if display clock is used; VBIOS will program DPREFCLK */
2422		/* We need to convert from KHz units into 10KHz units */
2423		params.asParam.ulDCEClkFreq = cpu_to_le32(bp_params->target_clock_frequency / 10);
2424
2425	if (EXEC_BIOS_CMD_TABLE(SetDCEClock, params)) {
2426		/* Convert from 10KHz units back to KHz */
2427		bp_params->target_clock_frequency = le32_to_cpu(params.asParam.ulDCEClkFreq) * 10;
2428		result = BP_RESULT_OK;
2429	}
2430
2431	return result;
2432}
v4.17
   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 "include/bios_parser_interface.h"
  31
  32#include "command_table.h"
  33#include "command_table_helper.h"
  34#include "bios_parser_helper.h"
  35#include "bios_parser_types_internal.h"
  36
  37#define EXEC_BIOS_CMD_TABLE(command, params)\
  38	(cgs_atom_exec_cmd_table(bp->base.ctx->cgs_device, \
  39		GetIndexIntoMasterTable(COMMAND, command), \
  40		&params) == 0)
  41
  42#define BIOS_CMD_TABLE_REVISION(command, frev, crev)\
  43	cgs_atom_get_cmd_table_revs(bp->base.ctx->cgs_device, \
  44		GetIndexIntoMasterTable(COMMAND, command), &frev, &crev)
  45
  46#define BIOS_CMD_TABLE_PARA_REVISION(command)\
  47	bios_cmd_table_para_revision(bp->base.ctx->cgs_device, \
  48		GetIndexIntoMasterTable(COMMAND, command))
  49
  50static void init_dig_encoder_control(struct bios_parser *bp);
  51static void init_transmitter_control(struct bios_parser *bp);
  52static void init_set_pixel_clock(struct bios_parser *bp);
  53static void init_enable_spread_spectrum_on_ppll(struct bios_parser *bp);
  54static void init_adjust_display_pll(struct bios_parser *bp);
  55static void init_dac_encoder_control(struct bios_parser *bp);
  56static void init_dac_output_control(struct bios_parser *bp);
  57static void init_set_crtc_timing(struct bios_parser *bp);
  58static void init_select_crtc_source(struct bios_parser *bp);
  59static void init_enable_crtc(struct bios_parser *bp);
  60static void init_enable_crtc_mem_req(struct bios_parser *bp);
  61static void init_external_encoder_control(struct bios_parser *bp);
  62static void init_enable_disp_power_gating(struct bios_parser *bp);
  63static void init_program_clock(struct bios_parser *bp);
  64static void init_set_dce_clock(struct bios_parser *bp);
  65
  66void dal_bios_parser_init_cmd_tbl(struct bios_parser *bp)
  67{
  68	init_dig_encoder_control(bp);
  69	init_transmitter_control(bp);
  70	init_set_pixel_clock(bp);
  71	init_enable_spread_spectrum_on_ppll(bp);
  72	init_adjust_display_pll(bp);
  73	init_dac_encoder_control(bp);
  74	init_dac_output_control(bp);
  75	init_set_crtc_timing(bp);
  76	init_select_crtc_source(bp);
  77	init_enable_crtc(bp);
  78	init_enable_crtc_mem_req(bp);
  79	init_program_clock(bp);
  80	init_external_encoder_control(bp);
  81	init_enable_disp_power_gating(bp);
  82	init_set_dce_clock(bp);
  83}
  84
  85static uint32_t bios_cmd_table_para_revision(void *cgs_device,
  86					     uint32_t index)
  87{
 
  88	uint8_t frev, crev;
  89
  90	if (cgs_atom_get_cmd_table_revs(cgs_device,
  91					index,
  92					&frev, &crev) != 0)
 
 
  93		return 0;
  94	return crev;
  95}
  96
  97/*******************************************************************************
  98 ********************************************************************************
  99 **
 100 **                  D I G E N C O D E R C O N T R O L
 101 **
 102 ********************************************************************************
 103 *******************************************************************************/
 104static enum bp_result encoder_control_digx_v3(
 105	struct bios_parser *bp,
 106	struct bp_encoder_control *cntl);
 107
 108static enum bp_result encoder_control_digx_v4(
 109	struct bios_parser *bp,
 110	struct bp_encoder_control *cntl);
 111
 112static enum bp_result encoder_control_digx_v5(
 113	struct bios_parser *bp,
 114	struct bp_encoder_control *cntl);
 115
 116static void init_encoder_control_dig_v1(struct bios_parser *bp);
 117
 118static void init_dig_encoder_control(struct bios_parser *bp)
 119{
 120	uint32_t version =
 121		BIOS_CMD_TABLE_PARA_REVISION(DIGxEncoderControl);
 122
 123	switch (version) {
 124	case 2:
 125		bp->cmd_tbl.dig_encoder_control = encoder_control_digx_v3;
 126		break;
 127	case 4:
 128		bp->cmd_tbl.dig_encoder_control = encoder_control_digx_v4;
 129		break;
 130
 131	case 5:
 132		bp->cmd_tbl.dig_encoder_control = encoder_control_digx_v5;
 133		break;
 134
 135	default:
 136		init_encoder_control_dig_v1(bp);
 137		break;
 138	}
 139}
 140
 141static enum bp_result encoder_control_dig_v1(
 142	struct bios_parser *bp,
 143	struct bp_encoder_control *cntl);
 144static enum bp_result encoder_control_dig1_v1(
 145	struct bios_parser *bp,
 146	struct bp_encoder_control *cntl);
 147static enum bp_result encoder_control_dig2_v1(
 148	struct bios_parser *bp,
 149	struct bp_encoder_control *cntl);
 150
 151static void init_encoder_control_dig_v1(struct bios_parser *bp)
 152{
 153	struct cmd_tbl *cmd_tbl = &bp->cmd_tbl;
 154
 155	if (1 == BIOS_CMD_TABLE_PARA_REVISION(DIG1EncoderControl))
 156		cmd_tbl->encoder_control_dig1 = encoder_control_dig1_v1;
 157	else
 158		cmd_tbl->encoder_control_dig1 = NULL;
 159
 160	if (1 == BIOS_CMD_TABLE_PARA_REVISION(DIG2EncoderControl))
 161		cmd_tbl->encoder_control_dig2 = encoder_control_dig2_v1;
 162	else
 163		cmd_tbl->encoder_control_dig2 = NULL;
 164
 165	cmd_tbl->dig_encoder_control = encoder_control_dig_v1;
 166}
 167
 168static enum bp_result encoder_control_dig_v1(
 169	struct bios_parser *bp,
 170	struct bp_encoder_control *cntl)
 171{
 172	enum bp_result result = BP_RESULT_FAILURE;
 173	struct cmd_tbl *cmd_tbl = &bp->cmd_tbl;
 174
 175	if (cntl != NULL)
 176		switch (cntl->engine_id) {
 177		case ENGINE_ID_DIGA:
 178			if (cmd_tbl->encoder_control_dig1 != NULL)
 179				result =
 180					cmd_tbl->encoder_control_dig1(bp, cntl);
 181			break;
 182		case ENGINE_ID_DIGB:
 183			if (cmd_tbl->encoder_control_dig2 != NULL)
 184				result =
 185					cmd_tbl->encoder_control_dig2(bp, cntl);
 186			break;
 187
 188		default:
 189			break;
 190		}
 191
 192	return result;
 193}
 194
 195static enum bp_result encoder_control_dig1_v1(
 196	struct bios_parser *bp,
 197	struct bp_encoder_control *cntl)
 198{
 199	enum bp_result result = BP_RESULT_FAILURE;
 200	DIG_ENCODER_CONTROL_PARAMETERS_V2 params = {0};
 201
 202	bp->cmd_helper->assign_control_parameter(bp->cmd_helper, cntl, &params);
 203
 204	if (EXEC_BIOS_CMD_TABLE(DIG1EncoderControl, params))
 205		result = BP_RESULT_OK;
 206
 207	return result;
 208}
 209
 210static enum bp_result encoder_control_dig2_v1(
 211	struct bios_parser *bp,
 212	struct bp_encoder_control *cntl)
 213{
 214	enum bp_result result = BP_RESULT_FAILURE;
 215	DIG_ENCODER_CONTROL_PARAMETERS_V2 params = {0};
 216
 217	bp->cmd_helper->assign_control_parameter(bp->cmd_helper, cntl, &params);
 218
 219	if (EXEC_BIOS_CMD_TABLE(DIG2EncoderControl, params))
 220		result = BP_RESULT_OK;
 221
 222	return result;
 223}
 224
 225static enum bp_result encoder_control_digx_v3(
 226	struct bios_parser *bp,
 227	struct bp_encoder_control *cntl)
 228{
 229	enum bp_result result = BP_RESULT_FAILURE;
 230	DIG_ENCODER_CONTROL_PARAMETERS_V3 params = {0};
 231
 232	if (LANE_COUNT_FOUR < cntl->lanes_number)
 233		params.acConfig.ucDPLinkRate = 1; /* dual link 2.7GHz */
 234	else
 235		params.acConfig.ucDPLinkRate = 0; /* single link 1.62GHz */
 236
 237	params.acConfig.ucDigSel = (uint8_t)(cntl->engine_id);
 238
 239	/* We need to convert from KHz units into 10KHz units */
 240	params.ucAction = bp->cmd_helper->encoder_action_to_atom(cntl->action);
 241	params.usPixelClock = cpu_to_le16((uint16_t)(cntl->pixel_clock / 10));
 242	params.ucEncoderMode =
 243			(uint8_t)bp->cmd_helper->encoder_mode_bp_to_atom(
 244					cntl->signal,
 245					cntl->enable_dp_audio);
 246	params.ucLaneNum = (uint8_t)(cntl->lanes_number);
 247
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 248	if (EXEC_BIOS_CMD_TABLE(DIGxEncoderControl, params))
 249		result = BP_RESULT_OK;
 250
 251	return result;
 252}
 253
 254static enum bp_result encoder_control_digx_v4(
 255	struct bios_parser *bp,
 256	struct bp_encoder_control *cntl)
 257{
 258	enum bp_result result = BP_RESULT_FAILURE;
 259	DIG_ENCODER_CONTROL_PARAMETERS_V4 params = {0};
 260
 261	if (LANE_COUNT_FOUR < cntl->lanes_number)
 262		params.acConfig.ucDPLinkRate = 1; /* dual link 2.7GHz */
 263	else
 264		params.acConfig.ucDPLinkRate = 0; /* single link 1.62GHz */
 265
 266	params.acConfig.ucDigSel = (uint8_t)(cntl->engine_id);
 267
 268	/* We need to convert from KHz units into 10KHz units */
 269	params.ucAction = bp->cmd_helper->encoder_action_to_atom(cntl->action);
 270	params.usPixelClock = cpu_to_le16((uint16_t)(cntl->pixel_clock / 10));
 271	params.ucEncoderMode =
 272			(uint8_t)(bp->cmd_helper->encoder_mode_bp_to_atom(
 273					cntl->signal,
 274					cntl->enable_dp_audio));
 275	params.ucLaneNum = (uint8_t)(cntl->lanes_number);
 276
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 277	if (EXEC_BIOS_CMD_TABLE(DIGxEncoderControl, params))
 278		result = BP_RESULT_OK;
 279
 280	return result;
 281}
 282
 283static enum bp_result encoder_control_digx_v5(
 284	struct bios_parser *bp,
 285	struct bp_encoder_control *cntl)
 286{
 287	enum bp_result result = BP_RESULT_FAILURE;
 288	ENCODER_STREAM_SETUP_PARAMETERS_V5 params = {0};
 289
 290	params.ucDigId = (uint8_t)(cntl->engine_id);
 291	params.ucAction = bp->cmd_helper->encoder_action_to_atom(cntl->action);
 292
 293	params.ulPixelClock = cntl->pixel_clock / 10;
 294	params.ucDigMode =
 295			(uint8_t)(bp->cmd_helper->encoder_mode_bp_to_atom(
 296					cntl->signal,
 297					cntl->enable_dp_audio));
 298	params.ucLaneNum = (uint8_t)(cntl->lanes_number);
 299
 300	switch (cntl->color_depth) {
 301	case COLOR_DEPTH_888:
 302		params.ucBitPerColor = PANEL_8BIT_PER_COLOR;
 303		break;
 304	case COLOR_DEPTH_101010:
 305		params.ucBitPerColor = PANEL_10BIT_PER_COLOR;
 306		break;
 307	case COLOR_DEPTH_121212:
 308		params.ucBitPerColor = PANEL_12BIT_PER_COLOR;
 309		break;
 310	case COLOR_DEPTH_161616:
 311		params.ucBitPerColor = PANEL_16BIT_PER_COLOR;
 312		break;
 313	default:
 314		break;
 315	}
 316
 317	if (cntl->signal == SIGNAL_TYPE_HDMI_TYPE_A)
 318		switch (cntl->color_depth) {
 319		case COLOR_DEPTH_101010:
 320			params.ulPixelClock =
 321				(params.ulPixelClock * 30) / 24;
 322			break;
 323		case COLOR_DEPTH_121212:
 324			params.ulPixelClock =
 325				(params.ulPixelClock * 36) / 24;
 326			break;
 327		case COLOR_DEPTH_161616:
 328			params.ulPixelClock =
 329				(params.ulPixelClock * 48) / 24;
 330			break;
 331		default:
 332			break;
 333		}
 334
 335	if (EXEC_BIOS_CMD_TABLE(DIGxEncoderControl, params))
 336		result = BP_RESULT_OK;
 337
 338	return result;
 339}
 340
 341/*******************************************************************************
 342 ********************************************************************************
 343 **
 344 **                  TRANSMITTER CONTROL
 345 **
 346 ********************************************************************************
 347 *******************************************************************************/
 348
 349static enum bp_result transmitter_control_v2(
 350	struct bios_parser *bp,
 351	struct bp_transmitter_control *cntl);
 352static enum bp_result transmitter_control_v3(
 353	struct bios_parser *bp,
 354	struct bp_transmitter_control *cntl);
 355static enum bp_result transmitter_control_v4(
 356	struct bios_parser *bp,
 357	struct bp_transmitter_control *cntl);
 358static enum bp_result transmitter_control_v1_5(
 359	struct bios_parser *bp,
 360	struct bp_transmitter_control *cntl);
 361static enum bp_result transmitter_control_v1_6(
 362	struct bios_parser *bp,
 363	struct bp_transmitter_control *cntl);
 364
 365static void init_transmitter_control(struct bios_parser *bp)
 366{
 367	uint8_t frev;
 368	uint8_t crev;
 369
 370	if (BIOS_CMD_TABLE_REVISION(UNIPHYTransmitterControl,
 371			frev, crev) != 0)
 372		BREAK_TO_DEBUGGER();
 373	switch (crev) {
 374	case 2:
 375		bp->cmd_tbl.transmitter_control = transmitter_control_v2;
 376		break;
 377	case 3:
 378		bp->cmd_tbl.transmitter_control = transmitter_control_v3;
 379		break;
 380	case 4:
 381		bp->cmd_tbl.transmitter_control = transmitter_control_v4;
 382		break;
 383	case 5:
 384		bp->cmd_tbl.transmitter_control = transmitter_control_v1_5;
 385		break;
 386	case 6:
 387		bp->cmd_tbl.transmitter_control = transmitter_control_v1_6;
 388		break;
 389	default:
 390		dm_output_to_console("Don't have transmitter_control for v%d\n", crev);
 391		bp->cmd_tbl.transmitter_control = NULL;
 392		break;
 393	}
 394}
 395
 396static enum bp_result transmitter_control_v2(
 397	struct bios_parser *bp,
 398	struct bp_transmitter_control *cntl)
 399{
 400	enum bp_result result = BP_RESULT_FAILURE;
 401	DIG_TRANSMITTER_CONTROL_PARAMETERS_V2 params;
 402	enum connector_id connector_id =
 403		dal_graphics_object_id_get_connector_id(cntl->connector_obj_id);
 404
 405	memset(&params, 0, sizeof(params));
 406
 407	switch (cntl->transmitter) {
 408	case TRANSMITTER_UNIPHY_A:
 409	case TRANSMITTER_UNIPHY_B:
 410	case TRANSMITTER_UNIPHY_C:
 411	case TRANSMITTER_UNIPHY_D:
 412	case TRANSMITTER_UNIPHY_E:
 413	case TRANSMITTER_UNIPHY_F:
 414	case TRANSMITTER_TRAVIS_LCD:
 415		break;
 416	default:
 417		return BP_RESULT_BADINPUT;
 418	}
 419
 420	switch (cntl->action) {
 421	case TRANSMITTER_CONTROL_INIT:
 422		if ((CONNECTOR_ID_DUAL_LINK_DVII == connector_id) ||
 423				(CONNECTOR_ID_DUAL_LINK_DVID == connector_id))
 424			/* on INIT this bit should be set according to the
 425			 * phisycal connector
 426			 * Bit0: dual link connector flag
 427			 * =0 connector is single link connector
 428			 * =1 connector is dual link connector
 429			 */
 430			params.acConfig.fDualLinkConnector = 1;
 431
 432		/* connector object id */
 433		params.usInitInfo =
 434				cpu_to_le16((uint8_t)cntl->connector_obj_id.id);
 435		break;
 436	case TRANSMITTER_CONTROL_SET_VOLTAGE_AND_PREEMPASIS:
 437		/* votage swing and pre-emphsis */
 438		params.asMode.ucLaneSel = (uint8_t)cntl->lane_select;
 439		params.asMode.ucLaneSet = (uint8_t)cntl->lane_settings;
 440		break;
 441	default:
 442		/* if dual-link */
 443		if (LANE_COUNT_FOUR < cntl->lanes_number) {
 444			/* on ENABLE/DISABLE this bit should be set according to
 445			 * actual timing (number of lanes)
 446			 * Bit0: dual link connector flag
 447			 * =0 connector is single link connector
 448			 * =1 connector is dual link connector
 449			 */
 450			params.acConfig.fDualLinkConnector = 1;
 451
 452			/* link rate, half for dual link
 453			 * We need to convert from KHz units into 20KHz units
 454			 */
 455			params.usPixelClock =
 456					cpu_to_le16((uint16_t)(cntl->pixel_clock / 20));
 457		} else
 458			/* link rate, half for dual link
 459			 * We need to convert from KHz units into 10KHz units
 460			 */
 461			params.usPixelClock =
 462					cpu_to_le16((uint16_t)(cntl->pixel_clock / 10));
 463		break;
 464	}
 465
 466	/* 00 - coherent mode
 467	 * 01 - incoherent mode
 468	 */
 469
 470	params.acConfig.fCoherentMode = cntl->coherent;
 471
 472	if ((TRANSMITTER_UNIPHY_B == cntl->transmitter)
 473			|| (TRANSMITTER_UNIPHY_D == cntl->transmitter)
 474			|| (TRANSMITTER_UNIPHY_F == cntl->transmitter))
 475		/* Bit2: Transmitter Link selection
 476		 * =0 when bit0=0, single link A/C/E, when bit0=1,
 477		 * master link A/C/E
 478		 * =1 when bit0=0, single link B/D/F, when bit0=1,
 479		 * master link B/D/F
 480		 */
 481		params.acConfig.ucLinkSel = 1;
 482
 483	if (ENGINE_ID_DIGB == cntl->engine_id)
 484		/* Bit3: Transmitter data source selection
 485		 * =0 DIGA is data source.
 486		 * =1 DIGB is data source.
 487		 * This bit is only useful when ucAction= ATOM_ENABLE
 488		 */
 489		params.acConfig.ucEncoderSel = 1;
 490
 491	if (CONNECTOR_ID_DISPLAY_PORT == connector_id)
 
 492		/* Bit4: DP connector flag
 493		 * =0 connector is none-DP connector
 494		 * =1 connector is DP connector
 495		 */
 496		params.acConfig.fDPConnector = 1;
 497
 498	/* Bit[7:6]: Transmitter selection
 499	 * =0 UNIPHY_ENCODER: UNIPHYA/B
 500	 * =1 UNIPHY1_ENCODER: UNIPHYC/D
 501	 * =2 UNIPHY2_ENCODER: UNIPHYE/F
 502	 * =3 reserved
 503	 */
 504	params.acConfig.ucTransmitterSel =
 505			(uint8_t)bp->cmd_helper->transmitter_bp_to_atom(
 506					cntl->transmitter);
 507
 508	params.ucAction = (uint8_t)cntl->action;
 509
 510	if (EXEC_BIOS_CMD_TABLE(UNIPHYTransmitterControl, params))
 511		result = BP_RESULT_OK;
 512
 513	return result;
 514}
 515
 516static enum bp_result transmitter_control_v3(
 517	struct bios_parser *bp,
 518	struct bp_transmitter_control *cntl)
 519{
 520	enum bp_result result = BP_RESULT_FAILURE;
 521	DIG_TRANSMITTER_CONTROL_PARAMETERS_V3 params;
 522	uint32_t pll_id;
 523	enum connector_id conn_id =
 524			dal_graphics_object_id_get_connector_id(cntl->connector_obj_id);
 525	const struct command_table_helper *cmd = bp->cmd_helper;
 526	bool dual_link_conn = (CONNECTOR_ID_DUAL_LINK_DVII == conn_id)
 527					|| (CONNECTOR_ID_DUAL_LINK_DVID == conn_id);
 528
 529	memset(&params, 0, sizeof(params));
 530
 531	switch (cntl->transmitter) {
 532	case TRANSMITTER_UNIPHY_A:
 533	case TRANSMITTER_UNIPHY_B:
 534	case TRANSMITTER_UNIPHY_C:
 535	case TRANSMITTER_UNIPHY_D:
 536	case TRANSMITTER_UNIPHY_E:
 537	case TRANSMITTER_UNIPHY_F:
 538	case TRANSMITTER_TRAVIS_LCD:
 539		break;
 540	default:
 541		return BP_RESULT_BADINPUT;
 542	}
 543
 544	if (!cmd->clock_source_id_to_atom(cntl->pll_id, &pll_id))
 545		return BP_RESULT_BADINPUT;
 546
 547	/* fill information based on the action */
 548	switch (cntl->action) {
 549	case TRANSMITTER_CONTROL_INIT:
 550		if (dual_link_conn) {
 551			/* on INIT this bit should be set according to the
 552			 * phisycal connector
 553			 * Bit0: dual link connector flag
 554			 * =0 connector is single link connector
 555			 * =1 connector is dual link connector
 556			 */
 557			params.acConfig.fDualLinkConnector = 1;
 558		}
 559
 560		/* connector object id */
 561		params.usInitInfo =
 562				cpu_to_le16((uint8_t)(cntl->connector_obj_id.id));
 563		break;
 564	case TRANSMITTER_CONTROL_SET_VOLTAGE_AND_PREEMPASIS:
 565		/* votage swing and pre-emphsis */
 566		params.asMode.ucLaneSel = (uint8_t)cntl->lane_select;
 567		params.asMode.ucLaneSet = (uint8_t)cntl->lane_settings;
 568		break;
 569	default:
 570		if (dual_link_conn && cntl->multi_path)
 571			/* on ENABLE/DISABLE this bit should be set according to
 572			 * actual timing (number of lanes)
 573			 * Bit0: dual link connector flag
 574			 * =0 connector is single link connector
 575			 * =1 connector is dual link connector
 576			 */
 577			params.acConfig.fDualLinkConnector = 1;
 578
 579		/* if dual-link */
 580		if (LANE_COUNT_FOUR < cntl->lanes_number) {
 581			/* on ENABLE/DISABLE this bit should be set according to
 582			 * actual timing (number of lanes)
 583			 * Bit0: dual link connector flag
 584			 * =0 connector is single link connector
 585			 * =1 connector is dual link connector
 586			 */
 587			params.acConfig.fDualLinkConnector = 1;
 588
 589			/* link rate, half for dual link
 590			 * We need to convert from KHz units into 20KHz units
 591			 */
 592			params.usPixelClock =
 593					cpu_to_le16((uint16_t)(cntl->pixel_clock / 20));
 594		} else {
 595			/* link rate, half for dual link
 596			 * We need to convert from KHz units into 10KHz units
 597			 */
 598			params.usPixelClock =
 599					cpu_to_le16((uint16_t)(cntl->pixel_clock / 10));
 600		}
 601		break;
 602	}
 603
 604	/* 00 - coherent mode
 605	 * 01 - incoherent mode
 606	 */
 607
 608	params.acConfig.fCoherentMode = cntl->coherent;
 609
 610	if ((TRANSMITTER_UNIPHY_B == cntl->transmitter)
 611		|| (TRANSMITTER_UNIPHY_D == cntl->transmitter)
 612		|| (TRANSMITTER_UNIPHY_F == cntl->transmitter))
 613		/* Bit2: Transmitter Link selection
 614		 * =0 when bit0=0, single link A/C/E, when bit0=1,
 615		 * master link A/C/E
 616		 * =1 when bit0=0, single link B/D/F, when bit0=1,
 617		 * master link B/D/F
 618		 */
 619		params.acConfig.ucLinkSel = 1;
 620
 621	if (ENGINE_ID_DIGB == cntl->engine_id)
 622		/* Bit3: Transmitter data source selection
 623		 * =0 DIGA is data source.
 624		 * =1 DIGB is data source.
 625		 * This bit is only useful when ucAction= ATOM_ENABLE
 626		 */
 627		params.acConfig.ucEncoderSel = 1;
 628
 629	/* Bit[7:6]: Transmitter selection
 630	 * =0 UNIPHY_ENCODER: UNIPHYA/B
 631	 * =1 UNIPHY1_ENCODER: UNIPHYC/D
 632	 * =2 UNIPHY2_ENCODER: UNIPHYE/F
 633	 * =3 reserved
 634	 */
 635	params.acConfig.ucTransmitterSel =
 636			(uint8_t)cmd->transmitter_bp_to_atom(cntl->transmitter);
 637
 638	params.ucLaneNum = (uint8_t)cntl->lanes_number;
 639
 640	params.acConfig.ucRefClkSource = (uint8_t)pll_id;
 641
 642	params.ucAction = (uint8_t)cntl->action;
 643
 644	if (EXEC_BIOS_CMD_TABLE(UNIPHYTransmitterControl, params))
 645		result = BP_RESULT_OK;
 646
 647	return result;
 648}
 649
 650static enum bp_result transmitter_control_v4(
 651	struct bios_parser *bp,
 652	struct bp_transmitter_control *cntl)
 653{
 654	enum bp_result result = BP_RESULT_FAILURE;
 655	DIG_TRANSMITTER_CONTROL_PARAMETERS_V4 params;
 656	uint32_t ref_clk_src_id;
 657	enum connector_id conn_id =
 658			dal_graphics_object_id_get_connector_id(cntl->connector_obj_id);
 659	const struct command_table_helper *cmd = bp->cmd_helper;
 660
 661	memset(&params, 0, sizeof(params));
 662
 663	switch (cntl->transmitter) {
 664	case TRANSMITTER_UNIPHY_A:
 665	case TRANSMITTER_UNIPHY_B:
 666	case TRANSMITTER_UNIPHY_C:
 667	case TRANSMITTER_UNIPHY_D:
 668	case TRANSMITTER_UNIPHY_E:
 669	case TRANSMITTER_UNIPHY_F:
 670	case TRANSMITTER_TRAVIS_LCD:
 671		break;
 672	default:
 673		return BP_RESULT_BADINPUT;
 674	}
 675
 676	if (!cmd->clock_source_id_to_ref_clk_src(cntl->pll_id, &ref_clk_src_id))
 677		return BP_RESULT_BADINPUT;
 678
 679	switch (cntl->action) {
 680	case TRANSMITTER_CONTROL_INIT:
 681	{
 682		if ((CONNECTOR_ID_DUAL_LINK_DVII == conn_id) ||
 683				(CONNECTOR_ID_DUAL_LINK_DVID == conn_id))
 684			/* on INIT this bit should be set according to the
 685			 * phisycal connector
 686			 * Bit0: dual link connector flag
 687			 * =0 connector is single link connector
 688			 * =1 connector is dual link connector
 689			 */
 690			params.acConfig.fDualLinkConnector = 1;
 691
 692		/* connector object id */
 693		params.usInitInfo =
 694				cpu_to_le16((uint8_t)(cntl->connector_obj_id.id));
 695	}
 696	break;
 697	case TRANSMITTER_CONTROL_SET_VOLTAGE_AND_PREEMPASIS:
 698		/* votage swing and pre-emphsis */
 699		params.asMode.ucLaneSel = (uint8_t)(cntl->lane_select);
 700		params.asMode.ucLaneSet = (uint8_t)(cntl->lane_settings);
 701		break;
 702	default:
 703		if ((CONNECTOR_ID_DUAL_LINK_DVII == conn_id) ||
 704				(CONNECTOR_ID_DUAL_LINK_DVID == conn_id))
 705			/* on ENABLE/DISABLE this bit should be set according to
 706			 * actual timing (number of lanes)
 707			 * Bit0: dual link connector flag
 708			 * =0 connector is single link connector
 709			 * =1 connector is dual link connector
 710			 */
 711			params.acConfig.fDualLinkConnector = 1;
 712
 713		/* if dual-link */
 714		if (LANE_COUNT_FOUR < cntl->lanes_number)
 715			/* link rate, half for dual link
 716			 * We need to convert from KHz units into 20KHz units
 717			 */
 718			params.usPixelClock =
 719					cpu_to_le16((uint16_t)(cntl->pixel_clock / 20));
 720		else {
 721			/* link rate, half for dual link
 722			 * We need to convert from KHz units into 10KHz units
 723			 */
 724			params.usPixelClock =
 725					cpu_to_le16((uint16_t)(cntl->pixel_clock / 10));
 726		}
 727		break;
 728	}
 729
 730	/* 00 - coherent mode
 731	 * 01 - incoherent mode
 732	 */
 733
 734	params.acConfig.fCoherentMode = cntl->coherent;
 735
 736	if ((TRANSMITTER_UNIPHY_B == cntl->transmitter)
 737		|| (TRANSMITTER_UNIPHY_D == cntl->transmitter)
 738		|| (TRANSMITTER_UNIPHY_F == cntl->transmitter))
 739		/* Bit2: Transmitter Link selection
 740		 * =0 when bit0=0, single link A/C/E, when bit0=1,
 741		 * master link A/C/E
 742		 * =1 when bit0=0, single link B/D/F, when bit0=1,
 743		 * master link B/D/F
 744		 */
 745		params.acConfig.ucLinkSel = 1;
 746
 747	if (ENGINE_ID_DIGB == cntl->engine_id)
 748		/* Bit3: Transmitter data source selection
 749		 * =0 DIGA is data source.
 750		 * =1 DIGB is data source.
 751		 * This bit is only useful when ucAction= ATOM_ENABLE
 752		 */
 753		params.acConfig.ucEncoderSel = 1;
 754
 755	/* Bit[7:6]: Transmitter selection
 756	 * =0 UNIPHY_ENCODER: UNIPHYA/B
 757	 * =1 UNIPHY1_ENCODER: UNIPHYC/D
 758	 * =2 UNIPHY2_ENCODER: UNIPHYE/F
 759	 * =3 reserved
 760	 */
 761	params.acConfig.ucTransmitterSel =
 762		(uint8_t)(cmd->transmitter_bp_to_atom(cntl->transmitter));
 763	params.ucLaneNum = (uint8_t)(cntl->lanes_number);
 764	params.acConfig.ucRefClkSource = (uint8_t)(ref_clk_src_id);
 765	params.ucAction = (uint8_t)(cntl->action);
 766
 767	if (EXEC_BIOS_CMD_TABLE(UNIPHYTransmitterControl, params))
 768		result = BP_RESULT_OK;
 769
 770	return result;
 771}
 772
 773static enum bp_result transmitter_control_v1_5(
 774	struct bios_parser *bp,
 775	struct bp_transmitter_control *cntl)
 776{
 777	enum bp_result result = BP_RESULT_FAILURE;
 778	const struct command_table_helper *cmd = bp->cmd_helper;
 779	DIG_TRANSMITTER_CONTROL_PARAMETERS_V1_5 params;
 780
 781	memset(&params, 0, sizeof(params));
 782	params.ucPhyId = cmd->phy_id_to_atom(cntl->transmitter);
 783	params.ucAction = (uint8_t)cntl->action;
 784	params.ucLaneNum = (uint8_t)cntl->lanes_number;
 785	params.ucConnObjId = (uint8_t)cntl->connector_obj_id.id;
 786
 787	params.ucDigMode =
 788		cmd->signal_type_to_atom_dig_mode(cntl->signal);
 789	params.asConfig.ucPhyClkSrcId =
 790		cmd->clock_source_id_to_atom_phy_clk_src_id(cntl->pll_id);
 791	/* 00 - coherent mode */
 792	params.asConfig.ucCoherentMode = cntl->coherent;
 793	params.asConfig.ucHPDSel =
 794		cmd->hpd_sel_to_atom(cntl->hpd_sel);
 795	params.ucDigEncoderSel =
 796		cmd->dig_encoder_sel_to_atom(cntl->engine_id);
 797	params.ucDPLaneSet = (uint8_t) cntl->lane_settings;
 798	params.usSymClock = cpu_to_le16((uint16_t) (cntl->pixel_clock / 10));
 799	/*
 800	 * In SI/TN case, caller have to set usPixelClock as following:
 801	 * DP mode: usPixelClock = DP_LINK_CLOCK/10
 802	 * (DP_LINK_CLOCK = 1.62GHz, 2.7GHz, 5.4GHz)
 803	 * DVI single link mode: usPixelClock = pixel clock
 804	 * DVI dual link mode: usPixelClock = pixel clock
 805	 * HDMI mode: usPixelClock = pixel clock * deep_color_ratio
 806	 * (=1: 8bpp, =1.25: 10bpp, =1.5:12bpp, =2: 16bpp)
 807	 * LVDS mode: usPixelClock = pixel clock
 808	 */
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 809
 810	if (EXEC_BIOS_CMD_TABLE(UNIPHYTransmitterControl, params))
 811		result = BP_RESULT_OK;
 812
 813	return result;
 814}
 815
 816static enum bp_result transmitter_control_v1_6(
 817	struct bios_parser *bp,
 818	struct bp_transmitter_control *cntl)
 819{
 820	enum bp_result result = BP_RESULT_FAILURE;
 821	const struct command_table_helper *cmd = bp->cmd_helper;
 822	DIG_TRANSMITTER_CONTROL_PARAMETERS_V1_6 params;
 823
 824	memset(&params, 0, sizeof(params));
 825	params.ucPhyId = cmd->phy_id_to_atom(cntl->transmitter);
 826	params.ucAction = (uint8_t)cntl->action;
 827
 828	if (cntl->action == TRANSMITTER_CONTROL_SET_VOLTAGE_AND_PREEMPASIS)
 829		params.ucDPLaneSet = (uint8_t)cntl->lane_settings;
 830	else
 831		params.ucDigMode = cmd->signal_type_to_atom_dig_mode(cntl->signal);
 832
 833	params.ucLaneNum = (uint8_t)cntl->lanes_number;
 834	params.ucHPDSel = cmd->hpd_sel_to_atom(cntl->hpd_sel);
 835	params.ucDigEncoderSel = cmd->dig_encoder_sel_to_atom(cntl->engine_id);
 836	params.ucConnObjId = (uint8_t)cntl->connector_obj_id.id;
 837	params.ulSymClock = cntl->pixel_clock/10;
 838
 839	/*
 840	 * In SI/TN case, caller have to set usPixelClock as following:
 841	 * DP mode: usPixelClock = DP_LINK_CLOCK/10
 842	 * (DP_LINK_CLOCK = 1.62GHz, 2.7GHz, 5.4GHz)
 843	 * DVI single link mode: usPixelClock = pixel clock
 844	 * DVI dual link mode: usPixelClock = pixel clock
 845	 * HDMI mode: usPixelClock = pixel clock * deep_color_ratio
 846	 * (=1: 8bpp, =1.25: 10bpp, =1.5:12bpp, =2: 16bpp)
 847	 * LVDS mode: usPixelClock = pixel clock
 848	 */
 849	switch (cntl->signal) {
 850	case SIGNAL_TYPE_HDMI_TYPE_A:
 851		switch (cntl->color_depth) {
 852		case COLOR_DEPTH_101010:
 853			params.ulSymClock =
 854				cpu_to_le16((le16_to_cpu(params.ulSymClock) * 30) / 24);
 855			break;
 856		case COLOR_DEPTH_121212:
 857			params.ulSymClock =
 858				cpu_to_le16((le16_to_cpu(params.ulSymClock) * 36) / 24);
 859			break;
 860		case COLOR_DEPTH_161616:
 861			params.ulSymClock =
 862				cpu_to_le16((le16_to_cpu(params.ulSymClock) * 48) / 24);
 863			break;
 864		default:
 865			break;
 866		}
 867		break;
 868		default:
 869			break;
 870	}
 871
 872	if (EXEC_BIOS_CMD_TABLE(UNIPHYTransmitterControl, params))
 873		result = BP_RESULT_OK;
 874	return result;
 875}
 876
 877/*******************************************************************************
 878 ********************************************************************************
 879 **
 880 **                  SET PIXEL CLOCK
 881 **
 882 ********************************************************************************
 883 *******************************************************************************/
 884
 885static enum bp_result set_pixel_clock_v3(
 886	struct bios_parser *bp,
 887	struct bp_pixel_clock_parameters *bp_params);
 888static enum bp_result set_pixel_clock_v5(
 889	struct bios_parser *bp,
 890	struct bp_pixel_clock_parameters *bp_params);
 891static enum bp_result set_pixel_clock_v6(
 892	struct bios_parser *bp,
 893	struct bp_pixel_clock_parameters *bp_params);
 894static enum bp_result set_pixel_clock_v7(
 895	struct bios_parser *bp,
 896	struct bp_pixel_clock_parameters *bp_params);
 897
 898static void init_set_pixel_clock(struct bios_parser *bp)
 899{
 900	switch (BIOS_CMD_TABLE_PARA_REVISION(SetPixelClock)) {
 901	case 3:
 902		bp->cmd_tbl.set_pixel_clock = set_pixel_clock_v3;
 903		break;
 904	case 5:
 905		bp->cmd_tbl.set_pixel_clock = set_pixel_clock_v5;
 906		break;
 907	case 6:
 908		bp->cmd_tbl.set_pixel_clock = set_pixel_clock_v6;
 909		break;
 910	case 7:
 911		bp->cmd_tbl.set_pixel_clock = set_pixel_clock_v7;
 912		break;
 913	default:
 914		dm_output_to_console("Don't have set_pixel_clock for v%d\n",
 915			 BIOS_CMD_TABLE_PARA_REVISION(SetPixelClock));
 916		bp->cmd_tbl.set_pixel_clock = NULL;
 917		break;
 918	}
 919}
 920
 921static enum bp_result set_pixel_clock_v3(
 922	struct bios_parser *bp,
 923	struct bp_pixel_clock_parameters *bp_params)
 924{
 925	enum bp_result result = BP_RESULT_FAILURE;
 926	PIXEL_CLOCK_PARAMETERS_V3 *params;
 927	SET_PIXEL_CLOCK_PS_ALLOCATION allocation;
 928
 929	memset(&allocation, 0, sizeof(allocation));
 930
 931	if (CLOCK_SOURCE_ID_PLL1 == bp_params->pll_id)
 932		allocation.sPCLKInput.ucPpll = ATOM_PPLL1;
 933	else if (CLOCK_SOURCE_ID_PLL2 == bp_params->pll_id)
 934		allocation.sPCLKInput.ucPpll = ATOM_PPLL2;
 935	else
 936		return BP_RESULT_BADINPUT;
 937
 938	allocation.sPCLKInput.usRefDiv =
 939			cpu_to_le16((uint16_t)bp_params->reference_divider);
 940	allocation.sPCLKInput.usFbDiv =
 941			cpu_to_le16((uint16_t)bp_params->feedback_divider);
 942	allocation.sPCLKInput.ucFracFbDiv =
 943			(uint8_t)bp_params->fractional_feedback_divider;
 944	allocation.sPCLKInput.ucPostDiv =
 945			(uint8_t)bp_params->pixel_clock_post_divider;
 946
 947	/* We need to convert from KHz units into 10KHz units */
 948	allocation.sPCLKInput.usPixelClock =
 949			cpu_to_le16((uint16_t)(bp_params->target_pixel_clock / 10));
 950
 951	params = (PIXEL_CLOCK_PARAMETERS_V3 *)&allocation.sPCLKInput;
 952	params->ucTransmitterId =
 953			bp->cmd_helper->encoder_id_to_atom(
 954					dal_graphics_object_id_get_encoder_id(
 955							bp_params->encoder_object_id));
 956	params->ucEncoderMode =
 957			(uint8_t)(bp->cmd_helper->encoder_mode_bp_to_atom(
 958					bp_params->signal_type, false));
 959
 960	if (bp_params->flags.FORCE_PROGRAMMING_OF_PLL)
 961		params->ucMiscInfo |= PIXEL_CLOCK_MISC_FORCE_PROG_PPLL;
 962
 963	if (bp_params->flags.USE_E_CLOCK_AS_SOURCE_FOR_D_CLOCK)
 964		params->ucMiscInfo |= PIXEL_CLOCK_MISC_USE_ENGINE_FOR_DISPCLK;
 965
 966	if (CONTROLLER_ID_D1 != bp_params->controller_id)
 967		params->ucMiscInfo |= PIXEL_CLOCK_MISC_CRTC_SEL_CRTC2;
 968
 969	if (EXEC_BIOS_CMD_TABLE(SetPixelClock, allocation))
 970		result = BP_RESULT_OK;
 971
 972	return result;
 973}
 974
 975#ifndef SET_PIXEL_CLOCK_PS_ALLOCATION_V5
 976/* video bios did not define this: */
 977typedef struct _SET_PIXEL_CLOCK_PS_ALLOCATION_V5 {
 978	PIXEL_CLOCK_PARAMETERS_V5 sPCLKInput;
 979	/* Caller doesn't need to init this portion */
 980	ENABLE_SPREAD_SPECTRUM_ON_PPLL sReserved;
 981} SET_PIXEL_CLOCK_PS_ALLOCATION_V5;
 982#endif
 983
 984#ifndef SET_PIXEL_CLOCK_PS_ALLOCATION_V6
 985/* video bios did not define this: */
 986typedef struct _SET_PIXEL_CLOCK_PS_ALLOCATION_V6 {
 987	PIXEL_CLOCK_PARAMETERS_V6 sPCLKInput;
 988	/* Caller doesn't need to init this portion */
 989	ENABLE_SPREAD_SPECTRUM_ON_PPLL sReserved;
 990} SET_PIXEL_CLOCK_PS_ALLOCATION_V6;
 991#endif
 992
 993static enum bp_result set_pixel_clock_v5(
 994	struct bios_parser *bp,
 995	struct bp_pixel_clock_parameters *bp_params)
 996{
 997	enum bp_result result = BP_RESULT_FAILURE;
 998	SET_PIXEL_CLOCK_PS_ALLOCATION_V5 clk;
 999	uint8_t controller_id;
1000	uint32_t pll_id;
1001
1002	memset(&clk, 0, sizeof(clk));
1003
1004	if (bp->cmd_helper->clock_source_id_to_atom(bp_params->pll_id, &pll_id)
1005			&& bp->cmd_helper->controller_id_to_atom(
1006					bp_params->controller_id, &controller_id)) {
1007		clk.sPCLKInput.ucCRTC = controller_id;
1008		clk.sPCLKInput.ucPpll = (uint8_t)pll_id;
1009		clk.sPCLKInput.ucRefDiv =
1010				(uint8_t)(bp_params->reference_divider);
1011		clk.sPCLKInput.usFbDiv =
1012				cpu_to_le16((uint16_t)(bp_params->feedback_divider));
1013		clk.sPCLKInput.ulFbDivDecFrac =
1014				cpu_to_le32(bp_params->fractional_feedback_divider);
1015		clk.sPCLKInput.ucPostDiv =
1016				(uint8_t)(bp_params->pixel_clock_post_divider);
1017		clk.sPCLKInput.ucTransmitterID =
1018				bp->cmd_helper->encoder_id_to_atom(
1019						dal_graphics_object_id_get_encoder_id(
1020								bp_params->encoder_object_id));
1021		clk.sPCLKInput.ucEncoderMode =
1022				(uint8_t)bp->cmd_helper->encoder_mode_bp_to_atom(
1023						bp_params->signal_type, false);
1024
1025		/* We need to convert from KHz units into 10KHz units */
1026		clk.sPCLKInput.usPixelClock =
1027				cpu_to_le16((uint16_t)(bp_params->target_pixel_clock / 10));
1028
1029		if (bp_params->flags.FORCE_PROGRAMMING_OF_PLL)
1030			clk.sPCLKInput.ucMiscInfo |=
1031					PIXEL_CLOCK_MISC_FORCE_PROG_PPLL;
1032
1033		if (bp_params->flags.SET_EXTERNAL_REF_DIV_SRC)
1034			clk.sPCLKInput.ucMiscInfo |=
1035					PIXEL_CLOCK_MISC_REF_DIV_SRC;
1036
1037		/* clkV5.ucMiscInfo bit[3:2]= HDMI panel bit depth: =0: 24bpp
1038		 * =1:30bpp, =2:32bpp
1039		 * driver choose program it itself, i.e. here we program it
1040		 * to 888 by default.
1041		 */
 
 
 
 
 
 
 
 
 
 
 
 
 
1042
1043		if (EXEC_BIOS_CMD_TABLE(SetPixelClock, clk))
1044			result = BP_RESULT_OK;
1045	}
1046
1047	return result;
1048}
1049
1050static enum bp_result set_pixel_clock_v6(
1051	struct bios_parser *bp,
1052	struct bp_pixel_clock_parameters *bp_params)
1053{
1054	enum bp_result result = BP_RESULT_FAILURE;
1055	SET_PIXEL_CLOCK_PS_ALLOCATION_V6 clk;
1056	uint8_t controller_id;
1057	uint32_t pll_id;
1058
1059	memset(&clk, 0, sizeof(clk));
1060
1061	if (bp->cmd_helper->clock_source_id_to_atom(bp_params->pll_id, &pll_id)
1062			&& bp->cmd_helper->controller_id_to_atom(
1063					bp_params->controller_id, &controller_id)) {
1064		/* Note: VBIOS still wants to use ucCRTC name which is now
1065		 * 1 byte in ULONG
1066		 *typedef struct _CRTC_PIXEL_CLOCK_FREQ
1067		 *{
1068		 * target the pixel clock to drive the CRTC timing.
1069		 * ULONG ulPixelClock:24;
1070		 * 0 means disable PPLL/DCPLL. Expanded to 24 bits comparing to
1071		 * previous version.
1072		 * ATOM_CRTC1~6, indicate the CRTC controller to
1073		 * ULONG ucCRTC:8;
1074		 * drive the pixel clock. not used for DCPLL case.
1075		 *}CRTC_PIXEL_CLOCK_FREQ;
1076		 *union
1077		 *{
1078		 * pixel clock and CRTC id frequency
1079		 * CRTC_PIXEL_CLOCK_FREQ ulCrtcPclkFreq;
1080		 * ULONG ulDispEngClkFreq; dispclk frequency
1081		 *};
1082		 */
1083		clk.sPCLKInput.ulCrtcPclkFreq.ucCRTC = controller_id;
1084		clk.sPCLKInput.ucPpll = (uint8_t) pll_id;
1085		clk.sPCLKInput.ucRefDiv =
1086				(uint8_t) bp_params->reference_divider;
1087		clk.sPCLKInput.usFbDiv =
1088				cpu_to_le16((uint16_t) bp_params->feedback_divider);
1089		clk.sPCLKInput.ulFbDivDecFrac =
1090				cpu_to_le32(bp_params->fractional_feedback_divider);
1091		clk.sPCLKInput.ucPostDiv =
1092				(uint8_t) bp_params->pixel_clock_post_divider;
1093		clk.sPCLKInput.ucTransmitterID =
1094				bp->cmd_helper->encoder_id_to_atom(
1095						dal_graphics_object_id_get_encoder_id(
1096								bp_params->encoder_object_id));
1097		clk.sPCLKInput.ucEncoderMode =
1098				(uint8_t) bp->cmd_helper->encoder_mode_bp_to_atom(
1099						bp_params->signal_type, false);
1100
1101		/* We need to convert from KHz units into 10KHz units */
1102		clk.sPCLKInput.ulCrtcPclkFreq.ulPixelClock =
1103				cpu_to_le32(bp_params->target_pixel_clock / 10);
1104
1105		if (bp_params->flags.FORCE_PROGRAMMING_OF_PLL) {
1106			clk.sPCLKInput.ucMiscInfo |=
1107					PIXEL_CLOCK_V6_MISC_FORCE_PROG_PPLL;
1108		}
1109
1110		if (bp_params->flags.SET_EXTERNAL_REF_DIV_SRC) {
1111			clk.sPCLKInput.ucMiscInfo |=
1112					PIXEL_CLOCK_V6_MISC_REF_DIV_SRC;
1113		}
1114
1115		/* clkV6.ucMiscInfo bit[3:2]= HDMI panel bit depth: =0:
1116		 * 24bpp =1:30bpp, =2:32bpp
1117		 * driver choose program it itself, i.e. here we pass required
1118		 * target rate that includes deep color.
1119		 */
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1120
1121		if (EXEC_BIOS_CMD_TABLE(SetPixelClock, clk))
1122			result = BP_RESULT_OK;
1123	}
1124
1125	return result;
1126}
1127
1128static enum bp_result set_pixel_clock_v7(
1129	struct bios_parser *bp,
1130	struct bp_pixel_clock_parameters *bp_params)
1131{
1132	enum bp_result result = BP_RESULT_FAILURE;
1133	PIXEL_CLOCK_PARAMETERS_V7 clk;
1134	uint8_t controller_id;
1135	uint32_t pll_id;
1136
1137	memset(&clk, 0, sizeof(clk));
1138
1139	if (bp->cmd_helper->clock_source_id_to_atom(bp_params->pll_id, &pll_id)
1140			&& bp->cmd_helper->controller_id_to_atom(bp_params->controller_id, &controller_id)) {
1141		/* Note: VBIOS still wants to use ucCRTC name which is now
1142		 * 1 byte in ULONG
1143		 *typedef struct _CRTC_PIXEL_CLOCK_FREQ
1144		 *{
1145		 * target the pixel clock to drive the CRTC timing.
1146		 * ULONG ulPixelClock:24;
1147		 * 0 means disable PPLL/DCPLL. Expanded to 24 bits comparing to
1148		 * previous version.
1149		 * ATOM_CRTC1~6, indicate the CRTC controller to
1150		 * ULONG ucCRTC:8;
1151		 * drive the pixel clock. not used for DCPLL case.
1152		 *}CRTC_PIXEL_CLOCK_FREQ;
1153		 *union
1154		 *{
1155		 * pixel clock and CRTC id frequency
1156		 * CRTC_PIXEL_CLOCK_FREQ ulCrtcPclkFreq;
1157		 * ULONG ulDispEngClkFreq; dispclk frequency
1158		 *};
1159		 */
1160		clk.ucCRTC = controller_id;
1161		clk.ucPpll = (uint8_t) pll_id;
1162		clk.ucTransmitterID = bp->cmd_helper->encoder_id_to_atom(dal_graphics_object_id_get_encoder_id(bp_params->encoder_object_id));
1163		clk.ucEncoderMode = (uint8_t) bp->cmd_helper->encoder_mode_bp_to_atom(bp_params->signal_type, false);
1164
1165		/* We need to convert from KHz units into 10KHz units */
1166		clk.ulPixelClock = cpu_to_le32(bp_params->target_pixel_clock * 10);
1167
1168		clk.ucDeepColorRatio = (uint8_t) bp->cmd_helper->transmitter_color_depth_to_atom(bp_params->color_depth);
1169
1170		if (bp_params->flags.FORCE_PROGRAMMING_OF_PLL)
1171			clk.ucMiscInfo |= PIXEL_CLOCK_V7_MISC_FORCE_PROG_PPLL;
1172
1173		if (bp_params->flags.SET_EXTERNAL_REF_DIV_SRC)
1174			clk.ucMiscInfo |= PIXEL_CLOCK_V7_MISC_REF_DIV_SRC;
1175
1176		if (bp_params->flags.PROGRAM_PHY_PLL_ONLY)
1177			clk.ucMiscInfo |= PIXEL_CLOCK_V7_MISC_PROG_PHYPLL;
1178
1179		if (bp_params->flags.SUPPORT_YUV_420)
1180			clk.ucMiscInfo |= PIXEL_CLOCK_V7_MISC_YUV420_MODE;
1181
1182		if (bp_params->flags.SET_XTALIN_REF_SRC)
1183			clk.ucMiscInfo |= PIXEL_CLOCK_V7_MISC_REF_DIV_SRC_XTALIN;
1184
1185		if (bp_params->flags.SET_GENLOCK_REF_DIV_SRC)
1186			clk.ucMiscInfo |= PIXEL_CLOCK_V7_MISC_REF_DIV_SRC_GENLK;
1187
1188		if (bp_params->signal_type == SIGNAL_TYPE_DVI_DUAL_LINK)
1189			clk.ucMiscInfo |= PIXEL_CLOCK_V7_MISC_DVI_DUALLINK_EN;
1190
1191		if (EXEC_BIOS_CMD_TABLE(SetPixelClock, clk))
1192			result = BP_RESULT_OK;
1193	}
1194	return result;
1195}
1196
1197/*******************************************************************************
1198 ********************************************************************************
1199 **
1200 **                  ENABLE PIXEL CLOCK SS
1201 **
1202 ********************************************************************************
1203 *******************************************************************************/
1204static enum bp_result enable_spread_spectrum_on_ppll_v1(
1205	struct bios_parser *bp,
1206	struct bp_spread_spectrum_parameters *bp_params,
1207	bool enable);
1208static enum bp_result enable_spread_spectrum_on_ppll_v2(
1209	struct bios_parser *bp,
1210	struct bp_spread_spectrum_parameters *bp_params,
1211	bool enable);
1212static enum bp_result enable_spread_spectrum_on_ppll_v3(
1213	struct bios_parser *bp,
1214	struct bp_spread_spectrum_parameters *bp_params,
1215	bool enable);
1216
1217static void init_enable_spread_spectrum_on_ppll(struct bios_parser *bp)
1218{
1219	switch (BIOS_CMD_TABLE_PARA_REVISION(EnableSpreadSpectrumOnPPLL)) {
1220	case 1:
1221		bp->cmd_tbl.enable_spread_spectrum_on_ppll =
1222				enable_spread_spectrum_on_ppll_v1;
1223		break;
1224	case 2:
1225		bp->cmd_tbl.enable_spread_spectrum_on_ppll =
1226				enable_spread_spectrum_on_ppll_v2;
1227		break;
1228	case 3:
1229		bp->cmd_tbl.enable_spread_spectrum_on_ppll =
1230				enable_spread_spectrum_on_ppll_v3;
1231		break;
1232	default:
1233		dm_output_to_console("Don't have enable_spread_spectrum_on_ppll for v%d\n",
1234			 BIOS_CMD_TABLE_PARA_REVISION(EnableSpreadSpectrumOnPPLL));
1235		bp->cmd_tbl.enable_spread_spectrum_on_ppll = NULL;
1236		break;
1237	}
1238}
1239
1240static enum bp_result enable_spread_spectrum_on_ppll_v1(
1241	struct bios_parser *bp,
1242	struct bp_spread_spectrum_parameters *bp_params,
1243	bool enable)
1244{
1245	enum bp_result result = BP_RESULT_FAILURE;
1246	ENABLE_SPREAD_SPECTRUM_ON_PPLL params;
1247
1248	memset(&params, 0, sizeof(params));
1249
1250	if ((enable == true) && (bp_params->percentage > 0))
1251		params.ucEnable = ATOM_ENABLE;
1252	else
1253		params.ucEnable = ATOM_DISABLE;
1254
1255	params.usSpreadSpectrumPercentage =
1256			cpu_to_le16((uint16_t)bp_params->percentage);
1257	params.ucSpreadSpectrumStep =
1258			(uint8_t)bp_params->ver1.step;
1259	params.ucSpreadSpectrumDelay =
1260			(uint8_t)bp_params->ver1.delay;
1261	/* convert back to unit of 10KHz */
1262	params.ucSpreadSpectrumRange =
1263			(uint8_t)(bp_params->ver1.range / 10000);
1264
1265	if (bp_params->flags.EXTERNAL_SS)
1266		params.ucSpreadSpectrumType |= ATOM_EXTERNAL_SS_MASK;
1267
1268	if (bp_params->flags.CENTER_SPREAD)
1269		params.ucSpreadSpectrumType |= ATOM_SS_CENTRE_SPREAD_MODE;
1270
1271	if (bp_params->pll_id == CLOCK_SOURCE_ID_PLL1)
1272		params.ucPpll = ATOM_PPLL1;
1273	else if (bp_params->pll_id == CLOCK_SOURCE_ID_PLL2)
1274		params.ucPpll = ATOM_PPLL2;
1275	else
1276		BREAK_TO_DEBUGGER(); /* Unexpected PLL value!! */
1277
1278	if (EXEC_BIOS_CMD_TABLE(EnableSpreadSpectrumOnPPLL, params))
1279		result = BP_RESULT_OK;
1280
1281	return result;
1282}
1283
1284static enum bp_result enable_spread_spectrum_on_ppll_v2(
1285	struct bios_parser *bp,
1286	struct bp_spread_spectrum_parameters *bp_params,
1287	bool enable)
1288{
1289	enum bp_result result = BP_RESULT_FAILURE;
1290	ENABLE_SPREAD_SPECTRUM_ON_PPLL_V2 params;
1291
1292	memset(&params, 0, sizeof(params));
1293
1294	if (bp_params->pll_id == CLOCK_SOURCE_ID_PLL1)
1295		params.ucSpreadSpectrumType = ATOM_PPLL_SS_TYPE_V2_P1PLL;
1296	else if (bp_params->pll_id == CLOCK_SOURCE_ID_PLL2)
1297		params.ucSpreadSpectrumType = ATOM_PPLL_SS_TYPE_V2_P2PLL;
1298	else
1299		BREAK_TO_DEBUGGER(); /* Unexpected PLL value!! */
1300
1301	if ((enable == true) && (bp_params->percentage > 0)) {
1302		params.ucEnable = ATOM_ENABLE;
1303
1304		params.usSpreadSpectrumPercentage =
1305				cpu_to_le16((uint16_t)(bp_params->percentage));
1306		params.usSpreadSpectrumStep =
1307				cpu_to_le16((uint16_t)(bp_params->ds.ds_frac_size));
1308
1309		if (bp_params->flags.EXTERNAL_SS)
1310			params.ucSpreadSpectrumType |=
1311					ATOM_PPLL_SS_TYPE_V2_EXT_SPREAD;
1312
1313		if (bp_params->flags.CENTER_SPREAD)
1314			params.ucSpreadSpectrumType |=
1315					ATOM_PPLL_SS_TYPE_V2_CENTRE_SPREAD;
1316
1317		/* Both amounts need to be left shifted first before bit
1318		 * comparison. Otherwise, the result will always be zero here
1319		 */
1320		params.usSpreadSpectrumAmount = cpu_to_le16((uint16_t)(
1321				((bp_params->ds.feedback_amount <<
1322						ATOM_PPLL_SS_AMOUNT_V2_FBDIV_SHIFT) &
1323						ATOM_PPLL_SS_AMOUNT_V2_FBDIV_MASK) |
1324						((bp_params->ds.nfrac_amount <<
1325								ATOM_PPLL_SS_AMOUNT_V2_NFRAC_SHIFT) &
1326								ATOM_PPLL_SS_AMOUNT_V2_NFRAC_MASK)));
1327	} else
1328		params.ucEnable = ATOM_DISABLE;
1329
1330	if (EXEC_BIOS_CMD_TABLE(EnableSpreadSpectrumOnPPLL, params))
1331		result = BP_RESULT_OK;
1332
1333	return result;
1334}
1335
1336static enum bp_result enable_spread_spectrum_on_ppll_v3(
1337	struct bios_parser *bp,
1338	struct bp_spread_spectrum_parameters *bp_params,
1339	bool enable)
1340{
1341	enum bp_result result = BP_RESULT_FAILURE;
1342	ENABLE_SPREAD_SPECTRUM_ON_PPLL_V3 params;
1343
1344	memset(&params, 0, sizeof(params));
1345
1346	switch (bp_params->pll_id) {
1347	case CLOCK_SOURCE_ID_PLL0:
1348		/* ATOM_PPLL_SS_TYPE_V3_P0PLL; this is pixel clock only,
1349		 * not for SI display clock.
1350		 */
1351		params.ucSpreadSpectrumType = ATOM_PPLL_SS_TYPE_V3_DCPLL;
1352		break;
1353	case CLOCK_SOURCE_ID_PLL1:
1354		params.ucSpreadSpectrumType = ATOM_PPLL_SS_TYPE_V3_P1PLL;
1355		break;
1356
1357	case CLOCK_SOURCE_ID_PLL2:
1358		params.ucSpreadSpectrumType = ATOM_PPLL_SS_TYPE_V3_P2PLL;
1359		break;
1360
1361	case CLOCK_SOURCE_ID_DCPLL:
1362		params.ucSpreadSpectrumType = ATOM_PPLL_SS_TYPE_V3_DCPLL;
1363		break;
1364
1365	default:
1366		BREAK_TO_DEBUGGER();
1367		/* Unexpected PLL value!! */
1368		return result;
1369	}
1370
1371	if (enable == true) {
1372		params.ucEnable = ATOM_ENABLE;
1373
1374		params.usSpreadSpectrumAmountFrac =
1375				cpu_to_le16((uint16_t)(bp_params->ds_frac_amount));
1376		params.usSpreadSpectrumStep =
1377				cpu_to_le16((uint16_t)(bp_params->ds.ds_frac_size));
1378
1379		if (bp_params->flags.EXTERNAL_SS)
1380			params.ucSpreadSpectrumType |=
1381					ATOM_PPLL_SS_TYPE_V3_EXT_SPREAD;
1382		if (bp_params->flags.CENTER_SPREAD)
1383			params.ucSpreadSpectrumType |=
1384					ATOM_PPLL_SS_TYPE_V3_CENTRE_SPREAD;
1385
1386		/* Both amounts need to be left shifted first before bit
1387		 * comparison. Otherwise, the result will always be zero here
1388		 */
1389		params.usSpreadSpectrumAmount = cpu_to_le16((uint16_t)(
1390				((bp_params->ds.feedback_amount <<
1391						ATOM_PPLL_SS_AMOUNT_V3_FBDIV_SHIFT) &
1392						ATOM_PPLL_SS_AMOUNT_V3_FBDIV_MASK) |
1393						((bp_params->ds.nfrac_amount <<
1394								ATOM_PPLL_SS_AMOUNT_V3_NFRAC_SHIFT) &
1395								ATOM_PPLL_SS_AMOUNT_V3_NFRAC_MASK)));
1396	} else
1397		params.ucEnable = ATOM_DISABLE;
1398
1399	if (EXEC_BIOS_CMD_TABLE(EnableSpreadSpectrumOnPPLL, params))
1400		result = BP_RESULT_OK;
1401
1402	return result;
1403}
1404
1405/*******************************************************************************
1406 ********************************************************************************
1407 **
1408 **                  ADJUST DISPLAY PLL
1409 **
1410 ********************************************************************************
1411 *******************************************************************************/
1412
1413static enum bp_result adjust_display_pll_v2(
1414	struct bios_parser *bp,
1415	struct bp_adjust_pixel_clock_parameters *bp_params);
1416static enum bp_result adjust_display_pll_v3(
1417	struct bios_parser *bp,
1418	struct bp_adjust_pixel_clock_parameters *bp_params);
1419
1420static void init_adjust_display_pll(struct bios_parser *bp)
1421{
1422	switch (BIOS_CMD_TABLE_PARA_REVISION(AdjustDisplayPll)) {
1423	case 2:
1424		bp->cmd_tbl.adjust_display_pll = adjust_display_pll_v2;
1425		break;
1426	case 3:
1427		bp->cmd_tbl.adjust_display_pll = adjust_display_pll_v3;
1428		break;
1429	default:
1430		dm_output_to_console("Don't have adjust_display_pll for v%d\n",
1431			 BIOS_CMD_TABLE_PARA_REVISION(AdjustDisplayPll));
1432		bp->cmd_tbl.adjust_display_pll = NULL;
1433		break;
1434	}
1435}
1436
1437static enum bp_result adjust_display_pll_v2(
1438	struct bios_parser *bp,
1439	struct bp_adjust_pixel_clock_parameters *bp_params)
1440{
1441	enum bp_result result = BP_RESULT_FAILURE;
1442	ADJUST_DISPLAY_PLL_PS_ALLOCATION params = { 0 };
1443
1444	/* We need to convert from KHz units into 10KHz units and then convert
1445	 * output pixel clock back 10KHz-->KHz */
1446	uint32_t pixel_clock_10KHz_in = bp_params->pixel_clock / 10;
1447
1448	params.usPixelClock = cpu_to_le16((uint16_t)(pixel_clock_10KHz_in));
1449	params.ucTransmitterID =
1450			bp->cmd_helper->encoder_id_to_atom(
1451					dal_graphics_object_id_get_encoder_id(
1452							bp_params->encoder_object_id));
1453	params.ucEncodeMode =
1454			(uint8_t)bp->cmd_helper->encoder_mode_bp_to_atom(
1455					bp_params->signal_type, false);
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1456	return result;
1457}
1458
1459static enum bp_result adjust_display_pll_v3(
1460	struct bios_parser *bp,
1461	struct bp_adjust_pixel_clock_parameters *bp_params)
1462{
1463	enum bp_result result = BP_RESULT_FAILURE;
1464	ADJUST_DISPLAY_PLL_PS_ALLOCATION_V3 params;
1465	uint32_t pixel_clk_10_kHz_in = bp_params->pixel_clock / 10;
1466
1467	memset(&params, 0, sizeof(params));
1468
1469	/* We need to convert from KHz units into 10KHz units and then convert
1470	 * output pixel clock back 10KHz-->KHz */
1471	params.sInput.usPixelClock = cpu_to_le16((uint16_t)pixel_clk_10_kHz_in);
1472	params.sInput.ucTransmitterID =
1473			bp->cmd_helper->encoder_id_to_atom(
1474					dal_graphics_object_id_get_encoder_id(
1475							bp_params->encoder_object_id));
1476	params.sInput.ucEncodeMode =
1477			(uint8_t)bp->cmd_helper->encoder_mode_bp_to_atom(
1478					bp_params->signal_type, false);
1479
1480	if (bp_params->ss_enable == true)
1481		params.sInput.ucDispPllConfig |= DISPPLL_CONFIG_SS_ENABLE;
1482
1483	if (bp_params->signal_type == SIGNAL_TYPE_DVI_DUAL_LINK)
1484		params.sInput.ucDispPllConfig |= DISPPLL_CONFIG_DUAL_LINK;
1485
1486	if (EXEC_BIOS_CMD_TABLE(AdjustDisplayPll, params)) {
1487		/* Convert output pixel clock back 10KHz-->KHz: multiply
1488		 * original pixel clock in KHz by ratio
1489		 * [output pxlClk/input pxlClk] */
1490		uint64_t pixel_clk_10_khz_out =
1491				(uint64_t)le32_to_cpu(params.sOutput.ulDispPllFreq);
1492		uint64_t pixel_clk = (uint64_t)bp_params->pixel_clock;
1493
1494		if (pixel_clk_10_kHz_in != 0) {
1495			bp_params->adjusted_pixel_clock =
1496					div_u64(pixel_clk * pixel_clk_10_khz_out,
1497							pixel_clk_10_kHz_in);
1498		} else {
1499			bp_params->adjusted_pixel_clock = 0;
1500			BREAK_TO_DEBUGGER();
1501		}
1502
1503		bp_params->reference_divider = params.sOutput.ucRefDiv;
1504		bp_params->pixel_clock_post_divider = params.sOutput.ucPostDiv;
1505
1506		result = BP_RESULT_OK;
1507	}
1508
1509	return result;
1510}
1511
1512/*******************************************************************************
1513 ********************************************************************************
1514 **
1515 **                  DAC ENCODER CONTROL
1516 **
1517 ********************************************************************************
1518 *******************************************************************************/
1519
1520static enum bp_result dac1_encoder_control_v1(
1521	struct bios_parser *bp,
1522	bool enable,
1523	uint32_t pixel_clock,
1524	uint8_t dac_standard);
1525static enum bp_result dac2_encoder_control_v1(
1526	struct bios_parser *bp,
1527	bool enable,
1528	uint32_t pixel_clock,
1529	uint8_t dac_standard);
1530
1531static void init_dac_encoder_control(struct bios_parser *bp)
1532{
1533	switch (BIOS_CMD_TABLE_PARA_REVISION(DAC1EncoderControl)) {
1534	case 1:
1535		bp->cmd_tbl.dac1_encoder_control = dac1_encoder_control_v1;
1536		break;
1537	default:
1538		bp->cmd_tbl.dac1_encoder_control = NULL;
1539		break;
1540	}
1541	switch (BIOS_CMD_TABLE_PARA_REVISION(DAC2EncoderControl)) {
1542	case 1:
1543		bp->cmd_tbl.dac2_encoder_control = dac2_encoder_control_v1;
1544		break;
1545	default:
1546		bp->cmd_tbl.dac2_encoder_control = NULL;
1547		break;
1548	}
1549}
1550
1551static void dac_encoder_control_prepare_params(
1552	DAC_ENCODER_CONTROL_PS_ALLOCATION *params,
1553	bool enable,
1554	uint32_t pixel_clock,
1555	uint8_t dac_standard)
1556{
1557	params->ucDacStandard = dac_standard;
1558	if (enable)
1559		params->ucAction = ATOM_ENABLE;
1560	else
1561		params->ucAction = ATOM_DISABLE;
1562
1563	/* We need to convert from KHz units into 10KHz units
1564	 * it looks as if the TvControl do not care about pixel clock
1565	 */
1566	params->usPixelClock = cpu_to_le16((uint16_t)(pixel_clock / 10));
1567}
1568
1569static enum bp_result dac1_encoder_control_v1(
1570	struct bios_parser *bp,
1571	bool enable,
1572	uint32_t pixel_clock,
1573	uint8_t dac_standard)
1574{
1575	enum bp_result result = BP_RESULT_FAILURE;
1576	DAC_ENCODER_CONTROL_PS_ALLOCATION params;
1577
1578	dac_encoder_control_prepare_params(
1579		&params,
1580		enable,
1581		pixel_clock,
1582		dac_standard);
1583
1584	if (EXEC_BIOS_CMD_TABLE(DAC1EncoderControl, params))
1585		result = BP_RESULT_OK;
1586
1587	return result;
1588}
1589
1590static enum bp_result dac2_encoder_control_v1(
1591	struct bios_parser *bp,
1592	bool enable,
1593	uint32_t pixel_clock,
1594	uint8_t dac_standard)
1595{
1596	enum bp_result result = BP_RESULT_FAILURE;
1597	DAC_ENCODER_CONTROL_PS_ALLOCATION params;
1598
1599	dac_encoder_control_prepare_params(
1600		&params,
1601		enable,
1602		pixel_clock,
1603		dac_standard);
1604
1605	if (EXEC_BIOS_CMD_TABLE(DAC2EncoderControl, params))
1606		result = BP_RESULT_OK;
1607
1608	return result;
1609}
1610
1611/*******************************************************************************
1612 ********************************************************************************
1613 **
1614 **                  DAC OUTPUT CONTROL
1615 **
1616 ********************************************************************************
1617 *******************************************************************************/
1618static enum bp_result dac1_output_control_v1(
1619	struct bios_parser *bp,
1620	bool enable);
1621static enum bp_result dac2_output_control_v1(
1622	struct bios_parser *bp,
1623	bool enable);
1624
1625static void init_dac_output_control(struct bios_parser *bp)
1626{
1627	switch (BIOS_CMD_TABLE_PARA_REVISION(DAC1OutputControl)) {
1628	case 1:
1629		bp->cmd_tbl.dac1_output_control = dac1_output_control_v1;
1630		break;
1631	default:
1632		bp->cmd_tbl.dac1_output_control = NULL;
1633		break;
1634	}
1635	switch (BIOS_CMD_TABLE_PARA_REVISION(DAC2OutputControl)) {
1636	case 1:
1637		bp->cmd_tbl.dac2_output_control = dac2_output_control_v1;
1638		break;
1639	default:
1640		bp->cmd_tbl.dac2_output_control = NULL;
1641		break;
1642	}
1643}
1644
1645static enum bp_result dac1_output_control_v1(
1646	struct bios_parser *bp, bool enable)
1647{
1648	enum bp_result result = BP_RESULT_FAILURE;
1649	DISPLAY_DEVICE_OUTPUT_CONTROL_PS_ALLOCATION params;
1650
1651	if (enable)
1652		params.ucAction = ATOM_ENABLE;
1653	else
1654		params.ucAction = ATOM_DISABLE;
1655
1656	if (EXEC_BIOS_CMD_TABLE(DAC1OutputControl, params))
1657		result = BP_RESULT_OK;
1658
1659	return result;
1660}
1661
1662static enum bp_result dac2_output_control_v1(
1663	struct bios_parser *bp, bool enable)
1664{
1665	enum bp_result result = BP_RESULT_FAILURE;
1666	DISPLAY_DEVICE_OUTPUT_CONTROL_PS_ALLOCATION params;
1667
1668	if (enable)
1669		params.ucAction = ATOM_ENABLE;
1670	else
1671		params.ucAction = ATOM_DISABLE;
1672
1673	if (EXEC_BIOS_CMD_TABLE(DAC2OutputControl, params))
1674		result = BP_RESULT_OK;
1675
1676	return result;
1677}
1678
1679/*******************************************************************************
1680 ********************************************************************************
1681 **
1682 **                  SET CRTC TIMING
1683 **
1684 ********************************************************************************
1685 *******************************************************************************/
1686
1687static enum bp_result set_crtc_using_dtd_timing_v3(
1688	struct bios_parser *bp,
1689	struct bp_hw_crtc_timing_parameters *bp_params);
1690static enum bp_result set_crtc_timing_v1(
1691	struct bios_parser *bp,
1692	struct bp_hw_crtc_timing_parameters *bp_params);
1693
1694static void init_set_crtc_timing(struct bios_parser *bp)
1695{
1696	uint32_t dtd_version =
1697			BIOS_CMD_TABLE_PARA_REVISION(SetCRTC_UsingDTDTiming);
1698	if (dtd_version > 2)
1699		switch (dtd_version) {
1700		case 3:
1701			bp->cmd_tbl.set_crtc_timing =
1702					set_crtc_using_dtd_timing_v3;
1703			break;
1704		default:
1705			dm_output_to_console("Don't have set_crtc_timing for dtd v%d\n",
1706				 dtd_version);
1707			bp->cmd_tbl.set_crtc_timing = NULL;
1708			break;
1709		}
1710	else
1711		switch (BIOS_CMD_TABLE_PARA_REVISION(SetCRTC_Timing)) {
1712		case 1:
1713			bp->cmd_tbl.set_crtc_timing = set_crtc_timing_v1;
1714			break;
1715		default:
1716			dm_output_to_console("Don't have set_crtc_timing for v%d\n",
1717				 BIOS_CMD_TABLE_PARA_REVISION(SetCRTC_Timing));
1718			bp->cmd_tbl.set_crtc_timing = NULL;
1719			break;
1720		}
1721}
1722
1723static enum bp_result set_crtc_timing_v1(
1724	struct bios_parser *bp,
1725	struct bp_hw_crtc_timing_parameters *bp_params)
1726{
1727	enum bp_result result = BP_RESULT_FAILURE;
1728	SET_CRTC_TIMING_PARAMETERS_PS_ALLOCATION params = {0};
1729	uint8_t atom_controller_id;
1730
1731	if (bp->cmd_helper->controller_id_to_atom(
1732			bp_params->controller_id, &atom_controller_id))
1733		params.ucCRTC = atom_controller_id;
1734
1735	params.usH_Total = cpu_to_le16((uint16_t)(bp_params->h_total));
1736	params.usH_Disp = cpu_to_le16((uint16_t)(bp_params->h_addressable));
1737	params.usH_SyncStart = cpu_to_le16((uint16_t)(bp_params->h_sync_start));
1738	params.usH_SyncWidth = cpu_to_le16((uint16_t)(bp_params->h_sync_width));
1739	params.usV_Total = cpu_to_le16((uint16_t)(bp_params->v_total));
1740	params.usV_Disp = cpu_to_le16((uint16_t)(bp_params->v_addressable));
1741	params.usV_SyncStart =
1742			cpu_to_le16((uint16_t)(bp_params->v_sync_start));
1743	params.usV_SyncWidth =
1744			cpu_to_le16((uint16_t)(bp_params->v_sync_width));
1745
1746	/* VBIOS does not expect any value except zero into this call, for
1747	 * underscan use another entry ProgramOverscan call but when mode
1748	 * 1776x1000 with the overscan 72x44 .e.i. 1920x1080 @30 DAL2 is ok,
1749	 * but when same ,but 60 Hz there is corruption
1750	 * DAL1 does not allow the mode 1776x1000@60
1751	 */
1752	params.ucOverscanRight = (uint8_t)bp_params->h_overscan_right;
1753	params.ucOverscanLeft = (uint8_t)bp_params->h_overscan_left;
1754	params.ucOverscanBottom = (uint8_t)bp_params->v_overscan_bottom;
1755	params.ucOverscanTop = (uint8_t)bp_params->v_overscan_top;
1756
1757	if (0 == bp_params->flags.HSYNC_POSITIVE_POLARITY)
1758		params.susModeMiscInfo.usAccess =
1759				cpu_to_le16(le16_to_cpu(params.susModeMiscInfo.usAccess) | ATOM_HSYNC_POLARITY);
1760
1761	if (0 == bp_params->flags.VSYNC_POSITIVE_POLARITY)
1762		params.susModeMiscInfo.usAccess =
1763				cpu_to_le16(le16_to_cpu(params.susModeMiscInfo.usAccess) | ATOM_VSYNC_POLARITY);
1764
1765	if (bp_params->flags.INTERLACE) {
1766		params.susModeMiscInfo.usAccess =
1767				cpu_to_le16(le16_to_cpu(params.susModeMiscInfo.usAccess) | ATOM_INTERLACE);
1768
1769		/* original DAL code has this condition to apply tis for
1770		 * non-TV/CV only due to complex MV testing for possible
1771		 * impact
1772		 * if (pACParameters->signal != SignalType_YPbPr &&
1773		 *  pACParameters->signal != SignalType_Composite &&
1774		 *  pACParameters->signal != SignalType_SVideo)
1775		 */
1776		/* HW will deduct 0.5 line from 2nd feild.
1777		 * i.e. for 1080i, it is 2 lines for 1st field, 2.5
1778		 * lines for the 2nd feild. we need input as 5 instead
1779		 * of 4, but it is 4 either from Edid data
1780		 * (spec CEA 861) or CEA timing table.
1781		 */
1782		params.usV_SyncStart =
1783				cpu_to_le16((uint16_t)(bp_params->v_sync_start + 1));
1784	}
1785
1786	if (bp_params->flags.HORZ_COUNT_BY_TWO)
1787		params.susModeMiscInfo.usAccess =
1788				cpu_to_le16(le16_to_cpu(params.susModeMiscInfo.usAccess) | ATOM_DOUBLE_CLOCK_MODE);
1789
1790	if (EXEC_BIOS_CMD_TABLE(SetCRTC_Timing, params))
1791		result = BP_RESULT_OK;
1792
1793	return result;
1794}
1795
1796static enum bp_result set_crtc_using_dtd_timing_v3(
1797	struct bios_parser *bp,
1798	struct bp_hw_crtc_timing_parameters *bp_params)
1799{
1800	enum bp_result result = BP_RESULT_FAILURE;
1801	SET_CRTC_USING_DTD_TIMING_PARAMETERS params = {0};
1802	uint8_t atom_controller_id;
1803
1804	if (bp->cmd_helper->controller_id_to_atom(
1805			bp_params->controller_id, &atom_controller_id))
1806		params.ucCRTC = atom_controller_id;
1807
1808	/* bios usH_Size wants h addressable size */
1809	params.usH_Size = cpu_to_le16((uint16_t)bp_params->h_addressable);
1810	/* bios usH_Blanking_Time wants borders included in blanking */
1811	params.usH_Blanking_Time =
1812			cpu_to_le16((uint16_t)(bp_params->h_total - bp_params->h_addressable));
1813	/* bios usV_Size wants v addressable size */
1814	params.usV_Size = cpu_to_le16((uint16_t)bp_params->v_addressable);
1815	/* bios usV_Blanking_Time wants borders included in blanking */
1816	params.usV_Blanking_Time =
1817			cpu_to_le16((uint16_t)(bp_params->v_total - bp_params->v_addressable));
1818	/* bios usHSyncOffset is the offset from the end of h addressable,
1819	 * our horizontalSyncStart is the offset from the beginning
1820	 * of h addressable */
1821	params.usH_SyncOffset =
1822			cpu_to_le16((uint16_t)(bp_params->h_sync_start - bp_params->h_addressable));
1823	params.usH_SyncWidth = cpu_to_le16((uint16_t)bp_params->h_sync_width);
1824	/* bios usHSyncOffset is the offset from the end of v addressable,
1825	 * our verticalSyncStart is the offset from the beginning of
1826	 * v addressable */
1827	params.usV_SyncOffset =
1828			cpu_to_le16((uint16_t)(bp_params->v_sync_start - bp_params->v_addressable));
1829	params.usV_SyncWidth = cpu_to_le16((uint16_t)bp_params->v_sync_width);
1830
1831	/* we assume that overscan from original timing does not get bigger
1832	 * than 255
1833	 * we will program all the borders in the Set CRTC Overscan call below
1834	 */
1835
1836	if (0 == bp_params->flags.HSYNC_POSITIVE_POLARITY)
1837		params.susModeMiscInfo.usAccess =
1838				cpu_to_le16(le16_to_cpu(params.susModeMiscInfo.usAccess) | ATOM_HSYNC_POLARITY);
1839
1840	if (0 == bp_params->flags.VSYNC_POSITIVE_POLARITY)
1841		params.susModeMiscInfo.usAccess =
1842				cpu_to_le16(le16_to_cpu(params.susModeMiscInfo.usAccess) | ATOM_VSYNC_POLARITY);
1843
1844	if (bp_params->flags.INTERLACE)	{
1845		params.susModeMiscInfo.usAccess =
1846				cpu_to_le16(le16_to_cpu(params.susModeMiscInfo.usAccess) | ATOM_INTERLACE);
1847
1848		/* original DAL code has this condition to apply this
1849		 * for non-TV/CV only
1850		 * due to complex MV testing for possible impact
1851		 * if ( pACParameters->signal != SignalType_YPbPr &&
1852		 *  pACParameters->signal != SignalType_Composite &&
1853		 *  pACParameters->signal != SignalType_SVideo)
1854		 */
1855		{
1856			/* HW will deduct 0.5 line from 2nd feild.
1857			 * i.e. for 1080i, it is 2 lines for 1st field,
1858			 * 2.5 lines for the 2nd feild. we need input as 5
1859			 * instead of 4.
1860			 * but it is 4 either from Edid data (spec CEA 861)
1861			 * or CEA timing table.
1862			 */
1863			params.usV_SyncOffset =
1864					cpu_to_le16(le16_to_cpu(params.usV_SyncOffset) + 1);
1865
1866		}
1867	}
1868
1869	if (bp_params->flags.HORZ_COUNT_BY_TWO)
1870		params.susModeMiscInfo.usAccess =
1871				cpu_to_le16(le16_to_cpu(params.susModeMiscInfo.usAccess) | ATOM_DOUBLE_CLOCK_MODE);
1872
1873	if (EXEC_BIOS_CMD_TABLE(SetCRTC_UsingDTDTiming, params))
1874		result = BP_RESULT_OK;
1875
1876	return result;
1877}
1878
1879/*******************************************************************************
1880 ********************************************************************************
1881 **
1882 **                  SELECT CRTC SOURCE
1883 **
1884 ********************************************************************************
1885 *******************************************************************************/
1886
1887static enum bp_result select_crtc_source_v2(
1888	struct bios_parser *bp,
1889	struct bp_crtc_source_select *bp_params);
1890static enum bp_result select_crtc_source_v3(
1891	struct bios_parser *bp,
1892	struct bp_crtc_source_select *bp_params);
1893
1894static void init_select_crtc_source(struct bios_parser *bp)
1895{
1896	switch (BIOS_CMD_TABLE_PARA_REVISION(SelectCRTC_Source)) {
1897	case 2:
1898		bp->cmd_tbl.select_crtc_source = select_crtc_source_v2;
1899		break;
1900	case 3:
1901		bp->cmd_tbl.select_crtc_source = select_crtc_source_v3;
1902		break;
1903	default:
1904		dm_output_to_console("Don't select_crtc_source enable_crtc for v%d\n",
1905			 BIOS_CMD_TABLE_PARA_REVISION(SelectCRTC_Source));
1906		bp->cmd_tbl.select_crtc_source = NULL;
1907		break;
1908	}
1909}
1910
1911static enum bp_result select_crtc_source_v2(
1912	struct bios_parser *bp,
1913	struct bp_crtc_source_select *bp_params)
1914{
1915	enum bp_result result = BP_RESULT_FAILURE;
1916	SELECT_CRTC_SOURCE_PARAMETERS_V2 params;
1917	uint8_t atom_controller_id;
1918	uint32_t atom_engine_id;
1919	enum signal_type s = bp_params->signal;
1920
1921	memset(&params, 0, sizeof(params));
1922
1923	/* set controller id */
1924	if (bp->cmd_helper->controller_id_to_atom(
1925			bp_params->controller_id, &atom_controller_id))
1926		params.ucCRTC = atom_controller_id;
1927	else
1928		return BP_RESULT_FAILURE;
1929
1930	/* set encoder id */
1931	if (bp->cmd_helper->engine_bp_to_atom(
1932			bp_params->engine_id, &atom_engine_id))
1933		params.ucEncoderID = (uint8_t)atom_engine_id;
1934	else
1935		return BP_RESULT_FAILURE;
1936
1937	if (SIGNAL_TYPE_EDP == s ||
1938			(SIGNAL_TYPE_DISPLAY_PORT == s &&
1939					SIGNAL_TYPE_LVDS == bp_params->sink_signal))
1940		s = SIGNAL_TYPE_LVDS;
1941
1942	params.ucEncodeMode =
1943			(uint8_t)bp->cmd_helper->encoder_mode_bp_to_atom(
1944					s, bp_params->enable_dp_audio);
1945
1946	if (EXEC_BIOS_CMD_TABLE(SelectCRTC_Source, params))
1947		result = BP_RESULT_OK;
1948
1949	return result;
1950}
1951
1952static enum bp_result select_crtc_source_v3(
1953	struct bios_parser *bp,
1954	struct bp_crtc_source_select *bp_params)
1955{
1956	bool result = BP_RESULT_FAILURE;
1957	SELECT_CRTC_SOURCE_PARAMETERS_V3 params;
1958	uint8_t atom_controller_id;
1959	uint32_t atom_engine_id;
1960	enum signal_type s = bp_params->signal;
1961
1962	memset(&params, 0, sizeof(params));
1963
1964	if (bp->cmd_helper->controller_id_to_atom(bp_params->controller_id,
1965			&atom_controller_id))
1966		params.ucCRTC = atom_controller_id;
1967	else
1968		return result;
1969
1970	if (bp->cmd_helper->engine_bp_to_atom(bp_params->engine_id,
1971			&atom_engine_id))
1972		params.ucEncoderID = (uint8_t)atom_engine_id;
1973	else
1974		return result;
1975
1976	if (SIGNAL_TYPE_EDP == s ||
1977			(SIGNAL_TYPE_DISPLAY_PORT == s &&
1978					SIGNAL_TYPE_LVDS == bp_params->sink_signal))
1979		s = SIGNAL_TYPE_LVDS;
1980
1981	params.ucEncodeMode =
1982			bp->cmd_helper->encoder_mode_bp_to_atom(
1983					s, bp_params->enable_dp_audio);
1984	/* Needed for VBIOS Random Spatial Dithering feature */
1985	params.ucDstBpc = (uint8_t)(bp_params->display_output_bit_depth);
1986
1987	if (EXEC_BIOS_CMD_TABLE(SelectCRTC_Source, params))
1988		result = BP_RESULT_OK;
1989
1990	return result;
1991}
1992
1993/*******************************************************************************
1994 ********************************************************************************
1995 **
1996 **                  ENABLE CRTC
1997 **
1998 ********************************************************************************
1999 *******************************************************************************/
2000
2001static enum bp_result enable_crtc_v1(
2002	struct bios_parser *bp,
2003	enum controller_id controller_id,
2004	bool enable);
2005
2006static void init_enable_crtc(struct bios_parser *bp)
2007{
2008	switch (BIOS_CMD_TABLE_PARA_REVISION(EnableCRTC)) {
2009	case 1:
2010		bp->cmd_tbl.enable_crtc = enable_crtc_v1;
2011		break;
2012	default:
2013		dm_output_to_console("Don't have enable_crtc for v%d\n",
2014			 BIOS_CMD_TABLE_PARA_REVISION(EnableCRTC));
2015		bp->cmd_tbl.enable_crtc = NULL;
2016		break;
2017	}
2018}
2019
2020static enum bp_result enable_crtc_v1(
2021	struct bios_parser *bp,
2022	enum controller_id controller_id,
2023	bool enable)
2024{
2025	bool result = BP_RESULT_FAILURE;
2026	ENABLE_CRTC_PARAMETERS params = {0};
2027	uint8_t id;
2028
2029	if (bp->cmd_helper->controller_id_to_atom(controller_id, &id))
2030		params.ucCRTC = id;
2031	else
2032		return BP_RESULT_BADINPUT;
2033
2034	if (enable)
2035		params.ucEnable = ATOM_ENABLE;
2036	else
2037		params.ucEnable = ATOM_DISABLE;
2038
2039	if (EXEC_BIOS_CMD_TABLE(EnableCRTC, params))
2040		result = BP_RESULT_OK;
2041
2042	return result;
2043}
2044
2045/*******************************************************************************
2046 ********************************************************************************
2047 **
2048 **                  ENABLE CRTC MEM REQ
2049 **
2050 ********************************************************************************
2051 *******************************************************************************/
2052
2053static enum bp_result enable_crtc_mem_req_v1(
2054	struct bios_parser *bp,
2055	enum controller_id controller_id,
2056	bool enable);
2057
2058static void init_enable_crtc_mem_req(struct bios_parser *bp)
2059{
2060	switch (BIOS_CMD_TABLE_PARA_REVISION(EnableCRTCMemReq)) {
2061	case 1:
2062		bp->cmd_tbl.enable_crtc_mem_req = enable_crtc_mem_req_v1;
2063		break;
2064	default:
2065		bp->cmd_tbl.enable_crtc_mem_req = NULL;
2066		break;
2067	}
2068}
2069
2070static enum bp_result enable_crtc_mem_req_v1(
2071	struct bios_parser *bp,
2072	enum controller_id controller_id,
2073	bool enable)
2074{
2075	bool result = BP_RESULT_BADINPUT;
2076	ENABLE_CRTC_PARAMETERS params = {0};
2077	uint8_t id;
2078
2079	if (bp->cmd_helper->controller_id_to_atom(controller_id, &id)) {
2080		params.ucCRTC = id;
2081
2082		if (enable)
2083			params.ucEnable = ATOM_ENABLE;
2084		else
2085			params.ucEnable = ATOM_DISABLE;
2086
2087		if (EXEC_BIOS_CMD_TABLE(EnableCRTCMemReq, params))
2088			result = BP_RESULT_OK;
2089		else
2090			result = BP_RESULT_FAILURE;
2091	}
2092
2093	return result;
2094}
2095
2096/*******************************************************************************
2097 ********************************************************************************
2098 **
2099 **                  DISPLAY PLL
2100 **
2101 ********************************************************************************
2102 *******************************************************************************/
2103
2104static enum bp_result program_clock_v5(
2105	struct bios_parser *bp,
2106	struct bp_pixel_clock_parameters *bp_params);
2107static enum bp_result program_clock_v6(
2108	struct bios_parser *bp,
2109	struct bp_pixel_clock_parameters *bp_params);
2110
2111static void init_program_clock(struct bios_parser *bp)
2112{
2113	switch (BIOS_CMD_TABLE_PARA_REVISION(SetPixelClock)) {
2114	case 5:
2115		bp->cmd_tbl.program_clock = program_clock_v5;
2116		break;
2117	case 6:
2118		bp->cmd_tbl.program_clock = program_clock_v6;
2119		break;
2120	default:
2121		dm_output_to_console("Don't have program_clock for v%d\n",
2122			 BIOS_CMD_TABLE_PARA_REVISION(SetPixelClock));
2123		bp->cmd_tbl.program_clock = NULL;
2124		break;
2125	}
2126}
2127
2128static enum bp_result program_clock_v5(
2129	struct bios_parser *bp,
2130	struct bp_pixel_clock_parameters *bp_params)
2131{
2132	enum bp_result result = BP_RESULT_FAILURE;
2133
2134	SET_PIXEL_CLOCK_PS_ALLOCATION_V5 params;
2135	uint32_t atom_pll_id;
2136
2137	memset(&params, 0, sizeof(params));
2138	if (!bp->cmd_helper->clock_source_id_to_atom(
2139			bp_params->pll_id, &atom_pll_id)) {
2140		BREAK_TO_DEBUGGER(); /* Invalid Inpute!! */
2141		return BP_RESULT_BADINPUT;
2142	}
2143
2144	/* We need to convert from KHz units into 10KHz units */
2145	params.sPCLKInput.ucPpll = (uint8_t) atom_pll_id;
2146	params.sPCLKInput.usPixelClock =
2147			cpu_to_le16((uint16_t) (bp_params->target_pixel_clock / 10));
2148	params.sPCLKInput.ucCRTC = (uint8_t) ATOM_CRTC_INVALID;
2149
2150	if (bp_params->flags.SET_EXTERNAL_REF_DIV_SRC)
2151		params.sPCLKInput.ucMiscInfo |= PIXEL_CLOCK_MISC_REF_DIV_SRC;
2152
2153	if (EXEC_BIOS_CMD_TABLE(SetPixelClock, params))
2154		result = BP_RESULT_OK;
2155
2156	return result;
2157}
2158
2159static enum bp_result program_clock_v6(
2160	struct bios_parser *bp,
2161	struct bp_pixel_clock_parameters *bp_params)
2162{
2163	enum bp_result result = BP_RESULT_FAILURE;
2164
2165	SET_PIXEL_CLOCK_PS_ALLOCATION_V6 params;
2166	uint32_t atom_pll_id;
2167
2168	memset(&params, 0, sizeof(params));
2169
2170	if (!bp->cmd_helper->clock_source_id_to_atom(
2171			bp_params->pll_id, &atom_pll_id)) {
2172		BREAK_TO_DEBUGGER(); /*Invalid Input!!*/
2173		return BP_RESULT_BADINPUT;
2174	}
2175
2176	/* We need to convert from KHz units into 10KHz units */
2177	params.sPCLKInput.ucPpll = (uint8_t)atom_pll_id;
2178	params.sPCLKInput.ulDispEngClkFreq =
2179			cpu_to_le32(bp_params->target_pixel_clock / 10);
2180
2181	if (bp_params->flags.SET_EXTERNAL_REF_DIV_SRC)
2182		params.sPCLKInput.ucMiscInfo |= PIXEL_CLOCK_MISC_REF_DIV_SRC;
 
 
 
2183
2184	if (EXEC_BIOS_CMD_TABLE(SetPixelClock, params)) {
2185		/* True display clock is returned by VBIOS if DFS bypass
2186		 * is enabled. */
2187		bp_params->dfs_bypass_display_clock =
2188				(uint32_t)(le32_to_cpu(params.sPCLKInput.ulDispEngClkFreq) * 10);
2189		result = BP_RESULT_OK;
2190	}
2191
2192	return result;
2193}
2194
2195/*******************************************************************************
2196 ********************************************************************************
2197 **
2198 **                  EXTERNAL ENCODER CONTROL
2199 **
2200 ********************************************************************************
2201 *******************************************************************************/
2202
2203static enum bp_result external_encoder_control_v3(
2204	struct bios_parser *bp,
2205	struct bp_external_encoder_control *cntl);
2206
2207static void init_external_encoder_control(
2208	struct bios_parser *bp)
2209{
2210	switch (BIOS_CMD_TABLE_PARA_REVISION(ExternalEncoderControl)) {
2211	case 3:
2212		bp->cmd_tbl.external_encoder_control =
2213				external_encoder_control_v3;
2214		break;
2215	default:
2216		bp->cmd_tbl.external_encoder_control = NULL;
2217		break;
2218	}
2219}
2220
2221static enum bp_result external_encoder_control_v3(
2222	struct bios_parser *bp,
2223	struct bp_external_encoder_control *cntl)
2224{
2225	enum bp_result result = BP_RESULT_FAILURE;
2226
2227	/* we need use _PS_Alloc struct */
2228	EXTERNAL_ENCODER_CONTROL_PS_ALLOCATION_V3 params;
2229	EXTERNAL_ENCODER_CONTROL_PARAMETERS_V3 *cntl_params;
2230	struct graphics_object_id encoder;
2231	bool is_input_signal_dp = false;
2232
2233	memset(&params, 0, sizeof(params));
2234
2235	cntl_params = &params.sExtEncoder;
2236
2237	encoder = cntl->encoder_id;
2238
2239	/* check if encoder supports external encoder control table */
2240	switch (dal_graphics_object_id_get_encoder_id(encoder)) {
2241	case ENCODER_ID_EXTERNAL_NUTMEG:
2242	case ENCODER_ID_EXTERNAL_TRAVIS:
2243		is_input_signal_dp = true;
2244		break;
2245
2246	default:
2247		BREAK_TO_DEBUGGER();
2248		return BP_RESULT_BADINPUT;
2249	}
2250
2251	/* Fill information based on the action
2252	 *
2253	 * Bit[6:4]: indicate external encoder, applied to all functions.
2254	 * =0: external encoder1, mapped to external encoder enum id1
2255	 * =1: external encoder2, mapped to external encoder enum id2
2256	 *
2257	 * enum ObjectEnumId
2258	 * {
2259	 *  EnumId_Unknown = 0,
2260	 *  EnumId_1,
2261	 *  EnumId_2,
2262	 * };
2263	 */
2264	cntl_params->ucConfig = (uint8_t)((encoder.enum_id - 1) << 4);
2265
2266	switch (cntl->action) {
2267	case EXTERNAL_ENCODER_CONTROL_INIT:
2268		/* output display connector type. Only valid in encoder
2269		 * initialization */
2270		cntl_params->usConnectorId =
2271				cpu_to_le16((uint16_t)cntl->connector_obj_id.id);
2272		break;
2273	case EXTERNAL_ENCODER_CONTROL_SETUP:
2274		/* EXTERNAL_ENCODER_CONTROL_PARAMETERS_V3 pixel clock unit in
2275		 * 10KHz
2276		 * output display device pixel clock frequency in unit of 10KHz.
2277		 * Only valid in setup and enableoutput
2278		 */
2279		cntl_params->usPixelClock =
2280				cpu_to_le16((uint16_t)(cntl->pixel_clock / 10));
2281		/* Indicate display output signal type drive by external
2282		 * encoder, only valid in setup and enableoutput */
2283		cntl_params->ucEncoderMode =
2284				(uint8_t)bp->cmd_helper->encoder_mode_bp_to_atom(
2285						cntl->signal, false);
2286
2287		if (is_input_signal_dp) {
2288			/* Bit[0]: indicate link rate, =1: 2.7Ghz, =0: 1.62Ghz,
2289			 * only valid in encoder setup with DP mode. */
2290			if (LINK_RATE_HIGH == cntl->link_rate)
2291				cntl_params->ucConfig |= 1;
2292			/* output color depth Indicate encoder data bpc format
2293			 * in DP mode, only valid in encoder setup in DP mode.
2294			 */
2295			cntl_params->ucBitPerColor =
2296					(uint8_t)(cntl->color_depth);
2297		}
2298		/* Indicate how many lanes used by external encoder, only valid
2299		 * in encoder setup and enableoutput. */
2300		cntl_params->ucLaneNum = (uint8_t)(cntl->lanes_number);
2301		break;
2302	case EXTERNAL_ENCODER_CONTROL_ENABLE:
2303		cntl_params->usPixelClock =
2304				cpu_to_le16((uint16_t)(cntl->pixel_clock / 10));
2305		cntl_params->ucEncoderMode =
2306				(uint8_t)bp->cmd_helper->encoder_mode_bp_to_atom(
2307						cntl->signal, false);
2308		cntl_params->ucLaneNum = (uint8_t)cntl->lanes_number;
2309		break;
2310	default:
2311		break;
2312	}
2313
2314	cntl_params->ucAction = (uint8_t)cntl->action;
2315
2316	if (EXEC_BIOS_CMD_TABLE(ExternalEncoderControl, params))
2317		result = BP_RESULT_OK;
2318
2319	return result;
2320}
2321
2322/*******************************************************************************
2323 ********************************************************************************
2324 **
2325 **                  ENABLE DISPLAY POWER GATING
2326 **
2327 ********************************************************************************
2328 *******************************************************************************/
2329
2330static enum bp_result enable_disp_power_gating_v2_1(
2331	struct bios_parser *bp,
2332	enum controller_id crtc_id,
2333	enum bp_pipe_control_action action);
2334
2335static void init_enable_disp_power_gating(
2336	struct bios_parser *bp)
2337{
2338	switch (BIOS_CMD_TABLE_PARA_REVISION(EnableDispPowerGating)) {
2339	case 1:
2340		bp->cmd_tbl.enable_disp_power_gating =
2341				enable_disp_power_gating_v2_1;
2342		break;
2343	default:
2344		dm_output_to_console("Don't enable_disp_power_gating enable_crtc for v%d\n",
2345			 BIOS_CMD_TABLE_PARA_REVISION(EnableDispPowerGating));
2346		bp->cmd_tbl.enable_disp_power_gating = NULL;
2347		break;
2348	}
2349}
2350
2351static enum bp_result enable_disp_power_gating_v2_1(
2352	struct bios_parser *bp,
2353	enum controller_id crtc_id,
2354	enum bp_pipe_control_action action)
2355{
2356	enum bp_result result = BP_RESULT_FAILURE;
2357
2358	ENABLE_DISP_POWER_GATING_PS_ALLOCATION params = {0};
2359	uint8_t atom_crtc_id;
2360
2361	if (bp->cmd_helper->controller_id_to_atom(crtc_id, &atom_crtc_id))
2362		params.ucDispPipeId = atom_crtc_id;
2363	else
2364		return BP_RESULT_BADINPUT;
2365
2366	params.ucEnable =
2367			bp->cmd_helper->disp_power_gating_action_to_atom(action);
2368
2369	if (EXEC_BIOS_CMD_TABLE(EnableDispPowerGating, params))
2370		result = BP_RESULT_OK;
2371
2372	return result;
2373}
2374
2375/*******************************************************************************
2376 ********************************************************************************
2377 **
2378 **                  SET DCE CLOCK
2379 **
2380 ********************************************************************************
2381 *******************************************************************************/
2382static enum bp_result set_dce_clock_v2_1(
2383	struct bios_parser *bp,
2384	struct bp_set_dce_clock_parameters *bp_params);
2385
2386static void init_set_dce_clock(struct bios_parser *bp)
2387{
2388	switch (BIOS_CMD_TABLE_PARA_REVISION(SetDCEClock)) {
2389	case 1:
2390		bp->cmd_tbl.set_dce_clock = set_dce_clock_v2_1;
2391		break;
2392	default:
2393		dm_output_to_console("Don't have set_dce_clock for v%d\n",
2394			 BIOS_CMD_TABLE_PARA_REVISION(SetDCEClock));
2395		bp->cmd_tbl.set_dce_clock = NULL;
2396		break;
2397	}
2398}
2399
2400static enum bp_result set_dce_clock_v2_1(
2401	struct bios_parser *bp,
2402	struct bp_set_dce_clock_parameters *bp_params)
2403{
2404	enum bp_result result = BP_RESULT_FAILURE;
2405
2406	SET_DCE_CLOCK_PS_ALLOCATION_V2_1 params;
2407	uint32_t atom_pll_id;
2408	uint32_t atom_clock_type;
2409	const struct command_table_helper *cmd = bp->cmd_helper;
2410
2411	memset(&params, 0, sizeof(params));
2412
2413	if (!cmd->clock_source_id_to_atom(bp_params->pll_id, &atom_pll_id) ||
2414			!cmd->dc_clock_type_to_atom(bp_params->clock_type, &atom_clock_type))
2415		return BP_RESULT_BADINPUT;
2416
2417	params.asParam.ucDCEClkSrc  = atom_pll_id;
2418	params.asParam.ucDCEClkType = atom_clock_type;
2419
2420	if (bp_params->clock_type == DCECLOCK_TYPE_DPREFCLK) {
2421		if (bp_params->flags.USE_GENLOCK_AS_SOURCE_FOR_DPREFCLK)
2422			params.asParam.ucDCEClkFlag |= DCE_CLOCK_FLAG_PLL_REFCLK_SRC_GENLK;
2423
2424		if (bp_params->flags.USE_PCIE_AS_SOURCE_FOR_DPREFCLK)
2425			params.asParam.ucDCEClkFlag |= DCE_CLOCK_FLAG_PLL_REFCLK_SRC_PCIE;
2426
2427		if (bp_params->flags.USE_XTALIN_AS_SOURCE_FOR_DPREFCLK)
2428			params.asParam.ucDCEClkFlag |= DCE_CLOCK_FLAG_PLL_REFCLK_SRC_XTALIN;
2429
2430		if (bp_params->flags.USE_GENERICA_AS_SOURCE_FOR_DPREFCLK)
2431			params.asParam.ucDCEClkFlag |= DCE_CLOCK_FLAG_PLL_REFCLK_SRC_GENERICA;
2432	}
2433	else
2434		/* only program clock frequency if display clock is used; VBIOS will program DPREFCLK */
2435		/* We need to convert from KHz units into 10KHz units */
2436		params.asParam.ulDCEClkFreq = cpu_to_le32(bp_params->target_clock_frequency / 10);
2437
2438	if (EXEC_BIOS_CMD_TABLE(SetDCEClock, params)) {
2439		/* Convert from 10KHz units back to KHz */
2440		bp_params->target_clock_frequency = le32_to_cpu(params.asParam.ulDCEClkFreq) * 10;
2441		result = BP_RESULT_OK;
2442	}
2443
2444	return result;
2445}