使用单片机对STC8G,8H,8A进行ISP下载程序

通过第三方程序实现对STC单片机的程序下载,可以方便进行现场的调试和更新。特别是对于设计远程程序更新、无线程序下载与调试等功能有帮助。

本文给出了下载相关的一些程序设计。

STC单片机ISP下载协议

在手边并没有成文的对STC单片机ISP下载协议,仅仅在STC官方网站的STC8G,8H,8A单片机手册中给出了“使用第三方MCU对STC8G,8H, 8A”系列的但潘辰进行ISP下载的范例程序。参考这些程序,可以完成对STC8G系列的单片机内部的FLASH程序的更新。

▲ STC8G技术手册中的下载协议

▲ STC8G技术手册中的下载协议

对于STC单片机内部的“硬件选项”,手册建议还是使用官方的原来的STC-ISP下载程序完成配置,这比较可靠。

▲ STC单片机的硬件选项

▲ STC单片机的硬件选项

STC的ISP下载程序

下面给出了使用STC8G1K17单片机中,C语言实现的ISP程序。

下面给出单片机借助的外部资源以及相应的源码。

1. UART2

程序使用了STC8G1k的UART2完成对下载MCU发送相关的指令。需要注意的是,在ISP通信过程UART2需要设置为9bit的工作模式,这在开始的我并没有注意,造成前期的调试受阻。

为了避免串口线对被调试单片机反向供电,特别是在外部单片机掉电进行冷启动的过程中,由于串口的电压可能仍然能够满足被下载单片机的工作电源需要,通过对UART2对应的输出管脚置为高阻状态来避免反向供电。这一点,并没有像STC8G1K08手册中推荐的使用电阻和二极管来进行电源个的的方案。

▲ STC单片机数据手册中推荐的串口电源隔离方案

▲ STC单片机数据手册中推荐的串口电源隔离方案

2. CCP2

使用了STC8G1K08的CCP2产生PWM波形来整流出高压( 2 V C ≈ 2 × 5 V ≈ 10 V 2V_C \approx 2 \times 5V \approx 10V 2VC2×5V10V)来驱动MOS管。详细的方案可以参见博文如何利用单片机IO口产生两倍的电源电压中的描述。

在对被下载单片机电源控制的同时,也需要改变UART2的RXD,TXD管脚的输出状态。在关闭电源的时候,将RXD,TXD设置为INPUT的高阻模式,可以避免对ISP单片机端口供电。下面代码显示了这里的控制。

//------------------------------------------------------------------------------
void ispPowerOn(void) {
    P10V_ON;
    PM_BIDIR(RXD2_PIN);
    PM_PP(TXD2_PIN);
}

void ispPowerOff(void) {
    PM_INPUT(RXD2_PIN);
    PM_INPUT(TXD2_PIN);
    P10V_OFF;    
}
3. 完整的源码

程序的其它部分可以参见数据手册中的范例程序进行阅读。由于没有正式的问题,在这里也就不再对下载的协议反向解释了。

程序的头文件。

/*
**==============================================================================
** STCISP.H:            -- by Dr. ZhuoQing, 2020-04-30
**
**  Description:
**
**==============================================================================
*/
#ifndef __STCISP__
#define __STCISP__
//------------------------------------------------------------------------------
#ifdef STCISP_GLOBALS
   #define STCISP_EXT
#else
   #define STCISP_EXT extern
#endif // STCISP_GLOBALS
//------------------------------------------------------------------------------
//==============================================================================

#define PWM_PIN                 3, 7
#define P5V_PIN                 3, 6                                                                

#define RXD2_PIN                1, 0
#define TXD2_PIN                1, 1
                                                               
                               
//------------------------------------------------------------------------------
#define P10V_ON                 ON(P5V_PIN), PWM3SetPWM(0x7f)        		
#define P10V_OFF                OFF(P5V_PIN), PWM3SetPWM(0xff)

//------------------------------------------------------------------------------

void ispInit(unsigned long lnBaud);

void ispPowerOn(void);
void ispPowerOff(void);

#define ISP_POWEROFF_TIME      200         // unit :ms

STCISP_EXT unsigned int g_nIspPowerOffTime;

void ispSetBaud(unsigned long lnBaud, unsigned char uc9BitFlag);
                                            // uc9BitFlag: 1 : Set UART2 as 9bit mode
                                            //             0 : as 8 bit mode

//------------------------------------------------------------------------------
STCISP_EXT unsigned int g_nispReceiveSum;
STCISP_EXT unsigned char g_ucispReceiveIndex, g_ucispReceiveCount, g_ucispReceiveStep;
STCISP_EXT unsigned char xdata g_ucispRxBuffer[64], g_ucispTxBuffer[150];
STCISP_EXT unsigned char g_ucispReceiveFlag;

//------------------------------------------------------------------------------
STCISP_EXT unsigned int g_nispProgramStart;
STCISP_EXT unsigned char g_ucispProgramLength;

#define ISP_PROGRAM_BUFFER          256
STCISP_EXT unsigned char xdata g_ucispProgramBuffer[ISP_PROGRAM_BUFFER];

//------------------------------------------------------------------------------

void ispProcChar(void);
void ispCommunicationInit(void);
unsigned char ispSendChar(unsigned char ucChar);
void ispCommandSend(unsigned char ucSize);
void ispRun(unsigned long lnBaud, unsigned int nTime);


//------------------------------------------------------------------------------
void ispShowRxBuffer(void);

//------------------------------------------------------------------------------
#define FALSE                       1
#define TRUE                        0

typedef bit                         BOOL;
typedef unsigned char               BYTE;
typedef unsigned short              WORD;

#define LOBYTE(w)                   ((BYTE)(WORD)(w))
#define HIBYTE(w)                   ((BYTE)((WORD)(w) >> 8))


#define FUSER                       35000000L
#define RL(n)                       (0x10000 - FUSER/4/(n))

#define ISP_DOWNLOAD_CHECKLOOP      100
#define ISP_DOWNLOAD_BAUD           115200L
#define ISP_DOWNLOAD_TIMEOUT        2000
#define ISP_DOWNLOAD_TIMEOUT_ERASE  2000
#define ISP_DOWNLOAD_PAGESIZE       128


//------------------------------------------------------------------------------
#define ISP_ARG7_8H                 0x97
#define ISP_ARG7_8G                 0x97
#define ISP_ARG7_8A                 0x81

unsigned char ispEnterISP(unsigned long lnStartBaud, unsigned long lnBaud, 
                          unsigned char ucArg7, unsigned long lnFosc);
                          
unsigned char ispEraseMCU(void);
unsigned char ispDownloadCode(unsigned char * pucCode, unsigned int nSize, 
                              unsigned int nStartAddress);

//------------------------------------------------------------------------------





//==============================================================================
//             END OF THE FILE : STCISP.H
//------------------------------------------------------------------------------
#endif // __STCISP__

程序的C语言文件。

/*
**==============================================================================
** STCISP.C:             -- by Dr. ZhuoQing, 2020-04-30
**
**==============================================================================
*/

//------------------------------------------------------------------------------
#define STCISP_GLOBALS        1              // Define the global variables
#include "STCISP.H"
#include "STC8G.H"
#include "C51BASIC.H"
#include <STDIO.H>

//------------------------------------------------------------------------------

void ispInit(unsigned long lnBaud) {
    //--------------------------------------------------------------------------    
    ispSetBaud(lnBaud, 0);
    
    //--------------------------------------------------------------------------
    ispCommunicationInit();

    //--------------------------------------------------------------------------
    
    PM_PP(PWM_PIN);
    PM_PP(P5V_PIN);

    P10V_ON;  
    
    g_nIspPowerOffTime = ISP_POWEROFF_TIME;
    
}

//------------------------------------------------------------------------------
void ispRun(unsigned long lnBaud, unsigned int nTime) {
    ispPowerOff();
    WaitTime(nTime);
    ispPowerOn();
    
    ispSetBaud(lnBaud, 0);
    UART2_CLEAR;
}


//------------------------------------------------------------------------------
void ispPowerOn(void) {
    P10V_ON;
    PM_BIDIR(RXD2_PIN);
    PM_PP(TXD2_PIN);
}

void ispPowerOff(void) {
    PM_INPUT(RXD2_PIN);
    PM_INPUT(TXD2_PIN);
    P10V_OFF;    
}

void ispSetBaud(unsigned long lnBaud, unsigned char uc9BitFlag) {
    unsigned int nNumber;
    nNumber = Baud2TimeReload(lnBaud, 35000000L);

    if(uc9BitFlag)    
        S2CON |= 0x80;                          // Enable bit9 communication
    else S2CON &= ~0x80;
    
    
    T2H = nNumber >> 8;
    T2L = nNumber;
}

void ispCommunicationInit(void) {
    unsigned int i;
    
    for(i = 0; i < sizeof(g_ucispRxBuffer); i ++)
        g_ucispRxBuffer[i] = 0;
        
    g_nispReceiveSum = 0;
    g_ucispReceiveIndex = g_ucispReceiveCount = g_ucispReceiveStep = 0;
    g_ucispReceiveFlag = 0;
    
}


//------------------------------------------------------------------------------
void ispProcChar(void) {
    unsigned char ucChar;
    
    if(!UART2_CANRECE) return;    
    UART2ReceChar(&ucChar);
    
    //--------------------------------------------------------------------------
//    printf("%bx ", ucChar);
    
    //------------------------------------------------------------------------        
    switch(g_ucispReceiveStep) {
        case 1:
            if(ucChar != 0xb9) goto L_CheckFirst;
            g_ucispReceiveStep ++;
            break;
           
        case 2:
            if(ucChar != 0x68) goto L_CheckFirst;
            g_ucispReceiveStep ++;
            break;
        
        case 3:
            if(ucChar != 0x0) goto L_CheckFirst;
            g_ucispReceiveStep ++;
            break;
            
        case 4:
            g_nispReceiveSum = 0x68 + ucChar;
            g_ucispReceiveCount = ucChar - 6;
            g_ucispReceiveIndex = 0;
            g_ucispReceiveStep ++;
            break;
            
        case 5:
            g_nispReceiveSum += ucChar;
            g_ucispRxBuffer[g_ucispReceiveIndex ++] = ucChar;
            if(g_ucispReceiveIndex == g_ucispReceiveCount)
                g_ucispReceiveStep ++;
            break;
            
        case 6:
            if(ucChar != (unsigned char)(g_nispReceiveSum >> 8))
                goto L_CheckFirst;
            g_ucispReceiveStep ++;
            break;
        
        case 7:
            if(ucChar != (unsigned char)(g_nispReceiveSum & 0xff))
                goto L_CheckFirst;
            g_ucispReceiveStep ++;
            break;
            
        case 8:
            if(ucChar != 0x16) goto L_CheckFirst;
            g_ucispReceiveFlag = 1;
            g_ucispReceiveStep ++;
            break;
            
L_CheckFirst:
        case 0:
        default:
            ispCommunicationInit();
            if(ucChar == 0x46) g_ucispReceiveStep = 1;
            else g_ucispReceiveStep = 0;
            
            break;
    }
}

//------------------------------------------------------------------------------
/*
void ispShowRxBuffer(void) {
    unsigned int i;
    
    printf("Count:%bd\r\n", g_ucispReceiveCount);
    for(i = 0; i < g_ucispReceiveCount; i ++) 
        printf("%bx ", g_ucispRxBuffer[i]);
    
    printf("\r\n");
}
*/

//------------------------------------------------------------------------------
unsigned char ispSendChar(unsigned char ucChar) {
    g_ucUART2IntFlag = 0;
    
    ACC = ucChar;
    
    S2CON &= ~0x8;
    if(P) S2CON |= 0x8;
    
    S2BUF = ACC;
    
    while(1) {
        if(g_ucUART2IntFlag) break;
    }
    
    return ucChar;    
}


//------------------------------------------------------------------------------
void ispCommandSend(unsigned char ucSize) {
    unsigned int nSum;
    unsigned char i, c;
        
    ispSendChar(0x46);
    ispSendChar(0xb9);
    ispSendChar(0x6a);
    ispSendChar(0x00);
    
    nSum = ucSize + 6 + 0x6a;
    ispSendChar(ucSize + 6);
    
    for(i = 0; i < ucSize; i ++) {
        c = g_ucispTxBuffer[i];
        nSum += c;
        ispSendChar(c);
    }
    
    ispSendChar((unsigned char)(nSum >> 8));
    ispSendChar((unsigned char)(nSum & 0xff));
    ispSendChar(0x16);
    
    
    ispCommunicationInit();
    
}

//------------------------------------------------------------------------------
unsigned char ispEnterISP(unsigned long lnStartBaud, unsigned long lnBaud, 
                          unsigned char ucArg7, unsigned long lnFosc) {
    unsigned char ucArg;
    unsigned int nCheckLoop;
    unsigned int nMS10;
    unsigned int nBaudTimeReload;
    
    
    //----------------------------------------------------------------------            
    ispCommunicationInit();
    ispSetBaud(lnStartBaud, 1);
    
//    printf("Set baud:%ld\r\n", lnStartBaud);
    
    //--------------------------------------------------------------------------
    ispPowerOff();
    WaitTime(g_nIspPowerOffTime);
    ispPowerOn();
    
    
    nMS10 = (unsigned int)(GetClickMS() + 10);
    nCheckLoop = 0;
    
    for(nCheckLoop = 0; nCheckLoop < ISP_DOWNLOAD_CHECKLOOP;) {
        ispProcChar();        
        if(g_ucispReceiveStep == 0) {
            if(GetClickMS() == nMS10) {
                ispSendChar(0x7f);
                nMS10 = (unsigned int)(GetClickMS() + 10);
                nCheckLoop ++;
            }           
        }
        
        if(g_ucispReceiveFlag) {
            ucArg = g_ucispRxBuffer[4];
            if(g_ucispRxBuffer[0] == 0x50)
                break;
            
            return 1;                       // return error 
        }
        

    }
    
    
    if(nCheckLoop >= ISP_DOWNLOAD_CHECKLOOP) return 2;
    
    //--------------------------------------------------------------------------
    // Set new baud
//    printf("\r\nSet new Buad:%ld/%d\r\n", lnBaud, nCheckLoop);
    
    nBaudTimeReload = Baud2TimeReload(lnBaud, lnFosc);
//    printf("%x\r\n", nBaudTimeReload);
    
    
    ucArg = g_ucispRxBuffer[4];
    g_ucispTxBuffer[0] = 0x01;
    g_ucispTxBuffer[1] = ucArg;
    g_ucispTxBuffer[2] = 0x40;
    g_ucispTxBuffer[3] = HIBYTE(nBaudTimeReload);
    g_ucispTxBuffer[4] = LOBYTE(nBaudTimeReload);
    g_ucispTxBuffer[5] = 0x00;
    g_ucispTxBuffer[6] = 0x00;
    g_ucispTxBuffer[7] = ucArg7;//0x97;//0x81;//0x97;                // 0x81: STC8A
                                                            // 0x97: STC8G,8H

    nMS10 = (unsigned int)(GetClickMS() + ISP_DOWNLOAD_TIMEOUT);
    ispCommandSend(8);
    
    for(;;) {
        ispProcChar();
        if(g_ucispReceiveFlag) {
            if(g_ucispRxBuffer[0] == 0x01) break;
            return 3;
        }
        
        if(GetClickMS() == nMS10) return 4;
        
    }
    

    //----------------------------------------------------------------------
    // Prepare the new download baud    
    ispSetBaud(lnBaud, 1);
    
//    WaitTime(10);
    
    g_ucispTxBuffer[0] = 0x05;
    g_ucispTxBuffer[1] = 0x00;
    g_ucispTxBuffer[2] = 0x00;
    g_ucispTxBuffer[3] = 0x5a;
    g_ucispTxBuffer[4] = 0xa5;
        
    nMS10 = (unsigned int)(GetClickMS() + ISP_DOWNLOAD_TIMEOUT);
    ispCommandSend(5);
    for(;;) {
        ispProcChar();
        
        if(g_ucispReceiveFlag) {
            if(g_ucispRxBuffer[0] == 0x05) break;
            return 5;
        }
        
        if(GetClickMS() == nMS10) return 6;

    }
    

    return 0;

}

//------------------------------------------------------------------------------
unsigned char ispEraseMCU(void) {
    unsigned int nMS10;
    
    //----------------------------------------------------------------------    
    // Erase the IC
    WaitTime(10);
    
    g_ucispTxBuffer[0] = 0x03;
    g_ucispTxBuffer[1] = 0x00;
    g_ucispTxBuffer[2] = 0x00;
    g_ucispTxBuffer[3] = 0x5a;
    g_ucispTxBuffer[4] = 0xa5;
    
    
    nMS10 = (unsigned int)(GetClickMS() + ISP_DOWNLOAD_TIMEOUT_ERASE);
    ispCommandSend(5);
    
    for(;;) {
        ispProcChar();
        
        if(g_ucispReceiveFlag) {
            if(g_ucispRxBuffer[0] == 0x03) break;
            return 1;
        }

        if(GetClickMS() == nMS10) return 2;
    }
    
    
    return 0;                               // OK

}

//------------------------------------------------------------------------------s
unsigned char ispDownloadCode(unsigned char * pucCode, unsigned int nSize, unsigned int nStartAddress) {
    unsigned int i, nStep, nLength, j;
    unsigned int nDownloadAddress, nStart, nEnd, nOffset;
    unsigned int nPoint;
    unsigned int nMS10;
    
    
    nStep = (int)((nSize + ISP_DOWNLOAD_PAGESIZE - 1) / ISP_DOWNLOAD_PAGESIZE);
    
    g_ucispTxBuffer[0] = 0x22;
    g_ucispTxBuffer[3] = 0x5a;
    g_ucispTxBuffer[4] = 0xa5;
    
    nOffset = 5;
    nPoint = 0;
    
    for(i = 0; i < nStep; i ++) {
        nStart = i * ISP_DOWNLOAD_PAGESIZE;
        nEnd = nStart + ISP_DOWNLOAD_PAGESIZE;
        if(nEnd > nSize) nEnd = nSize;
        nLength = nEnd - nStart;
        
        nDownloadAddress = nStartAddress + nStart;
        g_ucispTxBuffer[1] = HIBYTE(nDownloadAddress);
        g_ucispTxBuffer[2] = LOBYTE(nDownloadAddress);
        
        for(j = 0; j < nLength; j ++) {
            g_ucispTxBuffer[nOffset + j] = *(pucCode + nPoint);
            nPoint ++;        
        }
        
        nMS10 = (unsigned int)(GetClickMS() + ISP_DOWNLOAD_TIMEOUT);
        
        ispCommandSend(nLength + nOffset);
        
        
        for(;;) {
            ispProcChar();
            
            if(g_ucispReceiveFlag) {
                if(g_ucispRxBuffer[0] == 0x2 &&
                   g_ucispRxBuffer[1] == 'T') break;
                   
                return 1;                   
            }
            if(GetClickMS() == nMS10) return 2;
        }
        
        g_ucispTxBuffer[0] = 0x02;
        
    }
    
    return 0;

}


//==============================================================================
//                END OF THE FILE : STCISP.C
//------------------------------------------------------------------------------
/*---------------------------------------------------------------------*/ /* --- STC MCU Limited ------------------------------------------------*/ /* --- 使用主芯片对从芯片(限STC15系列)进行ISP下载举例 -----------------*/ /* --- Mobile: (86)13922805190 ----------------------------------------*/ /* --- Fax: 86-755-82905966 -------------------------------------------*/ /* --- Tel: 86-755-82948412 -------------------------------------------*/ /* --- Web: www.STCMCU.com --------------------------------------------*/ /* 如果要在程序使用此代码,请在程序中注明使用了宏晶科技的资料及程序 */ /* 如果要在文章中应用此代码,请在文章中注明使用了宏晶科技的资料及程序 */ /*---------------------------------------------------------------------*/ //本示例在Keil开发环境下请选择Intel的8058芯片型号进行编译 //假定测试芯片的工作频率为11.0592MHz //注意:使用本代码对STC15系列的单片机进行下载时,必须要执行了Download代码之后, //才能给目标芯片上电,否则目标芯片将无法正确下载 #include "reg51.h" typedef bit BOOL; typedef unsigned char BYTE; typedef unsigned short WORD; typedef unsigned long DWORD; //宏、常量定义 #define FALSE 0 #define TRUE 1 #define LOBYTE(w) ((BYTE)(WORD)(w)) #define HIBYTE(w) ((BYTE)((WORD)(w) >> 8)) #define MINBAUD 2400L #define MAXBAUD 115200L #define FOSC 11059200L //主控芯片工作频率 #define BR(n) (65536 - FOSC/4/(n)) //主控芯片串口波特率计算公式 #define T1MS (65536 - FOSC/1000) //主控芯片1ms定时初值 #define FUSER 24000000L //15系列目标芯片工作频率 #define RL(n) (65536 - FUSER/4/(n)) //15系列目标芯片串口波特率计算公式 //SFR定义 sfr AUXR = 0x8e; //变量定义 BOOL f1ms; //1ms标志位 BOOL UartBusy; //串口发送忙标志位 BOOL UartReceived; //串口数据接收完成标志位 BYTE UartRecvStep; //串口数据接收控制 BYTE TimeOut; //串口通讯超时计数器 BYTE xdata TxBuffer[256]; //串口数据发送缓冲区 BYTE xdata RxBuffer[256]; //串口数据接收缓冲区 char code DEMO[256]; //演示代码数据 //函数声明 void Initial(void); void DelayXms(WORD x); BYTE UartSend(BYTE dat); void CommInit(void); void CommSend(BYTE size); BOOL Download(BYTE *pdat, long size); //主函数入口 void main(void) { while (1) { Initial(); if (Download(DEMO, 0x0100)) { //下载成功 P3 = 0xff; DelayXms(500); P3 = 0x00; DelayXms(500); P3 = 0xff; DelayXms(500); P3 = 0x00; DelayXms(500); P3 = 0xff; DelayXms(500); P3 = 0x00; DelayXms(500); P3 = 0xff; } else { //下载失败 P3 = 0xff; DelayXms(500); P3 = 0xf3; DelayXms(500); P3 = 0xff; DelayXms(500); P3 = 0xf3; DelayXms(500); P3 = 0xff; DelayXms(500); P3 = 0xf3; DelayXms(500); P3 = 0xff; } } } //1ms定时器中断服务程序 void tm0(void) interrupt 1 using 1 { static BYTE Counter100; f1ms = TRUE; if (Counter100-- == 0) { Counter100 = 100; if (TimeOut) TimeOut--; } } //串口中断服务程序 void uart(void) interrupt 4 using 1 { static WORD RecvSum; static BYTE RecvIndex; static BYTE RecvCount; BYTE dat; if (TI) { TI = 0; UartBusy = FALSE; } if (RI) { RI = 0; dat = SBUF; switch (UartRecvStep) { case 1: if (dat != 0xb9) goto L_CheckFirst; UartRecvStep++; break; case 2: if (dat != 0x68) goto L_CheckFirst; UartRecvStep++; break; case 3: if (dat != 0x00) goto L_CheckFirst; UartRecvStep++; break; case 4: RecvSum = 0x68 + dat; RecvCount = dat - 6; RecvIndex = 0; UartRecvStep++; break; case 5: RecvSum += dat; RxBuffer[RecvIndex++] = dat; if (RecvIndex == RecvCount) UartRecvStep++; break; case 6: if (dat != HIBYTE(RecvSum)) goto L_CheckFirst; UartRecvStep++; break; case 7: if (dat != LOBYTE(RecvSum)) goto L_CheckFirst; UartRecvStep++; break; case 8: if (dat != 0x16) goto L_CheckFirst; UartReceived = TRUE; UartRecvStep++; break; L_CheckFirst: case 0: default: CommInit(); UartRecvStep = (dat == 0x46 ? 1 : 0); break; } } } //系统初始化 void Initial(void) { UartBusy = FALSE; SCON = 0xd0; //串口数据模式必须为8位数据+1位偶检验 AUXR = 0xc0; TMOD = 0x00; TH0 = HIBYTE(T1MS); TL0 = LOBYTE(T1MS); TR0 = 1; TH1 = HIBYTE(BR(MINBAUD)); TL1 = LOBYTE(BR(MINBAUD)); TR1 = 1; ET0 = 1; ES = 1; EA = 1; } //Xms延时程序 void DelayXms(WORD x) { do { f1ms = FALSE; while (!f1ms); } while (x--); } //串口数据发送程序 BYTE UartSend(BYTE dat) { while (UartBusy); UartBusy = TRUE; ACC = dat; TB8 = P; SBUF = ACC; return dat; } //串口通讯初始化 void CommInit(void) { UartRecvStep = 0; TimeOut = 20; UartReceived = FALSE; } //发送串口通讯数据包 void CommSend(BYTE size) { WORD sum; BYTE i; UartSend(0x46); UartSend(0xb9); UartSend(0x6a); UartSend(0x00); sum = size + 6 + 0x6a; UartSend(size + 6); for (i=0; i<size; i++) { sum += UartSend(TxBuffer[i]); } UartSend(HIBYTE(sum)); UartSend(LOBYTE(sum)); UartSend(0x16); while (UartBusy); CommInit(); } //对STC15系列的芯片进行数据下载程序 BOOL Download(BYTE *pdat, long size) { BYTE arg; BYTE cnt; WORD addr; //握手 CommInit(); while (1) { if (UartRecvStep == 0) { UartSend(0x7f); DelayXms(10); } if (UartReceived) { arg = RxBuffer[4]; if (RxBuffer[0] == 0x50) break; return FALSE; } } //设置参数(设置从芯片使用最高的波特率以及擦除等待时间等参数) TxBuffer[0] = 0x01; TxBuffer[1] = arg; TxBuffer[2] = 0x40; TxBuffer[3] = HIBYTE(RL(MAXBAUD)); TxBuffer[4] = LOBYTE(RL(MAXBAUD)); TxBuffer[5] = 0x00; TxBuffer[6] = 0x00; TxBuffer[7] = 0xc3; CommSend(8); while (1) { if (TimeOut == 0) return FALSE; if (UartReceived) { if (RxBuffer[0] == 0x01) break; return FALSE; } } //准备 TH1 = HIBYTE(BR(MAXBAUD)); TL1 = LOBYTE(BR(MAXBAUD)); DelayXms(10); TxBuffer[0] = 0x05; CommSend(1); while (1) { if (TimeOut == 0) return FALSE; if (UartReceived) { if (RxBuffer[0] == 0x05) break; return FALSE; } } //擦除 DelayXms(10); TxBuffer[0] = 0x03; TxBuffer[1] = 0x00; CommSend(2); TimeOut = 100; while (1) { if (TimeOut == 0) return FALSE; if (UartReceived) { if (RxBuffer[0] == 0x03) break; return FALSE; } } //写用户代码 DelayXms(10); addr = 0; TxBuffer[0] = 0x22; while (addr < size) { TxBuffer[1] = HIBYTE(addr); TxBuffer[2] = LOBYTE(addr); cnt = 0; while (addr = 128) break; } CommSend(cnt + 3); while (1) { if (TimeOut == 0) return FALSE; if (UartReceived) { if ((RxBuffer[0] == 0x02) && (RxBuffer[1] == 'T')) break; return FALSE; } } TxBuffer[0] = 0x02; } ////写硬件选项 ////如果不需要修改硬件选项,此步骤可直接跳过,此时所有的硬件选项 ////都维持不变,MCU的频率为上一次所调节频率 ////若写硬件选项,MCU的内部IRC频率将被固定写为24M, ////建议:第一次使用STC-ISP下载软件将从芯片的硬件选项设置好 //// 以后再使用主芯片对从芯片下载程序时不写硬件选项 //DelayXms(10); //for (cnt=0; cnt<128; cnt++) //{ // TxBuffer[cnt] = 0xff; //} //TxBuffer[0] = 0x04; //TxBuffer[1] = 0x00; //TxBuffer[2] = 0x00; //TxBuffer[34] = 0xfd; //TxBuffer[62] = arg; //TxBuffer[63] = 0x7f; //TxBuffer[64] = 0xf7; //TxBuffer[65] = 0x7b; //TxBuffer[66] = 0x1f; //CommSend(67); //while (1) //{ // if (TimeOut == 0) return FALSE; // if (UartReceived) // { // if ((RxBuffer[0] == 0x04) && (RxBuffer[1] == 'T')) break; // return FALSE; // } //} //下载完成 return TRUE; } char code DEMO[256] = { 0x02,0x00,0x5E,0x12,0x00,0x4B,0x75,0xB0, 0xEF,0x12,0x00,0x2C,0x75,0xB0,0xDF,0x12, 0x00,0x2C,0x75,0xB0,0xFE,0x12,0x00,0x2C, 0x75,0xB0,0xFD,0x12,0x00,0x2C,0x75,0xB0, 0xFB,0x12,0x00,0x2C,0x75,0xB0,0xF7,0x12, 0x00,0x2C,0x80,0xDA,0xE4,0xFF,0xFE,0xE4, 0xFD,0xFC,0x0D,0xBD,0x00,0x01,0x0C,0xBC, 0x01,0xF8,0xBD,0xF4,0xF5,0x0F,0xBF,0x00, 0x01,0x0E,0xBE,0x03,0xEA,0xBF,0xE8,0xE7, 0x02,0x00,0x4B,0x75,0x80,0xFF,0x75,0x90, 0xFF,0x75,0xA0,0xFF,0x75,0xB0,0xFF,0x75, 0xC0,0xFF,0x75,0xC8,0xFF,0x22,0x78,0x7F, 0xE4,0xF6,0xD8,0xFD,0x75,0x81,0x07,0x02, 0x00,0x03,0x00,0x00,0x00,0x00,0x00,0x00, 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00, 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00, 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00, 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00, 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00, 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00, 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00, 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00, 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00, 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00, 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00, 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00, 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00, 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00, 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00, 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00, 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00, 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00, };
评论 7
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

卓晴

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值