1 // SPDX-License-Identifier: MIT
2 /*
3  * Copyright © 2023 Intel Corporation
4  */
5 
6 #include "gt/intel_context.h"
7 #include "gt/intel_engine_pm.h"
8 #include "gt/intel_gpu_commands.h"
9 #include "gt/intel_gt.h"
10 #include "gt/intel_ring.h"
11 #include "intel_gsc_uc_heci_cmd_submit.h"
12 
13 struct gsc_heci_pkt {
14 	u64 addr_in;
15 	u32 size_in;
16 	u64 addr_out;
17 	u32 size_out;
18 };
19 
emit_gsc_heci_pkt(struct i915_request * rq,struct gsc_heci_pkt * pkt)20 static int emit_gsc_heci_pkt(struct i915_request *rq, struct gsc_heci_pkt *pkt)
21 {
22 	u32 *cs;
23 
24 	cs = intel_ring_begin(rq, 8);
25 	if (IS_ERR(cs))
26 		return PTR_ERR(cs);
27 
28 	*cs++ = GSC_HECI_CMD_PKT;
29 	*cs++ = lower_32_bits(pkt->addr_in);
30 	*cs++ = upper_32_bits(pkt->addr_in);
31 	*cs++ = pkt->size_in;
32 	*cs++ = lower_32_bits(pkt->addr_out);
33 	*cs++ = upper_32_bits(pkt->addr_out);
34 	*cs++ = pkt->size_out;
35 	*cs++ = 0;
36 
37 	intel_ring_advance(rq, cs);
38 
39 	return 0;
40 }
41 
intel_gsc_uc_heci_cmd_submit_packet(struct intel_gsc_uc * gsc,u64 addr_in,u32 size_in,u64 addr_out,u32 size_out)42 int intel_gsc_uc_heci_cmd_submit_packet(struct intel_gsc_uc *gsc, u64 addr_in,
43 					u32 size_in, u64 addr_out,
44 					u32 size_out)
45 {
46 	struct intel_context *ce = gsc->ce;
47 	struct i915_request *rq;
48 	struct gsc_heci_pkt pkt = {
49 	.addr_in = addr_in,
50 	.size_in = size_in,
51 	.addr_out = addr_out,
52 	.size_out = size_out
53 	};
54 	int err;
55 
56 	if (!ce)
57 		return -ENODEV;
58 
59 	rq = i915_request_create(ce);
60 	if (IS_ERR(rq))
61 		return PTR_ERR(rq);
62 
63 	if (ce->engine->emit_init_breadcrumb) {
64 		err = ce->engine->emit_init_breadcrumb(rq);
65 		if (err)
66 			goto out_rq;
67 	}
68 
69 	err = emit_gsc_heci_pkt(rq, &pkt);
70 
71 	if (err)
72 		goto out_rq;
73 
74 	err = ce->engine->emit_flush(rq, 0);
75 
76 out_rq:
77 	i915_request_get(rq);
78 
79 	if (unlikely(err))
80 		i915_request_set_error_once(rq, err);
81 
82 	i915_request_add(rq);
83 
84 	if (!err) {
85 		/*
86 		 * Start timeout for i915_request_wait only after considering one possible
87 		 * pending GSC-HECI submission cycle on the other (non-privileged) path.
88 		 */
89 		if (wait_for(i915_request_started(rq), GSC_HECI_REPLY_LATENCY_MS))
90 			drm_dbg(&gsc_uc_to_gt(gsc)->i915->drm,
91 				"Delay in gsc-heci-priv submission to gsccs-hw");
92 		if (i915_request_wait(rq, 0, msecs_to_jiffies(GSC_HECI_REPLY_LATENCY_MS)) < 0)
93 			err = -ETIME;
94 	}
95 
96 	i915_request_put(rq);
97 
98 	if (err)
99 		drm_err(&gsc_uc_to_gt(gsc)->i915->drm,
100 			"Request submission for GSC heci cmd failed (%d)\n",
101 			err);
102 
103 	return err;
104 }
105 
intel_gsc_uc_heci_cmd_emit_mtl_header(struct intel_gsc_mtl_header * header,u8 heci_client_id,u32 message_size,u64 host_session_id)106 void intel_gsc_uc_heci_cmd_emit_mtl_header(struct intel_gsc_mtl_header *header,
107 					   u8 heci_client_id, u32 message_size,
108 					   u64 host_session_id)
109 {
110 	host_session_id &= ~HOST_SESSION_MASK;
111 	if (host_session_id && heci_client_id == HECI_MEADDRESS_PXP)
112 		host_session_id |= HOST_SESSION_PXP_SINGLE;
113 
114 	header->validity_marker = GSC_HECI_VALIDITY_MARKER;
115 	header->heci_client_id = heci_client_id;
116 	header->host_session_handle = host_session_id;
117 	header->header_version = MTL_GSC_HEADER_VERSION;
118 	header->message_size = message_size;
119 }
120 
121 static void
emit_gsc_heci_pkt_nonpriv(u32 * cmd,struct intel_gsc_heci_non_priv_pkt * pkt)122 emit_gsc_heci_pkt_nonpriv(u32 *cmd, struct intel_gsc_heci_non_priv_pkt *pkt)
123 {
124 	*cmd++ = GSC_HECI_CMD_PKT;
125 	*cmd++ = lower_32_bits(pkt->addr_in);
126 	*cmd++ = upper_32_bits(pkt->addr_in);
127 	*cmd++ = pkt->size_in;
128 	*cmd++ = lower_32_bits(pkt->addr_out);
129 	*cmd++ = upper_32_bits(pkt->addr_out);
130 	*cmd++ = pkt->size_out;
131 	*cmd++ = 0;
132 	*cmd++ = MI_BATCH_BUFFER_END;
133 }
134 
135 int
intel_gsc_uc_heci_cmd_submit_nonpriv(struct intel_gsc_uc * gsc,struct intel_context * ce,struct intel_gsc_heci_non_priv_pkt * pkt,u32 * cmd,int timeout_ms)136 intel_gsc_uc_heci_cmd_submit_nonpriv(struct intel_gsc_uc *gsc,
137 				     struct intel_context *ce,
138 				     struct intel_gsc_heci_non_priv_pkt *pkt,
139 				     u32 *cmd, int timeout_ms)
140 {
141 	struct intel_engine_cs *engine;
142 	struct i915_gem_ww_ctx ww;
143 	struct i915_request *rq;
144 	int err, trials = 0;
145 
146 	i915_gem_ww_ctx_init(&ww, false);
147 retry:
148 	err = i915_gem_object_lock(pkt->bb_vma->obj, &ww);
149 	if (err)
150 		goto out_ww;
151 	err = i915_gem_object_lock(pkt->heci_pkt_vma->obj, &ww);
152 	if (err)
153 		goto out_ww;
154 	err = intel_context_pin_ww(ce, &ww);
155 	if (err)
156 		goto out_ww;
157 
158 	rq = i915_request_create(ce);
159 	if (IS_ERR(rq)) {
160 		err = PTR_ERR(rq);
161 		goto out_unpin_ce;
162 	}
163 
164 	emit_gsc_heci_pkt_nonpriv(cmd, pkt);
165 
166 	err = i915_vma_move_to_active(pkt->bb_vma, rq, 0);
167 	if (err)
168 		goto out_rq;
169 	err = i915_vma_move_to_active(pkt->heci_pkt_vma, rq, EXEC_OBJECT_WRITE);
170 	if (err)
171 		goto out_rq;
172 
173 	engine = rq->context->engine;
174 	if (engine->emit_init_breadcrumb) {
175 		err = engine->emit_init_breadcrumb(rq);
176 		if (err)
177 			goto out_rq;
178 	}
179 
180 	err = engine->emit_bb_start(rq, i915_vma_offset(pkt->bb_vma), PAGE_SIZE, 0);
181 	if (err)
182 		goto out_rq;
183 
184 	err = ce->engine->emit_flush(rq, 0);
185 	if (err)
186 		drm_err(&gsc_uc_to_gt(gsc)->i915->drm,
187 			"Failed emit-flush for gsc-heci-non-priv-pkterr=%d\n", err);
188 
189 out_rq:
190 	i915_request_get(rq);
191 
192 	if (unlikely(err))
193 		i915_request_set_error_once(rq, err);
194 
195 	i915_request_add(rq);
196 
197 	if (!err) {
198 		/*
199 		 * Start timeout for i915_request_wait only after considering one possible
200 		 * pending GSC-HECI submission cycle on the other (privileged) path.
201 		 */
202 		if (wait_for(i915_request_started(rq), GSC_HECI_REPLY_LATENCY_MS))
203 			drm_dbg(&gsc_uc_to_gt(gsc)->i915->drm,
204 				"Delay in gsc-heci-non-priv submission to gsccs-hw");
205 		if (i915_request_wait(rq, I915_WAIT_INTERRUPTIBLE,
206 				      msecs_to_jiffies(timeout_ms)) < 0)
207 			err = -ETIME;
208 	}
209 
210 	i915_request_put(rq);
211 
212 out_unpin_ce:
213 	intel_context_unpin(ce);
214 out_ww:
215 	if (err == -EDEADLK) {
216 		err = i915_gem_ww_ctx_backoff(&ww);
217 		if (!err) {
218 			if (++trials < 10)
219 				goto retry;
220 			else
221 				err = -EAGAIN;
222 		}
223 	}
224 	i915_gem_ww_ctx_fini(&ww);
225 
226 	return err;
227 }
228