blob: 3d5ff951eb63a03b8630d8dcf43f06db4288e7d3 [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{
Patrick Delaunayf82f9e42022-12-15 10:15:50 +010022 int err;
23
Simon Glassd911f3b2023-02-05 15:44:20 -070024 if (!IS_ENABLED(CONFIG_UDP_FUNCTION_FASTBOOT)) {
Patrick Delaunayf82f9e42022-12-15 10:15:50 +010025 pr_err("Fastboot UDP not enabled\n");
26 return CMD_RET_FAILURE;
27 }
28
Dmitrii Merkurev308252d2023-04-12 19:49:30 +010029 err = net_loop(FASTBOOT_UDP);
Alex Kiernand5aa57c2018-05-29 15:30:53 +000030
31 if (err < 0) {
32 printf("fastboot udp error: %d\n", err);
33 return CMD_RET_FAILURE;
34 }
35
36 return CMD_RET_SUCCESS;
Alex Kiernand5aa57c2018-05-29 15:30:53 +000037}
38
Dmitrii Merkurev308252d2023-04-12 19:49:30 +010039static int do_fastboot_tcp(int argc, char *const argv[],
40 uintptr_t buf_addr, size_t buf_size)
41{
42 int err;
43
44 if (!IS_ENABLED(CONFIG_TCP_FUNCTION_FASTBOOT)) {
45 pr_err("Fastboot TCP not enabled\n");
46 return CMD_RET_FAILURE;
47 }
48
49 err = net_loop(FASTBOOT_TCP);
50
51 if (err < 0) {
52 printf("fastboot tcp error: %d\n", err);
53 return CMD_RET_FAILURE;
54 }
55
56 return CMD_RET_SUCCESS;
57}
58
Alex Kiernand5aa57c2018-05-29 15:30:53 +000059static int do_fastboot_usb(int argc, char *const argv[],
60 uintptr_t buf_addr, size_t buf_size)
61{
Paul Kocialkowskid48378f2015-06-12 19:57:00 +020062 int controller_index;
63 char *usb_controller;
Sam Protsenko29f57342018-06-29 21:59:40 +030064 char *endp;
Sebastian Siewior9d4471e2014-05-05 15:08:10 -050065 int ret;
66
Simon Glassf597ebe2023-02-05 15:44:23 -070067 if (!IS_ENABLED(CONFIG_USB_FUNCTION_FASTBOOT)) {
Patrick Delaunayf82f9e42022-12-15 10:15:50 +010068 pr_err("Fastboot USB not enabled\n");
69 return CMD_RET_FAILURE;
70 }
71
Paul Kocialkowskid48378f2015-06-12 19:57:00 +020072 if (argc < 2)
73 return CMD_RET_USAGE;
74
75 usb_controller = argv[1];
Sam Protsenko29f57342018-06-29 21:59:40 +030076 controller_index = simple_strtoul(usb_controller, &endp, 0);
77 if (*endp != '\0') {
78 pr_err("Error: Wrong USB controller index format\n");
79 return CMD_RET_FAILURE;
80 }
Paul Kocialkowskid48378f2015-06-12 19:57:00 +020081
Jean-Jacques Hiblot585e1992018-11-29 10:52:41 +010082 ret = usb_gadget_initialize(controller_index);
Paul Kocialkowskid48378f2015-06-12 19:57:00 +020083 if (ret) {
Seung-Woo Kim591188e2018-06-04 16:04:51 +090084 pr_err("USB init failed: %d\n", ret);
Paul Kocialkowskid48378f2015-06-12 19:57:00 +020085 return CMD_RET_FAILURE;
86 }
87
Rob Herring5aba3982014-12-10 14:43:04 -060088 g_dnl_clear_detach();
Sebastian Siewior9d4471e2014-05-05 15:08:10 -050089 ret = g_dnl_register("usb_dnl_fastboot");
90 if (ret)
91 return ret;
92
Rob Herringce9d8ee2015-01-26 15:14:41 -060093 if (!g_dnl_board_usb_cable_connected()) {
94 puts("\rUSB cable not detected.\n" \
95 "Command exit.\n");
Paul Kocialkowskid48378f2015-06-12 19:57:00 +020096 ret = CMD_RET_FAILURE;
97 goto exit;
Rob Herringce9d8ee2015-01-26 15:14:41 -060098 }
99
Sebastian Siewior9d4471e2014-05-05 15:08:10 -0500100 while (1) {
Rob Herring5aba3982014-12-10 14:43:04 -0600101 if (g_dnl_detach())
102 break;
Sebastian Siewior9d4471e2014-05-05 15:08:10 -0500103 if (ctrlc())
104 break;
Stefan Roese80877fa2022-09-02 14:10:46 +0200105 schedule();
Paul Kocialkowski6b9f7282015-05-24 11:41:04 +0200106 usb_gadget_handle_interrupts(controller_index);
Sebastian Siewior9d4471e2014-05-05 15:08:10 -0500107 }
108
Paul Kocialkowskid48378f2015-06-12 19:57:00 +0200109 ret = CMD_RET_SUCCESS;
110
111exit:
Mattijs Korpershoeke9621d82022-10-07 11:38:22 +0200112 usb_gadget_release(controller_index);
Sebastian Siewior9d4471e2014-05-05 15:08:10 -0500113 g_dnl_unregister();
Rob Herring5aba3982014-12-10 14:43:04 -0600114 g_dnl_clear_detach();
Paul Kocialkowskid48378f2015-06-12 19:57:00 +0200115
116 return ret;
Alex Kiernand5aa57c2018-05-29 15:30:53 +0000117}
118
Simon Glassed38aef2020-05-10 11:40:03 -0600119static int do_fastboot(struct cmd_tbl *cmdtp, int flag, int argc,
120 char *const argv[])
Alex Kiernand5aa57c2018-05-29 15:30:53 +0000121{
122 uintptr_t buf_addr = (uintptr_t)NULL;
123 size_t buf_size = 0;
124
125 if (argc < 2)
126 return CMD_RET_USAGE;
127
128 while (argc > 1 && **(argv + 1) == '-') {
129 char *arg = *++argv;
130
131 --argc;
132 while (*++arg) {
133 switch (*arg) {
134 case 'l':
135 if (--argc <= 0)
136 return CMD_RET_USAGE;
Simon Glass3ff49ec2021-07-24 09:03:29 -0600137 buf_addr = hextoul(*++argv, NULL);
Alex Kiernand5aa57c2018-05-29 15:30:53 +0000138 goto NXTARG;
139
140 case 's':
141 if (--argc <= 0)
142 return CMD_RET_USAGE;
Simon Glass3ff49ec2021-07-24 09:03:29 -0600143 buf_size = hextoul(*++argv, NULL);
Alex Kiernand5aa57c2018-05-29 15:30:53 +0000144 goto NXTARG;
145
146 default:
147 return CMD_RET_USAGE;
148 }
149 }
150NXTARG:
151 ;
152 }
153
Sam Protsenko29f57342018-06-29 21:59:40 +0300154 /* Handle case when USB controller param is just '-' */
155 if (argc == 1) {
156 pr_err("Error: Incorrect USB controller index\n");
157 return CMD_RET_USAGE;
158 }
159
Alex Kiernand5aa57c2018-05-29 15:30:53 +0000160 fastboot_init((void *)buf_addr, buf_size);
161
162 if (!strcmp(argv[1], "udp"))
163 return do_fastboot_udp(argc, argv, buf_addr, buf_size);
Dmitrii Merkurev308252d2023-04-12 19:49:30 +0100164 if (!strcmp(argv[1], "tcp"))
165 return do_fastboot_tcp(argc, argv, buf_addr, buf_size);
Alex Kiernand5aa57c2018-05-29 15:30:53 +0000166 if (!strcmp(argv[1], "usb")) {
167 argv++;
168 argc--;
169 }
170
171 return do_fastboot_usb(argc, argv, buf_addr, buf_size);
Sebastian Siewior9d4471e2014-05-05 15:08:10 -0500172}
173
Patrick Delaunayf82f9e42022-12-15 10:15:50 +0100174U_BOOT_CMD(
175 fastboot, CONFIG_SYS_MAXARGS, 1, do_fastboot,
176 "run as a fastboot usb or udp device",
Alex Kiernand5aa57c2018-05-29 15:30:53 +0000177 "[-l addr] [-s size] usb <controller> | udp\n"
178 "\taddr - address of buffer used during data transfers ("
179 __stringify(CONFIG_FASTBOOT_BUF_ADDR) ")\n"
180 "\tsize - size of buffer used during data transfers ("
181 __stringify(CONFIG_FASTBOOT_BUF_SIZE) ")"
Sebastian Siewior9d4471e2014-05-05 15:08:10 -0500182);