blob: 0be83b78ac130fda28873596fcee64274913f516 [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>
Sebastian Siewior9d4471e2014-05-05 15:08:10 -050016
Alex Kiernand5aa57c2018-05-29 15:30:53 +000017static int do_fastboot_udp(int argc, char *const argv[],
18 uintptr_t buf_addr, size_t buf_size)
Sebastian Siewior9d4471e2014-05-05 15:08:10 -050019{
Alex Kiernand5aa57c2018-05-29 15:30:53 +000020#if CONFIG_IS_ENABLED(UDP_FUNCTION_FASTBOOT)
21 int err = net_loop(FASTBOOT);
22
23 if (err < 0) {
24 printf("fastboot udp error: %d\n", err);
25 return CMD_RET_FAILURE;
26 }
27
28 return CMD_RET_SUCCESS;
29#else
30 pr_err("Fastboot UDP not enabled\n");
31 return CMD_RET_FAILURE;
32#endif
33}
34
35static int do_fastboot_usb(int argc, char *const argv[],
36 uintptr_t buf_addr, size_t buf_size)
37{
38#if CONFIG_IS_ENABLED(USB_FUNCTION_FASTBOOT)
Paul Kocialkowskid48378f2015-06-12 19:57:00 +020039 int controller_index;
40 char *usb_controller;
Sam Protsenko29f57342018-06-29 21:59:40 +030041 char *endp;
Sebastian Siewior9d4471e2014-05-05 15:08:10 -050042 int ret;
43
Paul Kocialkowskid48378f2015-06-12 19:57:00 +020044 if (argc < 2)
45 return CMD_RET_USAGE;
46
47 usb_controller = argv[1];
Sam Protsenko29f57342018-06-29 21:59:40 +030048 controller_index = simple_strtoul(usb_controller, &endp, 0);
49 if (*endp != '\0') {
50 pr_err("Error: Wrong USB controller index format\n");
51 return CMD_RET_FAILURE;
52 }
Paul Kocialkowskid48378f2015-06-12 19:57:00 +020053
Jean-Jacques Hiblot585e1992018-11-29 10:52:41 +010054 ret = usb_gadget_initialize(controller_index);
Paul Kocialkowskid48378f2015-06-12 19:57:00 +020055 if (ret) {
Seung-Woo Kim591188e2018-06-04 16:04:51 +090056 pr_err("USB init failed: %d\n", ret);
Paul Kocialkowskid48378f2015-06-12 19:57:00 +020057 return CMD_RET_FAILURE;
58 }
59
Rob Herring5aba3982014-12-10 14:43:04 -060060 g_dnl_clear_detach();
Sebastian Siewior9d4471e2014-05-05 15:08:10 -050061 ret = g_dnl_register("usb_dnl_fastboot");
62 if (ret)
63 return ret;
64
Rob Herringce9d8ee2015-01-26 15:14:41 -060065 if (!g_dnl_board_usb_cable_connected()) {
66 puts("\rUSB cable not detected.\n" \
67 "Command exit.\n");
Paul Kocialkowskid48378f2015-06-12 19:57:00 +020068 ret = CMD_RET_FAILURE;
69 goto exit;
Rob Herringce9d8ee2015-01-26 15:14:41 -060070 }
71
Sebastian Siewior9d4471e2014-05-05 15:08:10 -050072 while (1) {
Rob Herring5aba3982014-12-10 14:43:04 -060073 if (g_dnl_detach())
74 break;
Sebastian Siewior9d4471e2014-05-05 15:08:10 -050075 if (ctrlc())
76 break;
Paul Kocialkowski6b9f7282015-05-24 11:41:04 +020077 usb_gadget_handle_interrupts(controller_index);
Sebastian Siewior9d4471e2014-05-05 15:08:10 -050078 }
79
Paul Kocialkowskid48378f2015-06-12 19:57:00 +020080 ret = CMD_RET_SUCCESS;
81
82exit:
Sebastian Siewior9d4471e2014-05-05 15:08:10 -050083 g_dnl_unregister();
Rob Herring5aba3982014-12-10 14:43:04 -060084 g_dnl_clear_detach();
Jean-Jacques Hiblot585e1992018-11-29 10:52:41 +010085 usb_gadget_release(controller_index);
Paul Kocialkowskid48378f2015-06-12 19:57:00 +020086
87 return ret;
Alex Kiernand5aa57c2018-05-29 15:30:53 +000088#else
89 pr_err("Fastboot USB not enabled\n");
90 return CMD_RET_FAILURE;
91#endif
92}
93
94static int do_fastboot(cmd_tbl_t *cmdtp, int flag, int argc, char *const argv[])
95{
96 uintptr_t buf_addr = (uintptr_t)NULL;
97 size_t buf_size = 0;
98
99 if (argc < 2)
100 return CMD_RET_USAGE;
101
102 while (argc > 1 && **(argv + 1) == '-') {
103 char *arg = *++argv;
104
105 --argc;
106 while (*++arg) {
107 switch (*arg) {
108 case 'l':
109 if (--argc <= 0)
110 return CMD_RET_USAGE;
111 buf_addr = simple_strtoul(*++argv, NULL, 16);
112 goto NXTARG;
113
114 case 's':
115 if (--argc <= 0)
116 return CMD_RET_USAGE;
117 buf_size = simple_strtoul(*++argv, NULL, 16);
118 goto NXTARG;
119
120 default:
121 return CMD_RET_USAGE;
122 }
123 }
124NXTARG:
125 ;
126 }
127
Sam Protsenko29f57342018-06-29 21:59:40 +0300128 /* Handle case when USB controller param is just '-' */
129 if (argc == 1) {
130 pr_err("Error: Incorrect USB controller index\n");
131 return CMD_RET_USAGE;
132 }
133
Alex Kiernand5aa57c2018-05-29 15:30:53 +0000134 fastboot_init((void *)buf_addr, buf_size);
135
136 if (!strcmp(argv[1], "udp"))
137 return do_fastboot_udp(argc, argv, buf_addr, buf_size);
138
139 if (!strcmp(argv[1], "usb")) {
140 argv++;
141 argc--;
142 }
143
144 return do_fastboot_usb(argc, argv, buf_addr, buf_size);
Sebastian Siewior9d4471e2014-05-05 15:08:10 -0500145}
146
Alex Kiernand5aa57c2018-05-29 15:30:53 +0000147#ifdef CONFIG_SYS_LONGHELP
148static char fastboot_help_text[] =
149 "[-l addr] [-s size] usb <controller> | udp\n"
150 "\taddr - address of buffer used during data transfers ("
151 __stringify(CONFIG_FASTBOOT_BUF_ADDR) ")\n"
152 "\tsize - size of buffer used during data transfers ("
153 __stringify(CONFIG_FASTBOOT_BUF_SIZE) ")"
154 ;
155#endif
156
Sebastian Siewior9d4471e2014-05-05 15:08:10 -0500157U_BOOT_CMD(
Alex Kiernand5aa57c2018-05-29 15:30:53 +0000158 fastboot, CONFIG_SYS_MAXARGS, 1, do_fastboot,
159 "run as a fastboot usb or udp device", fastboot_help_text
Sebastian Siewior9d4471e2014-05-05 15:08:10 -0500160);