1 // SPDX-License-Identifier: MIT
3 * Copyright(c) 2020, Intel Corporation. All rights reserved.
7 #include "intel_pxp_cmd.h"
8 #include "intel_pxp_session.h"
9 #include "gt/intel_context.h"
10 #include "gt/intel_engine_pm.h"
11 #include "gt/intel_gpu_commands.h"
12 #include "gt/intel_ring.h"
14 #include "i915_trace.h"
16 /* stall until prior PXP and MFX/HCP/HUC objects are cmopleted */
17 #define MFX_WAIT_PXP (MFX_WAIT | \
18 MFX_WAIT_DW0_PXP_SYNC_CONTROL_FLAG | \
19 MFX_WAIT_DW0_MFX_SYNC_CONTROL_FLAG)
21 static u32 *pxp_emit_session_selection(u32 *cs, u32 idx)
31 *cs++ = MI_SET_APPID | MI_SET_APPID_SESSION_ID(idx);
36 *cs++ = MI_FLUSH_DW | MI_FLUSH_DW_PROTECTED_MEM_EN |
37 MI_FLUSH_DW_OP_STOREDW | MI_FLUSH_DW_STORE_INDEX;
38 *cs++ = I915_GEM_HWS_PXP_ADDR | MI_FLUSH_DW_USE_GTT;
46 static u32 *pxp_emit_inline_termination(u32 *cs)
48 /* session inline termination */
49 *cs++ = CRYPTO_KEY_EXCHANGE;
55 static u32 *pxp_emit_session_termination(u32 *cs, u32 idx)
57 cs = pxp_emit_session_selection(cs, idx);
58 cs = pxp_emit_inline_termination(cs);
63 static u32 *pxp_emit_wait(u32 *cs)
65 /* wait for cmds to go through */
73 * if we ever need to terminate more than one session, we can submit multiple
74 * selections and terminations back-to-back with a single wait at the end
76 #define SELECTION_LEN 10
77 #define TERMINATION_LEN 2
78 #define SESSION_TERMINATION_LEN(x) ((SELECTION_LEN + TERMINATION_LEN) * (x))
81 static void pxp_request_commit(struct i915_request *rq)
83 struct i915_sched_attr attr = { .priority = I915_PRIORITY_MAX };
84 struct intel_timeline * const tl = i915_request_timeline(rq);
86 lockdep_unpin_lock(&tl->mutex, rq->cookie);
88 trace_i915_request_add(rq);
89 __i915_request_commit(rq);
90 __i915_request_queue(rq, &attr);
92 mutex_unlock(&tl->mutex);
95 int intel_pxp_terminate_session(struct intel_pxp *pxp, u32 id)
97 struct i915_request *rq;
98 struct intel_context *ce = pxp->ce;
102 if (!intel_pxp_is_enabled(pxp))
105 rq = i915_request_create(ce);
109 if (ce->engine->emit_init_breadcrumb) {
110 err = ce->engine->emit_init_breadcrumb(rq);
115 cs = intel_ring_begin(rq, SESSION_TERMINATION_LEN(1) + WAIT_LEN);
121 cs = pxp_emit_session_termination(cs, id);
122 cs = pxp_emit_wait(cs);
124 intel_ring_advance(rq, cs);
127 i915_request_get(rq);
130 i915_request_set_error_once(rq, err);
132 pxp_request_commit(rq);
134 if (!err && i915_request_wait(rq, 0, HZ / 5) < 0)
137 i915_request_put(rq);