/* 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 /* standard in/out procedures */ #include /* defines system calls */ #include /* necessary for memset */ #include #include /* used for all socket calls */ #include /* used for sockaddr_in6 */ #include #include #include #include #include #include #include #include #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; }