2022-11-25 21:16:25 +01:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2019-2022 Mesotic SAS
|
|
|
|
*
|
|
|
|
* This file is subject to the terms and conditions of the GNU Lesser
|
|
|
|
* General Public License v2.1. See the file LICENSE in the top level
|
|
|
|
* directory for more details.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @ingroup tests
|
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file
|
|
|
|
* @brief Test application for demonstrating USBUS MSC implementation
|
|
|
|
*
|
|
|
|
* @author Dylan Laduranty <dylan.laduranty@mesotic.com>
|
|
|
|
*
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <errno.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <string.h>
|
|
|
|
|
2023-04-04 23:49:27 +02:00
|
|
|
#include "board.h"
|
|
|
|
|
|
|
|
#if defined(MODULE_MTD_EMULATED)
|
|
|
|
|
|
|
|
#include "mtd_emulated.h"
|
|
|
|
|
|
|
|
/* The following parameters are only suitable for testing the basic functions
|
|
|
|
* of USB MSC. To create a partition with a FAT file system, at least 64 sectors
|
|
|
|
* have to be used. */
|
|
|
|
|
|
|
|
#ifndef SECTOR_COUNT
|
|
|
|
#define SECTOR_COUNT 64
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifndef PAGES_PER_SECTOR
|
|
|
|
#define PAGES_PER_SECTOR 4
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifndef PAGE_SIZE
|
|
|
|
#define PAGE_SIZE 128
|
|
|
|
#endif
|
|
|
|
|
|
|
|
MTD_EMULATED_DEV(0, SECTOR_COUNT, PAGES_PER_SECTOR, PAGE_SIZE);
|
|
|
|
|
|
|
|
#endif /* MODULE_MTD_EMULATED */
|
|
|
|
|
2023-11-10 19:24:18 +01:00
|
|
|
#include "mtd.h"
|
2022-11-25 21:16:25 +01:00
|
|
|
#include "shell.h"
|
|
|
|
#include "usb/usbus.h"
|
|
|
|
#include "usb/usbus/msc.h"
|
|
|
|
#include "ztimer.h"
|
|
|
|
|
2023-03-07 00:47:01 +01:00
|
|
|
static usbus_t *usbus;
|
2022-11-25 21:16:25 +01:00
|
|
|
|
|
|
|
static int _cmd_add_lun(int argc, char **argv)
|
|
|
|
{
|
|
|
|
int dev, ret;
|
|
|
|
mtd_dev_t *mtd_dev;
|
|
|
|
|
|
|
|
if (argc < 2) {
|
|
|
|
printf("usage: %s <mtd dev>\n", argv[0]);
|
|
|
|
puts("\tMTD devices available:");
|
|
|
|
for (int i = 0; i < (int)MTD_NUMOF; i++) {
|
|
|
|
printf("\t\t%i: MTD_DEV(%i)\n", i, i);
|
|
|
|
}
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* parse the given MTD device */
|
|
|
|
dev = atol(argv[1]);
|
|
|
|
if (dev < 0 || dev >= (int)MTD_NUMOF) {
|
|
|
|
puts("error: invalid MTD device specified");
|
|
|
|
return -2;
|
|
|
|
}
|
2023-11-10 19:24:18 +01:00
|
|
|
mtd_dev = mtd_dev_get(dev);
|
2023-03-07 00:47:01 +01:00
|
|
|
ret = usbus_msc_add_lun(usbus, mtd_dev);
|
2022-11-25 21:16:25 +01:00
|
|
|
if (ret != 0) {
|
|
|
|
printf("Cannot add LUN device (error:%d %s)\n", ret, strerror(-ret));
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int _cmd_remove_lun(int argc, char **argv)
|
|
|
|
{
|
|
|
|
int dev, ret;
|
|
|
|
mtd_dev_t *mtd_dev;
|
|
|
|
|
|
|
|
if (argc < 2) {
|
|
|
|
printf("usage: %s <mtd dev>\n", argv[0]);
|
|
|
|
puts("\tMTD devices available:");
|
|
|
|
for (int i = 0; i < (int)MTD_NUMOF; i++) {
|
|
|
|
printf("\t\t%i: MTD_DEV(%i)\n", i, i);
|
|
|
|
}
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* parse the given MTD device */
|
|
|
|
dev = atol(argv[1]);
|
|
|
|
if (dev < 0 || dev >= (int)MTD_NUMOF) {
|
|
|
|
puts("error: invalid MTD device specified");
|
|
|
|
return -2;
|
|
|
|
}
|
2023-11-10 19:24:18 +01:00
|
|
|
mtd_dev = mtd_dev_get(dev);
|
2023-03-07 00:47:01 +01:00
|
|
|
ret = usbus_msc_remove_lun(usbus, mtd_dev);
|
2022-11-25 21:16:25 +01:00
|
|
|
if (ret == -EAGAIN) {
|
|
|
|
printf("MTD device was not registered\n");
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int _cmd_usb_attach(int argc, char **argv)
|
|
|
|
{
|
|
|
|
(void)argc;
|
|
|
|
(void)argv;
|
|
|
|
static const usbopt_enable_t _enable = USBOPT_ENABLE;
|
|
|
|
|
2023-03-07 00:47:01 +01:00
|
|
|
usbdev_set(usbus->dev, USBOPT_ATTACH, &_enable,
|
2022-11-25 21:16:25 +01:00
|
|
|
sizeof(usbopt_enable_t));
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int _cmd_usb_detach(int argc, char **argv)
|
|
|
|
{
|
|
|
|
(void)argc;
|
|
|
|
(void)argv;
|
|
|
|
static const usbopt_enable_t _enable = USBOPT_DISABLE;
|
|
|
|
|
2023-03-07 00:47:01 +01:00
|
|
|
usbdev_set(usbus->dev, USBOPT_ATTACH, &_enable,
|
2022-11-25 21:16:25 +01:00
|
|
|
sizeof(usbopt_enable_t));
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int _cmd_usb_reset(int argc, char **argv)
|
|
|
|
{
|
|
|
|
_cmd_usb_detach(argc, argv);
|
|
|
|
ztimer_sleep(ZTIMER_MSEC, 100);
|
|
|
|
_cmd_usb_attach(argc, argv);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static const shell_command_t shell_commands[] = {
|
|
|
|
{ "add_lun", "Add a MTD device as new LUN", _cmd_add_lun },
|
|
|
|
{ "remove_lun", "Remove existing LUN", _cmd_remove_lun },
|
|
|
|
{ "usb_attach", "Attach USB to host", _cmd_usb_attach },
|
|
|
|
{ "usb_detach", "Detach USB from host", _cmd_usb_detach },
|
|
|
|
{ "usb_reset", "Combine Detach and Attach with a 100ms delay", _cmd_usb_reset },
|
|
|
|
{ NULL, NULL, NULL }
|
|
|
|
};
|
|
|
|
|
|
|
|
int main(void)
|
|
|
|
{
|
|
|
|
puts("RIOT USB MSC test application");
|
|
|
|
puts("Add one or more MTD devices as USB LUN");
|
|
|
|
puts("Then use the attach command to connect");
|
|
|
|
puts("your device and start USB operation");
|
|
|
|
|
|
|
|
/* Get driver context */
|
|
|
|
usbdev_t *usbdev = usbdev_get_ctx(0);
|
|
|
|
assert(usbdev);
|
|
|
|
|
2023-03-07 00:47:01 +01:00
|
|
|
usbus_t *usbus_auto_init_get(void);
|
|
|
|
usbus = usbus_auto_init_get();
|
2022-11-25 21:16:25 +01:00
|
|
|
|
|
|
|
/* start shell */
|
|
|
|
puts("All up, running the shell now");
|
|
|
|
char line_buf[SHELL_DEFAULT_BUFSIZE];
|
|
|
|
|
|
|
|
shell_run(shell_commands, line_buf, SHELL_DEFAULT_BUFSIZE);
|
|
|
|
|
|
|
|
/* should be never reached */
|
|
|
|
return 0;
|
|
|
|
}
|