3      Copyright (C) Dean Camera, 2009. 
   5   dean [at] fourwalledcubicle [dot] com 
   6       www.fourwalledcubicle.com 
  10   Copyright 2009  Dean Camera (dean [at] fourwalledcubicle [dot] com) 
  12   Permission to use, copy, modify, and distribute this software 
  13   and its documentation for any purpose and without fee is hereby 
  14   granted, provided that the above copyright notice appear in all 
  15   copies and that both that the copyright notice and this 
  16   permission notice and warranty disclaimer appear in supporting 
  17   documentation, and that the name of the author not be used in 
  18   advertising or publicity pertaining to distribution of the 
  19   software without specific, written prior permission. 
  21   The author disclaim all warranties with regard to this 
  22   software, including all implied warranties of merchantability 
  23   and fitness.  In no event shall the author be liable for any 
  24   special, indirect or consequential damages or any damages 
  25   whatsoever resulting from loss of use, data or profits, whether 
  26   in an action of contract, negligence or other tortious action, 
  27   arising out of or in connection with the use or performance of 
  31 #include "../../HighLevel/USBMode.h" 
  32 #if defined(USB_CAN_BE_DEVICE) 
  34 #define  INCLUDE_FROM_MS_CLASS_DEVICE_C 
  35 #include "MassStorage.h" 
  37 static USB_ClassInfo_MS_Device_t
* CallbackMSInterfaceInfo
; 
  39 void MS_Device_ProcessControlRequest(USB_ClassInfo_MS_Device_t
* const MSInterfaceInfo
) 
  41         if (!(Endpoint_IsSETUPReceived())) 
  44         if (USB_ControlRequest
.wIndex 
!= MSInterfaceInfo
->Config
.InterfaceNumber
) 
  47         switch (USB_ControlRequest
.bRequest
) 
  49                 case REQ_MassStorageReset
: 
  50                         if (USB_ControlRequest
.bmRequestType 
== (REQDIR_HOSTTODEVICE 
| REQTYPE_CLASS 
| REQREC_INTERFACE
)) 
  52                                 Endpoint_ClearSETUP(); 
  54                                 MSInterfaceInfo
->State
.IsMassStoreReset 
= true;                  
  56                                 Endpoint_ClearStatusStage(); 
  61                         if (USB_ControlRequest
.bmRequestType 
== (REQDIR_DEVICETOHOST 
| REQTYPE_CLASS 
| REQREC_INTERFACE
)) 
  63                                 Endpoint_ClearSETUP(); 
  65                                 Endpoint_Write_Byte(MSInterfaceInfo
->Config
.TotalLUNs 
- 1);                              
  68                                 Endpoint_ClearStatusStage(); 
  75 bool MS_Device_ConfigureEndpoints(USB_ClassInfo_MS_Device_t
* const MSInterfaceInfo
) 
  77         memset(&MSInterfaceInfo
->State
, 0x00, sizeof(MSInterfaceInfo
->State
)); 
  79         if (!(Endpoint_ConfigureEndpoint(MSInterfaceInfo
->Config
.DataINEndpointNumber
, EP_TYPE_BULK
, 
  80                                                                  ENDPOINT_DIR_IN
, MSInterfaceInfo
->Config
.DataINEndpointSize
, 
  81                                                                  ENDPOINT_BANK_SINGLE
))) 
  86         if (!(Endpoint_ConfigureEndpoint(MSInterfaceInfo
->Config
.DataOUTEndpointNumber
, EP_TYPE_BULK
, 
  87                                          ENDPOINT_DIR_OUT
, MSInterfaceInfo
->Config
.DataOUTEndpointSize
, 
  88                                          ENDPOINT_BANK_SINGLE
))) 
  96 void MS_Device_USBTask(USB_ClassInfo_MS_Device_t
* const MSInterfaceInfo
) 
  98         if (USB_DeviceState 
!= DEVICE_STATE_Configured
) 
 101         Endpoint_SelectEndpoint(MSInterfaceInfo
->Config
.DataOUTEndpointNumber
); 
 103         if (Endpoint_IsReadWriteAllowed()) 
 105                 if (MS_Device_ReadInCommandBlock(MSInterfaceInfo
)) 
 107                         if (MSInterfaceInfo
->State
.CommandBlock
.Flags 
& MS_COMMAND_DIR_DATA_IN
) 
 108                           Endpoint_SelectEndpoint(MSInterfaceInfo
->Config
.DataINEndpointNumber
); 
 110                         MSInterfaceInfo
->State
.CommandStatus
.Status 
= CALLBACK_MS_Device_SCSICommandReceived(MSInterfaceInfo
) ?
 
 111                                                                                          SCSI_Command_Pass 
: SCSI_Command_Fail
; 
 112                         MSInterfaceInfo
->State
.CommandStatus
.Signature           
= MS_CSW_SIGNATURE
; 
 113                         MSInterfaceInfo
->State
.CommandStatus
.Tag                 
= MSInterfaceInfo
->State
.CommandBlock
.Tag
; 
 114                         MSInterfaceInfo
->State
.CommandStatus
.DataTransferResidue 
= MSInterfaceInfo
->State
.CommandBlock
.DataTransferLength
; 
 116                         if ((MSInterfaceInfo
->State
.CommandStatus
.Status 
== SCSI_Command_Fail
) && 
 117                             (MSInterfaceInfo
->State
.CommandStatus
.DataTransferResidue
)) 
 119                                 Endpoint_StallTransaction(); 
 122                         MS_Device_ReturnCommandStatus(MSInterfaceInfo
); 
 124                         if (MSInterfaceInfo
->State
.IsMassStoreReset
) 
 126                                 Endpoint_ResetFIFO(MSInterfaceInfo
->Config
.DataOUTEndpointNumber
); 
 127                                 Endpoint_ResetFIFO(MSInterfaceInfo
->Config
.DataINEndpointNumber
); 
 129                                 Endpoint_SelectEndpoint(MSInterfaceInfo
->Config
.DataOUTEndpointNumber
); 
 130                                 Endpoint_ClearStall(); 
 131                                 Endpoint_SelectEndpoint(MSInterfaceInfo
->Config
.DataINEndpointNumber
); 
 132                                 Endpoint_ClearStall(); 
 137         MSInterfaceInfo
->State
.IsMassStoreReset 
= false; 
 140 static bool MS_Device_ReadInCommandBlock(USB_ClassInfo_MS_Device_t
* const MSInterfaceInfo
) 
 142         Endpoint_SelectEndpoint(MSInterfaceInfo
->Config
.DataOUTEndpointNumber
); 
 144         CallbackMSInterfaceInfo 
= MSInterfaceInfo
; 
 145         Endpoint_Read_Stream_LE(&MSInterfaceInfo
->State
.CommandBlock
, 
 146                                 (sizeof(MS_CommandBlockWrapper_t
) - 16), 
 147                                 StreamCallback_MS_Device_AbortOnMassStoreReset
); 
 149         if ((MSInterfaceInfo
->State
.CommandBlock
.Signature         
!= MS_CBW_SIGNATURE
)                  || 
 150             (MSInterfaceInfo
->State
.CommandBlock
.LUN               
>= MSInterfaceInfo
->Config
.TotalLUNs
) || 
 151                 (MSInterfaceInfo
->State
.CommandBlock
.Flags              
& 0x1F)                              || 
 152                 (MSInterfaceInfo
->State
.CommandBlock
.SCSICommandLength 
== 0)                                 || 
 153                 (MSInterfaceInfo
->State
.CommandBlock
.SCSICommandLength 
>  16)) 
 155                 Endpoint_StallTransaction(); 
 156                 Endpoint_SelectEndpoint(MSInterfaceInfo
->Config
.DataINEndpointNumber
); 
 157                 Endpoint_StallTransaction(); 
 162         CallbackMSInterfaceInfo 
= MSInterfaceInfo
; 
 163         Endpoint_Read_Stream_LE(&MSInterfaceInfo
->State
.CommandBlock
.SCSICommandData
, 
 164                                 MSInterfaceInfo
->State
.CommandBlock
.SCSICommandLength
, 
 165                                 StreamCallback_MS_Device_AbortOnMassStoreReset
); 
 169         if (MSInterfaceInfo
->State
.IsMassStoreReset
) 
 175 static void MS_Device_ReturnCommandStatus(USB_ClassInfo_MS_Device_t
* const MSInterfaceInfo
) 
 177         Endpoint_SelectEndpoint(MSInterfaceInfo
->Config
.DataOUTEndpointNumber
); 
 179         while (Endpoint_IsStalled()) 
 183                 if (MSInterfaceInfo
->State
.IsMassStoreReset
) 
 187         Endpoint_SelectEndpoint(MSInterfaceInfo
->Config
.DataINEndpointNumber
); 
 189         while (Endpoint_IsStalled()) 
 193                 if (MSInterfaceInfo
->State
.IsMassStoreReset
) 
 197         CallbackMSInterfaceInfo 
= MSInterfaceInfo
; 
 198         Endpoint_Write_Stream_LE(&MSInterfaceInfo
->State
.CommandStatus
, sizeof(MS_CommandStatusWrapper_t
), 
 199                                  StreamCallback_MS_Device_AbortOnMassStoreReset
); 
 203         if (MSInterfaceInfo
->State
.IsMassStoreReset
) 
 207 static uint8_t StreamCallback_MS_Device_AbortOnMassStoreReset(void) 
 209         MS_Device_USBTask(CallbackMSInterfaceInfo
); 
 211         if (CallbackMSInterfaceInfo
->State
.IsMassStoreReset
) 
 212           return STREAMCALLBACK_Abort
; 
 214           return STREAMCALLBACK_Continue
;