blob: 9888ed4f17563face2a0233d343a236b834f809d [file] [log] [blame]
Lukasz Majewskib528f712013-03-05 12:10:17 +01001/*
2 * Copyright (C) 2011 Samsung Electronics
3 * Lukasz Majewski <l.majewski@samsung.com>
4 *
Wolfgang Denk1a459662013-07-08 09:37:19 +02005 * SPDX-License-Identifier: GPL-2.0+
Lukasz Majewskib528f712013-03-05 12:10:17 +01006 */
7
Przemyslaw Marczak351e9b22013-10-23 14:30:46 +02008#include <errno.h>
Lukasz Majewskib528f712013-03-05 12:10:17 +01009#include <common.h>
10#include <command.h>
11#include <g_dnl.h>
Stephen Warrenabfe8af2014-05-05 10:40:15 -060012#include <mmc.h>
13#include <part.h>
Mateusz Zalega16297cf2013-10-04 19:22:26 +020014#include <usb.h>
Lukasz Majewskib528f712013-03-05 12:10:17 +010015#include <usb_mass_storage.h>
16
Stephen Warrenabfe8af2014-05-05 10:40:15 -060017static int ums_read_sector(struct ums *ums_dev,
18 ulong start, lbaint_t blkcnt, void *buf)
19{
20 block_dev_desc_t *block_dev = ums_dev->block_dev;
21 lbaint_t blkstart = start + ums_dev->start_sector;
22 int dev_num = block_dev->dev;
23
24 return block_dev->block_read(dev_num, blkstart, blkcnt, buf);
25}
26
27static int ums_write_sector(struct ums *ums_dev,
28 ulong start, lbaint_t blkcnt, const void *buf)
29{
30 block_dev_desc_t *block_dev = ums_dev->block_dev;
31 lbaint_t blkstart = start + ums_dev->start_sector;
32 int dev_num = block_dev->dev;
33
34 return block_dev->block_write(dev_num, blkstart, blkcnt, buf);
35}
36
37static struct ums ums_dev = {
38 .read_sector = ums_read_sector,
39 .write_sector = ums_write_sector,
40 .name = "UMS disk",
41};
42
43struct ums *ums_init(unsigned int dev_num)
44{
45 struct mmc *mmc = NULL;
46
47 mmc = find_mmc_device(dev_num);
48 if (!mmc || mmc_init(mmc))
49 return NULL;
50
51 ums_dev.block_dev = &mmc->block_dev;
52 ums_dev.start_sector = 0;
53 ums_dev.num_sectors = mmc->capacity / SECTOR_SIZE;
54
55 printf("UMS: disk start sector: %#x, count: %#x\n",
56 ums_dev.start_sector, ums_dev.num_sectors);
57
58 return &ums_dev;
59}
60
Lukasz Majewskib528f712013-03-05 12:10:17 +010061int do_usb_mass_storage(cmd_tbl_t *cmdtp, int flag,
62 int argc, char * const argv[])
63{
Mateusz Zalega16297cf2013-10-04 19:22:26 +020064 if (argc < 3)
65 return CMD_RET_USAGE;
Lukasz Majewskib528f712013-03-05 12:10:17 +010066
Mateusz Zalega16297cf2013-10-04 19:22:26 +020067 const char *usb_controller = argv[1];
68 const char *mmc_devstring = argv[2];
Lukasz Majewskib528f712013-03-05 12:10:17 +010069
Przemyslaw Marczakf4dacf72013-10-23 14:30:43 +020070 unsigned int dev_num = simple_strtoul(mmc_devstring, NULL, 0);
71
72 struct ums *ums = ums_init(dev_num);
73 if (!ums)
74 return CMD_RET_FAILURE;
Lukasz Majewskib528f712013-03-05 12:10:17 +010075
Mateusz Zalega16297cf2013-10-04 19:22:26 +020076 unsigned int controller_index = (unsigned int)(simple_strtoul(
77 usb_controller, NULL, 0));
78 if (board_usb_init(controller_index, USB_INIT_DEVICE)) {
79 error("Couldn't init USB controller.");
Przemyslaw Marczak93c813b2013-10-23 14:30:42 +020080 return CMD_RET_FAILURE;
Mateusz Zalega16297cf2013-10-04 19:22:26 +020081 }
Lukasz Majewskib528f712013-03-05 12:10:17 +010082
Przemyslaw Marczak93c813b2013-10-23 14:30:42 +020083 int rc = fsg_init(ums);
Mateusz Zalega16297cf2013-10-04 19:22:26 +020084 if (rc) {
85 error("fsg_init failed");
Przemyslaw Marczak93c813b2013-10-23 14:30:42 +020086 return CMD_RET_FAILURE;
Mateusz Zalega16297cf2013-10-04 19:22:26 +020087 }
88
Stephen Warren66b88b02014-05-01 15:42:10 -060089 rc = g_dnl_register("usb_dnl_ums");
90 if (rc) {
91 error("g_dnl_register failed");
92 return CMD_RET_FAILURE;
93 }
Lukasz Majewskib528f712013-03-05 12:10:17 +010094
Przemyslaw Marczak3603e312014-01-07 15:08:37 +010095 /* Timeout unit: seconds */
96 int cable_ready_timeout = UMS_CABLE_READY_TIMEOUT;
97
Mateusz Zalega75504e92014-04-30 13:07:48 +020098 if (!g_dnl_board_usb_cable_connected()) {
99 /*
100 * Won't execute if we don't know whether the cable is
101 * connected.
102 */
Przemyslaw Marczak3603e312014-01-07 15:08:37 +0100103 puts("Please connect USB cable.\n");
104
Mateusz Zalega75504e92014-04-30 13:07:48 +0200105 while (!g_dnl_board_usb_cable_connected()) {
Przemyslaw Marczak3603e312014-01-07 15:08:37 +0100106 if (ctrlc()) {
107 puts("\rCTRL+C - Operation aborted.\n");
108 goto exit;
109 }
110 if (!cable_ready_timeout) {
111 puts("\rUSB cable not detected.\n" \
112 "Command exit.\n");
113 goto exit;
114 }
115
116 printf("\rAuto exit in: %.2d s.", cable_ready_timeout);
117 mdelay(1000);
118 cable_ready_timeout--;
119 }
120 puts("\r\n");
121 }
122
Lukasz Majewskib528f712013-03-05 12:10:17 +0100123 while (1) {
Przemyslaw Marczak351e9b22013-10-23 14:30:46 +0200124 usb_gadget_handle_interrupts();
125
126 rc = fsg_main_thread(NULL);
127 if (rc) {
128 /* Check I/O error */
129 if (rc == -EIO)
130 printf("\rCheck USB cable connection\n");
131
132 /* Check CTRL+C */
133 if (rc == -EPIPE)
134 printf("\rCTRL+C - Operation aborted\n");
135
Lukasz Majewskib528f712013-03-05 12:10:17 +0100136 goto exit;
137 }
Lukasz Majewskib528f712013-03-05 12:10:17 +0100138 }
139exit:
140 g_dnl_unregister();
Przemyslaw Marczak93c813b2013-10-23 14:30:42 +0200141 return CMD_RET_SUCCESS;
Lukasz Majewskib528f712013-03-05 12:10:17 +0100142}
143
144U_BOOT_CMD(ums, CONFIG_SYS_MAXARGS, 1, do_usb_mass_storage,
145 "Use the UMS [User Mass Storage]",
Przemyslaw Marczak93c813b2013-10-23 14:30:42 +0200146 "ums <USB_controller> <mmc_dev> e.g. ums 0 0"
Lukasz Majewskib528f712013-03-05 12:10:17 +0100147);