reworking MTP API, adding callback, getting GetDeviceInfo working

This commit is contained in:
hathach
2025-09-19 11:44:21 +07:00
parent aea56dc776
commit f99f203c28
4 changed files with 794 additions and 729 deletions

File diff suppressed because it is too large Load Diff

View File

@ -38,7 +38,6 @@
extern "C" {
#endif
#define TU_ARRAY_LEN(a) (sizeof(a)/sizeof(a[0]))
#define STORAGE_ID(physical_id, logical_id) ( (((uint32_t)physical_id & 0xFFFF) << 16) | ((uint32_t)logical_id & 0x0000FFFF) )
typedef uint16_t wchar16_t;
@ -583,19 +582,32 @@ typedef enum
} mtp_object_handles_t;
// Datatypes
typedef enum
{
MTP_TYPE_UNDEFINED = 0x0000u,
MTP_TYPE_INT8 = 0x0001u,
MTP_TYPE_UINT8 = 0x0002u,
MTP_TYPE_INT16 = 0x0003u,
MTP_TYPE_UINT16 = 0x0004u,
MTP_TYPE_INT32 = 0x0005u,
MTP_TYPE_UINT32 = 0x0006u,
MTP_TYPE_INT64 = 0x0007u,
MTP_TYPE_UINT64 = 0x0008u,
MTP_TYPE_STR = 0xFFFFu,
} mtp_datatypes_t;
typedef enum {
MTP_DATA_TYPE_UNDEFINED = 0x0000u,
// scalars
MTP_DATA_TYPE_INT8 = 0x0001u,
MTP_DATA_TYPE_UINT8 = 0x0002u,
MTP_DATA_TYPE_INT16 = 0x0003u,
MTP_DATA_TYPE_UINT16 = 0x0004u,
MTP_DATA_TYPE_INT32 = 0x0005u,
MTP_DATA_TYPE_UINT32 = 0x0006u,
MTP_DATA_TYPE_INT64 = 0x0007u,
MTP_DATA_TYPE_UINT64 = 0x0008u,
MTP_DATA_TYPE_INT128 = 0x0009u,
MTP_DATA_TYPE_UINT128 = 0x000Au,
// array
MTP_DATA_TYPE_AINT8 = 0x4001u,
MTP_DATA_TYPE_AUINT8 = 0x4002u,
MTP_DATA_TYPE_AINT16 = 0x4003u,
MTP_DATA_TYPE_AUINT16 = 0x4004u,
MTP_DATA_TYPE_AINT32 = 0x4005u,
MTP_DATA_TYPE_AUINT32 = 0x4006u,
MTP_DATA_TYPE_AINT64 = 0x4007u,
MTP_DATA_TYPE_AUINT64 = 0x4008u,
MTP_DATA_TYPE_AINT128 = 0x4009u,
MTP_DATA_TYPE_AUINT128 = 0x400Au,
MTP_DATA_TYPE_STR = 0xFFFFu,
} mtp_data_type_t;
// Get/Set
typedef enum
@ -707,46 +719,24 @@ typedef struct TU_ATTR_PACKED {
uint32_t data[MTP_MAX_PACKET_SIZE / sizeof(uint32_t)];
} mtp_generic_container_t;
// DeviceInfo Dataset
#define MTP_EXTENSIONS "microsoft.com: 1.0; "
#define mtp_string_t(_nchars) \
struct TU_ATTR_PACKED { \
uint8_t count; /* in characters including null */ \
uint16_t utf16[_nchars]; \
}
#define mtp_array_t(_type, _count) \
struct TU_ATTR_PACKED { \
uint32_t count; \
_type arr[_count];\
}
#define mtp_auint16_t(_count) mtp_array_t(uint16_t, _count)
typedef struct TU_ATTR_PACKED {
uint16_t standard_version;
uint32_t mtp_vendor_extension_id;
uint16_t mtp_version;
uint8_t mtp_extensions_len;
wchar16_t mtp_extensions[TU_ARRAY_LEN(MTP_EXTENSIONS)] TU_ATTR_PACKED;
uint16_t functional_mode;
/* Operations supported */
uint32_t operations_supported_len;
uint16_t operations_supported[TU_ARRAY_LEN(mtp_operations_supported)] TU_ATTR_PACKED;
/* Events supported */
uint32_t events_supported_len;
uint16_t events_supported[TU_ARRAY_LEN(mtp_events_supported)] TU_ATTR_PACKED;
/* Device properties supported */
uint32_t device_properties_supported_len;
uint16_t device_properties_supported[TU_ARRAY_LEN(mtp_device_properties_supported)] TU_ATTR_PACKED;
/* Capture formats */
uint32_t capture_formats_len;
uint16_t capture_formats[TU_ARRAY_LEN(mtp_capture_formats)] TU_ATTR_PACKED;
/* Playback formats */
uint32_t playback_formats_len;
uint16_t playback_formats[TU_ARRAY_LEN(mtp_playback_formats)] TU_ATTR_PACKED;
} mtp_device_info_t;
// The following fields will be dynamically added to the struct at runtime:
// - wstring manufacturer
// - wstring model
// - wstring device_version
// - wstring serial_number
#define MTP_STRING_DEF(name, string) \
uint8_t name##_len; \
wchar16_t name[TU_ARRAY_LEN(string)];
#define MTP_ARRAY_DEF(name, array) \
uint16_t name##_len; \
typeof(name) name[TU_ARRAY_LEN(array)];
uint8_t count;
uint16_t utf16[];
} mtp_flexible_string_t;
// StorageInfo dataset
typedef struct TU_ATTR_PACKED {
@ -813,6 +803,95 @@ typedef struct TU_ATTR_PACKED {
uint32_t parent_object_handle;
} mtp_basic_object_info_t;
//--------------------------------------------------------------------+
// Generic Container function
//--------------------------------------------------------------------+
TU_ATTR_ALWAYS_INLINE static inline uint32_t mtp_container_add(mtp_generic_container_t* p_container, mtp_data_type_t type, const void* data) {
TU_VERIFY(type != MTP_DATA_TYPE_UNDEFINED, 0);
uint8_t scalar_size; // size of single scalar
uint8_t count_width; // size of count field (0, 1 or 4 bytes)
if (type == MTP_DATA_TYPE_STR) {
scalar_size = 2;
count_width = 1;
} else {
uint8_t scalar_type = type & 0x3F;
count_width = (type & 0x4000u) ? 4 : 0;
scalar_size = 1u << ((scalar_type - 1u) >> 1);
}
uint32_t data_len;
if (count_width) {
const uint32_t count = *(const uint32_t*) data;
data_len = count_width + count*scalar_size;
} else {
data_len = scalar_size;
}
memcpy(((uint8_t*)p_container) + p_container->len, data, data_len);
p_container->len += data_len;
return data_len;
}
TU_ATTR_ALWAYS_INLINE static inline uint32_t mtp_container_add_field(mtp_generic_container_t* p_container, uint8_t scalar_size, uint32_t count, const void* data) {
const uint32_t prev_len = p_container->len;
uint8_t* container8 = (uint8_t*) p_container;
if (count == 0) {
// count = 0 means scalar
memcpy(container8 + p_container->len, data, scalar_size);
p_container->len += scalar_size;
} else {
tu_unaligned_write32(container8 + p_container->len, count);
p_container->len += 4;
memcpy(container8 + p_container->len, data, count * scalar_size);
}
return p_container->len - prev_len;
}
TU_ATTR_ALWAYS_INLINE static inline uint32_t mtp_container_add_string(mtp_generic_container_t* p_container, uint8_t count, uint16_t* utf16) {
const uint32_t prev_len = p_container->len;
uint8_t* container8 = (uint8_t*) p_container;
*(container8 + p_container->len) = count;
p_container->len += 1;
memcpy(container8 + p_container->len, utf16, 2 * count);
p_container->len += 2 * count;
return p_container->len - prev_len;
}
TU_ATTR_ALWAYS_INLINE static inline uint32_t mtp_container_add_uint8(mtp_generic_container_t* p_container, uint8_t data) {
return mtp_container_add_field(p_container, sizeof(uint8_t), 0, &data);
}
TU_ATTR_ALWAYS_INLINE static inline uint32_t mtp_container_add_uint16(mtp_generic_container_t* p_container, uint16_t data) {
return mtp_container_add_field(p_container, sizeof(uint16_t), 0, &data);
}
TU_ATTR_ALWAYS_INLINE static inline uint32_t mtp_container_add_uint32(mtp_generic_container_t* p_container, uint32_t data) {
return mtp_container_add_field(p_container, sizeof(uint32_t), 0, &data);
}
TU_ATTR_ALWAYS_INLINE static inline uint32_t mtp_container_add_uint64(mtp_generic_container_t* p_container, uint64_t data) {
return mtp_container_add_field(p_container, 8, 0, &data);
}
TU_ATTR_ALWAYS_INLINE static inline uint32_t mtp_container_add_auint8(mtp_generic_container_t* p_container, uint32_t count, const uint8_t* data) {
return mtp_container_add_field(p_container, sizeof(uint8_t), count, data);
}
TU_ATTR_ALWAYS_INLINE static inline uint32_t mtp_container_add_auint16(mtp_generic_container_t* p_container, uint32_t count, const uint16_t* data) {
return mtp_container_add_field(p_container, sizeof(uint16_t), count, data);
}
TU_ATTR_ALWAYS_INLINE static inline uint32_t mtp_container_add_auint32(mtp_generic_container_t* p_container, uint32_t count, const uint32_t* data) {
return mtp_container_add_field(p_container, sizeof(uint32_t), count, data);
}
#ifdef __cplusplus
}
#endif

View File

@ -50,8 +50,8 @@
//--------------------------------------------------------------------+
// STRUCT
//--------------------------------------------------------------------+
typedef struct
{
typedef struct {
uint8_t rhport;
uint8_t itf_num;
uint8_t ep_in;
uint8_t ep_out;
@ -82,7 +82,7 @@ static mtp_phase_type_t mtpd_chk_generic(const char *func_name, const bool err_c
static mtp_phase_type_t mtpd_chk_session_open(const char *func_name);
// MTP commands
static mtp_phase_type_t mtpd_handle_cmd(void);
static mtp_phase_type_t mtpd_handle_cmd(mtpd_interface_t* p_mtp);
static mtp_phase_type_t mtpd_handle_data(void);
static mtp_phase_type_t mtpd_handle_cmd_get_device_info(void);
static mtp_phase_type_t mtpd_handle_cmd_open_session(void);
@ -118,9 +118,9 @@ CFG_TUD_MEM_SECTION CFG_TUSB_MEM_ALIGN static char _mtp_datestr[20];
// Helper
//--------------------------------------------------------------------+
static bool prepare_new_command(uint8_t rhport, mtpd_interface_t* p_mtp) {
static bool prepare_new_command(mtpd_interface_t* p_mtp) {
p_mtp->phase = MTP_PHASE_IDLE;
return usbd_edpt_xfer(rhport, p_mtp->ep_out, (uint8_t *)(&_mtpd_epbuf.container), sizeof(mtp_generic_container_t));
return usbd_edpt_xfer(p_mtp->rhport, p_mtp->ep_out, (uint8_t *)(&_mtpd_epbuf.container), sizeof(mtp_generic_container_t));
}
@ -157,6 +157,8 @@ uint16_t mtpd_open(uint8_t rhport, tusb_desc_interface_t const* itf_desc, uint16
// Max length must be at least 1 interface + 3 endpoints
TU_ASSERT(itf_desc->bNumEndpoints == 3 && max_len >= mtpd_itf_size);
mtpd_interface_t* p_mtp = &_mtpd_itf;
tu_memclr(p_mtp, sizeof(mtpd_interface_t));
p_mtp->rhport = rhport;
p_mtp->itf_num = itf_desc->bInterfaceNumber;
// Open interrupt IN endpoint
@ -168,7 +170,7 @@ uint16_t mtpd_open(uint8_t rhport, tusb_desc_interface_t const* itf_desc, uint16
// Open endpoint pair
TU_ASSERT(usbd_open_edpt_pair(rhport, tu_desc_next(ep_desc), 2, TUSB_XFER_BULK, &p_mtp->ep_out, &p_mtp->ep_in), 0);
TU_ASSERT(prepare_new_command(rhport, p_mtp), 0);
TU_ASSERT(prepare_new_command(p_mtp), 0);
return mtpd_itf_size;
}
@ -216,6 +218,29 @@ bool mtpd_control_xfer_cb(uint8_t rhport, uint8_t stage, tusb_control_request_t
return true;
}
bool tud_mtp_data_send(mtp_generic_container_t* data_block) {
mtpd_interface_t* p_mtp = &_mtpd_itf;
p_mtp->phase = MTP_PHASE_DATA;
p_mtp->total_len = data_block->len;
p_mtp->xferred_len = 0;
p_mtp->handled_len = 0;
p_mtp->xfer_completed = false;
data_block->type = MTP_CONTAINER_TYPE_DATA_BLOCK;
data_block->transaction_id = p_mtp->cmd_header.transaction_id;
TU_ASSERT(usbd_edpt_xfer(p_mtp->rhport, p_mtp->ep_in, (uint8_t*) data_block, (uint16_t)data_block->len));
return true;
}
bool tud_mtp_response_send(mtp_generic_container_t* resp_block) {
mtpd_interface_t* p_mtp = &_mtpd_itf;
p_mtp->phase = MTP_PHASE_RESPONSE_QUEUED;
resp_block->type = MTP_CONTAINER_TYPE_RESPONSE_BLOCK;
resp_block->transaction_id = p_mtp->cmd_header.transaction_id;
TU_ASSERT(usbd_edpt_xfer(p_mtp->rhport, p_mtp->ep_in, (uint8_t*) resp_block, (uint16_t)resp_block->len));
return true;
}
// Transfer on bulk endpoints
bool mtpd_xfer_cb(uint8_t rhport, uint8_t ep_addr, xfer_result_t event, uint32_t xferred_bytes) {
TU_ASSERT(event == XFER_RESULT_SUCCESS);
@ -235,23 +260,25 @@ bool mtpd_xfer_cb(uint8_t rhport, uint8_t ep_addr, xfer_result_t event, uint32_t
p_mtp->phase = MTP_PHASE_COMMAND;
TU_ATTR_FALLTHROUGH; // handle in the next case
case MTP_PHASE_COMMAND:
// Handle command block
memcpy(&p_mtp->cmd_header, p_container, sizeof(mtp_container_header_t));
p_mtp->phase = mtpd_handle_cmd();
if (p_mtp->phase == MTP_PHASE_DATA_IN) {
TU_ASSERT(usbd_edpt_xfer(rhport, p_mtp->ep_in, (uint8_t*) p_container, (uint16_t)p_mtp->queued_len));
p_mtp->total_len = p_container->len;
p_mtp->xferred_len = 0;
p_mtp->handled_len = 0;
p_mtp->xfer_completed = false;
} else if (p_mtp->phase == MTP_PHASE_DATA_OUT) {
p_mtp->xferred_len = 0;
p_mtp->handled_len = 0;
p_mtp->xfer_completed = false;
TU_ASSERT(usbd_edpt_xfer(rhport, p_mtp->ep_out, (uint8_t*) p_container, sizeof(mtp_generic_container_t)), 0);
case MTP_PHASE_COMMAND: {
mtpd_handle_cmd(p_mtp);
break;
}
case MTP_PHASE_DATA: {
const uint16_t bulk_mps = (tud_speed_get() == TUSB_SPEED_HIGH) ? 512 : 64;
p_mtp->xferred_len += xferred_bytes;
// transfer complete if ZLP or short packet or overflow
if (xferred_bytes == 0 || // ZLP
(xferred_bytes & (bulk_mps - 1)) || // short packet
p_mtp->xferred_len > p_mtp->total_len) {
tud_mtp_data_complete_cb(0, &p_mtp->cmd_header, p_container, event, p_mtp->xferred_len);
} else {
TU_ASSERT(false);
}
break;
}
case MTP_PHASE_DATA_IN:
p_mtp->xferred_len += xferred_bytes;
@ -328,7 +355,7 @@ bool mtpd_xfer_cb(uint8_t rhport, uint8_t ep_addr, xfer_result_t event, uint32_t
case MTP_PHASE_RESPONSE_QUEUED:
// response phase is complete -> prepare for new command
TU_ASSERT(ep_addr == p_mtp->ep_in);
prepare_new_command(rhport, p_mtp);
prepare_new_command(p_mtp);
break;
case MTP_PHASE_RESPONSE:
@ -339,151 +366,16 @@ bool mtpd_xfer_cb(uint8_t rhport, uint8_t ep_addr, xfer_result_t event, uint32_t
}
if (p_mtp->phase == MTP_PHASE_RESPONSE) {
p_mtp->phase = MTP_PHASE_RESPONSE_QUEUED;
p_container->type = MTP_CONTAINER_TYPE_RESPONSE_BLOCK;
p_container->transaction_id = p_mtp->cmd_header.transaction_id;
TU_ASSERT(usbd_edpt_xfer(rhport, p_mtp->ep_in, (uint8_t*) p_container, (uint16_t)p_container->len), 0);
// p_mtp->phase = MTP_PHASE_RESPONSE_QUEUED;
// p_container->type = MTP_CONTAINER_TYPE_RESPONSE_BLOCK;
// p_container->transaction_id = p_mtp->cmd_header.transaction_id;
// TU_ASSERT(usbd_edpt_xfer(rhport, p_mtp->ep_in, (uint8_t*) p_container, (uint16_t)p_container->len), 0);
} else if (p_mtp->phase == MTP_PHASE_ERROR) {
// stall both IN & OUT endpoints
usbd_edpt_stall(rhport, p_mtp->ep_out);
usbd_edpt_stall(rhport, p_mtp->ep_in);
}
#if 0
// IN transfer completed
if (ep_addr == p_mtp->ep_in) {
if (p_mtp->phase == MTP_PHASE_RESPONSE) {
// IN transfer completed, prepare for a new command
TU_ASSERT(usbd_edpt_xfer(rhport, p_mtp->ep_out, (uint8_t*) p_container, CFG_MTP_EP_SIZE), 0);
p_mtp->phase = MTP_PHASE_IDLE;
} else if (p_mtp->phase == MTP_PHASE_DATA_IN) {
p_mtp->xferred_len += xferred_bytes;
p_mtp->handled_len = p_mtp->xferred_len;
// Check if transfer completed.
if (p_mtp->xferred_len >= p_mtp->total_len && (xferred_bytes == 0 || (xferred_bytes % CFG_MTP_EP_SIZE) != 0)) {
p_mtp->phase = MTP_PHASE_RESPONSE;
p_container->type = MTP_CONTAINER_TYPE_RESPONSE_BLOCK;
p_container->code = MTP_RESP_OK;
p_container->len = MTP_CONTAINER_HEADER_LENGTH;
p_container->transaction_id = p_mtp->context.transaction_id;
if (p_mtp->session_id != 0) {
p_container->data[0] = p_mtp->session_id;
p_container->len += sizeof(uint32_t);
}
TU_ASSERT(usbd_edpt_xfer(rhport, p_mtp->ep_in, (uint8_t*) p_container, (uint16_t)p_container->len), 0);
} else {
// Send next block of DATA
// Send Zero-Length Packet
if (p_mtp->xferred_len == p_mtp->total_len) {
TU_ASSERT(usbd_edpt_xfer(rhport, p_mtp->ep_in, ((uint8_t *)(&p_container->data)), 0 ));
} else {
p_mtp->phase = mtpd_handle_data();
if (p_mtp->phase == MTP_PHASE_RESPONSE) {
TU_ASSERT(usbd_edpt_xfer(rhport, p_mtp->ep_in, (uint8_t*) p_container, (uint16_t)p_container->len));
} else {
TU_ASSERT(usbd_edpt_xfer(rhport, p_mtp->ep_in, ((uint8_t *)(&p_container->data)), (uint16_t)p_mtp->queued_len));
}
}
}
} else {
return false;
}
}
if (ep_addr == p_mtp->ep_out) {
if (p_mtp->phase == MTP_PHASE_IDLE) {
// A new command has been received. Ensure this is the last of the sequence.
p_mtp->total_len = p_container->len;
// Stall in case of unexpected block
if (p_container->type != MTP_CONTAINER_TYPE_COMMAND_BLOCK) {
return false;
}
p_mtp->phase = MTP_PHASE_COMMAND;
p_mtp->total_len = p_container->len;
p_mtp->xferred_len = xferred_bytes;
p_mtp->handled_len = 0;
p_mtp->xfer_completed = false;
TU_ASSERT(p_mtp->total_len < sizeof(mtp_generic_container_t));
}
if (p_mtp->phase == MTP_PHASE_COMMAND) {
// A zero-length or a short packet termination is expected
if (xferred_bytes == CFG_MTP_EP_SIZE || (p_mtp->total_len - p_mtp->xferred_len) > 0) {
TU_ASSERT(usbd_edpt_xfer(rhport, p_mtp->ep_out, (uint8_t*) p_container + p_mtp->xferred_len, (uint16_t)(p_mtp->total_len - p_mtp->xferred_len)));
} else {
// Handle command block
p_mtp->phase = mtpd_handle_cmd();
if (p_mtp->phase == MTP_PHASE_RESPONSE) {
TU_ASSERT(usbd_edpt_xfer(rhport, p_mtp->ep_in, (uint8_t*) p_container, (uint16_t)p_container->len));
} else if (p_mtp->phase == MTP_PHASE_DATA_IN) {
TU_ASSERT(usbd_edpt_xfer(rhport, p_mtp->ep_in, (uint8_t*) p_container, (uint16_t)p_mtp->queued_len));
p_mtp->total_len = p_container->len;
p_mtp->xferred_len = 0;
p_mtp->handled_len = 0;
p_mtp->xfer_completed = false;
} else if (p_mtp->phase == MTP_PHASE_DATA_OUT) {
TU_ASSERT(usbd_edpt_xfer(rhport, p_mtp->ep_out, (uint8_t*) p_container, sizeof(mtp_generic_container_t)), 0);
p_mtp->xferred_len = 0;
p_mtp->handled_len = 0;
p_mtp->xfer_completed = false;
} else {
usbd_edpt_stall(rhport, p_mtp->ep_out);
usbd_edpt_stall(rhport, p_mtp->ep_in);
}
}
return true;
}
if (p_mtp->phase == MTP_PHASE_DATA_OUT) {
// First block of data
if (p_mtp->xferred_len == 0) {
p_mtp->total_len = p_container->len;
p_mtp->handled_len = 0;
p_mtp->xfer_completed = false;
}
p_mtp->xferred_len += xferred_bytes;
// Stall in case of unexpected block
if (p_container->type != MTP_CONTAINER_TYPE_DATA_BLOCK) { return false; }
// A zero-length or a short packet termination
if (xferred_bytes < CFG_MTP_EP_SIZE) {
p_mtp->xfer_completed = true;
// Handle data block
p_mtp->phase = mtpd_handle_data();
if (p_mtp->phase == MTP_PHASE_DATA_IN || p_mtp->phase == MTP_PHASE_RESPONSE) {
TU_ASSERT(usbd_edpt_xfer(rhport, p_mtp->ep_in, (uint8_t*) p_container, (uint16_t)p_container->len));
} else if (p_mtp->phase == MTP_PHASE_DATA_OUT) {
TU_ASSERT(usbd_edpt_xfer(rhport, p_mtp->ep_out, (uint8_t*) p_container, sizeof(mtp_generic_container_t)), 0);
p_mtp->xferred_len = 0;
p_mtp->xfer_completed = false;
} else {
usbd_edpt_stall(rhport, p_mtp->ep_out);
usbd_edpt_stall(rhport, p_mtp->ep_in);
}
} else {
// Handle data block when container is full
if (p_mtp->xferred_len - p_mtp->handled_len >= MTP_MAX_PACKET_SIZE - CFG_MTP_EP_SIZE) {
p_mtp->phase = mtpd_handle_data();
p_mtp->handled_len = p_mtp->xferred_len;
}
// Transfer completed: wait for zero-lenght packet
// Some platforms may not respect EP size and xferred_bytes may be more than CFG_MTP_EP_SIZE if
// the OUT EP is waiting for more data. Ensure we are not waiting for more than CFG_MTP_EP_SIZE.
if (p_mtp->total_len == p_mtp->xferred_len) {
TU_ASSERT(usbd_edpt_xfer(rhport, p_mtp->ep_out, ((uint8_t *)(&p_container->data)), CFG_MTP_EP_SIZE), 0);
} else if (p_mtp->handled_len == 0) {
// First data block includes container header + container data
TU_ASSERT(usbd_edpt_xfer(rhport, p_mtp->ep_out, (uint8_t*) p_container + p_mtp->xferred_len, (uint16_t)TU_MIN(p_mtp->total_len - p_mtp->xferred_len, CFG_MTP_EP_SIZE)));
} else {
// Successive data block includes only container data
TU_ASSERT(usbd_edpt_xfer(rhport, p_mtp->ep_out, ((uint8_t *)(&p_container->data)) + p_mtp->xferred_len - p_mtp->handled_len, (uint16_t)TU_MIN(p_mtp->total_len - p_mtp->xferred_len, CFG_MTP_EP_SIZE)));
}
}
}
}
#endif
return true;
}
@ -492,26 +384,77 @@ bool mtpd_xfer_cb(uint8_t rhport, uint8_t ep_addr, xfer_result_t event, uint32_t
//--------------------------------------------------------------------+
// Decode command and prepare response
mtp_phase_type_t mtpd_handle_cmd(void) {
mtp_phase_type_t mtpd_handle_cmd(mtpd_interface_t* p_mtp) {
mtp_generic_container_t* p_container = &_mtpd_epbuf.container;
TU_ASSERT(p_container->type == MTP_CONTAINER_TYPE_COMMAND_BLOCK);
mtp_generic_container_t cmd_block; // copy command block for callback
memcpy(&cmd_block, p_container, p_container->len);
memcpy(&p_mtp->cmd_header, p_container, sizeof(mtp_container_header_t));
// p_container->len = MTP_CONTAINER_HEADER_LENGTH; // default data/response length
if (p_container->code != MTP_OP_SEND_OBJECT) {
_mtpd_soi.object_handle = 0;
}
mtp_phase_type_t ret = MTP_PHASE_RESPONSE;
switch (p_container->code) {
case MTP_OP_GET_DEVICE_INFO:
case MTP_OP_GET_DEVICE_INFO: {
TU_LOG_DRV(" MTP command: MTP_OP_GET_DEVICE_INFO\n");
return mtpd_handle_cmd_get_device_info();
tud_mtp_device_info_t dev_info = {
.standard_version = 100,
.mtp_vendor_extension_id = 0xFFFFFFFFU,
.mtp_version = 100,
.mtp_extensions = {
.count = sizeof(CFG_TUD_MTP_DEVICEINFO_EXTENSIONS),
.utf16 = { 0 }
},
.functional_mode = 0x0000,
.supported_operations = {
.count = TU_ARGS_NUM(CFG_TUD_MTP_DEVICEINFO_SUPPORTED_OPERATIONS),
.arr = { CFG_TUD_MTP_DEVICEINFO_SUPPORTED_OPERATIONS }
},
.supported_events = {
.count = TU_ARGS_NUM(CFG_TUD_MTP_DEVICEINFO_SUPPORTED_EVENTS),
.arr = { CFG_TUD_MTP_DEVICEINFO_SUPPORTED_EVENTS }
},
.supported_device_properties = {
.count = TU_ARGS_NUM(CFG_TUD_MTP_DEVICEINFO_SUPPORTED_DEVICE_PROPERTIES),
.arr = { CFG_TUD_MTP_DEVICEINFO_SUPPORTED_DEVICE_PROPERTIES }
},
.capture_formats = {
.count = TU_ARGS_NUM(CFG_TUD_MTP_DEVICEINFO_CAPTURE_FORMATS),
.arr = { CFG_TUD_MTP_DEVICEINFO_CAPTURE_FORMATS }
},
.playback_formats = {
.count = TU_ARGS_NUM(CFG_TUD_MTP_DEVICEINFO_PLAYBACK_FORMATS),
.arr = { CFG_TUD_MTP_DEVICEINFO_PLAYBACK_FORMATS }
}
};
for (uint8_t i=0; i < dev_info.mtp_extensions.count; i++) {
dev_info.mtp_extensions.utf16[i] = (uint16_t)CFG_TUD_MTP_DEVICEINFO_EXTENSIONS[i];
}
p_container->len = MTP_CONTAINER_HEADER_LENGTH + sizeof(tud_mtp_device_info_t);
p_container->type = MTP_CONTAINER_TYPE_DATA_BLOCK;
p_container->code = MTP_OP_GET_DEVICE_INFO;
memcpy(p_container->data, &dev_info, sizeof(tud_mtp_device_info_t));
ret = MTP_PHASE_RESPONSE;
break;
}
case MTP_OP_OPEN_SESSION:
TU_LOG_DRV(" MTP command: MTP_OP_OPEN_SESSION\n");
return mtpd_handle_cmd_open_session();
break;
case MTP_OP_CLOSE_SESSION:
TU_LOG_DRV(" MTP command: MTP_OP_CLOSE_SESSION\n");
return mtpd_handle_cmd_close_session();
case MTP_OP_GET_STORAGE_IDS:
TU_LOG_DRV(" MTP command: MTP_OP_GET_STORAGE_IDS\n");
return mtpd_handle_cmd_get_storage_ids();
case MTP_OP_GET_STORAGE_INFO:
TU_LOG_DRV(" MTP command: MTP_OP_GET_STORAGE_INFO for ID=%lu\n", p_container->data[0]);
return mtpd_handle_cmd_get_storage_info();
@ -546,7 +489,9 @@ mtp_phase_type_t mtpd_handle_cmd(void) {
TU_LOG_DRV(" MTP command: MTP_OP_UNKNOWN_COMMAND %x!!!!\n", p_container->code);
return false;
}
return true;
tud_mtp_command_received_cb(0, &cmd_block, p_container);
return ret;
}
mtp_phase_type_t mtpd_handle_data(void)
@ -572,40 +517,6 @@ mtp_phase_type_t mtpd_handle_data(void)
return true;
}
mtp_phase_type_t mtpd_handle_cmd_get_device_info(void)
{
TU_VERIFY_STATIC(sizeof(mtp_device_info_t) < MTP_MAX_PACKET_SIZE, "mtp_device_info_t shall fit in MTP_MAX_PACKET_SIZE");
mtp_generic_container_t* p_container = &_mtpd_epbuf.container;
p_container->len = MTP_CONTAINER_HEADER_LENGTH + sizeof(mtp_device_info_t);
p_container->type = MTP_CONTAINER_TYPE_DATA_BLOCK;
p_container->code = MTP_OP_GET_DEVICE_INFO;
mtp_device_info_t *d = (mtp_device_info_t *)p_container->data;
d->standard_version = 100;
d->mtp_vendor_extension_id = 0x06;
d->mtp_version = 100;
d->mtp_extensions_len = TU_ARRAY_LEN(MTP_EXTENSIONS);
mtpd_wc16cpy((uint8_t *)d->mtp_extensions, MTP_EXTENSIONS);
d->functional_mode = 0x0000;
d->operations_supported_len = TU_ARRAY_LEN(mtp_operations_supported);
memcpy(d->operations_supported, mtp_operations_supported, sizeof(mtp_operations_supported));
d->events_supported_len = TU_ARRAY_LEN(mtp_events_supported);
memcpy(d->events_supported, mtp_events_supported, sizeof(mtp_events_supported));
d->device_properties_supported_len = TU_ARRAY_LEN(mtp_device_properties_supported);
memcpy(d->device_properties_supported, mtp_device_properties_supported, sizeof(mtp_device_properties_supported));
d->capture_formats_len = TU_ARRAY_LEN(mtp_capture_formats);
memcpy(d->capture_formats, mtp_capture_formats, sizeof(mtp_capture_formats));
d->playback_formats_len = TU_ARRAY_LEN(mtp_playback_formats);
memcpy(d->playback_formats, mtp_playback_formats, sizeof(mtp_playback_formats));
mtpd_gct_append_wstring(CFG_TUD_MANUFACTURER);
mtpd_gct_append_wstring(CFG_TUD_MODEL);
mtpd_gct_append_wstring(CFG_MTP_DEVICE_VERSION);
mtpd_gct_append_wstring(CFG_MTP_SERIAL_NUMBER);
_mtpd_itf.queued_len = p_container->len;
return MTP_PHASE_DATA_IN;
}
mtp_phase_type_t mtpd_handle_cmd_open_session(void)
{
mtp_generic_container_t* p_container = &_mtpd_epbuf.container;
@ -851,7 +762,7 @@ mtp_phase_type_t mtpd_handle_cmd_get_device_prop_desc(void)
p_container->len = MTP_CONTAINER_HEADER_LENGTH + sizeof(mtp_device_prop_desc_t);
mtp_device_prop_desc_t *d = (mtp_device_prop_desc_t *)p_container->data;
d->device_property_code = (uint16_t)(device_prop_code);
d->datatype = MTP_TYPE_STR;
d->datatype = MTP_DATA_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

View File

@ -18,14 +18,14 @@
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN0
* THE SOFTWARE.
*
* This file is part of the TinyUSB stack.
*/
#ifndef _TUSB_MTP_DEVICE_H_
#define _TUSB_MTP_DEVICE_H_
#ifndef TUSB_MTP_DEVICE_H_
#define TUSB_MTP_DEVICE_H_
#include "common/tusb_common.h"
#include "mtp.h"
@ -36,15 +36,51 @@
extern "C" {
#endif
typedef struct {
const mtp_container_header_t* cmd_header;
tusb_xfer_result_t xfer_result;
uint32_t xferred_bytes;
} tud_mtp_cb_complete_data_t;
// Number of supported operations, events, device properties, capture formats, playback formats
// and max number of characters for strings manufacturer, model, device_version, serial_number
#define MTP_DEVICE_INFO_TYPEDEF(_extension_nchars, _op_count, _event_count, _devprop_count, _capture_count, _playback_count) \
struct TU_ATTR_PACKED { \
uint16_t standard_version; \
uint32_t mtp_vendor_extension_id; \
uint16_t mtp_version; \
mtp_string_t(_extension_nchars) mtp_extensions; \
uint16_t functional_mode; \
mtp_auint16_t(_op_count) supported_operations; \
mtp_auint16_t(_event_count) supported_events; \
mtp_auint16_t(_devprop_count) supported_device_properties; \
mtp_auint16_t(_capture_count) capture_formats; \
mtp_auint16_t(_playback_count) playback_formats; \
/* string fields will be added using append function */ \
}
typedef MTP_DEVICE_INFO_TYPEDEF(
sizeof(CFG_TUD_MTP_DEVICEINFO_EXTENSIONS), TU_ARGS_NUM(CFG_TUD_MTP_DEVICEINFO_SUPPORTED_OPERATIONS),
TU_ARGS_NUM(CFG_TUD_MTP_DEVICEINFO_SUPPORTED_EVENTS), TU_ARGS_NUM(CFG_TUD_MTP_DEVICEINFO_SUPPORTED_DEVICE_PROPERTIES),
TU_ARGS_NUM(CFG_TUD_MTP_DEVICEINFO_CAPTURE_FORMATS), TU_ARGS_NUM(CFG_TUD_MTP_DEVICEINFO_PLAYBACK_FORMATS)
) tud_mtp_device_info_t;
//--------------------------------------------------------------------+
// Internal Class Driver API
// Application API
//--------------------------------------------------------------------+
void mtpd_init (void);
bool mtpd_deinit (void);
void mtpd_reset (uint8_t rhport);
uint16_t mtpd_open (uint8_t rhport, tusb_desc_interface_t const *itf_desc, uint16_t max_len);
bool mtpd_control_xfer_cb (uint8_t rhport, uint8_t stage, tusb_control_request_t const *p_request);
bool mtpd_xfer_cb (uint8_t rhport, uint8_t ep_addr, xfer_result_t event, uint32_t xferred_bytes);
bool tud_mtp_data_send(mtp_generic_container_t* data_block);
// bool tud_mtp_block_data_receive();
bool tud_mtp_response_send(mtp_generic_container_t* resp_block);
//--------------------------------------------------------------------+
// Application Callbacks
//--------------------------------------------------------------------+
// Invoked when new command is received
int32_t tud_mtp_command_received_cb(uint8_t idx, mtp_generic_container_t* cmd_block, mtp_generic_container_t* out_block);
// Invoked when data phase is complete
int32_t tud_mtp_data_complete_cb(uint8_t idx, mtp_container_header_t* cmd_header, mtp_generic_container_t* resp_block, tusb_xfer_result_t xfer_result, uint32_t xferred_bytes);
//--------------------------------------------------------------------+
// Helper functions
@ -66,10 +102,19 @@ bool mtpd_gct_append_array(uint32_t array_size, const void *data, size_t type_si
// The function returns true if the data fits in the available buffer space.
bool mtpd_gct_append_date(struct tm *timeinfo);
//--------------------------------------------------------------------+
// Internal Class Driver API
//--------------------------------------------------------------------+
void mtpd_init (void);
bool mtpd_deinit (void);
void mtpd_reset (uint8_t rhport);
uint16_t mtpd_open (uint8_t rhport, tusb_desc_interface_t const *itf_desc, uint16_t max_len);
bool mtpd_control_xfer_cb (uint8_t rhport, uint8_t stage, tusb_control_request_t const *p_request);
bool mtpd_xfer_cb (uint8_t rhport, uint8_t ep_addr, xfer_result_t event, uint32_t xferred_bytes);
#ifdef __cplusplus
}
#endif
#endif /* CFG_TUD_ENABLED && CFG_TUD_MTP */
#endif /* _TUSB_MTP_DEVICE_H_ */
#endif
#endif