xref: /openbsd-src/sys/dev/pci/drm/amd/display/modules/hdcp/hdcp.c (revision 1bb76ff151c0aba8e3312a604e4cd2e5195cf4b7)
1c349dbc7Sjsg /*
2c349dbc7Sjsg  * Copyright 2019 Advanced Micro Devices, Inc.
3c349dbc7Sjsg  *
4c349dbc7Sjsg  * Permission is hereby granted, free of charge, to any person obtaining a
5c349dbc7Sjsg  * copy of this software and associated documentation files (the "Software"),
6c349dbc7Sjsg  * to deal in the Software without restriction, including without limitation
7c349dbc7Sjsg  * the rights to use, copy, modify, merge, publish, distribute, sublicense,
8c349dbc7Sjsg  * and/or sell copies of the Software, and to permit persons to whom the
9c349dbc7Sjsg  * Software is furnished to do so, subject to the following conditions:
10c349dbc7Sjsg  *
11c349dbc7Sjsg  * The above copyright notice and this permission notice shall be included in
12c349dbc7Sjsg  * all copies or substantial portions of the Software.
13c349dbc7Sjsg  *
14c349dbc7Sjsg  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
15c349dbc7Sjsg  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
16c349dbc7Sjsg  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL
17c349dbc7Sjsg  * THE COPYRIGHT HOLDER(S) OR AUTHOR(S) BE LIABLE FOR ANY CLAIM, DAMAGES OR
18c349dbc7Sjsg  * OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
19c349dbc7Sjsg  * ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
20c349dbc7Sjsg  * OTHER DEALINGS IN THE SOFTWARE.
21c349dbc7Sjsg  *
22c349dbc7Sjsg  * Authors: AMD
23c349dbc7Sjsg  *
24c349dbc7Sjsg  */
25c349dbc7Sjsg 
26c349dbc7Sjsg #include "hdcp.h"
27c349dbc7Sjsg 
push_error_status(struct mod_hdcp * hdcp,enum mod_hdcp_status status)28c349dbc7Sjsg static void push_error_status(struct mod_hdcp *hdcp,
29c349dbc7Sjsg 		enum mod_hdcp_status status)
30c349dbc7Sjsg {
31c349dbc7Sjsg 	struct mod_hdcp_trace *trace = &hdcp->connection.trace;
32c349dbc7Sjsg 
33c349dbc7Sjsg 	if (trace->error_count < MAX_NUM_OF_ERROR_TRACE) {
34c349dbc7Sjsg 		trace->errors[trace->error_count].status = status;
35c349dbc7Sjsg 		trace->errors[trace->error_count].state_id = hdcp->state.id;
36c349dbc7Sjsg 		trace->error_count++;
37c349dbc7Sjsg 		HDCP_ERROR_TRACE(hdcp, status);
38c349dbc7Sjsg 	}
39c349dbc7Sjsg 
40c349dbc7Sjsg 	if (is_hdcp1(hdcp)) {
41c349dbc7Sjsg 		hdcp->connection.hdcp1_retry_count++;
425ca02815Sjsg 		if (hdcp->connection.hdcp1_retry_count == MAX_NUM_OF_ATTEMPTS)
435ca02815Sjsg 			hdcp->connection.link.adjust.hdcp1.disable = 1;
44c349dbc7Sjsg 	} else if (is_hdcp2(hdcp)) {
45c349dbc7Sjsg 		hdcp->connection.hdcp2_retry_count++;
465ca02815Sjsg 		if (hdcp->connection.hdcp2_retry_count == MAX_NUM_OF_ATTEMPTS)
475ca02815Sjsg 			hdcp->connection.link.adjust.hdcp2.disable = 1;
48c349dbc7Sjsg 	}
49c349dbc7Sjsg }
50c349dbc7Sjsg 
is_cp_desired_hdcp1(struct mod_hdcp * hdcp)51c349dbc7Sjsg static uint8_t is_cp_desired_hdcp1(struct mod_hdcp *hdcp)
52c349dbc7Sjsg {
53c349dbc7Sjsg 	int i, is_auth_needed = 0;
54c349dbc7Sjsg 
55c349dbc7Sjsg 	/* if all displays on the link don't need authentication,
56c349dbc7Sjsg 	 * hdcp is not desired
57c349dbc7Sjsg 	 */
58c349dbc7Sjsg 	for (i = 0; i < MAX_NUM_OF_DISPLAYS; i++) {
59c349dbc7Sjsg 		if (hdcp->displays[i].state != MOD_HDCP_DISPLAY_INACTIVE &&
605ca02815Sjsg 				hdcp->displays[i].adjust.disable != MOD_HDCP_DISPLAY_DISABLE_AUTHENTICATION) {
61c349dbc7Sjsg 			is_auth_needed = 1;
62c349dbc7Sjsg 			break;
63c349dbc7Sjsg 		}
64c349dbc7Sjsg 	}
65c349dbc7Sjsg 
665ca02815Sjsg 	return is_auth_needed &&
67c349dbc7Sjsg 			!hdcp->connection.link.adjust.hdcp1.disable &&
68c349dbc7Sjsg 			!hdcp->connection.is_hdcp1_revoked;
69c349dbc7Sjsg }
70c349dbc7Sjsg 
is_cp_desired_hdcp2(struct mod_hdcp * hdcp)71c349dbc7Sjsg static uint8_t is_cp_desired_hdcp2(struct mod_hdcp *hdcp)
72c349dbc7Sjsg {
73c349dbc7Sjsg 	int i, is_auth_needed = 0;
74c349dbc7Sjsg 
75c349dbc7Sjsg 	/* if all displays on the link don't need authentication,
76c349dbc7Sjsg 	 * hdcp is not desired
77c349dbc7Sjsg 	 */
78c349dbc7Sjsg 	for (i = 0; i < MAX_NUM_OF_DISPLAYS; i++) {
79c349dbc7Sjsg 		if (hdcp->displays[i].state != MOD_HDCP_DISPLAY_INACTIVE &&
805ca02815Sjsg 				hdcp->displays[i].adjust.disable != MOD_HDCP_DISPLAY_DISABLE_AUTHENTICATION) {
81c349dbc7Sjsg 			is_auth_needed = 1;
82c349dbc7Sjsg 			break;
83c349dbc7Sjsg 		}
84c349dbc7Sjsg 	}
85c349dbc7Sjsg 
865ca02815Sjsg 	return is_auth_needed &&
87c349dbc7Sjsg 			!hdcp->connection.link.adjust.hdcp2.disable &&
88c349dbc7Sjsg 			!hdcp->connection.is_hdcp2_revoked;
89c349dbc7Sjsg }
90c349dbc7Sjsg 
execution(struct mod_hdcp * hdcp,struct mod_hdcp_event_context * event_ctx,union mod_hdcp_transition_input * input)91c349dbc7Sjsg static enum mod_hdcp_status execution(struct mod_hdcp *hdcp,
92c349dbc7Sjsg 		struct mod_hdcp_event_context *event_ctx,
93c349dbc7Sjsg 		union mod_hdcp_transition_input *input)
94c349dbc7Sjsg {
95c349dbc7Sjsg 	enum mod_hdcp_status status = MOD_HDCP_STATUS_SUCCESS;
96c349dbc7Sjsg 
97c349dbc7Sjsg 	if (is_in_initialized_state(hdcp)) {
98c349dbc7Sjsg 		if (event_ctx->event != MOD_HDCP_EVENT_CALLBACK) {
99c349dbc7Sjsg 			event_ctx->unexpected_event = 1;
100c349dbc7Sjsg 			goto out;
101c349dbc7Sjsg 		}
102c349dbc7Sjsg 		/* initialize transition input */
103c349dbc7Sjsg 		memset(input, 0, sizeof(union mod_hdcp_transition_input));
104c349dbc7Sjsg 	} else if (is_in_cp_not_desired_state(hdcp)) {
105c349dbc7Sjsg 		if (event_ctx->event != MOD_HDCP_EVENT_CALLBACK) {
106c349dbc7Sjsg 			event_ctx->unexpected_event = 1;
107c349dbc7Sjsg 			goto out;
108c349dbc7Sjsg 		}
109c349dbc7Sjsg 	} else if (is_in_hdcp1_states(hdcp)) {
110c349dbc7Sjsg 		status = mod_hdcp_hdcp1_execution(hdcp, event_ctx, &input->hdcp1);
111c349dbc7Sjsg 	} else if (is_in_hdcp1_dp_states(hdcp)) {
112c349dbc7Sjsg 		status = mod_hdcp_hdcp1_dp_execution(hdcp,
113c349dbc7Sjsg 				event_ctx, &input->hdcp1);
114c349dbc7Sjsg 	} else if (is_in_hdcp2_states(hdcp)) {
115c349dbc7Sjsg 		status = mod_hdcp_hdcp2_execution(hdcp, event_ctx, &input->hdcp2);
116c349dbc7Sjsg 	} else if (is_in_hdcp2_dp_states(hdcp)) {
117c349dbc7Sjsg 		status = mod_hdcp_hdcp2_dp_execution(hdcp,
118c349dbc7Sjsg 				event_ctx, &input->hdcp2);
119c349dbc7Sjsg 	} else {
120c349dbc7Sjsg 		event_ctx->unexpected_event = 1;
121c349dbc7Sjsg 		goto out;
122c349dbc7Sjsg 	}
123c349dbc7Sjsg out:
124c349dbc7Sjsg 	return status;
125c349dbc7Sjsg }
126c349dbc7Sjsg 
transition(struct mod_hdcp * hdcp,struct mod_hdcp_event_context * event_ctx,union mod_hdcp_transition_input * input,struct mod_hdcp_output * output)127c349dbc7Sjsg static enum mod_hdcp_status transition(struct mod_hdcp *hdcp,
128c349dbc7Sjsg 		struct mod_hdcp_event_context *event_ctx,
129c349dbc7Sjsg 		union mod_hdcp_transition_input *input,
130c349dbc7Sjsg 		struct mod_hdcp_output *output)
131c349dbc7Sjsg {
132c349dbc7Sjsg 	enum mod_hdcp_status status = MOD_HDCP_STATUS_SUCCESS;
133c349dbc7Sjsg 
134c349dbc7Sjsg 	if (event_ctx->unexpected_event)
135c349dbc7Sjsg 		goto out;
136c349dbc7Sjsg 
137c349dbc7Sjsg 	if (is_in_initialized_state(hdcp)) {
138c349dbc7Sjsg 		if (is_dp_hdcp(hdcp))
139c349dbc7Sjsg 			if (is_cp_desired_hdcp2(hdcp)) {
140c349dbc7Sjsg 				callback_in_ms(0, output);
141c349dbc7Sjsg 				set_state_id(hdcp, output, D2_A0_DETERMINE_RX_HDCP_CAPABLE);
142c349dbc7Sjsg 			} else if (is_cp_desired_hdcp1(hdcp)) {
143c349dbc7Sjsg 				callback_in_ms(0, output);
144c349dbc7Sjsg 				set_state_id(hdcp, output, D1_A0_DETERMINE_RX_HDCP_CAPABLE);
145c349dbc7Sjsg 			} else {
146c349dbc7Sjsg 				callback_in_ms(0, output);
147c349dbc7Sjsg 				set_state_id(hdcp, output, HDCP_CP_NOT_DESIRED);
1485ca02815Sjsg 				set_auth_complete(hdcp, output);
149c349dbc7Sjsg 			}
150c349dbc7Sjsg 		else if (is_hdmi_dvi_sl_hdcp(hdcp))
151c349dbc7Sjsg 			if (is_cp_desired_hdcp2(hdcp)) {
152c349dbc7Sjsg 				callback_in_ms(0, output);
153c349dbc7Sjsg 				set_state_id(hdcp, output, H2_A0_KNOWN_HDCP2_CAPABLE_RX);
154c349dbc7Sjsg 			} else if (is_cp_desired_hdcp1(hdcp)) {
155c349dbc7Sjsg 				callback_in_ms(0, output);
156c349dbc7Sjsg 				set_state_id(hdcp, output, H1_A0_WAIT_FOR_ACTIVE_RX);
157c349dbc7Sjsg 			} else {
158c349dbc7Sjsg 				callback_in_ms(0, output);
159c349dbc7Sjsg 				set_state_id(hdcp, output, HDCP_CP_NOT_DESIRED);
1605ca02815Sjsg 				set_auth_complete(hdcp, output);
161c349dbc7Sjsg 			}
162c349dbc7Sjsg 		else {
163c349dbc7Sjsg 			callback_in_ms(0, output);
164c349dbc7Sjsg 			set_state_id(hdcp, output, HDCP_CP_NOT_DESIRED);
1655ca02815Sjsg 			set_auth_complete(hdcp, output);
166c349dbc7Sjsg 		}
167c349dbc7Sjsg 	} else if (is_in_cp_not_desired_state(hdcp)) {
168c349dbc7Sjsg 		increment_stay_counter(hdcp);
169c349dbc7Sjsg 	} else if (is_in_hdcp1_states(hdcp)) {
170c349dbc7Sjsg 		status = mod_hdcp_hdcp1_transition(hdcp,
171c349dbc7Sjsg 				event_ctx, &input->hdcp1, output);
172c349dbc7Sjsg 	} else if (is_in_hdcp1_dp_states(hdcp)) {
173c349dbc7Sjsg 		status = mod_hdcp_hdcp1_dp_transition(hdcp,
174c349dbc7Sjsg 				event_ctx, &input->hdcp1, output);
175c349dbc7Sjsg 	} else if (is_in_hdcp2_states(hdcp)) {
176c349dbc7Sjsg 		status = mod_hdcp_hdcp2_transition(hdcp,
177c349dbc7Sjsg 				event_ctx, &input->hdcp2, output);
178c349dbc7Sjsg 	} else if (is_in_hdcp2_dp_states(hdcp)) {
179c349dbc7Sjsg 		status = mod_hdcp_hdcp2_dp_transition(hdcp,
180c349dbc7Sjsg 				event_ctx, &input->hdcp2, output);
181c349dbc7Sjsg 	} else {
182c349dbc7Sjsg 		status = MOD_HDCP_STATUS_INVALID_STATE;
183c349dbc7Sjsg 	}
184c349dbc7Sjsg out:
185c349dbc7Sjsg 	return status;
186c349dbc7Sjsg }
187c349dbc7Sjsg 
reset_authentication(struct mod_hdcp * hdcp,struct mod_hdcp_output * output)188c349dbc7Sjsg static enum mod_hdcp_status reset_authentication(struct mod_hdcp *hdcp,
189c349dbc7Sjsg 		struct mod_hdcp_output *output)
190c349dbc7Sjsg {
191c349dbc7Sjsg 	enum mod_hdcp_status status = MOD_HDCP_STATUS_SUCCESS;
192c349dbc7Sjsg 
193c349dbc7Sjsg 	if (is_hdcp1(hdcp)) {
194c349dbc7Sjsg 		if (hdcp->auth.trans_input.hdcp1.create_session != UNKNOWN) {
195c349dbc7Sjsg 			/* TODO - update psp to unify create session failure
196c349dbc7Sjsg 			 * recovery between hdcp1 and 2.
197c349dbc7Sjsg 			 */
198c349dbc7Sjsg 			mod_hdcp_hdcp1_destroy_session(hdcp);
199c349dbc7Sjsg 
200c349dbc7Sjsg 		}
201c349dbc7Sjsg 
202c349dbc7Sjsg 		HDCP_TOP_RESET_AUTH_TRACE(hdcp);
203c349dbc7Sjsg 		memset(&hdcp->auth, 0, sizeof(struct mod_hdcp_authentication));
204c349dbc7Sjsg 		memset(&hdcp->state, 0, sizeof(struct mod_hdcp_state));
205c349dbc7Sjsg 		set_state_id(hdcp, output, HDCP_INITIALIZED);
206c349dbc7Sjsg 	} else if (is_hdcp2(hdcp)) {
207c349dbc7Sjsg 		if (hdcp->auth.trans_input.hdcp2.create_session == PASS) {
208c349dbc7Sjsg 			status = mod_hdcp_hdcp2_destroy_session(hdcp);
209c349dbc7Sjsg 			if (status != MOD_HDCP_STATUS_SUCCESS) {
210c349dbc7Sjsg 				output->callback_needed = 0;
211c349dbc7Sjsg 				output->watchdog_timer_needed = 0;
212c349dbc7Sjsg 				goto out;
213c349dbc7Sjsg 			}
214c349dbc7Sjsg 		}
215c349dbc7Sjsg 
216c349dbc7Sjsg 		HDCP_TOP_RESET_AUTH_TRACE(hdcp);
217c349dbc7Sjsg 		memset(&hdcp->auth, 0, sizeof(struct mod_hdcp_authentication));
218c349dbc7Sjsg 		memset(&hdcp->state, 0, sizeof(struct mod_hdcp_state));
219c349dbc7Sjsg 		set_state_id(hdcp, output, HDCP_INITIALIZED);
220c349dbc7Sjsg 	} else if (is_in_cp_not_desired_state(hdcp)) {
221c349dbc7Sjsg 		HDCP_TOP_RESET_AUTH_TRACE(hdcp);
222c349dbc7Sjsg 		memset(&hdcp->auth, 0, sizeof(struct mod_hdcp_authentication));
223c349dbc7Sjsg 		memset(&hdcp->state, 0, sizeof(struct mod_hdcp_state));
224c349dbc7Sjsg 		set_state_id(hdcp, output, HDCP_INITIALIZED);
225c349dbc7Sjsg 	}
226c349dbc7Sjsg 
227c349dbc7Sjsg out:
228c349dbc7Sjsg 	/* stop callback and watchdog requests from previous authentication*/
229c349dbc7Sjsg 	output->watchdog_timer_stop = 1;
230c349dbc7Sjsg 	output->callback_stop = 1;
231c349dbc7Sjsg 	return status;
232c349dbc7Sjsg }
233c349dbc7Sjsg 
reset_connection(struct mod_hdcp * hdcp,struct mod_hdcp_output * output)234c349dbc7Sjsg static enum mod_hdcp_status reset_connection(struct mod_hdcp *hdcp,
235c349dbc7Sjsg 		struct mod_hdcp_output *output)
236c349dbc7Sjsg {
237c349dbc7Sjsg 	enum mod_hdcp_status status = MOD_HDCP_STATUS_SUCCESS;
238c349dbc7Sjsg 
239c349dbc7Sjsg 	memset(output, 0, sizeof(struct mod_hdcp_output));
240c349dbc7Sjsg 
241c349dbc7Sjsg 	status = reset_authentication(hdcp, output);
242c349dbc7Sjsg 	if (status != MOD_HDCP_STATUS_SUCCESS)
243c349dbc7Sjsg 		goto out;
244c349dbc7Sjsg 
245c349dbc7Sjsg 	if (current_state(hdcp) != HDCP_UNINITIALIZED) {
246c349dbc7Sjsg 		HDCP_TOP_RESET_CONN_TRACE(hdcp);
247c349dbc7Sjsg 		set_state_id(hdcp, output, HDCP_UNINITIALIZED);
248c349dbc7Sjsg 	}
249c349dbc7Sjsg 	memset(&hdcp->connection, 0, sizeof(hdcp->connection));
250c349dbc7Sjsg out:
251c349dbc7Sjsg 	return status;
252c349dbc7Sjsg }
253c349dbc7Sjsg 
update_display_adjustments(struct mod_hdcp * hdcp,struct mod_hdcp_display * display,struct mod_hdcp_display_adjustment * adj)254*1bb76ff1Sjsg static enum mod_hdcp_status update_display_adjustments(struct mod_hdcp *hdcp,
255*1bb76ff1Sjsg 		struct mod_hdcp_display *display,
256*1bb76ff1Sjsg 		struct mod_hdcp_display_adjustment *adj)
257*1bb76ff1Sjsg {
258*1bb76ff1Sjsg 	enum mod_hdcp_status status = MOD_HDCP_STATUS_NOT_IMPLEMENTED;
259*1bb76ff1Sjsg 
260*1bb76ff1Sjsg 	if (is_in_authenticated_states(hdcp) &&
261*1bb76ff1Sjsg 			is_dp_mst_hdcp(hdcp) &&
262*1bb76ff1Sjsg 			display->adjust.disable == true &&
263*1bb76ff1Sjsg 			adj->disable == false) {
264*1bb76ff1Sjsg 		display->adjust.disable = false;
265*1bb76ff1Sjsg 		if (is_hdcp1(hdcp))
266*1bb76ff1Sjsg 			status = mod_hdcp_hdcp1_enable_dp_stream_encryption(hdcp);
267*1bb76ff1Sjsg 		else if (is_hdcp2(hdcp))
268*1bb76ff1Sjsg 			status = mod_hdcp_hdcp2_enable_dp_stream_encryption(hdcp);
269*1bb76ff1Sjsg 
270*1bb76ff1Sjsg 		if (status != MOD_HDCP_STATUS_SUCCESS)
271*1bb76ff1Sjsg 			display->adjust.disable = true;
272*1bb76ff1Sjsg 	}
273*1bb76ff1Sjsg 
274*1bb76ff1Sjsg 	if (status == MOD_HDCP_STATUS_SUCCESS &&
275*1bb76ff1Sjsg 		memcmp(adj, &display->adjust,
276*1bb76ff1Sjsg 		sizeof(struct mod_hdcp_display_adjustment)) != 0)
277*1bb76ff1Sjsg 		status = MOD_HDCP_STATUS_NOT_IMPLEMENTED;
278*1bb76ff1Sjsg 
279*1bb76ff1Sjsg 	return status;
280*1bb76ff1Sjsg }
281c349dbc7Sjsg /*
282c349dbc7Sjsg  * Implementation of functions in mod_hdcp.h
283c349dbc7Sjsg  */
mod_hdcp_get_memory_size(void)284c349dbc7Sjsg size_t mod_hdcp_get_memory_size(void)
285c349dbc7Sjsg {
286c349dbc7Sjsg 	return sizeof(struct mod_hdcp);
287c349dbc7Sjsg }
288c349dbc7Sjsg 
mod_hdcp_setup(struct mod_hdcp * hdcp,struct mod_hdcp_config * config)289c349dbc7Sjsg enum mod_hdcp_status mod_hdcp_setup(struct mod_hdcp *hdcp,
290c349dbc7Sjsg 		struct mod_hdcp_config *config)
291c349dbc7Sjsg {
292c349dbc7Sjsg 	struct mod_hdcp_output output;
293c349dbc7Sjsg 	enum mod_hdcp_status status = MOD_HDCP_STATUS_SUCCESS;
294c349dbc7Sjsg 
295c349dbc7Sjsg 	memset(&output, 0, sizeof(output));
296c349dbc7Sjsg 	hdcp->config = *config;
297c349dbc7Sjsg 	HDCP_TOP_INTERFACE_TRACE(hdcp);
298c349dbc7Sjsg 	status = reset_connection(hdcp, &output);
299c349dbc7Sjsg 	if (status != MOD_HDCP_STATUS_SUCCESS)
300c349dbc7Sjsg 		push_error_status(hdcp, status);
301c349dbc7Sjsg 	return status;
302c349dbc7Sjsg }
303c349dbc7Sjsg 
mod_hdcp_teardown(struct mod_hdcp * hdcp)304c349dbc7Sjsg enum mod_hdcp_status mod_hdcp_teardown(struct mod_hdcp *hdcp)
305c349dbc7Sjsg {
306c349dbc7Sjsg 	enum mod_hdcp_status status = MOD_HDCP_STATUS_SUCCESS;
307c349dbc7Sjsg 	struct mod_hdcp_output output;
308c349dbc7Sjsg 
309c349dbc7Sjsg 	HDCP_TOP_INTERFACE_TRACE(hdcp);
310c349dbc7Sjsg 	memset(&output, 0,  sizeof(output));
311c349dbc7Sjsg 	status = reset_connection(hdcp, &output);
312c349dbc7Sjsg 	if (status == MOD_HDCP_STATUS_SUCCESS)
313c349dbc7Sjsg 		memset(hdcp, 0, sizeof(struct mod_hdcp));
314c349dbc7Sjsg 	else
315c349dbc7Sjsg 		push_error_status(hdcp, status);
316c349dbc7Sjsg 	return status;
317c349dbc7Sjsg }
318c349dbc7Sjsg 
mod_hdcp_add_display(struct mod_hdcp * hdcp,struct mod_hdcp_link * link,struct mod_hdcp_display * display,struct mod_hdcp_output * output)319c349dbc7Sjsg enum mod_hdcp_status mod_hdcp_add_display(struct mod_hdcp *hdcp,
320c349dbc7Sjsg 		struct mod_hdcp_link *link, struct mod_hdcp_display *display,
321c349dbc7Sjsg 		struct mod_hdcp_output *output)
322c349dbc7Sjsg {
323c349dbc7Sjsg 	enum mod_hdcp_status status = MOD_HDCP_STATUS_SUCCESS;
324c349dbc7Sjsg 	struct mod_hdcp_display *display_container = NULL;
325c349dbc7Sjsg 
326c349dbc7Sjsg 	HDCP_TOP_INTERFACE_TRACE_WITH_INDEX(hdcp, display->index);
327c349dbc7Sjsg 	memset(output, 0, sizeof(struct mod_hdcp_output));
328c349dbc7Sjsg 
329c349dbc7Sjsg 	/* skip inactive display */
330c349dbc7Sjsg 	if (display->state != MOD_HDCP_DISPLAY_ACTIVE) {
331c349dbc7Sjsg 		status = MOD_HDCP_STATUS_SUCCESS;
332c349dbc7Sjsg 		goto out;
333c349dbc7Sjsg 	}
334c349dbc7Sjsg 
335c349dbc7Sjsg 	/* check existing display container */
336c349dbc7Sjsg 	if (get_active_display_at_index(hdcp, display->index)) {
337c349dbc7Sjsg 		status = MOD_HDCP_STATUS_SUCCESS;
338c349dbc7Sjsg 		goto out;
339c349dbc7Sjsg 	}
340c349dbc7Sjsg 
341c349dbc7Sjsg 	/* find an empty display container */
342c349dbc7Sjsg 	display_container = get_empty_display_container(hdcp);
343c349dbc7Sjsg 	if (!display_container) {
344c349dbc7Sjsg 		status = MOD_HDCP_STATUS_DISPLAY_OUT_OF_BOUND;
345c349dbc7Sjsg 		goto out;
346c349dbc7Sjsg 	}
347c349dbc7Sjsg 
348c349dbc7Sjsg 	/* reset existing authentication status */
349c349dbc7Sjsg 	status = reset_authentication(hdcp, output);
350c349dbc7Sjsg 	if (status != MOD_HDCP_STATUS_SUCCESS)
351c349dbc7Sjsg 		goto out;
352c349dbc7Sjsg 
353c349dbc7Sjsg 	/* reset retry counters */
354c349dbc7Sjsg 	reset_retry_counts(hdcp);
355c349dbc7Sjsg 
356c349dbc7Sjsg 	/* reset error trace */
357c349dbc7Sjsg 	memset(&hdcp->connection.trace, 0, sizeof(hdcp->connection.trace));
358c349dbc7Sjsg 
359c349dbc7Sjsg 	/* add display to connection */
360c349dbc7Sjsg 	hdcp->connection.link = *link;
361c349dbc7Sjsg 	*display_container = *display;
362ad8b1aafSjsg 	status = mod_hdcp_add_display_to_topology(hdcp, display_container);
363ad8b1aafSjsg 
364c349dbc7Sjsg 	if (status != MOD_HDCP_STATUS_SUCCESS)
365c349dbc7Sjsg 		goto out;
366c349dbc7Sjsg 
367c349dbc7Sjsg 	/* request authentication */
368c349dbc7Sjsg 	if (current_state(hdcp) != HDCP_INITIALIZED)
369c349dbc7Sjsg 		set_state_id(hdcp, output, HDCP_INITIALIZED);
370c349dbc7Sjsg 	callback_in_ms(hdcp->connection.link.adjust.auth_delay * 1000, output);
371c349dbc7Sjsg out:
372c349dbc7Sjsg 	if (status != MOD_HDCP_STATUS_SUCCESS)
373c349dbc7Sjsg 		push_error_status(hdcp, status);
374c349dbc7Sjsg 
375c349dbc7Sjsg 	return status;
376c349dbc7Sjsg }
377c349dbc7Sjsg 
mod_hdcp_remove_display(struct mod_hdcp * hdcp,uint8_t index,struct mod_hdcp_output * output)378c349dbc7Sjsg enum mod_hdcp_status mod_hdcp_remove_display(struct mod_hdcp *hdcp,
379c349dbc7Sjsg 		uint8_t index, struct mod_hdcp_output *output)
380c349dbc7Sjsg {
381c349dbc7Sjsg 	enum mod_hdcp_status status = MOD_HDCP_STATUS_SUCCESS;
382c349dbc7Sjsg 	struct mod_hdcp_display *display = NULL;
383c349dbc7Sjsg 
384c349dbc7Sjsg 	HDCP_TOP_INTERFACE_TRACE_WITH_INDEX(hdcp, index);
385c349dbc7Sjsg 	memset(output, 0, sizeof(struct mod_hdcp_output));
386c349dbc7Sjsg 
387c349dbc7Sjsg 	/* find display in connection */
388c349dbc7Sjsg 	display = get_active_display_at_index(hdcp, index);
389c349dbc7Sjsg 	if (!display) {
390c349dbc7Sjsg 		status = MOD_HDCP_STATUS_SUCCESS;
391c349dbc7Sjsg 		goto out;
392c349dbc7Sjsg 	}
393c349dbc7Sjsg 
394c349dbc7Sjsg 	/* stop current authentication */
395c349dbc7Sjsg 	status = reset_authentication(hdcp, output);
396c349dbc7Sjsg 	if (status != MOD_HDCP_STATUS_SUCCESS)
397c349dbc7Sjsg 		goto out;
398c349dbc7Sjsg 
399c349dbc7Sjsg 	/* clear retry counters */
400c349dbc7Sjsg 	reset_retry_counts(hdcp);
401c349dbc7Sjsg 
402c349dbc7Sjsg 	/* reset error trace */
403c349dbc7Sjsg 	memset(&hdcp->connection.trace, 0, sizeof(hdcp->connection.trace));
404c349dbc7Sjsg 
405c349dbc7Sjsg 	/* remove display */
406c349dbc7Sjsg 	status = mod_hdcp_remove_display_from_topology(hdcp, index);
407c349dbc7Sjsg 	if (status != MOD_HDCP_STATUS_SUCCESS)
408c349dbc7Sjsg 		goto out;
409ad8b1aafSjsg 	memset(display, 0, sizeof(struct mod_hdcp_display));
410c349dbc7Sjsg 
411c349dbc7Sjsg 	/* request authentication when connection is not reset */
412c349dbc7Sjsg 	if (current_state(hdcp) != HDCP_UNINITIALIZED)
413c349dbc7Sjsg 		callback_in_ms(hdcp->connection.link.adjust.auth_delay * 1000,
414c349dbc7Sjsg 				output);
415c349dbc7Sjsg out:
416c349dbc7Sjsg 	if (status != MOD_HDCP_STATUS_SUCCESS)
417c349dbc7Sjsg 		push_error_status(hdcp, status);
418c349dbc7Sjsg 	return status;
419c349dbc7Sjsg }
420c349dbc7Sjsg 
mod_hdcp_update_display(struct mod_hdcp * hdcp,uint8_t index,struct mod_hdcp_link_adjustment * link_adjust,struct mod_hdcp_display_adjustment * display_adjust,struct mod_hdcp_output * output)421*1bb76ff1Sjsg enum mod_hdcp_status mod_hdcp_update_display(struct mod_hdcp *hdcp,
4225ca02815Sjsg 		uint8_t index,
4235ca02815Sjsg 		struct mod_hdcp_link_adjustment *link_adjust,
4245ca02815Sjsg 		struct mod_hdcp_display_adjustment *display_adjust,
4255ca02815Sjsg 		struct mod_hdcp_output *output)
4265ca02815Sjsg {
4275ca02815Sjsg 	enum mod_hdcp_status status = MOD_HDCP_STATUS_SUCCESS;
4285ca02815Sjsg 	struct mod_hdcp_display *display = NULL;
4295ca02815Sjsg 
4305ca02815Sjsg 	HDCP_TOP_INTERFACE_TRACE_WITH_INDEX(hdcp, index);
4315ca02815Sjsg 	memset(output, 0, sizeof(struct mod_hdcp_output));
4325ca02815Sjsg 
4335ca02815Sjsg 	/* find display in connection */
4345ca02815Sjsg 	display = get_active_display_at_index(hdcp, index);
4355ca02815Sjsg 	if (!display) {
4365ca02815Sjsg 		status = MOD_HDCP_STATUS_DISPLAY_NOT_FOUND;
4375ca02815Sjsg 		goto out;
4385ca02815Sjsg 	}
4395ca02815Sjsg 
4405ca02815Sjsg 	/* skip if no changes */
4415ca02815Sjsg 	if (memcmp(link_adjust, &hdcp->connection.link.adjust,
4425ca02815Sjsg 			sizeof(struct mod_hdcp_link_adjustment)) == 0 &&
4435ca02815Sjsg 			memcmp(display_adjust, &display->adjust,
4445ca02815Sjsg 					sizeof(struct mod_hdcp_display_adjustment)) == 0) {
4455ca02815Sjsg 		status = MOD_HDCP_STATUS_SUCCESS;
4465ca02815Sjsg 		goto out;
4475ca02815Sjsg 	}
4485ca02815Sjsg 
449*1bb76ff1Sjsg 	if (memcmp(link_adjust, &hdcp->connection.link.adjust,
450*1bb76ff1Sjsg 			sizeof(struct mod_hdcp_link_adjustment)) == 0 &&
451*1bb76ff1Sjsg 			memcmp(display_adjust, &display->adjust,
452*1bb76ff1Sjsg 					sizeof(struct mod_hdcp_display_adjustment)) != 0) {
453*1bb76ff1Sjsg 		status = update_display_adjustments(hdcp, display, display_adjust);
454*1bb76ff1Sjsg 		if (status != MOD_HDCP_STATUS_NOT_IMPLEMENTED)
455*1bb76ff1Sjsg 			goto out;
456*1bb76ff1Sjsg 	}
457*1bb76ff1Sjsg 
4585ca02815Sjsg 	/* stop current authentication */
4595ca02815Sjsg 	status = reset_authentication(hdcp, output);
4605ca02815Sjsg 	if (status != MOD_HDCP_STATUS_SUCCESS)
4615ca02815Sjsg 		goto out;
4625ca02815Sjsg 
4635ca02815Sjsg 	/* clear retry counters */
4645ca02815Sjsg 	reset_retry_counts(hdcp);
4655ca02815Sjsg 
4665ca02815Sjsg 	/* reset error trace */
4675ca02815Sjsg 	memset(&hdcp->connection.trace, 0, sizeof(hdcp->connection.trace));
4685ca02815Sjsg 
4695ca02815Sjsg 	/* set new adjustment */
4705ca02815Sjsg 	hdcp->connection.link.adjust = *link_adjust;
4715ca02815Sjsg 	display->adjust = *display_adjust;
4725ca02815Sjsg 
4735ca02815Sjsg 	/* request authentication when connection is not reset */
4745ca02815Sjsg 	if (current_state(hdcp) != HDCP_UNINITIALIZED)
4755ca02815Sjsg 		/* wait 100ms to debounce simultaneous updates for different indices */
4765ca02815Sjsg 		callback_in_ms(100, output);
4775ca02815Sjsg 
4785ca02815Sjsg out:
4795ca02815Sjsg 	if (status != MOD_HDCP_STATUS_SUCCESS)
4805ca02815Sjsg 		push_error_status(hdcp, status);
4815ca02815Sjsg 	return status;
4825ca02815Sjsg }
4835ca02815Sjsg 
mod_hdcp_query_display(struct mod_hdcp * hdcp,uint8_t index,struct mod_hdcp_display_query * query)484c349dbc7Sjsg enum mod_hdcp_status mod_hdcp_query_display(struct mod_hdcp *hdcp,
485c349dbc7Sjsg 		uint8_t index, struct mod_hdcp_display_query *query)
486c349dbc7Sjsg {
487c349dbc7Sjsg 	enum mod_hdcp_status status = MOD_HDCP_STATUS_SUCCESS;
488c349dbc7Sjsg 	struct mod_hdcp_display *display = NULL;
489c349dbc7Sjsg 
490c349dbc7Sjsg 	/* find display in connection */
491c349dbc7Sjsg 	display = get_active_display_at_index(hdcp, index);
492c349dbc7Sjsg 	if (!display) {
493c349dbc7Sjsg 		status = MOD_HDCP_STATUS_DISPLAY_NOT_FOUND;
494c349dbc7Sjsg 		goto out;
495c349dbc7Sjsg 	}
496c349dbc7Sjsg 
497c349dbc7Sjsg 	/* populate query */
498c349dbc7Sjsg 	query->link = &hdcp->connection.link;
499c349dbc7Sjsg 	query->display = display;
500c349dbc7Sjsg 	query->trace = &hdcp->connection.trace;
501c349dbc7Sjsg 	query->encryption_status = MOD_HDCP_ENCRYPTION_STATUS_HDCP_OFF;
502c349dbc7Sjsg 
503c349dbc7Sjsg 	if (is_display_encryption_enabled(display)) {
504c349dbc7Sjsg 		if (is_hdcp1(hdcp)) {
505c349dbc7Sjsg 			query->encryption_status = MOD_HDCP_ENCRYPTION_STATUS_HDCP1_ON;
506c349dbc7Sjsg 		} else if (is_hdcp2(hdcp)) {
507c349dbc7Sjsg 			if (query->link->adjust.hdcp2.force_type == MOD_HDCP_FORCE_TYPE_0)
508c349dbc7Sjsg 				query->encryption_status = MOD_HDCP_ENCRYPTION_STATUS_HDCP2_TYPE0_ON;
509c349dbc7Sjsg 			else if (query->link->adjust.hdcp2.force_type == MOD_HDCP_FORCE_TYPE_1)
510c349dbc7Sjsg 				query->encryption_status = MOD_HDCP_ENCRYPTION_STATUS_HDCP2_TYPE1_ON;
511c349dbc7Sjsg 			else
512c349dbc7Sjsg 				query->encryption_status = MOD_HDCP_ENCRYPTION_STATUS_HDCP2_ON;
513c349dbc7Sjsg 		}
514c349dbc7Sjsg 	} else {
515c349dbc7Sjsg 		query->encryption_status = MOD_HDCP_ENCRYPTION_STATUS_HDCP_OFF;
516c349dbc7Sjsg 	}
517c349dbc7Sjsg 
518c349dbc7Sjsg out:
519c349dbc7Sjsg 	return status;
520c349dbc7Sjsg }
521c349dbc7Sjsg 
mod_hdcp_reset_connection(struct mod_hdcp * hdcp,struct mod_hdcp_output * output)522c349dbc7Sjsg enum mod_hdcp_status mod_hdcp_reset_connection(struct mod_hdcp *hdcp,
523c349dbc7Sjsg 		struct mod_hdcp_output *output)
524c349dbc7Sjsg {
525c349dbc7Sjsg 	enum mod_hdcp_status status = MOD_HDCP_STATUS_SUCCESS;
526c349dbc7Sjsg 
527c349dbc7Sjsg 	HDCP_TOP_INTERFACE_TRACE(hdcp);
528c349dbc7Sjsg 	status = reset_connection(hdcp, output);
529c349dbc7Sjsg 	if (status != MOD_HDCP_STATUS_SUCCESS)
530c349dbc7Sjsg 		push_error_status(hdcp, status);
531c349dbc7Sjsg 
532c349dbc7Sjsg 	return status;
533c349dbc7Sjsg }
534c349dbc7Sjsg 
mod_hdcp_process_event(struct mod_hdcp * hdcp,enum mod_hdcp_event event,struct mod_hdcp_output * output)535c349dbc7Sjsg enum mod_hdcp_status mod_hdcp_process_event(struct mod_hdcp *hdcp,
536c349dbc7Sjsg 		enum mod_hdcp_event event, struct mod_hdcp_output *output)
537c349dbc7Sjsg {
538c349dbc7Sjsg 	enum mod_hdcp_status exec_status, trans_status, reset_status, status;
539c349dbc7Sjsg 	struct mod_hdcp_event_context event_ctx;
540c349dbc7Sjsg 
541c349dbc7Sjsg 	HDCP_EVENT_TRACE(hdcp, event);
542c349dbc7Sjsg 	memset(output, 0, sizeof(struct mod_hdcp_output));
543c349dbc7Sjsg 	memset(&event_ctx, 0, sizeof(struct mod_hdcp_event_context));
544c349dbc7Sjsg 	event_ctx.event = event;
545c349dbc7Sjsg 
546c349dbc7Sjsg 	/* execute and transition */
547c349dbc7Sjsg 	exec_status = execution(hdcp, &event_ctx, &hdcp->auth.trans_input);
548c349dbc7Sjsg 	trans_status = transition(
549c349dbc7Sjsg 			hdcp, &event_ctx, &hdcp->auth.trans_input, output);
550c349dbc7Sjsg 	if (trans_status == MOD_HDCP_STATUS_SUCCESS) {
551c349dbc7Sjsg 		status = MOD_HDCP_STATUS_SUCCESS;
552c349dbc7Sjsg 	} else if (exec_status == MOD_HDCP_STATUS_SUCCESS) {
553c349dbc7Sjsg 		status = MOD_HDCP_STATUS_INTERNAL_POLICY_FAILURE;
554c349dbc7Sjsg 		push_error_status(hdcp, status);
555c349dbc7Sjsg 	} else {
556c349dbc7Sjsg 		status = exec_status;
557c349dbc7Sjsg 		push_error_status(hdcp, status);
558c349dbc7Sjsg 	}
559c349dbc7Sjsg 
560c349dbc7Sjsg 	/* reset authentication if needed */
561c349dbc7Sjsg 	if (trans_status == MOD_HDCP_STATUS_RESET_NEEDED) {
5625ca02815Sjsg 		mod_hdcp_log_ddc_trace(hdcp);
563c349dbc7Sjsg 		reset_status = reset_authentication(hdcp, output);
564c349dbc7Sjsg 		if (reset_status != MOD_HDCP_STATUS_SUCCESS)
565c349dbc7Sjsg 			push_error_status(hdcp, reset_status);
566c349dbc7Sjsg 	}
567ad8b1aafSjsg 
568ad8b1aafSjsg 	/* Clear CP_IRQ status if needed */
569ad8b1aafSjsg 	if (event_ctx.event == MOD_HDCP_EVENT_CPIRQ) {
570ad8b1aafSjsg 		status = mod_hdcp_clear_cp_irq_status(hdcp);
571ad8b1aafSjsg 		if (status != MOD_HDCP_STATUS_SUCCESS)
572ad8b1aafSjsg 			push_error_status(hdcp, status);
573ad8b1aafSjsg 	}
574ad8b1aafSjsg 
575c349dbc7Sjsg 	return status;
576c349dbc7Sjsg }
577c349dbc7Sjsg 
mod_hdcp_signal_type_to_operation_mode(enum amd_signal_type signal)578c349dbc7Sjsg enum mod_hdcp_operation_mode mod_hdcp_signal_type_to_operation_mode(
579c349dbc7Sjsg 		enum amd_signal_type signal)
580c349dbc7Sjsg {
581c349dbc7Sjsg 	enum mod_hdcp_operation_mode mode = MOD_HDCP_MODE_OFF;
582c349dbc7Sjsg 
583c349dbc7Sjsg 	switch (signal) {
584c349dbc7Sjsg 	case SIGNAL_TYPE_DVI_SINGLE_LINK:
585c349dbc7Sjsg 	case SIGNAL_TYPE_HDMI_TYPE_A:
586c349dbc7Sjsg 		mode = MOD_HDCP_MODE_DEFAULT;
587c349dbc7Sjsg 		break;
588c349dbc7Sjsg 	case SIGNAL_TYPE_EDP:
589c349dbc7Sjsg 	case SIGNAL_TYPE_DISPLAY_PORT:
590c349dbc7Sjsg 	case SIGNAL_TYPE_DISPLAY_PORT_MST:
591c349dbc7Sjsg 		mode = MOD_HDCP_MODE_DP;
592c349dbc7Sjsg 		break;
593c349dbc7Sjsg 	default:
594c349dbc7Sjsg 		break;
595c349dbc7Sjsg 	}
596c349dbc7Sjsg 
597c349dbc7Sjsg 	return mode;
598c349dbc7Sjsg }
599