Loading...
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 *)¶ms, 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, ¶ms);
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, ¶ms);
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(¶ms, 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(¶ms, 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(¶ms, 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(¶ms, 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(¶ms, 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(¶ms, 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(¶ms, 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(¶ms, 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(¶ms, 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 ¶ms,
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 ¶ms,
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(¶ms.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(¶ms, 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(¶ms, 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(¶ms, 0, sizeof(params));
2221
2222 cntl_params = ¶ms.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(¶ms, 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}
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 ¶ms) == 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, ¶ms);
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, ¶ms);
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(¶ms, 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(¶ms, 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(¶ms, 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(¶ms, 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(¶ms, 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(¶ms, 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(¶ms, 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(¶ms, 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(¶ms, 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 ¶ms,
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 ¶ms,
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(¶ms, 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(¶ms, 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(¶ms, 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(¶ms, 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(¶ms, 0, sizeof(params));
2234
2235 cntl_params = ¶ms.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(¶ms, 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}