/*
* ekermit-test.c
* --------------
* A test program for the E-Kermit library.
*
* Copyright (c) 2016, NORDUnet A/S All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are
* met:
* - Redistributions of source code must retain the above copyright notice,
* this list of conditions and the following disclaimer.
*
* - Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
*
* - Neither the name of the NORDUnet nor the names of its contributors may
* be used to endorse or promote products derived from this software
* without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS
* IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
* TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A
* PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
* HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED
* TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
* PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
* LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
* NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
/* This receives one or more files from a kermit client (e.g. C-Kermit),
* and stores them in a minimal file system on SDRAM.
*/
#include <stdio.h>
#include <string.h>
#include <ctype.h>
#include "stm-init.h"
#include "stm-sdram.h"
#include "sdram-malloc.h"
#include "stm-kermit.h"
#include "stm-uart.h"
#include "ringbuf.h"
/* alignment for file allocations */
#define ALIGN 16
/* maximum length of file name */
#define NAMELEN 32
/* maximum number of files */
#define NFILE 4
typedef struct {
char name[NAMELEN];
uint8_t *loc;
size_t len;
} filetab_t;
filetab_t filetab[NFILE], *curfile;
int nfile;
static ringbuf_t uart_ringbuf;
/* current character received from UART */
static uint8_t uart_rx;
/* Callback for HAL_UART_Receive_IT(). Must be public. */
void HAL_UART1_RxCpltCallback(UART_HandleTypeDef *huart)
{
ringbuf_write_char(&uart_ringbuf, uart_rx);
HAL_UART_Receive_IT(huart, &uart_rx, 1);
}
void uart1_init(void)
{
ringbuf_init(&uart_ringbuf);
HAL_UART_Receive_IT(&huart_mgmt, &uart_rx, 1);
}
int uart1_read_char(uint8_t *c)
{
return ringbuf_read_char(&uart_ringbuf, c);
}
#ifdef DUMP
static void hexdump(uint8_t *buf, int len)
{
for ( ; len > 0; buf += 16, len -= 16) {
int plen = (len > 16) ? 16 : len;
int i;
for (i = 0; i < plen; ++i)
printf("%02x ", buf[i]);
for ( ; i < 16; ++i)
printf(" ");
printf("| ");
for (i = 0; i < plen; ++i)
printf("%c", isprint(buf[i]) ? buf[i] : ' ');
printf("\n");
}
}
#endif
/* Override weak function defintions for callouts in stm-kermit.c */
/* Called when kermit receives an F (file) packet
*/
int kermit_openfile(unsigned char *name)
{
if (nfile == NFILE) {
curfile = NULL;
return -1;
}
curfile = &filetab[nfile];
strncpy(curfile->name, (char *)name, sizeof(curfile->name));
curfile->loc = sdram_malloc(0);
if (curfile->loc == NULL) {
curfile = NULL;
return -1;
}
curfile->len = 0;
++nfile;
return 0;
}
/* Called when kermit's receive buffer is full, after some number of D
* (data) packets
*/
int kermit_writefile(unsigned char *buf, int len)
{
if (curfile == NULL)
return -1;
uint8_t *writeptr = sdram_malloc(len);
if (writeptr == NULL)
return -1;
memcpy(writeptr, buf, len);
curfile->len += len;
return 0;
}
/* Called when kermit receives a Z (EOF) packet
*/
int kermit_closefile(void)
{
if (curfile == NULL)
return -1;
int pad = ALIGN - (curfile->len & (ALIGN - 1));
if (sdram_malloc(pad) == NULL)
return -1;
curfile = NULL;
return 0;
}
/* Called when kermit receives a Z (EOF) packet with a D (discard) flag
*/
int kermit_cancelfile(void)
{
if (curfile == NULL)
return -1;
sdram_free(curfile->loc);
memset(curfile, 0, sizeof(*curfile));
--nfile;
return 0;
}
/* The main function is pretty simple, because the kermit library does all
* the work out of sight.
*/
int main(void)
{
stm_init();
sdram_init();
uart1_init();
while (1) {
memset((void *)filetab, 0, sizeof(filetab));
curfile = NULL;
nfile = 0;
kermit_main();
for (int i = 0; i < nfile; ++i) {
filetab_t *f = &filetab[i];
printf("%d. %-16s %p %d\n", i, f->name, f->loc, f->len);
#ifdef DUMP
hexdump(f->loc, f->len);
#endif
sdram_free(f->loc);
}
}
}