-
Notifications
You must be signed in to change notification settings - Fork 1.1k
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
base: master
Are you sure you want to change the base?
Conversation
b734659
to
8c0a674
Compare
src/class/mtp/mtp.h
Outdated
MTP_DEVP_DEVICE_FRIENDLY_NAME, | ||
}; | ||
|
||
tu_static const uint16_t mtp_object_properties_supported[] = { |
Check notice
Code scanning / CodeQL
Unused static variable
src/class/mtp/mtp.h
Outdated
// wstring factory_def_value; | ||
// wstring current_value_len; | ||
// uint8_t form_flag; |
Check notice
Code scanning / CodeQL
Commented-out code
src/class/mtp/mtp_device.c
Outdated
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
src/class/mtp/mtp_device.c
Outdated
// usbd_edpt_stall(rhport, _mtpd_itf.ep_out); | ||
// usbd_edpt_stall(rhport, _mtpd_itf.ep_in); |
Check notice
Code scanning / CodeQL
Commented-out code
src/class/mtp/mtp_device.c
Outdated
// usbd_edpt_stall(rhport, _mtpd_itf.ep_out); | ||
// usbd_edpt_stall(rhport, _mtpd_itf.ep_in); |
Check notice
Code scanning / CodeQL
Commented-out code
src/class/mtp/mtp_device.c
Outdated
} | ||
|
||
// 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
src/class/mtp/mtp_device.c
Outdated
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
Congrats! |
@WerWolv thank you for sharing! 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:
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:
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. |
Hi @roundby |
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