blob: ad5c06491183a4ac30a5b4bc4a49bca8384651ec [file] [log] [blame]
Tom Rini10e47792018-05-06 17:58:06 -04001// SPDX-License-Identifier: GPL-2.0+
Sebastian Siewior9d4471e2014-05-05 15:08:10 -05002/*
3 * Copyright 2008 - 2009 Windriver, <www.windriver.com>
4 * Author: Tom Rix <Tom.Rix@windriver.com>
5 *
6 * (C) Copyright 2014 Linaro, Ltd.
7 * Rob Herring <robh@kernel.org>
Sebastian Siewior9d4471e2014-05-05 15:08:10 -05008 */
9#include <common.h>
10#include <command.h>
Simon Glassa73bda42015-11-08 23:47:45 -070011#include <console.h>
Sebastian Siewior9d4471e2014-05-05 15:08:10 -050012#include <g_dnl.h>
Alex Kiernand5aa57c2018-05-29 15:30:53 +000013#include <fastboot.h>
14#include <net.h>
Paul Kocialkowskid48378f2015-06-12 19:57:00 +020015#include <usb.h>
Sean Nyekjaer674f6622019-01-24 15:17:21 +010016#include <watchdog.h>
Simon Glassfb64e362020-05-10 11:40:09 -060017#include <linux/stringify.h>
Sebastian Siewior9d4471e2014-05-05 15:08:10 -050018
Alex Kiernand5aa57c2018-05-29 15:30:53 +000019static int do_fastboot_udp(int argc, char *const argv[],
20 uintptr_t buf_addr, size_t buf_size)
Sebastian Siewior9d4471e2014-05-05 15:08:10 -050021{
Alex Kiernand5aa57c2018-05-29 15:30:53 +000022#if CONFIG_IS_ENABLED(UDP_FUNCTION_FASTBOOT)
23 int err = net_loop(FASTBOOT);
24
25 if (err < 0) {
26 printf("fastboot udp error: %d\n", err);
27 return CMD_RET_FAILURE;
28 }
29
30 return CMD_RET_SUCCESS;
31#else
32 pr_err("Fastboot UDP not enabled\n");
33 return CMD_RET_FAILURE;
34#endif
35}
36
37static int do_fastboot_usb(int argc, char *const argv[],
38 uintptr_t buf_addr, size_t buf_size)
39{
40#if CONFIG_IS_ENABLED(USB_FUNCTION_FASTBOOT)
Paul Kocialkowskid48378f2015-06-12 19:57:00 +020041 int controller_index;
42 char *usb_controller;
Sam Protsenko29f57342018-06-29 21:59:40 +030043 char *endp;
Sebastian Siewior9d4471e2014-05-05 15:08:10 -050044 int ret;
45
Paul Kocialkowskid48378f2015-06-12 19:57:00 +020046 if (argc < 2)
47 return CMD_RET_USAGE;
48
49 usb_controller = argv[1];
Sam Protsenko29f57342018-06-29 21:59:40 +030050 controller_index = simple_strtoul(usb_controller, &endp, 0);
51 if (*endp != '\0') {
52 pr_err("Error: Wrong USB controller index format\n");
53 return CMD_RET_FAILURE;
54 }
Paul Kocialkowskid48378f2015-06-12 19:57:00 +020055
Jean-Jacques Hiblot585e1992018-11-29 10:52:41 +010056 ret = usb_gadget_initialize(controller_index);
Paul Kocialkowskid48378f2015-06-12 19:57:00 +020057 if (ret) {
Seung-Woo Kim591188e2018-06-04 16:04:51 +090058 pr_err("USB init failed: %d\n", ret);
Paul Kocialkowskid48378f2015-06-12 19:57:00 +020059 return CMD_RET_FAILURE;
60 }
61
Rob Herring5aba3982014-12-10 14:43:04 -060062 g_dnl_clear_detach();
Sebastian Siewior9d4471e2014-05-05 15:08:10 -050063 ret = g_dnl_register("usb_dnl_fastboot");
64 if (ret)
65 return ret;
66
Rob Herringce9d8ee2015-01-26 15:14:41 -060067 if (!g_dnl_board_usb_cable_connected()) {
68 puts("\rUSB cable not detected.\n" \
69 "Command exit.\n");
Paul Kocialkowskid48378f2015-06-12 19:57:00 +020070 ret = CMD_RET_FAILURE;
71 goto exit;
Rob Herringce9d8ee2015-01-26 15:14:41 -060072 }
73
Sebastian Siewior9d4471e2014-05-05 15:08:10 -050074 while (1) {
Rob Herring5aba3982014-12-10 14:43:04 -060075 if (g_dnl_detach())
76 break;
Sebastian Siewior9d4471e2014-05-05 15:08:10 -050077 if (ctrlc())
78 break;
Sean Nyekjaer674f6622019-01-24 15:17:21 +010079 WATCHDOG_RESET();
Paul Kocialkowski6b9f7282015-05-24 11:41:04 +020080 usb_gadget_handle_interrupts(controller_index);
Sebastian Siewior9d4471e2014-05-05 15:08:10 -050081 }
82
Paul Kocialkowskid48378f2015-06-12 19:57:00 +020083 ret = CMD_RET_SUCCESS;
84
85exit:
Sebastian Siewior9d4471e2014-05-05 15:08:10 -050086 g_dnl_unregister();
Rob Herring5aba3982014-12-10 14:43:04 -060087 g_dnl_clear_detach();
Jean-Jacques Hiblot585e1992018-11-29 10:52:41 +010088 usb_gadget_release(controller_index);
Paul Kocialkowskid48378f2015-06-12 19:57:00 +020089
90 return ret;
Alex Kiernand5aa57c2018-05-29 15:30:53 +000091#else
92 pr_err("Fastboot USB not enabled\n");
93 return CMD_RET_FAILURE;
94#endif
95}
96
Simon Glassed38aef2020-05-10 11:40:03 -060097static int do_fastboot(struct cmd_tbl *cmdtp, int flag, int argc,
98 char *const argv[])
Alex Kiernand5aa57c2018-05-29 15:30:53 +000099{
100 uintptr_t buf_addr = (uintptr_t)NULL;
101 size_t buf_size = 0;
102
103 if (argc < 2)
104 return CMD_RET_USAGE;
105
106 while (argc > 1 && **(argv + 1) == '-') {
107 char *arg = *++argv;
108
109 --argc;
110 while (*++arg) {
111 switch (*arg) {
112 case 'l':
113 if (--argc <= 0)
114 return CMD_RET_USAGE;
115 buf_addr = simple_strtoul(*++argv, NULL, 16);
116 goto NXTARG;
117
118 case 's':
119 if (--argc <= 0)
120 return CMD_RET_USAGE;
121 buf_size = simple_strtoul(*++argv, NULL, 16);
122 goto NXTARG;
123
124 default:
125 return CMD_RET_USAGE;
126 }
127 }
128NXTARG:
129 ;
130 }
131
Sam Protsenko29f57342018-06-29 21:59:40 +0300132 /* Handle case when USB controller param is just '-' */
133 if (argc == 1) {
134 pr_err("Error: Incorrect USB controller index\n");
135 return CMD_RET_USAGE;
136 }
137
Alex Kiernand5aa57c2018-05-29 15:30:53 +0000138 fastboot_init((void *)buf_addr, buf_size);
139
140 if (!strcmp(argv[1], "udp"))
141 return do_fastboot_udp(argc, argv, buf_addr, buf_size);
142
143 if (!strcmp(argv[1], "usb")) {
144 argv++;
145 argc--;
146 }
147
148 return do_fastboot_usb(argc, argv, buf_addr, buf_size);
Sebastian Siewior9d4471e2014-05-05 15:08:10 -0500149}
150
Alex Kiernand5aa57c2018-05-29 15:30:53 +0000151#ifdef CONFIG_SYS_LONGHELP
152static char fastboot_help_text[] =
153 "[-l addr] [-s size] usb <controller> | udp\n"
154 "\taddr - address of buffer used during data transfers ("
155 __stringify(CONFIG_FASTBOOT_BUF_ADDR) ")\n"
156 "\tsize - size of buffer used during data transfers ("
157 __stringify(CONFIG_FASTBOOT_BUF_SIZE) ")"
158 ;
159#endif
160
Sebastian Siewior9d4471e2014-05-05 15:08:10 -0500161U_BOOT_CMD(
Alex Kiernand5aa57c2018-05-29 15:30:53 +0000162 fastboot, CONFIG_SYS_MAXARGS, 1, do_fastboot,
163 "run as a fastboot usb or udp device", fastboot_help_text
Sebastian Siewior9d4471e2014-05-05 15:08:10 -0500164);