mirror of https://github.com/wolfSSL/wolfBoot.git
255 lines
6.8 KiB
C
255 lines
6.8 KiB
C
/* test-update-server.c
|
|
*
|
|
* Copyright (C) 2006-2021 wolfSSL Inc.
|
|
*
|
|
* This file is part of wolfSSL. (formerly known as CyaSSL)
|
|
*
|
|
* wolfSSL is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation; either version 3 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* wolfSSL is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program; if not, write to the Free Software
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
|
|
*
|
|
*=============================================================================
|
|
*
|
|
* OTA Upgrade mechanism implemented using UART
|
|
*
|
|
*/
|
|
|
|
#define _XOPEN_SOURCE 600
|
|
|
|
#include <stdio.h> /* standard in/out procedures */
|
|
#include <stdlib.h> /* defines system calls */
|
|
#include <string.h> /* necessary for memset */
|
|
#include <netdb.h>
|
|
#include <sys/socket.h> /* used for all socket calls */
|
|
#include <netinet/in.h> /* used for sockaddr_in6 */
|
|
#include <arpa/inet.h>
|
|
#include <errno.h>
|
|
#include <unistd.h>
|
|
#include <fcntl.h>
|
|
#include <sys/stat.h>
|
|
#include <termios.h>
|
|
#include <signal.h>
|
|
#include <errno.h>
|
|
|
|
#define MSGLEN (4 + 4 + 8)
|
|
#ifndef UART_DEV
|
|
#ifdef __MACH__
|
|
#define UART_DEV "/dev/cu.usbmodem1411"
|
|
#else
|
|
#define UART_DEV "/dev/ttyACM0"
|
|
#endif
|
|
#endif
|
|
#ifndef B115200
|
|
#define B115200 115200
|
|
#endif
|
|
|
|
static volatile int cleanup; /* To handle shutdown */
|
|
union usb_ack {
|
|
uint32_t offset;
|
|
uint8_t u8[4];
|
|
};
|
|
|
|
|
|
static uint8_t pktbuf[MSGLEN];
|
|
static unsigned int pktbuf_size = 0;
|
|
static int serialfd = -1;
|
|
static uint32_t high_ack;
|
|
|
|
|
|
void alarm_handler(int signo)
|
|
{
|
|
if (serialfd >= 0 && pktbuf_size > 0) {
|
|
write(serialfd, pktbuf, pktbuf_size);
|
|
printf("retransmitting...\n");
|
|
alarm(2);
|
|
}
|
|
}
|
|
|
|
static int recv_ack(union usb_ack *ack)
|
|
{
|
|
unsigned char c;
|
|
int res;
|
|
int err = 0;
|
|
while (1) {
|
|
res = read(serialfd, &c, 1);
|
|
if (res <= 0) {
|
|
continue;
|
|
}
|
|
if (c == '#') {
|
|
int i = 0;
|
|
err = 0;
|
|
ack->offset = 0;
|
|
while (i < 4) {
|
|
res = read(serialfd, &c, 1);
|
|
if (res < 1) {
|
|
usleep(10000);
|
|
continue;
|
|
}
|
|
ack->u8[i++] = c;
|
|
}
|
|
return 0;
|
|
}
|
|
if (c == '!') {
|
|
if (++err > 3) {
|
|
return -1;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
static void check(uint8_t *pkt, int size)
|
|
{
|
|
uint16_t *c = (uint16_t *)(pkt + 2);
|
|
int i;
|
|
uint16_t *p = (uint16_t *)(pkt + 4);
|
|
*c = 0;
|
|
pkt[0] = 0xA5;
|
|
pkt[1] = 0x5A;
|
|
for (i = 0; i < ((size - 4) >> 1); i++)
|
|
*c += p[i];
|
|
}
|
|
|
|
|
|
int main(int argc, char** argv)
|
|
{
|
|
/* Variables for awaiting datagram */
|
|
int res = 1;
|
|
uint32_t len, tot_len;
|
|
int ffd; /* Firmware file descriptor */
|
|
struct stat st;
|
|
union usb_ack ack;
|
|
struct termios tty;
|
|
sigset(SIGALRM, alarm_handler);
|
|
|
|
if (argc != 2) {
|
|
printf("Usage: %s firmware_filename\n", argv[0]);
|
|
exit(1);
|
|
}
|
|
|
|
/* open file and get size */
|
|
ffd = open(argv[1], O_RDONLY);
|
|
if (ffd < 0) {
|
|
perror("opening file");
|
|
exit(2);
|
|
}
|
|
res = fstat(ffd, &st);
|
|
if (res != 0) {
|
|
perror("fstat file");
|
|
exit(2);
|
|
}
|
|
tot_len = st.st_size;
|
|
|
|
/* open UART */
|
|
printf("Opening %s UART\n", UART_DEV);
|
|
serialfd = open(UART_DEV, O_RDWR | O_NOCTTY);
|
|
if (serialfd < 0) {
|
|
fprintf(stderr, "failed opening serial %s\n", UART_DEV);
|
|
exit(2);
|
|
}
|
|
tcgetattr(serialfd, &tty);
|
|
cfsetospeed(&tty, B115200);
|
|
cfsetispeed(&tty, B115200);
|
|
tty.c_cflag = (tty.c_cflag & ~CSIZE) | (CS8);
|
|
tty.c_iflag &= ~(IGNBRK | IXON | IXOFF | IXANY| INLCR | ICRNL);
|
|
tty.c_oflag &= ~OPOST;
|
|
tty.c_oflag &= ~(ONLCR|OCRNL);
|
|
tty.c_cflag &= ~(PARENB | PARODD | CSTOPB);
|
|
tty.c_lflag &= ~(ICANON | ECHO | ECHOE | ISIG);
|
|
tty.c_iflag &= ~ISTRIP;
|
|
tty.c_cc[VMIN] = 0;
|
|
tty.c_cc[VTIME] = 5;
|
|
tcsetattr(serialfd, TCSANOW, &tty);
|
|
|
|
/* Wait for start hash (asterisk) */
|
|
while (1) {
|
|
char c;
|
|
|
|
res = read(serialfd, &c, 1);
|
|
if (res <= 0) {
|
|
usleep(10000);
|
|
continue;
|
|
}
|
|
if (c == '*') {
|
|
break;
|
|
}
|
|
else {
|
|
printf("%c",c);
|
|
fflush(stdout);
|
|
}
|
|
|
|
}
|
|
printf("Target connected.\n");
|
|
usleep(500000);
|
|
printf("Starting update.\n");
|
|
|
|
do {
|
|
uint8_t hdr[2] = { 0xA5, 0x5A};
|
|
len = 0;
|
|
lseek(ffd, 0, SEEK_SET);
|
|
write(serialfd, &hdr, 2);
|
|
write(serialfd, &tot_len, sizeof(uint32_t));
|
|
printf("Sent image file size (%d)\n", tot_len);
|
|
while (len < tot_len) {
|
|
res = recv_ack(&ack);
|
|
if (res == 0) {
|
|
if (ack.offset > tot_len) {
|
|
printf("Ignore bogus ack...\n");
|
|
continue;
|
|
}
|
|
if (ack.offset < high_ack) {
|
|
printf("Ignore low ack...\n");
|
|
continue;
|
|
}
|
|
high_ack = ack.offset;
|
|
pktbuf_size = 0;
|
|
if (ack.offset != len) {
|
|
printf("buf rewind %u\n", ack.offset);
|
|
lseek(ffd, ack.offset, SEEK_SET);
|
|
len = ack.offset;
|
|
}
|
|
memcpy(pktbuf + 4, &len, sizeof(len));
|
|
res = read(ffd, pktbuf + 4 + sizeof(uint32_t), MSGLEN - (4 + sizeof(uint32_t)));
|
|
if (res < 0) {
|
|
printf("EOF\r\n");
|
|
cleanup = 1;
|
|
break;
|
|
}
|
|
pktbuf_size = res + 4 + sizeof(uint32_t);
|
|
check(pktbuf, pktbuf_size);
|
|
write(serialfd, pktbuf, pktbuf_size);
|
|
len += res;
|
|
|
|
printf("Sent bytes: %d/%d %02x:%02x:%02x:%02x:%02x:%02x:%02x:%02x \r",
|
|
len, tot_len, pktbuf[0], pktbuf[1], pktbuf[2], pktbuf[3], pktbuf[4], pktbuf[5], pktbuf[6], pktbuf[7]);
|
|
|
|
fflush(stdout);
|
|
alarm(2);
|
|
}
|
|
}
|
|
printf("\n\n");
|
|
} while (0);
|
|
|
|
printf("waiting for last ack...\n");
|
|
while(!cleanup) {
|
|
res = recv_ack(&ack);
|
|
if ((res == 0 ) && (ack.offset == tot_len)) {
|
|
printf("Transfer complete.\n");
|
|
break;
|
|
}
|
|
}
|
|
printf("All done.\n");
|
|
close(serialfd);
|
|
|
|
return 0;
|
|
}
|