1 /*
2  * Copyright 2019 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 "dmub_abm.h"
27 #include "dce_abm.h"
28 #include "dc.h"
29 #include "dc_dmub_srv.h"
30 #include "dmub/dmub_srv.h"
31 #include "core_types.h"
32 #include "dm_services.h"
33 #include "reg_helper.h"
34 #include "fixed31_32.h"
35 
36 #include "atom.h"
37 
38 #define TO_DMUB_ABM(abm)\
39 	container_of(abm, struct dce_abm, base)
40 
41 #define REG(reg) \
42 	(dce_abm->regs->reg)
43 
44 #undef FN
45 #define FN(reg_name, field_name) \
46 	dce_abm->abm_shift->field_name, dce_abm->abm_mask->field_name
47 
48 #define CTX \
49 	dce_abm->base.ctx
50 
51 #define DISABLE_ABM_IMMEDIATELY 255
52 
53 
54 
55 static void dmub_abm_enable_fractional_pwm(struct dc_context *dc)
56 {
57 	union dmub_rb_cmd cmd;
58 	uint32_t fractional_pwm = (dc->dc->config.disable_fractional_pwm == false) ? 1 : 0;
59 
60 	memset(&cmd, 0, sizeof(cmd));
61 	cmd.abm_set_pwm_frac.header.type = DMUB_CMD__ABM;
62 	cmd.abm_set_pwm_frac.header.sub_type = DMUB_CMD__ABM_SET_PWM_FRAC;
63 	cmd.abm_set_pwm_frac.abm_set_pwm_frac_data.fractional_pwm = fractional_pwm;
64 	cmd.abm_set_pwm_frac.header.payload_bytes = sizeof(struct dmub_cmd_abm_set_pwm_frac_data);
65 
66 	dc_dmub_srv_cmd_queue(dc->dmub_srv, &cmd);
67 	dc_dmub_srv_cmd_execute(dc->dmub_srv);
68 	dc_dmub_srv_wait_idle(dc->dmub_srv);
69 }
70 
71 static void dmub_abm_init(struct abm *abm, uint32_t backlight)
72 {
73 	struct dce_abm *dce_abm = TO_DMUB_ABM(abm);
74 
75 	REG_WRITE(DC_ABM1_HG_SAMPLE_RATE, 0x3);
76 	REG_WRITE(DC_ABM1_HG_SAMPLE_RATE, 0x1);
77 	REG_WRITE(DC_ABM1_LS_SAMPLE_RATE, 0x3);
78 	REG_WRITE(DC_ABM1_LS_SAMPLE_RATE, 0x1);
79 	REG_WRITE(BL1_PWM_BL_UPDATE_SAMPLE_RATE, 0x1);
80 
81 	REG_SET_3(DC_ABM1_HG_MISC_CTRL, 0,
82 			ABM1_HG_NUM_OF_BINS_SEL, 0,
83 			ABM1_HG_VMAX_SEL, 1,
84 			ABM1_HG_BIN_BITWIDTH_SIZE_SEL, 0);
85 
86 	REG_SET_3(DC_ABM1_IPCSC_COEFF_SEL, 0,
87 			ABM1_IPCSC_COEFF_SEL_R, 2,
88 			ABM1_IPCSC_COEFF_SEL_G, 4,
89 			ABM1_IPCSC_COEFF_SEL_B, 2);
90 
91 	REG_UPDATE(BL1_PWM_CURRENT_ABM_LEVEL,
92 			BL1_PWM_CURRENT_ABM_LEVEL, backlight);
93 
94 	REG_UPDATE(BL1_PWM_TARGET_ABM_LEVEL,
95 			BL1_PWM_TARGET_ABM_LEVEL, backlight);
96 
97 	REG_UPDATE(BL1_PWM_USER_LEVEL,
98 			BL1_PWM_USER_LEVEL, backlight);
99 
100 	REG_UPDATE_2(DC_ABM1_LS_MIN_MAX_PIXEL_VALUE_THRES,
101 			ABM1_LS_MIN_PIXEL_VALUE_THRES, 0,
102 			ABM1_LS_MAX_PIXEL_VALUE_THRES, 1000);
103 
104 	REG_SET_3(DC_ABM1_HGLS_REG_READ_PROGRESS, 0,
105 			ABM1_HG_REG_READ_MISSED_FRAME_CLEAR, 1,
106 			ABM1_LS_REG_READ_MISSED_FRAME_CLEAR, 1,
107 			ABM1_BL_REG_READ_MISSED_FRAME_CLEAR, 1);
108 
109 	dmub_abm_enable_fractional_pwm(abm->ctx);
110 }
111 
112 static unsigned int dmub_abm_get_current_backlight(struct abm *abm)
113 {
114 	struct dce_abm *dce_abm = TO_DMUB_ABM(abm);
115 	unsigned int backlight = REG_READ(BL1_PWM_CURRENT_ABM_LEVEL);
116 
117 	/* return backlight in hardware format which is unsigned 17 bits, with
118 	 * 1 bit integer and 16 bit fractional
119 	 */
120 	return backlight;
121 }
122 
123 static unsigned int dmub_abm_get_target_backlight(struct abm *abm)
124 {
125 	struct dce_abm *dce_abm = TO_DMUB_ABM(abm);
126 	unsigned int backlight = REG_READ(BL1_PWM_TARGET_ABM_LEVEL);
127 
128 	/* return backlight in hardware format which is unsigned 17 bits, with
129 	 * 1 bit integer and 16 bit fractional
130 	 */
131 	return backlight;
132 }
133 
134 static bool dmub_abm_set_level(struct abm *abm, uint32_t level)
135 {
136 	union dmub_rb_cmd cmd;
137 	struct dc_context *dc = abm->ctx;
138 
139 	memset(&cmd, 0, sizeof(cmd));
140 	cmd.abm_set_level.header.type = DMUB_CMD__ABM;
141 	cmd.abm_set_level.header.sub_type = DMUB_CMD__ABM_SET_LEVEL;
142 	cmd.abm_set_level.abm_set_level_data.level = level;
143 	cmd.abm_set_level.header.payload_bytes = sizeof(struct dmub_cmd_abm_set_level_data);
144 
145 	dc_dmub_srv_cmd_queue(dc->dmub_srv, &cmd);
146 	dc_dmub_srv_cmd_execute(dc->dmub_srv);
147 	dc_dmub_srv_wait_idle(dc->dmub_srv);
148 
149 	return true;
150 }
151 
152 static bool dmub_abm_init_config(struct abm *abm,
153 	const char *src,
154 	unsigned int bytes)
155 {
156 	union dmub_rb_cmd cmd;
157 	struct dc_context *dc = abm->ctx;
158 
159 	// TODO: Optimize by only reading back final 4 bytes
160 	dmub_flush_buffer_mem(&dc->dmub_srv->dmub->scratch_mem_fb);
161 
162 	// Copy iramtable into cw7
163 	memcpy(dc->dmub_srv->dmub->scratch_mem_fb.cpu_addr, (void *)src, bytes);
164 
165 	memset(&cmd, 0, sizeof(cmd));
166 	// Fw will copy from cw7 to fw_state
167 	cmd.abm_init_config.header.type = DMUB_CMD__ABM;
168 	cmd.abm_init_config.header.sub_type = DMUB_CMD__ABM_INIT_CONFIG;
169 	cmd.abm_init_config.abm_init_config_data.src.quad_part = dc->dmub_srv->dmub->scratch_mem_fb.gpu_addr;
170 	cmd.abm_init_config.abm_init_config_data.bytes = bytes;
171 	cmd.abm_init_config.header.payload_bytes = sizeof(struct dmub_cmd_abm_init_config_data);
172 
173 	dc_dmub_srv_cmd_queue(dc->dmub_srv, &cmd);
174 	dc_dmub_srv_cmd_execute(dc->dmub_srv);
175 	dc_dmub_srv_wait_idle(dc->dmub_srv);
176 
177 	return true;
178 }
179 
180 static const struct abm_funcs abm_funcs = {
181 	.abm_init = dmub_abm_init,
182 	.set_abm_level = dmub_abm_set_level,
183 	.get_current_backlight = dmub_abm_get_current_backlight,
184 	.get_target_backlight = dmub_abm_get_target_backlight,
185 	.init_abm_config = dmub_abm_init_config,
186 };
187 
188 static void dmub_abm_construct(
189 	struct dce_abm *abm_dce,
190 	struct dc_context *ctx,
191 	const struct dce_abm_registers *regs,
192 	const struct dce_abm_shift *abm_shift,
193 	const struct dce_abm_mask *abm_mask)
194 {
195 	struct abm *base = &abm_dce->base;
196 
197 	base->ctx = ctx;
198 	base->funcs = &abm_funcs;
199 	base->dmcu_is_running = false;
200 
201 	abm_dce->regs = regs;
202 	abm_dce->abm_shift = abm_shift;
203 	abm_dce->abm_mask = abm_mask;
204 }
205 
206 struct abm *dmub_abm_create(
207 	struct dc_context *ctx,
208 	const struct dce_abm_registers *regs,
209 	const struct dce_abm_shift *abm_shift,
210 	const struct dce_abm_mask *abm_mask)
211 {
212 	struct dce_abm *abm_dce = kzalloc(sizeof(*abm_dce), GFP_KERNEL);
213 
214 	if (abm_dce == NULL) {
215 		BREAK_TO_DEBUGGER();
216 		return NULL;
217 	}
218 
219 	dmub_abm_construct(abm_dce, ctx, regs, abm_shift, abm_mask);
220 
221 	return &abm_dce->base;
222 }
223 
224 void dmub_abm_destroy(struct abm **abm)
225 {
226 	struct dce_abm *abm_dce = TO_DMUB_ABM(*abm);
227 
228 	kfree(abm_dce);
229 	*abm = NULL;
230 }
231