xref: /linux/drivers/media/pci/intel/ivsc/mei_csi.c (revision 576d7fed09c7edbae7600f29a8a3ed6c1ead904f)
1 // SPDX-License-Identifier: GPL-2.0-only
2 /*
3  * Copyright (C) 2023 Intel Corporation. All rights reserved.
4  * Intel Visual Sensing Controller CSI Linux driver
5  */
6 
7 /*
8  * To set ownership of CSI-2 link and to configure CSI-2 link, there
9  * are specific commands, which are sent via MEI protocol. The send
10  * command function uses "completion" as a synchronization mechanism.
11  * The response for command is received via a mei callback which wakes
12  * up the caller. There can be only one outstanding command at a time.
13  */
14 
15 #include <linux/completion.h>
16 #include <linux/delay.h>
17 #include <linux/kernel.h>
18 #include <linux/math64.h>
19 #include <linux/mei_cl_bus.h>
20 #include <linux/module.h>
21 #include <linux/mutex.h>
22 #include <linux/pm_runtime.h>
23 #include <linux/slab.h>
24 #include <linux/units.h>
25 #include <linux/uuid.h>
26 #include <linux/workqueue.h>
27 
28 #include <media/v4l2-async.h>
29 #include <media/v4l2-ctrls.h>
30 #include <media/v4l2-fwnode.h>
31 #include <media/v4l2-subdev.h>
32 
33 #define MEI_CSI_ENTITY_NAME "Intel IVSC CSI"
34 
35 #define MEI_CSI_LINK_FREQ_400MHZ 400000000ULL
36 
37 /* the 5s used here is based on experiment */
38 #define CSI_CMD_TIMEOUT (5 * HZ)
39 /* to setup CSI-2 link an extra delay needed and determined experimentally */
40 #define CSI_FW_READY_DELAY_MS 100
41 /* link frequency unit is 100kHz */
42 #define CSI_LINK_FREQ(x) ((u32)(div_u64(x, 100 * HZ_PER_KHZ)))
43 
44 /*
45  * identify the command id supported by firmware
46  * IPC, as well as the privacy notification id
47  * used when processing privacy event.
48  */
49 enum csi_cmd_id {
50 	/* used to set csi ownership */
51 	CSI_SET_OWNER = 0,
52 
53 	/* used to configure CSI-2 link */
54 	CSI_SET_CONF = 2,
55 
56 	/* privacy notification id used when privacy state changes */
57 	CSI_PRIVACY_NOTIF = 6,
58 };
59 
60 /* CSI-2 link ownership definition */
61 enum csi_link_owner {
62 	CSI_LINK_IVSC,
63 	CSI_LINK_HOST,
64 };
65 
66 /* privacy status definition */
67 enum ivsc_privacy_status {
68 	CSI_PRIVACY_OFF,
69 	CSI_PRIVACY_ON,
70 	CSI_PRIVACY_MAX,
71 };
72 
73 enum csi_pads {
74 	CSI_PAD_SOURCE,
75 	CSI_PAD_SINK,
76 	CSI_NUM_PADS
77 };
78 
79 /* configuration of the CSI-2 link between host and IVSC */
80 struct csi_link_cfg {
81 	/* number of data lanes used on the CSI-2 link */
82 	u32 nr_of_lanes;
83 
84 	/* frequency of the CSI-2 link */
85 	u32 link_freq;
86 
87 	/* for future use */
88 	u32 rsvd[2];
89 } __packed;
90 
91 /* CSI command structure */
92 struct csi_cmd {
93 	u32 cmd_id;
94 	union _cmd_param {
95 		u32 param;
96 		struct csi_link_cfg conf;
97 	} param;
98 } __packed;
99 
100 /* CSI notification structure */
101 struct csi_notif {
102 	u32 cmd_id;
103 	int status;
104 	union _resp_cont {
105 		u32 cont;
106 		struct csi_link_cfg conf;
107 	} cont;
108 } __packed;
109 
110 struct mei_csi {
111 	struct mei_cl_device *cldev;
112 
113 	/* command response */
114 	struct csi_notif cmd_response;
115 	/* used to wait for command response from firmware */
116 	struct completion cmd_completion;
117 	/* protect command download */
118 	struct mutex lock;
119 
120 	struct v4l2_subdev subdev;
121 	struct v4l2_subdev *remote;
122 	struct v4l2_async_notifier notifier;
123 	struct v4l2_ctrl_handler ctrl_handler;
124 	struct v4l2_ctrl *freq_ctrl;
125 	struct v4l2_ctrl *privacy_ctrl;
126 	unsigned int remote_pad;
127 	/* start streaming or not */
128 	int streaming;
129 
130 	struct media_pad pads[CSI_NUM_PADS];
131 	struct v4l2_mbus_framefmt format_mbus[CSI_NUM_PADS];
132 
133 	/* number of data lanes used on the CSI-2 link */
134 	u32 nr_of_lanes;
135 	/* frequency of the CSI-2 link */
136 	u64 link_freq;
137 
138 	/* privacy status */
139 	enum ivsc_privacy_status status;
140 };
141 
142 static const struct v4l2_mbus_framefmt mei_csi_format_mbus_default = {
143 	.width = 1,
144 	.height = 1,
145 	.code = MEDIA_BUS_FMT_Y8_1X8,
146 	.field = V4L2_FIELD_NONE,
147 };
148 
149 static s64 link_freq_menu_items[] = {
150 	MEI_CSI_LINK_FREQ_400MHZ
151 };
152 
153 static inline struct mei_csi *notifier_to_csi(struct v4l2_async_notifier *n)
154 {
155 	return container_of(n, struct mei_csi, notifier);
156 }
157 
158 static inline struct mei_csi *sd_to_csi(struct v4l2_subdev *sd)
159 {
160 	return container_of(sd, struct mei_csi, subdev);
161 }
162 
163 static inline struct mei_csi *ctrl_to_csi(struct v4l2_ctrl *ctrl)
164 {
165 	return container_of(ctrl->handler, struct mei_csi, ctrl_handler);
166 }
167 
168 /* send a command to firmware and mutex must be held by caller */
169 static int mei_csi_send(struct mei_csi *csi, u8 *buf, size_t len)
170 {
171 	struct csi_cmd *cmd = (struct csi_cmd *)buf;
172 	int ret;
173 
174 	reinit_completion(&csi->cmd_completion);
175 
176 	ret = mei_cldev_send(csi->cldev, buf, len);
177 	if (ret < 0)
178 		goto out;
179 
180 	ret = wait_for_completion_killable_timeout(&csi->cmd_completion,
181 						   CSI_CMD_TIMEOUT);
182 	if (ret < 0) {
183 		goto out;
184 	} else if (!ret) {
185 		ret = -ETIMEDOUT;
186 		goto out;
187 	}
188 
189 	/* command response status */
190 	ret = csi->cmd_response.status;
191 	if (ret) {
192 		ret = -EINVAL;
193 		goto out;
194 	}
195 
196 	if (csi->cmd_response.cmd_id != cmd->cmd_id)
197 		ret = -EINVAL;
198 
199 out:
200 	return ret;
201 }
202 
203 /* set CSI-2 link ownership */
204 static int csi_set_link_owner(struct mei_csi *csi, enum csi_link_owner owner)
205 {
206 	struct csi_cmd cmd = { 0 };
207 	size_t cmd_size;
208 	int ret;
209 
210 	cmd.cmd_id = CSI_SET_OWNER;
211 	cmd.param.param = owner;
212 	cmd_size = sizeof(cmd.cmd_id) + sizeof(cmd.param.param);
213 
214 	mutex_lock(&csi->lock);
215 
216 	ret = mei_csi_send(csi, (u8 *)&cmd, cmd_size);
217 
218 	mutex_unlock(&csi->lock);
219 
220 	return ret;
221 }
222 
223 /* configure CSI-2 link between host and IVSC */
224 static int csi_set_link_cfg(struct mei_csi *csi)
225 {
226 	struct csi_cmd cmd = { 0 };
227 	size_t cmd_size;
228 	int ret;
229 
230 	cmd.cmd_id = CSI_SET_CONF;
231 	cmd.param.conf.nr_of_lanes = csi->nr_of_lanes;
232 	cmd.param.conf.link_freq = CSI_LINK_FREQ(csi->link_freq);
233 	cmd_size = sizeof(cmd.cmd_id) + sizeof(cmd.param.conf);
234 
235 	mutex_lock(&csi->lock);
236 
237 	ret = mei_csi_send(csi, (u8 *)&cmd, cmd_size);
238 	/*
239 	 * wait configuration ready if download success. placing
240 	 * delay under mutex is to make sure current command flow
241 	 * completed before starting a possible new one.
242 	 */
243 	if (!ret)
244 		msleep(CSI_FW_READY_DELAY_MS);
245 
246 	mutex_unlock(&csi->lock);
247 
248 	return ret;
249 }
250 
251 /* callback for receive */
252 static void mei_csi_rx(struct mei_cl_device *cldev)
253 {
254 	struct mei_csi *csi = mei_cldev_get_drvdata(cldev);
255 	struct csi_notif notif = { 0 };
256 	int ret;
257 
258 	ret = mei_cldev_recv(cldev, (u8 *)&notif, sizeof(notif));
259 	if (ret < 0) {
260 		dev_err(&cldev->dev, "recv error: %d\n", ret);
261 		return;
262 	}
263 
264 	switch (notif.cmd_id) {
265 	case CSI_PRIVACY_NOTIF:
266 		if (notif.cont.cont < CSI_PRIVACY_MAX) {
267 			csi->status = notif.cont.cont;
268 			v4l2_ctrl_s_ctrl(csi->privacy_ctrl, csi->status);
269 		}
270 		break;
271 	case CSI_SET_OWNER:
272 	case CSI_SET_CONF:
273 		memcpy(&csi->cmd_response, &notif, ret);
274 
275 		complete(&csi->cmd_completion);
276 		break;
277 	default:
278 		break;
279 	}
280 }
281 
282 static int mei_csi_set_stream(struct v4l2_subdev *sd, int enable)
283 {
284 	struct mei_csi *csi = sd_to_csi(sd);
285 	s64 freq;
286 	int ret;
287 
288 	if (enable && csi->streaming == 0) {
289 		freq = v4l2_get_link_freq(csi->remote->ctrl_handler, 0, 0);
290 		if (freq < 0) {
291 			dev_err(&csi->cldev->dev,
292 				"error %lld, invalid link_freq\n", freq);
293 			ret = freq;
294 			goto err;
295 		}
296 		csi->link_freq = freq;
297 
298 		/* switch CSI-2 link to host */
299 		ret = csi_set_link_owner(csi, CSI_LINK_HOST);
300 		if (ret < 0)
301 			goto err;
302 
303 		/* configure CSI-2 link */
304 		ret = csi_set_link_cfg(csi);
305 		if (ret < 0)
306 			goto err_switch;
307 
308 		ret = v4l2_subdev_call(csi->remote, video, s_stream, 1);
309 		if (ret)
310 			goto err_switch;
311 	} else if (!enable && csi->streaming == 1) {
312 		v4l2_subdev_call(csi->remote, video, s_stream, 0);
313 
314 		/* switch CSI-2 link to IVSC */
315 		ret = csi_set_link_owner(csi, CSI_LINK_IVSC);
316 		if (ret < 0)
317 			dev_warn(&csi->cldev->dev,
318 				 "failed to switch CSI2 link: %d\n", ret);
319 	}
320 
321 	csi->streaming = enable;
322 
323 	return 0;
324 
325 err_switch:
326 	csi_set_link_owner(csi, CSI_LINK_IVSC);
327 
328 err:
329 	return ret;
330 }
331 
332 static struct v4l2_mbus_framefmt *
333 mei_csi_get_pad_format(struct v4l2_subdev *sd,
334 		       struct v4l2_subdev_state *sd_state,
335 		       unsigned int pad, u32 which)
336 {
337 	struct mei_csi *csi = sd_to_csi(sd);
338 
339 	switch (which) {
340 	case V4L2_SUBDEV_FORMAT_TRY:
341 		return v4l2_subdev_get_try_format(sd, sd_state, pad);
342 	case V4L2_SUBDEV_FORMAT_ACTIVE:
343 		return &csi->format_mbus[pad];
344 	default:
345 		return NULL;
346 	}
347 }
348 
349 static int mei_csi_init_cfg(struct v4l2_subdev *sd,
350 			    struct v4l2_subdev_state *sd_state)
351 {
352 	struct v4l2_mbus_framefmt *mbusformat;
353 	struct mei_csi *csi = sd_to_csi(sd);
354 	unsigned int i;
355 
356 	mutex_lock(&csi->lock);
357 
358 	for (i = 0; i < sd->entity.num_pads; i++) {
359 		mbusformat = v4l2_subdev_get_try_format(sd, sd_state, i);
360 		*mbusformat = mei_csi_format_mbus_default;
361 	}
362 
363 	mutex_unlock(&csi->lock);
364 
365 	return 0;
366 }
367 
368 static int mei_csi_get_fmt(struct v4l2_subdev *sd,
369 			   struct v4l2_subdev_state *sd_state,
370 			   struct v4l2_subdev_format *format)
371 {
372 	struct v4l2_mbus_framefmt *mbusformat;
373 	struct mei_csi *csi = sd_to_csi(sd);
374 
375 	mutex_lock(&csi->lock);
376 
377 	mbusformat = mei_csi_get_pad_format(sd, sd_state, format->pad,
378 					    format->which);
379 	if (mbusformat)
380 		format->format = *mbusformat;
381 
382 	mutex_unlock(&csi->lock);
383 
384 	return 0;
385 }
386 
387 static int mei_csi_set_fmt(struct v4l2_subdev *sd,
388 			   struct v4l2_subdev_state *sd_state,
389 			   struct v4l2_subdev_format *format)
390 {
391 	struct v4l2_mbus_framefmt *source_mbusformat;
392 	struct v4l2_mbus_framefmt *mbusformat;
393 	struct mei_csi *csi = sd_to_csi(sd);
394 	struct media_pad *pad;
395 
396 	mbusformat = mei_csi_get_pad_format(sd, sd_state, format->pad,
397 					    format->which);
398 	if (!mbusformat)
399 		return -EINVAL;
400 
401 	source_mbusformat = mei_csi_get_pad_format(sd, sd_state, CSI_PAD_SOURCE,
402 						   format->which);
403 	if (!source_mbusformat)
404 		return -EINVAL;
405 
406 	v4l_bound_align_image(&format->format.width, 1, 65536, 0,
407 			      &format->format.height, 1, 65536, 0, 0);
408 
409 	switch (format->format.code) {
410 	case MEDIA_BUS_FMT_RGB444_1X12:
411 	case MEDIA_BUS_FMT_RGB444_2X8_PADHI_BE:
412 	case MEDIA_BUS_FMT_RGB444_2X8_PADHI_LE:
413 	case MEDIA_BUS_FMT_RGB555_2X8_PADHI_BE:
414 	case MEDIA_BUS_FMT_RGB555_2X8_PADHI_LE:
415 	case MEDIA_BUS_FMT_RGB565_1X16:
416 	case MEDIA_BUS_FMT_BGR565_2X8_BE:
417 	case MEDIA_BUS_FMT_BGR565_2X8_LE:
418 	case MEDIA_BUS_FMT_RGB565_2X8_BE:
419 	case MEDIA_BUS_FMT_RGB565_2X8_LE:
420 	case MEDIA_BUS_FMT_RGB666_1X18:
421 	case MEDIA_BUS_FMT_RBG888_1X24:
422 	case MEDIA_BUS_FMT_RGB666_1X24_CPADHI:
423 	case MEDIA_BUS_FMT_BGR888_1X24:
424 	case MEDIA_BUS_FMT_GBR888_1X24:
425 	case MEDIA_BUS_FMT_RGB888_1X24:
426 	case MEDIA_BUS_FMT_RGB888_2X12_BE:
427 	case MEDIA_BUS_FMT_RGB888_2X12_LE:
428 	case MEDIA_BUS_FMT_ARGB8888_1X32:
429 	case MEDIA_BUS_FMT_RGB888_1X32_PADHI:
430 	case MEDIA_BUS_FMT_RGB101010_1X30:
431 	case MEDIA_BUS_FMT_RGB121212_1X36:
432 	case MEDIA_BUS_FMT_RGB161616_1X48:
433 	case MEDIA_BUS_FMT_Y8_1X8:
434 	case MEDIA_BUS_FMT_UV8_1X8:
435 	case MEDIA_BUS_FMT_UYVY8_1_5X8:
436 	case MEDIA_BUS_FMT_VYUY8_1_5X8:
437 	case MEDIA_BUS_FMT_YUYV8_1_5X8:
438 	case MEDIA_BUS_FMT_YVYU8_1_5X8:
439 	case MEDIA_BUS_FMT_UYVY8_2X8:
440 	case MEDIA_BUS_FMT_VYUY8_2X8:
441 	case MEDIA_BUS_FMT_YUYV8_2X8:
442 	case MEDIA_BUS_FMT_YVYU8_2X8:
443 	case MEDIA_BUS_FMT_Y10_1X10:
444 	case MEDIA_BUS_FMT_UYVY10_2X10:
445 	case MEDIA_BUS_FMT_VYUY10_2X10:
446 	case MEDIA_BUS_FMT_YUYV10_2X10:
447 	case MEDIA_BUS_FMT_YVYU10_2X10:
448 	case MEDIA_BUS_FMT_Y12_1X12:
449 	case MEDIA_BUS_FMT_UYVY12_2X12:
450 	case MEDIA_BUS_FMT_VYUY12_2X12:
451 	case MEDIA_BUS_FMT_YUYV12_2X12:
452 	case MEDIA_BUS_FMT_YVYU12_2X12:
453 	case MEDIA_BUS_FMT_UYVY8_1X16:
454 	case MEDIA_BUS_FMT_VYUY8_1X16:
455 	case MEDIA_BUS_FMT_YUYV8_1X16:
456 	case MEDIA_BUS_FMT_YVYU8_1X16:
457 	case MEDIA_BUS_FMT_YDYUYDYV8_1X16:
458 	case MEDIA_BUS_FMT_UYVY10_1X20:
459 	case MEDIA_BUS_FMT_VYUY10_1X20:
460 	case MEDIA_BUS_FMT_YUYV10_1X20:
461 	case MEDIA_BUS_FMT_YVYU10_1X20:
462 	case MEDIA_BUS_FMT_VUY8_1X24:
463 	case MEDIA_BUS_FMT_YUV8_1X24:
464 	case MEDIA_BUS_FMT_UYYVYY8_0_5X24:
465 	case MEDIA_BUS_FMT_UYVY12_1X24:
466 	case MEDIA_BUS_FMT_VYUY12_1X24:
467 	case MEDIA_BUS_FMT_YUYV12_1X24:
468 	case MEDIA_BUS_FMT_YVYU12_1X24:
469 	case MEDIA_BUS_FMT_YUV10_1X30:
470 	case MEDIA_BUS_FMT_UYYVYY10_0_5X30:
471 	case MEDIA_BUS_FMT_AYUV8_1X32:
472 	case MEDIA_BUS_FMT_UYYVYY12_0_5X36:
473 	case MEDIA_BUS_FMT_YUV12_1X36:
474 	case MEDIA_BUS_FMT_YUV16_1X48:
475 	case MEDIA_BUS_FMT_UYYVYY16_0_5X48:
476 	case MEDIA_BUS_FMT_JPEG_1X8:
477 	case MEDIA_BUS_FMT_AHSV8888_1X32:
478 	case MEDIA_BUS_FMT_SBGGR8_1X8:
479 	case MEDIA_BUS_FMT_SGBRG8_1X8:
480 	case MEDIA_BUS_FMT_SGRBG8_1X8:
481 	case MEDIA_BUS_FMT_SRGGB8_1X8:
482 	case MEDIA_BUS_FMT_SBGGR10_1X10:
483 	case MEDIA_BUS_FMT_SGBRG10_1X10:
484 	case MEDIA_BUS_FMT_SGRBG10_1X10:
485 	case MEDIA_BUS_FMT_SRGGB10_1X10:
486 	case MEDIA_BUS_FMT_SBGGR12_1X12:
487 	case MEDIA_BUS_FMT_SGBRG12_1X12:
488 	case MEDIA_BUS_FMT_SGRBG12_1X12:
489 	case MEDIA_BUS_FMT_SRGGB12_1X12:
490 	case MEDIA_BUS_FMT_SBGGR14_1X14:
491 	case MEDIA_BUS_FMT_SGBRG14_1X14:
492 	case MEDIA_BUS_FMT_SGRBG14_1X14:
493 	case MEDIA_BUS_FMT_SRGGB14_1X14:
494 	case MEDIA_BUS_FMT_SBGGR16_1X16:
495 	case MEDIA_BUS_FMT_SGBRG16_1X16:
496 	case MEDIA_BUS_FMT_SGRBG16_1X16:
497 	case MEDIA_BUS_FMT_SRGGB16_1X16:
498 		break;
499 	default:
500 		format->format.code = MEDIA_BUS_FMT_Y8_1X8;
501 		break;
502 	}
503 
504 	if (format->format.field == V4L2_FIELD_ANY)
505 		format->format.field = V4L2_FIELD_NONE;
506 
507 	mutex_lock(&csi->lock);
508 
509 	pad = &csi->pads[format->pad];
510 	if (pad->flags & MEDIA_PAD_FL_SOURCE)
511 		format->format = csi->format_mbus[CSI_PAD_SINK];
512 
513 	*mbusformat = format->format;
514 
515 	if (pad->flags & MEDIA_PAD_FL_SINK)
516 		*source_mbusformat = format->format;
517 
518 	mutex_unlock(&csi->lock);
519 
520 	return 0;
521 }
522 
523 static int mei_csi_g_volatile_ctrl(struct v4l2_ctrl *ctrl)
524 {
525 	struct mei_csi *csi = ctrl_to_csi(ctrl);
526 	s64 freq;
527 
528 	if (ctrl->id == V4L2_CID_LINK_FREQ) {
529 		if (!csi->remote)
530 			return -EINVAL;
531 
532 		freq = v4l2_get_link_freq(csi->remote->ctrl_handler, 0, 0);
533 		if (freq < 0) {
534 			dev_err(&csi->cldev->dev,
535 				"error %lld, invalid link_freq\n", freq);
536 			return -EINVAL;
537 		}
538 
539 		link_freq_menu_items[0] = freq;
540 		ctrl->val = 0;
541 
542 		return 0;
543 	}
544 
545 	return -EINVAL;
546 }
547 
548 static const struct v4l2_ctrl_ops mei_csi_ctrl_ops = {
549 	.g_volatile_ctrl = mei_csi_g_volatile_ctrl,
550 };
551 
552 static const struct v4l2_subdev_video_ops mei_csi_video_ops = {
553 	.s_stream = mei_csi_set_stream,
554 };
555 
556 static const struct v4l2_subdev_pad_ops mei_csi_pad_ops = {
557 	.init_cfg = mei_csi_init_cfg,
558 	.get_fmt = mei_csi_get_fmt,
559 	.set_fmt = mei_csi_set_fmt,
560 };
561 
562 static const struct v4l2_subdev_ops mei_csi_subdev_ops = {
563 	.video = &mei_csi_video_ops,
564 	.pad = &mei_csi_pad_ops,
565 };
566 
567 static const struct media_entity_operations mei_csi_entity_ops = {
568 	.link_validate = v4l2_subdev_link_validate,
569 };
570 
571 static int mei_csi_notify_bound(struct v4l2_async_notifier *notifier,
572 				struct v4l2_subdev *subdev,
573 				struct v4l2_async_connection *asd)
574 {
575 	struct mei_csi *csi = notifier_to_csi(notifier);
576 	int pad;
577 
578 	pad = media_entity_get_fwnode_pad(&subdev->entity, asd->match.fwnode,
579 					  MEDIA_PAD_FL_SOURCE);
580 	if (pad < 0)
581 		return pad;
582 
583 	csi->remote = subdev;
584 	csi->remote_pad = pad;
585 
586 	return media_create_pad_link(&subdev->entity, pad,
587 				     &csi->subdev.entity, 1,
588 				     MEDIA_LNK_FL_ENABLED |
589 				     MEDIA_LNK_FL_IMMUTABLE);
590 }
591 
592 static void mei_csi_notify_unbind(struct v4l2_async_notifier *notifier,
593 				  struct v4l2_subdev *subdev,
594 				  struct v4l2_async_connection *asd)
595 {
596 	struct mei_csi *csi = notifier_to_csi(notifier);
597 
598 	csi->remote = NULL;
599 }
600 
601 static const struct v4l2_async_notifier_operations mei_csi_notify_ops = {
602 	.bound = mei_csi_notify_bound,
603 	.unbind = mei_csi_notify_unbind,
604 };
605 
606 static int mei_csi_init_controls(struct mei_csi *csi)
607 {
608 	u32 max;
609 	int ret;
610 
611 	ret = v4l2_ctrl_handler_init(&csi->ctrl_handler, 2);
612 	if (ret)
613 		return ret;
614 
615 	csi->ctrl_handler.lock = &csi->lock;
616 
617 	max = ARRAY_SIZE(link_freq_menu_items) - 1;
618 	csi->freq_ctrl = v4l2_ctrl_new_int_menu(&csi->ctrl_handler,
619 						&mei_csi_ctrl_ops,
620 						V4L2_CID_LINK_FREQ,
621 						max,
622 						0,
623 						link_freq_menu_items);
624 	if (csi->freq_ctrl)
625 		csi->freq_ctrl->flags |= V4L2_CTRL_FLAG_READ_ONLY |
626 					 V4L2_CTRL_FLAG_VOLATILE;
627 
628 	csi->privacy_ctrl = v4l2_ctrl_new_std(&csi->ctrl_handler, NULL,
629 					      V4L2_CID_PRIVACY, 0, 1, 1, 0);
630 	if (csi->privacy_ctrl)
631 		csi->privacy_ctrl->flags |= V4L2_CTRL_FLAG_READ_ONLY;
632 
633 	if (csi->ctrl_handler.error)
634 		return csi->ctrl_handler.error;
635 
636 	csi->subdev.ctrl_handler = &csi->ctrl_handler;
637 
638 	return 0;
639 }
640 
641 static int mei_csi_parse_firmware(struct mei_csi *csi)
642 {
643 	struct v4l2_fwnode_endpoint v4l2_ep = {
644 		.bus_type = V4L2_MBUS_CSI2_DPHY,
645 	};
646 	struct device *dev = &csi->cldev->dev;
647 	struct v4l2_async_connection *asd;
648 	struct fwnode_handle *fwnode;
649 	struct fwnode_handle *ep;
650 	int ret;
651 
652 	ep = fwnode_graph_get_endpoint_by_id(dev_fwnode(dev), 0, 0, 0);
653 	if (!ep) {
654 		dev_err(dev, "not connected to subdevice\n");
655 		return -EINVAL;
656 	}
657 
658 	ret = v4l2_fwnode_endpoint_parse(ep, &v4l2_ep);
659 	if (ret) {
660 		dev_err(dev, "could not parse v4l2 endpoint\n");
661 		fwnode_handle_put(ep);
662 		return -EINVAL;
663 	}
664 
665 	fwnode = fwnode_graph_get_remote_endpoint(ep);
666 	fwnode_handle_put(ep);
667 
668 	v4l2_async_subdev_nf_init(&csi->notifier, &csi->subdev);
669 	csi->notifier.ops = &mei_csi_notify_ops;
670 
671 	asd = v4l2_async_nf_add_fwnode(&csi->notifier, fwnode,
672 				       struct v4l2_async_connection);
673 	if (IS_ERR(asd)) {
674 		fwnode_handle_put(fwnode);
675 		return PTR_ERR(asd);
676 	}
677 
678 	ret = v4l2_fwnode_endpoint_alloc_parse(fwnode, &v4l2_ep);
679 	fwnode_handle_put(fwnode);
680 	if (ret)
681 		return ret;
682 	csi->nr_of_lanes = v4l2_ep.bus.mipi_csi2.num_data_lanes;
683 
684 	ret = v4l2_async_nf_register(&csi->notifier);
685 	if (ret)
686 		v4l2_async_nf_cleanup(&csi->notifier);
687 
688 	v4l2_fwnode_endpoint_free(&v4l2_ep);
689 
690 	return ret;
691 }
692 
693 static int mei_csi_probe(struct mei_cl_device *cldev,
694 			 const struct mei_cl_device_id *id)
695 {
696 	struct device *dev = &cldev->dev;
697 	struct mei_csi *csi;
698 	int ret;
699 
700 	if (!dev_fwnode(dev))
701 		return -EPROBE_DEFER;
702 
703 	csi = devm_kzalloc(dev, sizeof(struct mei_csi), GFP_KERNEL);
704 	if (!csi)
705 		return -ENOMEM;
706 
707 	csi->cldev = cldev;
708 	mutex_init(&csi->lock);
709 	init_completion(&csi->cmd_completion);
710 
711 	mei_cldev_set_drvdata(cldev, csi);
712 
713 	ret = mei_cldev_enable(cldev);
714 	if (ret < 0) {
715 		dev_err(dev, "mei_cldev_enable failed: %d\n", ret);
716 		goto destroy_mutex;
717 	}
718 
719 	ret = mei_cldev_register_rx_cb(cldev, mei_csi_rx);
720 	if (ret) {
721 		dev_err(dev, "event cb registration failed: %d\n", ret);
722 		goto err_disable;
723 	}
724 
725 	ret = mei_csi_parse_firmware(csi);
726 	if (ret)
727 		goto err_disable;
728 
729 	csi->subdev.dev = &cldev->dev;
730 	v4l2_subdev_init(&csi->subdev, &mei_csi_subdev_ops);
731 	v4l2_set_subdevdata(&csi->subdev, csi);
732 	csi->subdev.flags = V4L2_SUBDEV_FL_HAS_DEVNODE |
733 			    V4L2_SUBDEV_FL_HAS_EVENTS;
734 	csi->subdev.entity.function = MEDIA_ENT_F_VID_IF_BRIDGE;
735 	csi->subdev.entity.ops = &mei_csi_entity_ops;
736 
737 	snprintf(csi->subdev.name, sizeof(csi->subdev.name),
738 		 MEI_CSI_ENTITY_NAME);
739 
740 	ret = mei_csi_init_controls(csi);
741 	if (ret)
742 		goto err_ctrl_handler;
743 
744 	csi->format_mbus[CSI_PAD_SOURCE] = mei_csi_format_mbus_default;
745 	csi->format_mbus[CSI_PAD_SINK] = mei_csi_format_mbus_default;
746 
747 	csi->pads[CSI_PAD_SOURCE].flags = MEDIA_PAD_FL_SOURCE;
748 	csi->pads[CSI_PAD_SINK].flags = MEDIA_PAD_FL_SINK;
749 	ret = media_entity_pads_init(&csi->subdev.entity, CSI_NUM_PADS,
750 				     csi->pads);
751 	if (ret)
752 		goto err_ctrl_handler;
753 
754 	ret = v4l2_subdev_init_finalize(&csi->subdev);
755 	if (ret < 0)
756 		goto err_entity;
757 
758 	ret = v4l2_async_register_subdev(&csi->subdev);
759 	if (ret < 0)
760 		goto err_subdev;
761 
762 	pm_runtime_enable(&cldev->dev);
763 
764 	return 0;
765 
766 err_subdev:
767 	v4l2_subdev_cleanup(&csi->subdev);
768 
769 err_entity:
770 	media_entity_cleanup(&csi->subdev.entity);
771 
772 err_ctrl_handler:
773 	v4l2_ctrl_handler_free(&csi->ctrl_handler);
774 	v4l2_async_nf_unregister(&csi->notifier);
775 	v4l2_async_nf_cleanup(&csi->notifier);
776 
777 err_disable:
778 	mei_cldev_disable(cldev);
779 
780 destroy_mutex:
781 	mutex_destroy(&csi->lock);
782 
783 	return ret;
784 }
785 
786 static void mei_csi_remove(struct mei_cl_device *cldev)
787 {
788 	struct mei_csi *csi = mei_cldev_get_drvdata(cldev);
789 
790 	v4l2_async_nf_unregister(&csi->notifier);
791 	v4l2_async_nf_cleanup(&csi->notifier);
792 	v4l2_ctrl_handler_free(&csi->ctrl_handler);
793 	v4l2_async_unregister_subdev(&csi->subdev);
794 	v4l2_subdev_cleanup(&csi->subdev);
795 	media_entity_cleanup(&csi->subdev.entity);
796 
797 	pm_runtime_disable(&cldev->dev);
798 
799 	mutex_destroy(&csi->lock);
800 }
801 
802 #define MEI_CSI_UUID UUID_LE(0x92335FCF, 0x3203, 0x4472, \
803 			     0xAF, 0x93, 0x7b, 0x44, 0x53, 0xAC, 0x29, 0xDA)
804 
805 static const struct mei_cl_device_id mei_csi_tbl[] = {
806 	{ .uuid = MEI_CSI_UUID, .version = MEI_CL_VERSION_ANY },
807 	{ /* sentinel */ }
808 };
809 MODULE_DEVICE_TABLE(mei, mei_csi_tbl);
810 
811 static struct mei_cl_driver mei_csi_driver = {
812 	.id_table = mei_csi_tbl,
813 	.name = KBUILD_MODNAME,
814 
815 	.probe = mei_csi_probe,
816 	.remove = mei_csi_remove,
817 };
818 
819 module_mei_cl_driver(mei_csi_driver);
820 
821 MODULE_AUTHOR("Wentong Wu <wentong.wu@intel.com>");
822 MODULE_AUTHOR("Zhifeng Wang <zhifeng.wang@intel.com>");
823 MODULE_DESCRIPTION("Device driver for IVSC CSI");
824 MODULE_LICENSE("GPL");
825