Home | History | Annotate | Line # | Download | only in hdcp
      1 /*	$NetBSD: amdgpu_hdcp1_execution.c,v 1.3 2021/12/19 12:02:39 riastradh Exp $	*/
      2 
      3 /*
      4  * Copyright 2019 Advanced Micro Devices, Inc.
      5  *
      6  * Permission is hereby granted, free of charge, to any person obtaining a
      7  * copy of this software and associated documentation files (the "Software"),
      8  * to deal in the Software without restriction, including without limitation
      9  * the rights to use, copy, modify, merge, publish, distribute, sublicense,
     10  * and/or sell copies of the Software, and to permit persons to whom the
     11  * Software is furnished to do so, subject to the following conditions:
     12  *
     13  * The above copyright notice and this permission notice shall be included in
     14  * all copies or substantial portions of the Software.
     15  *
     16  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
     17  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
     18  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL
     19  * THE COPYRIGHT HOLDER(S) OR AUTHOR(S) BE LIABLE FOR ANY CLAIM, DAMAGES OR
     20  * OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
     21  * ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
     22  * OTHER DEALINGS IN THE SOFTWARE.
     23  *
     24  * Authors: AMD
     25  *
     26  */
     27 
     28 #include <sys/cdefs.h>
     29 __KERNEL_RCSID(0, "$NetBSD: amdgpu_hdcp1_execution.c,v 1.3 2021/12/19 12:02:39 riastradh Exp $");
     30 
     31 #include "hdcp.h"
     32 
     33 static inline enum mod_hdcp_status validate_bksv(struct mod_hdcp *hdcp)
     34 {
     35 	uint64_t n = 0;
     36 	uint8_t count = 0;
     37 
     38 	memcpy(&n, hdcp->auth.msg.hdcp1.bksv, sizeof(uint64_t));
     39 
     40 	while (n) {
     41 		count++;
     42 		n &= (n - 1);
     43 	}
     44 	return (count == 20) ? MOD_HDCP_STATUS_SUCCESS :
     45 			MOD_HDCP_STATUS_HDCP1_INVALID_BKSV;
     46 }
     47 
     48 static inline enum mod_hdcp_status check_ksv_ready(struct mod_hdcp *hdcp)
     49 {
     50 	if (is_dp_hdcp(hdcp))
     51 		return (hdcp->auth.msg.hdcp1.bstatus & DP_BSTATUS_READY) ?
     52 				MOD_HDCP_STATUS_SUCCESS :
     53 				MOD_HDCP_STATUS_HDCP1_KSV_LIST_NOT_READY;
     54 	return (hdcp->auth.msg.hdcp1.bcaps & DRM_HDCP_DDC_BCAPS_KSV_FIFO_READY) ?
     55 			MOD_HDCP_STATUS_SUCCESS :
     56 			MOD_HDCP_STATUS_HDCP1_KSV_LIST_NOT_READY;
     57 }
     58 
     59 static inline enum mod_hdcp_status check_hdcp_capable_dp(struct mod_hdcp *hdcp)
     60 {
     61 	return (hdcp->auth.msg.hdcp1.bcaps & DP_BCAPS_HDCP_CAPABLE) ?
     62 			MOD_HDCP_STATUS_SUCCESS :
     63 			MOD_HDCP_STATUS_HDCP1_NOT_CAPABLE;
     64 }
     65 
     66 static inline enum mod_hdcp_status check_r0p_available_dp(struct mod_hdcp *hdcp)
     67 {
     68 	enum mod_hdcp_status status;
     69 	if (is_dp_hdcp(hdcp)) {
     70 		status = (hdcp->auth.msg.hdcp1.bstatus &
     71 				DP_BSTATUS_R0_PRIME_READY) ?
     72 			MOD_HDCP_STATUS_SUCCESS :
     73 			MOD_HDCP_STATUS_HDCP1_R0_PRIME_PENDING;
     74 	} else {
     75 		status = MOD_HDCP_STATUS_INVALID_OPERATION;
     76 	}
     77 	return status;
     78 }
     79 
     80 static inline enum mod_hdcp_status check_link_integrity_dp(
     81 		struct mod_hdcp *hdcp)
     82 {
     83 	return (hdcp->auth.msg.hdcp1.bstatus &
     84 			DP_BSTATUS_LINK_FAILURE) ?
     85 			MOD_HDCP_STATUS_HDCP1_LINK_INTEGRITY_FAILURE :
     86 			MOD_HDCP_STATUS_SUCCESS;
     87 }
     88 
     89 static inline enum mod_hdcp_status check_no_reauthentication_request_dp(
     90 		struct mod_hdcp *hdcp)
     91 {
     92 	return (hdcp->auth.msg.hdcp1.bstatus & DP_BSTATUS_REAUTH_REQ) ?
     93 			MOD_HDCP_STATUS_HDCP1_REAUTH_REQUEST_ISSUED :
     94 			MOD_HDCP_STATUS_SUCCESS;
     95 }
     96 
     97 static inline enum mod_hdcp_status check_no_max_cascade(struct mod_hdcp *hdcp)
     98 {
     99 	enum mod_hdcp_status status;
    100 
    101 	if (is_dp_hdcp(hdcp))
    102 		status = DRM_HDCP_MAX_CASCADE_EXCEEDED(hdcp->auth.msg.hdcp1.binfo_dp >> 8)
    103 				 ? MOD_HDCP_STATUS_HDCP1_MAX_CASCADE_EXCEEDED_FAILURE
    104 				 : MOD_HDCP_STATUS_SUCCESS;
    105 	else
    106 		status = DRM_HDCP_MAX_CASCADE_EXCEEDED(hdcp->auth.msg.hdcp1.bstatus >> 8)
    107 				 ? MOD_HDCP_STATUS_HDCP1_MAX_CASCADE_EXCEEDED_FAILURE
    108 				 : MOD_HDCP_STATUS_SUCCESS;
    109 	return status;
    110 }
    111 
    112 static inline enum mod_hdcp_status check_no_max_devs(struct mod_hdcp *hdcp)
    113 {
    114 	enum mod_hdcp_status status;
    115 
    116 	if (is_dp_hdcp(hdcp))
    117 		status = DRM_HDCP_MAX_DEVICE_EXCEEDED(hdcp->auth.msg.hdcp1.binfo_dp) ?
    118 				MOD_HDCP_STATUS_HDCP1_MAX_DEVS_EXCEEDED_FAILURE :
    119 				MOD_HDCP_STATUS_SUCCESS;
    120 	else
    121 		status = DRM_HDCP_MAX_DEVICE_EXCEEDED(hdcp->auth.msg.hdcp1.bstatus) ?
    122 				MOD_HDCP_STATUS_HDCP1_MAX_DEVS_EXCEEDED_FAILURE :
    123 				MOD_HDCP_STATUS_SUCCESS;
    124 	return status;
    125 }
    126 
    127 static inline uint8_t get_device_count(struct mod_hdcp *hdcp)
    128 {
    129 	return is_dp_hdcp(hdcp) ?
    130 			DRM_HDCP_NUM_DOWNSTREAM(hdcp->auth.msg.hdcp1.binfo_dp) :
    131 			DRM_HDCP_NUM_DOWNSTREAM(hdcp->auth.msg.hdcp1.bstatus);
    132 }
    133 
    134 static inline enum mod_hdcp_status check_device_count(struct mod_hdcp *hdcp)
    135 {
    136 	/* device count must be greater than or equal to tracked hdcp displays */
    137 	return (get_device_count(hdcp) < get_added_display_count(hdcp)) ?
    138 			MOD_HDCP_STATUS_HDCP1_DEVICE_COUNT_MISMATCH_FAILURE :
    139 			MOD_HDCP_STATUS_SUCCESS;
    140 }
    141 
    142 static enum mod_hdcp_status wait_for_active_rx(struct mod_hdcp *hdcp,
    143 		struct mod_hdcp_event_context *event_ctx,
    144 		struct mod_hdcp_transition_input_hdcp1 *input)
    145 {
    146 	enum mod_hdcp_status status = MOD_HDCP_STATUS_SUCCESS;
    147 
    148 	if (event_ctx->event != MOD_HDCP_EVENT_CALLBACK) {
    149 		event_ctx->unexpected_event = 1;
    150 		goto out;
    151 	}
    152 
    153 	if (!mod_hdcp_execute_and_set(mod_hdcp_read_bksv,
    154 			&input->bksv_read, &status,
    155 			hdcp, "bksv_read"))
    156 		goto out;
    157 	if (!mod_hdcp_execute_and_set(mod_hdcp_read_bcaps,
    158 			&input->bcaps_read, &status,
    159 			hdcp, "bcaps_read"))
    160 		goto out;
    161 out:
    162 	return status;
    163 }
    164 
    165 static enum mod_hdcp_status exchange_ksvs(struct mod_hdcp *hdcp,
    166 		struct mod_hdcp_event_context *event_ctx,
    167 		struct mod_hdcp_transition_input_hdcp1 *input)
    168 {
    169 	enum mod_hdcp_status status = MOD_HDCP_STATUS_SUCCESS;
    170 
    171 	if (event_ctx->event != MOD_HDCP_EVENT_CALLBACK) {
    172 		event_ctx->unexpected_event = 1;
    173 		goto out;
    174 	}
    175 
    176 	if (!mod_hdcp_execute_and_set(mod_hdcp_add_display_topology,
    177 			&input->add_topology, &status,
    178 			hdcp, "add_topology"))
    179 		goto out;
    180 	if (!mod_hdcp_execute_and_set(mod_hdcp_hdcp1_create_session,
    181 			&input->create_session, &status,
    182 			hdcp, "create_session"))
    183 		goto out;
    184 	if (!mod_hdcp_execute_and_set(mod_hdcp_write_an,
    185 			&input->an_write, &status,
    186 			hdcp, "an_write"))
    187 		goto out;
    188 	if (!mod_hdcp_execute_and_set(mod_hdcp_write_aksv,
    189 			&input->aksv_write, &status,
    190 			hdcp, "aksv_write"))
    191 		goto out;
    192 	if (!mod_hdcp_execute_and_set(mod_hdcp_read_bksv,
    193 			&input->bksv_read, &status,
    194 			hdcp, "bksv_read"))
    195 		goto out;
    196 	if (!mod_hdcp_execute_and_set(validate_bksv,
    197 			&input->bksv_validation, &status,
    198 			hdcp, "bksv_validation"))
    199 		goto out;
    200 	if (hdcp->auth.msg.hdcp1.ainfo) {
    201 		if (!mod_hdcp_execute_and_set(mod_hdcp_write_ainfo,
    202 				&input->ainfo_write, &status,
    203 				hdcp, "ainfo_write"))
    204 			goto out;
    205 	}
    206 out:
    207 	return status;
    208 }
    209 
    210 static enum mod_hdcp_status computations_validate_rx_test_for_repeater(
    211 		struct mod_hdcp *hdcp,
    212 		struct mod_hdcp_event_context *event_ctx,
    213 		struct mod_hdcp_transition_input_hdcp1 *input)
    214 {
    215 	enum mod_hdcp_status status = MOD_HDCP_STATUS_SUCCESS;
    216 
    217 	if (event_ctx->event != MOD_HDCP_EVENT_CALLBACK) {
    218 		event_ctx->unexpected_event = 1;
    219 		goto out;
    220 	}
    221 
    222 	if (!mod_hdcp_execute_and_set(mod_hdcp_read_r0p,
    223 			&input->r0p_read, &status,
    224 			hdcp, "r0p_read"))
    225 		goto out;
    226 	if (!mod_hdcp_execute_and_set(mod_hdcp_hdcp1_validate_rx,
    227 			&input->rx_validation, &status,
    228 			hdcp, "rx_validation"))
    229 		goto out;
    230 	if (hdcp->connection.is_repeater) {
    231 		if (!hdcp->connection.link.adjust.hdcp1.postpone_encryption)
    232 			if (!mod_hdcp_execute_and_set(
    233 					mod_hdcp_hdcp1_enable_encryption,
    234 					&input->encryption, &status,
    235 					hdcp, "encryption"))
    236 				goto out;
    237 	} else {
    238 		if (!mod_hdcp_execute_and_set(mod_hdcp_hdcp1_enable_encryption,
    239 				&input->encryption, &status,
    240 				hdcp, "encryption"))
    241 			goto out;
    242 		if (is_dp_mst_hdcp(hdcp))
    243 			if (!mod_hdcp_execute_and_set(
    244 					mod_hdcp_hdcp1_enable_dp_stream_encryption,
    245 					&input->stream_encryption_dp, &status,
    246 					hdcp, "stream_encryption_dp"))
    247 				goto out;
    248 	}
    249 out:
    250 	return status;
    251 }
    252 
    253 static enum mod_hdcp_status authenticated(struct mod_hdcp *hdcp,
    254 		struct mod_hdcp_event_context *event_ctx,
    255 		struct mod_hdcp_transition_input_hdcp1 *input)
    256 {
    257 	enum mod_hdcp_status status = MOD_HDCP_STATUS_SUCCESS;
    258 
    259 	if (event_ctx->event != MOD_HDCP_EVENT_CALLBACK) {
    260 		event_ctx->unexpected_event = 1;
    261 		goto out;
    262 	}
    263 
    264 	if (!mod_hdcp_execute_and_set(mod_hdcp_hdcp1_link_maintenance,
    265 			&input->link_maintenance, &status,
    266 			hdcp, "link_maintenance"))
    267 		goto out;
    268 out:
    269 	return status;
    270 }
    271 
    272 static enum mod_hdcp_status wait_for_ready(struct mod_hdcp *hdcp,
    273 		struct mod_hdcp_event_context *event_ctx,
    274 		struct mod_hdcp_transition_input_hdcp1 *input)
    275 {
    276 	enum mod_hdcp_status status = MOD_HDCP_STATUS_SUCCESS;
    277 
    278 	if (event_ctx->event != MOD_HDCP_EVENT_CALLBACK &&
    279 			event_ctx->event != MOD_HDCP_EVENT_CPIRQ &&
    280 			event_ctx->event != MOD_HDCP_EVENT_WATCHDOG_TIMEOUT) {
    281 		event_ctx->unexpected_event = 1;
    282 		goto out;
    283 	}
    284 
    285 	if (is_dp_hdcp(hdcp)) {
    286 		if (!mod_hdcp_execute_and_set(mod_hdcp_read_bstatus,
    287 				&input->bstatus_read, &status,
    288 				hdcp, "bstatus_read"))
    289 			goto out;
    290 		if (!mod_hdcp_execute_and_set(check_link_integrity_dp,
    291 				&input->link_integrity_check, &status,
    292 				hdcp, "link_integrity_check"))
    293 			goto out;
    294 		if (!mod_hdcp_execute_and_set(check_no_reauthentication_request_dp,
    295 				&input->reauth_request_check, &status,
    296 				hdcp, "reauth_request_check"))
    297 			goto out;
    298 	} else {
    299 		if (!mod_hdcp_execute_and_set(mod_hdcp_read_bcaps,
    300 				&input->bcaps_read, &status,
    301 				hdcp, "bcaps_read"))
    302 			goto out;
    303 	}
    304 	if (!mod_hdcp_execute_and_set(check_ksv_ready,
    305 			&input->ready_check, &status,
    306 			hdcp, "ready_check"))
    307 		goto out;
    308 out:
    309 	return status;
    310 }
    311 
    312 static enum mod_hdcp_status read_ksv_list(struct mod_hdcp *hdcp,
    313 		struct mod_hdcp_event_context *event_ctx,
    314 		struct mod_hdcp_transition_input_hdcp1 *input)
    315 {
    316 	enum mod_hdcp_status status = MOD_HDCP_STATUS_SUCCESS;
    317 	uint8_t device_count;
    318 
    319 	if (event_ctx->event != MOD_HDCP_EVENT_CALLBACK) {
    320 		event_ctx->unexpected_event = 1;
    321 		goto out;
    322 	}
    323 
    324 	if (is_dp_hdcp(hdcp)) {
    325 		if (!mod_hdcp_execute_and_set(mod_hdcp_read_binfo,
    326 				&input->binfo_read_dp, &status,
    327 				hdcp, "binfo_read_dp"))
    328 			goto out;
    329 	} else {
    330 		if (!mod_hdcp_execute_and_set(mod_hdcp_read_bstatus,
    331 				&input->bstatus_read, &status,
    332 				hdcp, "bstatus_read"))
    333 			goto out;
    334 	}
    335 	if (!mod_hdcp_execute_and_set(check_no_max_cascade,
    336 			&input->max_cascade_check, &status,
    337 			hdcp, "max_cascade_check"))
    338 		goto out;
    339 	if (!mod_hdcp_execute_and_set(check_no_max_devs,
    340 			&input->max_devs_check, &status,
    341 			hdcp, "max_devs_check"))
    342 		goto out;
    343 	if (!mod_hdcp_execute_and_set(check_device_count,
    344 			&input->device_count_check, &status,
    345 			hdcp, "device_count_check"))
    346 		goto out;
    347 	device_count = get_device_count(hdcp);
    348 	hdcp->auth.msg.hdcp1.ksvlist_size = device_count*5;
    349 	if (!mod_hdcp_execute_and_set(mod_hdcp_read_ksvlist,
    350 			&input->ksvlist_read, &status,
    351 			hdcp, "ksvlist_read"))
    352 		goto out;
    353 	if (!mod_hdcp_execute_and_set(mod_hdcp_read_vp,
    354 			&input->vp_read, &status,
    355 			hdcp, "vp_read"))
    356 		goto out;
    357 	if (!mod_hdcp_execute_and_set(mod_hdcp_hdcp1_validate_ksvlist_vp,
    358 			&input->ksvlist_vp_validation, &status,
    359 			hdcp, "ksvlist_vp_validation"))
    360 		goto out;
    361 	if (input->encryption != PASS)
    362 		if (!mod_hdcp_execute_and_set(mod_hdcp_hdcp1_enable_encryption,
    363 				&input->encryption, &status,
    364 				hdcp, "encryption"))
    365 			goto out;
    366 	if (is_dp_mst_hdcp(hdcp))
    367 		if (!mod_hdcp_execute_and_set(
    368 				mod_hdcp_hdcp1_enable_dp_stream_encryption,
    369 				&input->stream_encryption_dp, &status,
    370 				hdcp, "stream_encryption_dp"))
    371 			goto out;
    372 out:
    373 	return status;
    374 }
    375 
    376 static enum mod_hdcp_status determine_rx_hdcp_capable_dp(struct mod_hdcp *hdcp,
    377 		struct mod_hdcp_event_context *event_ctx,
    378 		struct mod_hdcp_transition_input_hdcp1 *input)
    379 {
    380 	enum mod_hdcp_status status = MOD_HDCP_STATUS_SUCCESS;
    381 
    382 	if (event_ctx->event != MOD_HDCP_EVENT_CALLBACK) {
    383 		event_ctx->unexpected_event = 1;
    384 		goto out;
    385 	}
    386 
    387 	if (!mod_hdcp_execute_and_set(mod_hdcp_read_bcaps,
    388 			&input->bcaps_read, &status,
    389 			hdcp, "bcaps_read"))
    390 		goto out;
    391 	if (!mod_hdcp_execute_and_set(check_hdcp_capable_dp,
    392 			&input->hdcp_capable_dp, &status,
    393 			hdcp, "hdcp_capable_dp"))
    394 		goto out;
    395 out:
    396 	return status;
    397 }
    398 
    399 static enum mod_hdcp_status wait_for_r0_prime_dp(struct mod_hdcp *hdcp,
    400 		struct mod_hdcp_event_context *event_ctx,
    401 		struct mod_hdcp_transition_input_hdcp1 *input)
    402 {
    403 	enum mod_hdcp_status status = MOD_HDCP_STATUS_SUCCESS;
    404 
    405 	if (event_ctx->event != MOD_HDCP_EVENT_CPIRQ &&
    406 			event_ctx->event != MOD_HDCP_EVENT_WATCHDOG_TIMEOUT) {
    407 		event_ctx->unexpected_event = 1;
    408 		goto out;
    409 	}
    410 
    411 	if (!mod_hdcp_execute_and_set(mod_hdcp_read_bstatus,
    412 			&input->bstatus_read, &status,
    413 			hdcp, "bstatus_read"))
    414 		goto out;
    415 	if (!mod_hdcp_execute_and_set(check_r0p_available_dp,
    416 			&input->r0p_available_dp, &status,
    417 			hdcp, "r0p_available_dp"))
    418 		goto out;
    419 out:
    420 	return status;
    421 }
    422 
    423 static enum mod_hdcp_status authenticated_dp(struct mod_hdcp *hdcp,
    424 		struct mod_hdcp_event_context *event_ctx,
    425 		struct mod_hdcp_transition_input_hdcp1 *input)
    426 {
    427 	enum mod_hdcp_status status = MOD_HDCP_STATUS_SUCCESS;
    428 
    429 	if (event_ctx->event != MOD_HDCP_EVENT_CPIRQ) {
    430 		event_ctx->unexpected_event = 1;
    431 		goto out;
    432 	}
    433 
    434 	if (!mod_hdcp_execute_and_set(mod_hdcp_read_bstatus,
    435 			&input->bstatus_read, &status,
    436 			hdcp, "bstatus_read"))
    437 		goto out;
    438 	if (!mod_hdcp_execute_and_set(check_link_integrity_dp,
    439 			&input->link_integrity_check, &status,
    440 			hdcp, "link_integrity_check"))
    441 		goto out;
    442 	if (!mod_hdcp_execute_and_set(check_no_reauthentication_request_dp,
    443 			&input->reauth_request_check, &status,
    444 			hdcp, "reauth_request_check"))
    445 		goto out;
    446 out:
    447 	return status;
    448 }
    449 
    450 uint8_t mod_hdcp_execute_and_set(
    451 		mod_hdcp_action func, uint8_t *flag,
    452 		enum mod_hdcp_status *status, struct mod_hdcp *hdcp, const char *str)
    453 {
    454 	*status = func(hdcp);
    455 	if (*status == MOD_HDCP_STATUS_SUCCESS && *flag != PASS) {
    456 		HDCP_INPUT_PASS_TRACE(hdcp, str);
    457 		*flag = PASS;
    458 	} else if (*status != MOD_HDCP_STATUS_SUCCESS && *flag != FAIL) {
    459 		HDCP_INPUT_FAIL_TRACE(hdcp, str);
    460 		*flag = FAIL;
    461 	}
    462 	return (*status == MOD_HDCP_STATUS_SUCCESS);
    463 }
    464 
    465 enum mod_hdcp_status mod_hdcp_hdcp1_execution(struct mod_hdcp *hdcp,
    466 		struct mod_hdcp_event_context *event_ctx,
    467 		struct mod_hdcp_transition_input_hdcp1 *input)
    468 {
    469 	enum mod_hdcp_status status = MOD_HDCP_STATUS_SUCCESS;
    470 
    471 	switch (current_state(hdcp)) {
    472 	case H1_A0_WAIT_FOR_ACTIVE_RX:
    473 		status = wait_for_active_rx(hdcp, event_ctx, input);
    474 		break;
    475 	case H1_A1_EXCHANGE_KSVS:
    476 		status = exchange_ksvs(hdcp, event_ctx, input);
    477 		break;
    478 	case H1_A2_COMPUTATIONS_A3_VALIDATE_RX_A6_TEST_FOR_REPEATER:
    479 		status = computations_validate_rx_test_for_repeater(hdcp,
    480 				event_ctx, input);
    481 		break;
    482 	case H1_A45_AUTHENTICATED:
    483 		status = authenticated(hdcp, event_ctx, input);
    484 		break;
    485 	case H1_A8_WAIT_FOR_READY:
    486 		status = wait_for_ready(hdcp, event_ctx, input);
    487 		break;
    488 	case H1_A9_READ_KSV_LIST:
    489 		status = read_ksv_list(hdcp, event_ctx, input);
    490 		break;
    491 	default:
    492 		status = MOD_HDCP_STATUS_INVALID_STATE;
    493 		break;
    494 	}
    495 
    496 	return status;
    497 }
    498 
    499 extern enum mod_hdcp_status mod_hdcp_hdcp1_dp_execution(struct mod_hdcp *hdcp,
    500 		struct mod_hdcp_event_context *event_ctx,
    501 		struct mod_hdcp_transition_input_hdcp1 *input)
    502 {
    503 	enum mod_hdcp_status status = MOD_HDCP_STATUS_SUCCESS;
    504 
    505 	switch (current_state(hdcp)) {
    506 	case D1_A0_DETERMINE_RX_HDCP_CAPABLE:
    507 		status = determine_rx_hdcp_capable_dp(hdcp, event_ctx, input);
    508 		break;
    509 	case D1_A1_EXCHANGE_KSVS:
    510 		status = exchange_ksvs(hdcp, event_ctx, input);
    511 		break;
    512 	case D1_A23_WAIT_FOR_R0_PRIME:
    513 		status = wait_for_r0_prime_dp(hdcp, event_ctx, input);
    514 		break;
    515 	case D1_A2_COMPUTATIONS_A3_VALIDATE_RX_A5_TEST_FOR_REPEATER:
    516 		status = computations_validate_rx_test_for_repeater(
    517 				hdcp, event_ctx, input);
    518 		break;
    519 	case D1_A4_AUTHENTICATED:
    520 		status = authenticated_dp(hdcp, event_ctx, input);
    521 		break;
    522 	case D1_A6_WAIT_FOR_READY:
    523 		status = wait_for_ready(hdcp, event_ctx, input);
    524 		break;
    525 	case D1_A7_READ_KSV_LIST:
    526 		status = read_ksv_list(hdcp, event_ctx, input);
    527 		break;
    528 	default:
    529 		status = MOD_HDCP_STATUS_INVALID_STATE;
    530 		break;
    531 	}
    532 
    533 	return status;
    534 }
    535