116012806SWyatt Wood /*
216012806SWyatt Wood * Copyright 2019 Advanced Micro Devices, Inc.
316012806SWyatt Wood *
416012806SWyatt Wood * Permission is hereby granted, free of charge, to any person obtaining a
516012806SWyatt Wood * copy of this software and associated documentation files (the "Software"),
616012806SWyatt Wood * to deal in the Software without restriction, including without limitation
716012806SWyatt Wood * the rights to use, copy, modify, merge, publish, distribute, sublicense,
816012806SWyatt Wood * and/or sell copies of the Software, and to permit persons to whom the
916012806SWyatt Wood * Software is furnished to do so, subject to the following conditions:
1016012806SWyatt Wood *
1116012806SWyatt Wood * The above copyright notice and this permission notice shall be included in
1216012806SWyatt Wood * all copies or substantial portions of the Software.
1316012806SWyatt Wood *
1416012806SWyatt Wood * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
1516012806SWyatt Wood * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
1616012806SWyatt Wood * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
1716012806SWyatt Wood * THE COPYRIGHT HOLDER(S) OR AUTHOR(S) BE LIABLE FOR ANY CLAIM, DAMAGES OR
1816012806SWyatt Wood * OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
1916012806SWyatt Wood * ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
2016012806SWyatt Wood * OTHER DEALINGS IN THE SOFTWARE.
2116012806SWyatt Wood *
2216012806SWyatt Wood * Authors: AMD
2316012806SWyatt Wood *
2416012806SWyatt Wood */
2516012806SWyatt Wood
2616012806SWyatt Wood #include "dmub_abm.h"
27b8fe5637SLeon Huang #include "dmub_abm_lcd.h"
2816012806SWyatt Wood #include "dc.h"
2916012806SWyatt Wood #include "core_types.h"
30da915efaSReza Amini #include "dmub_cmd.h"
31*4a8396d5SChun-Liang Chang #include "dc_dmub_srv.h"
32*4a8396d5SChun-Liang Chang #include "dmub/dmub_srv.h"
3316012806SWyatt Wood
3416012806SWyatt Wood #define TO_DMUB_ABM(abm)\
3516012806SWyatt Wood container_of(abm, struct dce_abm, base)
3616012806SWyatt Wood
37b8fe5637SLeon Huang #define ABM_FEATURE_NO_SUPPORT 0
38b8fe5637SLeon Huang #define ABM_LCD_SUPPORT 1
3916012806SWyatt Wood
abm_feature_support(struct abm * abm,unsigned int panel_inst)40b8fe5637SLeon Huang static unsigned int abm_feature_support(struct abm *abm, unsigned int panel_inst)
41dd5a94ceSWyatt Wood {
4216012806SWyatt Wood struct dc_context *dc = abm->ctx;
4363de4f04SJake Wang struct dc_link *edp_links[MAX_NUM_EDP];
4463de4f04SJake Wang int i;
4563de4f04SJake Wang int edp_num;
46b8fe5637SLeon Huang unsigned int ret = ABM_FEATURE_NO_SUPPORT;
4763de4f04SJake Wang
487ae1dbe6SWenjing Liu dc_get_edp_links(dc->dc, edp_links, &edp_num);
4963de4f04SJake Wang
5063de4f04SJake Wang for (i = 0; i < edp_num; i++) {
51b8fe5637SLeon Huang if (panel_inst == i)
52b8fe5637SLeon Huang break;
5363de4f04SJake Wang }
5416012806SWyatt Wood
55b8fe5637SLeon Huang if (i < edp_num) {
56b8fe5637SLeon Huang ret = ABM_LCD_SUPPORT;
5716012806SWyatt Wood }
5816012806SWyatt Wood
59b8fe5637SLeon Huang return ret;
60b8fe5637SLeon Huang }
61b8fe5637SLeon Huang
dmub_abm_init_ex(struct abm * abm,uint32_t backlight,uint32_t user_level)62ee8ed250SCamille Cho static void dmub_abm_init_ex(struct abm *abm, uint32_t backlight, uint32_t user_level)
63b8fe5637SLeon Huang {
64ee8ed250SCamille Cho dmub_abm_init(abm, backlight, user_level);
65b8fe5637SLeon Huang }
66b8fe5637SLeon Huang
dmub_abm_get_current_backlight_ex(struct abm * abm)67b8fe5637SLeon Huang static unsigned int dmub_abm_get_current_backlight_ex(struct abm *abm)
68b8fe5637SLeon Huang {
69fdb0ad2fSSung Joon Kim dc_allow_idle_optimizations(abm->ctx->dc, false);
70fdb0ad2fSSung Joon Kim
71b8fe5637SLeon Huang return dmub_abm_get_current_backlight(abm);
72b8fe5637SLeon Huang }
73b8fe5637SLeon Huang
dmub_abm_get_target_backlight_ex(struct abm * abm)74b8fe5637SLeon Huang static unsigned int dmub_abm_get_target_backlight_ex(struct abm *abm)
75b8fe5637SLeon Huang {
76fdb0ad2fSSung Joon Kim dc_allow_idle_optimizations(abm->ctx->dc, false);
77fdb0ad2fSSung Joon Kim
78b8fe5637SLeon Huang return dmub_abm_get_target_backlight(abm);
79b8fe5637SLeon Huang }
80b8fe5637SLeon Huang
dmub_abm_set_level_ex(struct abm * abm,uint32_t level)81b8fe5637SLeon Huang static bool dmub_abm_set_level_ex(struct abm *abm, uint32_t level)
82b8fe5637SLeon Huang {
83b8fe5637SLeon Huang bool ret = false;
84b8fe5637SLeon Huang unsigned int feature_support, i;
85b8fe5637SLeon Huang uint8_t panel_mask0 = 0;
86b8fe5637SLeon Huang
87b8fe5637SLeon Huang for (i = 0; i < MAX_NUM_EDP; i++) {
88b8fe5637SLeon Huang feature_support = abm_feature_support(abm, i);
89b8fe5637SLeon Huang
90b8fe5637SLeon Huang if (feature_support == ABM_LCD_SUPPORT)
91b8fe5637SLeon Huang panel_mask0 |= (0x01 << i);
92b8fe5637SLeon Huang }
93b8fe5637SLeon Huang
94b8fe5637SLeon Huang if (panel_mask0)
95b8fe5637SLeon Huang ret = dmub_abm_set_level(abm, level, panel_mask0);
96b8fe5637SLeon Huang
97b8fe5637SLeon Huang return ret;
98b8fe5637SLeon Huang }
99b8fe5637SLeon Huang
dmub_abm_init_config_ex(struct abm * abm,const char * src,unsigned int bytes,unsigned int inst)100b8fe5637SLeon Huang static bool dmub_abm_init_config_ex(struct abm *abm,
10116012806SWyatt Wood const char *src,
1026e568e43SJake Wang unsigned int bytes,
1036e568e43SJake Wang unsigned int inst)
10416012806SWyatt Wood {
105b8fe5637SLeon Huang unsigned int feature_support;
106c5d5b0ecSWyatt Wood
107b8fe5637SLeon Huang feature_support = abm_feature_support(abm, inst);
108c5d5b0ecSWyatt Wood
109b8fe5637SLeon Huang if (feature_support == ABM_LCD_SUPPORT)
110b8fe5637SLeon Huang dmub_abm_init_config(abm, src, bytes, inst);
111c5d5b0ecSWyatt Wood
11216012806SWyatt Wood return true;
11316012806SWyatt Wood }
11416012806SWyatt Wood
dmub_abm_set_pause_ex(struct abm * abm,bool pause,unsigned int panel_inst,unsigned int stream_inst)115b8fe5637SLeon Huang static bool dmub_abm_set_pause_ex(struct abm *abm, bool pause, unsigned int panel_inst, unsigned int stream_inst)
116b629a824SEric Yang {
117b8fe5637SLeon Huang bool ret = false;
118b8fe5637SLeon Huang unsigned int feature_support;
119b629a824SEric Yang
120b8fe5637SLeon Huang feature_support = abm_feature_support(abm, panel_inst);
121b629a824SEric Yang
122b8fe5637SLeon Huang if (feature_support == ABM_LCD_SUPPORT)
123b8fe5637SLeon Huang ret = dmub_abm_set_pause(abm, pause, panel_inst, stream_inst);
124b629a824SEric Yang
125b8fe5637SLeon Huang return ret;
126b8fe5637SLeon Huang }
127b8fe5637SLeon Huang
128da915efaSReza Amini /*****************************************************************************
129da915efaSReza Amini * dmub_abm_save_restore_ex() - calls dmub_abm_save_restore for preserving DMUB's
130da915efaSReza Amini * Varibright states for LCD only. OLED is TBD
131da915efaSReza Amini * @abm: used to check get dc context
132da915efaSReza Amini * @panel_inst: panel instance index
133da915efaSReza Amini * @pData: contains command to pause/un-pause abm and abm parameters
134da915efaSReza Amini *
135da915efaSReza Amini *
136da915efaSReza Amini ***************************************************************************/
dmub_abm_save_restore_ex(struct abm * abm,unsigned int panel_inst,struct abm_save_restore * pData)137da915efaSReza Amini static bool dmub_abm_save_restore_ex(
138da915efaSReza Amini struct abm *abm,
139da915efaSReza Amini unsigned int panel_inst,
140da915efaSReza Amini struct abm_save_restore *pData)
141da915efaSReza Amini {
142da915efaSReza Amini bool ret = false;
143da915efaSReza Amini unsigned int feature_support;
144da915efaSReza Amini struct dc_context *dc = abm->ctx;
145da915efaSReza Amini
146da915efaSReza Amini feature_support = abm_feature_support(abm, panel_inst);
147da915efaSReza Amini
148da915efaSReza Amini if (feature_support == ABM_LCD_SUPPORT)
149da915efaSReza Amini ret = dmub_abm_save_restore(dc, panel_inst, pData);
150da915efaSReza Amini
151da915efaSReza Amini return ret;
152da915efaSReza Amini }
153da915efaSReza Amini
dmub_abm_set_pipe_ex(struct abm * abm,uint32_t otg_inst,uint32_t option,uint32_t panel_inst,uint32_t pwrseq_inst)154b17ef04bSLewis Huang static bool dmub_abm_set_pipe_ex(struct abm *abm,
155b17ef04bSLewis Huang uint32_t otg_inst,
156b17ef04bSLewis Huang uint32_t option,
157b17ef04bSLewis Huang uint32_t panel_inst,
158b17ef04bSLewis Huang uint32_t pwrseq_inst)
159b8fe5637SLeon Huang {
160b8fe5637SLeon Huang bool ret = false;
161b8fe5637SLeon Huang unsigned int feature_support;
162b8fe5637SLeon Huang
163b8fe5637SLeon Huang feature_support = abm_feature_support(abm, panel_inst);
164b8fe5637SLeon Huang
165b8fe5637SLeon Huang if (feature_support == ABM_LCD_SUPPORT)
166b17ef04bSLewis Huang ret = dmub_abm_set_pipe(abm, otg_inst, option, panel_inst, pwrseq_inst);
167b8fe5637SLeon Huang
168b8fe5637SLeon Huang return ret;
169b8fe5637SLeon Huang }
170b8fe5637SLeon Huang
dmub_abm_set_backlight_level_pwm_ex(struct abm * abm,unsigned int backlight_pwm_u16_16,unsigned int frame_ramp,unsigned int controller_id,unsigned int panel_inst)171b8fe5637SLeon Huang static bool dmub_abm_set_backlight_level_pwm_ex(struct abm *abm,
172b8fe5637SLeon Huang unsigned int backlight_pwm_u16_16,
173b8fe5637SLeon Huang unsigned int frame_ramp,
174b8fe5637SLeon Huang unsigned int controller_id,
175b8fe5637SLeon Huang unsigned int panel_inst)
176b8fe5637SLeon Huang {
177b8fe5637SLeon Huang bool ret = false;
178b8fe5637SLeon Huang unsigned int feature_support;
179b8fe5637SLeon Huang
180b8fe5637SLeon Huang feature_support = abm_feature_support(abm, panel_inst);
181b8fe5637SLeon Huang
182b8fe5637SLeon Huang if (feature_support == ABM_LCD_SUPPORT)
183b8fe5637SLeon Huang ret = dmub_abm_set_backlight_level(abm, backlight_pwm_u16_16, frame_ramp, panel_inst);
184b8fe5637SLeon Huang
185b8fe5637SLeon Huang return ret;
186b629a824SEric Yang }
187b629a824SEric Yang
18816012806SWyatt Wood static const struct abm_funcs abm_funcs = {
189b8fe5637SLeon Huang .abm_init = dmub_abm_init_ex,
190b8fe5637SLeon Huang .set_abm_level = dmub_abm_set_level_ex,
191b8fe5637SLeon Huang .get_current_backlight = dmub_abm_get_current_backlight_ex,
192b8fe5637SLeon Huang .get_target_backlight = dmub_abm_get_target_backlight_ex,
193b8fe5637SLeon Huang .init_abm_config = dmub_abm_init_config_ex,
194b8fe5637SLeon Huang .set_abm_pause = dmub_abm_set_pause_ex,
195da915efaSReza Amini .save_restore = dmub_abm_save_restore_ex,
196b8fe5637SLeon Huang .set_pipe_ex = dmub_abm_set_pipe_ex,
197b8fe5637SLeon Huang .set_backlight_level_pwm = dmub_abm_set_backlight_level_pwm_ex,
19816012806SWyatt Wood };
19916012806SWyatt Wood
dmub_abm_construct(struct dce_abm * abm_dce,struct dc_context * ctx,const struct dce_abm_registers * regs,const struct dce_abm_shift * abm_shift,const struct dce_abm_mask * abm_mask)20016012806SWyatt Wood static void dmub_abm_construct(
20116012806SWyatt Wood struct dce_abm *abm_dce,
20216012806SWyatt Wood struct dc_context *ctx,
20316012806SWyatt Wood const struct dce_abm_registers *regs,
20416012806SWyatt Wood const struct dce_abm_shift *abm_shift,
20516012806SWyatt Wood const struct dce_abm_mask *abm_mask)
20616012806SWyatt Wood {
20716012806SWyatt Wood struct abm *base = &abm_dce->base;
20816012806SWyatt Wood
20916012806SWyatt Wood base->ctx = ctx;
21016012806SWyatt Wood base->funcs = &abm_funcs;
21116012806SWyatt Wood base->dmcu_is_running = false;
21216012806SWyatt Wood
21316012806SWyatt Wood abm_dce->regs = regs;
21416012806SWyatt Wood abm_dce->abm_shift = abm_shift;
21516012806SWyatt Wood abm_dce->abm_mask = abm_mask;
21616012806SWyatt Wood }
21716012806SWyatt Wood
dmub_abm_create(struct dc_context * ctx,const struct dce_abm_registers * regs,const struct dce_abm_shift * abm_shift,const struct dce_abm_mask * abm_mask)21816012806SWyatt Wood struct abm *dmub_abm_create(
21916012806SWyatt Wood struct dc_context *ctx,
22016012806SWyatt Wood const struct dce_abm_registers *regs,
22116012806SWyatt Wood const struct dce_abm_shift *abm_shift,
22216012806SWyatt Wood const struct dce_abm_mask *abm_mask)
22316012806SWyatt Wood {
224b8fe5637SLeon Huang if (ctx->dc->caps.dmcub_support) {
22516012806SWyatt Wood struct dce_abm *abm_dce = kzalloc(sizeof(*abm_dce), GFP_KERNEL);
22616012806SWyatt Wood
22716012806SWyatt Wood if (abm_dce == NULL) {
22816012806SWyatt Wood BREAK_TO_DEBUGGER();
22916012806SWyatt Wood return NULL;
23016012806SWyatt Wood }
23116012806SWyatt Wood
23216012806SWyatt Wood dmub_abm_construct(abm_dce, ctx, regs, abm_shift, abm_mask);
23316012806SWyatt Wood
23416012806SWyatt Wood return &abm_dce->base;
23516012806SWyatt Wood }
236b8fe5637SLeon Huang return NULL;
237b8fe5637SLeon Huang }
23816012806SWyatt Wood
dmub_abm_destroy(struct abm ** abm)23916012806SWyatt Wood void dmub_abm_destroy(struct abm **abm)
24016012806SWyatt Wood {
24116012806SWyatt Wood struct dce_abm *abm_dce = TO_DMUB_ABM(*abm);
24216012806SWyatt Wood
24316012806SWyatt Wood kfree(abm_dce);
24416012806SWyatt Wood *abm = NULL;
24516012806SWyatt Wood }
246