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