Implemented xmodem data sending
This commit is contained in:
parent
e56cf52f28
commit
7e3131d9d1
|
|
@ -28,8 +28,26 @@
|
||||||
extern "C" {
|
extern "C" {
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Send an XMODEM packet over the serial port.
|
||||||
|
*
|
||||||
|
* @param data: pointer to payload data.
|
||||||
|
* @param size: data size, must be either 128 or 1024 byte.
|
||||||
|
* @param blockNum: packet sequence number.
|
||||||
|
*/
|
||||||
void xmodem_sendPacket(const void *data, size_t size, uint8_t blockNum);
|
void xmodem_sendPacket(const void *data, size_t size, uint8_t blockNum);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Send data using the XMODEM protocol, blocking function.
|
||||||
|
* Data transfer begins when the start command from the receiving endpoint is
|
||||||
|
* detected.
|
||||||
|
*
|
||||||
|
* @param size: data size.
|
||||||
|
* @param callback: pointer to a callback function in charge of providing data
|
||||||
|
* for the new packets being sent.
|
||||||
|
*/
|
||||||
|
ssize_t xmodem_sendData(size_t size, int (*callback)(uint8_t *, size_t));
|
||||||
|
|
||||||
#ifdef __cplusplus
|
#ifdef __cplusplus
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
|
||||||
|
|
@ -18,11 +18,12 @@
|
||||||
* along with this program; if not, see <http://www.gnu.org/licenses/> *
|
* along with this program; if not, see <http://www.gnu.org/licenses/> *
|
||||||
***************************************************************************/
|
***************************************************************************/
|
||||||
|
|
||||||
#include <crc.h>
|
#include <usb_vcom.h>
|
||||||
|
#include <stdbool.h>
|
||||||
|
#include <stdlib.h>
|
||||||
#include <xmodem.h>
|
#include <xmodem.h>
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
#include <stdio.h>
|
#include <crc.h>
|
||||||
#include <usb_vcom.h>
|
|
||||||
|
|
||||||
#define SOH (0x01) /* start of 128-byte data packet */
|
#define SOH (0x01) /* start of 128-byte data packet */
|
||||||
#define STX (0x02) /* start of 1024-byte data packet */
|
#define STX (0x02) /* start of 1024-byte data packet */
|
||||||
|
|
@ -35,11 +36,29 @@
|
||||||
#define ABT2 (0x61) /* 'a' == 0x61, assume try abort by user typing */
|
#define ABT2 (0x61) /* 'a' == 0x61, assume try abort by user typing */
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @internal
|
||||||
|
* Collect a given amount of data from serial port.
|
||||||
|
*
|
||||||
|
* @param ptr: pointer to destination buffer.
|
||||||
|
* @param size: number of bytes to be retrieved.
|
||||||
|
*/
|
||||||
|
static void waitForData(uint8_t *ptr, size_t size)
|
||||||
|
{
|
||||||
|
size_t curSize = 0;
|
||||||
|
|
||||||
|
while(curSize < size)
|
||||||
|
{
|
||||||
|
ssize_t recvd = vcom_readBlock(ptr + curSize, size - curSize);
|
||||||
|
if(recvd >= 0) curSize += recvd;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
void xmodem_sendPacket(const void *data, size_t size, uint8_t blockNum)
|
void xmodem_sendPacket(const void *data, size_t size, uint8_t blockNum)
|
||||||
{
|
{
|
||||||
// Bad payload size, null block number or null data pointer: do not send
|
// Bad payload size, null block number or null data pointer: do not send
|
||||||
if(((size != 128) && (size != 1024)) || (blockNum == 0) || (data == NULL))
|
if(((size != 128) && (size != 1024)) || (data == NULL))
|
||||||
{
|
{
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
@ -66,3 +85,76 @@ void xmodem_sendPacket(const void *data, size_t size, uint8_t blockNum)
|
||||||
buf[1] = crc & 0xFF;
|
buf[1] = crc & 0xFF;
|
||||||
vcom_writeBlock(buf, 2);
|
vcom_writeBlock(buf, 2);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
ssize_t xmodem_sendData(size_t size, ssize_t (*callback)(uint8_t *, size_t))
|
||||||
|
{
|
||||||
|
// Wait for the start command from the receiver, only CRC mode is supported.
|
||||||
|
uint8_t cmd = 0;
|
||||||
|
while(cmd != CRC)
|
||||||
|
{
|
||||||
|
waitForData(&cmd, 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Send data.
|
||||||
|
uint8_t dataBuf[1024];
|
||||||
|
uint8_t blockNum = 1;
|
||||||
|
size_t sentSize = 0;
|
||||||
|
|
||||||
|
while(sentSize < size)
|
||||||
|
{
|
||||||
|
size_t remaining = size - sentSize;
|
||||||
|
size_t blockSize = 1024;
|
||||||
|
if(remaining < blockSize) blockSize = remaining;
|
||||||
|
|
||||||
|
// Request data, stop transfer on failure
|
||||||
|
if(callback(dataBuf, blockSize) < 0)
|
||||||
|
{
|
||||||
|
cmd = CAN;
|
||||||
|
vcom_writeBlock(&cmd, 1);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Pad data to 128 or 1024 bytes, if necessary
|
||||||
|
size_t padSize = 0;
|
||||||
|
if(blockSize < 128)
|
||||||
|
{
|
||||||
|
padSize = 128 - blockSize;
|
||||||
|
}
|
||||||
|
else if(blockSize < 1024)
|
||||||
|
{
|
||||||
|
padSize = 1024 - blockSize;
|
||||||
|
}
|
||||||
|
|
||||||
|
uint8_t *ptr = dataBuf + padSize + 1;
|
||||||
|
memset(ptr, 0x1A, padSize);
|
||||||
|
|
||||||
|
// Send packet and wait for ACK, resend on NACK.
|
||||||
|
bool ok = false;
|
||||||
|
do
|
||||||
|
{
|
||||||
|
blockSize += padSize;
|
||||||
|
xmodem_sendPacket(dataBuf, blockSize, blockNum);
|
||||||
|
|
||||||
|
cmd = 0;
|
||||||
|
while((cmd != ACK) && (cmd != NAK))
|
||||||
|
{
|
||||||
|
waitForData(&cmd, 1);
|
||||||
|
if(cmd == ACK) ok = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
while(ok == false);
|
||||||
|
|
||||||
|
sentSize += blockSize - padSize;
|
||||||
|
blockNum++;
|
||||||
|
}
|
||||||
|
|
||||||
|
// End of transfer
|
||||||
|
cmd = EOT;
|
||||||
|
vcom_writeBlock(&cmd, 1);
|
||||||
|
while(cmd != ACK)
|
||||||
|
{
|
||||||
|
waitForData(&cmd, 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
return sentSize;
|
||||||
|
}
|
||||||
|
|
|
||||||
|
|
@ -24,20 +24,18 @@
|
||||||
#include <interfaces/platform.h>
|
#include <interfaces/platform.h>
|
||||||
#include <interfaces/delays.h>
|
#include <interfaces/delays.h>
|
||||||
#include "W25Qx.h"
|
#include "W25Qx.h"
|
||||||
#include "usb_vcom.h"
|
|
||||||
|
|
||||||
#define SOH (0x01) /* start of 128-byte data packet */
|
|
||||||
#define STX (0x02) /* start of 1024-byte data packet */
|
|
||||||
#define EOT (0x04) /* End Of Transmission */
|
|
||||||
#define ACK (0x06) /* ACKnowledge, receive OK */
|
|
||||||
#define NAK (0x15) /* Negative ACKnowledge, receiver ERROR, retry */
|
|
||||||
#define CAN (0x18) /* two CAN in succession will abort transfer */
|
|
||||||
#define CRC (0x43) /* 'C' == 0x43, request 16-bit CRC, use in place of first NAK for CRC mode */
|
|
||||||
#define ABT1 (0x41) /* 'A' == 0x41, assume try abort by user typing */
|
|
||||||
#define ABT2 (0x61) /* 'a' == 0x61, assume try abort by user typing */
|
|
||||||
|
|
||||||
static const size_t FLASH_SIZE = 16*1024*1024; /* 16MB */
|
static const size_t FLASH_SIZE = 16*1024*1024; /* 16MB */
|
||||||
uint8_t blockData[1024];
|
size_t addr = 0;
|
||||||
|
|
||||||
|
int callback(uint8_t *ptr, size_t size)
|
||||||
|
{
|
||||||
|
if((addr + size) > FLASH_SIZE) return -1;
|
||||||
|
W25Qx_readData(addr, ptr, size);
|
||||||
|
addr += size;
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
int main()
|
int main()
|
||||||
{
|
{
|
||||||
|
|
@ -45,52 +43,7 @@ int main()
|
||||||
W25Qx_init();
|
W25Qx_init();
|
||||||
W25Qx_wakeup();
|
W25Qx_wakeup();
|
||||||
|
|
||||||
uint8_t cmd = 0;
|
xmodem_sendData(FLASH_SIZE, callback);
|
||||||
while(cmd != 'C')
|
|
||||||
{
|
|
||||||
platform_ledOn(GREEN);
|
|
||||||
sleepFor(0,50);
|
|
||||||
platform_ledOff(GREEN);
|
|
||||||
sleepFor(0,50);
|
|
||||||
vcom_readBlock(&cmd, 1);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
uint8_t block = 1;
|
|
||||||
for(size_t addr = 0; addr < FLASH_SIZE; )
|
|
||||||
{
|
|
||||||
W25Qx_readData(addr, blockData, 1024);
|
|
||||||
|
|
||||||
bool ok = false;
|
|
||||||
do
|
|
||||||
{
|
|
||||||
xmodem_sendPacket(blockData, 1024, block);
|
|
||||||
|
|
||||||
cmd = 0;
|
|
||||||
while((cmd != ACK) && (cmd != NAK))
|
|
||||||
{
|
|
||||||
platform_ledOn(RED);
|
|
||||||
sleepFor(0,50);
|
|
||||||
platform_ledOff(RED);
|
|
||||||
sleepFor(0,50);
|
|
||||||
vcom_readBlock(&cmd, 1);
|
|
||||||
|
|
||||||
if(cmd == ACK) ok = true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
while(ok == false);
|
|
||||||
|
|
||||||
block++;
|
|
||||||
if(block == 255) block = 1;
|
|
||||||
addr += 1024;
|
|
||||||
}
|
|
||||||
|
|
||||||
cmd = EOT;
|
|
||||||
vcom_writeBlock(&cmd, 1);
|
|
||||||
while(cmd != ACK)
|
|
||||||
{
|
|
||||||
vcom_readBlock(&cmd, 1);
|
|
||||||
}
|
|
||||||
|
|
||||||
while(1)
|
while(1)
|
||||||
{
|
{
|
||||||
|
|
|
||||||
Loading…
Reference in New Issue