forked from stm/stm32-mw-usb-device
Compare commits
3 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
947131e000 | ||
|
|
d6e6bd2c66 | ||
|
|
2a0a3521ac |
@@ -1,31 +1,46 @@
|
|||||||
## Contributing guide
|
# Contributing guide
|
||||||
This document serves as a checklist before contributing to this repository.
|
|
||||||
It includes links to read up on if topics are unclear to you.
|
|
||||||
|
|
||||||
This guide mainly focuses on the proper use of Git.
|
This guide mainly focuses on the steps to follow to submit an issue or a pull request.
|
||||||
|
|
||||||
### 1. Before opening an issue
|
## 1. Issues
|
||||||
Please check the following boxes before posting an issue:
|
|
||||||
- [ ] `Make sure you are using the latest commit (major releases are Tagged, but corrections are available as new commits).`
|
|
||||||
- [ ] `Make sure your issue is a question/feedback/suggestions RELATED TO the software provided in this repository.` Otherwise, it should be discussed on the [ST Community/STM32 MCUs forum](https://community.st.com/s/group/0F90X000000AXsASAW/stm32-mcus).
|
|
||||||
- [ ] `Make sure your issue is not already reported/fixed on GitHub or discussed on a previous issue.` Please refer to this [dashboard](https://github.com/orgs/STMicroelectronics/projects/5) for the list of issues and pull-requests. Do not forget to browse into the **closed** issues.
|
|
||||||
|
|
||||||
### 2. Posting the issue
|
### 1.1 Before opening an issue
|
||||||
When you have checked the previous boxes. You will find two templates (Bug Report or Other Issue) available in the **Issues** tab of the repository ([link](https://github.com/STMicroelectronics/stm32_mw_usb_device/issues/new/choose)).
|
|
||||||
|
Before posting an issue, please ensure:
|
||||||
|
* You are using the latest commit.
|
||||||
|
* Your issue is **not** a vulnerability. Otherwise, please refer to section [3](CONTRIBUTING.md#3-vulnerabilities) below.
|
||||||
|
* Your issue is **related to** the software provided in this repository. Otherwise, please refer to section [4](CONTRIBUTING.md#4-support-requests-and-miscellaneous) below.
|
||||||
|
* Your issue is not already reported, fixed or discussed in a previous one. Remember to browse the **closed** issues.
|
||||||
|
|
||||||
|
### 1.2 Posting the issue
|
||||||
|
|
||||||
|
When you have checked the previous points, create a new report from the **Issues** tab of this repository. A couple of templates are available [here](../../issues/new/choose).
|
||||||
|
|
||||||
|
## 2. Pull Requests
|
||||||
|
|
||||||
|
### 2.1 Before opening a pull request
|
||||||
|
|
||||||
### 3. Pull Requests
|
|
||||||
STMicrolectronics is happy to receive contributions from the community, based on an initial Contributor License Agreement (CLA) procedure.
|
STMicrolectronics is happy to receive contributions from the community, based on an initial Contributor License Agreement (CLA) procedure.
|
||||||
|
|
||||||
* If you are an individual writing original source code and you are sure **you own the intellectual property**, then you need to sign an Individual CLA (https://cla.st.com).
|
* If you are an individual writing original source code and you are sure **you own the intellectual property**, then you need to sign an **Individual** [CLA](https://cla.st.com).
|
||||||
* If you work for a company that wants also to allow you to contribute with your work, your company needs to provide a Corporate CLA (https://cla.st.com) mentioning your GitHub account name.
|
* If you work for a company that wants also to allow you to contribute with your work, your company needs to provide a **Corporate** [CLA](https://cla.st.com) mentioning your GitHub account name.
|
||||||
* If you are not sure that a CLA (Individual or Corporate) has been signed for your GitHub account you can check here (https://cla.st.com).
|
* If you are not sure that a CLA (Individual or Corporate) has been signed for your GitHub account you can check the [CLA](https://cla.st.com) dedicated page.
|
||||||
|
|
||||||
Please note that:
|
> [!IMPORTANT]
|
||||||
* The Corporate CLA will always take precedence over the Individual CLA.
|
> Please note that:
|
||||||
* One CLA submission is sufficient, for any project proposed by STMicroelectronics.
|
> * The Corporate CLA will always take precedence over the Individual CLA.
|
||||||
|
> * One CLA submission is sufficient, for any project proposed by STMicroelectronics.
|
||||||
|
|
||||||
#### How to proceed
|
### 2.2 How to proceed
|
||||||
|
|
||||||
* We recommend to engage first a communication thru an issue, in order to present your proposal. Just to confirm that it corresponds to STMicroelectronics domain or scope.
|
* We recommend to engage first a communication thru an issue, in order to present your proposal, just to confirm that it corresponds to STMicroelectronics' domain or scope.
|
||||||
* Then fork the project to your GitHub account to further develop your contribution. Please use the latest commit version.
|
* Then fork the project to your GitHub account to further develop your contribution. Please use the latest commit version.
|
||||||
* Please, submit one Pull Request for one new feature or proposal. This will ease the analysis and the final merge if accepted.
|
* Please, submit one pull request per new feature or proposal. This will ease the analysis and the final merge if accepted.
|
||||||
|
|
||||||
|
## 3. Vulnerabilities
|
||||||
|
|
||||||
|
To report a **vulnerability**, please refer to the [SECURITY.md](./SECURITY.md) file for instructions.
|
||||||
|
|
||||||
|
## 4. Support requests and miscellaneous
|
||||||
|
|
||||||
|
For support requests or any other topics not related to the content of this repository, you can submit a post to the **ST Community** on the appropriate topic [page](https://community.st.com/s/topiccatalog).
|
||||||
|
|||||||
@@ -71,6 +71,7 @@ extern "C" {
|
|||||||
#define USB_DEVICE_CLASS_AUDIO 0x01U
|
#define USB_DEVICE_CLASS_AUDIO 0x01U
|
||||||
#define AUDIO_SUBCLASS_AUDIOCONTROL 0x01U
|
#define AUDIO_SUBCLASS_AUDIOCONTROL 0x01U
|
||||||
#define AUDIO_SUBCLASS_AUDIOSTREAMING 0x02U
|
#define AUDIO_SUBCLASS_AUDIOSTREAMING 0x02U
|
||||||
|
#define AUDIO_SUBCLASS_MIDISTREAMING 0x03U
|
||||||
#define AUDIO_PROTOCOL_UNDEFINED 0x00U
|
#define AUDIO_PROTOCOL_UNDEFINED 0x00U
|
||||||
#define AUDIO_STREAMING_GENERAL 0x01U
|
#define AUDIO_STREAMING_GENERAL 0x01U
|
||||||
#define AUDIO_STREAMING_FORMAT_TYPE 0x02U
|
#define AUDIO_STREAMING_FORMAT_TYPE 0x02U
|
||||||
|
|||||||
@@ -162,6 +162,9 @@ USBD_ClassTypeDef USBD_AUDIO =
|
|||||||
USBD_AUDIO_GetCfgDesc,
|
USBD_AUDIO_GetCfgDesc,
|
||||||
USBD_AUDIO_GetDeviceQualifierDesc,
|
USBD_AUDIO_GetDeviceQualifierDesc,
|
||||||
#endif /* USE_USBD_COMPOSITE */
|
#endif /* USE_USBD_COMPOSITE */
|
||||||
|
#if (USBD_SUPPORT_USER_STRING_DESC == 1U)
|
||||||
|
NULL,
|
||||||
|
#endif /* USBD_SUPPORT_USER_STRING_DESC */
|
||||||
};
|
};
|
||||||
|
|
||||||
#ifndef USE_USBD_COMPOSITE
|
#ifndef USE_USBD_COMPOSITE
|
||||||
|
|||||||
@@ -60,10 +60,12 @@ extern "C" {
|
|||||||
#define CCID_CMD_FS_BINTERVAL 0x10U
|
#define CCID_CMD_FS_BINTERVAL 0x10U
|
||||||
#endif /* CCID_CMD_FS_BINTERVAL */
|
#endif /* CCID_CMD_FS_BINTERVAL */
|
||||||
|
|
||||||
|
#ifndef CCID_CMD_PACKET_SIZE
|
||||||
|
#define CCID_CMD_PACKET_SIZE 8U /* Control Endpoint Packet size */
|
||||||
|
#endif /* CCID_CMD_PACKET_SIZE */
|
||||||
|
|
||||||
#define CCID_DATA_HS_MAX_PACKET_SIZE 512U /* Endpoint IN & OUT Packet size */
|
#define CCID_DATA_HS_MAX_PACKET_SIZE 512U /* Endpoint IN & OUT Packet size */
|
||||||
#define CCID_DATA_FS_MAX_PACKET_SIZE 64U /* Endpoint IN & OUT Packet size */
|
#define CCID_DATA_FS_MAX_PACKET_SIZE 64U /* Endpoint IN & OUT Packet size */
|
||||||
#define CCID_CMD_PACKET_SIZE 8U /* Control Endpoint Packet size */
|
|
||||||
|
|
||||||
#define USB_CCID_CONFIG_DESC_SIZ 93U
|
#define USB_CCID_CONFIG_DESC_SIZ 93U
|
||||||
#define CCID_DATA_HS_IN_PACKET_SIZE CCID_DATA_HS_MAX_PACKET_SIZE
|
#define CCID_DATA_HS_IN_PACKET_SIZE CCID_DATA_HS_MAX_PACKET_SIZE
|
||||||
|
|||||||
@@ -129,6 +129,9 @@ USBD_ClassTypeDef USBD_CCID =
|
|||||||
USBD_CCID_GetOtherSpeedCfgDesc,
|
USBD_CCID_GetOtherSpeedCfgDesc,
|
||||||
USBD_CCID_GetDeviceQualifierDescriptor,
|
USBD_CCID_GetDeviceQualifierDescriptor,
|
||||||
#endif /* USE_USBD_COMPOSITE */
|
#endif /* USE_USBD_COMPOSITE */
|
||||||
|
#if (USBD_SUPPORT_USER_STRING_DESC == 1U)
|
||||||
|
NULL,
|
||||||
|
#endif /* USBD_SUPPORT_USER_STRING_DESC */
|
||||||
};
|
};
|
||||||
|
|
||||||
#ifndef USE_USBD_COMPOSITE
|
#ifndef USE_USBD_COMPOSITE
|
||||||
|
|||||||
@@ -42,10 +42,10 @@ static void CCID_UpdateCommandStatus(USBD_HandleTypeDef *pdev, uint8_t cmd_stat
|
|||||||
uint8_t PC_to_RDR_IccPowerOn(USBD_HandleTypeDef *pdev)
|
uint8_t PC_to_RDR_IccPowerOn(USBD_HandleTypeDef *pdev)
|
||||||
{
|
{
|
||||||
/* Apply the ICC VCC
|
/* Apply the ICC VCC
|
||||||
Fills the Response buffer with ICC ATR
|
* Fills the Response buffer with ICC ATR
|
||||||
This Command is returned with RDR_to_PC_DataBlock();
|
* This Command is returned with RDR_to_PC_DataBlock();
|
||||||
*/
|
*/
|
||||||
USBD_CCID_HandleTypeDef *hccid = (USBD_CCID_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId];
|
USBD_CCID_HandleTypeDef *hccid = (USBD_CCID_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId];
|
||||||
uint8_t voltage;
|
uint8_t voltage;
|
||||||
uint8_t sc_voltage = 0U;
|
uint8_t sc_voltage = 0U;
|
||||||
uint8_t index;
|
uint8_t index;
|
||||||
|
|||||||
@@ -58,10 +58,13 @@ extern "C" {
|
|||||||
#define CDC_FS_BINTERVAL 0x10U
|
#define CDC_FS_BINTERVAL 0x10U
|
||||||
#endif /* CDC_FS_BINTERVAL */
|
#endif /* CDC_FS_BINTERVAL */
|
||||||
|
|
||||||
|
#ifndef CDC_CMD_PACKET_SIZE
|
||||||
|
#define CDC_CMD_PACKET_SIZE 8U /* Control Endpoint Packet size */
|
||||||
|
#endif /* CDC_CMD_PACKET_SIZE */
|
||||||
|
|
||||||
/* CDC Endpoints parameters: you can fine tune these values depending on the needed baudrates and performance. */
|
/* CDC Endpoints parameters: you can fine tune these values depending on the needed baudrates and performance. */
|
||||||
#define CDC_DATA_HS_MAX_PACKET_SIZE 512U /* Endpoint IN & OUT Packet size */
|
#define CDC_DATA_HS_MAX_PACKET_SIZE 512U /* Endpoint IN & OUT Packet size */
|
||||||
#define CDC_DATA_FS_MAX_PACKET_SIZE 64U /* Endpoint IN & OUT Packet size */
|
#define CDC_DATA_FS_MAX_PACKET_SIZE 64U /* Endpoint IN & OUT Packet size */
|
||||||
#define CDC_CMD_PACKET_SIZE 8U /* Control Endpoint Packet size */
|
|
||||||
|
|
||||||
#define USB_CDC_CONFIG_DESC_SIZ 67U
|
#define USB_CDC_CONFIG_DESC_SIZ 67U
|
||||||
#define CDC_DATA_HS_IN_PACKET_SIZE CDC_DATA_HS_MAX_PACKET_SIZE
|
#define CDC_DATA_HS_IN_PACKET_SIZE CDC_DATA_HS_MAX_PACKET_SIZE
|
||||||
|
|||||||
@@ -161,6 +161,9 @@ USBD_ClassTypeDef USBD_CDC =
|
|||||||
USBD_CDC_GetOtherSpeedCfgDesc,
|
USBD_CDC_GetOtherSpeedCfgDesc,
|
||||||
USBD_CDC_GetDeviceQualifierDescriptor,
|
USBD_CDC_GetDeviceQualifierDescriptor,
|
||||||
#endif /* USE_USBD_COMPOSITE */
|
#endif /* USE_USBD_COMPOSITE */
|
||||||
|
#if (USBD_SUPPORT_USER_STRING_DESC == 1U)
|
||||||
|
NULL,
|
||||||
|
#endif /* USBD_SUPPORT_USER_STRING_DESC */
|
||||||
};
|
};
|
||||||
|
|
||||||
#ifndef USE_USBD_COMPOSITE
|
#ifndef USE_USBD_COMPOSITE
|
||||||
|
|||||||
@@ -68,6 +68,10 @@ extern "C" {
|
|||||||
#define CDC_ECM_FS_BINTERVAL 0x10U
|
#define CDC_ECM_FS_BINTERVAL 0x10U
|
||||||
#endif /* CDC_ECM_FS_BINTERVAL */
|
#endif /* CDC_ECM_FS_BINTERVAL */
|
||||||
|
|
||||||
|
#ifndef CDC_ECM_CMD_PACKET_SIZE
|
||||||
|
#define CDC_ECM_CMD_PACKET_SIZE 16U /* Control Endpoint Packet size */
|
||||||
|
#endif /* CDC_ECM_CMD_PACKET_SIZE */
|
||||||
|
|
||||||
#ifndef USBD_SUPPORT_USER_STRING_DESC
|
#ifndef USBD_SUPPORT_USER_STRING_DESC
|
||||||
#define USBD_SUPPORT_USER_STRING_DESC 1U
|
#define USBD_SUPPORT_USER_STRING_DESC 1U
|
||||||
#endif /* USBD_SUPPORT_USER_STRING_DESC */
|
#endif /* USBD_SUPPORT_USER_STRING_DESC */
|
||||||
@@ -75,7 +79,6 @@ extern "C" {
|
|||||||
/* CDC_ECM Endpoints parameters: you can fine tune these values depending on the needed baudrates and performance. */
|
/* CDC_ECM Endpoints parameters: you can fine tune these values depending on the needed baudrates and performance. */
|
||||||
#define CDC_ECM_DATA_HS_MAX_PACKET_SIZE 512U /* Endpoint IN & OUT Packet size */
|
#define CDC_ECM_DATA_HS_MAX_PACKET_SIZE 512U /* Endpoint IN & OUT Packet size */
|
||||||
#define CDC_ECM_DATA_FS_MAX_PACKET_SIZE 64U /* Endpoint IN & OUT Packet size */
|
#define CDC_ECM_DATA_FS_MAX_PACKET_SIZE 64U /* Endpoint IN & OUT Packet size */
|
||||||
#define CDC_ECM_CMD_PACKET_SIZE 16U /* Control Endpoint Packet size */
|
|
||||||
|
|
||||||
#define CDC_ECM_CONFIG_DESC_SIZ 79U
|
#define CDC_ECM_CONFIG_DESC_SIZ 79U
|
||||||
|
|
||||||
|
|||||||
@@ -66,12 +66,14 @@ extern "C" {
|
|||||||
#define CDC_RNDIS_FS_BINTERVAL 0x10U
|
#define CDC_RNDIS_FS_BINTERVAL 0x10U
|
||||||
#endif /* CDC_RNDIS_FS_BINTERVAL */
|
#endif /* CDC_RNDIS_FS_BINTERVAL */
|
||||||
|
|
||||||
|
#ifndef CDC_RNDIS_CMD_PACKET_SIZE
|
||||||
|
#define CDC_RNDIS_CMD_PACKET_SIZE 16U /* Control Endpoint Packet size */
|
||||||
|
#endif /* CDC_RNDIS_CMD_PACKET_SIZE */
|
||||||
|
|
||||||
/* CDC_RNDIS Endpoints parameters: you can fine tune these values
|
/* CDC_RNDIS Endpoints parameters: you can fine tune these values
|
||||||
depending on the needed baudrates and performance. */
|
depending on the needed baudrates and performance. */
|
||||||
#define CDC_RNDIS_DATA_HS_MAX_PACKET_SIZE 512U /* Endpoint IN & OUT Packet size */
|
#define CDC_RNDIS_DATA_HS_MAX_PACKET_SIZE 512U /* Endpoint IN & OUT Packet size */
|
||||||
#define CDC_RNDIS_DATA_FS_MAX_PACKET_SIZE 64U /* Endpoint IN & OUT Packet size */
|
#define CDC_RNDIS_DATA_FS_MAX_PACKET_SIZE 64U /* Endpoint IN & OUT Packet size */
|
||||||
#define CDC_RNDIS_CMD_PACKET_SIZE 16U /* Control Endpoint Packet size */
|
|
||||||
|
|
||||||
#define CDC_RNDIS_CONFIG_DESC_SIZ 75U
|
#define CDC_RNDIS_CONFIG_DESC_SIZ 75U
|
||||||
#define CDC_RNDIS_DATA_HS_IN_PACKET_SIZE CDC_RNDIS_DATA_HS_MAX_PACKET_SIZE
|
#define CDC_RNDIS_DATA_HS_IN_PACKET_SIZE CDC_RNDIS_DATA_HS_MAX_PACKET_SIZE
|
||||||
|
|||||||
@@ -1452,7 +1452,7 @@ static uint8_t USBD_CDC_RNDIS_ProcessQueryMsg(USBD_HandleTypeDef *pdev,
|
|||||||
|
|
||||||
case OID_GEN_CURRENT_PACKET_FILTER:
|
case OID_GEN_CURRENT_PACKET_FILTER:
|
||||||
QueryResponse->InfoBufLength = sizeof(uint32_t);
|
QueryResponse->InfoBufLength = sizeof(uint32_t);
|
||||||
QueryResponse->InfoBuf[0] = 0xFFFFFFU; /* USBD_CDC_RNDIS_DEVICE.packetFilter; */
|
QueryResponse->InfoBuf[0] = 0xFFFFFFU; /* USBD_CDC_RNDIS_DEVICE.packetFilter */
|
||||||
QueryResponse->Status = CDC_RNDIS_STATUS_SUCCESS;
|
QueryResponse->Status = CDC_RNDIS_STATUS_SUCCESS;
|
||||||
break;
|
break;
|
||||||
|
|
||||||
|
|||||||
@@ -511,11 +511,11 @@ uint8_t USBD_CMPSIT_AddToConfDesc(USBD_HandleTypeDef *pdev)
|
|||||||
|
|
||||||
/* Set IN endpoint slot */
|
/* Set IN endpoint slot */
|
||||||
iEp = pdev->tclasslist[pdev->classId].EpAdd[0];
|
iEp = pdev->tclasslist[pdev->classId].EpAdd[0];
|
||||||
USBD_CMPSIT_AssignEp(pdev, iEp, USBD_EP_TYPE_INTR, pdev->tclasslist[pdev->classId].CurrPcktSze);
|
USBD_CMPSIT_AssignEp(pdev, iEp, USBD_EP_TYPE_INTR,CUSTOM_HID_EPIN_SIZE);
|
||||||
|
|
||||||
/* Set OUT endpoint slot */
|
/* Set OUT endpoint slot */
|
||||||
iEp = pdev->tclasslist[pdev->classId].EpAdd[1];
|
iEp = pdev->tclasslist[pdev->classId].EpAdd[1];
|
||||||
USBD_CMPSIT_AssignEp(pdev, iEp, USBD_EP_TYPE_INTR, pdev->tclasslist[pdev->classId].CurrPcktSze);
|
USBD_CMPSIT_AssignEp(pdev, iEp, USBD_EP_TYPE_INTR, CUSTOM_HID_EPOUT_SIZE);
|
||||||
|
|
||||||
/* Configure and Append the Descriptor */
|
/* Configure and Append the Descriptor */
|
||||||
USBD_CMPSIT_CUSTOMHIDDesc(pdev, (uint32_t)pCmpstFSConfDesc, &CurrFSConfDescSz, (uint8_t)USBD_SPEED_FULL);
|
USBD_CMPSIT_CUSTOMHIDDesc(pdev, (uint32_t)pCmpstFSConfDesc, &CurrFSConfDescSz, (uint8_t)USBD_SPEED_FULL);
|
||||||
@@ -1393,7 +1393,12 @@ static void USBD_CMPSIT_CUSTOMHIDDesc(USBD_HandleTypeDef *pdev, uint32_t pConf,
|
|||||||
pDesc->bCountryCode = 0x00U;
|
pDesc->bCountryCode = 0x00U;
|
||||||
pDesc->bNumDescriptors = 0x01U;
|
pDesc->bNumDescriptors = 0x01U;
|
||||||
pDesc->bDescriptorType = 0x22U;
|
pDesc->bDescriptorType = 0x22U;
|
||||||
|
#ifdef USBD_CUSTOMHID_REPORT_DESC_SIZE_ENABLED
|
||||||
|
pDesc->wItemLength = ((USBD_CUSTOM_HID_ItfTypeDef *)pdev->pUserData[pdev->classId])->wReportDescLen;
|
||||||
|
#else
|
||||||
pDesc->wItemLength = USBD_CUSTOM_HID_REPORT_DESC_SIZE;
|
pDesc->wItemLength = USBD_CUSTOM_HID_REPORT_DESC_SIZE;
|
||||||
|
#endif /* USBD_CUSTOMHID_REPORT_DESC_SIZE_ENABLED */
|
||||||
|
|
||||||
*Sze += (uint32_t)sizeof(USBD_DescTypeDef);
|
*Sze += (uint32_t)sizeof(USBD_DescTypeDef);
|
||||||
|
|
||||||
/* Descriptor of Custom HID endpoints */
|
/* Descriptor of Custom HID endpoints */
|
||||||
@@ -1403,7 +1408,7 @@ static void USBD_CMPSIT_CUSTOMHIDDesc(USBD_HandleTypeDef *pdev, uint32_t pConf,
|
|||||||
|
|
||||||
/* Append Endpoint descriptor to Configuration descriptor */
|
/* Append Endpoint descriptor to Configuration descriptor */
|
||||||
__USBD_CMPSIT_SET_EP(pdev->tclasslist[pdev->classId].Eps[1].add, \
|
__USBD_CMPSIT_SET_EP(pdev->tclasslist[pdev->classId].Eps[1].add, \
|
||||||
USBD_EP_TYPE_INTR, CUSTOM_HID_EPIN_SIZE, CUSTOM_HID_HS_BINTERVAL, CUSTOM_HID_FS_BINTERVAL);
|
USBD_EP_TYPE_INTR, CUSTOM_HID_EPOUT_SIZE, CUSTOM_HID_HS_BINTERVAL, CUSTOM_HID_FS_BINTERVAL);
|
||||||
|
|
||||||
/* Update Config Descriptor and IAD descriptor */
|
/* Update Config Descriptor and IAD descriptor */
|
||||||
((USBD_ConfigDescTypeDef *)pConf)->bNumInterfaces += 1U;
|
((USBD_ConfigDescTypeDef *)pConf)->bNumInterfaces += 1U;
|
||||||
|
|||||||
@@ -103,9 +103,16 @@ typedef enum
|
|||||||
typedef struct _USBD_CUSTOM_HID_Itf
|
typedef struct _USBD_CUSTOM_HID_Itf
|
||||||
{
|
{
|
||||||
uint8_t *pReport;
|
uint8_t *pReport;
|
||||||
|
#ifdef USBD_CUSTOMHID_REPORT_DESC_SIZE_ENABLED
|
||||||
|
uint16_t wReportDescLen;
|
||||||
|
#endif /* USBD_CUSTOMHID_REPORT_DESC_SIZE_ENABLED */
|
||||||
int8_t (* Init)(void);
|
int8_t (* Init)(void);
|
||||||
int8_t (* DeInit)(void);
|
int8_t (* DeInit)(void);
|
||||||
|
#ifdef USBD_CUSTOMHID_REPORT_BUFFER_EVENT_ENABLED
|
||||||
|
int8_t (* OutEvent)(uint8_t *report_buffer);
|
||||||
|
#else
|
||||||
int8_t (* OutEvent)(uint8_t event_idx, uint8_t state);
|
int8_t (* OutEvent)(uint8_t event_idx, uint8_t state);
|
||||||
|
#endif /* USBD_CUSTOMHID_REPORT_BUFFER_EVENT_ENABLED */
|
||||||
#ifdef USBD_CUSTOMHID_CTRL_REQ_COMPLETE_CALLBACK_ENABLED
|
#ifdef USBD_CUSTOMHID_CTRL_REQ_COMPLETE_CALLBACK_ENABLED
|
||||||
int8_t (* CtrlReqComplete)(uint8_t request, uint16_t wLength);
|
int8_t (* CtrlReqComplete)(uint8_t request, uint16_t wLength);
|
||||||
#endif /* USBD_CUSTOMHID_CTRL_REQ_COMPLETE_CALLBACK_ENABLED */
|
#endif /* USBD_CUSTOMHID_CTRL_REQ_COMPLETE_CALLBACK_ENABLED */
|
||||||
|
|||||||
@@ -128,6 +128,9 @@ USBD_ClassTypeDef USBD_CUSTOM_HID =
|
|||||||
USBD_CUSTOM_HID_GetOtherSpeedCfgDesc,
|
USBD_CUSTOM_HID_GetOtherSpeedCfgDesc,
|
||||||
USBD_CUSTOM_HID_GetDeviceQualifierDesc,
|
USBD_CUSTOM_HID_GetDeviceQualifierDesc,
|
||||||
#endif /* USE_USBD_COMPOSITE */
|
#endif /* USE_USBD_COMPOSITE */
|
||||||
|
#if (USBD_SUPPORT_USER_STRING_DESC == 1U)
|
||||||
|
NULL,
|
||||||
|
#endif /* USBD_SUPPORT_USER_STRING_DESC */
|
||||||
};
|
};
|
||||||
|
|
||||||
#ifndef USE_USBD_COMPOSITE
|
#ifndef USE_USBD_COMPOSITE
|
||||||
@@ -284,6 +287,11 @@ static uint8_t USBD_CUSTOM_HID_Init(USBD_HandleTypeDef *pdev, uint8_t cfgidx)
|
|||||||
|
|
||||||
pdev->ep_in[CUSTOMHIDInEpAdd & 0xFU].is_used = 1U;
|
pdev->ep_in[CUSTOMHIDInEpAdd & 0xFU].is_used = 1U;
|
||||||
|
|
||||||
|
if (USBD_CUSTOMHID_OUTREPORT_BUF_SIZE < CUSTOM_HID_EPOUT_SIZE)
|
||||||
|
{
|
||||||
|
return (uint8_t)USBD_FAIL;
|
||||||
|
}
|
||||||
|
|
||||||
/* Open EP OUT */
|
/* Open EP OUT */
|
||||||
(void)USBD_LL_OpenEP(pdev, CUSTOMHIDOutEpAdd, USBD_EP_TYPE_INTR,
|
(void)USBD_LL_OpenEP(pdev, CUSTOMHIDOutEpAdd, USBD_EP_TYPE_INTR,
|
||||||
CUSTOM_HID_EPOUT_SIZE);
|
CUSTOM_HID_EPOUT_SIZE);
|
||||||
@@ -397,9 +405,17 @@ static uint8_t USBD_CUSTOM_HID_Setup(USBD_HandleTypeDef *pdev,
|
|||||||
}
|
}
|
||||||
#endif /* USBD_CUSTOMHID_CTRL_REQ_COMPLETE_CALLBACK_ENABLED */
|
#endif /* USBD_CUSTOMHID_CTRL_REQ_COMPLETE_CALLBACK_ENABLED */
|
||||||
#ifndef USBD_CUSTOMHID_EP0_OUT_PREPARE_RECEIVE_DISABLED
|
#ifndef USBD_CUSTOMHID_EP0_OUT_PREPARE_RECEIVE_DISABLED
|
||||||
|
|
||||||
|
if (req->wLength > USBD_CUSTOMHID_OUTREPORT_BUF_SIZE)
|
||||||
|
{
|
||||||
|
/* Stall EP0 */
|
||||||
|
USBD_CtlError(pdev, req);
|
||||||
|
return USBD_FAIL;
|
||||||
|
}
|
||||||
|
|
||||||
hhid->IsReportAvailable = 1U;
|
hhid->IsReportAvailable = 1U;
|
||||||
(void)USBD_CtlPrepareRx(pdev, hhid->Report_buf,
|
|
||||||
MIN(req->wLength, USBD_CUSTOMHID_OUTREPORT_BUF_SIZE));
|
(void)USBD_CtlPrepareRx(pdev, hhid->Report_buf, req->wLength);
|
||||||
#endif /* USBD_CUSTOMHID_EP0_OUT_PREPARE_RECEIVE_DISABLED */
|
#endif /* USBD_CUSTOMHID_EP0_OUT_PREPARE_RECEIVE_DISABLED */
|
||||||
break;
|
break;
|
||||||
#ifdef USBD_CUSTOMHID_CTRL_REQ_GET_REPORT_ENABLED
|
#ifdef USBD_CUSTOMHID_CTRL_REQ_GET_REPORT_ENABLED
|
||||||
@@ -701,8 +717,13 @@ static uint8_t USBD_CUSTOM_HID_DataOut(USBD_HandleTypeDef *pdev, uint8_t epnum)
|
|||||||
|
|
||||||
/* USB data will be immediately processed, this allow next USB traffic being
|
/* USB data will be immediately processed, this allow next USB traffic being
|
||||||
NAKed till the end of the application processing */
|
NAKed till the end of the application processing */
|
||||||
|
|
||||||
|
#ifdef USBD_CUSTOMHID_REPORT_BUFFER_EVENT_ENABLED
|
||||||
|
((USBD_CUSTOM_HID_ItfTypeDef *)pdev->pUserData[pdev->classId])->OutEvent(hhid->Report_buf);
|
||||||
|
#else
|
||||||
((USBD_CUSTOM_HID_ItfTypeDef *)pdev->pUserData[pdev->classId])->OutEvent(hhid->Report_buf[0],
|
((USBD_CUSTOM_HID_ItfTypeDef *)pdev->pUserData[pdev->classId])->OutEvent(hhid->Report_buf[0],
|
||||||
hhid->Report_buf[1]);
|
hhid->Report_buf[1]);
|
||||||
|
#endif /* USBD_CUSTOMHID_REPORT_BUFFER_EVENT_ENABLED */
|
||||||
|
|
||||||
return (uint8_t)USBD_OK;
|
return (uint8_t)USBD_OK;
|
||||||
}
|
}
|
||||||
@@ -755,8 +776,12 @@ static uint8_t USBD_CUSTOM_HID_EP0_RxReady(USBD_HandleTypeDef *pdev)
|
|||||||
|
|
||||||
if (hhid->IsReportAvailable == 1U)
|
if (hhid->IsReportAvailable == 1U)
|
||||||
{
|
{
|
||||||
|
#ifdef USBD_CUSTOMHID_REPORT_BUFFER_EVENT_ENABLED
|
||||||
|
((USBD_CUSTOM_HID_ItfTypeDef *)pdev->pUserData[pdev->classId])->OutEvent(hhid->Report_buf);
|
||||||
|
#else
|
||||||
((USBD_CUSTOM_HID_ItfTypeDef *)pdev->pUserData[pdev->classId])->OutEvent(hhid->Report_buf[0],
|
((USBD_CUSTOM_HID_ItfTypeDef *)pdev->pUserData[pdev->classId])->OutEvent(hhid->Report_buf[0],
|
||||||
hhid->Report_buf[1]);
|
hhid->Report_buf[1]);
|
||||||
|
#endif /* USBD_CUSTOMHID_REPORT_BUFFER_EVENT_ENABLED */
|
||||||
hhid->IsReportAvailable = 0U;
|
hhid->IsReportAvailable = 0U;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -32,7 +32,11 @@ EndBSPDependencies */
|
|||||||
|
|
||||||
static int8_t TEMPLATE_CUSTOM_HID_Init(void);
|
static int8_t TEMPLATE_CUSTOM_HID_Init(void);
|
||||||
static int8_t TEMPLATE_CUSTOM_HID_DeInit(void);
|
static int8_t TEMPLATE_CUSTOM_HID_DeInit(void);
|
||||||
|
#ifdef USBD_CUSTOMHID_REPORT_BUFFER_EVENT_ENABLED
|
||||||
|
static int8_t TEMPLATE_CUSTOM_HID_OutEvent(uint8_t *report_buffer);
|
||||||
|
#else
|
||||||
static int8_t TEMPLATE_CUSTOM_HID_OutEvent(uint8_t event_idx, uint8_t state);
|
static int8_t TEMPLATE_CUSTOM_HID_OutEvent(uint8_t event_idx, uint8_t state);
|
||||||
|
#endif /* USBD_CUSTOMHID_REPORT_BUFFER_EVENT_ENABLED */
|
||||||
|
|
||||||
#ifdef USBD_CUSTOMHID_CTRL_REQ_COMPLETE_CALLBACK_ENABLED
|
#ifdef USBD_CUSTOMHID_CTRL_REQ_COMPLETE_CALLBACK_ENABLED
|
||||||
static int8_t TEMPLATE_CUSTOM_HID_CtrlReqComplete(uint8_t request, uint16_t wLength);
|
static int8_t TEMPLATE_CUSTOM_HID_CtrlReqComplete(uint8_t request, uint16_t wLength);
|
||||||
@@ -49,6 +53,9 @@ __ALIGN_BEGIN static uint8_t TEMPLATE_CUSTOM_HID_ReportDesc[USBD_CUSTOM_HID_REPO
|
|||||||
USBD_CUSTOM_HID_ItfTypeDef USBD_CustomHID_template_fops =
|
USBD_CUSTOM_HID_ItfTypeDef USBD_CustomHID_template_fops =
|
||||||
{
|
{
|
||||||
TEMPLATE_CUSTOM_HID_ReportDesc,
|
TEMPLATE_CUSTOM_HID_ReportDesc,
|
||||||
|
#ifdef USBD_CUSTOMHID_REPORT_DESC_SIZE_ENABLED
|
||||||
|
USBD_CUSTOM_HID_REPORT_DESC_SIZE,
|
||||||
|
#endif /* USBD_CUSTOMHID_REPORT_DESC_SIZE_ENABLED */
|
||||||
TEMPLATE_CUSTOM_HID_Init,
|
TEMPLATE_CUSTOM_HID_Init,
|
||||||
TEMPLATE_CUSTOM_HID_DeInit,
|
TEMPLATE_CUSTOM_HID_DeInit,
|
||||||
TEMPLATE_CUSTOM_HID_OutEvent,
|
TEMPLATE_CUSTOM_HID_OutEvent,
|
||||||
@@ -95,10 +102,17 @@ static int8_t TEMPLATE_CUSTOM_HID_DeInit(void)
|
|||||||
* @param state: event state
|
* @param state: event state
|
||||||
* @retval Result of the operation: USBD_OK if all operations are OK else USBD_FAIL
|
* @retval Result of the operation: USBD_OK if all operations are OK else USBD_FAIL
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
#ifdef USBD_CUSTOMHID_REPORT_BUFFER_EVENT_ENABLED
|
||||||
|
static int8_t TEMPLATE_CUSTOM_HID_OutEvent(uint8_t *report_buffer)
|
||||||
|
{
|
||||||
|
UNUSED(report_buffer);
|
||||||
|
#else
|
||||||
static int8_t TEMPLATE_CUSTOM_HID_OutEvent(uint8_t event_idx, uint8_t state)
|
static int8_t TEMPLATE_CUSTOM_HID_OutEvent(uint8_t event_idx, uint8_t state)
|
||||||
{
|
{
|
||||||
UNUSED(event_idx);
|
UNUSED(event_idx);
|
||||||
UNUSED(state);
|
UNUSED(state);
|
||||||
|
#endif /* USBD_CUSTOMHID_REPORT_BUFFER_EVENT_ENABLED */
|
||||||
|
|
||||||
/* Start next USB packet transfer once data processing is completed */
|
/* Start next USB packet transfer once data processing is completed */
|
||||||
if (USBD_CUSTOM_HID_ReceivePacket(&USBD_Device) != (uint8_t)USBD_OK)
|
if (USBD_CUSTOM_HID_ReceivePacket(&USBD_Device) != (uint8_t)USBD_OK)
|
||||||
@@ -151,7 +165,7 @@ static int8_t TEMPLATE_CUSTOM_HID_CtrlReqComplete(uint8_t request, uint16_t wLen
|
|||||||
static uint8_t *TEMPLATE_CUSTOM_HID_GetReport(uint16_t *ReportLength)
|
static uint8_t *TEMPLATE_CUSTOM_HID_GetReport(uint16_t *ReportLength)
|
||||||
{
|
{
|
||||||
UNUSED(ReportLength);
|
UNUSED(ReportLength);
|
||||||
uint8_t *pbuff;
|
uint8_t *pbuff = NULL;
|
||||||
|
|
||||||
return (pbuff);
|
return (pbuff);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -940,6 +940,12 @@ static void DFU_Download(USBD_HandleTypeDef *pdev, USBD_SetupReqTypedef *req)
|
|||||||
if ((hdfu->dev_state == DFU_STATE_DNLOAD_IDLE) || (hdfu->dev_state == DFU_STATE_IDLE))
|
if ((hdfu->dev_state == DFU_STATE_DNLOAD_IDLE) || (hdfu->dev_state == DFU_STATE_IDLE))
|
||||||
{
|
{
|
||||||
#if (USBD_DFU_VENDOR_CHECK_ENABLED == 1U)
|
#if (USBD_DFU_VENDOR_CHECK_ENABLED == 1U)
|
||||||
|
/* Fill the buffer to be checked with the address stored in data buffer */
|
||||||
|
hdfu->buffer.d8[1] = (uint8_t)(hdfu->data_ptr & 0xFFU);
|
||||||
|
hdfu->buffer.d8[2] = (uint8_t)((hdfu->data_ptr & 0xFF00U) >> 8U);
|
||||||
|
hdfu->buffer.d8[3] = (uint8_t)((hdfu->data_ptr & 0xFF0000U) >> 16U);
|
||||||
|
hdfu->buffer.d8[4] = (uint8_t)((hdfu->data_ptr & 0xFF000000U) >> 24U);
|
||||||
|
|
||||||
if (DfuInterface->VendorCheck(hdfu->buffer.d8, IS_DFU_SETADDRESSPOINTER, &VendorStatus) != USBD_OK)
|
if (DfuInterface->VendorCheck(hdfu->buffer.d8, IS_DFU_SETADDRESSPOINTER, &VendorStatus) != USBD_OK)
|
||||||
{
|
{
|
||||||
/* Update the state machine */
|
/* Update the state machine */
|
||||||
|
|||||||
@@ -128,6 +128,9 @@ USBD_ClassTypeDef USBD_HID =
|
|||||||
USBD_HID_GetOtherSpeedCfgDesc,
|
USBD_HID_GetOtherSpeedCfgDesc,
|
||||||
USBD_HID_GetDeviceQualifierDesc,
|
USBD_HID_GetDeviceQualifierDesc,
|
||||||
#endif /* USE_USBD_COMPOSITE */
|
#endif /* USE_USBD_COMPOSITE */
|
||||||
|
#if (USBD_SUPPORT_USER_STRING_DESC == 1U)
|
||||||
|
NULL,
|
||||||
|
#endif /* USBD_SUPPORT_USER_STRING_DESC */
|
||||||
};
|
};
|
||||||
|
|
||||||
#ifndef USE_USBD_COMPOSITE
|
#ifndef USE_USBD_COMPOSITE
|
||||||
|
|||||||
@@ -62,6 +62,10 @@ extern "C" {
|
|||||||
#define MSC_EPOUT_ADDR 0x01U
|
#define MSC_EPOUT_ADDR 0x01U
|
||||||
#endif /* MSC_EPOUT_ADDR */
|
#endif /* MSC_EPOUT_ADDR */
|
||||||
|
|
||||||
|
#ifndef MSC_BOT_MAX_LUN
|
||||||
|
#define MSC_BOT_MAX_LUN 0x2U
|
||||||
|
#endif /* MSC_BOT_MAX_LUN */
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @}
|
* @}
|
||||||
*/
|
*/
|
||||||
@@ -82,6 +86,13 @@ typedef struct _USBD_STORAGE
|
|||||||
|
|
||||||
} USBD_StorageTypeDef;
|
} USBD_StorageTypeDef;
|
||||||
|
|
||||||
|
typedef struct
|
||||||
|
{
|
||||||
|
uint16_t size;
|
||||||
|
uint32_t nbr;
|
||||||
|
uint32_t addr;
|
||||||
|
uint32_t len;
|
||||||
|
} USBD_MSC_BOT_LUN_TypeDef;
|
||||||
|
|
||||||
typedef struct
|
typedef struct
|
||||||
{
|
{
|
||||||
@@ -99,11 +110,7 @@ typedef struct
|
|||||||
uint8_t scsi_sense_tail;
|
uint8_t scsi_sense_tail;
|
||||||
uint8_t scsi_medium_state;
|
uint8_t scsi_medium_state;
|
||||||
|
|
||||||
uint16_t scsi_blk_size;
|
USBD_MSC_BOT_LUN_TypeDef scsi_blk[MSC_BOT_MAX_LUN];
|
||||||
uint32_t scsi_blk_nbr;
|
|
||||||
|
|
||||||
uint32_t scsi_blk_addr;
|
|
||||||
uint32_t scsi_blk_len;
|
|
||||||
} USBD_MSC_BOT_HandleTypeDef;
|
} USBD_MSC_BOT_HandleTypeDef;
|
||||||
|
|
||||||
/* Structure for MSC process */
|
/* Structure for MSC process */
|
||||||
|
|||||||
@@ -44,6 +44,8 @@ extern "C" {
|
|||||||
#define LENGTH_INQUIRY_PAGE00 0x06U
|
#define LENGTH_INQUIRY_PAGE00 0x06U
|
||||||
#define LENGTH_INQUIRY_PAGE80 0x08U
|
#define LENGTH_INQUIRY_PAGE80 0x08U
|
||||||
#define LENGTH_FORMAT_CAPACITIES 0x14U
|
#define LENGTH_FORMAT_CAPACITIES 0x14U
|
||||||
|
#define DIAGNOSTIC_DATA_LEN 0x08U
|
||||||
|
#define LOG_PAGE_DATA_LEN 0x10U
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @}
|
* @}
|
||||||
@@ -74,6 +76,8 @@ extern uint8_t MSC_Page00_Inquiry_Data[LENGTH_INQUIRY_PAGE00];
|
|||||||
extern uint8_t MSC_Page80_Inquiry_Data[LENGTH_INQUIRY_PAGE80];
|
extern uint8_t MSC_Page80_Inquiry_Data[LENGTH_INQUIRY_PAGE80];
|
||||||
extern uint8_t MSC_Mode_Sense6_data[MODE_SENSE6_LEN];
|
extern uint8_t MSC_Mode_Sense6_data[MODE_SENSE6_LEN];
|
||||||
extern uint8_t MSC_Mode_Sense10_data[MODE_SENSE10_LEN];
|
extern uint8_t MSC_Mode_Sense10_data[MODE_SENSE10_LEN];
|
||||||
|
extern uint8_t MSC_Diagnostic_Data[DIAGNOSTIC_DATA_LEN];
|
||||||
|
extern uint8_t MSC_Log_Page_Data[LOG_PAGE_DATA_LEN];
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @}
|
* @}
|
||||||
|
|||||||
@@ -70,8 +70,10 @@ extern "C" {
|
|||||||
#define SCSI_VERIFY12 0xAFU
|
#define SCSI_VERIFY12 0xAFU
|
||||||
#define SCSI_VERIFY16 0x8FU
|
#define SCSI_VERIFY16 0x8FU
|
||||||
|
|
||||||
#define SCSI_SEND_DIAGNOSTIC 0x1DU
|
|
||||||
#define SCSI_READ_FORMAT_CAPACITIES 0x23U
|
#define SCSI_READ_FORMAT_CAPACITIES 0x23U
|
||||||
|
#define SCSI_RECEIVE_DIAGNOSTIC_RESULTS 0x1CU
|
||||||
|
#define SCSI_SEND_DIAGNOSTIC 0x1DU
|
||||||
|
#define SCSI_REPORT_LUNS 0xA0U
|
||||||
|
|
||||||
#define NO_SENSE 0U
|
#define NO_SENSE 0U
|
||||||
#define RECOVERED_ERROR 1U
|
#define RECOVERED_ERROR 1U
|
||||||
@@ -88,9 +90,8 @@ extern "C" {
|
|||||||
#define VOLUME_OVERFLOW 13U
|
#define VOLUME_OVERFLOW 13U
|
||||||
#define MISCOMPARE 14U
|
#define MISCOMPARE 14U
|
||||||
|
|
||||||
|
|
||||||
#define INVALID_CDB 0x20U
|
#define INVALID_CDB 0x20U
|
||||||
#define INVALID_FIELED_IN_COMMAND 0x24U
|
#define INVALID_FIELD_IN_COMMAND 0x24U
|
||||||
#define PARAMETER_LIST_LENGTH_ERROR 0x1AU
|
#define PARAMETER_LIST_LENGTH_ERROR 0x1AU
|
||||||
#define INVALID_FIELD_IN_PARAMETER_LIST 0x26U
|
#define INVALID_FIELD_IN_PARAMETER_LIST 0x26U
|
||||||
#define ADDRESS_OUT_OF_RANGE 0x21U
|
#define ADDRESS_OUT_OF_RANGE 0x21U
|
||||||
|
|||||||
@@ -125,6 +125,9 @@ USBD_ClassTypeDef USBD_MSC =
|
|||||||
USBD_MSC_GetOtherSpeedCfgDesc,
|
USBD_MSC_GetOtherSpeedCfgDesc,
|
||||||
USBD_MSC_GetDeviceQualifierDescriptor,
|
USBD_MSC_GetDeviceQualifierDescriptor,
|
||||||
#endif /* USE_USBD_COMPOSITE */
|
#endif /* USE_USBD_COMPOSITE */
|
||||||
|
#if (USBD_SUPPORT_USER_STRING_DESC == 1U)
|
||||||
|
NULL,
|
||||||
|
#endif /* USBD_SUPPORT_USER_STRING_DESC */
|
||||||
};
|
};
|
||||||
|
|
||||||
/* USB Mass storage device Configuration Descriptor */
|
/* USB Mass storage device Configuration Descriptor */
|
||||||
@@ -309,6 +312,7 @@ uint8_t USBD_MSC_Setup(USBD_HandleTypeDef *pdev, USBD_SetupReqTypedef *req)
|
|||||||
{
|
{
|
||||||
USBD_MSC_BOT_HandleTypeDef *hmsc = (USBD_MSC_BOT_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId];
|
USBD_MSC_BOT_HandleTypeDef *hmsc = (USBD_MSC_BOT_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId];
|
||||||
USBD_StatusTypeDef ret = USBD_OK;
|
USBD_StatusTypeDef ret = USBD_OK;
|
||||||
|
uint32_t max_lun;
|
||||||
uint16_t status_info = 0U;
|
uint16_t status_info = 0U;
|
||||||
|
|
||||||
#ifdef USE_USBD_COMPOSITE
|
#ifdef USE_USBD_COMPOSITE
|
||||||
@@ -332,7 +336,8 @@ uint8_t USBD_MSC_Setup(USBD_HandleTypeDef *pdev, USBD_SetupReqTypedef *req)
|
|||||||
if ((req->wValue == 0U) && (req->wLength == 1U) &&
|
if ((req->wValue == 0U) && (req->wLength == 1U) &&
|
||||||
((req->bmRequest & 0x80U) == 0x80U))
|
((req->bmRequest & 0x80U) == 0x80U))
|
||||||
{
|
{
|
||||||
hmsc->max_lun = (uint32_t)((USBD_StorageTypeDef *)pdev->pUserData[pdev->classId])->GetMaxLun();
|
max_lun = (uint32_t)((USBD_StorageTypeDef *)pdev->pUserData[pdev->classId])->GetMaxLun();
|
||||||
|
hmsc->max_lun = (max_lun > MSC_BOT_MAX_LUN) ? MSC_BOT_MAX_LUN : max_lun;
|
||||||
(void)USBD_CtlSendData(pdev, (uint8_t *)&hmsc->max_lun, 1U);
|
(void)USBD_CtlSendData(pdev, (uint8_t *)&hmsc->max_lun, 1U);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
|
|||||||
@@ -273,7 +273,7 @@ static void MSC_BOT_CBW_Decode(USBD_HandleTypeDef *pdev)
|
|||||||
|
|
||||||
if ((USBD_LL_GetRxDataSize(pdev, MSCOutEpAdd) != USBD_BOT_CBW_LENGTH) ||
|
if ((USBD_LL_GetRxDataSize(pdev, MSCOutEpAdd) != USBD_BOT_CBW_LENGTH) ||
|
||||||
(hmsc->cbw.dSignature != USBD_BOT_CBW_SIGNATURE) ||
|
(hmsc->cbw.dSignature != USBD_BOT_CBW_SIGNATURE) ||
|
||||||
(hmsc->cbw.bLUN > 1U) || (hmsc->cbw.bCBLength < 1U) ||
|
(hmsc->cbw.bLUN > hmsc->max_lun) || (hmsc->cbw.bCBLength < 1U) ||
|
||||||
(hmsc->cbw.bCBLength > 16U))
|
(hmsc->cbw.bCBLength > 16U))
|
||||||
{
|
{
|
||||||
SCSI_SenseCode(pdev, hmsc->cbw.bLUN, ILLEGAL_REQUEST, INVALID_CDB);
|
SCSI_SenseCode(pdev, hmsc->cbw.bLUN, ILLEGAL_REQUEST, INVALID_CDB);
|
||||||
|
|||||||
@@ -118,6 +118,19 @@ uint8_t MSC_Mode_Sense10_data[MODE_SENSE10_LEN] =
|
|||||||
0x00, /* BLOCK DESCRIPTOR LENGTH MSB. */
|
0x00, /* BLOCK DESCRIPTOR LENGTH MSB. */
|
||||||
0x00 /* BLOCK DESCRIPTOR LENGTH LSB. */
|
0x00 /* BLOCK DESCRIPTOR LENGTH LSB. */
|
||||||
};
|
};
|
||||||
|
|
||||||
|
uint8_t MSC_Diagnostic_Data[DIAGNOSTIC_DATA_LEN] =
|
||||||
|
{
|
||||||
|
0x00, /* Byte 0: ADDITIONAL LENGTH (MSB) */
|
||||||
|
0x00, /* Byte 1: ADDITIONAL LENGTH (LSB) */
|
||||||
|
0x00, /* Byte 2: FRU CODE (most probable) */
|
||||||
|
0x00, /* Byte 3: FRU CODE */
|
||||||
|
0x00, /* Byte 4: FRU CODE */
|
||||||
|
0x00, /* Byte 5: FRU CODE (least probable) */
|
||||||
|
0x00, /* Byte 6: ERROR CODE (MSB) */
|
||||||
|
0x00, /* Byte 7: ERROR CODE (LSB) */
|
||||||
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @}
|
* @}
|
||||||
*/
|
*/
|
||||||
|
|||||||
@@ -92,6 +92,8 @@ static int8_t SCSI_Write12(USBD_HandleTypeDef *pdev, uint8_t lun, uint8_t *param
|
|||||||
static int8_t SCSI_Read10(USBD_HandleTypeDef *pdev, uint8_t lun, uint8_t *params);
|
static int8_t SCSI_Read10(USBD_HandleTypeDef *pdev, uint8_t lun, uint8_t *params);
|
||||||
static int8_t SCSI_Read12(USBD_HandleTypeDef *pdev, uint8_t lun, uint8_t *params);
|
static int8_t SCSI_Read12(USBD_HandleTypeDef *pdev, uint8_t lun, uint8_t *params);
|
||||||
static int8_t SCSI_Verify10(USBD_HandleTypeDef *pdev, uint8_t lun, uint8_t *params);
|
static int8_t SCSI_Verify10(USBD_HandleTypeDef *pdev, uint8_t lun, uint8_t *params);
|
||||||
|
static int8_t SCSI_ReportLuns(USBD_HandleTypeDef *pdev, uint8_t lun, uint8_t *params);
|
||||||
|
static int8_t SCSI_ReceiveDiagnosticResults(USBD_HandleTypeDef *pdev, uint8_t lun, uint8_t *params);
|
||||||
static int8_t SCSI_CheckAddressRange(USBD_HandleTypeDef *pdev, uint8_t lun,
|
static int8_t SCSI_CheckAddressRange(USBD_HandleTypeDef *pdev, uint8_t lun,
|
||||||
uint32_t blk_offset, uint32_t blk_nbr);
|
uint32_t blk_offset, uint32_t blk_nbr);
|
||||||
|
|
||||||
@@ -190,9 +192,16 @@ int8_t SCSI_ProcessCmd(USBD_HandleTypeDef *pdev, uint8_t lun, uint8_t *cmd)
|
|||||||
ret = SCSI_Verify10(pdev, lun, cmd);
|
ret = SCSI_Verify10(pdev, lun, cmd);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
|
case SCSI_REPORT_LUNS:
|
||||||
|
ret = SCSI_ReportLuns(pdev, lun, cmd);
|
||||||
|
break;
|
||||||
|
|
||||||
|
case SCSI_RECEIVE_DIAGNOSTIC_RESULTS:
|
||||||
|
ret = SCSI_ReceiveDiagnosticResults(pdev, lun, cmd);
|
||||||
|
break;
|
||||||
|
|
||||||
default:
|
default:
|
||||||
SCSI_SenseCode(pdev, lun, ILLEGAL_REQUEST, INVALID_CDB);
|
SCSI_SenseCode(pdev, lun, ILLEGAL_REQUEST, INVALID_CDB);
|
||||||
hmsc->bot_status = USBD_BOT_STATUS_ERROR;
|
|
||||||
ret = -1;
|
ret = -1;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
@@ -283,7 +292,7 @@ static int8_t SCSI_Inquiry(USBD_HandleTypeDef *pdev, uint8_t lun, uint8_t *param
|
|||||||
else /* Request Not supported */
|
else /* Request Not supported */
|
||||||
{
|
{
|
||||||
SCSI_SenseCode(pdev, hmsc->cbw.bLUN, ILLEGAL_REQUEST,
|
SCSI_SenseCode(pdev, hmsc->cbw.bLUN, ILLEGAL_REQUEST,
|
||||||
INVALID_FIELED_IN_COMMAND);
|
INVALID_FIELD_IN_COMMAND);
|
||||||
|
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
@@ -319,14 +328,15 @@ static int8_t SCSI_ReadCapacity10(USBD_HandleTypeDef *pdev, uint8_t lun, uint8_t
|
|||||||
UNUSED(params);
|
UNUSED(params);
|
||||||
int8_t ret;
|
int8_t ret;
|
||||||
USBD_MSC_BOT_HandleTypeDef *hmsc = (USBD_MSC_BOT_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId];
|
USBD_MSC_BOT_HandleTypeDef *hmsc = (USBD_MSC_BOT_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId];
|
||||||
|
USBD_MSC_BOT_LUN_TypeDef *p_scsi_blk = &hmsc->scsi_blk[lun];
|
||||||
|
|
||||||
if (hmsc == NULL)
|
if (hmsc == NULL)
|
||||||
{
|
{
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
ret = ((USBD_StorageTypeDef *)pdev->pUserData[pdev->classId])->GetCapacity(lun, &hmsc->scsi_blk_nbr,
|
ret = ((USBD_StorageTypeDef *)pdev->pUserData[pdev->classId])->GetCapacity(lun, &p_scsi_blk->nbr,
|
||||||
&hmsc->scsi_blk_size);
|
&p_scsi_blk->size);
|
||||||
|
|
||||||
if ((ret != 0) || (hmsc->scsi_medium_state == SCSI_MEDIUM_EJECTED))
|
if ((ret != 0) || (hmsc->scsi_medium_state == SCSI_MEDIUM_EJECTED))
|
||||||
{
|
{
|
||||||
@@ -334,20 +344,19 @@ static int8_t SCSI_ReadCapacity10(USBD_HandleTypeDef *pdev, uint8_t lun, uint8_t
|
|||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
hmsc->bot_data[0] = (uint8_t)((hmsc->scsi_blk_nbr - 1U) >> 24);
|
hmsc->bot_data[0] = (uint8_t)((p_scsi_blk->nbr - 1U) >> 24);
|
||||||
hmsc->bot_data[1] = (uint8_t)((hmsc->scsi_blk_nbr - 1U) >> 16);
|
hmsc->bot_data[1] = (uint8_t)((p_scsi_blk->nbr - 1U) >> 16);
|
||||||
hmsc->bot_data[2] = (uint8_t)((hmsc->scsi_blk_nbr - 1U) >> 8);
|
hmsc->bot_data[2] = (uint8_t)((p_scsi_blk->nbr - 1U) >> 8);
|
||||||
hmsc->bot_data[3] = (uint8_t)(hmsc->scsi_blk_nbr - 1U);
|
hmsc->bot_data[3] = (uint8_t)(p_scsi_blk->nbr - 1U);
|
||||||
|
|
||||||
hmsc->bot_data[4] = (uint8_t)(hmsc->scsi_blk_size >> 24);
|
hmsc->bot_data[4] = (uint8_t)(p_scsi_blk->size >> 24);
|
||||||
hmsc->bot_data[5] = (uint8_t)(hmsc->scsi_blk_size >> 16);
|
hmsc->bot_data[5] = (uint8_t)(p_scsi_blk->size >> 16);
|
||||||
hmsc->bot_data[6] = (uint8_t)(hmsc->scsi_blk_size >> 8);
|
hmsc->bot_data[6] = (uint8_t)(p_scsi_blk->size >> 8);
|
||||||
hmsc->bot_data[7] = (uint8_t)(hmsc->scsi_blk_size);
|
hmsc->bot_data[7] = (uint8_t)(p_scsi_blk->size);
|
||||||
|
|
||||||
hmsc->bot_data_length = 8U;
|
hmsc->bot_data_length = 8U;
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@@ -364,14 +373,15 @@ static int8_t SCSI_ReadCapacity16(USBD_HandleTypeDef *pdev, uint8_t lun, uint8_t
|
|||||||
uint32_t idx;
|
uint32_t idx;
|
||||||
int8_t ret;
|
int8_t ret;
|
||||||
USBD_MSC_BOT_HandleTypeDef *hmsc = (USBD_MSC_BOT_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId];
|
USBD_MSC_BOT_HandleTypeDef *hmsc = (USBD_MSC_BOT_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId];
|
||||||
|
USBD_MSC_BOT_LUN_TypeDef *p_scsi_blk = &hmsc->scsi_blk[lun];
|
||||||
|
|
||||||
if (hmsc == NULL)
|
if (hmsc == NULL)
|
||||||
{
|
{
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
ret = ((USBD_StorageTypeDef *)pdev->pUserData[pdev->classId])->GetCapacity(lun, &hmsc->scsi_blk_nbr,
|
ret = ((USBD_StorageTypeDef *)pdev->pUserData[pdev->classId])->GetCapacity(lun, &p_scsi_blk->nbr,
|
||||||
&hmsc->scsi_blk_size);
|
&p_scsi_blk->size);
|
||||||
|
|
||||||
if ((ret != 0) || (hmsc->scsi_medium_state == SCSI_MEDIUM_EJECTED))
|
if ((ret != 0) || (hmsc->scsi_medium_state == SCSI_MEDIUM_EJECTED))
|
||||||
{
|
{
|
||||||
@@ -389,15 +399,15 @@ static int8_t SCSI_ReadCapacity16(USBD_HandleTypeDef *pdev, uint8_t lun, uint8_t
|
|||||||
hmsc->bot_data[idx] = 0U;
|
hmsc->bot_data[idx] = 0U;
|
||||||
}
|
}
|
||||||
|
|
||||||
hmsc->bot_data[4] = (uint8_t)((hmsc->scsi_blk_nbr - 1U) >> 24);
|
hmsc->bot_data[4] = (uint8_t)((p_scsi_blk->nbr - 1U) >> 24);
|
||||||
hmsc->bot_data[5] = (uint8_t)((hmsc->scsi_blk_nbr - 1U) >> 16);
|
hmsc->bot_data[5] = (uint8_t)((p_scsi_blk->nbr - 1U) >> 16);
|
||||||
hmsc->bot_data[6] = (uint8_t)((hmsc->scsi_blk_nbr - 1U) >> 8);
|
hmsc->bot_data[6] = (uint8_t)((p_scsi_blk->nbr - 1U) >> 8);
|
||||||
hmsc->bot_data[7] = (uint8_t)(hmsc->scsi_blk_nbr - 1U);
|
hmsc->bot_data[7] = (uint8_t)(p_scsi_blk->nbr - 1U);
|
||||||
|
|
||||||
hmsc->bot_data[8] = (uint8_t)(hmsc->scsi_blk_size >> 24);
|
hmsc->bot_data[8] = (uint8_t)(p_scsi_blk->size >> 24);
|
||||||
hmsc->bot_data[9] = (uint8_t)(hmsc->scsi_blk_size >> 16);
|
hmsc->bot_data[9] = (uint8_t)(p_scsi_blk->size >> 16);
|
||||||
hmsc->bot_data[10] = (uint8_t)(hmsc->scsi_blk_size >> 8);
|
hmsc->bot_data[10] = (uint8_t)(p_scsi_blk->size >> 8);
|
||||||
hmsc->bot_data[11] = (uint8_t)(hmsc->scsi_blk_size);
|
hmsc->bot_data[11] = (uint8_t)(p_scsi_blk->size);
|
||||||
|
|
||||||
hmsc->bot_data_length = ((uint32_t)params[10] << 24) |
|
hmsc->bot_data_length = ((uint32_t)params[10] << 24) |
|
||||||
((uint32_t)params[11] << 16) |
|
((uint32_t)params[11] << 16) |
|
||||||
@@ -480,7 +490,11 @@ static int8_t SCSI_ModeSense6(USBD_HandleTypeDef *pdev, uint8_t lun, uint8_t *pa
|
|||||||
/* Check If media is write-protected */
|
/* Check If media is write-protected */
|
||||||
if (((USBD_StorageTypeDef *)pdev->pUserData[pdev->classId])->IsWriteProtected(lun) != 0)
|
if (((USBD_StorageTypeDef *)pdev->pUserData[pdev->classId])->IsWriteProtected(lun) != 0)
|
||||||
{
|
{
|
||||||
MSC_Mode_Sense6_data[2] |= 0x80U;
|
MSC_Mode_Sense6_data[2] |= (0x1U << 7); /* Set the WP (write protection) bit */
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
MSC_Mode_Sense10_data[2] &= ~(0x1U << 7); /* Clear the WP (write protection) bit */
|
||||||
}
|
}
|
||||||
|
|
||||||
if (params[4] <= len)
|
if (params[4] <= len)
|
||||||
@@ -515,7 +529,11 @@ static int8_t SCSI_ModeSense10(USBD_HandleTypeDef *pdev, uint8_t lun, uint8_t *p
|
|||||||
/* Check If media is write-protected */
|
/* Check If media is write-protected */
|
||||||
if (((USBD_StorageTypeDef *)pdev->pUserData[pdev->classId])->IsWriteProtected(lun) != 0)
|
if (((USBD_StorageTypeDef *)pdev->pUserData[pdev->classId])->IsWriteProtected(lun) != 0)
|
||||||
{
|
{
|
||||||
MSC_Mode_Sense10_data[3] |= 0x80U;
|
MSC_Mode_Sense10_data[3] |= (0x1U << 7); /* Set the WP (write protection) bit */
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
MSC_Mode_Sense10_data[3] &= ~(0x1U << 7); /* Clear the WP (write protection) bit */
|
||||||
}
|
}
|
||||||
|
|
||||||
if (params[8] <= len)
|
if (params[8] <= len)
|
||||||
@@ -635,7 +653,7 @@ static int8_t SCSI_StartStopUnit(USBD_HandleTypeDef *pdev, uint8_t lun, uint8_t
|
|||||||
|
|
||||||
if ((hmsc->scsi_medium_state == SCSI_MEDIUM_LOCKED) && ((params[4] & 0x3U) == 2U))
|
if ((hmsc->scsi_medium_state == SCSI_MEDIUM_LOCKED) && ((params[4] & 0x3U) == 2U))
|
||||||
{
|
{
|
||||||
SCSI_SenseCode(pdev, lun, ILLEGAL_REQUEST, INVALID_FIELED_IN_COMMAND);
|
SCSI_SenseCode(pdev, lun, ILLEGAL_REQUEST, INVALID_FIELD_IN_COMMAND);
|
||||||
|
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
@@ -704,6 +722,7 @@ static int8_t SCSI_AllowPreventRemovable(USBD_HandleTypeDef *pdev, uint8_t lun,
|
|||||||
static int8_t SCSI_Read10(USBD_HandleTypeDef *pdev, uint8_t lun, uint8_t *params)
|
static int8_t SCSI_Read10(USBD_HandleTypeDef *pdev, uint8_t lun, uint8_t *params)
|
||||||
{
|
{
|
||||||
USBD_MSC_BOT_HandleTypeDef *hmsc = (USBD_MSC_BOT_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId];
|
USBD_MSC_BOT_HandleTypeDef *hmsc = (USBD_MSC_BOT_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId];
|
||||||
|
USBD_MSC_BOT_LUN_TypeDef *p_scsi_blk = &hmsc->scsi_blk[lun];
|
||||||
|
|
||||||
if (hmsc == NULL)
|
if (hmsc == NULL)
|
||||||
{
|
{
|
||||||
@@ -732,21 +751,20 @@ static int8_t SCSI_Read10(USBD_HandleTypeDef *pdev, uint8_t lun, uint8_t *params
|
|||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
hmsc->scsi_blk_addr = ((uint32_t)params[2] << 24) |
|
p_scsi_blk->addr = ((uint32_t)params[2] << 24) |
|
||||||
((uint32_t)params[3] << 16) |
|
((uint32_t)params[3] << 16) |
|
||||||
((uint32_t)params[4] << 8) |
|
((uint32_t)params[4] << 8) |
|
||||||
(uint32_t)params[5];
|
(uint32_t)params[5];
|
||||||
|
|
||||||
hmsc->scsi_blk_len = ((uint32_t)params[7] << 8) | (uint32_t)params[8];
|
p_scsi_blk->len = ((uint32_t)params[7] << 8) | (uint32_t)params[8];
|
||||||
|
|
||||||
if (SCSI_CheckAddressRange(pdev, lun, hmsc->scsi_blk_addr,
|
if (SCSI_CheckAddressRange(pdev, lun, p_scsi_blk->addr, p_scsi_blk->len) < 0)
|
||||||
hmsc->scsi_blk_len) < 0)
|
|
||||||
{
|
{
|
||||||
return -1; /* error */
|
return -1; /* error */
|
||||||
}
|
}
|
||||||
|
|
||||||
/* cases 4,5 : Hi <> Dn */
|
/* cases 4,5 : Hi <> Dn */
|
||||||
if (hmsc->cbw.dDataLength != (hmsc->scsi_blk_len * hmsc->scsi_blk_size))
|
if (hmsc->cbw.dDataLength != (p_scsi_blk->len * p_scsi_blk->size))
|
||||||
{
|
{
|
||||||
SCSI_SenseCode(pdev, hmsc->cbw.bLUN, ILLEGAL_REQUEST, INVALID_CDB);
|
SCSI_SenseCode(pdev, hmsc->cbw.bLUN, ILLEGAL_REQUEST, INVALID_CDB);
|
||||||
return -1;
|
return -1;
|
||||||
@@ -770,6 +788,7 @@ static int8_t SCSI_Read10(USBD_HandleTypeDef *pdev, uint8_t lun, uint8_t *params
|
|||||||
static int8_t SCSI_Read12(USBD_HandleTypeDef *pdev, uint8_t lun, uint8_t *params)
|
static int8_t SCSI_Read12(USBD_HandleTypeDef *pdev, uint8_t lun, uint8_t *params)
|
||||||
{
|
{
|
||||||
USBD_MSC_BOT_HandleTypeDef *hmsc = (USBD_MSC_BOT_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId];
|
USBD_MSC_BOT_HandleTypeDef *hmsc = (USBD_MSC_BOT_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId];
|
||||||
|
USBD_MSC_BOT_LUN_TypeDef *p_scsi_blk = &hmsc->scsi_blk[lun];
|
||||||
|
|
||||||
if (hmsc == NULL)
|
if (hmsc == NULL)
|
||||||
{
|
{
|
||||||
@@ -797,24 +816,23 @@ static int8_t SCSI_Read12(USBD_HandleTypeDef *pdev, uint8_t lun, uint8_t *params
|
|||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
hmsc->scsi_blk_addr = ((uint32_t)params[2] << 24) |
|
p_scsi_blk->addr = ((uint32_t)params[2] << 24) |
|
||||||
((uint32_t)params[3] << 16) |
|
((uint32_t)params[3] << 16) |
|
||||||
((uint32_t)params[4] << 8) |
|
((uint32_t)params[4] << 8) |
|
||||||
(uint32_t)params[5];
|
(uint32_t)params[5];
|
||||||
|
|
||||||
hmsc->scsi_blk_len = ((uint32_t)params[6] << 24) |
|
p_scsi_blk->len = ((uint32_t)params[6] << 24) |
|
||||||
((uint32_t)params[7] << 16) |
|
((uint32_t)params[7] << 16) |
|
||||||
((uint32_t)params[8] << 8) |
|
((uint32_t)params[8] << 8) |
|
||||||
(uint32_t)params[9];
|
(uint32_t)params[9];
|
||||||
|
|
||||||
if (SCSI_CheckAddressRange(pdev, lun, hmsc->scsi_blk_addr,
|
if (SCSI_CheckAddressRange(pdev, lun, p_scsi_blk->addr, p_scsi_blk->len) < 0)
|
||||||
hmsc->scsi_blk_len) < 0)
|
|
||||||
{
|
{
|
||||||
return -1; /* error */
|
return -1; /* error */
|
||||||
}
|
}
|
||||||
|
|
||||||
/* cases 4,5 : Hi <> Dn */
|
/* cases 4,5 : Hi <> Dn */
|
||||||
if (hmsc->cbw.dDataLength != (hmsc->scsi_blk_len * hmsc->scsi_blk_size))
|
if (hmsc->cbw.dDataLength != (p_scsi_blk->len * p_scsi_blk->size))
|
||||||
{
|
{
|
||||||
SCSI_SenseCode(pdev, hmsc->cbw.bLUN, ILLEGAL_REQUEST, INVALID_CDB);
|
SCSI_SenseCode(pdev, hmsc->cbw.bLUN, ILLEGAL_REQUEST, INVALID_CDB);
|
||||||
return -1;
|
return -1;
|
||||||
@@ -838,6 +856,7 @@ static int8_t SCSI_Read12(USBD_HandleTypeDef *pdev, uint8_t lun, uint8_t *params
|
|||||||
static int8_t SCSI_Write10(USBD_HandleTypeDef *pdev, uint8_t lun, uint8_t *params)
|
static int8_t SCSI_Write10(USBD_HandleTypeDef *pdev, uint8_t lun, uint8_t *params)
|
||||||
{
|
{
|
||||||
USBD_MSC_BOT_HandleTypeDef *hmsc = (USBD_MSC_BOT_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId];
|
USBD_MSC_BOT_HandleTypeDef *hmsc = (USBD_MSC_BOT_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId];
|
||||||
|
USBD_MSC_BOT_LUN_TypeDef *p_scsi_blk = &hmsc->scsi_blk[lun];
|
||||||
uint32_t len;
|
uint32_t len;
|
||||||
|
|
||||||
if (hmsc == NULL)
|
if (hmsc == NULL)
|
||||||
@@ -879,22 +898,21 @@ static int8_t SCSI_Write10(USBD_HandleTypeDef *pdev, uint8_t lun, uint8_t *param
|
|||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
hmsc->scsi_blk_addr = ((uint32_t)params[2] << 24) |
|
p_scsi_blk->addr = ((uint32_t)params[2] << 24) |
|
||||||
((uint32_t)params[3] << 16) |
|
((uint32_t)params[3] << 16) |
|
||||||
((uint32_t)params[4] << 8) |
|
((uint32_t)params[4] << 8) |
|
||||||
(uint32_t)params[5];
|
(uint32_t)params[5];
|
||||||
|
|
||||||
hmsc->scsi_blk_len = ((uint32_t)params[7] << 8) |
|
p_scsi_blk->len = ((uint32_t)params[7] << 8) |
|
||||||
(uint32_t)params[8];
|
(uint32_t)params[8];
|
||||||
|
|
||||||
/* check if LBA address is in the right range */
|
/* check if LBA address is in the right range */
|
||||||
if (SCSI_CheckAddressRange(pdev, lun, hmsc->scsi_blk_addr,
|
if (SCSI_CheckAddressRange(pdev, lun, p_scsi_blk->addr, p_scsi_blk->len) < 0)
|
||||||
hmsc->scsi_blk_len) < 0)
|
|
||||||
{
|
{
|
||||||
return -1; /* error */
|
return -1; /* error */
|
||||||
}
|
}
|
||||||
|
|
||||||
len = hmsc->scsi_blk_len * hmsc->scsi_blk_size;
|
len = p_scsi_blk->len * p_scsi_blk->size;
|
||||||
|
|
||||||
/* cases 3,11,13 : Hn,Ho <> D0 */
|
/* cases 3,11,13 : Hn,Ho <> D0 */
|
||||||
if (hmsc->cbw.dDataLength != len)
|
if (hmsc->cbw.dDataLength != len)
|
||||||
@@ -928,6 +946,7 @@ static int8_t SCSI_Write10(USBD_HandleTypeDef *pdev, uint8_t lun, uint8_t *param
|
|||||||
static int8_t SCSI_Write12(USBD_HandleTypeDef *pdev, uint8_t lun, uint8_t *params)
|
static int8_t SCSI_Write12(USBD_HandleTypeDef *pdev, uint8_t lun, uint8_t *params)
|
||||||
{
|
{
|
||||||
USBD_MSC_BOT_HandleTypeDef *hmsc = (USBD_MSC_BOT_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId];
|
USBD_MSC_BOT_HandleTypeDef *hmsc = (USBD_MSC_BOT_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId];
|
||||||
|
USBD_MSC_BOT_LUN_TypeDef *p_scsi_blk = &hmsc->scsi_blk[lun];
|
||||||
uint32_t len;
|
uint32_t len;
|
||||||
|
|
||||||
if (hmsc == NULL)
|
if (hmsc == NULL)
|
||||||
@@ -970,24 +989,23 @@ static int8_t SCSI_Write12(USBD_HandleTypeDef *pdev, uint8_t lun, uint8_t *param
|
|||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
hmsc->scsi_blk_addr = ((uint32_t)params[2] << 24) |
|
p_scsi_blk->addr = ((uint32_t)params[2] << 24) |
|
||||||
((uint32_t)params[3] << 16) |
|
((uint32_t)params[3] << 16) |
|
||||||
((uint32_t)params[4] << 8) |
|
((uint32_t)params[4] << 8) |
|
||||||
(uint32_t)params[5];
|
(uint32_t)params[5];
|
||||||
|
|
||||||
hmsc->scsi_blk_len = ((uint32_t)params[6] << 24) |
|
p_scsi_blk->len = ((uint32_t)params[6] << 24) |
|
||||||
((uint32_t)params[7] << 16) |
|
((uint32_t)params[7] << 16) |
|
||||||
((uint32_t)params[8] << 8) |
|
((uint32_t)params[8] << 8) |
|
||||||
(uint32_t)params[9];
|
(uint32_t)params[9];
|
||||||
|
|
||||||
/* check if LBA address is in the right range */
|
/* check if LBA address is in the right range */
|
||||||
if (SCSI_CheckAddressRange(pdev, lun, hmsc->scsi_blk_addr,
|
if (SCSI_CheckAddressRange(pdev, lun, p_scsi_blk->addr, p_scsi_blk->len) < 0)
|
||||||
hmsc->scsi_blk_len) < 0)
|
|
||||||
{
|
{
|
||||||
return -1; /* error */
|
return -1; /* error */
|
||||||
}
|
}
|
||||||
|
|
||||||
len = hmsc->scsi_blk_len * hmsc->scsi_blk_size;
|
len = p_scsi_blk->len * p_scsi_blk->size;
|
||||||
|
|
||||||
/* cases 3,11,13 : Hn,Ho <> D0 */
|
/* cases 3,11,13 : Hn,Ho <> D0 */
|
||||||
if (hmsc->cbw.dDataLength != len)
|
if (hmsc->cbw.dDataLength != len)
|
||||||
@@ -1021,6 +1039,7 @@ static int8_t SCSI_Write12(USBD_HandleTypeDef *pdev, uint8_t lun, uint8_t *param
|
|||||||
static int8_t SCSI_Verify10(USBD_HandleTypeDef *pdev, uint8_t lun, uint8_t *params)
|
static int8_t SCSI_Verify10(USBD_HandleTypeDef *pdev, uint8_t lun, uint8_t *params)
|
||||||
{
|
{
|
||||||
USBD_MSC_BOT_HandleTypeDef *hmsc = (USBD_MSC_BOT_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId];
|
USBD_MSC_BOT_HandleTypeDef *hmsc = (USBD_MSC_BOT_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId];
|
||||||
|
USBD_MSC_BOT_LUN_TypeDef *p_scsi_blk = &hmsc->scsi_blk[lun];
|
||||||
|
|
||||||
if (hmsc == NULL)
|
if (hmsc == NULL)
|
||||||
{
|
{
|
||||||
@@ -1029,11 +1048,11 @@ static int8_t SCSI_Verify10(USBD_HandleTypeDef *pdev, uint8_t lun, uint8_t *para
|
|||||||
|
|
||||||
if ((params[1] & 0x02U) == 0x02U)
|
if ((params[1] & 0x02U) == 0x02U)
|
||||||
{
|
{
|
||||||
SCSI_SenseCode(pdev, lun, ILLEGAL_REQUEST, INVALID_FIELED_IN_COMMAND);
|
SCSI_SenseCode(pdev, lun, ILLEGAL_REQUEST, INVALID_FIELD_IN_COMMAND);
|
||||||
return -1; /* Error, Verify Mode Not supported*/
|
return -1; /* Error, Verify Mode Not supported*/
|
||||||
}
|
}
|
||||||
|
|
||||||
if (SCSI_CheckAddressRange(pdev, lun, hmsc->scsi_blk_addr, hmsc->scsi_blk_len) < 0)
|
if (SCSI_CheckAddressRange(pdev, lun, p_scsi_blk->addr, p_scsi_blk->len) < 0)
|
||||||
{
|
{
|
||||||
return -1; /* error */
|
return -1; /* error */
|
||||||
}
|
}
|
||||||
@@ -1043,6 +1062,90 @@ static int8_t SCSI_Verify10(USBD_HandleTypeDef *pdev, uint8_t lun, uint8_t *para
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @brief SCSI_ReportLuns12
|
||||||
|
* Process ReportLuns command
|
||||||
|
* @retval status
|
||||||
|
*/
|
||||||
|
static int8_t SCSI_ReportLuns(USBD_HandleTypeDef *pdev, uint8_t lun, uint8_t *params)
|
||||||
|
{
|
||||||
|
USBD_MSC_BOT_HandleTypeDef *hmsc;
|
||||||
|
uint32_t lun_list_length;
|
||||||
|
uint32_t total_length;
|
||||||
|
uint8_t lun_idx;
|
||||||
|
|
||||||
|
UNUSED(lun);
|
||||||
|
UNUSED(params);
|
||||||
|
|
||||||
|
/* Define the report LUNs buffer Each LUN entry is 8 bytes */
|
||||||
|
static uint8_t lun_report[8U * (MSC_BOT_MAX_LUN + 1U)];
|
||||||
|
|
||||||
|
hmsc = (USBD_MSC_BOT_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId];
|
||||||
|
|
||||||
|
if (hmsc == NULL)
|
||||||
|
{
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Initialize the report LUNs buffer */
|
||||||
|
(void)USBD_memset(lun_report, 0, sizeof(lun_report));
|
||||||
|
|
||||||
|
/* Set the LUN list length in the first 4 bytes */
|
||||||
|
lun_list_length = 8U * (hmsc->max_lun + 1U);
|
||||||
|
lun_report[0] = (uint8_t)(lun_list_length >> 24);
|
||||||
|
lun_report[1] = (uint8_t)(lun_list_length >> 16);
|
||||||
|
lun_report[2] = (uint8_t)(lun_list_length >> 8);
|
||||||
|
lun_report[3] = (uint8_t)(lun_list_length & 0xFFU);
|
||||||
|
|
||||||
|
/* Update the LUN list */
|
||||||
|
for (lun_idx = 0U; lun_idx <= hmsc->max_lun; lun_idx++)
|
||||||
|
{
|
||||||
|
/* LUN identifier is placed at the second byte of each 8-byte entry */
|
||||||
|
lun_report[(8U * (lun_idx + 1U)) + 1U] = lun_idx;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Calculate the total length of the report LUNs buffer */
|
||||||
|
total_length = lun_list_length + 8U;
|
||||||
|
|
||||||
|
/* Update the BOT data with the report LUNs buffer */
|
||||||
|
(void)SCSI_UpdateBotData(hmsc, lun_report, (uint16_t)total_length);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @brief SCSI_ReceiveDiagnosticResults
|
||||||
|
* Process SCSI_Receive Diagnostic Results command
|
||||||
|
* @param lun: Logical unit number
|
||||||
|
* @param params: Command parameters
|
||||||
|
* @retval status
|
||||||
|
*/
|
||||||
|
static int8_t SCSI_ReceiveDiagnosticResults(USBD_HandleTypeDef *pdev, uint8_t lun, uint8_t *params)
|
||||||
|
{
|
||||||
|
UNUSED(lun);
|
||||||
|
USBD_MSC_BOT_HandleTypeDef *hmsc = (USBD_MSC_BOT_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId];
|
||||||
|
uint16_t allocation_length;
|
||||||
|
|
||||||
|
/* Extract the allocation length from the CDB */
|
||||||
|
allocation_length = (((uint16_t)params[3] << 8) | (uint16_t)params[4]);
|
||||||
|
|
||||||
|
if (allocation_length == 0U)
|
||||||
|
{
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Ensure the allocation length does not exceed the diagnostic data length */
|
||||||
|
if (allocation_length > DIAGNOSTIC_DATA_LEN)
|
||||||
|
{
|
||||||
|
allocation_length = DIAGNOSTIC_DATA_LEN;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Send the diagnostic data to the host */
|
||||||
|
(void)SCSI_UpdateBotData(hmsc, MSC_Diagnostic_Data, allocation_length);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @brief SCSI_CheckAddressRange
|
* @brief SCSI_CheckAddressRange
|
||||||
* Check address range
|
* Check address range
|
||||||
@@ -1055,13 +1158,14 @@ static int8_t SCSI_CheckAddressRange(USBD_HandleTypeDef *pdev, uint8_t lun,
|
|||||||
uint32_t blk_offset, uint32_t blk_nbr)
|
uint32_t blk_offset, uint32_t blk_nbr)
|
||||||
{
|
{
|
||||||
USBD_MSC_BOT_HandleTypeDef *hmsc = (USBD_MSC_BOT_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId];
|
USBD_MSC_BOT_HandleTypeDef *hmsc = (USBD_MSC_BOT_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId];
|
||||||
|
USBD_MSC_BOT_LUN_TypeDef *p_scsi_blk = &hmsc->scsi_blk[lun];
|
||||||
|
|
||||||
if (hmsc == NULL)
|
if (hmsc == NULL)
|
||||||
{
|
{
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
if ((blk_offset + blk_nbr) > hmsc->scsi_blk_nbr)
|
if ((blk_offset + blk_nbr) > p_scsi_blk->nbr)
|
||||||
{
|
{
|
||||||
SCSI_SenseCode(pdev, lun, ILLEGAL_REQUEST, ADDRESS_OUT_OF_RANGE);
|
SCSI_SenseCode(pdev, lun, ILLEGAL_REQUEST, ADDRESS_OUT_OF_RANGE);
|
||||||
return -1;
|
return -1;
|
||||||
@@ -1079,6 +1183,7 @@ static int8_t SCSI_CheckAddressRange(USBD_HandleTypeDef *pdev, uint8_t lun,
|
|||||||
static int8_t SCSI_ProcessRead(USBD_HandleTypeDef *pdev, uint8_t lun)
|
static int8_t SCSI_ProcessRead(USBD_HandleTypeDef *pdev, uint8_t lun)
|
||||||
{
|
{
|
||||||
USBD_MSC_BOT_HandleTypeDef *hmsc = (USBD_MSC_BOT_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId];
|
USBD_MSC_BOT_HandleTypeDef *hmsc = (USBD_MSC_BOT_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId];
|
||||||
|
USBD_MSC_BOT_LUN_TypeDef *p_scsi_blk = &hmsc->scsi_blk[lun];
|
||||||
uint32_t len;
|
uint32_t len;
|
||||||
|
|
||||||
if (hmsc == NULL)
|
if (hmsc == NULL)
|
||||||
@@ -1086,7 +1191,7 @@ static int8_t SCSI_ProcessRead(USBD_HandleTypeDef *pdev, uint8_t lun)
|
|||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
len = hmsc->scsi_blk_len * hmsc->scsi_blk_size;
|
len = p_scsi_blk->len * p_scsi_blk->size;
|
||||||
|
|
||||||
#ifdef USE_USBD_COMPOSITE
|
#ifdef USE_USBD_COMPOSITE
|
||||||
/* Get the Endpoints addresses allocated for this class instance */
|
/* Get the Endpoints addresses allocated for this class instance */
|
||||||
@@ -1096,8 +1201,8 @@ static int8_t SCSI_ProcessRead(USBD_HandleTypeDef *pdev, uint8_t lun)
|
|||||||
len = MIN(len, MSC_MEDIA_PACKET);
|
len = MIN(len, MSC_MEDIA_PACKET);
|
||||||
|
|
||||||
if (((USBD_StorageTypeDef *)pdev->pUserData[pdev->classId])->Read(lun, hmsc->bot_data,
|
if (((USBD_StorageTypeDef *)pdev->pUserData[pdev->classId])->Read(lun, hmsc->bot_data,
|
||||||
hmsc->scsi_blk_addr,
|
p_scsi_blk->addr,
|
||||||
(len / hmsc->scsi_blk_size)) < 0)
|
(len / p_scsi_blk->size)) < 0)
|
||||||
{
|
{
|
||||||
SCSI_SenseCode(pdev, lun, HARDWARE_ERROR, UNRECOVERED_READ_ERROR);
|
SCSI_SenseCode(pdev, lun, HARDWARE_ERROR, UNRECOVERED_READ_ERROR);
|
||||||
return -1;
|
return -1;
|
||||||
@@ -1105,13 +1210,13 @@ static int8_t SCSI_ProcessRead(USBD_HandleTypeDef *pdev, uint8_t lun)
|
|||||||
|
|
||||||
(void)USBD_LL_Transmit(pdev, MSCInEpAdd, hmsc->bot_data, len);
|
(void)USBD_LL_Transmit(pdev, MSCInEpAdd, hmsc->bot_data, len);
|
||||||
|
|
||||||
hmsc->scsi_blk_addr += (len / hmsc->scsi_blk_size);
|
p_scsi_blk->addr += (len / p_scsi_blk->size);
|
||||||
hmsc->scsi_blk_len -= (len / hmsc->scsi_blk_size);
|
p_scsi_blk->len -= (len / p_scsi_blk->size);
|
||||||
|
|
||||||
/* case 6 : Hi = Di */
|
/* case 6 : Hi = Di */
|
||||||
hmsc->csw.dDataResidue -= len;
|
hmsc->csw.dDataResidue -= len;
|
||||||
|
|
||||||
if (hmsc->scsi_blk_len == 0U)
|
if (p_scsi_blk->len == 0U)
|
||||||
{
|
{
|
||||||
hmsc->bot_state = USBD_BOT_LAST_DATA_IN;
|
hmsc->bot_state = USBD_BOT_LAST_DATA_IN;
|
||||||
}
|
}
|
||||||
@@ -1128,6 +1233,7 @@ static int8_t SCSI_ProcessRead(USBD_HandleTypeDef *pdev, uint8_t lun)
|
|||||||
static int8_t SCSI_ProcessWrite(USBD_HandleTypeDef *pdev, uint8_t lun)
|
static int8_t SCSI_ProcessWrite(USBD_HandleTypeDef *pdev, uint8_t lun)
|
||||||
{
|
{
|
||||||
USBD_MSC_BOT_HandleTypeDef *hmsc = (USBD_MSC_BOT_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId];
|
USBD_MSC_BOT_HandleTypeDef *hmsc = (USBD_MSC_BOT_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId];
|
||||||
|
USBD_MSC_BOT_LUN_TypeDef *p_scsi_blk = &hmsc->scsi_blk[lun];
|
||||||
uint32_t len;
|
uint32_t len;
|
||||||
|
|
||||||
if (hmsc == NULL)
|
if (hmsc == NULL)
|
||||||
@@ -1135,7 +1241,7 @@ static int8_t SCSI_ProcessWrite(USBD_HandleTypeDef *pdev, uint8_t lun)
|
|||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
len = hmsc->scsi_blk_len * hmsc->scsi_blk_size;
|
len = p_scsi_blk->len * p_scsi_blk->size;
|
||||||
|
|
||||||
#ifdef USE_USBD_COMPOSITE
|
#ifdef USE_USBD_COMPOSITE
|
||||||
/* Get the Endpoints addresses allocated for this class instance */
|
/* Get the Endpoints addresses allocated for this class instance */
|
||||||
@@ -1144,27 +1250,26 @@ static int8_t SCSI_ProcessWrite(USBD_HandleTypeDef *pdev, uint8_t lun)
|
|||||||
|
|
||||||
len = MIN(len, MSC_MEDIA_PACKET);
|
len = MIN(len, MSC_MEDIA_PACKET);
|
||||||
|
|
||||||
if (((USBD_StorageTypeDef *)pdev->pUserData[pdev->classId])->Write(lun, hmsc->bot_data,
|
if (((USBD_StorageTypeDef *)pdev->pUserData[pdev->classId])->Write(lun, hmsc->bot_data, p_scsi_blk->addr,
|
||||||
hmsc->scsi_blk_addr,
|
(len / p_scsi_blk->size)) < 0)
|
||||||
(len / hmsc->scsi_blk_size)) < 0)
|
|
||||||
{
|
{
|
||||||
SCSI_SenseCode(pdev, lun, HARDWARE_ERROR, WRITE_FAULT);
|
SCSI_SenseCode(pdev, lun, HARDWARE_ERROR, WRITE_FAULT);
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
hmsc->scsi_blk_addr += (len / hmsc->scsi_blk_size);
|
p_scsi_blk->addr += (len / p_scsi_blk->size);
|
||||||
hmsc->scsi_blk_len -= (len / hmsc->scsi_blk_size);
|
p_scsi_blk->len -= (len / p_scsi_blk->size);
|
||||||
|
|
||||||
/* case 12 : Ho = Do */
|
/* case 12 : Ho = Do */
|
||||||
hmsc->csw.dDataResidue -= len;
|
hmsc->csw.dDataResidue -= len;
|
||||||
|
|
||||||
if (hmsc->scsi_blk_len == 0U)
|
if (p_scsi_blk->len == 0U)
|
||||||
{
|
{
|
||||||
MSC_BOT_SendCSW(pdev, USBD_CSW_CMD_PASSED);
|
MSC_BOT_SendCSW(pdev, USBD_CSW_CMD_PASSED);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
len = MIN((hmsc->scsi_blk_len * hmsc->scsi_blk_size), MSC_MEDIA_PACKET);
|
len = MIN((p_scsi_blk->len * p_scsi_blk->size), MSC_MEDIA_PACKET);
|
||||||
|
|
||||||
/* Prepare EP to Receive next packet */
|
/* Prepare EP to Receive next packet */
|
||||||
(void)USBD_LL_PrepareReceive(pdev, MSCOutEpAdd, hmsc->bot_data, len);
|
(void)USBD_LL_PrepareReceive(pdev, MSCOutEpAdd, hmsc->bot_data, len);
|
||||||
|
|||||||
@@ -67,9 +67,12 @@ extern "C" {
|
|||||||
#define MTP_FS_BINTERVAL 0x10U
|
#define MTP_FS_BINTERVAL 0x10U
|
||||||
#endif /* MTP_FS_BINTERVAL */
|
#endif /* MTP_FS_BINTERVAL */
|
||||||
|
|
||||||
|
#ifndef MTP_CMD_PACKET_SIZE
|
||||||
|
#define MTP_CMD_PACKET_SIZE 8U /* Control Endpoint Packet size */
|
||||||
|
#endif /* MTP_CMD_PACKET_SIZE */
|
||||||
|
|
||||||
#define MTP_DATA_MAX_HS_PACKET_SIZE 512U
|
#define MTP_DATA_MAX_HS_PACKET_SIZE 512U
|
||||||
#define MTP_DATA_MAX_FS_PACKET_SIZE 64U /* Endpoint IN & OUT Packet size */
|
#define MTP_DATA_MAX_FS_PACKET_SIZE 64U /* Endpoint IN & OUT Packet size */
|
||||||
#define MTP_CMD_PACKET_SIZE 8U /* Control Endpoint Packet size */
|
|
||||||
|
|
||||||
#define MTP_MEDIA_PACKET 512U
|
#define MTP_MEDIA_PACKET 512U
|
||||||
#define MTP_CONT_HEADER_SIZE 12U
|
#define MTP_CONT_HEADER_SIZE 12U
|
||||||
|
|||||||
@@ -283,7 +283,7 @@ extern "C" {
|
|||||||
* Appendix C. Device Properties
|
* Appendix C. Device Properties
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/* MTP device properties code*/
|
/* MTP device properties code */
|
||||||
#define MTP_DEV_PROP_UNDEFINED 0x5000U
|
#define MTP_DEV_PROP_UNDEFINED 0x5000U
|
||||||
#define MTP_DEV_PROP_BATTERY_LEVEL 0x5001U
|
#define MTP_DEV_PROP_BATTERY_LEVEL 0x5001U
|
||||||
#define MTP_DEV_PROP_FUNCTIONAL_MODE 0x5002U
|
#define MTP_DEV_PROP_FUNCTIONAL_MODE 0x5002U
|
||||||
@@ -336,7 +336,7 @@ extern "C" {
|
|||||||
#define MTP_CONT_TYPE_EVENT 4U
|
#define MTP_CONT_TYPE_EVENT 4U
|
||||||
|
|
||||||
#ifndef MTP_STORAGE_ID
|
#ifndef MTP_STORAGE_ID
|
||||||
#define MTP_STORAGE_ID 0x00010001U /* SD card is inserted*/
|
#define MTP_STORAGE_ID 0x00010001U /* SD card is inserted */
|
||||||
#endif /* MTP_STORAGE_ID */
|
#endif /* MTP_STORAGE_ID */
|
||||||
|
|
||||||
#define MTP_NBR_STORAGE_ID 1U
|
#define MTP_NBR_STORAGE_ID 1U
|
||||||
|
|||||||
@@ -53,15 +53,16 @@ extern "C" {
|
|||||||
|
|
||||||
typedef enum
|
typedef enum
|
||||||
{
|
{
|
||||||
DATA_TYPE = 0x00,
|
DATA_TYPE = 0x00U,
|
||||||
REP_TYPE = 0x01,
|
REP_TYPE = 0x01U,
|
||||||
} MTP_CONTAINER_TYPE;
|
} MTP_CONTAINER_TYPE;
|
||||||
|
|
||||||
|
|
||||||
typedef enum
|
typedef enum
|
||||||
{
|
{
|
||||||
READ_FIRST_DATA = 0x00,
|
READ_FIRST_DATA = 0x00U,
|
||||||
READ_REST_OF_DATA = 0x01,
|
READ_REST_OF_DATA = 0x01U,
|
||||||
|
READ_SEND_ZLP_DATA = 0x02U
|
||||||
} MTP_READ_DATA_STATUS;
|
} MTP_READ_DATA_STATUS;
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -128,6 +128,9 @@ USBD_ClassTypeDef USBD_MTP =
|
|||||||
USBD_MTP_GetOtherSpeedCfgDesc,
|
USBD_MTP_GetOtherSpeedCfgDesc,
|
||||||
USBD_MTP_GetDeviceQualifierDescriptor,
|
USBD_MTP_GetDeviceQualifierDescriptor,
|
||||||
#endif /* USE_USBD_COMPOSITE */
|
#endif /* USE_USBD_COMPOSITE */
|
||||||
|
#if (USBD_SUPPORT_USER_STRING_DESC == 1U)
|
||||||
|
NULL,
|
||||||
|
#endif /* USBD_SUPPORT_USER_STRING_DESC */
|
||||||
};
|
};
|
||||||
|
|
||||||
#ifndef USE_USBD_COMPOSITE
|
#ifndef USE_USBD_COMPOSITE
|
||||||
|
|||||||
@@ -537,50 +537,30 @@ static void MTP_Get_PayloadContent(USBD_HandleTypeDef *pdev)
|
|||||||
{
|
{
|
||||||
USBD_MTP_HandleTypeDef *hmtp = (USBD_MTP_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId];
|
USBD_MTP_HandleTypeDef *hmtp = (USBD_MTP_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId];
|
||||||
USBD_MTP_ItfTypeDef *hmtpif = (USBD_MTP_ItfTypeDef *)pdev->pUserData[pdev->classId];
|
USBD_MTP_ItfTypeDef *hmtpif = (USBD_MTP_ItfTypeDef *)pdev->pUserData[pdev->classId];
|
||||||
uint8_t *buffer = hmtp->GenericContainer.data;
|
uint8_t *buffer;
|
||||||
uint32_t i;
|
|
||||||
uint32_t n_idx;
|
uint32_t n_idx;
|
||||||
|
|
||||||
switch (hmtp->OperationsContainer.code)
|
switch (hmtp->OperationsContainer.code)
|
||||||
{
|
{
|
||||||
case MTP_OP_GET_DEVICE_INFO:
|
case MTP_OP_GET_DEVICE_INFO:
|
||||||
(void)MTP_Get_DeviceInfo();
|
(void)MTP_Get_DeviceInfo();
|
||||||
(void)USBD_memcpy(buffer, (const uint8_t *)&MTP_DeviceInfo, sizeof(MTP_DeviceInfo));
|
(void)USBD_memcpy(hmtp->GenericContainer.data, (const uint8_t *)&MTP_DeviceInfo,
|
||||||
|
MIN(sizeof(MTP_DeviceInfo), MTP_MEDIA_PACKET));
|
||||||
for (i = 0U; i < sizeof(MTP_StorageIDS); i++)
|
|
||||||
{
|
|
||||||
hmtp->GenericContainer.data[i] = buffer[i];
|
|
||||||
}
|
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case MTP_OP_GET_STORAGE_IDS:
|
case MTP_OP_GET_STORAGE_IDS:
|
||||||
(void)MTP_Get_StorageIDS();
|
(void)MTP_Get_StorageIDS();
|
||||||
(void)USBD_memcpy(buffer, (const uint8_t *)&MTP_StorageIDS, sizeof(MTP_StorageIDS));
|
(void)USBD_memcpy(hmtp->GenericContainer.data, (const uint8_t *)&MTP_StorageIDS, sizeof(MTP_StorageIDS));
|
||||||
|
|
||||||
for (i = 0U; i < sizeof(MTP_StorageIDS); i++)
|
|
||||||
{
|
|
||||||
hmtp->GenericContainer.data[i] = buffer[i];
|
|
||||||
}
|
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case MTP_OP_GET_STORAGE_INFO:
|
case MTP_OP_GET_STORAGE_INFO:
|
||||||
(void)MTP_Get_StorageInfo(pdev);
|
(void)MTP_Get_StorageInfo(pdev);
|
||||||
(void)USBD_memcpy(buffer, (const uint8_t *)&MTP_StorageInfo, sizeof(MTP_StorageInfo));
|
(void)USBD_memcpy(hmtp->GenericContainer.data, (const uint8_t *)&MTP_StorageInfo, sizeof(MTP_StorageInfo));
|
||||||
|
|
||||||
for (i = 0U; i < sizeof(MTP_StorageInfo); i++)
|
|
||||||
{
|
|
||||||
hmtp->GenericContainer.data[i] = buffer[i];
|
|
||||||
}
|
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case MTP_OP_GET_OBJECT_HANDLES:
|
case MTP_OP_GET_OBJECT_HANDLES:
|
||||||
(void)MTP_Get_ObjectHandle(pdev);
|
(void)MTP_Get_ObjectHandle(pdev);
|
||||||
(void)USBD_memcpy(buffer, (const uint8_t *)&MTP_ObjectHandle, hmtp->ResponseLength);
|
(void)USBD_memcpy(hmtp->GenericContainer.data, (const uint8_t *)&MTP_ObjectHandle, hmtp->ResponseLength);
|
||||||
|
|
||||||
for (i = 0U; i < hmtp->ResponseLength; i++)
|
|
||||||
{
|
|
||||||
hmtp->GenericContainer.data[i] = buffer[i];
|
|
||||||
}
|
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case MTP_OP_GET_OBJECT_INFO:
|
case MTP_OP_GET_OBJECT_INFO:
|
||||||
@@ -589,12 +569,7 @@ static void MTP_Get_PayloadContent(USBD_HandleTypeDef *pdev)
|
|||||||
|
|
||||||
case MTP_OP_GET_OBJECT_PROPS_SUPPORTED:
|
case MTP_OP_GET_OBJECT_PROPS_SUPPORTED:
|
||||||
(void)MTP_Get_ObjectPropSupp();
|
(void)MTP_Get_ObjectPropSupp();
|
||||||
(void)USBD_memcpy(buffer, (const uint8_t *)&MTP_ObjectPropSupp, sizeof(MTP_ObjectPropSupp));
|
(void)USBD_memcpy(hmtp->GenericContainer.data, (const uint8_t *)&MTP_ObjectPropSupp, sizeof(MTP_ObjectPropSupp));
|
||||||
|
|
||||||
for (i = 0U; i < sizeof(MTP_ObjectPropSupp); i++)
|
|
||||||
{
|
|
||||||
hmtp->GenericContainer.data[i] = buffer[i];
|
|
||||||
}
|
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case MTP_OP_GET_OBJECT_PROP_DESC:
|
case MTP_OP_GET_OBJECT_PROP_DESC:
|
||||||
@@ -604,12 +579,7 @@ static void MTP_Get_PayloadContent(USBD_HandleTypeDef *pdev)
|
|||||||
|
|
||||||
case MTP_OP_GET_OBJECT_PROP_REFERENCES:
|
case MTP_OP_GET_OBJECT_PROP_REFERENCES:
|
||||||
MTP_Ref.ref_len = 0U;
|
MTP_Ref.ref_len = 0U;
|
||||||
(void)USBD_memcpy(buffer, (const uint8_t *)&MTP_Ref.ref_len, sizeof(MTP_Ref.ref_len));
|
(void)USBD_memcpy(hmtp->GenericContainer.data, (const uint8_t *)&MTP_Ref.ref_len, sizeof(MTP_Ref.ref_len));
|
||||||
|
|
||||||
for (i = 0U; i < sizeof(MTP_Ref.ref_len); i++)
|
|
||||||
{
|
|
||||||
hmtp->GenericContainer.data[i] = buffer[i];
|
|
||||||
}
|
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case MTP_OP_GET_OBJECT_PROPLIST:
|
case MTP_OP_GET_OBJECT_PROPLIST:
|
||||||
@@ -618,20 +588,13 @@ static void MTP_Get_PayloadContent(USBD_HandleTypeDef *pdev)
|
|||||||
|
|
||||||
case MTP_OP_GET_OBJECT_PROP_VALUE:
|
case MTP_OP_GET_OBJECT_PROP_VALUE:
|
||||||
buffer = MTP_Get_ObjectPropValue(pdev);
|
buffer = MTP_Get_ObjectPropValue(pdev);
|
||||||
for (i = 0U; i < hmtp->ResponseLength; i++)
|
(void)USBD_memcpy(hmtp->GenericContainer.data, (const uint8_t *)buffer, hmtp->ResponseLength);
|
||||||
{
|
|
||||||
hmtp->GenericContainer.data[i] = buffer[i];
|
|
||||||
}
|
|
||||||
|
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case MTP_OP_GET_DEVICE_PROP_DESC:
|
case MTP_OP_GET_DEVICE_PROP_DESC:
|
||||||
(void)MTP_Get_DevicePropDesc();
|
(void)MTP_Get_DevicePropDesc();
|
||||||
(void)USBD_memcpy(buffer, (const uint8_t *)&MTP_DevicePropDesc, sizeof(MTP_DevicePropDesc));
|
(void)USBD_memcpy(hmtp->GenericContainer.data, (const uint8_t *)&MTP_DevicePropDesc,
|
||||||
for (i = 0U; i < sizeof(MTP_DevicePropDesc); i++)
|
MIN(sizeof(MTP_DevicePropDesc), MTP_MEDIA_PACKET));
|
||||||
{
|
|
||||||
hmtp->GenericContainer.data[i] = buffer[i];
|
|
||||||
}
|
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case MTP_OP_SEND_OBJECT_INFO:
|
case MTP_OP_SEND_OBJECT_INFO:
|
||||||
@@ -837,7 +800,7 @@ static void MTP_Get_ObjectPropDesc(USBD_HandleTypeDef *pdev)
|
|||||||
|
|
||||||
case MTP_OB_PROP_PARENT_OBJECT :
|
case MTP_OB_PROP_PARENT_OBJECT :
|
||||||
MTP_ObjectPropDesc.ObjectPropertyCode = (uint16_t)(hmtp->OperationsContainer.Param1);
|
MTP_ObjectPropDesc.ObjectPropertyCode = (uint16_t)(hmtp->OperationsContainer.Param1);
|
||||||
MTP_ObjectPropDesc.DataType = MTP_DATATYPE_STR;
|
MTP_ObjectPropDesc.DataType = MTP_DATATYPE_UINT32;
|
||||||
MTP_ObjectPropDesc.GetSet = MTP_PROP_GET;
|
MTP_ObjectPropDesc.GetSet = MTP_PROP_GET;
|
||||||
MTP_ObjectPropDesc.DefValue = 0U;
|
MTP_ObjectPropDesc.DefValue = 0U;
|
||||||
MTP_ObjectPropDesc.GroupCode = 0U;
|
MTP_ObjectPropDesc.GroupCode = 0U;
|
||||||
@@ -951,8 +914,8 @@ static uint8_t *MTP_Get_ObjectPropValue(USBD_HandleTypeDef *pdev)
|
|||||||
*/
|
*/
|
||||||
static void MTP_Get_ObjectPropList(USBD_HandleTypeDef *pdev)
|
static void MTP_Get_ObjectPropList(USBD_HandleTypeDef *pdev)
|
||||||
{
|
{
|
||||||
USBD_MTP_HandleTypeDef *hmtp = (USBD_MTP_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId];
|
USBD_MTP_HandleTypeDef *hmtp = (USBD_MTP_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId];
|
||||||
USBD_MTP_ItfTypeDef *hmtpif = (USBD_MTP_ItfTypeDef *)pdev->pUserData[pdev->classId];
|
USBD_MTP_ItfTypeDef *hmtpif = (USBD_MTP_ItfTypeDef *)pdev->pUserData[pdev->classId];
|
||||||
uint16_t filename[255];
|
uint16_t filename[255];
|
||||||
uint32_t storageid = MTP_STORAGE_ID;
|
uint32_t storageid = MTP_STORAGE_ID;
|
||||||
uint32_t default_val = 0U;
|
uint32_t default_val = 0U;
|
||||||
@@ -960,6 +923,7 @@ static void MTP_Get_ObjectPropList(USBD_HandleTypeDef *pdev)
|
|||||||
uint16_t format;
|
uint16_t format;
|
||||||
uint64_t objsize;
|
uint64_t objsize;
|
||||||
uint32_t parent_proval;
|
uint32_t parent_proval;
|
||||||
|
uint8_t mtp_pers_uniq_obj_id[16] = {0};
|
||||||
|
|
||||||
MTP_PropertiesList.MTP_Properties_len = SUPP_OBJ_PROP_LEN;
|
MTP_PropertiesList.MTP_Properties_len = SUPP_OBJ_PROP_LEN;
|
||||||
hmtp->ResponseLength = 4U; /* size of MTP_PropertiesList.MTP_Properties_len */
|
hmtp->ResponseLength = 4U; /* size of MTP_PropertiesList.MTP_Properties_len */
|
||||||
@@ -1020,9 +984,13 @@ static void MTP_Get_ObjectPropList(USBD_HandleTypeDef *pdev)
|
|||||||
break;
|
break;
|
||||||
|
|
||||||
case MTP_OB_PROP_PERS_UNIQ_OBJ_IDEN :
|
case MTP_OB_PROP_PERS_UNIQ_OBJ_IDEN :
|
||||||
|
/* Fill in the first 4 bytes with Param1 (Object ID) */
|
||||||
|
(void)USBD_memcpy(mtp_pers_uniq_obj_id, &hmtp->OperationsContainer.Param1,
|
||||||
|
sizeof(hmtp->OperationsContainer.Param1));
|
||||||
|
|
||||||
MTP_PropertiesList.MTP_Properties[i].PropertyCode = MTP_OB_PROP_PERS_UNIQ_OBJ_IDEN;
|
MTP_PropertiesList.MTP_Properties[i].PropertyCode = MTP_OB_PROP_PERS_UNIQ_OBJ_IDEN;
|
||||||
MTP_PropertiesList.MTP_Properties[i].Datatype = MTP_DATATYPE_UINT128;
|
MTP_PropertiesList.MTP_Properties[i].Datatype = MTP_DATATYPE_UINT128;
|
||||||
MTP_PropertiesList.MTP_Properties[i].propval = (uint8_t *)&hmtp->OperationsContainer.Param1;
|
MTP_PropertiesList.MTP_Properties[i].propval = (uint8_t *)mtp_pers_uniq_obj_id;
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case MTP_OB_PROP_PROTECTION_STATUS :
|
case MTP_OB_PROP_PROTECTION_STATUS :
|
||||||
@@ -1187,7 +1155,8 @@ static uint32_t MTP_build_data_proplist(USBD_HandleTypeDef *pdev,
|
|||||||
MTP_PropertiesListTypedef proplist, uint32_t idx)
|
MTP_PropertiesListTypedef proplist, uint32_t idx)
|
||||||
{
|
{
|
||||||
USBD_MTP_HandleTypeDef *hmtp = (USBD_MTP_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId];
|
USBD_MTP_HandleTypeDef *hmtp = (USBD_MTP_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId];
|
||||||
uint8_t propval_size = (MTP_FileName.FileName_len * 2U) + 1U;
|
uint8_t *propval = proplist.MTP_Properties[idx].propval;
|
||||||
|
uint8_t propval_size = (propval[0] * 2U) + 1U;
|
||||||
uint32_t dataLength;
|
uint32_t dataLength;
|
||||||
|
|
||||||
dataLength = offsetof(MTP_PropertiesTypedef, propval);
|
dataLength = offsetof(MTP_PropertiesTypedef, propval);
|
||||||
|
|||||||
@@ -95,26 +95,38 @@ uint8_t USBD_MTP_STORAGE_ReadData(USBD_HandleTypeDef *pdev)
|
|||||||
{
|
{
|
||||||
USBD_MTP_HandleTypeDef *hmtp = (USBD_MTP_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId];
|
USBD_MTP_HandleTypeDef *hmtp = (USBD_MTP_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId];
|
||||||
uint32_t *data_buff;
|
uint32_t *data_buff;
|
||||||
|
uint32_t buffer_size;
|
||||||
|
|
||||||
/* Get the data buffer pointer from the low layer interface */
|
/* Get the data buffer pointer from the low layer interface */
|
||||||
data_buff = ((USBD_MTP_ItfTypeDef *)pdev->pUserData[pdev->classId])->ScratchBuff;
|
data_buff = ((USBD_MTP_ItfTypeDef *)pdev->pUserData[pdev->classId])->ScratchBuff;
|
||||||
|
|
||||||
|
/* Get Data Buffer Size */
|
||||||
|
buffer_size = ((USBD_MTP_ItfTypeDef *)pdev->pUserData[pdev->classId])->ScratchBuffSze;
|
||||||
|
|
||||||
|
if ((data_buff == NULL) || (buffer_size < MTP_CONT_HEADER_SIZE))
|
||||||
|
{
|
||||||
|
return (uint8_t)USBD_FAIL;
|
||||||
|
}
|
||||||
|
|
||||||
switch (ReadDataStatus)
|
switch (ReadDataStatus)
|
||||||
{
|
{
|
||||||
case READ_FIRST_DATA:
|
case READ_FIRST_DATA:
|
||||||
/* Reset the data length */
|
/* Reset the data length */
|
||||||
MTP_DataLength.temp_length = 0U;
|
MTP_DataLength.temp_length = 0U;
|
||||||
|
|
||||||
/* Perform the low layer read operation on the scratch buffer */
|
|
||||||
(void)((USBD_MTP_ItfTypeDef *)pdev->pUserData[pdev->classId])->ReadData(hmtp->OperationsContainer.Param1,
|
|
||||||
(uint8_t *)data_buff, &MTP_DataLength);
|
|
||||||
|
|
||||||
/* Add the container header to the data buffer */
|
/* Add the container header to the data buffer */
|
||||||
(void)USBD_memcpy((uint8_t *)data_buff, (uint8_t *)&hmtp->GenericContainer, MTP_CONT_HEADER_SIZE);
|
(void)USBD_memcpy((uint8_t *)data_buff, (uint8_t *)&hmtp->GenericContainer, MTP_CONT_HEADER_SIZE);
|
||||||
|
|
||||||
|
/* Perform the low layer read operation on the scratch buffer
|
||||||
|
* first packet expected data length: MPS - MTP_CONT_HEADER_SIZE
|
||||||
|
*/
|
||||||
|
(void)((USBD_MTP_ItfTypeDef *)pdev->pUserData[pdev->classId])->ReadData(hmtp->OperationsContainer.Param1,
|
||||||
|
(uint8_t *)data_buff +
|
||||||
|
MTP_CONT_HEADER_SIZE, &MTP_DataLength);
|
||||||
|
|
||||||
/* Start USB data transmission to the host */
|
/* Start USB data transmission to the host */
|
||||||
(void)USBD_MTP_STORAGE_SendData(pdev, (uint8_t *)data_buff,
|
(void)USBD_MTP_STORAGE_SendData(pdev, (uint8_t *)data_buff,
|
||||||
MTP_DataLength.readbytes + MTP_CONT_HEADER_SIZE);
|
MIN((MTP_DataLength.readbytes + MTP_CONT_HEADER_SIZE), buffer_size));
|
||||||
|
|
||||||
/* Check if this will be the last packet to send ? */
|
/* Check if this will be the last packet to send ? */
|
||||||
if (MTP_DataLength.readbytes < ((uint32_t)hmtp->MaxPcktLen - MTP_CONT_HEADER_SIZE))
|
if (MTP_DataLength.readbytes < ((uint32_t)hmtp->MaxPcktLen - MTP_CONT_HEADER_SIZE))
|
||||||
@@ -138,24 +150,45 @@ uint8_t USBD_MTP_STORAGE_ReadData(USBD_HandleTypeDef *pdev)
|
|||||||
if (MTP_DataLength.temp_length == MTP_DataLength.totallen)
|
if (MTP_DataLength.temp_length == MTP_DataLength.totallen)
|
||||||
{
|
{
|
||||||
/* Start USB data transmission to the host */
|
/* Start USB data transmission to the host */
|
||||||
(void)USBD_MTP_STORAGE_SendData(pdev, (uint8_t *)data_buff, MTP_DataLength.readbytes);
|
(void)USBD_MTP_STORAGE_SendData(pdev, (uint8_t *)data_buff, MIN(MTP_DataLength.readbytes, buffer_size));
|
||||||
|
|
||||||
/* Move to response phase */
|
if (((MTP_DataLength.totallen + MTP_CONT_HEADER_SIZE) % hmtp->MaxPcktLen) == 0U)
|
||||||
hmtp->MTP_ResponsePhase = MTP_RESPONSE_PHASE;
|
{
|
||||||
|
/* Send ZLP Packet */
|
||||||
|
ReadDataStatus = READ_SEND_ZLP_DATA;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
/* Move to response phase */
|
||||||
|
hmtp->MTP_ResponsePhase = MTP_RESPONSE_PHASE;
|
||||||
|
|
||||||
/* Reset the stat machine */
|
/* Reset the state machine */
|
||||||
ReadDataStatus = READ_FIRST_DATA;
|
ReadDataStatus = READ_FIRST_DATA;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
/* Start USB data transmission to the host */
|
/* Start USB data transmission to the host */
|
||||||
(void)USBD_MTP_STORAGE_SendData(pdev, (uint8_t *)data_buff, MTP_DataLength.readbytes);
|
(void)USBD_MTP_STORAGE_SendData(pdev, (uint8_t *)data_buff, MIN(MTP_DataLength.readbytes, buffer_size));
|
||||||
|
|
||||||
/* Keep the state machine into sending next packet of data */
|
/* Keep the state machine into sending next packet of data */
|
||||||
ReadDataStatus = READ_REST_OF_DATA;
|
ReadDataStatus = READ_REST_OF_DATA;
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
|
||||||
|
case READ_SEND_ZLP_DATA:
|
||||||
|
|
||||||
|
/* Send ZLP to the host */
|
||||||
|
(void)USBD_MTP_STORAGE_SendData(pdev, (uint8_t *)data_buff, 0U);
|
||||||
|
|
||||||
|
/* Move to response phase */
|
||||||
|
hmtp->MTP_ResponsePhase = MTP_RESPONSE_PHASE;
|
||||||
|
|
||||||
|
/* Reset the state machine */
|
||||||
|
ReadDataStatus = READ_FIRST_DATA;
|
||||||
|
|
||||||
|
break;
|
||||||
|
|
||||||
default:
|
default:
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
@@ -176,7 +209,8 @@ uint8_t USBD_MTP_STORAGE_SendContainer(USBD_HandleTypeDef *pdev, MTP_CONTAINER_
|
|||||||
{
|
{
|
||||||
case DATA_TYPE:
|
case DATA_TYPE:
|
||||||
/* send header + data : hmtp->ResponseLength = header size + data size */
|
/* send header + data : hmtp->ResponseLength = header size + data size */
|
||||||
(void)USBD_MTP_STORAGE_SendData(pdev, (uint8_t *)&hmtp->GenericContainer, hmtp->ResponseLength);
|
(void)USBD_MTP_STORAGE_SendData(pdev, (uint8_t *)&hmtp->GenericContainer, MIN(sizeof(MTP_GenericContainerTypeDef),
|
||||||
|
hmtp->ResponseLength));
|
||||||
break;
|
break;
|
||||||
case REP_TYPE:
|
case REP_TYPE:
|
||||||
/* send header without data */
|
/* send header without data */
|
||||||
@@ -185,7 +219,8 @@ uint8_t USBD_MTP_STORAGE_SendContainer(USBD_HandleTypeDef *pdev, MTP_CONTAINER_
|
|||||||
hmtp->GenericContainer.length = hmtp->ResponseLength;
|
hmtp->GenericContainer.length = hmtp->ResponseLength;
|
||||||
hmtp->GenericContainer.type = MTP_CONT_TYPE_RESPONSE;
|
hmtp->GenericContainer.type = MTP_CONT_TYPE_RESPONSE;
|
||||||
|
|
||||||
(void)USBD_MTP_STORAGE_SendData(pdev, (uint8_t *)&hmtp->GenericContainer, hmtp->ResponseLength);
|
(void)USBD_MTP_STORAGE_SendData(pdev, (uint8_t *)&hmtp->GenericContainer, MIN(sizeof(MTP_GenericContainerTypeDef),
|
||||||
|
hmtp->ResponseLength));
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
break;
|
break;
|
||||||
@@ -245,7 +280,7 @@ uint8_t USBD_MTP_STORAGE_ReceiveOpt(USBD_HandleTypeDef *pdev)
|
|||||||
*/
|
*/
|
||||||
uint8_t USBD_MTP_STORAGE_ReceiveData(USBD_HandleTypeDef *pdev)
|
uint8_t USBD_MTP_STORAGE_ReceiveData(USBD_HandleTypeDef *pdev)
|
||||||
{
|
{
|
||||||
USBD_MTP_HandleTypeDef *hmtp = (USBD_MTP_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId];
|
USBD_MTP_HandleTypeDef *hmtp = (USBD_MTP_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId];
|
||||||
switch (hmtp->RECEIVE_DATA_STATUS)
|
switch (hmtp->RECEIVE_DATA_STATUS)
|
||||||
{
|
{
|
||||||
case RECEIVE_COMMAND_DATA :
|
case RECEIVE_COMMAND_DATA :
|
||||||
@@ -410,15 +445,21 @@ static uint8_t USBD_MTP_STORAGE_DecodeOperations(USBD_HandleTypeDef *pdev)
|
|||||||
static uint8_t USBD_MTP_STORAGE_ReceiveContainer(USBD_HandleTypeDef *pdev,
|
static uint8_t USBD_MTP_STORAGE_ReceiveContainer(USBD_HandleTypeDef *pdev,
|
||||||
uint32_t *pDst, uint32_t len)
|
uint32_t *pDst, uint32_t len)
|
||||||
{
|
{
|
||||||
USBD_MTP_HandleTypeDef *hmtp = (USBD_MTP_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId];
|
USBD_MTP_HandleTypeDef *hmtp = (USBD_MTP_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId];
|
||||||
uint32_t Counter;
|
uint32_t Counter;
|
||||||
uint32_t *pdst = pDst;
|
uint32_t *pdst = pDst;
|
||||||
|
|
||||||
for (Counter = 0; Counter < len; Counter++)
|
if ((pDst == NULL) || (len > MTP_MEDIA_PACKET))
|
||||||
|
{
|
||||||
|
return (uint8_t)USBD_FAIL;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (Counter = 0U; Counter < len; Counter++)
|
||||||
{
|
{
|
||||||
*pdst = (hmtp->rx_buff[Counter]);
|
*pdst = (hmtp->rx_buff[Counter]);
|
||||||
pdst++;
|
pdst++;
|
||||||
}
|
}
|
||||||
|
|
||||||
return (uint8_t)USBD_OK;
|
return (uint8_t)USBD_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -456,17 +497,14 @@ void USBD_MTP_STORAGE_Cancel(USBD_HandleTypeDef *pdev,
|
|||||||
* @param len: Data Length
|
* @param len: Data Length
|
||||||
* @retval status value
|
* @retval status value
|
||||||
*/
|
*/
|
||||||
static uint8_t USBD_MTP_STORAGE_SendData(USBD_HandleTypeDef *pdev, uint8_t *buf,
|
static uint8_t USBD_MTP_STORAGE_SendData(USBD_HandleTypeDef *pdev, uint8_t *buf, uint32_t len)
|
||||||
uint32_t len)
|
|
||||||
{
|
{
|
||||||
USBD_MTP_HandleTypeDef *hmtp = (USBD_MTP_HandleTypeDef *)pdev->pClassDataCmsit[pdev->classId];
|
|
||||||
uint32_t length = MIN(hmtp->GenericContainer.length, len);
|
|
||||||
#ifdef USE_USBD_COMPOSITE
|
#ifdef USE_USBD_COMPOSITE
|
||||||
/* Get the Endpoints addresses allocated for this class instance */
|
/* Get the Endpoints addresses allocated for this class instance */
|
||||||
MTPInEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_IN, USBD_EP_TYPE_BULK, (uint8_t)pdev->classId);
|
MTPInEpAdd = USBD_CoreGetEPAdd(pdev, USBD_EP_IN, USBD_EP_TYPE_BULK, (uint8_t)pdev->classId);
|
||||||
#endif /* USE_USBD_COMPOSITE */
|
#endif /* USE_USBD_COMPOSITE */
|
||||||
|
|
||||||
(void)USBD_LL_Transmit(pdev, MTPInEpAdd, buf, length);
|
(void)USBD_LL_Transmit(pdev, MTPInEpAdd, buf, len);
|
||||||
|
|
||||||
return (uint8_t)USBD_OK;
|
return (uint8_t)USBD_OK;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -141,6 +141,9 @@ USBD_ClassTypeDef USBD_PRNT =
|
|||||||
USBD_PRNT_GetOtherSpeedCfgDesc,
|
USBD_PRNT_GetOtherSpeedCfgDesc,
|
||||||
USBD_PRNT_GetDeviceQualifierDescriptor,
|
USBD_PRNT_GetDeviceQualifierDescriptor,
|
||||||
#endif /* USE_USBD_COMPOSITE */
|
#endif /* USE_USBD_COMPOSITE */
|
||||||
|
#if (USBD_SUPPORT_USER_STRING_DESC == 1U)
|
||||||
|
NULL,
|
||||||
|
#endif /* USBD_SUPPORT_USER_STRING_DESC */
|
||||||
};
|
};
|
||||||
|
|
||||||
#ifndef USE_USBD_COMPOSITE
|
#ifndef USE_USBD_COMPOSITE
|
||||||
|
|||||||
@@ -140,6 +140,9 @@ USBD_ClassTypeDef USBD_VIDEO =
|
|||||||
USBD_VIDEO_GetOtherSpeedCfgDesc,
|
USBD_VIDEO_GetOtherSpeedCfgDesc,
|
||||||
USBD_VIDEO_GetDeviceQualifierDesc,
|
USBD_VIDEO_GetDeviceQualifierDesc,
|
||||||
#endif /* USE_USBD_COMPOSITE */
|
#endif /* USE_USBD_COMPOSITE */
|
||||||
|
#if (USBD_SUPPORT_USER_STRING_DESC == 1U)
|
||||||
|
NULL,
|
||||||
|
#endif /* USBD_SUPPORT_USER_STRING_DESC */
|
||||||
};
|
};
|
||||||
|
|
||||||
/* USB VIDEO device Configuration Descriptor (same for all speeds thanks to user defines) */
|
/* USB VIDEO device Configuration Descriptor (same for all speeds thanks to user defines) */
|
||||||
|
|||||||
@@ -79,10 +79,12 @@ extern "C" {
|
|||||||
#define USBD_CUSTOMHID_OUTREPORT_BUF_SIZE 0x02U
|
#define USBD_CUSTOMHID_OUTREPORT_BUF_SIZE 0x02U
|
||||||
#define USBD_CUSTOM_HID_REPORT_DESC_SIZE 163U
|
#define USBD_CUSTOM_HID_REPORT_DESC_SIZE 163U
|
||||||
|
|
||||||
|
/* #define USBD_CUSTOMHID_REPORT_DESC_SIZE_ENABLED */
|
||||||
/* #define USBD_CUSTOMHID_CTRL_REQ_GET_REPORT_ENABLED */
|
/* #define USBD_CUSTOMHID_CTRL_REQ_GET_REPORT_ENABLED */
|
||||||
/* #define USBD_CUSTOMHID_OUT_PREPARE_RECEIVE_DISABLED */
|
/* #define USBD_CUSTOMHID_OUT_PREPARE_RECEIVE_DISABLED */
|
||||||
/* #define USBD_CUSTOMHID_EP0_OUT_PREPARE_RECEIVE_DISABLED */
|
/* #define USBD_CUSTOMHID_EP0_OUT_PREPARE_RECEIVE_DISABLED */
|
||||||
/* #define USBD_CUSTOMHID_CTRL_REQ_COMPLETE_CALLBACK_ENABLED */
|
/* #define USBD_CUSTOMHID_CTRL_REQ_COMPLETE_CALLBACK_ENABLED */
|
||||||
|
/* #define USBD_CUSTOMHID_REPORT_BUFFER_EVENT_ENABLED */
|
||||||
|
|
||||||
/* VIDEO Class Config */
|
/* VIDEO Class Config */
|
||||||
#define UVC_1_1 /* #define UVC_1_0 */
|
#define UVC_1_1 /* #define UVC_1_0 */
|
||||||
|
|||||||
@@ -305,12 +305,13 @@ typedef struct
|
|||||||
/* USB Device handle structure */
|
/* USB Device handle structure */
|
||||||
typedef struct
|
typedef struct
|
||||||
{
|
{
|
||||||
uint32_t status;
|
|
||||||
uint32_t total_length;
|
uint32_t total_length;
|
||||||
uint32_t rem_length;
|
uint32_t rem_length;
|
||||||
uint32_t maxpacket;
|
uint32_t bInterval;
|
||||||
uint16_t is_used;
|
uint16_t maxpacket;
|
||||||
uint16_t bInterval;
|
uint8_t status;
|
||||||
|
uint8_t is_used;
|
||||||
|
uint8_t *pbuffer;
|
||||||
} USBD_EndpointTypeDef;
|
} USBD_EndpointTypeDef;
|
||||||
|
|
||||||
#ifdef USE_USBD_COMPOSITE
|
#ifdef USE_USBD_COMPOSITE
|
||||||
|
|||||||
@@ -25,6 +25,10 @@
|
|||||||
|
|
||||||
/* Exported types ------------------------------------------------------------*/
|
/* Exported types ------------------------------------------------------------*/
|
||||||
/* Exported constants --------------------------------------------------------*/
|
/* Exported constants --------------------------------------------------------*/
|
||||||
|
/*
|
||||||
|
* User to provide a unique ID to define the USB device serial number
|
||||||
|
* The use of UID_BASE register can be considered as an example
|
||||||
|
*/
|
||||||
#define DEVICE_ID1 (UID_BASE)
|
#define DEVICE_ID1 (UID_BASE)
|
||||||
#define DEVICE_ID2 (UID_BASE + 0x4U)
|
#define DEVICE_ID2 (UID_BASE + 0x4U)
|
||||||
#define DEVICE_ID3 (UID_BASE + 0x8U)
|
#define DEVICE_ID3 (UID_BASE + 0x8U)
|
||||||
|
|||||||
@@ -296,7 +296,7 @@ USBD_StatusTypeDef USBD_RegisterClassComposite(USBD_HandleTypeDef *pdev, USBD_Cl
|
|||||||
*/
|
*/
|
||||||
USBD_StatusTypeDef USBD_UnRegisterClassComposite(USBD_HandleTypeDef *pdev)
|
USBD_StatusTypeDef USBD_UnRegisterClassComposite(USBD_HandleTypeDef *pdev)
|
||||||
{
|
{
|
||||||
USBD_StatusTypeDef ret = USBD_FAIL;
|
USBD_StatusTypeDef ret = USBD_OK;
|
||||||
uint8_t idx1;
|
uint8_t idx1;
|
||||||
uint8_t idx2;
|
uint8_t idx2;
|
||||||
|
|
||||||
@@ -590,6 +590,8 @@ USBD_StatusTypeDef USBD_LL_DataOutStage(USBD_HandleTypeDef *pdev,
|
|||||||
USBD_StatusTypeDef ret = USBD_OK;
|
USBD_StatusTypeDef ret = USBD_OK;
|
||||||
uint8_t idx;
|
uint8_t idx;
|
||||||
|
|
||||||
|
UNUSED(pdata);
|
||||||
|
|
||||||
if (epnum == 0U)
|
if (epnum == 0U)
|
||||||
{
|
{
|
||||||
pep = &pdev->ep_out[0];
|
pep = &pdev->ep_out[0];
|
||||||
@@ -599,8 +601,9 @@ USBD_StatusTypeDef USBD_LL_DataOutStage(USBD_HandleTypeDef *pdev,
|
|||||||
if (pep->rem_length > pep->maxpacket)
|
if (pep->rem_length > pep->maxpacket)
|
||||||
{
|
{
|
||||||
pep->rem_length -= pep->maxpacket;
|
pep->rem_length -= pep->maxpacket;
|
||||||
|
pep->pbuffer += pep->maxpacket;
|
||||||
|
|
||||||
(void)USBD_CtlContinueRx(pdev, pdata, MIN(pep->rem_length, pep->maxpacket));
|
(void)USBD_CtlContinueRx(pdev, pep->pbuffer, MAX(pep->rem_length, pep->maxpacket));
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
@@ -685,6 +688,8 @@ USBD_StatusTypeDef USBD_LL_DataInStage(USBD_HandleTypeDef *pdev,
|
|||||||
USBD_StatusTypeDef ret;
|
USBD_StatusTypeDef ret;
|
||||||
uint8_t idx;
|
uint8_t idx;
|
||||||
|
|
||||||
|
UNUSED(pdata);
|
||||||
|
|
||||||
if (epnum == 0U)
|
if (epnum == 0U)
|
||||||
{
|
{
|
||||||
pep = &pdev->ep_in[0];
|
pep = &pdev->ep_in[0];
|
||||||
@@ -694,8 +699,9 @@ USBD_StatusTypeDef USBD_LL_DataInStage(USBD_HandleTypeDef *pdev,
|
|||||||
if (pep->rem_length > pep->maxpacket)
|
if (pep->rem_length > pep->maxpacket)
|
||||||
{
|
{
|
||||||
pep->rem_length -= pep->maxpacket;
|
pep->rem_length -= pep->maxpacket;
|
||||||
|
pep->pbuffer += pep->maxpacket;
|
||||||
|
|
||||||
(void)USBD_CtlContinueSendData(pdev, pdata, pep->rem_length);
|
(void)USBD_CtlContinueSendData(pdev, pep->pbuffer, pep->rem_length);
|
||||||
|
|
||||||
/* Prepare endpoint for premature end of transfer */
|
/* Prepare endpoint for premature end of transfer */
|
||||||
(void)USBD_LL_PrepareReceive(pdev, 0U, NULL, 0U);
|
(void)USBD_LL_PrepareReceive(pdev, 0U, NULL, 0U);
|
||||||
|
|||||||
@@ -65,7 +65,7 @@ USBD_DescriptorsTypeDef Class_Desc =
|
|||||||
USBD_Class_ConfigStrDescriptor,
|
USBD_Class_ConfigStrDescriptor,
|
||||||
USBD_Class_InterfaceStrDescriptor,
|
USBD_Class_InterfaceStrDescriptor,
|
||||||
#if (USBD_CLASS_USER_STRING_DESC == 1)
|
#if (USBD_CLASS_USER_STRING_DESC == 1)
|
||||||
USBD_CLASS_UserStrDescriptor,
|
USBD_Class_UserStrDescriptor,
|
||||||
#endif /* USB_CLASS_USER_STRING_DESC */
|
#endif /* USB_CLASS_USER_STRING_DESC */
|
||||||
|
|
||||||
#if ((USBD_LPM_ENABLED == 1) || (USBD_CLASS_BOS_ENABLED == 1))
|
#if ((USBD_LPM_ENABLED == 1) || (USBD_CLASS_BOS_ENABLED == 1))
|
||||||
|
|||||||
@@ -89,6 +89,7 @@ USBD_StatusTypeDef USBD_CtlSendData(USBD_HandleTypeDef *pdev,
|
|||||||
/* Set EP0 State */
|
/* Set EP0 State */
|
||||||
pdev->ep0_state = USBD_EP0_DATA_IN;
|
pdev->ep0_state = USBD_EP0_DATA_IN;
|
||||||
pdev->ep_in[0].total_length = len;
|
pdev->ep_in[0].total_length = len;
|
||||||
|
pdev->ep_in[0].pbuffer = pbuf;
|
||||||
|
|
||||||
#ifdef USBD_AVOID_PACKET_SPLIT_MPS
|
#ifdef USBD_AVOID_PACKET_SPLIT_MPS
|
||||||
pdev->ep_in[0].rem_length = 0U;
|
pdev->ep_in[0].rem_length = 0U;
|
||||||
@@ -133,6 +134,7 @@ USBD_StatusTypeDef USBD_CtlPrepareRx(USBD_HandleTypeDef *pdev,
|
|||||||
/* Set EP0 State */
|
/* Set EP0 State */
|
||||||
pdev->ep0_state = USBD_EP0_DATA_OUT;
|
pdev->ep0_state = USBD_EP0_DATA_OUT;
|
||||||
pdev->ep_out[0].total_length = len;
|
pdev->ep_out[0].total_length = len;
|
||||||
|
pdev->ep_out[0].pbuffer = pbuf;
|
||||||
|
|
||||||
#ifdef USBD_AVOID_PACKET_SPLIT_MPS
|
#ifdef USBD_AVOID_PACKET_SPLIT_MPS
|
||||||
pdev->ep_out[0].rem_length = 0U;
|
pdev->ep_out[0].rem_length = 0U;
|
||||||
|
|||||||
134
LICENSE.md
134
LICENSE.md
@@ -1,80 +1,94 @@
|
|||||||
SLA0044 Rev5/February 2018
|
This software component is provided to you as part of a software package and
|
||||||
|
applicable license terms are in the Package_license file. If you received this
|
||||||
|
software component outside of a package or without applicable license terms,
|
||||||
|
the terms of the SLA0044 license shall apply and are fully reproduced below:
|
||||||
|
|
||||||
|
SLA0044 Rev6/October 2025
|
||||||
|
|
||||||
## Software license agreement
|
## Software license agreement
|
||||||
|
|
||||||
### __ULTIMATE LIBERTY SOFTWARE LICENSE AGREEMENT__
|
### __ULTIMATE LIBERTY SOFTWARE LICENSE AGREEMENT__
|
||||||
|
|
||||||
BY INSTALLING, COPYING, DOWNLOADING, ACCESSING OR OTHERWISE USING THIS SOFTWARE
|
BY CLICKING ON THE "I ACCEPT" BUTTON OR BY UNZIPPING, INSTALLING, COPYING,
|
||||||
OR ANY PART THEREOF (AND THE RELATED DOCUMENTATION) FROM STMICROELECTRONICS
|
DOWNLOADING, ACCESSING OR OTHERWISE USING THIS SOFTWARE OR ANY PART THEREOF,
|
||||||
INTERNATIONAL N.V, SWISS BRANCH AND/OR ITS AFFILIATED COMPANIES
|
INCLUDING ANY RELATED DOCUMENTATION (collectively the “SOFTWARE”)
|
||||||
(STMICROELECTRONICS), THE RECIPIENT, ON BEHALF OF HIMSELF OR HERSELF, OR ON
|
FROM STMICROELECTRONICS INTERNATIONAL N.V, SWISS BRANCH AND/OR
|
||||||
BEHALF OF ANY ENTITY BY WHICH SUCH RECIPIENT IS EMPLOYED AND/OR ENGAGED AGREES
|
ITS AFFILIATED COMPANIES (collectively “STMICROELECTRONICS”),
|
||||||
TO BE BOUND BY THIS SOFTWARE LICENSE AGREEMENT.
|
YOU (hereinafter referred also to as “THE RECIPIENT”), ON BEHALF OF YOURSELF,
|
||||||
|
OR ON BEHALF OF ANY ENTITY BY WHICH YOU ARE EMPLOYED AND/OR ENGAGED,
|
||||||
|
AGREE TO BE BOUND BY THIS AGREEMENT.
|
||||||
|
|
||||||
Under STMicroelectronics’ intellectual property rights, the redistribution,
|
Under STMICROELECTRONICS’ intellectual property rights, the redistribution,
|
||||||
reproduction and use in source and binary forms of the software or any part
|
reproduction and use in source and binary forms of the SOFTWARE or any part
|
||||||
thereof, with or without modification, are permitted provided that the following
|
thereof, with or without modification, are permitted provided that the following
|
||||||
conditions are met:
|
conditions are met:
|
||||||
|
|
||||||
1. Redistribution of source code (modified or not) must retain any copyright
|
1. Redistribution of source code (modified or not) must retain any copyright
|
||||||
notice, this list of conditions and the disclaimer set forth below as items 10
|
notice accompanying the SOFTWARE, this list of conditions and the disclaimer below.
|
||||||
and 11.
|
|
||||||
|
|
||||||
2. Redistributions in binary form, except as embedded into microcontroller or
|
2. Redistributions in binary form, except as embedded into a processing unit device
|
||||||
microprocessor device manufactured by or for STMicroelectronics or a software
|
manufactured by or for STMicroelectronics or a software update for any such device,
|
||||||
update for such device, must reproduce any copyright notice provided with the
|
must reproduce the accompanying copyright notice, this list of conditions,
|
||||||
binary code, this list of conditions, and the disclaimer set forth below as
|
and the below disclaimer in capital type, in the documentation and/or
|
||||||
items 10 and 11, in documentation and/or other materials provided with the
|
other materials provided with the distribution.
|
||||||
distribution.
|
|
||||||
|
|
||||||
3. Neither the name of STMicroelectronics nor the names of other contributors to
|
3. Neither the name of STMicroelectronics nor the names of other contributors
|
||||||
this software may be used to endorse or promote products derived from this
|
to the SOFTWARE may be used to endorse or promote products derived
|
||||||
software or part thereof without specific written permission.
|
from the SOFTWARE or part thereof without specific written permission.
|
||||||
|
|
||||||
4. This software or any part thereof, including modifications and/or derivative
|
4. The SOFTWARE or any part thereof, including modifications and/or
|
||||||
works of this software, must be used and execute solely and exclusively on or in
|
derivative works of the SOFTWARE, must be used and execute solely
|
||||||
combination with a microcontroller or microprocessor device manufactured by or
|
and exclusively on or in combination with a processing unit device
|
||||||
for STMicroelectronics.
|
manufactured by or for STMicroelectronics.
|
||||||
|
|
||||||
5. No use, reproduction or redistribution of this software partially or totally
|
5. No use, reproduction or redistribution of the SOFTWARE partially
|
||||||
may be done in any manner that would subject this software to any Open Source
|
or totally may be done in any manner that would subject the SOFTWARE
|
||||||
Terms. “Open Source Terms” shall mean any open source license which requires as
|
to any Open Source Terms. “Open Source Terms” shall mean
|
||||||
part of distribution of software that the source code of such software is
|
any open source license which requires as part of distribution
|
||||||
distributed therewith or otherwise made available, or open source license that
|
of software that the source code of such software is distributed
|
||||||
substantially complies with the Open Source definition specified at
|
therewith or otherwise made available, or open source license
|
||||||
www.opensource.org and any other comparable open source license such as for
|
that substantially complies with the Open Source definition specified
|
||||||
example GNU General Public License (GPL), Eclipse Public License (EPL), Apache
|
at www.opensource.org and any other comparable open source license
|
||||||
Software License, BSD license or MIT license.
|
such as for example GNU General Public License (GPL),
|
||||||
|
Eclipse Public License (EPL), Apache Software License, BSD license
|
||||||
|
or MIT license.
|
||||||
|
|
||||||
6. STMicroelectronics has no obligation to provide any maintenance, support or
|
6. STMicroelectronics has no obligation to provide any maintenance,
|
||||||
updates for the software.
|
support or updates for the SOFTWARE.
|
||||||
|
|
||||||
7. The software is and will remain the exclusive property of STMicroelectronics
|
7. The SOFTWARE is and will remain the exclusive property of
|
||||||
and its licensors. The recipient will not take any action that jeopardizes
|
STMicroelectronics and its licensors. The RECIPIENT will not take
|
||||||
STMicroelectronics and its licensors' proprietary rights or acquire any rights
|
any action that jeopardizes STMicroelectronics and its
|
||||||
in the software, except the limited rights specified hereunder.
|
licensors' proprietary rights or acquire any rights in the SOFTWARE,
|
||||||
|
except the limited rights specified hereunder.
|
||||||
|
|
||||||
8. The recipient shall comply with all applicable laws and regulations affecting
|
8. The RECIPIENT shall comply with all applicable laws and regulations
|
||||||
the use of the software or any part thereof including any applicable export
|
affecting the use of the SOFTWARE or any part thereof including
|
||||||
control law or regulation.
|
any applicable export control law or regulation.
|
||||||
|
|
||||||
9. Redistribution and use of this software or any part thereof other than as
|
9. Redistribution and use of the SOFTWARE or any part thereof other
|
||||||
permitted under this license is void and will automatically terminate your
|
than as permitted under this AGREEMENT is void and will automatically
|
||||||
rights under this license.
|
terminate RECIPIENT’s rights under this AGREEMENT.
|
||||||
|
|
||||||
10. THIS SOFTWARE IS PROVIDED BY STMICROELECTRONICS AND CONTRIBUTORS "AS IS" AND
|
10. The RECIPIENT shall be solely liable to determine and verify that
|
||||||
ANY EXPRESS, IMPLIED OR STATUTORY WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
the SOFTWARE is fit for the RECIPIENT intended use, environment or
|
||||||
IMPLIED WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
application and comply with all regulatory, safety and security
|
||||||
NON-INFRINGEMENT OF THIRD PARTY INTELLECTUAL PROPERTY RIGHTS, WHICH ARE
|
related requirements concerning any use.
|
||||||
DISCLAIMED TO THE FULLEST EXTENT PERMITTED BY LAW. IN NO EVENT SHALL
|
|
||||||
STMICROELECTRONICS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
|
|
||||||
INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
|
||||||
LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
|
|
||||||
PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
|
|
||||||
LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE
|
|
||||||
OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
|
|
||||||
ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
||||||
|
|
||||||
11. EXCEPT AS EXPRESSLY PERMITTED HEREUNDER, NO LICENSE OR OTHER RIGHTS, WHETHER
|
DISCLAIMER:
|
||||||
EXPRESS OR IMPLIED, ARE GRANTED UNDER ANY PATENT OR OTHER INTELLECTUAL PROPERTY
|
|
||||||
RIGHTS OF STMICROELECTRONICS OR ANY THIRD PARTY.
|
THE SOFTWARE IS PROVIDED BY STMICROELECTRONICS AND CONTRIBUTORS "AS IS" AND
|
||||||
|
ANY EXPRESS, IMPLIED OR STATUTORY WARRANTIES, INCLUDING,BUT NOT LIMITED TO,
|
||||||
|
THE IMPLIED WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE
|
||||||
|
AND NON-INFRINGEMENT OF THIRD PARTY INTELLECTUAL PROPERTY RIGHTS, ARE
|
||||||
|
DISCLAIMED TO THE FULLEST EXTENT PERMITTED BY LAW.
|
||||||
|
IN NO EVENT SHALL STMICROELECTRONICS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT,
|
||||||
|
INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
||||||
|
(INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
||||||
|
LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
||||||
|
ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||||
|
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THE
|
||||||
|
SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
|
EXCEPT AS EXPRESSLY PERMITTED HEREUNDER, NO LICENSE OR OTHER RIGHTS,
|
||||||
|
WHETHER EXPRESS OR IMPLIED, ARE GRANTED UNDER ANY PATENT OR OTHER INTELLECTUAL
|
||||||
|
PROPERTY RIGHTS OF STMICROELECTRONICS OR ANY THIRD PARTY.
|
||||||
|
|||||||
48
README.md
48
README.md
@@ -1,36 +1,40 @@
|
|||||||
# Middleware USB Device MCU Component
|
# USB Device Middleware Library MCU Software Component
|
||||||
|
|
||||||

|

|
||||||
|
|
||||||
## Overview
|
## Overview of the STM32Cube MCU offer on GitHub
|
||||||
|
|
||||||
**STM32Cube** is an STMicroelectronics original initiative to ease developers' life by reducing efforts, time and cost.
|
**STM32Cube** is an original initiative by STMicroelectronics to **simplify** prototyping and development by **reducing** effort, time, and cost. It supports the entire ARM™ Cortex-based STM32 microcontroller portfolio and provides a **comprehensive** software solution including:
|
||||||
|
* The CMSIS Core and Device interfaces enabling access to processor core features and device-specific peripherals of STM32 microcontrollers.
|
||||||
|
* The STM32 HAL-LL drivers, an abstraction layer offering a set of APIs ensuring maximized portability across the STM32 portfolio.
|
||||||
|
* The BSP drivers enabling access to peripherals on the STM32 development boards, external to the microcontroller itself.
|
||||||
|
* A consistent set of middleware libraries offering standardized, high-level functionalities — such as USB, TCP/IP, file systems, and graphics.
|
||||||
|
* A full set of software projects (basic examples, applications, and demonstrations) that showcase specific functionalities or use cases, and provided with support for multiple IDEs.
|
||||||
|
|
||||||
**STM32Cube** covers the overall STM32 products portfolio. It includes a comprehensive embedded software platform delivered for each STM32 series.
|
The **STM32Cube embedded software** is available in two flavors:
|
||||||
* The CMSIS modules (core and device) corresponding to the ARM(tm) core implemented in this STM32 product.
|
* The **MCU Firmware** _monolithic_ offer, where **all** software components (Drivers, Middleware, Projects, Utilities) are included in a **single** repository for each STM32 series.
|
||||||
* The STM32 HAL-LL drivers, an abstraction layer offering a set of APIs ensuring maximized portability across the STM32 portfolio.
|
* The **MCU Software Components** _modular_ offer, where **each** software component (mainly Drivers and Middleware) is provided in a **dedicated** repository, allowing users to **select** only the components they need.
|
||||||
* The BSP drivers of each evaluation, demonstration or nucleo board provided for this STM32 series.
|
|
||||||
* A consistent set of middleware libraries such as RTOS, USB, FatFS, graphics, touch sensing library...
|
|
||||||
* A full set of software projects (basic examples, applications, and demonstrations) for each board provided for this STM32 series.
|
|
||||||
|
|
||||||
Two models of publication are proposed for the STM32Cube embedded software:
|
The complete list of repositories is available [here](https://github.com/STMicroelectronics/STM32Cube_MCU_Overall_Offer/blob/master/README.md#content).
|
||||||
* The monolithic **MCU Package**: all STM32Cube software modules of one STM32 series are present (Drivers, Middleware, Projects, Utilities) in the repository (usual name **STM32Cubexx**, xx corresponding to the STM32 series).
|
|
||||||
* The **MCU component**: each STM32Cube software module being part of the STM32Cube MCU Package, is delivered as an individual repository, allowing the user to select and get only the required software functions.
|
|
||||||
|
|
||||||
## Description
|
## Repository content
|
||||||
|
|
||||||
This **stm32_mw_usb_device** MCU component repository is one element **common to all** STM32Cube MCU embedded software packages, providing the **USB Device MCU Middleware** part.
|
This repository provides the STM32Cube **USB Device** Middleware Library, **common to every** STM32Cube MCU firmware that supports it.
|
||||||
|
|
||||||
It represents ST offer to ensure the support of USB Devices on STM32 MCUs.
|
This represents STMicroelectronics’ offering to support the USB Device role on STM32 MCUs. It includes two main modules:
|
||||||
It includes two main modules:
|
|
||||||
* **Core** module for the USB device standard peripheral control APIs. It includes the files ensuring USB 2.0 standard code implementation for an USB device.
|
- **Core**: provides USB Device standard peripheral control APIs implementing the USB 2.0 standard code for a USB device. These APIs are called in every USB Device application, regardless of the desired functionality.
|
||||||
These files’ APIs will be called within every USB device application regardless of the desired functionality.
|
|
||||||
* **Class** module for the commonly supported classes APIs. it includes the files including different USB device classes. All STM32 USB classes are implemented according to the USB 2.0 and every class’s specifications. These files’ APIs will be called within USB device applications according to the desired functionality.
|
- **Class**: provides APIs for commonly supported USB Device classes complying with USB 2.0 standard and their respective class specifications. These APIs are called in USB Device applications based on the desired functionality.
|
||||||
|
|
||||||
## Release note
|
## Release note
|
||||||
|
|
||||||
Details about the content of this release are available in the release note [here](https://htmlpreview.github.io/?https://github.com/STMicroelectronics/stm32_mw_usb_device/blob/master/Release_Notes.html).
|
Details about the content of this release are available in the release note [here](https://htmlpreview.github.io/?https://github.com/STMicroelectronics/stm32-mw-usb-device/blob/master/Release_Notes.html).
|
||||||
|
|
||||||
## Troubleshooting
|
## Compatibility information
|
||||||
|
|
||||||
|
Please refer to the **release note** in the firmware repository for the STM32 series you are using ([list](https://github.com/STMicroelectronics/STM32Cube_MCU_Overall_Offer/blob/master/README.md#stm32cube-mcu-packages)). It is **important** to use a **consistent set** of software component versions (i.e., CMSIS, HAL-LL, BSP, MW) as specified in the release note.
|
||||||
|
|
||||||
|
## Feedback and contributions
|
||||||
|
|
||||||
Please refer to the [CONTRIBUTING.md](CONTRIBUTING.md) guide.
|
Please refer to the [CONTRIBUTING.md](CONTRIBUTING.md) guide.
|
||||||
@@ -59,8 +59,91 @@ Page</a> : STM32Cube USB Wiki Page</li>
|
|||||||
<section id="update-history" class="col-sm-12 col-lg-8">
|
<section id="update-history" class="col-sm-12 col-lg-8">
|
||||||
<h1>Update History</h1>
|
<h1>Update History</h1>
|
||||||
<div class="collapse">
|
<div class="collapse">
|
||||||
<input type="checkbox" id="collapse-section23" checked aria-hidden="true">
|
<input type="checkbox" id="collapse-section26" checked aria-hidden="true">
|
||||||
<label for="collapse-section23" aria-hidden="true">V2.11.3 /
|
<label for="collapse-section26" aria-hidden="true">V2.11.5 /
|
||||||
|
26-September-2025</label>
|
||||||
|
<div>
|
||||||
|
<h2 id="main-changes">Main Changes</h2>
|
||||||
|
<table>
|
||||||
|
<thead>
|
||||||
|
<tr class="header">
|
||||||
|
<th style="text-align: left;">Headline</th>
|
||||||
|
</tr>
|
||||||
|
</thead>
|
||||||
|
<tbody>
|
||||||
|
<tr class="even">
|
||||||
|
<td style="text-align: left;"><strong>USB Core:</strong></td>
|
||||||
|
</tr>
|
||||||
|
<tr class="odd">
|
||||||
|
<td style="text-align: left;">SLA0044 Legal terms updated to latest revision (Rev 6 / October 2025)</td>
|
||||||
|
</tr>
|
||||||
|
<tr class="even">
|
||||||
|
<td style="text-align: left;"><strong>USB Classes:</strong></td>
|
||||||
|
</tr>
|
||||||
|
<tr class="even">
|
||||||
|
<td style="text-align: left;">Add missing NULL pointer for user string descriptor</td>
|
||||||
|
</tr>
|
||||||
|
<tr class="odd">
|
||||||
|
<td style="text-align: left;"><strong>USB MTP Class:</strong></td>
|
||||||
|
</tr>
|
||||||
|
<tr class="even">
|
||||||
|
<td style="text-align: left;">Minor enhancement; improve device data buffer management</td>
|
||||||
|
</tr>
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="collapse">
|
||||||
|
<input type="checkbox" id="collapse-section25" aria-hidden="true">
|
||||||
|
<label for="collapse-section25" aria-hidden="true">V2.11.4 /
|
||||||
|
11-April-2025</label>
|
||||||
|
<div>
|
||||||
|
<h2 id="main-changes">Main Changes</h2>
|
||||||
|
<table>
|
||||||
|
<thead>
|
||||||
|
<tr class="header">
|
||||||
|
<th style="text-align: left;">Headline</th>
|
||||||
|
</tr>
|
||||||
|
</thead>
|
||||||
|
<tbody>
|
||||||
|
<tr class="even">
|
||||||
|
<td style="text-align: left;"><strong>USB Core:</strong></td>
|
||||||
|
</tr>
|
||||||
|
<tr class="odd">
|
||||||
|
<td style="text-align: left;">Fix return value of USBD_UnRegisterClassComposite function</td>
|
||||||
|
</tr>
|
||||||
|
<tr class="even">
|
||||||
|
<td style="text-align: left;">Increment EP0 data buffer inside HAL data callbacks</td>
|
||||||
|
</tr>
|
||||||
|
<tr class="odd">
|
||||||
|
<td style="text-align: left;"><strong>USB MSC Class:</strong></td>
|
||||||
|
</tr>
|
||||||
|
<tr class="even">
|
||||||
|
<td style="text-align: left;">Adding multi LUN support</td>
|
||||||
|
</tr>
|
||||||
|
<tr class="odd">
|
||||||
|
<td style="text-align: left;"><strong>USB CustomHID Class:</strong></td>
|
||||||
|
</tr>
|
||||||
|
<tr class="even">
|
||||||
|
<td style="text-align: left;">Adding support of different CustomHID report descriptor sizes</td>
|
||||||
|
</tr>
|
||||||
|
<tr class="odd">
|
||||||
|
<td style="text-align: left;">Fix composite builder CustomHID EP size</td>
|
||||||
|
</tr>
|
||||||
|
<tr class="even">
|
||||||
|
<td style="text-align: left;"><strong>USB MTP Class:</strong></td>
|
||||||
|
</tr>
|
||||||
|
<tr class="odd">
|
||||||
|
<td style="text-align: left;">Minor enhancement; improve device data buffer management</td>
|
||||||
|
</tr>
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="collapse">
|
||||||
|
<input type="checkbox" id="collapse-section24" aria-hidden="true">
|
||||||
|
<label for="collapse-section24" aria-hidden="true">V2.11.3 /
|
||||||
20-December-2023</label>
|
20-December-2023</label>
|
||||||
<div>
|
<div>
|
||||||
<h2 id="main-changes">Main Changes</h2>
|
<h2 id="main-changes">Main Changes</h2>
|
||||||
@@ -332,7 +415,7 @@ Serial Bus Device Class Media Transfer Protocol Revision 1.1</td>
|
|||||||
Style</td>
|
Style</td>
|
||||||
</tr>
|
</tr>
|
||||||
<tr class="even">
|
<tr class="even">
|
||||||
<td style="text-align: left;">Update the way to declare licenses</td>
|
<td style="text-align: left;">Update the way to declare Legal terms</td>
|
||||||
</tr>
|
</tr>
|
||||||
<tr class="odd">
|
<tr class="odd">
|
||||||
<td style="text-align: left;"><strong>USB CDC/RNDIS/ECM
|
<td style="text-align: left;"><strong>USB CDC/RNDIS/ECM
|
||||||
@@ -710,8 +793,8 @@ NVIC_SystemReset() prototype change</td>
|
|||||||
</thead>
|
</thead>
|
||||||
<tbody>
|
<tbody>
|
||||||
<tr class="odd">
|
<tr class="odd">
|
||||||
<td style="text-align: left;">Update license section by adding path to
|
<td style="text-align: left;">Update Legal terms section by adding path to
|
||||||
get copy of ST Ultimate Liberty license</td>
|
get copy of ST Ultimate Liberty Legal terms</td>
|
||||||
</tr>
|
</tr>
|
||||||
<tr class="even">
|
<tr class="even">
|
||||||
<td style="text-align: left;">Core: Fix unexpected stall during status
|
<td style="text-align: left;">Core: Fix unexpected stall during status
|
||||||
@@ -747,7 +830,7 @@ Break on USBD_Template_Setup API</td>
|
|||||||
</thead>
|
</thead>
|
||||||
<tbody>
|
<tbody>
|
||||||
<tr class="odd">
|
<tr class="odd">
|
||||||
<td style="text-align: left;">Update license section</td>
|
<td style="text-align: left;">Update Legal terms section</td>
|
||||||
</tr>
|
</tr>
|
||||||
<tr class="even">
|
<tr class="even">
|
||||||
<td style="text-align: left;">Update some functions to be MISRAC 2004
|
<td style="text-align: left;">Update some functions to be MISRAC 2004
|
||||||
@@ -1094,8 +1177,8 @@ development</strong></p>
|
|||||||
<td style="text-align: left;">Official support of STM32F4xx devices</td>
|
<td style="text-align: left;">Official support of STM32F4xx devices</td>
|
||||||
</tr>
|
</tr>
|
||||||
<tr class="even">
|
<tr class="even">
|
||||||
<td style="text-align: left;">All source files: license disclaimer text
|
<td style="text-align: left;">All source files: Legal terms disclaimer text
|
||||||
update and add link to the License file on ST Internet.</td>
|
update and add link to the Legal terms file on ST Internet.</td>
|
||||||
</tr>
|
</tr>
|
||||||
<tr class="odd">
|
<tr class="odd">
|
||||||
<td style="text-align: left;">Handle test mode in the set feature
|
<td style="text-align: left;">Handle test mode in the set feature
|
||||||
|
|||||||
Reference in New Issue
Block a user