246 lines
5.4 KiB
C
246 lines
5.4 KiB
C
/**
|
|
******************************************************************************
|
|
* @file usbh_diskio.c
|
|
* @author MCD Application Team
|
|
* @brief USB Host Disk I/O driver (without internal DMA).
|
|
******************************************************************************
|
|
* @attention
|
|
*
|
|
* Copyright (c) 2018 STMicroelectronics.
|
|
* All rights reserved.
|
|
*
|
|
* This software is licensed under terms that can be found in the LICENSE file
|
|
* in the root directory of this software component.
|
|
* If no LICENSE file comes with this software, it is provided AS-IS.
|
|
*
|
|
******************************************************************************
|
|
*/
|
|
|
|
/* Includes ------------------------------------------------------------------*/
|
|
#include "ff_gen_drv.h"
|
|
#include "usbh_diskio.h"
|
|
|
|
/* Private typedef -----------------------------------------------------------*/
|
|
/* Private define ------------------------------------------------------------*/
|
|
|
|
#define USB_DEFAULT_BLOCK_SIZE 512
|
|
|
|
/* Private variables ---------------------------------------------------------*/
|
|
extern USBH_HandleTypeDef hUSBHost;
|
|
|
|
/* Private function prototypes -----------------------------------------------*/
|
|
DSTATUS USBH_initialize (BYTE);
|
|
DSTATUS USBH_status (BYTE);
|
|
DRESULT USBH_read (BYTE, BYTE*, DWORD, UINT);
|
|
|
|
#if _USE_WRITE == 1
|
|
DRESULT USBH_write (BYTE, const BYTE*, DWORD, UINT);
|
|
#endif /* _USE_WRITE == 1 */
|
|
|
|
#if _USE_IOCTL == 1
|
|
DRESULT USBH_ioctl (BYTE, BYTE, void*);
|
|
#endif /* _USE_IOCTL == 1 */
|
|
|
|
const Diskio_drvTypeDef USBH_Driver =
|
|
{
|
|
USBH_initialize,
|
|
USBH_status,
|
|
USBH_read,
|
|
#if _USE_WRITE == 1
|
|
USBH_write,
|
|
#endif /* _USE_WRITE == 1 */
|
|
#if _USE_IOCTL == 1
|
|
USBH_ioctl,
|
|
#endif /* _USE_IOCTL == 1 */
|
|
};
|
|
|
|
/* Private functions ---------------------------------------------------------*/
|
|
|
|
/**
|
|
* @brief Initializes a Drive
|
|
* @param lun : lun id
|
|
* @retval DSTATUS: Operation status
|
|
*/
|
|
DSTATUS USBH_initialize(BYTE lun)
|
|
{
|
|
/* CAUTION : USB Host library has to be initialized in the application */
|
|
|
|
return RES_OK;
|
|
}
|
|
|
|
/**
|
|
* @brief Gets Disk Status
|
|
* @param lun : lun id
|
|
* @retval DSTATUS: Operation status
|
|
*/
|
|
DSTATUS USBH_status(BYTE lun)
|
|
{
|
|
DRESULT res = RES_ERROR;
|
|
|
|
if(USBH_MSC_UnitIsReady(&hUSBHost, lun))
|
|
{
|
|
res = RES_OK;
|
|
}
|
|
else
|
|
{
|
|
res = RES_ERROR;
|
|
}
|
|
|
|
return res;
|
|
}
|
|
|
|
/**
|
|
* @brief Reads Sector(s)
|
|
* @param lun : lun id
|
|
* @param *buff: Data buffer to store read data
|
|
* @param sector: Sector address (LBA)
|
|
* @param count: Number of sectors to read (1..128)
|
|
* @retval DRESULT: Operation result
|
|
*/
|
|
DRESULT USBH_read(BYTE lun, BYTE *buff, DWORD sector, UINT count)
|
|
{
|
|
DRESULT res = RES_ERROR;
|
|
MSC_LUNTypeDef info;
|
|
|
|
if(USBH_MSC_Read(&hUSBHost, lun, sector, buff, count) == USBH_OK)
|
|
{
|
|
res = RES_OK;
|
|
}
|
|
else
|
|
{
|
|
USBH_MSC_GetLUNInfo(&hUSBHost, lun, &info);
|
|
|
|
switch (info.sense.asc)
|
|
{
|
|
case SCSI_ASC_LOGICAL_UNIT_NOT_READY:
|
|
case SCSI_ASC_MEDIUM_NOT_PRESENT:
|
|
case SCSI_ASC_NOT_READY_TO_READY_CHANGE:
|
|
USBH_ErrLog ("USB Disk is not ready!");
|
|
res = RES_NOTRDY;
|
|
break;
|
|
|
|
default:
|
|
res = RES_ERROR;
|
|
break;
|
|
}
|
|
}
|
|
|
|
return res;
|
|
}
|
|
|
|
/**
|
|
* @brief Writes Sector(s)
|
|
* @param lun : lun id
|
|
* @param *buff: Data to be written
|
|
* @param sector: Sector address (LBA)
|
|
* @param count: Number of sectors to write (1..128)
|
|
* @retval DRESULT: Operation result
|
|
*/
|
|
#if _USE_WRITE == 1
|
|
DRESULT USBH_write(BYTE lun, const BYTE *buff, DWORD sector, UINT count)
|
|
{
|
|
DRESULT res = RES_ERROR;
|
|
MSC_LUNTypeDef info;
|
|
|
|
if(USBH_MSC_Write(&hUSBHost, lun, sector, (BYTE *)buff, count) == USBH_OK)
|
|
{
|
|
res = RES_OK;
|
|
}
|
|
else
|
|
{
|
|
USBH_MSC_GetLUNInfo(&hUSBHost, lun, &info);
|
|
|
|
switch (info.sense.asc)
|
|
{
|
|
case SCSI_ASC_WRITE_PROTECTED:
|
|
USBH_ErrLog("USB Disk is Write protected!");
|
|
res = RES_WRPRT;
|
|
break;
|
|
|
|
case SCSI_ASC_LOGICAL_UNIT_NOT_READY:
|
|
case SCSI_ASC_MEDIUM_NOT_PRESENT:
|
|
case SCSI_ASC_NOT_READY_TO_READY_CHANGE:
|
|
USBH_ErrLog("USB Disk is not ready!");
|
|
res = RES_NOTRDY;
|
|
break;
|
|
|
|
default:
|
|
res = RES_ERROR;
|
|
break;
|
|
}
|
|
}
|
|
|
|
return res;
|
|
}
|
|
#endif /* _USE_WRITE == 1 */
|
|
|
|
/**
|
|
* @brief I/O control operation
|
|
* @param lun : lun id
|
|
* @param cmd: Control code
|
|
* @param *buff: Buffer to send/receive control data
|
|
* @retval DRESULT: Operation result
|
|
*/
|
|
#if _USE_IOCTL == 1
|
|
DRESULT USBH_ioctl(BYTE lun, BYTE cmd, void *buff)
|
|
{
|
|
DRESULT res = RES_ERROR;
|
|
MSC_LUNTypeDef info;
|
|
|
|
switch (cmd)
|
|
{
|
|
/* Make sure that no pending write process */
|
|
case CTRL_SYNC:
|
|
res = RES_OK;
|
|
break;
|
|
|
|
/* Get number of sectors on the disk (DWORD) */
|
|
case GET_SECTOR_COUNT :
|
|
if(USBH_MSC_GetLUNInfo(&hUSBHost, lun, &info) == USBH_OK)
|
|
{
|
|
*(DWORD*)buff = info.capacity.block_nbr;
|
|
res = RES_OK;
|
|
}
|
|
else
|
|
{
|
|
res = RES_ERROR;
|
|
}
|
|
break;
|
|
|
|
/* Get R/W sector size (WORD) */
|
|
case GET_SECTOR_SIZE :
|
|
if(USBH_MSC_GetLUNInfo(&hUSBHost, lun, &info) == USBH_OK)
|
|
{
|
|
*(DWORD*)buff = info.capacity.block_size;
|
|
res = RES_OK;
|
|
}
|
|
else
|
|
{
|
|
res = RES_ERROR;
|
|
}
|
|
break;
|
|
|
|
/* Get erase block size in unit of sector (DWORD) */
|
|
case GET_BLOCK_SIZE :
|
|
|
|
if(USBH_MSC_GetLUNInfo(&hUSBHost, lun, &info) == USBH_OK)
|
|
{
|
|
*(DWORD*)buff = info.capacity.block_size / USB_DEFAULT_BLOCK_SIZE;
|
|
res = RES_OK;
|
|
}
|
|
else
|
|
{
|
|
res = RES_ERROR;
|
|
}
|
|
break;
|
|
|
|
default:
|
|
res = RES_PARERR;
|
|
}
|
|
|
|
return res;
|
|
}
|
|
#endif /* _USE_IOCTL == 1 */
|
|
|
|
|