/* Number of bytes in PSP footer for firmware. */
 #define PSP_FOOTER_BYTES 0x100
 
-/*
- * DMUB Async to Sync Mechanism Status
- */
-#define DMUB_ASYNC_TO_SYNC_ACCESS_FAIL 1
-#define DMUB_ASYNC_TO_SYNC_ACCESS_TIMEOUT 2
-#define DMUB_ASYNC_TO_SYNC_ACCESS_SUCCESS 3
-#define DMUB_ASYNC_TO_SYNC_ACCESS_INVALID 4
-
 /**
  * DOC: overview
  *
        memset(&init_params, 0, sizeof(init_params));
 #endif
 
+       mutex_init(&adev->dm.dpia_aux_lock);
        mutex_init(&adev->dm.dc_lock);
        mutex_init(&adev->dm.audio_lock);
 
 
        mutex_destroy(&adev->dm.audio_lock);
        mutex_destroy(&adev->dm.dc_lock);
+       mutex_destroy(&adev->dm.dpia_aux_lock);
 
        return;
 }
        return value;
 }
 
-static int amdgpu_dm_set_dmub_async_sync_status(bool is_cmd_aux,
-                                               struct dc_context *ctx,
-                                               uint8_t status_type,
-                                               uint32_t *operation_result)
+int amdgpu_dm_process_dmub_aux_transfer_sync(
+               struct dc_context *ctx,
+               unsigned int link_index,
+               struct aux_payload *payload,
+               enum aux_return_code_type *operation_result)
 {
        struct amdgpu_device *adev = ctx->driver_context;
-       int return_status = -1;
        struct dmub_notification *p_notify = adev->dm.dmub_notify;
+       int ret = -1;
 
-       if (is_cmd_aux) {
-               if (status_type == DMUB_ASYNC_TO_SYNC_ACCESS_SUCCESS) {
-                       return_status = p_notify->aux_reply.length;
-                       *operation_result = p_notify->result;
-               } else if (status_type == DMUB_ASYNC_TO_SYNC_ACCESS_TIMEOUT) {
-                       *operation_result = AUX_RET_ERROR_TIMEOUT;
-               } else if (status_type == DMUB_ASYNC_TO_SYNC_ACCESS_FAIL) {
-                       *operation_result = AUX_RET_ERROR_ENGINE_ACQUIRE;
-               } else if (status_type == DMUB_ASYNC_TO_SYNC_ACCESS_INVALID) {
-                       *operation_result = AUX_RET_ERROR_INVALID_REPLY;
-               } else {
-                       *operation_result = AUX_RET_ERROR_UNKNOWN;
+       mutex_lock(&adev->dm.dpia_aux_lock);
+       if (!dc_process_dmub_aux_transfer_async(ctx->dc, link_index, payload)) {
+               *operation_result = AUX_RET_ERROR_ENGINE_ACQUIRE;
+               goto out;
+       }
+
+       if (!wait_for_completion_timeout(&adev->dm.dmub_aux_transfer_done, 10 * HZ)) {
+               DRM_ERROR("wait_for_completion_timeout timeout!");
+               *operation_result = AUX_RET_ERROR_TIMEOUT;
+               goto out;
+       }
+
+       if (p_notify->result != AUX_RET_SUCCESS) {
+               /*
+                * Transient states before tunneling is enabled could
+                * lead to this error. We can ignore this for now.
+                */
+               if (p_notify->result != AUX_RET_ERROR_PROTOCOL_ERROR) {
+                       DRM_WARN("DPIA AUX failed on 0x%x(%d), error %d\n",
+                                       payload->address, payload->length,
+                                       p_notify->result);
                }
-       } else {
-               if (status_type == DMUB_ASYNC_TO_SYNC_ACCESS_SUCCESS) {
-                       return_status = 0;
-                       *operation_result = p_notify->sc_status;
-               } else {
-                       *operation_result = SET_CONFIG_UNKNOWN_ERROR;
+               *operation_result = AUX_RET_ERROR_INVALID_REPLY;
+               goto out;
+       }
+
+
+       payload->reply[0] = adev->dm.dmub_notify->aux_reply.command;
+       if (!payload->write && p_notify->aux_reply.length &&
+                       (payload->reply[0] == AUX_TRANSACTION_REPLY_AUX_ACK)) {
+
+               if (payload->length != p_notify->aux_reply.length) {
+                       DRM_WARN("invalid read length %d from DPIA AUX 0x%x(%d)!\n",
+                               p_notify->aux_reply.length,
+                                       payload->address, payload->length);
+                       *operation_result = AUX_RET_ERROR_INVALID_REPLY;
+                       goto out;
                }
+
+               memcpy(payload->data, p_notify->aux_reply.data,
+                               p_notify->aux_reply.length);
        }
 
-       return return_status;
+       /* success */
+       ret = p_notify->aux_reply.length;
+       *operation_result = p_notify->result;
+out:
+       mutex_unlock(&adev->dm.dpia_aux_lock);
+       return ret;
 }
 
-int amdgpu_dm_process_dmub_aux_transfer_sync(bool is_cmd_aux, struct dc_context *ctx,
-       unsigned int link_index, void *cmd_payload, void *operation_result)
+int amdgpu_dm_process_dmub_set_config_sync(
+               struct dc_context *ctx,
+               unsigned int link_index,
+               struct set_config_cmd_payload *payload,
+               enum set_config_status *operation_result)
 {
        struct amdgpu_device *adev = ctx->driver_context;
-       int ret = 0;
+       bool is_cmd_complete;
+       int ret;
 
-       if (is_cmd_aux) {
-               dc_process_dmub_aux_transfer_async(ctx->dc,
-                       link_index, (struct aux_payload *)cmd_payload);
-       } else if (dc_process_dmub_set_config_async(ctx->dc, link_index,
-                                       (struct set_config_cmd_payload *)cmd_payload,
-                                       adev->dm.dmub_notify)) {
-               return amdgpu_dm_set_dmub_async_sync_status(is_cmd_aux,
-                                       ctx, DMUB_ASYNC_TO_SYNC_ACCESS_SUCCESS,
-                                       (uint32_t *)operation_result);
-       }
+       mutex_lock(&adev->dm.dpia_aux_lock);
+       is_cmd_complete = dc_process_dmub_set_config_async(ctx->dc,
+                       link_index, payload, adev->dm.dmub_notify);
 
-       ret = wait_for_completion_timeout(&adev->dm.dmub_aux_transfer_done, 10 * HZ);
-       if (ret == 0) {
+       if (is_cmd_complete || wait_for_completion_timeout(&adev->dm.dmub_aux_transfer_done, 10 * HZ)) {
+               ret = 0;
+               *operation_result = adev->dm.dmub_notify->sc_status;
+       } else {
                DRM_ERROR("wait_for_completion_timeout timeout!");
-               return amdgpu_dm_set_dmub_async_sync_status(is_cmd_aux,
-                               ctx, DMUB_ASYNC_TO_SYNC_ACCESS_TIMEOUT,
-                               (uint32_t *)operation_result);
-       }
-
-       if (is_cmd_aux) {
-               if (adev->dm.dmub_notify->result == AUX_RET_SUCCESS) {
-                       struct aux_payload *payload = (struct aux_payload *)cmd_payload;
-
-                       payload->reply[0] = adev->dm.dmub_notify->aux_reply.command;
-                       if (!payload->write && adev->dm.dmub_notify->aux_reply.length &&
-                           payload->reply[0] == AUX_TRANSACTION_REPLY_AUX_ACK) {
-
-                               if (payload->length != adev->dm.dmub_notify->aux_reply.length) {
-                                       DRM_WARN("invalid read from DPIA AUX %x(%d) got length %d!\n",
-                                                       payload->address, payload->length,
-                                                       adev->dm.dmub_notify->aux_reply.length);
-                                       return amdgpu_dm_set_dmub_async_sync_status(is_cmd_aux, ctx,
-                                                       DMUB_ASYNC_TO_SYNC_ACCESS_INVALID,
-                                                       (uint32_t *)operation_result);
-                               }
-
-                               memcpy(payload->data, adev->dm.dmub_notify->aux_reply.data,
-                                      adev->dm.dmub_notify->aux_reply.length);
-                       }
-               }
+               ret = -1;
+               *operation_result = SET_CONFIG_UNKNOWN_ERROR;
        }
 
-       return amdgpu_dm_set_dmub_async_sync_status(is_cmd_aux,
-                       ctx, DMUB_ASYNC_TO_SYNC_ACCESS_SUCCESS,
-                       (uint32_t *)operation_result);
+       mutex_unlock(&adev->dm.dpia_aux_lock);
+       return ret;
 }
 
 /*