Add TMC header read and write functions, so that TMC data can now be exchanged in both directions.

Minor update to the LowLevel MassStorage device demo, so that the ReadInCommandBlock() performs the data OUT endpoint selection and packet arrival test.
pull/1469/head
Dean Camera 15 years ago
parent 059307d89c
commit 9a97f16b07

@ -330,35 +330,35 @@ void TMC_Task(void)
if (USB_DeviceState != DEVICE_STATE_Configured) if (USB_DeviceState != DEVICE_STATE_Configured)
return; return;
Endpoint_SelectEndpoint(TMC_OUT_EPNUM); TMC_MessageHeader_t MessageHeader;
if (Endpoint_IsOUTReceived()) /* Check if a TMC packet has been received */
if (ReadTMCHeader(&MessageHeader))
{ {
TMC_MessageHeader_t MessageHeader; /* Indicate busy */
LEDs_SetAllLEDs(LEDMASK_USB_BUSY);
Endpoint_Read_Stream_LE(&MessageHeader, sizeof(MessageHeader), StreamCallback_AbortOUTOnRequest);
CurrentTransferTag = MessageHeader.Tag;
switch (MessageHeader.MessageID) switch (MessageHeader.MessageID)
{ {
case TMC_MESSAGEID_DEV_DEP_MSG_OUT: case TMC_MESSAGEID_DEV_DEP_MSG_OUT:
Endpoint_Discard_Stream(MessageHeader.TransferSize, StreamCallback_AbortOUTOnRequest);
Endpoint_ClearOUT();
break; break;
case TMC_MESSAGEID_DEV_DEP_MSG_IN: case TMC_MESSAGEID_DEV_DEP_MSG_IN:
Endpoint_ClearOUT();
break; MessageHeader.TransferSize = 3;
case TMC_MESSAGEID_DEV_VENDOR_OUT: WriteTMCHeader(&MessageHeader);
break;
case TMC_MESSAGEID_DEV_VENDOR_IN:
Endpoint_Write_Stream_LE("TMC", 3, StreamCallback_AbortINOnRequest);
Endpoint_ClearIN();
break; break;
default: default:
Endpoint_StallTransaction(); Endpoint_StallTransaction();
break; break;
} }
Endpoint_ClearOUT(); LEDs_SetAllLEDs(LEDMASK_USB_READY);
} }
/* All pending data has been processed - reset the data abort flags */ /* All pending data has been processed - reset the data abort flags */
@ -366,6 +366,45 @@ void TMC_Task(void)
IsTMCBulkOUTReset = false; IsTMCBulkOUTReset = false;
} }
bool ReadTMCHeader(TMC_MessageHeader_t* const MessageHeader)
{
/* Select the Data Out endpoint */
Endpoint_SelectEndpoint(TMC_OUT_EPNUM);
/* Abort if no command has been sent from the host */
if (!(Endpoint_IsOUTReceived()))
return false;
/* Read in the header of the command from the host */
Endpoint_Read_Stream_LE(MessageHeader, sizeof(TMC_MessageHeader_t), StreamCallback_AbortOUTOnRequest);
/* Store the new command tag value for later use */
CurrentTransferTag = MessageHeader->Tag;
/* Indicate if the command has been aborted or not */
return !IsTMCBulkOUTReset;
}
bool WriteTMCHeader(TMC_MessageHeader_t* const MessageHeader)
{
/* Compute the next transfer tag value, must be between 1 and 254 */
if (++CurrentTransferTag == 0xFF)
CurrentTransferTag = 1;
/* Set the message tag of the command header */
MessageHeader->Tag = CurrentTransferTag;
MessageHeader->InverseTag = ~CurrentTransferTag;
/* Select the Data In endpoint */
Endpoint_SelectEndpoint(TMC_IN_EPNUM);
/* Send the command header to the host */
Endpoint_Write_Stream_LE(MessageHeader, sizeof(TMC_MessageHeader_t), StreamCallback_AbortINOnRequest);
/* Indicate if the command has been aborted or not */
return !IsTMCBulkINReset;
}
/** Stream callback function for the Endpoint stream write functions. This callback will abort the current stream transfer /** Stream callback function for the Endpoint stream write functions. This callback will abort the current stream transfer
* if a TMC Abort Bulk IN request has been issued to the control endpoint. * if a TMC Abort Bulk IN request has been issued to the control endpoint.
*/ */

@ -45,16 +45,19 @@
/* Macros: */ /* Macros: */
/** LED mask for the library LED driver, to indicate that the USB interface is not ready. */ /** LED mask for the library LED driver, to indicate that the USB interface is not ready. */
#define LEDMASK_USB_NOTREADY LEDS_LED1 #define LEDMASK_USB_NOTREADY LEDS_LED1
/** LED mask for the library LED driver, to indicate that the USB interface is enumerating. */ /** LED mask for the library LED driver, to indicate that the USB interface is enumerating. */
#define LEDMASK_USB_ENUMERATING (LEDS_LED2 | LEDS_LED3) #define LEDMASK_USB_ENUMERATING (LEDS_LED2 | LEDS_LED3)
/** LED mask for the library LED driver, to indicate that the USB interface is ready. */ /** LED mask for the library LED driver, to indicate that the USB interface is ready. */
#define LEDMASK_USB_READY (LEDS_LED2 | LEDS_LED4) #define LEDMASK_USB_READY (LEDS_LED2 | LEDS_LED4)
/** LED mask for the library LED driver, to indicate that an error has occurred in the USB interface. */ /** LED mask for the library LED driver, to indicate that an error has occurred in the USB interface. */
#define LEDMASK_USB_ERROR (LEDS_LED1 | LEDS_LED3) #define LEDMASK_USB_ERROR (LEDS_LED1 | LEDS_LED3)
/** LED mask for the library LED driver, to indicate that the USB interface is busy. */
#define LEDMASK_USB_BUSY LEDS_LED2
#define Req_InitiateAbortBulkOut 0x01 #define Req_InitiateAbortBulkOut 0x01
#define Req_CheckAbortBulkOutStatus 0x02 #define Req_CheckAbortBulkOutStatus 0x02
@ -139,6 +142,8 @@
/* Function Prototypes: */ /* Function Prototypes: */
void SetupHardware(void); void SetupHardware(void);
void TMC_Task(void); void TMC_Task(void);
bool ReadTMCHeader(TMC_MessageHeader_t* const MessageHeader);
bool WriteTMCHeader(TMC_MessageHeader_t* const MessageHeader);
void EVENT_USB_Device_Connect(void); void EVENT_USB_Device_Connect(void);
void EVENT_USB_Device_Disconnect(void); void EVENT_USB_Device_Disconnect(void);

@ -174,46 +174,34 @@ void MassStorage_Task(void)
if (USB_DeviceState != DEVICE_STATE_Configured) if (USB_DeviceState != DEVICE_STATE_Configured)
return; return;
/* Select the Data Out Endpoint */ /* Process sent command block from the host if one has been sent */
Endpoint_SelectEndpoint(MASS_STORAGE_OUT_EPNUM); if (ReadInCommandBlock())
/* Check to see if a command from the host has been issued */
if (Endpoint_IsReadWriteAllowed())
{ {
/* Indicate busy */ /* Indicate busy */
LEDs_SetAllLEDs(LEDMASK_USB_BUSY); LEDs_SetAllLEDs(LEDMASK_USB_BUSY);
/* Process sent command block from the host */ /* Check direction of command, select Data IN endpoint if data is from the device */
if (ReadInCommandBlock()) if (CommandBlock.Flags & COMMAND_DIRECTION_DATA_IN)
{ Endpoint_SelectEndpoint(MASS_STORAGE_IN_EPNUM);
/* Check direction of command, select Data IN endpoint if data is from the device */
if (CommandBlock.Flags & COMMAND_DIRECTION_DATA_IN) /* Decode the received SCSI command, set returned status code */
Endpoint_SelectEndpoint(MASS_STORAGE_IN_EPNUM); CommandStatus.Status = SCSI_DecodeSCSICommand() ? Command_Pass : Command_Fail;
/* Decode the received SCSI command, set returned status code */ /* Load in the CBW tag into the CSW to link them together */
CommandStatus.Status = SCSI_DecodeSCSICommand() ? Command_Pass : Command_Fail; CommandStatus.Tag = CommandBlock.Tag;
/* Load in the CBW tag into the CSW to link them together */ /* Load in the data residue counter into the CSW */
CommandStatus.Tag = CommandBlock.Tag; CommandStatus.DataTransferResidue = CommandBlock.DataTransferLength;
/* Load in the data residue counter into the CSW */ /* Stall the selected data pipe if command failed (if data is still to be transferred) */
CommandStatus.DataTransferResidue = CommandBlock.DataTransferLength; if ((CommandStatus.Status == Command_Fail) && (CommandStatus.DataTransferResidue))
Endpoint_StallTransaction();
/* Stall the selected data pipe if command failed (if data is still to be transferred) */
if ((CommandStatus.Status == Command_Fail) && (CommandStatus.DataTransferResidue)) /* Return command status block to the host */
Endpoint_StallTransaction(); ReturnCommandStatus();
/* Return command status block to the host */ /* Indicate ready */
ReturnCommandStatus(); LEDs_SetAllLEDs(LEDMASK_USB_READY);
/* Indicate ready */
LEDs_SetAllLEDs(LEDMASK_USB_READY);
}
else
{
/* Indicate error reading in the command block from the host */
LEDs_SetAllLEDs(LEDMASK_USB_ERROR);
}
} }
/* Check if a Mass Storage Reset occurred */ /* Check if a Mass Storage Reset occurred */
@ -245,6 +233,10 @@ static bool ReadInCommandBlock(void)
/* Select the Data Out endpoint */ /* Select the Data Out endpoint */
Endpoint_SelectEndpoint(MASS_STORAGE_OUT_EPNUM); Endpoint_SelectEndpoint(MASS_STORAGE_OUT_EPNUM);
/* Abort if no command has been sent from the host */
if (!(Endpoint_IsOUTReceived()))
return false;
/* Read in command block header */ /* Read in command block header */
Endpoint_Read_Stream_LE(&CommandBlock, (sizeof(CommandBlock) - sizeof(CommandBlock.SCSICommandData)), Endpoint_Read_Stream_LE(&CommandBlock, (sizeof(CommandBlock) - sizeof(CommandBlock.SCSICommandData)),
StreamCallback_AbortOnMassStoreReset); StreamCallback_AbortOnMassStoreReset);

Loading…
Cancel
Save