[PATCH v1 10/11] usb: gadget: amlogic: implement Optimus protocol

Arseniy Krasnov avkrasnov at salutedevices.com
Wed Mar 19 21:20:44 CET 2025


From: Vladimir Mitrofanov <vvmitrofanov at salutedevices.com>

Add Amlogic's Optimus protocol support for USB transport.

Signed-off-by: Vladimir Mitrofanov <vvmitrofanov at salutedevices.com>
Signed-off-by: Arseniy Krasnov <avkrasnov at salutedevices.com>
---
 cmd/Kconfig                                   |   7 +
 cmd/meson/Makefile                            |   1 +
 cmd/meson/optimus.c                           |  21 +
 drivers/usb/gadget/Makefile                   |   1 +
 drivers/usb/gadget/amlogic/Kconfig            |   1 +
 drivers/usb/gadget/amlogic/optimus/Kconfig    |  12 +
 drivers/usb/gadget/amlogic/optimus/Makefile   |   7 +
 .../usb/gadget/amlogic/optimus/f_optimus.c    | 687 ++++++++++++++++++
 .../gadget/amlogic/optimus/optimus_download.c | 188 +++++
 .../gadget/amlogic/optimus/optimus_download.h |  86 +++
 10 files changed, 1011 insertions(+)
 create mode 100644 cmd/meson/optimus.c
 create mode 100644 drivers/usb/gadget/amlogic/optimus/Kconfig
 create mode 100644 drivers/usb/gadget/amlogic/optimus/Makefile
 create mode 100644 drivers/usb/gadget/amlogic/optimus/f_optimus.c
 create mode 100644 drivers/usb/gadget/amlogic/optimus/optimus_download.c
 create mode 100644 drivers/usb/gadget/amlogic/optimus/optimus_download.h

diff --git a/cmd/Kconfig b/cmd/Kconfig
index 2f195d03848..159e52d5298 100644
--- a/cmd/Kconfig
+++ b/cmd/Kconfig
@@ -1153,6 +1153,13 @@ config CMD_ADNL
           This command, "adnl", enables the ADNL listen mode, providing
 	  the ability to listen in ADNL mode.
 
+config CMD_OPTIMUS
+	bool "OPTIMUS - Amlogic download protocol"
+	depends on OPTIMUS
+	help
+          This command, "optimus", enables the OPTIMUS listen mode, providing
+	  the ability to communicate with host by OPTIMUS protocol.
+
 config CMD_FLASH
 	bool "flinfo, erase, protect"
 	default y
diff --git a/cmd/meson/Makefile b/cmd/meson/Makefile
index 1cf984378a3..611d1549736 100644
--- a/cmd/meson/Makefile
+++ b/cmd/meson/Makefile
@@ -4,3 +4,4 @@
 
 obj-y += sm.o
 obj-$(CONFIG_CMD_ADNL) += gadget.o adnl.o
+obj-$(CONFIG_CMD_OPTIMUS) += gadget.o optimus.o
diff --git a/cmd/meson/optimus.c b/cmd/meson/optimus.c
new file mode 100644
index 00000000000..97a1435c0ab
--- /dev/null
+++ b/cmd/meson/optimus.c
@@ -0,0 +1,21 @@
+// SPDX-License-Identifier: GPL-2.0+
+/*
+ * Copyright (c) 2024 SaluteDevices, Inc.
+ * Author: Vladimir Mitrofanov <vvmitrofanov at salutedevices.com>
+ */
+
+#include <command.h>
+
+#include "gadget.h"
+
+static int do_optimus(struct cmd_tbl *cmdtp, int flag, int argc,
+		      char *const argv[])
+{
+	return amlogic_gadget_run("usb_dnl_optimus", argc, argv);
+}
+
+U_BOOT_LONGHELP(optimus,
+		"- run as an OPTIMUS USB device\n\n");
+
+U_BOOT_CMD(optimus, 1, 1, do_optimus,
+	   "OPTIMUS protocol mode", optimus_help_text);
diff --git a/drivers/usb/gadget/Makefile b/drivers/usb/gadget/Makefile
index 2b6705c52be..f5899a96221 100644
--- a/drivers/usb/gadget/Makefile
+++ b/drivers/usb/gadget/Makefile
@@ -22,6 +22,7 @@ obj-$(CONFIG_USB_GADGET_DWC2_OTG) += dwc2_udc_otg.o
 obj-$(CONFIG_USB_GADGET_DWC2_OTG_PHY) += dwc2_udc_otg_phy.o
 obj-$(CONFIG_USB_GADGET_MAX3420) += max3420_udc.o
 obj-$(CONFIG_USB_RENESAS_USBHS) += rcar/
+obj-$(CONFIG_OPTIMUS) += f_fastboot_common.o amlogic/optimus/
 ifndef CONFIG_XPL_BUILD
 obj-$(CONFIG_USB_GADGET_DOWNLOAD) += g_dnl.o
 obj-$(CONFIG_USB_FUNCTION_THOR) += f_thor.o
diff --git a/drivers/usb/gadget/amlogic/Kconfig b/drivers/usb/gadget/amlogic/Kconfig
index 460dc6fe235..5a16fb25998 100644
--- a/drivers/usb/gadget/amlogic/Kconfig
+++ b/drivers/usb/gadget/amlogic/Kconfig
@@ -7,5 +7,6 @@ choice
 	optional
 
 source "drivers/usb/gadget/amlogic/adnl/Kconfig"
+source "drivers/usb/gadget/amlogic/optimus/Kconfig"
 
 endchoice
diff --git a/drivers/usb/gadget/amlogic/optimus/Kconfig b/drivers/usb/gadget/amlogic/optimus/Kconfig
new file mode 100644
index 00000000000..6a730670d34
--- /dev/null
+++ b/drivers/usb/gadget/amlogic/optimus/Kconfig
@@ -0,0 +1,12 @@
+# SPDX-License-Identifier: GPL-2.0+
+#
+# (C) Copyright 2024 SaluteDevices, Inc.
+
+config OPTIMUS
+	bool "Enable OPTIMUS protocol"
+	depends on USB_GADGET
+	depends on MESON_AXG
+	help
+	  This enables usb optimus function protocol
+	  "Optimus" protocol is used to flash Amlogic devices
+	  from bootRom or from U-Boot mode
diff --git a/drivers/usb/gadget/amlogic/optimus/Makefile b/drivers/usb/gadget/amlogic/optimus/Makefile
new file mode 100644
index 00000000000..f48806414aa
--- /dev/null
+++ b/drivers/usb/gadget/amlogic/optimus/Makefile
@@ -0,0 +1,7 @@
+# SPDX-License-Identifier: GPL-2.0+
+
+obj-y += f_optimus_cb.o
+
+subdir-ccflags-y += -I$(src)/../ -I$(src)/../../
+f_optimus_cb-y += f_optimus.o
+f_optimus_cb-y += optimus_download.o
diff --git a/drivers/usb/gadget/amlogic/optimus/f_optimus.c b/drivers/usb/gadget/amlogic/optimus/f_optimus.c
new file mode 100644
index 00000000000..9912a691301
--- /dev/null
+++ b/drivers/usb/gadget/amlogic/optimus/f_optimus.c
@@ -0,0 +1,687 @@
+// SPDX-License-Identifier: GPL-2.0
+/*
+ * Copyright (c) 2024 SaluteDevices, Inc.
+ * Author: Vladimir Mitrofanov <vvmitrofanov at salutedevices.com>
+ */
+/*
+ * OPTIMUS - is Amlogic protocol for updating firmware.
+ * This protocol is used for flashing devices based on Amlogic SoC.
+ * Optimus uses specific image format. In this implementation USB
+ * is used for data transfer.
+ *
+ * There are two ways to switch device to flashing mode:
+ * 1. Enter bootROM flashing mode (how this should be done depends on device)
+ * 2. Run from TPL (ex. from U-boot)
+ *
+ * Flashing steps
+ *  1. Device has to be switched to Optimus mode (as described earlier).
+ *  2. Host detects it as "Amlogic DNL" device, by sending identify request.
+ *  3. Host sends u-boot. BootROM loads it to RAM and executes u-boot.
+ *     Finally u-boot detects that we are in Optimus update process, so it enters
+ *     Optimus mode to continue fw update.
+ *  4. Now host starts to send protocol commands.
+ *     First host sends some preparation commands like "low_power", "echo 1234" etc.
+ *     The order of this commands depends on host software.
+ *     Host may start flashing from any partition of the image (depends on host)
+ *  5. Host sends "bootloader_is_old" and "erase_bootloader". Device is preparing
+ *     buffers and storage to receive "bootloader". "Bootloader" consists of two
+ *     parts "bl2" and "tpl". As "tpl" could be used u-boot.
+ *  6. Host sends "download" command that contains information about full size of
+ *     partition that will be sent. Partition is sending by chunks. Every chunk
+ *     writes to permanent memory.
+ *     "Bootloader" is exception. It should be fully received before writing to storage.
+ *     After getting full "bootloader" device starts flashing it in to permanent memory.
+ *  7. After sending each partition of firmware Host sends "download get_status".
+ *  8. For every success operation Device respond with "success" status.
+ *  9. Host repeats steps form 6 - 8 with other partitions (A, B, ...).
+ *     This partitions are sending by chunks, but in contrast ("bootloader") they
+ *     are transmitting and writing to permanent memory by block of 64KiB (the size
+ *     depends on host software).
+ * 10. After all partitions are received Host sends "save_setting" to restore
+ *     environment to default state.
+ *
+ * NOTES#1:
+ * Current implementation has limited set of OPTIMUS features and commands.
+ * It was tested only on A113 SoC with one image of specific layout/content
+ * (BL2, TPL, etc.).
+ *
+ * NOTES#2:
+ * "bootloader" partition has special layout: BL2 and TPL are saved as
+ * several copies of each other with specific alignment.
+ *
+ * NOTES#3:
+ * This implementation requires that data buffer address and size to be
+ * set in the config file. This buffers will be used for both IN and OUT
+ * USB data.
+ */
+
+#include <cli.h>
+#include <command.h>
+#include <env.h>
+#include <fastboot.h>
+#include <g_dnl.h>
+#include <log.h>
+#include <asm/arch-meson/sm.h>
+#include <asm/io.h>
+#include <linux/delay.h>
+#include <linux/mtd/mtd.h>
+#include <u-boot/sha1.h>
+
+#include "f_fastboot_common.h"
+#include "optimus_download.h"
+
+#define CMD_DOWNLOAD      "download"
+#define CMD_LOWPOWER      "low_power"
+#define CMD_BL_IS_OLD     "bootloader_is_old"
+#define CMD_ERASE_BL      "erase_bootloader"
+#define CMD_RESET         "reset"
+#define CMD_UPLOAD        "upload"
+#define CMD_VERIFY        "verify"
+#define CMD_DISK_INITIAL  "disk_initial"
+#define CMD_SAVE_SET      "save_setting"
+#define CMD_BURN_COMPL    "burn_complete"
+#define CMD_ECHO_1234     "echo"
+#define CMD_SETENV        "setenv"
+#define CMD_SAVE          "save"
+#define CMD_RPMB_RESET    "rpmb_reset"
+#define CMD_AMLMMC        "amlmmc"
+#define CMD_NAND          "nand"
+#define CMD_SUB_GETSTATUS "get_status"
+#define CMD_SUB_STORE     "store"
+
+#define MAX_RESPONSE_LEN  512
+#define MAX_CMD_LEN       60
+#define SPACE_IN_HEX      0x20
+
+#define REQ_WRITE_MEM     0x01
+#define REQ_READ_MEM      0x02
+#define REQ_RUN_IN_ADDR   0x05
+#define REQ_WR_LARGE_MEM  0x11
+#define REQ_RD_LARGE_MEM  0x12
+#define REQ_IDENTIFY_HOST 0x20
+#define REQ_TPL_CMD       0x30
+#define REQ_TPL_STAT      0x31
+#define REQ_BULKCMD       0x34
+#define REQ_WRITE_MEDIA   0x32
+#define REQ_READ_MEDIA    0x33
+
+typedef void (*cmd_handler) (const struct usb_request *req);
+
+static cmd_handler get_handler(const char *cmd);
+
+static const char opti_name[] = "WorldCup Device";
+static const char resp_success[] = "success";
+/* Pay attention \":\" is in the end. Protocol specific. */
+static const char resp_failed[] = "failed:";
+
+static struct usb_os_desc_ext_prop opti_ext_prop = {
+	.type = 1,		/* NULL-terminated Unicode String (REG_SZ) */
+	.name = "DeviceInterfaceGUID",
+	.data = "{4866319A-F4D6-4374-93B9-DC2DEB361BA9}",
+};
+
+/* 16 bytes of "Compatible ID" and "Subcompatible ID" */
+static char opti_cid[16] = {'W', 'I', 'N', 'U', 'S', 'B'};
+static struct usb_os_desc opti_os_desc = {
+	.ext_compat_id = opti_cid,
+};
+
+static struct usb_os_desc_table opti_os_desc_table = {
+	.os_desc = &opti_os_desc,
+};
+
+static struct fastboot_funcs *opti_func;
+
+static struct usb_string opti_string_defs[] = {
+	[0].s = opti_name,
+	{  } /* end of list */
+};
+
+static struct usb_gadget_strings stringtab_opti = {
+	.language	= 0x0409,	/* en-us */
+	.strings	= opti_string_defs,
+};
+
+static struct usb_gadget_strings *opti_strings[] = {
+	&stringtab_opti,
+	NULL,
+};
+
+/* This function is oriented to send string response of desired size ending by zero bytes */
+static int opti_tx_write(const char *buffer, size_t buffer_size)
+{
+	int ret;
+	struct usb_request *in_req = opti_func->in_req;
+	size_t resp_len = strnlen(buffer, MAX_RESPONSE_LEN);
+
+	if (!buffer_size || buffer_size > MAX_RESPONSE_LEN)
+		buffer_size = MAX_RESPONSE_LEN;
+
+	resp_len = min_t(size_t, resp_len, buffer_size);
+	memset(in_req->buf, 0, MAX_RESPONSE_LEN);
+	memcpy(in_req->buf, buffer, resp_len);
+
+	in_req->length = buffer_size;
+
+	ret = usb_ep_queue(opti_func->in_ep, in_req, 0);
+	if (ret)
+		OPTI_ERR("Response with size %zu failed: %d", buffer_size, ret);
+
+	return ret;
+}
+
+static void opti_rx_handler_command(struct usb_ep *ep, struct usb_request *req);
+
+static int opti_bind(struct usb_configuration *c, struct usb_function *f)
+{
+	return fastboot_common_bind(c, f, &opti_os_desc_table,
+				    &opti_os_desc, (struct usb_string **)&opti_string_defs,
+				    &opti_ext_prop);
+}
+
+static void opti_unbind(struct usb_configuration *c, struct usb_function *f)
+{
+	fastboot_common_unbind(c, f, &opti_os_desc);
+	memset(opti_func, 0, sizeof(*opti_func));
+}
+
+static void opti_disable(struct usb_function *f)
+{
+	fastboot_common_disable(f);
+}
+
+static int opti_set_alt(struct usb_function *f,
+			unsigned int interface, unsigned int alt)
+{
+	int ret;
+	struct usb_configuration *c;
+
+	if (interface)
+		return 0;
+
+	ret = fastboot_common_set_alt(f, interface, alt, opti_rx_handler_command);
+
+	c = f->config;
+
+	if (!c->interface[0])
+		return ret;
+
+	for (size_t i = 1; i < MAX_CONFIG_INTERFACES; i++) {
+		if (!c->interface[i])
+			c->interface[i] = c->interface[0];
+	}
+
+	return ret;
+}
+
+void complete_tpl_cmd(struct usb_ep *ep, struct usb_request *req)
+{
+	cmd_handler cb;
+
+	((char *)req->buf)[req->actual] = '\0';
+	cb = get_handler(req->buf);
+
+	if (!cb) {
+		OPTI_ERR("Unknown command: %.*s", MAX_CMD_LEN, (char *)req->buf);
+		return;
+	}
+
+	cb(req);
+}
+
+void complete_write_media(struct usb_ep *ep, struct usb_request *req)
+{
+	struct optimus_chunk in;
+	struct optimus_img *img;
+
+	in = *((struct optimus_chunk *)req->buf);
+
+	img = optimus_get_img();
+
+	img->cur_chunk_meta = in;
+
+	if (optimus_chunk_alloc_buf(img->fsize))
+		return;
+
+	opti_func->out_req->length = min_t(u32, in.data_length, EP_BUFFER_SIZE);
+	opti_func->out_req->complete = opti_rx_handler_command;
+
+	OPTI_MSG_ERASE("Image %s received:%*zd%%%c", img->name,
+		       4, img->dsize * 100 / img->fsize, SPACE_IN_HEX);
+
+	usb_ep_queue(opti_func->out_ep, opti_func->out_req, 0);
+}
+
+static int optimus_setup(struct usb_function *f, const struct usb_ctrlrequest *ctrl)
+{
+	struct usb_gadget *gadget = f->config->cdev->gadget;
+	struct usb_request *req = f->config->cdev->req;
+	u16 len = le16_to_cpu(ctrl->wLength);
+	u16 w_value = le16_to_cpu(ctrl->wValue);
+	int value;
+
+	u8 resp_buf[MAX_RESPONSE_LEN];
+	u8 host_identity[] = {0, 7, 0, 16, 0, 0, 0, 0};
+
+	switch (ctrl->bRequest) {
+	case REQ_IDENTIFY_HOST:
+		value = min_t(u16, len, MAX_RESPONSE_LEN);
+		memcpy(req->buf, host_identity, value);
+		break;
+	case REQ_TPL_CMD:
+		value = len;
+		req->complete = complete_tpl_cmd;
+		break;
+	case REQ_TPL_STAT:
+		value = (!len) ? 0 : len - 1;
+		memset(req->buf, 0, len);
+		memcpy(req->buf, resp_success, sizeof(resp_success) - 1);
+		break;
+	case REQ_BULKCMD:
+		value = len;
+		req->complete = complete_tpl_cmd;
+		break;
+	case REQ_READ_MEDIA:
+		value = len;
+		memset(resp_buf, 0, sizeof(resp_buf));
+		opti_tx_write(resp_buf, w_value);
+		break;
+	case REQ_WRITE_MEDIA:
+		/* This is main request to receive image chunks of data */
+		value = len;
+		req->complete = complete_write_media;
+		break;
+	case REQ_WRITE_MEM:
+	case REQ_READ_MEM:
+	case REQ_WR_LARGE_MEM:
+	case REQ_RUN_IN_ADDR:
+		value = len;
+		memset(req->buf, 0, len);
+		break;
+	default:
+		value = 0;
+		OPTI_ERR("\nUnknown request: bRequest %#x", ctrl->bRequest);
+	}
+
+	req->length = value;
+	req->zero = 0;
+
+	value = usb_ep_queue(gadget->ep0, req, GFP_KERNEL);
+	if (value < 0) {
+		OPTI_ERR("Response \"ep0\" failed (%d)", value);
+		req->status = 0;
+	}
+
+	return value;
+}
+
+int opti_add(struct usb_configuration *c)
+{
+	struct fastboot_funcs *f_opti = opti_func;
+	int status;
+
+	OPTI_MSG("Start usb \"optimus\"");
+	if (!f_opti) {
+		f_opti = memalign(CONFIG_SYS_CACHELINE_SIZE, sizeof(*f_opti));
+		if (!f_opti)
+			return -ENOMEM;
+
+		opti_func = f_opti;
+		memset(f_opti, 0, sizeof(*f_opti));
+	}
+
+	g_dnl_set_product("DNL");
+	f_opti->usb_function.name = "f_opti";
+	f_opti->usb_function.bind = opti_bind;
+	f_opti->usb_function.unbind = opti_unbind;
+	f_opti->usb_function.set_alt = opti_set_alt;
+	f_opti->usb_function.disable = opti_disable;
+	f_opti->usb_function.strings = opti_strings;
+	f_opti->usb_function.setup = optimus_setup;
+
+	status = usb_add_function(c, &f_opti->usb_function);
+	if (status) {
+		free(f_opti);
+		opti_func = NULL;
+	}
+
+	return status;
+}
+
+DECLARE_GADGET_BIND_CALLBACK(usb_dnl_optimus, opti_add);
+
+static void cb_download(const struct usb_request *req)
+{
+	char *cmd[10];
+	char *token;
+	u16 i;
+	unsigned int img_size;
+	const char delim[] = {SPACE_IN_HEX, '\x00'};
+	struct optimus_img *img;
+
+	/* Finalizing received buffer by \x00 */
+	token = req->buf;
+	token[req->actual] = '\x00';
+
+	i = 0;
+	token = strtok(req->buf, delim);
+	while (token && i < ARRAY_SIZE(cmd)) {
+		cmd[i] = token;
+		token = strtok(NULL, delim);
+		i++;
+	}
+
+	img = optimus_get_img();
+
+	if (i >= 1 && !strncmp(cmd[1], CMD_SUB_GETSTATUS, sizeof(CMD_SUB_GETSTATUS) - 1)) {
+		if (img->dsize == img->fsize) {
+			opti_tx_write(resp_success, img->cur_chunk_meta.ack_len);
+			OPTI_MSG("Partition \'%s\' is fully received", img->name);
+		} else {
+			opti_tx_write(resp_failed, img->cur_chunk_meta.ack_len);
+			OPTI_MSG("Partition \'%s\' receiving failed", img->name);
+		}
+	} else if (i >= 4 && !strcmp(cmd[1], CMD_SUB_STORE)) {
+		img_size = simple_strtoul(cmd[4], NULL, 10);
+		/* Init first image chunk by full image name and full image size. */
+		optimus_chunk_init(cmd[2], img_size);
+	}
+}
+
+static void cb_lowpower(const struct usb_request *req)
+{
+	struct optimus_img *img;
+
+	img = optimus_get_img();
+	opti_tx_write(resp_success, img->cur_chunk_meta.ack_len);
+}
+
+static void cb_bl_is_old(const struct usb_request *req)
+{
+	struct optimus_img *img;
+
+	img = optimus_get_img();
+	opti_tx_write(resp_success, img->cur_chunk_meta.ack_len);
+}
+
+static void cb_erase_bl(const struct usb_request *req)
+{
+	struct optimus_img *img;
+
+	img = optimus_get_img();
+	opti_tx_write(resp_success, img->cur_chunk_meta.ack_len);
+}
+
+static void cb_reset(const struct usb_request *req)
+{
+	meson_sm_set_usb_boot_mode(FORCE_USB_BOOT);
+	meson_sm_reboot(REBOOT_REASON_NORMAL);
+}
+
+static void cb_upload(const struct usb_request *req)
+{
+	const char **param_list;
+	size_t resp_len = 0;
+	char *req_buf_ch;
+
+	req_buf_ch = (char *)req->buf;
+	req_buf_ch[MAX_RESPONSE_LEN - 1] = '\0';
+	param_list = str_to_list(req_buf_ch);
+
+	if (param_list) {
+		size_t param_count = 0;
+
+		while (param_list[param_count])
+			param_count++;
+
+		if (param_count > 4)
+			resp_len = simple_strtoul(param_list[4], NULL, 0);
+	} else {
+		resp_len = 0x4;
+	}
+
+	str_free_list(param_list);
+
+	opti_tx_write(resp_success, resp_len);
+}
+
+static void cb_verify(const struct usb_request *req)
+{
+	struct optimus_img *img;
+
+	img = optimus_get_img();
+	opti_tx_write(resp_success, img->cur_chunk_meta.ack_len);
+}
+
+static bool opti_must_skip_block(struct mtd_info *mtd, off_t offset)
+{
+	if (mtd_block_isbad(mtd, offset))
+		return true;
+
+	if (mtd_block_isreserved(mtd, offset))
+		return true;
+
+	return false;
+}
+
+static void cb_disk_initial(const struct usb_request *req)
+{
+	struct optimus_img *img;
+	struct mtd_info *mtd;
+	off_t offset;
+
+	mtd = get_mtd_device_nm("bootloader");
+	if (!mtd || !mtd->parent)
+		return;
+
+	mtd = mtd->parent;
+
+	for (offset = 0; offset < mtd->size; offset += mtd->erasesize) {
+		struct erase_info erase_op = {};
+		int ret;
+
+		if (opti_must_skip_block(mtd, offset))
+			continue;
+
+		erase_op.mtd = mtd;
+		erase_op.addr = offset;
+		erase_op.len = mtd->erasesize;
+
+		ret = mtd_erase(mtd, &erase_op);
+		if (ret) {
+			OPTI_ERR("Can't erase block at %lx\n",
+				  offset);
+			return;
+		}
+	}
+
+	img = optimus_get_img();
+	opti_tx_write(resp_success, img->cur_chunk_meta.ack_len);
+}
+
+static void cb_save_setting(const struct usb_request *req)
+{
+	struct optimus_img *img;
+
+	img = optimus_get_img();
+	opti_tx_write(resp_success, img->cur_chunk_meta.ack_len);
+}
+
+static void cb_burn_complete(const struct usb_request *req)
+{
+	struct optimus_img *img;
+
+	img = optimus_get_img();
+	opti_tx_write(resp_success, img->cur_chunk_meta.ack_len);
+}
+
+static void cb_echo_1234(const struct usb_request *req)
+{
+	struct optimus_img *img;
+
+	img = optimus_get_img();
+	opti_tx_write(resp_success, img->cur_chunk_meta.ack_len);
+}
+
+static void cb_setenv(const struct usb_request *req)
+{
+	struct optimus_img *img;
+
+	img = optimus_get_img();
+	env_set("firstboot", "1");
+	opti_tx_write(resp_success, img->cur_chunk_meta.ack_len);
+}
+
+static void cb_save(const struct usb_request *req)
+{
+	struct optimus_img *img;
+
+	img = optimus_get_img();
+	env_save();
+	opti_tx_write(resp_success, img->cur_chunk_meta.ack_len);
+}
+
+static void cb_rpmb_reset(const struct usb_request *req)
+{
+	struct optimus_img *img;
+
+	img = optimus_get_img();
+	opti_tx_write(resp_success, img->cur_chunk_meta.ack_len);
+}
+
+static void cb_amlmmc(const struct usb_request *req)
+{
+	struct optimus_img *img;
+
+	img = optimus_get_img();
+	opti_tx_write(resp_success, img->cur_chunk_meta.ack_len);
+}
+
+static void cb_nand(const struct usb_request *req)
+{
+	struct optimus_img *img;
+
+	img = optimus_get_img();
+	opti_tx_write(resp_success, img->cur_chunk_meta.ack_len);
+}
+
+static void opti_rx_handler_command(struct usb_ep *ep, struct usb_request *req)
+{
+	enum optimus_chunk_state ret = optimus_chunk_process(req->buf, req->length);
+	struct optimus_img *img;
+
+	img = optimus_get_img();
+
+	switch (ret) {
+	case OPTI_CHUNK_STATE_COMPLETE:
+		req->complete = NULL;
+		opti_tx_write("OK!!", img->cur_chunk_meta.ack_len);
+		break;
+	case OPTI_CHUNK_STATE_IN_PROGRESS:
+		req->length = img->req_length;
+		req->complete = opti_rx_handler_command;
+		if (!usb_ep_queue(ep, req, 0))
+			break;
+	case OPTI_CHUNK_STATE_ERROR:
+		req->complete = NULL;
+		opti_tx_write("FALSE!!", img->cur_chunk_meta.ack_len);
+		break;
+	case OPTI_CHUNK_STATE_NEED_INIT:
+		break;
+	}
+}
+
+struct cmd_dispatch_info {
+	const char *cmd;
+	cmd_handler cb;
+};
+
+static const struct cmd_dispatch_info cmd_dispatch_info[] = {
+	{
+		.cmd = CMD_RESET,
+		.cb  = cb_reset,
+	},
+	{
+		.cmd = CMD_DOWNLOAD,
+		.cb = cb_download,
+	},
+	{
+		.cmd = CMD_LOWPOWER,
+		.cb = cb_lowpower,
+	},
+	{
+		.cmd = CMD_BL_IS_OLD,
+		.cb = cb_bl_is_old,
+	},
+	{
+		.cmd = CMD_ERASE_BL,
+		.cb = cb_erase_bl,
+	},
+	{
+		.cmd = CMD_UPLOAD,
+		.cb = cb_upload,
+	},
+	{
+		.cmd = CMD_VERIFY,
+		.cb = cb_verify,
+	},
+	{
+		.cmd = CMD_DISK_INITIAL,
+		.cb = cb_disk_initial,
+	},
+	{
+		.cmd = CMD_SAVE_SET,
+		.cb = cb_save_setting,
+	},
+	{
+		.cmd = CMD_BURN_COMPL,
+		.cb = cb_burn_complete,
+	},
+	{
+		.cmd = CMD_ECHO_1234,
+		.cb = cb_echo_1234,
+	},
+	{
+		.cmd = CMD_SETENV,
+		.cb = cb_setenv,
+	},
+	{
+		.cmd = CMD_SAVE,
+		.cb = cb_save,
+	},
+	{
+		.cmd = CMD_RPMB_RESET,
+		.cb = cb_rpmb_reset,
+	},
+	{
+		.cmd = CMD_AMLMMC,
+		.cb = cb_amlmmc,
+	},
+	{
+		.cmd = CMD_NAND,
+		.cb = cb_nand,
+	},
+};
+
+static cmd_handler get_handler(const char *cmd)
+{
+	u32 i;
+	const char *str_ptr;
+
+	if (!cmd)
+		return NULL;
+
+	str_ptr = skip_spaces(cmd);
+
+	for (i = 0; i < ARRAY_SIZE(cmd_dispatch_info); i++) {
+		int ret;
+
+		ret = strncmp(str_ptr, cmd_dispatch_info[i].cmd, strlen(cmd_dispatch_info[i].cmd));
+		if (!ret) {
+			OPTI_MSG("In cmd: '%s'", cmd_dispatch_info[i].cmd);
+			return cmd_dispatch_info[i].cb;
+		}
+	}
+
+	return NULL;
+}
diff --git a/drivers/usb/gadget/amlogic/optimus/optimus_download.c b/drivers/usb/gadget/amlogic/optimus/optimus_download.c
new file mode 100644
index 00000000000..c78e39ed23e
--- /dev/null
+++ b/drivers/usb/gadget/amlogic/optimus/optimus_download.c
@@ -0,0 +1,188 @@
+// SPDX-License-Identifier: GPL-2.0
+/*
+ * Copyright (c) 2024 SaluteDevices, Inc.
+ * Author: Vladimir Mitrofanov <vvmitrofanov at salutedevices.com>
+ */
+
+#include <command.h>
+#include <log.h>
+#include <mtd.h>
+#include <nand.h>
+#include <asm/arch/rawnand.h>
+#include <linux/err.h>
+#include <linux/delay.h>
+#include <linux/libfdt.h>
+#include <linux/mtd/nand.h>
+#include <linux/mtd/rawnand.h>
+
+#include "f_fastboot_common.h"
+#include "optimus_download.h"
+
+static struct optimus_img flash_img = { .ptr = NULL };
+
+struct optimus_img *optimus_get_img(void)
+{
+	return &flash_img;
+}
+
+static int optimus_write_normal_image(u32 data_sz, u8 *data)
+{
+	struct optimus_img *img;
+	size_t written_length;
+	size_t length;
+	struct mtd_info *mtd;
+	int ret;
+
+	if (!data_sz)
+		return 0;
+
+	img = optimus_get_img();
+
+	mtd = get_mtd_device_nm(img->name);
+
+	if (IS_ERR_OR_NULL(mtd)) {
+		OPTI_ERR("Failed to get mtd device \"%s\" ret: %ld", img->name, PTR_ERR(mtd));
+		return PTR_ERR(mtd);
+	}
+
+	/* Need use intermediate storage to write conversion of types pointers */
+	length = data_sz;
+
+	written_length = 0;
+	ret = nand_write_skip_bad(mtd, img->media_offset, &length,
+				  &written_length, mtd->size, data, 0);
+	img->media_offset += written_length;
+
+	return ret;
+}
+
+void optimus_chunk_init(const char *name, unsigned int size)
+{
+	struct optimus_img *img;
+
+	img = optimus_get_img();
+
+	img->fsize = size;
+
+	memset(img->name, 0, sizeof(img->name));
+	strncpy(img->name, name, sizeof(img->name) - 1);
+
+	img->dsize = 0;
+	img->media_offset = 0;
+}
+
+int optimus_chunk_alloc_buf(unsigned int size)
+{
+	struct optimus_img *img;
+	size_t full_size;
+
+	img = optimus_get_img();
+
+	img->fsize = size;
+	img->cur_chunk_data_len = 0;
+
+	if (!strcmp(img->name, BOOT_LOADER)) {
+		img->type = OPTI_IMG_TYPE_BOOTLOADER;
+		full_size = img->fsize;
+	} else {
+		img->type = OPTI_IMG_TYPE_NORMAL;
+		full_size = img->cur_chunk_meta.data_length;
+	}
+
+	if (!img->ptr)
+		img->ptr = (u8 *)malloc(full_size);
+
+	if (!img->ptr) {
+		OPTI_ERR("No memory for data chunk");
+		return -ENOMEM;
+	}
+
+	return 0;
+}
+
+static void optimus_chunk_free(void)
+{
+	struct optimus_img *img;
+
+	img = optimus_get_img();
+
+	free(img->ptr);
+	img->ptr = NULL;
+}
+
+static unsigned int optimus_csum_special(const void *p_buf, const unsigned int size)
+{
+	const unsigned int *data = (const unsigned int *)p_buf;
+	unsigned int word_len = size >> 2;
+	unsigned int rest = size & 0x3;
+	unsigned int sum = 0;
+
+	while (word_len--)
+		sum += *data++;
+
+	if (rest)
+		sum += (*data) & GENMASK(rest * 8, 0);
+
+	return sum;
+}
+
+enum optimus_chunk_state optimus_chunk_process(const void *data, unsigned int size)
+{
+	struct optimus_img *img;
+	int rw_ret;
+	u32 len_diff;
+	u32 check_sum;
+	u32 pos;
+
+	img = optimus_get_img();
+
+	if (!img->ptr)
+		return OPTI_CHUNK_STATE_NEED_INIT;
+
+	if (img->type == OPTI_IMG_TYPE_BOOTLOADER)
+		pos = img->dsize;
+	else
+		pos = img->cur_chunk_data_len;
+
+	if (img->ptr + pos + size <= img->ptr + img->fsize)
+		memcpy(img->ptr + pos, data, size);
+	else
+		return OPTI_CHUNK_STATE_ERROR;
+
+	img->dsize += size;
+	img->cur_chunk_data_len += size;
+	if (img->cur_chunk_data_len < img->cur_chunk_meta.data_length) {
+		len_diff = img->cur_chunk_meta.data_length - img->cur_chunk_data_len;
+		img->req_length = min_t(u32, len_diff, EP_BUFFER_SIZE);
+		return OPTI_CHUNK_STATE_IN_PROGRESS;
+	}
+
+	/* All chunk data received. */
+	if (img->type == OPTI_IMG_TYPE_BOOTLOADER)
+		pos = img->dsize - img->cur_chunk_data_len;
+	else
+		pos = 0;
+
+	check_sum = optimus_csum_special(img->ptr + pos, img->cur_chunk_data_len);
+	if (check_sum != img->cur_chunk_meta.chksum)
+		return OPTI_CHUNK_STATE_ERROR;
+
+	if (img->type == OPTI_IMG_TYPE_BOOTLOADER && size) {
+		/* Not full bootloader image. Chunk is only part. */
+		if (img->dsize < img->fsize)
+			return OPTI_CHUNK_STATE_COMPLETE;
+
+		/* We write bootloader after getting full image */
+		rw_ret = meson_bootloader_write(img->ptr, img->fsize);
+		optimus_chunk_free();
+		if (rw_ret)
+			return OPTI_CHUNK_STATE_ERROR;
+	} else {
+		rw_ret = optimus_write_normal_image((u32)img->cur_chunk_data_len, img->ptr);
+		optimus_chunk_free();
+		if (rw_ret)
+			return OPTI_CHUNK_STATE_ERROR;
+	}
+
+	return OPTI_CHUNK_STATE_COMPLETE;
+}
diff --git a/drivers/usb/gadget/amlogic/optimus/optimus_download.h b/drivers/usb/gadget/amlogic/optimus/optimus_download.h
new file mode 100644
index 00000000000..3a8682df045
--- /dev/null
+++ b/drivers/usb/gadget/amlogic/optimus/optimus_download.h
@@ -0,0 +1,86 @@
+/* SPDX-License-Identifier: GPL-2.0 */
+/*
+ * Copyright (C) 2024 SaluteDevices, Inc. All rights reserved.
+ * Author: Vladimir Mitrofanov <vvmitrofanov at salutedevices.com>
+ */
+
+#ifndef __OPTIMUS_DOWNLOAD_H__
+#define __OPTIMUS_DOWNLOAD_H__
+
+#define OPTI_PREF          "[OPTI][MSG] "
+#define ERASE_SEQ          "\x1B[2K\r"
+#define __OPTI_MSG(prefix, fmt, ...) printf(prefix fmt, ##__VA_ARGS__)
+
+#define OPTI_MSG(fmt, ...) __OPTI_MSG("\n" OPTI_PREF, fmt, ##__VA_ARGS__)
+#define OPTI_MSG_ERASE(fmt, ...) __OPTI_MSG(ERASE_SEQ "" OPTI_PREF, fmt, ##__VA_ARGS__)
+#define OPTI_ERR(fmt, ...) pr_err("\n[OPTI][ERR]: " fmt, ##__VA_ARGS__)
+
+enum optimus_img_type {
+	OPTI_IMG_TYPE_BOOTLOADER,
+	OPTI_IMG_TYPE_NORMAL,
+};
+
+enum optimus_chunk_state {
+	OPTI_CHUNK_STATE_COMPLETE,
+	OPTI_CHUNK_STATE_IN_PROGRESS,
+	OPTI_CHUNK_STATE_NEED_INIT,
+	OPTI_CHUNK_STATE_ERROR,
+};
+
+struct optimus_chunk {
+	u32	retry_times;	/* Unused in this version. */
+	u32	data_length;	/* Length of data in this chunk. */
+	u32	seq_num;	/* Unused in this version. */
+	u32	chksum;		/* Checksum (see 'optimus_csum_special()'. */
+	u16	chksum_alg;	/* Unused in this version. */
+	u16	ack_len;	/* ACK packet length, expected by sender. */
+} __packed;
+
+struct optimus_img {
+	u8	*ptr;			/* Data buffer. */
+	size_t	dsize;			/* Downloaded size. */
+	size_t	fsize;			/* Image full size. */
+	char	name[256];		/* Current partition name. */
+	enum	optimus_img_type type;	/* Current partition type. */
+
+	struct	optimus_chunk cur_chunk_meta;	/* Current chunk. */
+	u32	cur_chunk_data_len;		/* Downloaded size for current chunk. */
+	u32	req_length;			/* Rest to download, limited by EP buffer. */
+	loff_t	media_offset;			/* Current offset to write this image. */
+};
+
+/**
+ * optimus_get_img() - Get current Optimus image (context).
+ *
+ * @return: pointer to Optimus image (context).
+ */
+struct optimus_img *optimus_get_img(void);
+
+/**
+ * optimus_chunk_init() - Initialize "chunk" to receive data.
+ *
+ * @name: "chunk" name.
+ * @size: full size of data to receive.
+ */
+void optimus_chunk_init(const char *name, unsigned int size);
+
+/**
+ * optimus_chunk_alloc_buf() - Alloc buffer inside "chunk" for data.
+ *
+ * @size: full size of data.
+ *
+ * @return: 0 on success, -errno otherwise.
+ */
+int optimus_chunk_alloc_buf(unsigned int size);
+
+/**
+ * optimus_chunk_process() - Pass new portion of data to "chunk".
+ *
+ * @data: new portion of data.
+ * @size: size of data.
+ *
+ * @return: new state of "chunk".
+ */
+enum optimus_chunk_state optimus_chunk_process(const void *data, unsigned int size);
+
+#endif /* __OPTIMUS_DOWNLOAD_H__ */
-- 
2.30.1



More information about the U-Boot mailing list