Kept working on the USB protocol

This commit is contained in:
Doug Brown 2011-12-16 20:13:34 -08:00
parent 4f0d1a894c
commit ca84a1d562

View File

@ -11,10 +11,15 @@
#include "../external_mem.h" #include "../external_mem.h"
#include "../tests/simm_electrical_test.h" #include "../tests/simm_electrical_test.h"
#define CHIP_SIZE (512UL * 1024UL)
#define READ_CHUNK_SIZE_BYTES 1024UL #define READ_CHUNK_SIZE_BYTES 1024UL
#define WRITE_CHUNK_SIZE_BYTES 1024UL
#if ((READ_CHUNK_SIZE_BYTES % 4) != 0) #if ((READ_CHUNK_SIZE_BYTES % 4) != 0)
#error Read chunk size should be a multiple of 4 bytes #error Read chunk size should be a multiple of 4 bytes
#endif #endif
#if ((WRITE_CHUNK_SIZE_BYTES % 4) != 0)
#error Write chunk size should be a multiple of 4 bytes
#endif
USB_ClassInfo_CDC_Device_t VirtualSerial_CDC_Interface = USB_ClassInfo_CDC_Device_t VirtualSerial_CDC_Interface =
{ {
@ -47,6 +52,7 @@ typedef enum ProgrammerCommandState
ReadingByteWaitingForAddress, ReadingByteWaitingForAddress,
ReadingChips, ReadingChips,
ReadingChipsUnableSendError, ReadingChipsUnableSendError,
WritingChips,
} ProgrammerCommandState; } ProgrammerCommandState;
typedef enum ProgrammerCommand typedef enum ProgrammerCommand
@ -62,20 +68,58 @@ typedef enum ProgrammerCommand
typedef enum ProgrammerReply typedef enum ProgrammerReply
{ {
ReplyOK, CommandReplyOK,
ReplyError, CommandReplyError,
CommandReplyInvalid,
} ProgrammerReply; } ProgrammerReply;
typedef enum ComputerReadReply
{
ComputerReadOK,
ComputerReadCancel,
} ComputerReadReply;
typedef enum ProgrammerReadReply
{
ProgrammerReadOK,
ProgrammerReadError,
ProgrammerReadMoreData,
ProgrammerReadFinished,
ProgrammerReadConfirmCancel,
} ProgrammerReadReply;
typedef enum ComputerWriteReply
{
ComputerWriteMore,
ComputerWriteFinish,
ComputerWriteCancel,
} ComputerWriteReply;
typedef enum ProgrammerWriteReply
{
ProgrammerWriteOK,
ProgrammerWriteError,
ProgrammerWriteConfirmCancel,
};
static ProgrammerCommandState curCommandState = WaitingForCommand; static ProgrammerCommandState curCommandState = WaitingForCommand;
static uint8_t byteAddressReceiveCount = 0; static uint8_t byteAddressReceiveCount = 0;
static uint16_t curReadIndex; static uint16_t curReadIndex;
static int16_t writePosInChunk = -1;
static uint16_t curWriteIndex = 0;
void USBSerial_HandleWaitingForCommandByte(uint8_t byte); void USBSerial_HandleWaitingForCommandByte(uint8_t byte);
void USBSerial_HandleReadingChipsByte(uint8_t byte);
void USBSerial_SendReadDataChunk(void); void USBSerial_SendReadDataChunk(void);
void USBSerial_HandleWritingChipsByte(uint8_t byte);
#define SendByte(b) CDC_Device_SendByte(&VirtualSerial_CDC_Interface, b)
#define ReadByte() CDC_Device_ReceiveByte(&VirtualSerial_CDC_Interface)
#define SendData(d, l) CDC_Device_SendData(&VirtualSerial_CDC_Interface, d, l)
void USBSerial_Check(void) void USBSerial_Check(void)
{ {
if (USB_DeviceState == DEVICE_STATE_Configured) /*if (USB_DeviceState == DEVICE_STATE_Configured)
{ {
if (CDC_Device_BytesReceived(&VirtualSerial_CDC_Interface)) if (CDC_Device_BytesReceived(&VirtualSerial_CDC_Interface))
{ {
@ -150,12 +194,12 @@ void USBSerial_Check(void)
CDC_Device_SendString(&VirtualSerial_CDC_Interface, "Finished\r\n"); CDC_Device_SendString(&VirtualSerial_CDC_Interface, "Finished\r\n");
} }
} }
} }*/
/*if (USB_DeviceState == DEVICE_STATE_Configured) if (USB_DeviceState == DEVICE_STATE_Configured)
{ {
// Check for commands, etc... // Check for commands, etc...
int16_t recvByte = CDC_Device_ReceiveByte(&VirtualSerial_CDC_Interface); int16_t recvByte = ReadByte();
if (recvByte >= 0) if (recvByte >= 0)
{ {
@ -164,9 +208,15 @@ void USBSerial_Check(void)
case WaitingForCommand: case WaitingForCommand:
USBSerial_HandleWaitingForCommandByte((uint8_t)recvByte); USBSerial_HandleWaitingForCommandByte((uint8_t)recvByte);
break; break;
case ReadingChips:
USBSerial_HandleReadingChipsByte((uint8_t)recvByte);
break;
case WritingChips:
USBSerial_HandleWritingChipsByte((uint8_t)recvByte);
break;
}
} }
} }
}*/
CDC_Device_USBTask(&VirtualSerial_CDC_Interface); CDC_Device_USBTask(&VirtualSerial_CDC_Interface);
USB_USBTask(); USB_USBTask();
@ -180,14 +230,14 @@ void USBSerial_HandleWaitingForCommandByte(uint8_t byte)
curCommandState = WaitingForCommand; curCommandState = WaitingForCommand;
break; break;
case DoElectricalTest: case DoElectricalTest:
CDC_Device_SendByte(&VirtualSerial_CDC_Interface, ReplyOK); SendByte(CommandReplyOK);
SIMMElectricalTest_Run(); SIMMElectricalTest_Run();
curCommandState = WaitingForCommand; curCommandState = WaitingForCommand;
break; break;
case IdentifyChips: case IdentifyChips:
{ {
struct ChipID chips[4]; struct ChipID chips[4];
CDC_Device_SendByte(&VirtualSerial_CDC_Interface, ReplyOK); SendByte(CommandReplyOK);
ExternalMem_IdentifyChips(chips); ExternalMem_IdentifyChips(chips);
// TODO: Send chip ID info back to receiver // TODO: Send chip ID info back to receiver
break; break;
@ -195,16 +245,49 @@ void USBSerial_HandleWaitingForCommandByte(uint8_t byte)
case ReadByte: case ReadByte:
curCommandState = ReadingByteWaitingForAddress; curCommandState = ReadingByteWaitingForAddress;
byteAddressReceiveCount = 0; byteAddressReceiveCount = 0;
CDC_Device_SendByte(&VirtualSerial_CDC_Interface, ReplyOK); SendByte(CommandReplyOK);
break; break;
case ReadChips: case ReadChips:
curCommandState = ReadingChips; curCommandState = ReadingChips;
curReadIndex = 0; curReadIndex = 0;
CDC_Device_SendByte(&VirtualSerial_CDC_Interface, ReplyOK); SendByte(CommandReplyOK);
USBSerial_SendReadDataChunk(); USBSerial_SendReadDataChunk();
break; break;
case EraseChips: case EraseChips:
ExternalMem_EraseChips(ALL_CHIPS);
SendByte(CommandReplyOK);
break;
case WriteChips: case WriteChips:
curCommandState = WritingChips;
curWriteIndex = 0;
writePosInChunk = -1;
SendByte(CommandReplyOK);
break;
default:
SendByte(CommandReplyInvalid);
break;
}
}
void USBSerial_HandleReadingChipsByte(uint8_t byte)
{
switch (byte)
{
case ComputerReadOK:
if (curReadIndex >= (CHIP_SIZE / READ_CHUNK_SIZE_BYTES))
{
SendByte(ProgrammerReadFinished);
curCommandState = WaitingForCommand;
}
else
{
SendByte(ProgrammerReadMoreData);
USBSerial_SendReadDataChunk();
}
break;
case ComputerReadCancel:
SendByte(ProgrammerReadConfirmCancel);
curCommandState = WaitingForCommand;
break; break;
} }
} }
@ -222,7 +305,7 @@ void USBSerial_SendReadDataChunk(void)
} chunks; } chunks;
ExternalMem_Read(curReadIndex * (READ_CHUNK_SIZE_BYTES/4), chunks.readChunks, READ_CHUNK_SIZE_BYTES/4); ExternalMem_Read(curReadIndex * (READ_CHUNK_SIZE_BYTES/4), chunks.readChunks, READ_CHUNK_SIZE_BYTES/4);
uint8_t retVal = CDC_Device_SendData(&VirtualSerial_CDC_Interface, (const char *)chunks.readChunkBytes, READ_CHUNK_SIZE_BYTES); uint8_t retVal = SendData((const char *)chunks.readChunkBytes, READ_CHUNK_SIZE_BYTES);
if (retVal != ENDPOINT_RWSTREAM_NoError) if (retVal != ENDPOINT_RWSTREAM_NoError)
{ {
curCommandState = ReadingChipsUnableSendError; curCommandState = ReadingChipsUnableSendError;
@ -233,6 +316,54 @@ void USBSerial_SendReadDataChunk(void)
} }
} }
void USBSerial_HandleWritingChipsByte(uint8_t byte)
{
static union
{
uint32_t writeChunks[WRITE_CHUNK_SIZE_BYTES / 4];
uint8_t writeChunkBytes[WRITE_CHUNK_SIZE_BYTES];
} chunks;
if (writePosInChunk == -1)
{
switch (byte)
{
case ComputerWriteMore:
writePosInChunk = 0;
if (curWriteIndex < CHIP_SIZE / WRITE_CHUNK_SIZE_BYTES)
{
SendByte(ProgrammerWriteOK);
}
else
{
SendByte(ProgrammerWriteError);
}
break;
case ComputerWriteFinish:
// Just to confirm that we finished writing...
SendByte(ProgrammerWriteOK);
curCommandState = WaitingForCommand;
break;
case ComputerWriteCancel:
SendByte(ProgrammerWriteConfirmCancel);
curCommandState = WaitingForCommand;
break;
}
}
else
{
chunks.writeChunkBytes[writePosInChunk++] = byte;
if (writePosInChunk >= WRITE_CHUNK_SIZE_BYTES)
{
// Write this data
ExternalMem_Write(curWriteIndex * (WRITE_CHUNK_SIZE_BYTES/4), chunks.writeChunks, WRITE_CHUNK_SIZE_BYTES/4, ALL_CHIPS);
SendByte(ProgrammerWriteOK);
curWriteIndex++;
writePosInChunk = -1;
}
}
}
/** Event handler for the library USB Configuration Changed event. */ /** Event handler for the library USB Configuration Changed event. */