xref: /linux/drivers/misc/mei/platform-vsc.c (revision 8a5f956a9fb7d74fff681145082acfad5afa6bb8)
1 // SPDX-License-Identifier: GPL-2.0-only
2 /*
3  * Copyright (c) 2023, Intel Corporation.
4  * Intel Visual Sensing Controller Interface Linux driver
5  */
6 
7 #include <linux/align.h>
8 #include <linux/cache.h>
9 #include <linux/cleanup.h>
10 #include <linux/iopoll.h>
11 #include <linux/list.h>
12 #include <linux/mei.h>
13 #include <linux/module.h>
14 #include <linux/mutex.h>
15 #include <linux/overflow.h>
16 #include <linux/platform_device.h>
17 #include <linux/pm_runtime.h>
18 #include <linux/timekeeping.h>
19 #include <linux/types.h>
20 
21 #include <asm-generic/bug.h>
22 #include <linux/unaligned.h>
23 
24 #include "mei_dev.h"
25 #include "vsc-tp.h"
26 
27 #define MEI_VSC_DRV_NAME		"intel_vsc"
28 
29 #define MEI_VSC_MAX_MSG_SIZE		512
30 
31 #define MEI_VSC_POLL_DELAY_US		(100 * USEC_PER_MSEC)
32 #define MEI_VSC_POLL_TIMEOUT_US		(400 * USEC_PER_MSEC)
33 
34 #define mei_dev_to_vsc_hw(dev)		((struct mei_vsc_hw *)((dev)->hw))
35 
36 struct mei_vsc_host_timestamp {
37 	u64 realtime;
38 	u64 boottime;
39 };
40 
41 struct mei_vsc_hw {
42 	struct vsc_tp *tp;
43 
44 	bool fw_ready;
45 	bool host_ready;
46 
47 	atomic_t write_lock_cnt;
48 
49 	u32 rx_len;
50 	u32 rx_hdr;
51 
52 	/* buffer for tx */
53 	char tx_buf[MEI_VSC_MAX_MSG_SIZE + sizeof(struct mei_msg_hdr)] ____cacheline_aligned;
54 	/* buffer for rx */
55 	char rx_buf[MEI_VSC_MAX_MSG_SIZE + sizeof(struct mei_msg_hdr)] ____cacheline_aligned;
56 };
57 
58 static int mei_vsc_read_helper(struct mei_vsc_hw *hw, u8 *buf,
59 			       u32 max_len)
60 {
61 	struct mei_vsc_host_timestamp ts = {
62 		.realtime = ktime_to_ns(ktime_get_real()),
63 		.boottime = ktime_to_ns(ktime_get_boottime()),
64 	};
65 
66 	return vsc_tp_xfer(hw->tp, VSC_TP_CMD_READ, &ts, sizeof(ts),
67 			   buf, max_len);
68 }
69 
70 static int mei_vsc_write_helper(struct mei_vsc_hw *hw, u8 *buf, u32 len)
71 {
72 	u8 status;
73 
74 	return vsc_tp_xfer(hw->tp, VSC_TP_CMD_WRITE, buf, len, &status,
75 			   sizeof(status));
76 }
77 
78 static int mei_vsc_fw_status(struct mei_device *mei_dev,
79 			     struct mei_fw_status *fw_status)
80 {
81 	if (!fw_status)
82 		return -EINVAL;
83 
84 	fw_status->count = 0;
85 
86 	return 0;
87 }
88 
89 static inline enum mei_pg_state mei_vsc_pg_state(struct mei_device *mei_dev)
90 {
91 	return MEI_PG_OFF;
92 }
93 
94 static void mei_vsc_intr_enable(struct mei_device *mei_dev)
95 {
96 	struct mei_vsc_hw *hw = mei_dev_to_vsc_hw(mei_dev);
97 
98 	vsc_tp_intr_enable(hw->tp);
99 }
100 
101 static void mei_vsc_intr_disable(struct mei_device *mei_dev)
102 {
103 	struct mei_vsc_hw *hw = mei_dev_to_vsc_hw(mei_dev);
104 
105 	vsc_tp_intr_disable(hw->tp);
106 }
107 
108 /* mei framework requires this ops */
109 static void mei_vsc_intr_clear(struct mei_device *mei_dev)
110 {
111 }
112 
113 /* wait for pending irq handler */
114 static void mei_vsc_synchronize_irq(struct mei_device *mei_dev)
115 {
116 	struct mei_vsc_hw *hw = mei_dev_to_vsc_hw(mei_dev);
117 
118 	vsc_tp_intr_synchronize(hw->tp);
119 }
120 
121 static int mei_vsc_hw_config(struct mei_device *mei_dev)
122 {
123 	return 0;
124 }
125 
126 static bool mei_vsc_host_is_ready(struct mei_device *mei_dev)
127 {
128 	struct mei_vsc_hw *hw = mei_dev_to_vsc_hw(mei_dev);
129 
130 	return hw->host_ready;
131 }
132 
133 static bool mei_vsc_hw_is_ready(struct mei_device *mei_dev)
134 {
135 	struct mei_vsc_hw *hw = mei_dev_to_vsc_hw(mei_dev);
136 
137 	return hw->fw_ready;
138 }
139 
140 static int mei_vsc_hw_start(struct mei_device *mei_dev)
141 {
142 	struct mei_vsc_hw *hw = mei_dev_to_vsc_hw(mei_dev);
143 	int ret, rlen;
144 	u8 buf;
145 
146 	hw->host_ready = true;
147 
148 	vsc_tp_intr_enable(hw->tp);
149 
150 	ret = read_poll_timeout(mei_vsc_read_helper, rlen,
151 				rlen >= 0, MEI_VSC_POLL_DELAY_US,
152 				MEI_VSC_POLL_TIMEOUT_US, true,
153 				hw, &buf, sizeof(buf));
154 	if (ret) {
155 		dev_err(mei_dev->dev, "wait fw ready failed: %d\n", ret);
156 		return ret;
157 	}
158 
159 	hw->fw_ready = true;
160 
161 	return 0;
162 }
163 
164 static bool mei_vsc_hbuf_is_ready(struct mei_device *mei_dev)
165 {
166 	struct mei_vsc_hw *hw = mei_dev_to_vsc_hw(mei_dev);
167 
168 	return atomic_read(&hw->write_lock_cnt) == 0;
169 }
170 
171 static int mei_vsc_hbuf_empty_slots(struct mei_device *mei_dev)
172 {
173 	return MEI_VSC_MAX_MSG_SIZE / MEI_SLOT_SIZE;
174 }
175 
176 static u32 mei_vsc_hbuf_depth(const struct mei_device *mei_dev)
177 {
178 	return MEI_VSC_MAX_MSG_SIZE / MEI_SLOT_SIZE;
179 }
180 
181 static int mei_vsc_write(struct mei_device *mei_dev,
182 			 const void *hdr, size_t hdr_len,
183 			 const void *data, size_t data_len)
184 {
185 	struct mei_vsc_hw *hw = mei_dev_to_vsc_hw(mei_dev);
186 	char *buf = hw->tx_buf;
187 	int ret;
188 
189 	if (WARN_ON(!hdr || !IS_ALIGNED(hdr_len, 4)))
190 		return -EINVAL;
191 
192 	if (!data || data_len > MEI_VSC_MAX_MSG_SIZE)
193 		return -EINVAL;
194 
195 	atomic_inc(&hw->write_lock_cnt);
196 
197 	memcpy(buf, hdr, hdr_len);
198 	memcpy(buf + hdr_len, data, data_len);
199 
200 	ret = mei_vsc_write_helper(hw, buf, hdr_len + data_len);
201 
202 	atomic_dec_if_positive(&hw->write_lock_cnt);
203 
204 	return ret < 0 ? ret : 0;
205 }
206 
207 static inline u32 mei_vsc_read(const struct mei_device *mei_dev)
208 {
209 	struct mei_vsc_hw *hw = mei_dev_to_vsc_hw(mei_dev);
210 	int ret;
211 
212 	ret = mei_vsc_read_helper(hw, hw->rx_buf, sizeof(hw->rx_buf));
213 	if (ret < 0 || ret < sizeof(u32))
214 		return 0;
215 	hw->rx_len = ret;
216 
217 	hw->rx_hdr = get_unaligned_le32(hw->rx_buf);
218 
219 	return hw->rx_hdr;
220 }
221 
222 static int mei_vsc_count_full_read_slots(struct mei_device *mei_dev)
223 {
224 	return MEI_VSC_MAX_MSG_SIZE / MEI_SLOT_SIZE;
225 }
226 
227 static int mei_vsc_read_slots(struct mei_device *mei_dev, unsigned char *buf,
228 			      unsigned long len)
229 {
230 	struct mei_vsc_hw *hw = mei_dev_to_vsc_hw(mei_dev);
231 	struct mei_msg_hdr *hdr;
232 
233 	hdr = (struct mei_msg_hdr *)&hw->rx_hdr;
234 	if (len != hdr->length || hdr->length + sizeof(*hdr) != hw->rx_len)
235 		return -EINVAL;
236 
237 	memcpy(buf, hw->rx_buf + sizeof(*hdr), len);
238 
239 	return 0;
240 }
241 
242 static bool mei_vsc_pg_in_transition(struct mei_device *mei_dev)
243 {
244 	return mei_dev->pg_event >= MEI_PG_EVENT_WAIT &&
245 	       mei_dev->pg_event <= MEI_PG_EVENT_INTR_WAIT;
246 }
247 
248 static bool mei_vsc_pg_is_enabled(struct mei_device *mei_dev)
249 {
250 	return false;
251 }
252 
253 static int mei_vsc_hw_reset(struct mei_device *mei_dev, bool intr_enable)
254 {
255 	struct mei_vsc_hw *hw = mei_dev_to_vsc_hw(mei_dev);
256 
257 	vsc_tp_reset(hw->tp);
258 
259 	if (!intr_enable)
260 		return 0;
261 
262 	return vsc_tp_init(hw->tp, mei_dev->dev);
263 }
264 
265 static const struct mei_hw_ops mei_vsc_hw_ops = {
266 	.fw_status = mei_vsc_fw_status,
267 	.pg_state = mei_vsc_pg_state,
268 
269 	.host_is_ready = mei_vsc_host_is_ready,
270 	.hw_is_ready = mei_vsc_hw_is_ready,
271 	.hw_reset = mei_vsc_hw_reset,
272 	.hw_config = mei_vsc_hw_config,
273 	.hw_start = mei_vsc_hw_start,
274 
275 	.pg_in_transition = mei_vsc_pg_in_transition,
276 	.pg_is_enabled = mei_vsc_pg_is_enabled,
277 
278 	.intr_clear = mei_vsc_intr_clear,
279 	.intr_enable = mei_vsc_intr_enable,
280 	.intr_disable = mei_vsc_intr_disable,
281 	.synchronize_irq = mei_vsc_synchronize_irq,
282 
283 	.hbuf_free_slots = mei_vsc_hbuf_empty_slots,
284 	.hbuf_is_ready = mei_vsc_hbuf_is_ready,
285 	.hbuf_depth = mei_vsc_hbuf_depth,
286 	.write = mei_vsc_write,
287 
288 	.rdbuf_full_slots = mei_vsc_count_full_read_slots,
289 	.read_hdr = mei_vsc_read,
290 	.read = mei_vsc_read_slots,
291 };
292 
293 static void mei_vsc_event_cb(void *context)
294 {
295 	struct mei_device *mei_dev = context;
296 	struct mei_vsc_hw *hw = mei_dev_to_vsc_hw(mei_dev);
297 	struct list_head cmpl_list;
298 	s32 slots;
299 	int ret;
300 
301 	if (mei_dev->dev_state == MEI_DEV_RESETTING ||
302 	    mei_dev->dev_state == MEI_DEV_INITIALIZING)
303 		return;
304 
305 	INIT_LIST_HEAD(&cmpl_list);
306 
307 	guard(mutex)(&mei_dev->device_lock);
308 
309 	while (vsc_tp_need_read(hw->tp)) {
310 		/* check slots available for reading */
311 		slots = mei_count_full_read_slots(mei_dev);
312 
313 		ret = mei_irq_read_handler(mei_dev, &cmpl_list, &slots);
314 		if (ret) {
315 			if (ret != -ENODATA) {
316 				if (mei_dev->dev_state != MEI_DEV_RESETTING &&
317 				    mei_dev->dev_state != MEI_DEV_POWER_DOWN)
318 					schedule_work(&mei_dev->reset_work);
319 			}
320 
321 			return;
322 		}
323 	}
324 
325 	mei_dev->hbuf_is_ready = mei_hbuf_is_ready(mei_dev);
326 	ret = mei_irq_write_handler(mei_dev, &cmpl_list);
327 	if (ret)
328 		dev_err(mei_dev->dev, "dispatch write request failed: %d\n", ret);
329 
330 	mei_dev->hbuf_is_ready = mei_hbuf_is_ready(mei_dev);
331 	mei_irq_compl_handler(mei_dev, &cmpl_list);
332 }
333 
334 static int mei_vsc_probe(struct platform_device *pdev)
335 {
336 	struct device *dev = &pdev->dev;
337 	struct mei_device *mei_dev;
338 	struct mei_vsc_hw *hw;
339 	struct vsc_tp *tp;
340 	int ret;
341 
342 	tp = *(struct vsc_tp **)dev_get_platdata(dev);
343 	if (!tp)
344 		return dev_err_probe(dev, -ENODEV, "no platform data\n");
345 
346 	mei_dev = devm_kzalloc(dev, size_add(sizeof(*mei_dev), sizeof(*hw)),
347 			       GFP_KERNEL);
348 	if (!mei_dev)
349 		return -ENOMEM;
350 
351 	mei_device_init(mei_dev, dev, false, &mei_vsc_hw_ops);
352 	mei_dev->fw_f_fw_ver_supported = 0;
353 	mei_dev->kind = "ivsc";
354 
355 	hw = mei_dev_to_vsc_hw(mei_dev);
356 	atomic_set(&hw->write_lock_cnt, 0);
357 	hw->tp = tp;
358 
359 	platform_set_drvdata(pdev, mei_dev);
360 
361 	vsc_tp_register_event_cb(tp, mei_vsc_event_cb, mei_dev);
362 
363 	ret = mei_start(mei_dev);
364 	if (ret) {
365 		dev_err_probe(dev, ret, "init hw failed\n");
366 		goto err_cancel;
367 	}
368 
369 	ret = mei_register(mei_dev, dev);
370 	if (ret)
371 		goto err_stop;
372 
373 	pm_runtime_enable(mei_dev->dev);
374 
375 	return 0;
376 
377 err_stop:
378 	mei_stop(mei_dev);
379 
380 err_cancel:
381 	mei_cancel_work(mei_dev);
382 
383 	vsc_tp_register_event_cb(tp, NULL, NULL);
384 
385 	mei_disable_interrupts(mei_dev);
386 
387 	return ret;
388 }
389 
390 static void mei_vsc_remove(struct platform_device *pdev)
391 {
392 	struct mei_device *mei_dev = platform_get_drvdata(pdev);
393 	struct mei_vsc_hw *hw = mei_dev_to_vsc_hw(mei_dev);
394 
395 	pm_runtime_disable(mei_dev->dev);
396 
397 	mei_stop(mei_dev);
398 
399 	vsc_tp_register_event_cb(hw->tp, NULL, NULL);
400 
401 	mei_disable_interrupts(mei_dev);
402 
403 	mei_deregister(mei_dev);
404 }
405 
406 static int mei_vsc_suspend(struct device *dev)
407 {
408 	struct mei_device *mei_dev;
409 	int ret = 0;
410 
411 	mei_dev = dev_get_drvdata(dev);
412 	if (!mei_dev)
413 		return -ENODEV;
414 
415 	mutex_lock(&mei_dev->device_lock);
416 
417 	if (!mei_write_is_idle(mei_dev))
418 		ret = -EAGAIN;
419 
420 	mutex_unlock(&mei_dev->device_lock);
421 
422 	return ret;
423 }
424 
425 static int mei_vsc_resume(struct device *dev)
426 {
427 	struct mei_device *mei_dev;
428 
429 	mei_dev = dev_get_drvdata(dev);
430 	if (!mei_dev)
431 		return -ENODEV;
432 
433 	return 0;
434 }
435 
436 static DEFINE_SIMPLE_DEV_PM_OPS(mei_vsc_pm_ops, mei_vsc_suspend, mei_vsc_resume);
437 
438 static const struct platform_device_id mei_vsc_id_table[] = {
439 	{ MEI_VSC_DRV_NAME },
440 	{ /* sentinel */ }
441 };
442 MODULE_DEVICE_TABLE(platform, mei_vsc_id_table);
443 
444 static struct platform_driver mei_vsc_drv = {
445 	.probe = mei_vsc_probe,
446 	.remove = mei_vsc_remove,
447 	.id_table = mei_vsc_id_table,
448 	.driver = {
449 		.name = MEI_VSC_DRV_NAME,
450 		.pm = &mei_vsc_pm_ops,
451 		.probe_type = PROBE_PREFER_ASYNCHRONOUS,
452 	},
453 };
454 module_platform_driver(mei_vsc_drv);
455 
456 MODULE_AUTHOR("Wentong Wu <wentong.wu@intel.com>");
457 MODULE_AUTHOR("Zhifeng Wang <zhifeng.wang@intel.com>");
458 MODULE_DESCRIPTION("Intel Visual Sensing Controller Interface");
459 MODULE_LICENSE("GPL");
460 MODULE_IMPORT_NS("VSC_TP");
461