xref: /linux/drivers/misc/mei/vsc-fw-loader.c (revision 7a5f93ea5862da91488975acaa0c7abd508f192b)
1 // SPDX-License-Identifier: GPL-2.0-only
2 /*
3  * Copyright (c) 2023, Intel Corporation.
4  * Intel Visual Sensing Controller Transport Layer Linux driver
5  */
6 
7 #include <linux/acpi.h>
8 #include <linux/align.h>
9 #include <linux/bitfield.h>
10 #include <linux/bits.h>
11 #include <linux/cleanup.h>
12 #include <linux/firmware.h>
13 #include <linux/sizes.h>
14 #include <linux/slab.h>
15 #include <linux/string_helpers.h>
16 #include <linux/types.h>
17 
18 #include <linux/unaligned.h>
19 
20 #include "vsc-tp.h"
21 
22 #define VSC_MAGIC_NUM			0x49505343 /* IPSC */
23 #define VSC_MAGIC_FW			0x49574653 /* IWFS */
24 #define VSC_MAGIC_FILE			0x46564353 /* FVCS */
25 
26 #define VSC_ADDR_BASE			0xE0030000
27 #define VSC_EFUSE_ADDR			(VSC_ADDR_BASE + 0x038)
28 #define VSC_STRAP_ADDR			(VSC_ADDR_BASE + 0x100)
29 
30 #define VSC_MAINSTEPPING_VERSION_MASK	GENMASK(7, 4)
31 #define VSC_MAINSTEPPING_VERSION_A	0
32 
33 #define VSC_SUBSTEPPING_VERSION_MASK	GENMASK(3, 0)
34 #define VSC_SUBSTEPPING_VERSION_0	0
35 #define VSC_SUBSTEPPING_VERSION_1	2
36 
37 #define VSC_BOOT_IMG_OPTION_MASK	GENMASK(15, 0)
38 
39 #define VSC_SKU_CFG_LOCATION		0x5001A000
40 #define VSC_SKU_MAX_SIZE		4100u
41 
42 #define VSC_ACE_IMG_CNT			2
43 #define VSC_CSI_IMG_CNT			4
44 #define VSC_IMG_CNT_MAX			6
45 
46 #define VSC_ROM_PKG_SIZE		256u
47 #define VSC_FW_PKG_SIZE			512u
48 
49 #define VSC_IMAGE_DIR			"intel/vsc/"
50 
51 #define VSC_CSI_IMAGE_NAME		VSC_IMAGE_DIR "ivsc_fw.bin"
52 #define VSC_ACE_IMAGE_NAME_FMT		VSC_IMAGE_DIR "ivsc_pkg_%s_0.bin"
53 #define VSC_CFG_IMAGE_NAME_FMT		VSC_IMAGE_DIR "ivsc_skucfg_%s_0_1.bin"
54 
55 #define VSC_IMAGE_PATH_MAX_LEN		64
56 
57 #define VSC_SENSOR_NAME_MAX_LEN		16
58 
59 /* command id */
60 enum {
61 	VSC_CMD_QUERY = 0,
62 	VSC_CMD_DL_SET = 1,
63 	VSC_CMD_DL_START = 2,
64 	VSC_CMD_DL_CONT = 3,
65 	VSC_CMD_DUMP_MEM = 4,
66 	VSC_CMD_GET_CONT = 8,
67 	VSC_CMD_CAM_BOOT = 10,
68 };
69 
70 /* command ack token */
71 enum {
72 	VSC_TOKEN_BOOTLOADER_REQ = 1,
73 	VSC_TOKEN_DUMP_RESP = 4,
74 	VSC_TOKEN_ERROR = 7,
75 };
76 
77 /* image type */
78 enum {
79 	VSC_IMG_BOOTLOADER_TYPE = 1,
80 	VSC_IMG_CSI_EM7D_TYPE,
81 	VSC_IMG_CSI_SEM_TYPE,
82 	VSC_IMG_CSI_RUNTIME_TYPE,
83 	VSC_IMG_ACE_VISION_TYPE,
84 	VSC_IMG_ACE_CFG_TYPE,
85 	VSC_IMG_SKU_CFG_TYPE,
86 };
87 
88 /* image fragments */
89 enum {
90 	VSC_IMG_BOOTLOADER_FRAG,
91 	VSC_IMG_CSI_SEM_FRAG,
92 	VSC_IMG_CSI_RUNTIME_FRAG,
93 	VSC_IMG_ACE_VISION_FRAG,
94 	VSC_IMG_ACE_CFG_FRAG,
95 	VSC_IMG_CSI_EM7D_FRAG,
96 	VSC_IMG_SKU_CFG_FRAG,
97 	VSC_IMG_FRAG_MAX
98 };
99 
100 struct vsc_rom_cmd {
101 	__le32 magic;
102 	__u8 cmd_id;
103 	union {
104 		/* download start */
105 		struct {
106 			__u8 img_type;
107 			__le16 option;
108 			__le32 img_len;
109 			__le32 img_loc;
110 			__le32 crc;
111 			DECLARE_FLEX_ARRAY(__u8, res);
112 		} __packed dl_start;
113 		/* download set */
114 		struct {
115 			__u8 option;
116 			__le16 img_cnt;
117 			DECLARE_FLEX_ARRAY(__le32, payload);
118 		} __packed dl_set;
119 		/* download continue */
120 		struct {
121 			__u8 end_flag;
122 			__le16 len;
123 			/* 8 is the offset of payload */
124 			__u8 payload[VSC_ROM_PKG_SIZE - 8];
125 		} __packed dl_cont;
126 		/* dump memory */
127 		struct {
128 			__u8 res;
129 			__le16 len;
130 			__le32 addr;
131 			DECLARE_FLEX_ARRAY(__u8, payload);
132 		} __packed dump_mem;
133 		/* 5 is the offset of padding */
134 		__u8 padding[VSC_ROM_PKG_SIZE - 5];
135 	} data;
136 };
137 
138 struct vsc_rom_cmd_ack {
139 	__le32 magic;
140 	__u8 token;
141 	__u8 type;
142 	__u8 res[2];
143 	__u8 payload[];
144 };
145 
146 struct vsc_fw_cmd {
147 	__le32 magic;
148 	__u8 cmd_id;
149 	union {
150 		struct {
151 			__le16 option;
152 			__u8 img_type;
153 			__le32 img_len;
154 			__le32 img_loc;
155 			__le32 crc;
156 			DECLARE_FLEX_ARRAY(__u8, res);
157 		} __packed dl_start;
158 		struct {
159 			__le16 option;
160 			__u8 img_cnt;
161 			DECLARE_FLEX_ARRAY(__le32, payload);
162 		} __packed dl_set;
163 		struct {
164 			__le32 addr;
165 			__u8 len;
166 			DECLARE_FLEX_ARRAY(__u8, payload);
167 		} __packed dump_mem;
168 		struct {
169 			__u8 resv[3];
170 			__le32 crc;
171 			DECLARE_FLEX_ARRAY(__u8, payload);
172 		} __packed boot;
173 		/* 5 is the offset of padding */
174 		__u8 padding[VSC_FW_PKG_SIZE - 5];
175 	} data;
176 };
177 
178 struct vsc_img {
179 	__le32 magic;
180 	__le32 option;
181 	__le32 image_count;
182 	__le32 image_location[VSC_IMG_CNT_MAX];
183 };
184 
185 struct vsc_fw_sign {
186 	__le32 magic;
187 	__le32 image_size;
188 	__u8 image[];
189 };
190 
191 struct vsc_image_code_data {
192 	/* fragment index */
193 	u8 frag_index;
194 	/* image type */
195 	u8 image_type;
196 };
197 
198 struct vsc_img_frag {
199 	u8 type;
200 	u32 location;
201 	const u8 *data;
202 	u32 size;
203 };
204 
205 /**
206  * struct vsc_fw_loader - represent vsc firmware loader
207  * @dev: device used to request firmware
208  * @tp: transport layer used with the firmware loader
209  * @csi: CSI image
210  * @ace: ACE image
211  * @cfg: config image
212  * @tx_buf: tx buffer
213  * @rx_buf: rx buffer
214  * @option: command option
215  * @count: total image count
216  * @sensor_name: camera sensor name
217  * @frags: image fragments
218  */
219 struct vsc_fw_loader {
220 	struct device *dev;
221 	struct vsc_tp *tp;
222 
223 	const struct firmware *csi;
224 	const struct firmware *ace;
225 	const struct firmware *cfg;
226 
227 	void *tx_buf;
228 	void *rx_buf;
229 
230 	u16 option;
231 	u16 count;
232 
233 	char sensor_name[VSC_SENSOR_NAME_MAX_LEN];
234 
235 	struct vsc_img_frag frags[VSC_IMG_FRAG_MAX];
236 };
237 
238 static inline u32 vsc_sum_crc(void *data, size_t size)
239 {
240 	u32 crc = 0;
241 	size_t i;
242 
243 	for (i = 0; i < size; i++)
244 		crc += *((u8 *)data + i);
245 
246 	return crc;
247 }
248 
249 /* get sensor name to construct image name */
250 static int vsc_get_sensor_name(struct vsc_fw_loader *fw_loader,
251 			       struct device *dev)
252 {
253 	struct acpi_buffer buffer = { ACPI_ALLOCATE_BUFFER };
254 	union acpi_object obj = {
255 		.integer.type = ACPI_TYPE_INTEGER,
256 		.integer.value = 1,
257 	};
258 	struct acpi_object_list arg_list = {
259 		.count = 1,
260 		.pointer = &obj,
261 	};
262 	union acpi_object *ret_obj;
263 	acpi_handle handle;
264 	acpi_status status;
265 	int ret = 0;
266 
267 	handle = ACPI_HANDLE(dev);
268 	if (!handle)
269 		return -EINVAL;
270 
271 	status = acpi_evaluate_object(handle, "SID", &arg_list, &buffer);
272 	if (ACPI_FAILURE(status)) {
273 		dev_err(dev, "can't evaluate SID method: %d\n", status);
274 		return -ENODEV;
275 	}
276 
277 	ret_obj = buffer.pointer;
278 	if (!ret_obj) {
279 		dev_err(dev, "can't locate ACPI buffer\n");
280 		return -ENODEV;
281 	}
282 
283 	if (ret_obj->type != ACPI_TYPE_STRING) {
284 		dev_err(dev, "found non-string entry\n");
285 		ret = -ENODEV;
286 		goto out_free_buff;
287 	}
288 
289 	/* string length excludes trailing NUL */
290 	if (ret_obj->string.length >= sizeof(fw_loader->sensor_name)) {
291 		dev_err(dev, "sensor name buffer too small\n");
292 		ret = -EINVAL;
293 		goto out_free_buff;
294 	}
295 
296 	memcpy(fw_loader->sensor_name, ret_obj->string.pointer,
297 	       ret_obj->string.length);
298 
299 	string_lower(fw_loader->sensor_name, fw_loader->sensor_name);
300 
301 out_free_buff:
302 	ACPI_FREE(buffer.pointer);
303 
304 	return ret;
305 }
306 
307 static int vsc_identify_silicon(struct vsc_fw_loader *fw_loader)
308 {
309 	struct vsc_rom_cmd_ack *ack = fw_loader->rx_buf;
310 	struct vsc_rom_cmd *cmd = fw_loader->tx_buf;
311 	u8 version, sub_version;
312 	int ret;
313 
314 	/* identify stepping information */
315 	cmd->magic = cpu_to_le32(VSC_MAGIC_NUM);
316 	cmd->cmd_id = VSC_CMD_DUMP_MEM;
317 	cmd->data.dump_mem.addr = cpu_to_le32(VSC_EFUSE_ADDR);
318 	cmd->data.dump_mem.len = cpu_to_le16(sizeof(__le32));
319 	ret = vsc_tp_rom_xfer(fw_loader->tp, cmd, ack, VSC_ROM_PKG_SIZE);
320 	if (ret || ack->token == VSC_TOKEN_ERROR) {
321 		dev_err(fw_loader->dev, "CMD_DUMP_MEM error %d token %d\n", ret, ack->token);
322 		return ret ?: -EINVAL;
323 	}
324 
325 	cmd->magic = cpu_to_le32(VSC_MAGIC_NUM);
326 	cmd->cmd_id = VSC_CMD_GET_CONT;
327 	ret = vsc_tp_rom_xfer(fw_loader->tp, cmd, ack, VSC_ROM_PKG_SIZE);
328 	if (ret || ack->token != VSC_TOKEN_DUMP_RESP) {
329 		dev_err(fw_loader->dev, "CMD_GETCONT error %d token %d\n", ret, ack->token);
330 		return ret ?: -EINVAL;
331 	}
332 
333 	version = FIELD_GET(VSC_MAINSTEPPING_VERSION_MASK, ack->payload[0]);
334 	sub_version = FIELD_GET(VSC_SUBSTEPPING_VERSION_MASK, ack->payload[0]);
335 
336 	if (version != VSC_MAINSTEPPING_VERSION_A) {
337 		dev_err(fw_loader->dev, "mainstepping mismatch expected %d got %d\n",
338 			VSC_MAINSTEPPING_VERSION_A, version);
339 		return -EINVAL;
340 	}
341 
342 	if (sub_version != VSC_SUBSTEPPING_VERSION_0 &&
343 	    sub_version != VSC_SUBSTEPPING_VERSION_1) {
344 		dev_err(fw_loader->dev, "substepping %d is out of supported range %d - %d\n",
345 			sub_version, VSC_SUBSTEPPING_VERSION_0, VSC_SUBSTEPPING_VERSION_1);
346 		return -EINVAL;
347 	}
348 
349 	dev_info(fw_loader->dev, "silicon stepping version is %u:%u\n",
350 		 version, sub_version);
351 
352 	/* identify strap information */
353 	cmd->magic = cpu_to_le32(VSC_MAGIC_NUM);
354 	cmd->cmd_id = VSC_CMD_DUMP_MEM;
355 	cmd->data.dump_mem.addr = cpu_to_le32(VSC_STRAP_ADDR);
356 	cmd->data.dump_mem.len = cpu_to_le16(sizeof(__le32));
357 	ret = vsc_tp_rom_xfer(fw_loader->tp, cmd, ack, VSC_ROM_PKG_SIZE);
358 	if (ret)
359 		return ret;
360 	if (ack->token == VSC_TOKEN_ERROR)
361 		return -EINVAL;
362 
363 	cmd->magic = cpu_to_le32(VSC_MAGIC_NUM);
364 	cmd->cmd_id = VSC_CMD_GET_CONT;
365 	ret = vsc_tp_rom_xfer(fw_loader->tp, cmd, ack, VSC_ROM_PKG_SIZE);
366 	if (ret)
367 		return ret;
368 	if (ack->token != VSC_TOKEN_DUMP_RESP)
369 		return -EINVAL;
370 
371 	return 0;
372 }
373 
374 static int vsc_identify_csi_image(struct vsc_fw_loader *fw_loader)
375 {
376 	const struct firmware *image;
377 	struct vsc_fw_sign *sign;
378 	struct vsc_img *img;
379 	unsigned int i;
380 	int ret;
381 
382 	ret = request_firmware(&image, VSC_CSI_IMAGE_NAME, fw_loader->dev);
383 	if (ret)
384 		return ret;
385 
386 	img = (struct vsc_img *)image->data;
387 	if (!img) {
388 		ret = -ENOENT;
389 		goto err_release_image;
390 	}
391 
392 	if (le32_to_cpu(img->magic) != VSC_MAGIC_FILE) {
393 		ret = -EINVAL;
394 		goto err_release_image;
395 	}
396 
397 	if (le32_to_cpu(img->image_count) != VSC_CSI_IMG_CNT) {
398 		ret = -EINVAL;
399 		goto err_release_image;
400 	}
401 	fw_loader->count += le32_to_cpu(img->image_count) - 1;
402 
403 	fw_loader->option =
404 		FIELD_GET(VSC_BOOT_IMG_OPTION_MASK, le32_to_cpu(img->option));
405 
406 	sign = (struct vsc_fw_sign *)
407 		(img->image_location + le32_to_cpu(img->image_count));
408 
409 	for (i = 0; i < VSC_CSI_IMG_CNT; i++) {
410 		/* mapping from CSI image index to image code data */
411 		static const struct vsc_image_code_data csi_image_map[] = {
412 			{ VSC_IMG_BOOTLOADER_FRAG, VSC_IMG_BOOTLOADER_TYPE },
413 			{ VSC_IMG_CSI_SEM_FRAG, VSC_IMG_CSI_SEM_TYPE },
414 			{ VSC_IMG_CSI_RUNTIME_FRAG, VSC_IMG_CSI_RUNTIME_TYPE },
415 			{ VSC_IMG_CSI_EM7D_FRAG, VSC_IMG_CSI_EM7D_TYPE },
416 		};
417 		struct vsc_img_frag *frag;
418 
419 		if ((u8 *)sign + sizeof(*sign) > image->data + image->size) {
420 			ret = -EINVAL;
421 			goto err_release_image;
422 		}
423 
424 		if (le32_to_cpu(sign->magic) != VSC_MAGIC_FW) {
425 			ret = -EINVAL;
426 			goto err_release_image;
427 		}
428 
429 		if (!le32_to_cpu(img->image_location[i])) {
430 			ret = -EINVAL;
431 			goto err_release_image;
432 		}
433 
434 		frag = &fw_loader->frags[csi_image_map[i].frag_index];
435 
436 		frag->data = sign->image;
437 		frag->size = le32_to_cpu(sign->image_size);
438 		frag->location = le32_to_cpu(img->image_location[i]);
439 		frag->type = csi_image_map[i].image_type;
440 
441 		sign = (struct vsc_fw_sign *)
442 			(sign->image + le32_to_cpu(sign->image_size));
443 	}
444 
445 	fw_loader->csi = image;
446 
447 	return 0;
448 
449 err_release_image:
450 	release_firmware(image);
451 
452 	return ret;
453 }
454 
455 static int vsc_identify_ace_image(struct vsc_fw_loader *fw_loader)
456 {
457 	char path[VSC_IMAGE_PATH_MAX_LEN];
458 	const struct firmware *image;
459 	struct vsc_fw_sign *sign;
460 	struct vsc_img *img;
461 	unsigned int i;
462 	int ret;
463 
464 	snprintf(path, sizeof(path), VSC_ACE_IMAGE_NAME_FMT,
465 		 fw_loader->sensor_name);
466 
467 	ret = request_firmware(&image, path, fw_loader->dev);
468 	if (ret)
469 		return ret;
470 
471 	img = (struct vsc_img *)image->data;
472 	if (!img) {
473 		ret = -ENOENT;
474 		goto err_release_image;
475 	}
476 
477 	if (le32_to_cpu(img->magic) != VSC_MAGIC_FILE) {
478 		ret = -EINVAL;
479 		goto err_release_image;
480 	}
481 
482 	if (le32_to_cpu(img->image_count) != VSC_ACE_IMG_CNT) {
483 		ret = -EINVAL;
484 		goto err_release_image;
485 	}
486 	fw_loader->count += le32_to_cpu(img->image_count);
487 
488 	sign = (struct vsc_fw_sign *)
489 		(img->image_location + le32_to_cpu(img->image_count));
490 
491 	for (i = 0; i < VSC_ACE_IMG_CNT; i++) {
492 		/* mapping from ACE image index to image code data */
493 		static const struct vsc_image_code_data ace_image_map[] = {
494 			{ VSC_IMG_ACE_VISION_FRAG, VSC_IMG_ACE_VISION_TYPE },
495 			{ VSC_IMG_ACE_CFG_FRAG, VSC_IMG_ACE_CFG_TYPE },
496 		};
497 		struct vsc_img_frag *frag, *last_frag;
498 		u8 frag_index;
499 
500 		if ((u8 *)sign + sizeof(*sign) > image->data + image->size) {
501 			ret = -EINVAL;
502 			goto err_release_image;
503 		}
504 
505 		if (le32_to_cpu(sign->magic) != VSC_MAGIC_FW) {
506 			ret = -EINVAL;
507 			goto err_release_image;
508 		}
509 
510 		frag_index = ace_image_map[i].frag_index;
511 		frag = &fw_loader->frags[frag_index];
512 
513 		frag->data = sign->image;
514 		frag->size = le32_to_cpu(sign->image_size);
515 		frag->location = le32_to_cpu(img->image_location[i]);
516 		frag->type = ace_image_map[i].image_type;
517 
518 		if (!frag->location) {
519 			last_frag = &fw_loader->frags[frag_index - 1];
520 			frag->location =
521 				ALIGN(last_frag->location + last_frag->size, SZ_4K);
522 		}
523 
524 		sign = (struct vsc_fw_sign *)
525 			(sign->image + le32_to_cpu(sign->image_size));
526 	}
527 
528 	fw_loader->ace = image;
529 
530 	return 0;
531 
532 err_release_image:
533 	release_firmware(image);
534 
535 	return ret;
536 }
537 
538 static int vsc_identify_cfg_image(struct vsc_fw_loader *fw_loader)
539 {
540 	struct vsc_img_frag *frag = &fw_loader->frags[VSC_IMG_SKU_CFG_FRAG];
541 	char path[VSC_IMAGE_PATH_MAX_LEN];
542 	const struct firmware *image;
543 	u32 size;
544 	int ret;
545 
546 	snprintf(path, sizeof(path), VSC_CFG_IMAGE_NAME_FMT,
547 		 fw_loader->sensor_name);
548 
549 	ret = request_firmware(&image, path, fw_loader->dev);
550 	if (ret)
551 		return ret;
552 
553 	/* identify image size */
554 	if (image->size <= sizeof(u32) || image->size > VSC_SKU_MAX_SIZE) {
555 		ret = -EINVAL;
556 		goto err_release_image;
557 	}
558 
559 	size = le32_to_cpu(*((__le32 *)image->data)) + sizeof(u32);
560 	if (image->size != size) {
561 		ret = -EINVAL;
562 		goto err_release_image;
563 	}
564 
565 	frag->data = image->data;
566 	frag->size = image->size;
567 	frag->type = VSC_IMG_SKU_CFG_TYPE;
568 	frag->location = VSC_SKU_CFG_LOCATION;
569 
570 	fw_loader->cfg = image;
571 
572 	return 0;
573 
574 err_release_image:
575 	release_firmware(image);
576 
577 	return ret;
578 }
579 
580 static int vsc_download_bootloader(struct vsc_fw_loader *fw_loader)
581 {
582 	struct vsc_img_frag *frag = &fw_loader->frags[VSC_IMG_BOOTLOADER_FRAG];
583 	struct vsc_rom_cmd_ack *ack = fw_loader->rx_buf;
584 	struct vsc_rom_cmd *cmd = fw_loader->tx_buf;
585 	u32 len, c_len;
586 	size_t remain;
587 	const u8 *p;
588 	int ret;
589 
590 	cmd->magic = cpu_to_le32(VSC_MAGIC_NUM);
591 	cmd->cmd_id = VSC_CMD_QUERY;
592 	ret = vsc_tp_rom_xfer(fw_loader->tp, cmd, ack, VSC_ROM_PKG_SIZE);
593 	if (ret)
594 		return ret;
595 	if (ack->token != VSC_TOKEN_DUMP_RESP &&
596 	    ack->token != VSC_TOKEN_BOOTLOADER_REQ)
597 		return -EINVAL;
598 
599 	cmd->magic = cpu_to_le32(VSC_MAGIC_NUM);
600 	cmd->cmd_id = VSC_CMD_DL_START;
601 	cmd->data.dl_start.option = cpu_to_le16(fw_loader->option);
602 	cmd->data.dl_start.img_type = frag->type;
603 	cmd->data.dl_start.img_len = cpu_to_le32(frag->size);
604 	cmd->data.dl_start.img_loc = cpu_to_le32(frag->location);
605 
606 	c_len = offsetof(struct vsc_rom_cmd, data.dl_start.crc);
607 	cmd->data.dl_start.crc = cpu_to_le32(vsc_sum_crc(cmd, c_len));
608 
609 	ret = vsc_tp_rom_xfer(fw_loader->tp, cmd, NULL, VSC_ROM_PKG_SIZE);
610 	if (ret)
611 		return ret;
612 
613 	p = frag->data;
614 	remain = frag->size;
615 
616 	/* download image data */
617 	while (remain > 0) {
618 		len = min(remain, sizeof(cmd->data.dl_cont.payload));
619 
620 		cmd->magic = cpu_to_le32(VSC_MAGIC_NUM);
621 		cmd->cmd_id = VSC_CMD_DL_CONT;
622 		cmd->data.dl_cont.len = cpu_to_le16(len);
623 		cmd->data.dl_cont.end_flag = remain == len;
624 		memcpy(cmd->data.dl_cont.payload, p, len);
625 
626 		ret = vsc_tp_rom_xfer(fw_loader->tp, cmd, NULL, VSC_ROM_PKG_SIZE);
627 		if (ret)
628 			return ret;
629 
630 		p += len;
631 		remain -= len;
632 	}
633 
634 	return 0;
635 }
636 
637 static int vsc_download_firmware(struct vsc_fw_loader *fw_loader)
638 {
639 	struct vsc_fw_cmd *cmd = fw_loader->tx_buf;
640 	unsigned int i, index = 0;
641 	u32 c_len;
642 	int ret;
643 
644 	cmd->magic = cpu_to_le32(VSC_MAGIC_NUM);
645 	cmd->cmd_id = VSC_CMD_DL_SET;
646 	cmd->data.dl_set.img_cnt = cpu_to_le16(fw_loader->count);
647 	put_unaligned_le16(fw_loader->option, &cmd->data.dl_set.option);
648 
649 	for (i = VSC_IMG_CSI_SEM_FRAG; i <= VSC_IMG_CSI_EM7D_FRAG; i++) {
650 		struct vsc_img_frag *frag = &fw_loader->frags[i];
651 
652 		cmd->data.dl_set.payload[index++] = cpu_to_le32(frag->location);
653 		cmd->data.dl_set.payload[index++] = cpu_to_le32(frag->size);
654 	}
655 
656 	c_len = offsetof(struct vsc_fw_cmd, data.dl_set.payload[index]);
657 	cmd->data.dl_set.payload[index] = cpu_to_le32(vsc_sum_crc(cmd, c_len));
658 
659 	ret = vsc_tp_rom_xfer(fw_loader->tp, cmd, NULL, VSC_FW_PKG_SIZE);
660 	if (ret)
661 		return ret;
662 
663 	for (i = VSC_IMG_CSI_SEM_FRAG; i < VSC_IMG_FRAG_MAX; i++) {
664 		struct vsc_img_frag *frag = &fw_loader->frags[i];
665 		const u8 *p;
666 		u32 remain;
667 
668 		cmd->magic = cpu_to_le32(VSC_MAGIC_NUM);
669 		cmd->cmd_id = VSC_CMD_DL_START;
670 		cmd->data.dl_start.img_type = frag->type;
671 		cmd->data.dl_start.img_len = cpu_to_le32(frag->size);
672 		cmd->data.dl_start.img_loc = cpu_to_le32(frag->location);
673 		put_unaligned_le16(fw_loader->option, &cmd->data.dl_start.option);
674 
675 		c_len = offsetof(struct vsc_fw_cmd, data.dl_start.crc);
676 		cmd->data.dl_start.crc = cpu_to_le32(vsc_sum_crc(cmd, c_len));
677 
678 		ret = vsc_tp_rom_xfer(fw_loader->tp, cmd, NULL, VSC_FW_PKG_SIZE);
679 		if (ret)
680 			return ret;
681 
682 		p = frag->data;
683 		remain = frag->size;
684 
685 		/* download image data */
686 		while (remain > 0) {
687 			u32 len = min(remain, VSC_FW_PKG_SIZE);
688 
689 			memcpy(fw_loader->tx_buf, p, len);
690 			memset(fw_loader->tx_buf + len, 0, VSC_FW_PKG_SIZE - len);
691 
692 			ret = vsc_tp_rom_xfer(fw_loader->tp, fw_loader->tx_buf,
693 					      NULL, VSC_FW_PKG_SIZE);
694 			if (ret)
695 				break;
696 
697 			p += len;
698 			remain -= len;
699 		}
700 	}
701 
702 	cmd->magic = cpu_to_le32(VSC_MAGIC_NUM);
703 	cmd->cmd_id = VSC_CMD_CAM_BOOT;
704 
705 	c_len = offsetof(struct vsc_fw_cmd, data.dl_start.crc);
706 	cmd->data.boot.crc = cpu_to_le32(vsc_sum_crc(cmd, c_len));
707 
708 	return vsc_tp_rom_xfer(fw_loader->tp, cmd, NULL, VSC_FW_PKG_SIZE);
709 }
710 
711 /**
712  * vsc_tp_init - init vsc_tp
713  * @tp: vsc_tp device handle
714  * @dev: device node for mei vsc device
715  * Return: 0 in case of success, negative value in case of error
716  */
717 int vsc_tp_init(struct vsc_tp *tp, struct device *dev)
718 {
719 	struct vsc_fw_loader *fw_loader __free(kfree) = NULL;
720 	void *tx_buf __free(kfree) = NULL;
721 	void *rx_buf __free(kfree) = NULL;
722 	int ret;
723 
724 	fw_loader = kzalloc(sizeof(*fw_loader), GFP_KERNEL);
725 	if (!fw_loader)
726 		return -ENOMEM;
727 
728 	tx_buf = kzalloc(VSC_FW_PKG_SIZE, GFP_KERNEL);
729 	if (!tx_buf)
730 		return -ENOMEM;
731 
732 	rx_buf = kzalloc(VSC_FW_PKG_SIZE, GFP_KERNEL);
733 	if (!rx_buf)
734 		return -ENOMEM;
735 
736 	fw_loader->tx_buf = tx_buf;
737 	fw_loader->rx_buf = rx_buf;
738 
739 	fw_loader->tp = tp;
740 	fw_loader->dev = dev;
741 
742 	ret = vsc_get_sensor_name(fw_loader, dev);
743 	if (ret)
744 		return ret;
745 
746 	ret = vsc_identify_silicon(fw_loader);
747 	if (ret)
748 		return ret;
749 
750 	ret = vsc_identify_csi_image(fw_loader);
751 	if (ret)
752 		return ret;
753 
754 	ret = vsc_identify_ace_image(fw_loader);
755 	if (ret)
756 		goto err_release_csi;
757 
758 	ret = vsc_identify_cfg_image(fw_loader);
759 	if (ret)
760 		goto err_release_ace;
761 
762 	ret = vsc_download_bootloader(fw_loader);
763 	if (!ret)
764 		ret = vsc_download_firmware(fw_loader);
765 
766 	release_firmware(fw_loader->cfg);
767 
768 err_release_ace:
769 	release_firmware(fw_loader->ace);
770 
771 err_release_csi:
772 	release_firmware(fw_loader->csi);
773 
774 	return ret;
775 }
776 EXPORT_SYMBOL_NS_GPL(vsc_tp_init, "VSC_TP");
777