TN220

Implementing a Serial Download Manager for Two 256K Byte Flash Memories

Disclaimer

The programs described in this note are provided as a sample field reprogramming method only with no guarantees that they are fail-safe. How fail-safe a system needs to be is obviously application dependent. It is unlikely this sample or future samples will meet the needs of all users. Samples systems are provided as a starting point for programmers to implement their own field reprogramming solutions.

Introduction

This note describes a method of implementing a serial Download Manager (DLM) and downloaded program (DLP) as a pair of coresident programs on a Rabbit-based board having two 256K byte flash chips. The DLM resides in the primary flash hooked to CS0, and the DLP resides in the other flash. The sample code is released with Dynamic C.

Z-World and Rabbit Semiconductor manufacture many board types that have two 256K flash chips. Some examples are:

Why Use this Method?

The most robust way to field reprogram a remote Rabbit target is to have a separate device such as the Rabbitlink receive the program and reprogram the target using the Rabbit's bootstrap mode. That solution may be too expensive for some applications where low cost and small size are critical.

Internet Protocol Alternatives

The techniques used in this serial method could be easily adapted to an Internet protocol based DLM/DLP system. It would not be necessary to transmit the DLP in HEX format with checksums if a reliable transport protocol were used; a BIN image file would suffice. This would remove much of the complexity in the current sample, since the details of reliable transmission are hidden in the Z-World provided TCP/IP libraries. Z-World will provide Internet Protocol DLM/DLP samples in the future, possibly with Python script for the PC end, but users are free to write their own loading system now and several have.

Functional Overview

The Download Manager (DLM) is the primary program, the program that runs when the target is reset or powered on. The DLP (or secondary program) is transmitted using RS232 (over serial port B or C in these examples) in an Intel Hex file format (See Appendix). Both the DLM and DLP monitor the channel for non-ASCII characters (bytes with the high bit set) in order to detect a predefined/user-definable command string of user-definable length which is understood to be the restart command. When the restart command is detected, the DLM starts. A menu is displayed on the dumb terminal screen, giving the user a choice of actions. The first action either must be to enter a password, or restart the DLP. Restarting the DLP or entering a password are the only actions which may be performed without entering a password first. If no password is entered within a user-specifiable time limit, the DLM checks for the presence of a valid DLP and runs it if one is present, or restarts itself if one is not. The user can set the initial password at DLM compile time, and specify whether the password will be run-time changeable.

Other Software Needed

A dumb terminal program with the ability to send a raw ASCII or binary file and use software flow control (XON/XOFF) is needed. A good terminal program called Tera Term was used for to test these samples.

http://hp.vector.co.jp/authors/VA002416/teraterm.html

(In the author's opinion, HyperTerminal is notoriously cranky and should not be used).

Software flow control is required so that receiving the program data can be paused when a flash sector is written.

Running the DLM and DLP

The following files are used for this system:

The following numbered list walks through the hardware and software setup.

  1. Hardware Set Up (modemless)

    Make a three-wire connection between the target board and the PC by connecting TX of the serial channel being used on the Rabbit target board to the RX line of the PC COMM port being used. RX on the target goes to the TX line of the PC, and a common ground is the third connection. The samples are set up to use serial port B or C, uncomment the appropriate macro near the tops of the DLM and DLP samples to select the port.

    //*** uncomment one only! ***
    #define USE_SERIAL_PORT_B
    //#define USE_SERIAL_PORT_C

  2. Running the Dumb Terminal

    Set your dumb terminal program for 57600 baud, one stop bit, no parity, ANSI terminal emulation, XON/XOFF flow control and the appropriate COMM port. Slower boards may require a slower baud rate which can be changed with this macro in DLP_2_FLASH.C and DLM_SERIAL_2_FLASH.C:

    #define SERIAL_BAUDRATE 57600ul

  3. Compiling DLP_2_Flash.C

    Open the Options | Compiler dialog box from the main menu. Use the "Defines" button on the lower left corner to open the Defines dialog box. Type in the following macros:

    COMPILE_PRIMARY_PROGx ; COMPILE_SECONDARY_PROG 

    These macros are used in the BIOS source code to split the RAM and Flash between the DLM and DLP.

    Use Dynamic C's File menu to open DLP_2_Flash.C.

    Now compile DLP_2_Flash.C by selecting Compile | Compile to a .bin file. Choose the option, Compile with defined target configuration. When compilation finishes there should be a file called DLP_2_Flash.HEX in the same directory as DLP_2_Flash.C. This is the secondary program that will be coresident with the DLM.

  4. Compiling DLM_Serial_2_Flash.C

    To compile the DLM open DLM_Serial_2_Flash.C. Use the Options | Compiler dialog box and the Defines button to change the macros to this:

    COMPILE_PRIMARY_PROG; COMPILE_SECONDARY_PROGx;

    Hint: Just adding an `x' to the macro names when they are not needed saves a lot of typing. To save these settings without interfering with other programs, use the File menu to save a new project file.

    Now compile DLM_Serial_2_Flash.C to the target. After it loads, power down the target, remove the programming cable and repower the board.

Using the Download Manager Menu via the Dumb Terminal

The following menu should appear on the dumb terminal window.

Type "1" and then enter the default password, "123."

Now type 3 and see the prompt "Send HEX file as raw ASCII or Binary." Use the dumb terminal to send the file DLP_2_Flash.HEX. After it loads, you should see a continuous stream of `X' s on the dumb terminal screen.

To restart the DLM, use the terminal to send the file RESTART.BIN. This file has the default byte sequence that the DLM and DLP interpret as a restart command. The menu should appear refreshed on the terminal now. If you do nothing for 10 seconds, the DLP will start again. If you type option 4, the DLP will also run again. Or you can enter the password again and send another program or change the password.

Configuring the DLM

The following properties can be easily modified by changing macros at the top the DLM source code, see the source code for details:

In addition, the following functions may be redefined by the user to adapt the DLM for different serial channels or other communication means. See the source code for detailed specifications:

int PutChar(char ch)
int GetChar(char ch)
int InitComm()
int CloseComm()
int UserTask()
int PauseComm()
int ResumeComm()

Functional Details

BIOS Changes

The following conditional macro redefinitions are in the file RABBITBIOS.C

#ifdef  COMPILE_SECONDARY_PROG
#undef RAM_SIZE
#undef CS_FLASH
#define CS_FLASH 0x02           ;
This is the Chip Select (CS) for 2 flash boards
   ...
   #if (_RAM_SIZE_==0x80)           ; The rest of the directives
#undef _RAM_SIZE_             ;   split RAM and make the start of
#define _RAM_SIZE_ 0x40       ;   RAM for the DLP begin after
#else                            ;   the RAM used for the DLM
#if (_RAM_SIZE_==0x40)
#undef _RAM_SIZE_
#define _RAM_SIZE_ 0x20
#else
#if (_RAM_SIZE_==0x20)
#undef _RAM_SIZE_
#define _RAM_SIZE_ 0x10
#else
#error "unknown RAM size"
#endif
#endif
#endif
#undef RAM_START
#define RAM_SIZE _RAM_SIZE_
#define RAM_START 0x80+RAM_SIZE
#endif

The macros redefined here are used later in the BIOS to set up memory mapping information in ORG statements used by the compiler. A similar block of compiler directives exists for compiling the DLM:

#ifdef  COMPILE_PRIMARY_PROG
  ...

This can be viewed in the BIOS source code. For the DLM, the changes that the directives cause from a normal compilation is that the available RAM is cut in half to accommodate the DLP.

Monitoring for the DLM Signal

It is the responsibility of the DLP to monitor for the string that signals a DLM restart. The DLM is also set up to do this, but this is less critical since the DLM will time-out and restart itself, or the DLP, if a valid restart string is present, if no password is entered or no data is received

The following macros are needed in the DLP.

#define RESTARTSIGNAL "\xaa\xbb\xcc\xbb\xaa"
#define RESTARTSIGNAL_TIMEOUT 30

The following functions are also needed in the DLP. They can be cut and pasted from the DLM.

void ProcessRestartCommand(char ch)
void RestartDLM()

The sample DLPs provided set up the channel monitoring. The DLP must monitor the download channel and call ProcessRestartCommand() each time a byte with the high bit set is received. The sample DLPs hit a virtual watchdog timer in the tasks that check the serial channel for input so that the board will reset if it is not entered periodically.

The user can make the restart signal longer or shorter if desired. The requirement is that it be at least one byte long and all bytes have the high bit set. It can be sent from the terminal as keyboard combinations if the terminal program allows that, or as a file. An easy way to create a file containing the reset signal is to use the Dynamic C debug options to log STDOUT (without appending) to a file, and run this program:

main(){
printf("\xaa\xbb\xcc\xbb\xaa");
}

This file is provided with Dynamic C in /Samples/Down_load/RESTART.BIN.

Program Verification

The Intel HEX format file containing the program (DLP) has a one byte checksum for each 32 byte fragment of code. Verification of the checksum is performed as the data is received. An overall 16 bit CRC is also calculated as the data is received. The program size, starting address (always 40000h from the DLM's point of view), and CRC are stored in flash in the User ID Block area. A CRC check is made on the stored DLP program before attempting to run it. If the CRC is not valid, the DLM will be restarted instead.

Virtual Watchdog

The DLM and DLP both enable a single virtual watchdog timer to ensure that the programs don't enter a "hung" state. The periodic interrupt ISR normally hits the hardware watchdog timer in Dynamic C programs.

Possible Complications

There are several situations to take into consideration when creating a DLM and/or a DLP.

Stuck in a Loop

If the following code fragment is run, the Rabbit board will be locked in a tight loop that cannot be exited without recycling power or asserting the reset pin:

#asm
ipset 3      ;
turn off interrupts
tightLoop:      ; only a reset will get out of this!
call hitwd
jr tightLoop
#endasm

Interrupts are turned off, so the periodic ISR will not run and virtual watchdogs cannot time-out and cause a needed reset. The hardware watchdog gets hit in the tight loop, so it can't time-out and cause a reset either. There's no way out of the tight loop except asserting the reset externally or cycling power. It would be silly to have a piece of code that did this in either the DLM or DLP, but careless programming could result in a more complex set of instructions that have the same result. The best way to avoid this would be to use virtual watchdogs in your programs, let the periodic interrupt take care of hitting the hardware watchdog timer, and take great care not to create situations where interrupts could be shut off permanently.

Power Failure

The possibility of a power failure at the wrong time is something that any field reprogramming method should take into consideration. In this sample, both the DLM and DLP write only to the top half of the flash. Even huge sector flashes generally don't have a sector that crosses the 128K boundary, so there is no possibility that the DLM will be corrupted by a power failure while writing (or after erasing, but before writing) a sector in the DLM.

When the password is changed and written to flash, the DLM uses the writeUserBlock function. Starting with Dynamic C version 7.20, this function will do redundant writes to separate sectors to store persistent data to protect against data loss if a power failure occurs at the wrong moment, but in version 7.05, no such redundancy exists, so if the end user entered a new password, and a power failure occurred while the new password was being written, then it is possible the DLM would become unreachable. (Run-time password changing can be disabled.)

Note: the redundant copies in the user block are only used if the board has a version 3 System ID block. It is highly recommended to use a version 3 block. The only difference between a version 3 and version 2 block is the version number itself which can be examined in the structure member SysIDBlockType.tableVersion.

Modem Failure

These samples make no attempt to exercise any control over any modem device. This is left to the user if it is needed. A danger of using a modem for field reprogramming is that the modem could get into a hung state. It might be a good idea to use a modem with an external reset, and have the DLP occasionally assert the reset if no data has been received for a long while. The worst that could happen would be that a reset would occur just as a remote user was starting to reprogram, and they might need a retry in order to establish the connection.

Appendix: The Intel HEX File Format

The Hex file format consists of ASCII records of the following format:

:NNAAAATTDD1DD2DD3...DDNCC

A colon starts every record. Each letter represents a hexadecimal nibble with the following meanings.

NN - Number of data bytes in record. For Dynamic C generated hex files, this always either 02 for extended address records, 20 for data records, or 00 for EOF records.

AAAA - 16 bit address. This is the offset portion off the destination address using the Intel real-mode addressing. The segment portion of the real-mode address is given by the last extended address record in the HEX file previous to the data record. The physical offset into the memory device is computed by shifting the segment left 4 bits and adding the offset.

TT - Type of record. For Dynamic C generated hex files, this always either 02 for extended address records, 00 for data records, or 01 for EOF records.

DDi - Data byte

CC - 8 bit checksum of all previous bytes int the record. The two's compliment of the checksum is used.

Examples

There are three types of records: extended address, data and end of file. Each record starts with a colon. The Address field is only meaningful for data records.

1. Extended Address Record

This is the first record in a Dynamic C generated HEX file, and applies to all addresses that come after it until another extended address record is found.

:020000020000FC
Table 1. Extended Address Record
Data Length
Address
Record Type
Data
Checksum

02

0000

02

0000

FC

2. Data record

Most of the records in a hex file will be of this type.

:200000003D183090DE3DBD803D7B803D7E80FFFF3DF3EE00000000003DF390000000000025
Table 2. Data Record
Data Length
Address
Record Type
Data
Checksum

20

0000

00

3D1830 . . . 00

25

3. End of file record

This is the last record in a hex file

:00000001FF
Table 3. End of File Record
Data Length
Address
Record Type
Checksum

00

0000

01

FF


022-0068 Rev. A
Z-World
http://www.zworld.com
Voice: (530) 757-3737
FAX: (530) 757-3792
sales@zworld.com
Rabbit Semiconductor
http://www.rabbitsemiconductor.com
Voice: (530) 757-8400
FAX: (530) 757-8402
sales@rabbitsemiconductor.com