Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add support for MTP device class #3023

Open
wants to merge 1 commit into
base: master
Choose a base branch
from
Open

Conversation

roundby
Copy link

@roundby roundby commented Mar 12, 2025

Describe the PR
Add Media Transfer Protocol (MTP) device class support

Additional context
Feature request reference: #1079

The implementation has been tested and found functional at prototype level. We provide an example tested on STM32F3. It makes use of blulk and interrupt EPs only, so it should be supported by most platforms.

The class supports a single session only, but it allows multiple objects and nested associations (i.e. files and folders). We did not add support for object properties access. There are some other limitations but for the purpose of exposing file trees when mass storage class is not advised, the library is fully functional.

This implementation has been written from the ground up, with minimal requirements on buffers and keeping the code as simple as possible. Most of the buffer allocation for object I/O is demanded to the underlying implementation, as such, the RAM footprint of the library is limited.

The working mechanism of MTP (persistent 32 bit file handles) may not scale well on microcontrollers with constrained resources, depending on the application requirements. In the example, we just provide a fixed filesystem with files residing in RAM. It would be possible, depending on the application requirements, to build filesystem based I/O on top of this, by using the API provided in mtp_device.h.

The sources are available here

@roundby roundby marked this pull request as draft March 12, 2025 08:37
@nbmaurizio nbmaurizio force-pushed the mtp branch 4 times, most recently from b734659 to 8c0a674 Compare March 12, 2025 09:25
@roundby roundby marked this pull request as ready for review March 12, 2025 10:22
@roundby roundby changed the title Add support for MTP class device Add support for MTP device class Mar 13, 2025
MTP_DEVP_DEVICE_FRIENDLY_NAME,
};

tu_static const uint16_t mtp_object_properties_supported[] = {

Check notice

Code scanning / CodeQL

Unused static variable

Static variable mtp_object_properties_supported is never read.
Comment on lines 354 to 356
// wstring factory_def_value;
// wstring current_value_len;
// uint8_t form_flag;

Check notice

Code scanning / CodeQL

Commented-out code

This comment appears to contain commented-out code.
Comment on lines 240 to 287
switch (p_mtp->phase)
{
case MTP_PHASE_RESPONSE:
// IN transfer completed, prepare for a new command
TU_ASSERT(usbd_edpt_xfer(rhport, _mtpd_itf.ep_out, ((uint8_t *)(&_mtpd_gct)), CFG_MTP_EP_SIZE), 0);
p_mtp->phase = MTP_PHASE_IDLE;
break;
case MTP_PHASE_DATA_IN:
_mtpd_itf.xferred_len += xferred_bytes;
_mtpd_itf.handled_len = _mtpd_itf.xferred_len;

// Check if transfer completed
if (_mtpd_itf.xferred_len >= _mtpd_itf.total_len && (xferred_bytes == 0 || (xferred_bytes % CFG_MTP_EP_SIZE) != 0))
{
_mtpd_itf.phase = MTP_PHASE_RESPONSE;
_mtpd_gct.container_type = MTP_CONTAINER_TYPE_RESPONSE_BLOCK;
_mtpd_gct.code = MTP_RESC_OK;
_mtpd_gct.container_length = MTP_GENERIC_DATA_BLOCK_LENGTH;
_mtpd_gct.transaction_id = _mtpd_ctx.transaction_id;
if (_mtpd_ctx.session_id != 0)
{
_mtpd_gct.data[0] = _mtpd_ctx.session_id;
_mtpd_gct.container_length += sizeof(uint32_t);
}
TU_ASSERT(usbd_edpt_xfer(rhport, _mtpd_itf.ep_in, ((uint8_t *)(&_mtpd_gct)), _mtpd_gct.container_length), 0);
}
else
// Send next block of DATA
{
// Send Zero-Lenght Packet
if (_mtpd_itf.xferred_len == _mtpd_itf.total_len)
{
TU_ASSERT(usbd_edpt_xfer(rhport, _mtpd_itf.ep_in, ((uint8_t *)(&_mtpd_gct.data)), 0 ));
}
else
{
_mtpd_itf.phase = mtpd_handle_data();
if (_mtpd_itf.phase == MTP_PHASE_RESPONSE)
TU_ASSERT(usbd_edpt_xfer(rhport, _mtpd_itf.ep_in, ((uint8_t *)(&_mtpd_gct)), _mtpd_gct.container_length));
else
TU_ASSERT(usbd_edpt_xfer(rhport, _mtpd_itf.ep_in, ((uint8_t *)(&_mtpd_gct.data)), _mtpd_itf.queued_len ));
}
}
break;
default:
return false;
}

Check notice

Code scanning / CodeQL

Long switch case

Switch has at least one case that is too long: [MTP_PHASE_DATA_IN (36 lines)](1).
Comment on lines 298 to 299
// usbd_edpt_stall(rhport, _mtpd_itf.ep_out);
// usbd_edpt_stall(rhport, _mtpd_itf.ep_in);

Check notice

Code scanning / CodeQL

Commented-out code

This comment appears to contain commented-out code.
Comment on lines 362 to 363
// usbd_edpt_stall(rhport, _mtpd_itf.ep_out);
// usbd_edpt_stall(rhport, _mtpd_itf.ep_in);

Check notice

Code scanning / CodeQL

Commented-out code

This comment appears to contain commented-out code.
}

// A zero-length or a short packet termination
if (xferred_bytes < CFG_MTP_EP_SIZE || xferred_bytes == 0)

Check warning

Code scanning / CodeQL

Comparison result is always the same

Comparison is always false because xferred_bytes >= 64.
Comment on lines 729 to 746
switch(device_prop_code)
{
case MTP_DEVP_DEVICE_FRIENDLY_NAME:
{
TU_VERIFY_STATIC(sizeof(mtp_device_prop_desc_t) < MTP_MAX_PACKET_SIZE, "mtp_device_info_t shall fit in MTP_MAX_PACKET_SIZE");
_mtpd_gct.container_type = MTP_CONTAINER_TYPE_DATA_BLOCK;
_mtpd_gct.code = MTP_OPEC_GET_DEVICE_INFO;
_mtpd_gct.container_length = MTP_GENERIC_DATA_BLOCK_LENGTH + sizeof(mtp_device_prop_desc_t);
mtp_device_prop_desc_t *d = (mtp_device_prop_desc_t *)_mtpd_gct.data;
d->device_property_code = (uint16_t)(device_prop_code);
d->datatype = MTP_TYPE_STR;
d->get_set = MTP_MODE_GET;
mtpd_gct_append_wstring(CFG_TUD_MODEL); // factory_def_value
mtpd_gct_append_wstring(CFG_TUD_MODEL); // current_value_len
mtpd_gct_append_uint8(0x00); // form_flag
_mtpd_itf.queued_len = _mtpd_gct.container_length;
return MTP_PHASE_DATA_IN;
}
}

Check notice

Code scanning / CodeQL

No trivial switch statements

This switch statement should either handle more cases, or be rewritten as an if statement.
@WerWolv
Copy link

WerWolv commented Mar 14, 2025

Congrats!
I started working on my own MTP driver about a week ago and just got the first few things working now before I found your PR. Nowhere near done yet but here's my branch in case it helps anybody: https://github.com/WerWolv/tinyusb/tree/sicd

image

@roundby
Copy link
Author

roundby commented Mar 15, 2025

@WerWolv thank you for sharing!
I am giving some context here, as it might be beneficial for review. I had a short look at your code, the implementation is not that far from the one we're suggesting. Since you already have a hands-on experience on how the MTP class works, it would be great if you could take a look at our code too (feel free to add comments or propose corrections) and maybe perform a test on a physical device (since now, we tested it on STM32F3 and on a custom PSoC5LP port, on Windows hosts). Hopefully, the example should work out of the box on most of the already supported demo boards.

This code has been designed after the need to expose a set of files and folders stored in an SPI memory managed by littlefs, and we took the chance to do it within this library (we already used TinyUSB on some of our designs, and we hope this is a way to contribute back). While reading the code and the example, one should remind that the specification requires the device to provide a session unique 32 bit identifier for each object (e.g. file or directory), cached by the host when traversing storage content, able to be recalled any time and never re-used if an object is deleted. The implementation of this on the application side and in case of static files is easy, but it's not trivial on mutable file system where file I/O is performed by both host and (internally) by the device. Moreover, we found that, under Windows (appearing in contrast with MTP requirements) the object IDs are still cached if the session is closed and reopened, unless a bus reset or physical unplug is performed.

Comparing with your code:

  • mtp.h contains some of the definitions you have in sicd.h. We added packed data structures for fixed data as we found this easier to read and similar to what is already done (see mass storage device class).
  • We provided application callbacks in mtp_device_storage.h, in order for the application to provide the interface to enumerate, read and write objects. Comments should be self-explanatory, but feel free to ask in case of doubts.
  • mtp_device.c contains the actual implementation (similar to your sicd_device.c). We did not require a minimum EP size, as the buffer I/O is managed by the application (usually the filesystem itself), but we statically allocated a single buffer with size MTP_MAX_PACKET_SIZE (512). With this single buffer, the code supports arbitrary object sizes. Each command is handled in a single function and information is taken from tud_mtp_* callbacks.
  • The example code is under examples/device/mtp/src/mtp_fs_example.c. A few MTP class specific definitions, namely CFG_MTP_*, are added to tusb_config.h.
  • We did not add support for properties (we don't need them, and they would have increased the amount of code to review).

The code is functional, and it is expected to be ready for exposing file and directory level I/O when a block device is not advisable:

  • Storage is visible in Explorer
  • Objects (files and directories) can be enumerated
  • Object reading, creation and deletion works

I recognize that the introduction of a new class (and it's the first time we're contributing to TinyUSB) could require some time for a review and refinement, so, we'll wait for indications from the maintainers about the interest in merging this, asking for changes, moving to a different branch for testing/integration with additional features or any other suggestion. So far, we corrected all the notified build errors and most of the code scanning notices.

@simon88
Copy link

simon88 commented Mar 31, 2025

Hi @roundby
Great work, do you try it with littlefs?

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

4 participants