⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 flash.c

📁 eCos1.31版
💻 C
字号:
//==========================================================================////        flash.c////        ARM PID7 eval board FLASH program tool////==========================================================================//####COPYRIGHTBEGIN####//                                                                          // -------------------------------------------                              // The contents of this file are subject to the Red Hat eCos Public License // Version 1.1 (the "License"); you may not use this file except in         // compliance with the License.  You may obtain a copy of the License at    // http://www.redhat.com/                                                   //                                                                          // Software distributed under the License is distributed on an "AS IS"      // basis, WITHOUT WARRANTY OF ANY KIND, either express or implied.  See the // License for the specific language governing rights and limitations under // the License.                                                             //                                                                          // The Original Code is eCos - Embedded Configurable Operating System,      // released September 30, 1998.                                             //                                                                          // The Initial Developer of the Original Code is Red Hat.                   // Portions created by Red Hat are                                          // Copyright (C) 1998, 1999, 2000 Red Hat, Inc.                             // All Rights Reserved.                                                     // -------------------------------------------                              //                                                                          //####COPYRIGHTEND####//==========================================================================//#####DESCRIPTIONBEGIN####//// Author(s):     gthomas// Contributors:  gthomas// Date:          1998-11-18// Description:   Tool used to program onboard FLASH image//####DESCRIPTIONEND####//// This program will program the FLASH on the PID board//#include <pkgconf/libc.h>   // Configuration header#include <cyg/kernel/kapi.h>#include <stdlib.h>#include <ctype.h>#include <cyg/infra/testcase.h>#include <sys/cstartup.h>#ifndef FALSE#define FALSE 0#define TRUE  1#endif#define SYNC_COUNT 63extern void diag_printf(const char *, ...);int identify_FLASH(void);void write_sector(int, char *);bool load_srecords(char (*readc)(), CYG_ADDRESS *start, int *size);char dbuf[256];char *raw = (char *)0x10000;char *flash_buffer = (char *)0x30000;int pos, len;// FUNCTIONSexternC voidcyg_package_start( void ){#ifdef CYGPKG_LIBC    cyg_iso_c_start();#else    (void)main(0, NULL);#endif} // cyg_package_start()char nextch(void){    return (raw[pos++]);}intmain( int argc, char *argv[] ){    int i, j, size;    CYG_ADDRESS entry;    char c;    diag_printf("FLASH here!\n");    while (identify_FLASH() == 0) {        diag_printf("... Please change FLASH jumper - hit C/R to continue:");        do {            hal_diag_read_char(&c);        } while ((c != '\r') && (c != '\n'));        diag_printf("\n");    } restart:    diag_printf("Ready file - hit C/R to continue:");    while (TRUE) {        hal_diag_read_char(&c);        if (c == '>') break;    }    i = 0;  j = 0;    while (1) {        hal_diag_read_char(&c);        if (c == '!') {            diag_printf("... Reset\n");            goto restart;        }        raw[i++] = c;        if (++j == SYNC_COUNT) {            hal_diag_write_char(c);            j = 0;        }        if (c == ':') break;    }    diag_printf("\n");    pos = 0;  len = i;    if (load_srecords(nextch, &entry, &size)) {        diag_printf("Read %x bytes, entry: %x\n", size, entry);        dump_buf(flash_buffer, 128);        diag_printf("\nData loaded - hit '!' to continue:");        while (TRUE) {            hal_diag_read_char(&c);            if (c == '!') break;        }        diag_printf("\n");        diag_printf("...Programming FLASH\n");        pos = 0;  i = 0;        while (pos < size) {            write_sector(i++, flash_buffer+pos);            pos += 256;        }    } else {        // Display buffer around failure                dump_buf(&raw[pos-32], 64);    }    diag_printf("All done!\n");    while (1) ;}// Adapted from ARM sample code#define SEQ_ADD1                0x5555#define SEQ_ADD2                0xAAAA#define START_CMD1              0xAA#define START_CMD2              0x55#define ID_CMD                  0x90#define PROG_CMD                0xA0#define STOP_CMD                0xF0#define MAN_ATMEL               0x1F#define ATMEL_AT29C040_ID       0X5B#define ATMEL_AT29C040A_ID      0XA4#define ATMEL_AT29C1024_ID      0X25#define ATMEL_SECTOR_SIZE       256#define ATMEL_MAX_SECTORS       2048int manuf_code, device_code, sector_size, max_no_of_sectors, word_mode;volatile char *FLASH = (volatile char *)0x04000000;intidentify_FLASH(void ){    // Enter Software Product Identification Mode    FLASH[SEQ_ADD1] = START_CMD1;    FLASH[SEQ_ADD2] = START_CMD2;    FLASH[SEQ_ADD1] = ID_CMD;    // Wait at least 10ms    cyg_thread_delay(2);    // Read Manufacturer and device code from the device    manuf_code = FLASH[0];    device_code = FLASH[1];    diag_printf("manuf: %x, device: %x\n", manuf_code, device_code);    // Exit Software Product Identification Mode    FLASH[SEQ_ADD1] = START_CMD1;    FLASH[SEQ_ADD2] = START_CMD2;    FLASH[SEQ_ADD1] = STOP_CMD;    // Wait at least 10ms    cyg_thread_delay(5);    if (manuf_code != MAN_ATMEL) {        diag_printf ( "Error: Wrong Manufaturer: %02x\n",manuf_code );        return (0);    }    switch (device_code) {    case  ATMEL_AT29C040A_ID:        diag_printf ("AT29C040A recognised\n");        sector_size = ATMEL_SECTOR_SIZE;        max_no_of_sectors = ATMEL_MAX_SECTORS;        word_mode = FALSE;        break;    case  ATMEL_AT29C1024_ID:        diag_printf ("AT29C1024 recognised\n");        sector_size = ATMEL_SECTOR_SIZE;        max_no_of_sectors = ATMEL_MAX_SECTORS;        word_mode = TRUE;        break;    default :        diag_printf ( "Error: Unsupported device: %02x\n", device_code);        return (0);    }    return (1);}voidwrite_sector(int num, char *buf){    int i, cnt;    volatile char *wrt = (volatile int *)&FLASH[num*sector_size];//    diag_printf("Writing to %08x\n", wrt);    // Enter Program Mode    FLASH[SEQ_ADD1] = START_CMD1;    FLASH[SEQ_ADD2] = START_CMD2;    FLASH[SEQ_ADD1] = PROG_CMD;    // Note: write bytes as longs regardless of bus width    for (i = 0;  i < sector_size;  i++) {        wrt[i] = buf[i];    }    // Wait for sector to program    cnt = 0;    i = sector_size - 1;    while (wrt[i] != buf[i]) {        if (cnt++ > 0x01000000) break;    }//    diag_printf("Out - i: %d, wrt[i] = %08X.%08X, buf[i] = %08X, count = %x\n", i, &wrt[i], wrt[i], buf[i], cnt);    // Verify    for (i = 0;  i < sector_size;  i++) {        for (cnt = 0;  cnt < 10;  cnt++) {            if (*wrt == *buf) break;            cyg_thread_delay(1);        }        if (cnt == 10) {            diag_printf("Can't program at 0x%08X: %02X not %02X\n", wrt, *wrt, *buf);        }        wrt++;  buf++;    }}// S-record download code - viciously 'adapted' from "kernel/src/sload/sload.c"/*---------------------------------------------------------------------------*//*////      An srecord looks like this://// byte count-+     address// start ---+ |        |       data        +- checksum//          | |        |                   |//        S01000006F6B692D746573742E73726563E4//        S315000448600000000000000000FC00005900000000E9//        S31A0004000023C1400037DE00F023604000377B009020825000348D//        S30B0004485A0000000000004E//        S70500040000F6////      S<type><length><address><data><checksum>////      Where //      - length (2 characters)//        is the number of bytes following upto the checksum. Note that//        this is not the number of chars following, since it takes two//        chars to represent a byte.//      - type (2 characters)//        is one of://        0) header record//        1) two byte address data record//        2) three byte address data record//        3) four byte address data record//        5) record containing the number of S1, S2, or S3 records//        7) four byte address termination record//        8) three byte address termination record//        9) two byte address termination record//       //      - address (4, 6, or 8 characters)//        is the start address of the data following, or in the case of//        a termination record, the start address of the image//      - data (0-2n characters)//        is the data.//      - checksum (2 characters)//        is the sum of all the raw byte data in the record, from the length//        upwards, modulo 256 and subtracted from 255.//// Useful S-records for testing purposes://   Start record://      S00B0000737461303030447563//   This sets the default address to be 0x02005000://      S31A020050002700801481C4E0B0A15000000100000091D02000018F//      S31A0200501500000001000000010000002700801881C4E2E4A150C1//      S311020080A42407070A090B0A0781050000E1//   Termination record://      S70502005000A8//*/#define S0      0#define S1      1#define S2      2#define S3      3#define S5      5#define S7      7#define S8      8#define S9      9/*---------------------------------------------------------------------------*/int hex2digit(char c){    if( c & 0x40 ) c += 9;;    return c &0x0f;    //    return ( c <= '9' ? c - '0' ://             c <= 'Z' ? c - 'A' + 10 ://             c - 'a' + 10);}/*---------------------------------------------------------------------------*/bool load_srecords(char (*readc)(),                    CYG_ADDRESS *start,                   int *size){    CYG_ADDRESS addr, load_addr;    int addrsize;    int length;    int i;    cyg_uint8 chksum, ochksum;    cyg_uint8 val;    cyg_uint8 *tdata;        char s;    char type;    char len0;    char len1;    bool first = true;        do {        // Skip whitespace characters until we find something that        // might be an 'S'.        do {            s = readc();        } while( s == '\r' || s == '\n' || s == ' ');        // Check that this is an S record        if( s != 'S' ) {            diag_printf("Invalid 'S' record\n");            return false;        }        // First 4 bytes are standard S + type + len        type = readc();        len0 = readc();        len1 = readc();                // decode the type        type = hex2digit(type);        // determine address size        switch (type) {        case S0:                        // start records have no address            addrsize = 0;            break;        case S1:                        // two byte address        case S9:            addrsize = 4;            break;        case S2:                        // 3 byte address        case S8:            addrsize = 6;            break;           case S3:                        // 4 byte address        case S7:            addrsize = 8;            break;        }        length  = hex2digit (len0) << 4;        length |= hex2digit (len1);        chksum = length;        // read the address        addr = 0;        for (i = 0; i < addrsize; i++) {            val = hex2digit(readc());            addr = (addr << 4) | val;        }        // calculate the checksum, which is done by the byte, not the digit        for (i = 0; i < addrsize*4; i += 8) {            chksum += ((addr >>  i) & 0xff);        }        // decide where to load this data        if (first && (type != S0)) {            load_addr = addr;            first = false;        }        // read the data and put it directly into memory where it belongs        tdata = (cyg_uint8 *)((addr - load_addr) + flash_buffer);        if (type < S7) {            *size = (addr - load_addr);        }        val = 0;        for (i = 0; i < ((length - 1) * 2) - addrsize; i += 2 ) {            val  = hex2digit (readc()) << 4;            val |= hex2digit (readc());            chksum += val;            if( type != S0 ) *tdata++ = val;            if (type < S7) *size = *size + 1;        }        // now get the old checksum        ochksum = hex2digit(readc()) << 4;        ochksum |= hex2digit(readc());        chksum = ~chksum;        if (chksum != ochksum) {            diag_printf("Bad checksum - addr: %x\n", addr);            return false;        }            } while( type < S7 );    *start = addr;    return true;}

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -