xref: /dflybsd-src/sys/dev/drm/amd/display/dc/i2caux/i2caux.c (revision b843c749addef9340ee7d4e250b09fdd492602a1)
1*b843c749SSergey Zigachev /*
2*b843c749SSergey Zigachev  * Copyright 2012-15 Advanced Micro Devices, Inc.
3*b843c749SSergey Zigachev  *
4*b843c749SSergey Zigachev  * Permission is hereby granted, free of charge, to any person obtaining a
5*b843c749SSergey Zigachev  * copy of this software and associated documentation files (the "Software"),
6*b843c749SSergey Zigachev  * to deal in the Software without restriction, including without limitation
7*b843c749SSergey Zigachev  * the rights to use, copy, modify, merge, publish, distribute, sublicense,
8*b843c749SSergey Zigachev  * and/or sell copies of the Software, and to permit persons to whom the
9*b843c749SSergey Zigachev  * Software is furnished to do so, subject to the following conditions:
10*b843c749SSergey Zigachev  *
11*b843c749SSergey Zigachev  * The above copyright notice and this permission notice shall be included in
12*b843c749SSergey Zigachev  * all copies or substantial portions of the Software.
13*b843c749SSergey Zigachev  *
14*b843c749SSergey Zigachev  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
15*b843c749SSergey Zigachev  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
16*b843c749SSergey Zigachev  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL
17*b843c749SSergey Zigachev  * THE COPYRIGHT HOLDER(S) OR AUTHOR(S) BE LIABLE FOR ANY CLAIM, DAMAGES OR
18*b843c749SSergey Zigachev  * OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
19*b843c749SSergey Zigachev  * ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
20*b843c749SSergey Zigachev  * OTHER DEALINGS IN THE SOFTWARE.
21*b843c749SSergey Zigachev  *
22*b843c749SSergey Zigachev  * Authors: AMD
23*b843c749SSergey Zigachev  *
24*b843c749SSergey Zigachev  */
25*b843c749SSergey Zigachev 
26*b843c749SSergey Zigachev #include "dm_services.h"
27*b843c749SSergey Zigachev 
28*b843c749SSergey Zigachev /*
29*b843c749SSergey Zigachev  * Pre-requisites: headers required by header of this unit
30*b843c749SSergey Zigachev  */
31*b843c749SSergey Zigachev #include "include/i2caux_interface.h"
32*b843c749SSergey Zigachev #include "dc_bios_types.h"
33*b843c749SSergey Zigachev 
34*b843c749SSergey Zigachev /*
35*b843c749SSergey Zigachev  * Header of this unit
36*b843c749SSergey Zigachev  */
37*b843c749SSergey Zigachev 
38*b843c749SSergey Zigachev #include "i2caux.h"
39*b843c749SSergey Zigachev 
40*b843c749SSergey Zigachev /*
41*b843c749SSergey Zigachev  * Post-requisites: headers required by this unit
42*b843c749SSergey Zigachev  */
43*b843c749SSergey Zigachev 
44*b843c749SSergey Zigachev #include "engine.h"
45*b843c749SSergey Zigachev #include "i2c_engine.h"
46*b843c749SSergey Zigachev #include "aux_engine.h"
47*b843c749SSergey Zigachev 
48*b843c749SSergey Zigachev /*
49*b843c749SSergey Zigachev  * This unit
50*b843c749SSergey Zigachev  */
51*b843c749SSergey Zigachev 
52*b843c749SSergey Zigachev #include "dce80/i2caux_dce80.h"
53*b843c749SSergey Zigachev 
54*b843c749SSergey Zigachev #include "dce100/i2caux_dce100.h"
55*b843c749SSergey Zigachev 
56*b843c749SSergey Zigachev #include "dce110/i2caux_dce110.h"
57*b843c749SSergey Zigachev 
58*b843c749SSergey Zigachev #include "dce112/i2caux_dce112.h"
59*b843c749SSergey Zigachev 
60*b843c749SSergey Zigachev #include "dce120/i2caux_dce120.h"
61*b843c749SSergey Zigachev 
62*b843c749SSergey Zigachev #if defined(CONFIG_DRM_AMD_DC_DCN1_0)
63*b843c749SSergey Zigachev #include "dcn10/i2caux_dcn10.h"
64*b843c749SSergey Zigachev #endif
65*b843c749SSergey Zigachev 
66*b843c749SSergey Zigachev #include "diagnostics/i2caux_diag.h"
67*b843c749SSergey Zigachev 
68*b843c749SSergey Zigachev /*
69*b843c749SSergey Zigachev  * @brief
70*b843c749SSergey Zigachev  * Plain API, available publicly
71*b843c749SSergey Zigachev  */
72*b843c749SSergey Zigachev 
dal_i2caux_create(struct dc_context * ctx)73*b843c749SSergey Zigachev struct i2caux *dal_i2caux_create(
74*b843c749SSergey Zigachev 	struct dc_context *ctx)
75*b843c749SSergey Zigachev {
76*b843c749SSergey Zigachev 	if (IS_FPGA_MAXIMUS_DC(ctx->dce_environment)) {
77*b843c749SSergey Zigachev 		return dal_i2caux_diag_fpga_create(ctx);
78*b843c749SSergey Zigachev 	}
79*b843c749SSergey Zigachev 
80*b843c749SSergey Zigachev 	switch (ctx->dce_version) {
81*b843c749SSergey Zigachev 	case DCE_VERSION_8_0:
82*b843c749SSergey Zigachev 	case DCE_VERSION_8_1:
83*b843c749SSergey Zigachev 	case DCE_VERSION_8_3:
84*b843c749SSergey Zigachev 		return dal_i2caux_dce80_create(ctx);
85*b843c749SSergey Zigachev 	case DCE_VERSION_11_2:
86*b843c749SSergey Zigachev 	case DCE_VERSION_11_22:
87*b843c749SSergey Zigachev 		return dal_i2caux_dce112_create(ctx);
88*b843c749SSergey Zigachev 	case DCE_VERSION_11_0:
89*b843c749SSergey Zigachev 		return dal_i2caux_dce110_create(ctx);
90*b843c749SSergey Zigachev 	case DCE_VERSION_10_0:
91*b843c749SSergey Zigachev 		return dal_i2caux_dce100_create(ctx);
92*b843c749SSergey Zigachev 	case DCE_VERSION_12_0:
93*b843c749SSergey Zigachev 		return dal_i2caux_dce120_create(ctx);
94*b843c749SSergey Zigachev #if defined(CONFIG_DRM_AMD_DC_DCN1_0)
95*b843c749SSergey Zigachev 	case DCN_VERSION_1_0:
96*b843c749SSergey Zigachev 		return dal_i2caux_dcn10_create(ctx);
97*b843c749SSergey Zigachev #endif
98*b843c749SSergey Zigachev 
99*b843c749SSergey Zigachev 	default:
100*b843c749SSergey Zigachev 		BREAK_TO_DEBUGGER();
101*b843c749SSergey Zigachev 		return NULL;
102*b843c749SSergey Zigachev 	}
103*b843c749SSergey Zigachev }
104*b843c749SSergey Zigachev 
dal_i2caux_submit_i2c_command(struct i2caux * i2caux,struct ddc * ddc,struct i2c_command * cmd)105*b843c749SSergey Zigachev bool dal_i2caux_submit_i2c_command(
106*b843c749SSergey Zigachev 	struct i2caux *i2caux,
107*b843c749SSergey Zigachev 	struct ddc *ddc,
108*b843c749SSergey Zigachev 	struct i2c_command *cmd)
109*b843c749SSergey Zigachev {
110*b843c749SSergey Zigachev 	struct i2c_engine *engine;
111*b843c749SSergey Zigachev 	uint8_t index_of_payload = 0;
112*b843c749SSergey Zigachev 	bool result;
113*b843c749SSergey Zigachev 
114*b843c749SSergey Zigachev 	if (!ddc) {
115*b843c749SSergey Zigachev 		BREAK_TO_DEBUGGER();
116*b843c749SSergey Zigachev 		return false;
117*b843c749SSergey Zigachev 	}
118*b843c749SSergey Zigachev 
119*b843c749SSergey Zigachev 	if (!cmd) {
120*b843c749SSergey Zigachev 		BREAK_TO_DEBUGGER();
121*b843c749SSergey Zigachev 		return false;
122*b843c749SSergey Zigachev 	}
123*b843c749SSergey Zigachev 
124*b843c749SSergey Zigachev 	/*
125*b843c749SSergey Zigachev 	 * default will be SW, however there is a feature flag in adapter
126*b843c749SSergey Zigachev 	 * service that determines whether SW i2c_engine will be available or
127*b843c749SSergey Zigachev 	 * not, if sw i2c is not available we will fallback to hw. This feature
128*b843c749SSergey Zigachev 	 * flag is set to not creating sw i2c engine for every dce except dce80
129*b843c749SSergey Zigachev 	 * currently
130*b843c749SSergey Zigachev 	 */
131*b843c749SSergey Zigachev 	switch (cmd->engine) {
132*b843c749SSergey Zigachev 	case I2C_COMMAND_ENGINE_DEFAULT:
133*b843c749SSergey Zigachev 	case I2C_COMMAND_ENGINE_SW:
134*b843c749SSergey Zigachev 		/* try to acquire SW engine first,
135*b843c749SSergey Zigachev 		 * acquire HW engine if SW engine not available */
136*b843c749SSergey Zigachev 		engine = i2caux->funcs->acquire_i2c_sw_engine(i2caux, ddc);
137*b843c749SSergey Zigachev 
138*b843c749SSergey Zigachev 		if (!engine)
139*b843c749SSergey Zigachev 			engine = i2caux->funcs->acquire_i2c_hw_engine(
140*b843c749SSergey Zigachev 				i2caux, ddc);
141*b843c749SSergey Zigachev 	break;
142*b843c749SSergey Zigachev 	case I2C_COMMAND_ENGINE_HW:
143*b843c749SSergey Zigachev 	default:
144*b843c749SSergey Zigachev 		/* try to acquire HW engine first,
145*b843c749SSergey Zigachev 		 * acquire SW engine if HW engine not available */
146*b843c749SSergey Zigachev 		engine = i2caux->funcs->acquire_i2c_hw_engine(i2caux, ddc);
147*b843c749SSergey Zigachev 
148*b843c749SSergey Zigachev 		if (!engine)
149*b843c749SSergey Zigachev 			engine = i2caux->funcs->acquire_i2c_sw_engine(
150*b843c749SSergey Zigachev 				i2caux, ddc);
151*b843c749SSergey Zigachev 	}
152*b843c749SSergey Zigachev 
153*b843c749SSergey Zigachev 	if (!engine)
154*b843c749SSergey Zigachev 		return false;
155*b843c749SSergey Zigachev 
156*b843c749SSergey Zigachev 	engine->funcs->set_speed(engine, cmd->speed);
157*b843c749SSergey Zigachev 
158*b843c749SSergey Zigachev 	result = true;
159*b843c749SSergey Zigachev 
160*b843c749SSergey Zigachev 	while (index_of_payload < cmd->number_of_payloads) {
161*b843c749SSergey Zigachev 		bool mot = (index_of_payload != cmd->number_of_payloads - 1);
162*b843c749SSergey Zigachev 
163*b843c749SSergey Zigachev 		struct i2c_payload *payload = cmd->payloads + index_of_payload;
164*b843c749SSergey Zigachev 
165*b843c749SSergey Zigachev 		struct i2caux_transaction_request request = { 0 };
166*b843c749SSergey Zigachev 
167*b843c749SSergey Zigachev 		request.operation = payload->write ?
168*b843c749SSergey Zigachev 			I2CAUX_TRANSACTION_WRITE :
169*b843c749SSergey Zigachev 			I2CAUX_TRANSACTION_READ;
170*b843c749SSergey Zigachev 
171*b843c749SSergey Zigachev 		request.payload.address_space =
172*b843c749SSergey Zigachev 			I2CAUX_TRANSACTION_ADDRESS_SPACE_I2C;
173*b843c749SSergey Zigachev 		request.payload.address = (payload->address << 1) |
174*b843c749SSergey Zigachev 			!payload->write;
175*b843c749SSergey Zigachev 		request.payload.length = payload->length;
176*b843c749SSergey Zigachev 		request.payload.data = payload->data;
177*b843c749SSergey Zigachev 
178*b843c749SSergey Zigachev 		if (!engine->base.funcs->submit_request(
179*b843c749SSergey Zigachev 			&engine->base, &request, mot)) {
180*b843c749SSergey Zigachev 			result = false;
181*b843c749SSergey Zigachev 			break;
182*b843c749SSergey Zigachev 		}
183*b843c749SSergey Zigachev 
184*b843c749SSergey Zigachev 		++index_of_payload;
185*b843c749SSergey Zigachev 	}
186*b843c749SSergey Zigachev 
187*b843c749SSergey Zigachev 	i2caux->funcs->release_engine(i2caux, &engine->base);
188*b843c749SSergey Zigachev 
189*b843c749SSergey Zigachev 	return result;
190*b843c749SSergey Zigachev }
191*b843c749SSergey Zigachev 
dal_i2caux_submit_aux_command(struct i2caux * i2caux,struct ddc * ddc,struct aux_command * cmd)192*b843c749SSergey Zigachev bool dal_i2caux_submit_aux_command(
193*b843c749SSergey Zigachev 	struct i2caux *i2caux,
194*b843c749SSergey Zigachev 	struct ddc *ddc,
195*b843c749SSergey Zigachev 	struct aux_command *cmd)
196*b843c749SSergey Zigachev {
197*b843c749SSergey Zigachev 	struct aux_engine *engine;
198*b843c749SSergey Zigachev 	uint8_t index_of_payload = 0;
199*b843c749SSergey Zigachev 	bool result;
200*b843c749SSergey Zigachev 	bool mot;
201*b843c749SSergey Zigachev 
202*b843c749SSergey Zigachev 	if (!ddc) {
203*b843c749SSergey Zigachev 		BREAK_TO_DEBUGGER();
204*b843c749SSergey Zigachev 		return false;
205*b843c749SSergey Zigachev 	}
206*b843c749SSergey Zigachev 
207*b843c749SSergey Zigachev 	if (!cmd) {
208*b843c749SSergey Zigachev 		BREAK_TO_DEBUGGER();
209*b843c749SSergey Zigachev 		return false;
210*b843c749SSergey Zigachev 	}
211*b843c749SSergey Zigachev 
212*b843c749SSergey Zigachev 	engine = i2caux->funcs->acquire_aux_engine(i2caux, ddc);
213*b843c749SSergey Zigachev 
214*b843c749SSergey Zigachev 	if (!engine)
215*b843c749SSergey Zigachev 		return false;
216*b843c749SSergey Zigachev 
217*b843c749SSergey Zigachev 	engine->delay = cmd->defer_delay;
218*b843c749SSergey Zigachev 	engine->max_defer_write_retry = cmd->max_defer_write_retry;
219*b843c749SSergey Zigachev 
220*b843c749SSergey Zigachev 	result = true;
221*b843c749SSergey Zigachev 
222*b843c749SSergey Zigachev 	while (index_of_payload < cmd->number_of_payloads) {
223*b843c749SSergey Zigachev 		struct aux_payload *payload = cmd->payloads + index_of_payload;
224*b843c749SSergey Zigachev 		struct i2caux_transaction_request request = { 0 };
225*b843c749SSergey Zigachev 
226*b843c749SSergey Zigachev 		if (cmd->mot == I2C_MOT_UNDEF)
227*b843c749SSergey Zigachev 			mot = (index_of_payload != cmd->number_of_payloads - 1);
228*b843c749SSergey Zigachev 		else
229*b843c749SSergey Zigachev 			mot = (cmd->mot == I2C_MOT_TRUE);
230*b843c749SSergey Zigachev 
231*b843c749SSergey Zigachev 		request.operation = payload->write ?
232*b843c749SSergey Zigachev 			I2CAUX_TRANSACTION_WRITE :
233*b843c749SSergey Zigachev 			I2CAUX_TRANSACTION_READ;
234*b843c749SSergey Zigachev 
235*b843c749SSergey Zigachev 		if (payload->i2c_over_aux) {
236*b843c749SSergey Zigachev 			request.payload.address_space =
237*b843c749SSergey Zigachev 				I2CAUX_TRANSACTION_ADDRESS_SPACE_I2C;
238*b843c749SSergey Zigachev 
239*b843c749SSergey Zigachev 			request.payload.address = (payload->address << 1) |
240*b843c749SSergey Zigachev 				!payload->write;
241*b843c749SSergey Zigachev 		} else {
242*b843c749SSergey Zigachev 			request.payload.address_space =
243*b843c749SSergey Zigachev 				I2CAUX_TRANSACTION_ADDRESS_SPACE_DPCD;
244*b843c749SSergey Zigachev 
245*b843c749SSergey Zigachev 			request.payload.address = payload->address;
246*b843c749SSergey Zigachev 		}
247*b843c749SSergey Zigachev 
248*b843c749SSergey Zigachev 		request.payload.length = payload->length;
249*b843c749SSergey Zigachev 		request.payload.data = payload->data;
250*b843c749SSergey Zigachev 
251*b843c749SSergey Zigachev 		if (!engine->base.funcs->submit_request(
252*b843c749SSergey Zigachev 			&engine->base, &request, mot)) {
253*b843c749SSergey Zigachev 			result = false;
254*b843c749SSergey Zigachev 			break;
255*b843c749SSergey Zigachev 		}
256*b843c749SSergey Zigachev 
257*b843c749SSergey Zigachev 		++index_of_payload;
258*b843c749SSergey Zigachev 	}
259*b843c749SSergey Zigachev 
260*b843c749SSergey Zigachev 	i2caux->funcs->release_engine(i2caux, &engine->base);
261*b843c749SSergey Zigachev 
262*b843c749SSergey Zigachev 	return result;
263*b843c749SSergey Zigachev }
264*b843c749SSergey Zigachev 
get_hw_supported_ddc_line(struct ddc * ddc,enum gpio_ddc_line * line)265*b843c749SSergey Zigachev static bool get_hw_supported_ddc_line(
266*b843c749SSergey Zigachev 	struct ddc *ddc,
267*b843c749SSergey Zigachev 	enum gpio_ddc_line *line)
268*b843c749SSergey Zigachev {
269*b843c749SSergey Zigachev 	enum gpio_ddc_line line_found;
270*b843c749SSergey Zigachev 
271*b843c749SSergey Zigachev 	*line = GPIO_DDC_LINE_UNKNOWN;
272*b843c749SSergey Zigachev 
273*b843c749SSergey Zigachev 	if (!ddc) {
274*b843c749SSergey Zigachev 		BREAK_TO_DEBUGGER();
275*b843c749SSergey Zigachev 		return false;
276*b843c749SSergey Zigachev 	}
277*b843c749SSergey Zigachev 
278*b843c749SSergey Zigachev 	if (!ddc->hw_info.hw_supported)
279*b843c749SSergey Zigachev 		return false;
280*b843c749SSergey Zigachev 
281*b843c749SSergey Zigachev 	line_found = dal_ddc_get_line(ddc);
282*b843c749SSergey Zigachev 
283*b843c749SSergey Zigachev 	if (line_found >= GPIO_DDC_LINE_COUNT)
284*b843c749SSergey Zigachev 		return false;
285*b843c749SSergey Zigachev 
286*b843c749SSergey Zigachev 	*line = line_found;
287*b843c749SSergey Zigachev 
288*b843c749SSergey Zigachev 	return true;
289*b843c749SSergey Zigachev }
290*b843c749SSergey Zigachev 
dal_i2caux_configure_aux(struct i2caux * i2caux,struct ddc * ddc,union aux_config cfg)291*b843c749SSergey Zigachev void dal_i2caux_configure_aux(
292*b843c749SSergey Zigachev 	struct i2caux *i2caux,
293*b843c749SSergey Zigachev 	struct ddc *ddc,
294*b843c749SSergey Zigachev 	union aux_config cfg)
295*b843c749SSergey Zigachev {
296*b843c749SSergey Zigachev 	struct aux_engine *engine =
297*b843c749SSergey Zigachev 		i2caux->funcs->acquire_aux_engine(i2caux, ddc);
298*b843c749SSergey Zigachev 
299*b843c749SSergey Zigachev 	if (!engine)
300*b843c749SSergey Zigachev 		return;
301*b843c749SSergey Zigachev 
302*b843c749SSergey Zigachev 	engine->funcs->configure(engine, cfg);
303*b843c749SSergey Zigachev 
304*b843c749SSergey Zigachev 	i2caux->funcs->release_engine(i2caux, &engine->base);
305*b843c749SSergey Zigachev }
306*b843c749SSergey Zigachev 
dal_i2caux_destroy(struct i2caux ** i2caux)307*b843c749SSergey Zigachev void dal_i2caux_destroy(
308*b843c749SSergey Zigachev 	struct i2caux **i2caux)
309*b843c749SSergey Zigachev {
310*b843c749SSergey Zigachev 	if (!i2caux || !*i2caux) {
311*b843c749SSergey Zigachev 		BREAK_TO_DEBUGGER();
312*b843c749SSergey Zigachev 		return;
313*b843c749SSergey Zigachev 	}
314*b843c749SSergey Zigachev 
315*b843c749SSergey Zigachev 	(*i2caux)->funcs->destroy(i2caux);
316*b843c749SSergey Zigachev 
317*b843c749SSergey Zigachev 	*i2caux = NULL;
318*b843c749SSergey Zigachev }
319*b843c749SSergey Zigachev 
320*b843c749SSergey Zigachev /*
321*b843c749SSergey Zigachev  * @brief
322*b843c749SSergey Zigachev  * An utility function used by 'struct i2caux' and its descendants
323*b843c749SSergey Zigachev  */
324*b843c749SSergey Zigachev 
dal_i2caux_get_reference_clock(struct dc_bios * bios)325*b843c749SSergey Zigachev uint32_t dal_i2caux_get_reference_clock(
326*b843c749SSergey Zigachev 		struct dc_bios *bios)
327*b843c749SSergey Zigachev {
328*b843c749SSergey Zigachev 	struct dc_firmware_info info = { { 0 } };
329*b843c749SSergey Zigachev 
330*b843c749SSergey Zigachev 	if (bios->funcs->get_firmware_info(bios, &info) != BP_RESULT_OK)
331*b843c749SSergey Zigachev 		return 0;
332*b843c749SSergey Zigachev 
333*b843c749SSergey Zigachev 	return info.pll_info.crystal_frequency;
334*b843c749SSergey Zigachev }
335*b843c749SSergey Zigachev 
336*b843c749SSergey Zigachev /*
337*b843c749SSergey Zigachev  * @brief
338*b843c749SSergey Zigachev  * i2caux
339*b843c749SSergey Zigachev  */
340*b843c749SSergey Zigachev 
341*b843c749SSergey Zigachev enum {
342*b843c749SSergey Zigachev 	/* following are expressed in KHz */
343*b843c749SSergey Zigachev 	DEFAULT_I2C_SW_SPEED = 50,
344*b843c749SSergey Zigachev 	DEFAULT_I2C_HW_SPEED = 50,
345*b843c749SSergey Zigachev 
346*b843c749SSergey Zigachev 	DEFAULT_I2C_SW_SPEED_100KHZ = 100,
347*b843c749SSergey Zigachev 	DEFAULT_I2C_HW_SPEED_100KHZ = 100,
348*b843c749SSergey Zigachev 
349*b843c749SSergey Zigachev 	/* This is the timeout as defined in DP 1.2a,
350*b843c749SSergey Zigachev 	 * 2.3.4 "Detailed uPacket TX AUX CH State Description". */
351*b843c749SSergey Zigachev 	AUX_TIMEOUT_PERIOD = 400,
352*b843c749SSergey Zigachev 
353*b843c749SSergey Zigachev 	/* Ideally, the SW timeout should be just above 550usec
354*b843c749SSergey Zigachev 	 * which is programmed in HW.
355*b843c749SSergey Zigachev 	 * But the SW timeout of 600usec is not reliable,
356*b843c749SSergey Zigachev 	 * because on some systems, delay_in_microseconds()
357*b843c749SSergey Zigachev 	 * returns faster than it should.
358*b843c749SSergey Zigachev 	 * EPR #379763: by trial-and-error on different systems,
359*b843c749SSergey Zigachev 	 * 700usec is the minimum reliable SW timeout for polling
360*b843c749SSergey Zigachev 	 * the AUX_SW_STATUS.AUX_SW_DONE bit.
361*b843c749SSergey Zigachev 	 * This timeout expires *only* when there is
362*b843c749SSergey Zigachev 	 * AUX Error or AUX Timeout conditions - not during normal operation.
363*b843c749SSergey Zigachev 	 * During normal operation, AUX_SW_STATUS.AUX_SW_DONE bit is set
364*b843c749SSergey Zigachev 	 * at most within ~240usec. That means,
365*b843c749SSergey Zigachev 	 * increasing this timeout will not affect normal operation,
366*b843c749SSergey Zigachev 	 * and we'll timeout after
367*b843c749SSergey Zigachev 	 * SW_AUX_TIMEOUT_PERIOD_MULTIPLIER * AUX_TIMEOUT_PERIOD = 1600usec.
368*b843c749SSergey Zigachev 	 * This timeout is especially important for
369*b843c749SSergey Zigachev 	 * resume from S3 and CTS. */
370*b843c749SSergey Zigachev 	SW_AUX_TIMEOUT_PERIOD_MULTIPLIER = 4
371*b843c749SSergey Zigachev };
372*b843c749SSergey Zigachev 
dal_i2caux_acquire_i2c_sw_engine(struct i2caux * i2caux,struct ddc * ddc)373*b843c749SSergey Zigachev struct i2c_engine *dal_i2caux_acquire_i2c_sw_engine(
374*b843c749SSergey Zigachev 	struct i2caux *i2caux,
375*b843c749SSergey Zigachev 	struct ddc *ddc)
376*b843c749SSergey Zigachev {
377*b843c749SSergey Zigachev 	enum gpio_ddc_line line;
378*b843c749SSergey Zigachev 	struct i2c_engine *engine = NULL;
379*b843c749SSergey Zigachev 
380*b843c749SSergey Zigachev 	if (get_hw_supported_ddc_line(ddc, &line))
381*b843c749SSergey Zigachev 		engine = i2caux->i2c_sw_engines[line];
382*b843c749SSergey Zigachev 
383*b843c749SSergey Zigachev 	if (!engine)
384*b843c749SSergey Zigachev 		engine = i2caux->i2c_generic_sw_engine;
385*b843c749SSergey Zigachev 
386*b843c749SSergey Zigachev 	if (!engine)
387*b843c749SSergey Zigachev 		return NULL;
388*b843c749SSergey Zigachev 
389*b843c749SSergey Zigachev 	if (!engine->base.funcs->acquire(&engine->base, ddc))
390*b843c749SSergey Zigachev 		return NULL;
391*b843c749SSergey Zigachev 
392*b843c749SSergey Zigachev 	return engine;
393*b843c749SSergey Zigachev }
394*b843c749SSergey Zigachev 
dal_i2caux_acquire_aux_engine(struct i2caux * i2caux,struct ddc * ddc)395*b843c749SSergey Zigachev struct aux_engine *dal_i2caux_acquire_aux_engine(
396*b843c749SSergey Zigachev 	struct i2caux *i2caux,
397*b843c749SSergey Zigachev 	struct ddc *ddc)
398*b843c749SSergey Zigachev {
399*b843c749SSergey Zigachev 	enum gpio_ddc_line line;
400*b843c749SSergey Zigachev 	struct aux_engine *engine;
401*b843c749SSergey Zigachev 
402*b843c749SSergey Zigachev 	if (!get_hw_supported_ddc_line(ddc, &line))
403*b843c749SSergey Zigachev 		return NULL;
404*b843c749SSergey Zigachev 
405*b843c749SSergey Zigachev 	engine = i2caux->aux_engines[line];
406*b843c749SSergey Zigachev 
407*b843c749SSergey Zigachev 	if (!engine)
408*b843c749SSergey Zigachev 		return NULL;
409*b843c749SSergey Zigachev 
410*b843c749SSergey Zigachev 	if (!engine->base.funcs->acquire(&engine->base, ddc))
411*b843c749SSergey Zigachev 		return NULL;
412*b843c749SSergey Zigachev 
413*b843c749SSergey Zigachev 	return engine;
414*b843c749SSergey Zigachev }
415*b843c749SSergey Zigachev 
dal_i2caux_release_engine(struct i2caux * i2caux,struct engine * engine)416*b843c749SSergey Zigachev void dal_i2caux_release_engine(
417*b843c749SSergey Zigachev 	struct i2caux *i2caux,
418*b843c749SSergey Zigachev 	struct engine *engine)
419*b843c749SSergey Zigachev {
420*b843c749SSergey Zigachev 	engine->funcs->release_engine(engine);
421*b843c749SSergey Zigachev 
422*b843c749SSergey Zigachev 	dal_ddc_close(engine->ddc);
423*b843c749SSergey Zigachev 
424*b843c749SSergey Zigachev 	engine->ddc = NULL;
425*b843c749SSergey Zigachev }
426*b843c749SSergey Zigachev 
dal_i2caux_construct(struct i2caux * i2caux,struct dc_context * ctx)427*b843c749SSergey Zigachev void dal_i2caux_construct(
428*b843c749SSergey Zigachev 	struct i2caux *i2caux,
429*b843c749SSergey Zigachev 	struct dc_context *ctx)
430*b843c749SSergey Zigachev {
431*b843c749SSergey Zigachev 	uint32_t i = 0;
432*b843c749SSergey Zigachev 
433*b843c749SSergey Zigachev 	i2caux->ctx = ctx;
434*b843c749SSergey Zigachev 	do {
435*b843c749SSergey Zigachev 		i2caux->i2c_sw_engines[i] = NULL;
436*b843c749SSergey Zigachev 		i2caux->i2c_hw_engines[i] = NULL;
437*b843c749SSergey Zigachev 		i2caux->aux_engines[i] = NULL;
438*b843c749SSergey Zigachev 
439*b843c749SSergey Zigachev 		++i;
440*b843c749SSergey Zigachev 	} while (i < GPIO_DDC_LINE_COUNT);
441*b843c749SSergey Zigachev 
442*b843c749SSergey Zigachev 	i2caux->i2c_generic_sw_engine = NULL;
443*b843c749SSergey Zigachev 	i2caux->i2c_generic_hw_engine = NULL;
444*b843c749SSergey Zigachev 
445*b843c749SSergey Zigachev 	i2caux->aux_timeout_period =
446*b843c749SSergey Zigachev 		SW_AUX_TIMEOUT_PERIOD_MULTIPLIER * AUX_TIMEOUT_PERIOD;
447*b843c749SSergey Zigachev 
448*b843c749SSergey Zigachev 	if (ctx->dce_version >= DCE_VERSION_11_2) {
449*b843c749SSergey Zigachev 		i2caux->default_i2c_hw_speed = DEFAULT_I2C_HW_SPEED_100KHZ;
450*b843c749SSergey Zigachev 		i2caux->default_i2c_sw_speed = DEFAULT_I2C_SW_SPEED_100KHZ;
451*b843c749SSergey Zigachev 	} else {
452*b843c749SSergey Zigachev 		i2caux->default_i2c_hw_speed = DEFAULT_I2C_HW_SPEED;
453*b843c749SSergey Zigachev 		i2caux->default_i2c_sw_speed = DEFAULT_I2C_SW_SPEED;
454*b843c749SSergey Zigachev 	}
455*b843c749SSergey Zigachev }
456*b843c749SSergey Zigachev 
dal_i2caux_destruct(struct i2caux * i2caux)457*b843c749SSergey Zigachev void dal_i2caux_destruct(
458*b843c749SSergey Zigachev 	struct i2caux *i2caux)
459*b843c749SSergey Zigachev {
460*b843c749SSergey Zigachev 	uint32_t i = 0;
461*b843c749SSergey Zigachev 
462*b843c749SSergey Zigachev 	if (i2caux->i2c_generic_hw_engine)
463*b843c749SSergey Zigachev 		i2caux->i2c_generic_hw_engine->funcs->destroy(
464*b843c749SSergey Zigachev 			&i2caux->i2c_generic_hw_engine);
465*b843c749SSergey Zigachev 
466*b843c749SSergey Zigachev 	if (i2caux->i2c_generic_sw_engine)
467*b843c749SSergey Zigachev 		i2caux->i2c_generic_sw_engine->funcs->destroy(
468*b843c749SSergey Zigachev 			&i2caux->i2c_generic_sw_engine);
469*b843c749SSergey Zigachev 
470*b843c749SSergey Zigachev 	do {
471*b843c749SSergey Zigachev 		if (i2caux->aux_engines[i])
472*b843c749SSergey Zigachev 			i2caux->aux_engines[i]->funcs->destroy(
473*b843c749SSergey Zigachev 				&i2caux->aux_engines[i]);
474*b843c749SSergey Zigachev 
475*b843c749SSergey Zigachev 		if (i2caux->i2c_hw_engines[i])
476*b843c749SSergey Zigachev 			i2caux->i2c_hw_engines[i]->funcs->destroy(
477*b843c749SSergey Zigachev 				&i2caux->i2c_hw_engines[i]);
478*b843c749SSergey Zigachev 
479*b843c749SSergey Zigachev 		if (i2caux->i2c_sw_engines[i])
480*b843c749SSergey Zigachev 			i2caux->i2c_sw_engines[i]->funcs->destroy(
481*b843c749SSergey Zigachev 				&i2caux->i2c_sw_engines[i]);
482*b843c749SSergey Zigachev 
483*b843c749SSergey Zigachev 		++i;
484*b843c749SSergey Zigachev 	} while (i < GPIO_DDC_LINE_COUNT);
485*b843c749SSergey Zigachev }
486*b843c749SSergey Zigachev 
487