3198a62eb8
Currently the VPU firmware prepares for D0i3 every time the VPU is entering D0i2 Idle state. This is not optimal as we might not enter D0i3 every time we enter D0i2 Idle and this preparation is quite costly. This optimization moves D0i3 preparation to a dedicated message sent from the host driver only when the driver is about to enter D0i3 - this reduces power consumption and latency for certain workloads, for example audio workloads that submit inference every 10 ms. The VPU needs non zero time to enter IDLE state after responding to D0i3 entry message. If the driver does not wait for the VPU to enter IDLE state it could cause warm boot failures. Signed-off-by: Andrzej Kacprowski <andrzej.kacprowski@linux.intel.com> Reviewed-by: Stanislaw Gruszka <stanislaw.gruszka@linux.intel.com> Signed-off-by: Stanislaw Gruszka <stanislaw.gruszka@linux.intel.com> Link: https://patchwork.freedesktop.org/patch/msgid/20231028133415.1169975-12-stanislaw.gruszka@linux.intel.com
284 lines
9.5 KiB
C
284 lines
9.5 KiB
C
// SPDX-License-Identifier: GPL-2.0-only
|
|
/*
|
|
* Copyright (C) 2020-2023 Intel Corporation
|
|
*/
|
|
|
|
#include "ivpu_drv.h"
|
|
#include "ivpu_hw.h"
|
|
#include "ivpu_ipc.h"
|
|
#include "ivpu_jsm_msg.h"
|
|
|
|
const char *ivpu_jsm_msg_type_to_str(enum vpu_ipc_msg_type type)
|
|
{
|
|
#define IVPU_CASE_TO_STR(x) case x: return #x
|
|
switch (type) {
|
|
IVPU_CASE_TO_STR(VPU_JSM_MSG_UNKNOWN);
|
|
IVPU_CASE_TO_STR(VPU_JSM_MSG_ENGINE_RESET);
|
|
IVPU_CASE_TO_STR(VPU_JSM_MSG_ENGINE_PREEMPT);
|
|
IVPU_CASE_TO_STR(VPU_JSM_MSG_REGISTER_DB);
|
|
IVPU_CASE_TO_STR(VPU_JSM_MSG_UNREGISTER_DB);
|
|
IVPU_CASE_TO_STR(VPU_JSM_MSG_QUERY_ENGINE_HB);
|
|
IVPU_CASE_TO_STR(VPU_JSM_MSG_GET_POWER_LEVEL_COUNT);
|
|
IVPU_CASE_TO_STR(VPU_JSM_MSG_GET_POWER_LEVEL);
|
|
IVPU_CASE_TO_STR(VPU_JSM_MSG_SET_POWER_LEVEL);
|
|
IVPU_CASE_TO_STR(VPU_JSM_MSG_METRIC_STREAMER_OPEN);
|
|
IVPU_CASE_TO_STR(VPU_JSM_MSG_METRIC_STREAMER_CLOSE);
|
|
IVPU_CASE_TO_STR(VPU_JSM_MSG_TRACE_SET_CONFIG);
|
|
IVPU_CASE_TO_STR(VPU_JSM_MSG_TRACE_GET_CONFIG);
|
|
IVPU_CASE_TO_STR(VPU_JSM_MSG_TRACE_GET_CAPABILITY);
|
|
IVPU_CASE_TO_STR(VPU_JSM_MSG_TRACE_GET_NAME);
|
|
IVPU_CASE_TO_STR(VPU_JSM_MSG_SSID_RELEASE);
|
|
IVPU_CASE_TO_STR(VPU_JSM_MSG_METRIC_STREAMER_START);
|
|
IVPU_CASE_TO_STR(VPU_JSM_MSG_METRIC_STREAMER_STOP);
|
|
IVPU_CASE_TO_STR(VPU_JSM_MSG_METRIC_STREAMER_UPDATE);
|
|
IVPU_CASE_TO_STR(VPU_JSM_MSG_METRIC_STREAMER_INFO);
|
|
IVPU_CASE_TO_STR(VPU_JSM_MSG_SET_PRIORITY_BAND_SETUP);
|
|
IVPU_CASE_TO_STR(VPU_JSM_MSG_CREATE_CMD_QUEUE);
|
|
IVPU_CASE_TO_STR(VPU_JSM_MSG_DESTROY_CMD_QUEUE);
|
|
IVPU_CASE_TO_STR(VPU_JSM_MSG_SET_CONTEXT_SCHED_PROPERTIES);
|
|
IVPU_CASE_TO_STR(VPU_JSM_MSG_HWS_REGISTER_DB);
|
|
IVPU_CASE_TO_STR(VPU_JSM_MSG_HWS_RESUME_CMDQ);
|
|
IVPU_CASE_TO_STR(VPU_JSM_MSG_HWS_SUSPEND_CMDQ);
|
|
IVPU_CASE_TO_STR(VPU_JSM_MSG_HWS_RESUME_CMDQ_RSP);
|
|
IVPU_CASE_TO_STR(VPU_JSM_MSG_HWS_SUSPEND_CMDQ_DONE);
|
|
IVPU_CASE_TO_STR(VPU_JSM_MSG_HWS_SET_SCHEDULING_LOG);
|
|
IVPU_CASE_TO_STR(VPU_JSM_MSG_HWS_SET_SCHEDULING_LOG_RSP);
|
|
IVPU_CASE_TO_STR(VPU_JSM_MSG_HWS_SCHEDULING_LOG_NOTIFICATION);
|
|
IVPU_CASE_TO_STR(VPU_JSM_MSG_HWS_ENGINE_RESUME);
|
|
IVPU_CASE_TO_STR(VPU_JSM_MSG_HWS_RESUME_ENGINE_DONE);
|
|
IVPU_CASE_TO_STR(VPU_JSM_MSG_STATE_DUMP);
|
|
IVPU_CASE_TO_STR(VPU_JSM_MSG_STATE_DUMP_RSP);
|
|
IVPU_CASE_TO_STR(VPU_JSM_MSG_BLOB_DEINIT);
|
|
IVPU_CASE_TO_STR(VPU_JSM_MSG_DYNDBG_CONTROL);
|
|
IVPU_CASE_TO_STR(VPU_JSM_MSG_JOB_DONE);
|
|
IVPU_CASE_TO_STR(VPU_JSM_MSG_ENGINE_RESET_DONE);
|
|
IVPU_CASE_TO_STR(VPU_JSM_MSG_ENGINE_PREEMPT_DONE);
|
|
IVPU_CASE_TO_STR(VPU_JSM_MSG_REGISTER_DB_DONE);
|
|
IVPU_CASE_TO_STR(VPU_JSM_MSG_UNREGISTER_DB_DONE);
|
|
IVPU_CASE_TO_STR(VPU_JSM_MSG_QUERY_ENGINE_HB_DONE);
|
|
IVPU_CASE_TO_STR(VPU_JSM_MSG_GET_POWER_LEVEL_COUNT_DONE);
|
|
IVPU_CASE_TO_STR(VPU_JSM_MSG_GET_POWER_LEVEL_DONE);
|
|
IVPU_CASE_TO_STR(VPU_JSM_MSG_SET_POWER_LEVEL_DONE);
|
|
IVPU_CASE_TO_STR(VPU_JSM_MSG_METRIC_STREAMER_OPEN_DONE);
|
|
IVPU_CASE_TO_STR(VPU_JSM_MSG_METRIC_STREAMER_CLOSE_DONE);
|
|
IVPU_CASE_TO_STR(VPU_JSM_MSG_TRACE_SET_CONFIG_RSP);
|
|
IVPU_CASE_TO_STR(VPU_JSM_MSG_TRACE_GET_CONFIG_RSP);
|
|
IVPU_CASE_TO_STR(VPU_JSM_MSG_TRACE_GET_CAPABILITY_RSP);
|
|
IVPU_CASE_TO_STR(VPU_JSM_MSG_TRACE_GET_NAME_RSP);
|
|
IVPU_CASE_TO_STR(VPU_JSM_MSG_SSID_RELEASE_DONE);
|
|
IVPU_CASE_TO_STR(VPU_JSM_MSG_METRIC_STREAMER_START_DONE);
|
|
IVPU_CASE_TO_STR(VPU_JSM_MSG_METRIC_STREAMER_STOP_DONE);
|
|
IVPU_CASE_TO_STR(VPU_JSM_MSG_METRIC_STREAMER_UPDATE_DONE);
|
|
IVPU_CASE_TO_STR(VPU_JSM_MSG_METRIC_STREAMER_INFO_DONE);
|
|
IVPU_CASE_TO_STR(VPU_JSM_MSG_METRIC_STREAMER_NOTIFICATION);
|
|
IVPU_CASE_TO_STR(VPU_JSM_MSG_SET_PRIORITY_BAND_SETUP_RSP);
|
|
IVPU_CASE_TO_STR(VPU_JSM_MSG_CREATE_CMD_QUEUE_RSP);
|
|
IVPU_CASE_TO_STR(VPU_JSM_MSG_DESTROY_CMD_QUEUE_RSP);
|
|
IVPU_CASE_TO_STR(VPU_JSM_MSG_SET_CONTEXT_SCHED_PROPERTIES_RSP);
|
|
IVPU_CASE_TO_STR(VPU_JSM_MSG_BLOB_DEINIT_DONE);
|
|
IVPU_CASE_TO_STR(VPU_JSM_MSG_DYNDBG_CONTROL_RSP);
|
|
IVPU_CASE_TO_STR(VPU_JSM_MSG_PWR_D0I3_ENTER);
|
|
IVPU_CASE_TO_STR(VPU_JSM_MSG_PWR_D0I3_ENTER_DONE);
|
|
IVPU_CASE_TO_STR(VPU_JSM_MSG_DCT_ENABLE);
|
|
IVPU_CASE_TO_STR(VPU_JSM_MSG_DCT_ENABLE_DONE);
|
|
IVPU_CASE_TO_STR(VPU_JSM_MSG_DCT_DISABLE);
|
|
IVPU_CASE_TO_STR(VPU_JSM_MSG_DCT_DISABLE_DONE);
|
|
}
|
|
#undef IVPU_CASE_TO_STR
|
|
|
|
return "Unknown JSM message type";
|
|
}
|
|
|
|
int ivpu_jsm_register_db(struct ivpu_device *vdev, u32 ctx_id, u32 db_id,
|
|
u64 jobq_base, u32 jobq_size)
|
|
{
|
|
struct vpu_jsm_msg req = { .type = VPU_JSM_MSG_REGISTER_DB };
|
|
struct vpu_jsm_msg resp;
|
|
int ret = 0;
|
|
|
|
req.payload.register_db.db_idx = db_id;
|
|
req.payload.register_db.jobq_base = jobq_base;
|
|
req.payload.register_db.jobq_size = jobq_size;
|
|
req.payload.register_db.host_ssid = ctx_id;
|
|
|
|
ret = ivpu_ipc_send_receive(vdev, &req, VPU_JSM_MSG_REGISTER_DB_DONE, &resp,
|
|
VPU_IPC_CHAN_ASYNC_CMD, vdev->timeout.jsm);
|
|
if (ret) {
|
|
ivpu_err_ratelimited(vdev, "Failed to register doorbell %d: %d\n", db_id, ret);
|
|
return ret;
|
|
}
|
|
|
|
ivpu_dbg(vdev, JSM, "Doorbell %d registered to context %d\n", db_id, ctx_id);
|
|
|
|
return 0;
|
|
}
|
|
|
|
int ivpu_jsm_unregister_db(struct ivpu_device *vdev, u32 db_id)
|
|
{
|
|
struct vpu_jsm_msg req = { .type = VPU_JSM_MSG_UNREGISTER_DB };
|
|
struct vpu_jsm_msg resp;
|
|
int ret = 0;
|
|
|
|
req.payload.unregister_db.db_idx = db_id;
|
|
|
|
ret = ivpu_ipc_send_receive(vdev, &req, VPU_JSM_MSG_UNREGISTER_DB_DONE, &resp,
|
|
VPU_IPC_CHAN_ASYNC_CMD, vdev->timeout.jsm);
|
|
if (ret) {
|
|
ivpu_warn_ratelimited(vdev, "Failed to unregister doorbell %d: %d\n", db_id, ret);
|
|
return ret;
|
|
}
|
|
|
|
ivpu_dbg(vdev, JSM, "Doorbell %d unregistered\n", db_id);
|
|
|
|
return 0;
|
|
}
|
|
|
|
int ivpu_jsm_get_heartbeat(struct ivpu_device *vdev, u32 engine, u64 *heartbeat)
|
|
{
|
|
struct vpu_jsm_msg req = { .type = VPU_JSM_MSG_QUERY_ENGINE_HB };
|
|
struct vpu_jsm_msg resp;
|
|
int ret;
|
|
|
|
if (engine > VPU_ENGINE_COPY)
|
|
return -EINVAL;
|
|
|
|
req.payload.query_engine_hb.engine_idx = engine;
|
|
|
|
ret = ivpu_ipc_send_receive(vdev, &req, VPU_JSM_MSG_QUERY_ENGINE_HB_DONE, &resp,
|
|
VPU_IPC_CHAN_ASYNC_CMD, vdev->timeout.jsm);
|
|
if (ret) {
|
|
ivpu_err_ratelimited(vdev, "Failed to get heartbeat from engine %d: %d\n",
|
|
engine, ret);
|
|
return ret;
|
|
}
|
|
|
|
*heartbeat = resp.payload.query_engine_hb_done.heartbeat;
|
|
return ret;
|
|
}
|
|
|
|
int ivpu_jsm_reset_engine(struct ivpu_device *vdev, u32 engine)
|
|
{
|
|
struct vpu_jsm_msg req = { .type = VPU_JSM_MSG_ENGINE_RESET };
|
|
struct vpu_jsm_msg resp;
|
|
int ret;
|
|
|
|
if (engine > VPU_ENGINE_COPY)
|
|
return -EINVAL;
|
|
|
|
req.payload.engine_reset.engine_idx = engine;
|
|
|
|
ret = ivpu_ipc_send_receive(vdev, &req, VPU_JSM_MSG_ENGINE_RESET_DONE, &resp,
|
|
VPU_IPC_CHAN_ASYNC_CMD, vdev->timeout.jsm);
|
|
if (ret)
|
|
ivpu_err_ratelimited(vdev, "Failed to reset engine %d: %d\n", engine, ret);
|
|
|
|
return ret;
|
|
}
|
|
|
|
int ivpu_jsm_preempt_engine(struct ivpu_device *vdev, u32 engine, u32 preempt_id)
|
|
{
|
|
struct vpu_jsm_msg req = { .type = VPU_JSM_MSG_ENGINE_PREEMPT };
|
|
struct vpu_jsm_msg resp;
|
|
int ret;
|
|
|
|
if (engine > VPU_ENGINE_COPY)
|
|
return -EINVAL;
|
|
|
|
req.payload.engine_preempt.engine_idx = engine;
|
|
req.payload.engine_preempt.preempt_id = preempt_id;
|
|
|
|
ret = ivpu_ipc_send_receive(vdev, &req, VPU_JSM_MSG_ENGINE_PREEMPT_DONE, &resp,
|
|
VPU_IPC_CHAN_ASYNC_CMD, vdev->timeout.jsm);
|
|
if (ret)
|
|
ivpu_err_ratelimited(vdev, "Failed to preempt engine %d: %d\n", engine, ret);
|
|
|
|
return ret;
|
|
}
|
|
|
|
int ivpu_jsm_dyndbg_control(struct ivpu_device *vdev, char *command, size_t size)
|
|
{
|
|
struct vpu_jsm_msg req = { .type = VPU_JSM_MSG_DYNDBG_CONTROL };
|
|
struct vpu_jsm_msg resp;
|
|
int ret;
|
|
|
|
strscpy(req.payload.dyndbg_control.dyndbg_cmd, command, VPU_DYNDBG_CMD_MAX_LEN);
|
|
|
|
ret = ivpu_ipc_send_receive(vdev, &req, VPU_JSM_MSG_DYNDBG_CONTROL_RSP, &resp,
|
|
VPU_IPC_CHAN_ASYNC_CMD, vdev->timeout.jsm);
|
|
if (ret)
|
|
ivpu_warn_ratelimited(vdev, "Failed to send command \"%s\": ret %d\n",
|
|
command, ret);
|
|
|
|
return ret;
|
|
}
|
|
|
|
int ivpu_jsm_trace_get_capability(struct ivpu_device *vdev, u32 *trace_destination_mask,
|
|
u64 *trace_hw_component_mask)
|
|
{
|
|
struct vpu_jsm_msg req = { .type = VPU_JSM_MSG_TRACE_GET_CAPABILITY };
|
|
struct vpu_jsm_msg resp;
|
|
int ret;
|
|
|
|
ret = ivpu_ipc_send_receive(vdev, &req, VPU_JSM_MSG_TRACE_GET_CAPABILITY_RSP, &resp,
|
|
VPU_IPC_CHAN_ASYNC_CMD, vdev->timeout.jsm);
|
|
if (ret) {
|
|
ivpu_warn_ratelimited(vdev, "Failed to get trace capability: %d\n", ret);
|
|
return ret;
|
|
}
|
|
|
|
*trace_destination_mask = resp.payload.trace_capability.trace_destination_mask;
|
|
*trace_hw_component_mask = resp.payload.trace_capability.trace_hw_component_mask;
|
|
|
|
return ret;
|
|
}
|
|
|
|
int ivpu_jsm_trace_set_config(struct ivpu_device *vdev, u32 trace_level, u32 trace_destination_mask,
|
|
u64 trace_hw_component_mask)
|
|
{
|
|
struct vpu_jsm_msg req = { .type = VPU_JSM_MSG_TRACE_SET_CONFIG };
|
|
struct vpu_jsm_msg resp;
|
|
int ret;
|
|
|
|
req.payload.trace_config.trace_level = trace_level;
|
|
req.payload.trace_config.trace_destination_mask = trace_destination_mask;
|
|
req.payload.trace_config.trace_hw_component_mask = trace_hw_component_mask;
|
|
|
|
ret = ivpu_ipc_send_receive(vdev, &req, VPU_JSM_MSG_TRACE_SET_CONFIG_RSP, &resp,
|
|
VPU_IPC_CHAN_ASYNC_CMD, vdev->timeout.jsm);
|
|
if (ret)
|
|
ivpu_warn_ratelimited(vdev, "Failed to set config: %d\n", ret);
|
|
|
|
return ret;
|
|
}
|
|
|
|
int ivpu_jsm_context_release(struct ivpu_device *vdev, u32 host_ssid)
|
|
{
|
|
struct vpu_jsm_msg req = { .type = VPU_JSM_MSG_SSID_RELEASE };
|
|
struct vpu_jsm_msg resp;
|
|
|
|
req.payload.ssid_release.host_ssid = host_ssid;
|
|
|
|
return ivpu_ipc_send_receive(vdev, &req, VPU_JSM_MSG_SSID_RELEASE_DONE, &resp,
|
|
VPU_IPC_CHAN_ASYNC_CMD, vdev->timeout.jsm);
|
|
}
|
|
|
|
int ivpu_jsm_pwr_d0i3_enter(struct ivpu_device *vdev)
|
|
{
|
|
struct vpu_jsm_msg req = { .type = VPU_JSM_MSG_PWR_D0I3_ENTER };
|
|
struct vpu_jsm_msg resp;
|
|
int ret;
|
|
|
|
if (IVPU_WA(disable_d0i3_msg))
|
|
return 0;
|
|
|
|
req.payload.pwr_d0i3_enter.send_response = 1;
|
|
|
|
ret = ivpu_ipc_send_receive_active(vdev, &req, VPU_JSM_MSG_PWR_D0I3_ENTER_DONE,
|
|
&resp, VPU_IPC_CHAN_GEN_CMD,
|
|
vdev->timeout.d0i3_entry_msg);
|
|
if (ret)
|
|
return ret;
|
|
|
|
return ivpu_hw_wait_for_idle(vdev);
|
|
}
|