blob: b94dbd548843835ee53bc1b40410291999aa74f8 [file] [log] [blame]
Tom Rini83d290c2018-05-06 17:58:06 -04001// SPDX-License-Identifier: GPL-2.0+
Sebastian Siewior3aab70a2014-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 Siewior3aab70a2014-05-05 15:08:10 -05008 */
9#include <common.h>
10#include <command.h>
Simon Glass24b852a2015-11-08 23:47:45 -070011#include <console.h>
Sebastian Siewior3aab70a2014-05-05 15:08:10 -050012#include <g_dnl.h>
Alex Kiernanf73a7df2018-05-29 15:30:53 +000013#include <fastboot.h>
14#include <net.h>
Paul Kocialkowski8d2f0032015-06-12 19:57:00 +020015#include <usb.h>
Sean Nyekjaer731785d2019-01-24 15:17:21 +010016#include <watchdog.h>
Simon Glass1af3c7f2020-05-10 11:40:09 -060017#include <linux/stringify.h>
Sebastian Siewior3aab70a2014-05-05 15:08:10 -050018
Alex Kiernanf73a7df2018-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 Siewior3aab70a2014-05-05 15:08:10 -050021{
Patrick Delaunayd0379902022-12-15 10:15:50 +010022 int err;
23
24 if (!CONFIG_IS_ENABLED(UDP_FUNCTION_FASTBOOT)) {
25 pr_err("Fastboot UDP not enabled\n");
26 return CMD_RET_FAILURE;
27 }
28
29 err = net_loop(FASTBOOT);
Alex Kiernanf73a7df2018-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 Kiernanf73a7df2018-05-29 15:30:53 +000037}
38
39static int do_fastboot_usb(int argc, char *const argv[],
40 uintptr_t buf_addr, size_t buf_size)
41{
Paul Kocialkowski8d2f0032015-06-12 19:57:00 +020042 int controller_index;
43 char *usb_controller;
Sam Protsenkoaa515792018-06-29 21:59:40 +030044 char *endp;
Sebastian Siewior3aab70a2014-05-05 15:08:10 -050045 int ret;
46
Patrick Delaunayd0379902022-12-15 10:15:50 +010047 if (!CONFIG_IS_ENABLED(USB_FUNCTION_FASTBOOT)) {
48 pr_err("Fastboot USB not enabled\n");
49 return CMD_RET_FAILURE;
50 }
51
Paul Kocialkowski8d2f0032015-06-12 19:57:00 +020052 if (argc < 2)
53 return CMD_RET_USAGE;
54
55 usb_controller = argv[1];
Sam Protsenkoaa515792018-06-29 21:59:40 +030056 controller_index = simple_strtoul(usb_controller, &endp, 0);
57 if (*endp != '\0') {
58 pr_err("Error: Wrong USB controller index format\n");
59 return CMD_RET_FAILURE;
60 }
Paul Kocialkowski8d2f0032015-06-12 19:57:00 +020061
Jean-Jacques Hiblota06955a2018-11-29 10:52:41 +010062 ret = usb_gadget_initialize(controller_index);
Paul Kocialkowski8d2f0032015-06-12 19:57:00 +020063 if (ret) {
Seung-Woo Kim71002b52018-06-04 16:04:51 +090064 pr_err("USB init failed: %d\n", ret);
Paul Kocialkowski8d2f0032015-06-12 19:57:00 +020065 return CMD_RET_FAILURE;
66 }
67
Rob Herring267abc62014-12-10 14:43:04 -060068 g_dnl_clear_detach();
Sebastian Siewior3aab70a2014-05-05 15:08:10 -050069 ret = g_dnl_register("usb_dnl_fastboot");
70 if (ret)
71 return ret;
72
Rob Herring7c23bcb2015-01-26 15:14:41 -060073 if (!g_dnl_board_usb_cable_connected()) {
74 puts("\rUSB cable not detected.\n" \
75 "Command exit.\n");
Paul Kocialkowski8d2f0032015-06-12 19:57:00 +020076 ret = CMD_RET_FAILURE;
77 goto exit;
Rob Herring7c23bcb2015-01-26 15:14:41 -060078 }
79
Sebastian Siewior3aab70a2014-05-05 15:08:10 -050080 while (1) {
Rob Herring267abc62014-12-10 14:43:04 -060081 if (g_dnl_detach())
82 break;
Sebastian Siewior3aab70a2014-05-05 15:08:10 -050083 if (ctrlc())
84 break;
Stefan Roese29caf932022-09-02 14:10:46 +020085 schedule();
Paul Kocialkowskieca76b72015-05-24 11:41:04 +020086 usb_gadget_handle_interrupts(controller_index);
Sebastian Siewior3aab70a2014-05-05 15:08:10 -050087 }
88
Paul Kocialkowski8d2f0032015-06-12 19:57:00 +020089 ret = CMD_RET_SUCCESS;
90
91exit:
Mattijs Korpershoek5f7e01e2022-10-07 11:38:22 +020092 usb_gadget_release(controller_index);
Sebastian Siewior3aab70a2014-05-05 15:08:10 -050093 g_dnl_unregister();
Rob Herring267abc62014-12-10 14:43:04 -060094 g_dnl_clear_detach();
Paul Kocialkowski8d2f0032015-06-12 19:57:00 +020095
96 return ret;
Sebastian Siewior3aab70a2014-05-05 15:08:10 -050097}
98
Simon Glass09140112020-05-10 11:40:03 -060099static int do_fastboot(struct cmd_tbl *cmdtp, int flag, int argc,
100 char *const argv[])
Alex Kiernanf73a7df2018-05-29 15:30:53 +0000101{
102 uintptr_t buf_addr = (uintptr_t)NULL;
103 size_t buf_size = 0;
104
105 if (argc < 2)
106 return CMD_RET_USAGE;
107
108 while (argc > 1 && **(argv + 1) == '-') {
109 char *arg = *++argv;
110
111 --argc;
112 while (*++arg) {
113 switch (*arg) {
114 case 'l':
115 if (--argc <= 0)
116 return CMD_RET_USAGE;
Simon Glass7e5f4602021-07-24 09:03:29 -0600117 buf_addr = hextoul(*++argv, NULL);
Alex Kiernanf73a7df2018-05-29 15:30:53 +0000118 goto NXTARG;
119
120 case 's':
121 if (--argc <= 0)
122 return CMD_RET_USAGE;
Simon Glass7e5f4602021-07-24 09:03:29 -0600123 buf_size = hextoul(*++argv, NULL);
Alex Kiernanf73a7df2018-05-29 15:30:53 +0000124 goto NXTARG;
125
126 default:
127 return CMD_RET_USAGE;
128 }
129 }
130NXTARG:
131 ;
132 }
133
Sam Protsenkoaa515792018-06-29 21:59:40 +0300134 /* Handle case when USB controller param is just '-' */
135 if (argc == 1) {
136 pr_err("Error: Incorrect USB controller index\n");
137 return CMD_RET_USAGE;
138 }
139
Alex Kiernanf73a7df2018-05-29 15:30:53 +0000140 fastboot_init((void *)buf_addr, buf_size);
141
142 if (!strcmp(argv[1], "udp"))
143 return do_fastboot_udp(argc, argv, buf_addr, buf_size);
144
145 if (!strcmp(argv[1], "usb")) {
146 argv++;
147 argc--;
148 }
149
150 return do_fastboot_usb(argc, argv, buf_addr, buf_size);
151}
152
Patrick Delaunayd0379902022-12-15 10:15:50 +0100153U_BOOT_CMD(
154 fastboot, CONFIG_SYS_MAXARGS, 1, do_fastboot,
155 "run as a fastboot usb or udp device",
Alex Kiernanf73a7df2018-05-29 15:30:53 +0000156 "[-l addr] [-s size] usb <controller> | udp\n"
157 "\taddr - address of buffer used during data transfers ("
158 __stringify(CONFIG_FASTBOOT_BUF_ADDR) ")\n"
159 "\tsize - size of buffer used during data transfers ("
160 __stringify(CONFIG_FASTBOOT_BUF_SIZE) ")"
Sebastian Siewior3aab70a2014-05-05 15:08:10 -0500161);