On 18:55 Sun 04 Nov     , Robert Jarzmik wrote:
> The current Y-Modem implementation has some limitations:
>  - Y-Modem/G protocol is not supported
>  - Multiple files (aka. batch) transfers are not supported
>  - Transfer speed over fast lines (USB console) is slow
>  - Code is not trivial to maintain (personnal opinion)
> 
> This implementation tries to address all these points by
> introducing loady2 command.
> 
> The effects are :
>  - transfer speed for Y-Modem over USB jumps from 2kBytes/s
>    to 180kBytes/s
>  - transfer speed for Y-Modem/G jumps to 200kBytes/s
>  - multiple file transfers are possible
> 
> This command was tested on a USB console and UART 9600bps
> serial line :
>  - NAKs (and retransmissions) were tested for faulty
>    serial lines
>  - multiple file transfers were tested
>  - Y-Modem, Y-Modem/G and X-Modem transfers were tested
> 
> Signed-off-by: Robert Jarzmik <[email protected]>
> 
> ---
> Since V1:
>  - add input fifo for small fifo hardwares
> Add a FIFO so that each getc will empty the hardware
> FIFO. This is very similar to the generic console code,
> except that the getc won't block each time for 100us,
> enabling faster lines (USB) to benefit their full speed.
>  - fix CRC calculation for big endian architectures
>    Thanks a lot Antony for the many patches testing !
>  - added some documentation
>  - amended the split as Sascha recommended
> ---
>  commands/Kconfig  |    1 +
>  commands/Makefile |    2 +-
>  commands/loadxy.c |  238 +++++++++++++++++++++
>  include/xymodem.h |   25 +++
>  lib/Kconfig       |    3 +
>  lib/Makefile      |    1 +
>  lib/xymodem.c     |  591 
> +++++++++++++++++++++++++++++++++++++++++++++++++++++
>  7 files changed, 860 insertions(+), 1 deletion(-)
>  create mode 100644 commands/loadxy.c
>  create mode 100644 include/xymodem.h
>  create mode 100644 lib/xymodem.c
> 
> diff --git a/commands/Kconfig b/commands/Kconfig
> index a52a01a..a7e9974 100644
> --- a/commands/Kconfig
> +++ b/commands/Kconfig
> @@ -261,6 +261,7 @@ config CMD_LOADB
>  
>  config CMD_LOADY
>       select CRC16
> +     select XYMODEM
>       tristate
>       prompt "loady"
>  
> diff --git a/commands/Makefile b/commands/Makefile
> index ff98051..44ad904 100644
> --- a/commands/Makefile
> +++ b/commands/Makefile
> @@ -3,7 +3,7 @@ obj-$(CONFIG_CMD_BOOTM)               += bootm.o
>  obj-$(CONFIG_CMD_UIMAGE)     += uimage.o
>  obj-$(CONFIG_CMD_LINUX16)    += linux16.o
>  obj-$(CONFIG_CMD_LOADB)              += loadb.o xyzModem.o
> -obj-$(CONFIG_CMD_LOADY)              += loadb.o xyzModem.o
> +obj-$(CONFIG_CMD_LOADY)              += loadb.o xyzModem.o loadxy.o
>  obj-$(CONFIG_CMD_LOADS)              += loads.o
>  obj-$(CONFIG_CMD_ECHO)               += echo.o
>  obj-$(CONFIG_CMD_MEMORY)     += mem.o
> diff --git a/commands/loadxy.c b/commands/loadxy.c
> new file mode 100644
> index 0000000..141bd7b
> --- /dev/null
> +++ b/commands/loadxy.c
> @@ -0,0 +1,238 @@
> +/**
> + * @file
> + * @brief loady and loadx  support.
> + *
> + * Provides loadx (over X-Modem) and loady(over Y-Modem) support to download
> + * images.
> + *
> + * FileName: commands/loadxy.c
> + */
> +/*
> + * (C) Copyright 2012 Robert Jarzmik <[email protected]>
> + *
> + * See file CREDITS for list of people who contributed to this
> + * project.
> + *
> + * This program 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 2 of
> + * the License, or (at your option) any later version.
> + *
> + * This program 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.
> + *
> + */
> +
> +/*
> + * Serial up- and download support
> + */
> +#include <common.h>
> +#include <command.h>
> +#include <console.h>
> +#include <xymodem.h>
> +#include <errno.h>
> +#include <getopt.h>
> +#include <fcntl.h>
> +#include <fs.h>
> +#include <malloc.h>
> +
> +#define DEF_FILE     "image.bin"
> +
> +/**
> + * @brief returns current used console device
> + *
> + * @return console device which is registered with CONSOLE_STDIN and
> + * CONSOLE_STDOUT
> + */
> +static struct console_device *get_current_console(void)
> +{
> +     struct console_device *cdev;
> +     /*
> +      * Assumption to have BOTH CONSOLE_STDIN AND STDOUT in the
> +      * same output console
> +      */
> +     for_each_console(cdev) {
> +             if ((cdev->f_active & (CONSOLE_STDIN | CONSOLE_STDOUT)))
> +                     return cdev;
> +     }
> +     return NULL;
> +}
> +
> +static int console_change_speed(struct console_device *cdev, int baudrate)
> +{
> +     int current_baudrate;
> +
> +     current_baudrate =
> +             (int)simple_strtoul(dev_get_param(&cdev->class_dev,
> +                                               "baudrate"), NULL, 10);
> +     if (baudrate && baudrate != current_baudrate) {
> +             printf("## Switch baudrate from %d to %d bps and press ENTER 
> ...\n",
> +                    current_baudrate, baudrate);
> +             mdelay(50);
> +             cdev->setbrg(cdev, baudrate);
> +             mdelay(50);
> +     }
> +     return current_baudrate;
> +}
> +
> +/**
> + * @brief provide the loady(Y-Modem or Y-Modem/G) support
> + *
> + * @param argc number of arguments
> + * @param argv arguments of loady command
> + *
> + * @return success or failure
> + */
> +static int do_loady(int argc, char *argv[])
> +{
> +     int is_ymodemg = 0, rc = 0, opt, rcode = 0;
> +     int load_baudrate = 0, current_baudrate;
> +     struct console_device *cdev = NULL;
> +
> +     while ((opt = getopt(argc, argv, "b:g")) > 0) {
> +             switch (opt) {
> +             case 'b':
> +                     load_baudrate = (int)simple_strtoul(optarg, NULL, 10);
> +                     break;
> +             case 'g':
> +                     is_ymodemg = 1;
> +                     break;
> +             default:
> +                     perror(argv[0]);
> +                     return 1;
> +             }
> +     }
> +
> +     cdev = get_current_console();
> +     if (NULL == cdev) {
this really look wired
        if (!cdev)
> +             printf("%s:No console device with STDIN and STDOUT\n", argv[0]);
> +             return -ENODEV;
> +     }
> +
> +     current_baudrate = console_change_speed(cdev, load_baudrate);
> +     printf("## Ready for binary (ymodem) download at %d bps...\n",
> +            load_baudrate ? load_baudrate : current_baudrate);
> +
> +     if (is_ymodemg)
> +             rc = do_load_serial_ymodemg(cdev);
> +     else
> +             rc = do_load_serial_ymodem(cdev);
> +
> +     if (rc < 0) {
> +             printf("## Binary (ymodem) download aborted (%d)\n", rc);
> +             rcode = 1;
> +     }
> +
> +     console_change_speed(cdev, current_baudrate);
> +
> +     return rcode;
> +}
> +
> +/**
> + * @brief provide the loadx(X-Modem) support
> + *
> + * @param argc number of arguments
> + * @param argv arguments of loadx command
> + *
> + * @return success or failure
> + */
> +static int do_loadx(int argc, char *argv[])
> +{
> +     ulong offset = 0;
> +     int load_baudrate = 0, current_baudrate, ofd, opt, rcode = 0;
> +     int open_mode = O_WRONLY;
> +     char *output_file = NULL;
> +     struct console_device *cdev = NULL;
> +
> +     while ((opt = getopt(argc, argv, "f:b:o:c")) > 0) {
> +             switch (opt) {
> +             case 'f':
> +                     output_file = optarg;
> +                     break;
> +             case 'b':
> +                     load_baudrate = (int)simple_strtoul(optarg, NULL, 10);
> +                     break;
> +             case 'o':
> +                     offset = (int)simple_strtoul(optarg, NULL, 10);
> +                     break;
> +             case 'c':
> +                     open_mode |= O_CREAT;
> +                     break;
> +             default:
> +                     perror(argv[0]);
> +                     return 1;
> +             }
> +     }
> +
> +     cdev = get_current_console();
> +     if (NULL == cdev) {
        ditto
> +             printf("%s:No console device with STDIN and STDOUT\n", argv[0]);
> +             return -ENODEV;
> +     }
> +
> +     /* Load Defaults */
> +     if (NULL == output_file)
        ditto
> +             output_file = DEF_FILE;
> +
> +     /* File should exist */
> +     ofd = open(output_file, open_mode);
> +     if (ofd < 0) {
> +             perror(argv[0]);
> +             return 3;
> +     }
> +     /* Seek to the right offset */
> +     if (offset) {
> +             int seek = lseek(ofd, offset, SEEK_SET);
> +             if (seek != offset) {
> +                     close(ofd);
> +                     ofd = 0;
> +                     perror(argv[0]);
> +                     return 4;
> +             }
> +     }
> +
> +     current_baudrate = console_change_speed(cdev, load_baudrate);
> +     printf("## Ready for binary (X-Modem) download "
> +            "to 0x%08lX offset on %s device at %d bps...\n", offset,
> +            output_file, load_baudrate);
> +     rcode = do_load_serial_ymodem(cdev);
> +     if (rcode < 0) {
> +             printf("## Binary (kermit) download aborted (%d)\n", rcode);
> +             rcode = 1;
> +     }
> +     console_change_speed(cdev, current_baudrate);
> +
> +     return rcode;
> +}

> +static void xy_flush(struct console_device *cdev, struct kfifo *fifo)
> +{
> +     while (cdev->tstc(cdev))
no timeout?
> +             cdev->getc(cdev);
> +     mdelay(250);
> +     while (cdev->tstc(cdev))
ditto
> +             cdev->getc(cdev);
> +     kfifo_reset(fifo);
> +}
> +

> + */
> +static ssize_t xy_read_block(struct xyz_ctxt *proto, struct xy_block *blk,
> +     uint64_t timeout)
> +{
> +     ssize_t rc, data_len = 0;
> +     unsigned char hdr, seqs[2], crcs[2];
> +     int crc = 0, hdr_found = 0;
> +     uint64_t start = get_time_ns();
> +
> +     while (!hdr_found) {
> +             rc = xy_gets(proto->cdev, proto->fifo, &hdr, 1, timeout);
> +             xy_dbg("read 0x%x(%c) -> %d\n", hdr, hdr, rc);
> +             if (rc < 0)
> +                     goto out;
> +             if (is_timeout(start, timeout))
> +                     goto timeout;
> +             switch (hdr) {
> +             case SOH:
> +                     data_len = 128;
> +                     hdr_found = 1;
> +                     proto->total_SOH++;
no capital please
> +                     break;
> +             case STX:
> +                     data_len = 1024;
> +                     hdr_found = 1;
boolean
> +                     proto->total_STX++;
> +                     break;
> +             case CAN:
> +                     rc = -ECONNABORTED;
> +                     if (proto->total_CAN++ > MAX_CAN_BEFORE_ABORT)
> +                             goto out;
> +                     break;
> +             case EOT:
> +                     rc = 0;
> +                     blk->len = 0;
> +                     goto out;
> +             default:
> +                     break;
> +             }
> +     }
> +
> +     blk->seq = 0;
> +     rc = xy_gets(proto->cdev, proto->fifo, seqs, 2, timeout);
> +     if (rc < 0)
> +             goto out;
> +     blk->seq = seqs[0];
> +     if (255 - seqs[0] != seqs[1])
> +             return -EBADMSG;
> +
> +     rc = xy_gets(proto->cdev, proto->fifo, blk->buf, data_len, timeout);
> +     if (rc < 0)
> +             goto out;
> +     blk->len = rc;
> +
> +     switch (proto->crc_mode) {
> +     case CRC_ADD8:
> +             rc = xy_gets(proto->cdev, proto->fifo, crcs, 1, timeout);
> +             crc = crcs[0];
> +             break;
> +     case CRC_CRC16:
> +             rc = xy_gets(proto->cdev, proto->fifo, crcs, 2, timeout);
> +             crc = be16_to_cpu(*(uint16_t *)crcs);
> +             break;
> +     case CRC_NONE:
> +             rc = 0;
> +             break;
> +     }
> +     if (rc < 0)
> +             goto out;
> +
> +     rc = check_crc(blk->buf, data_len, crc, proto->crc_mode);
> +     if (rc < 0)
> +             goto out;
> +     return data_len;
> +timeout:
> +     return -ETIMEDOUT;
> +out:
> +     return rc;
> +}
> +
> +static int check_blk_seq(struct xyz_ctxt *proto, struct xy_block *blk,
> +     int read_rc)
> +{
> +     if (blk->seq == ((proto->next_blk - 1) % 256))
> +             return -EALREADY;
> +     if (blk->seq != proto->next_blk)
> +             return -EILSEQ;
> +     return read_rc;
> +}
> +
> +static int parse_first_block(struct xyz_ctxt *proto, struct xy_block *blk)
> +{
> +     int filename_len;
> +     char *str_num;
> +
> +     filename_len = strlen(blk->buf);
> +     if (filename_len > blk->len)
> +             return -EINVAL;
> +     memset(proto->filename, 0, sizeof(proto->filename));
no need just add 0 at the end
> +     strncpy(proto->filename, blk->buf, filename_len);
> +     str_num = blk->buf + filename_len + 1;
> +     strsep(&str_num, " ");
> +     proto->file_len = simple_strtoul(blk->buf + filename_len + 1, NULL, 10);
> +     return 1;
> +}
Best Regards,
J.

_______________________________________________
barebox mailing list
[email protected]
http://lists.infradead.org/mailman/listinfo/barebox

Reply via email to