摘  要:為解決特殊場合DSP程序升級困難的問題,以TMS320F28035為例,介紹了一種基於串口通信的適合於TMS320C2000系列DSP實現程序更新的在線升級方法。描述了該在線升級方法的基本思想和實現步驟,給出了關鍵部分的程序代碼。實驗證明,該方法簡單可靠,可用於嵌入式設備軟件程序的升級更新中。

關鍵詞: 在線升級; DSP;串口通信; Flash

    TMS320C2000系列DSP是美國德州儀器公司(簡稱TI)推出的集微控制器和高性能DSP特點於一身的DSP系列。該系列的DSP具有強大的控制信號處理能力[1],能夠實現複雜的控制算法。隨着電子技術的不斷髮展以及用户需求的不斷提升,可能需要經常對已經投入使用的嵌入式設備程序進行更新,而目前一般的程序升級方法是實地取下設備,露出JTAG端口後通過仿真器來更新程序[2-4]。這種方法雖然簡單有效,但對於某些特殊場合,會給程序升級帶來了極大的不便[2]。本文以TMS320F28035為例,描述了一種可以脱離JTAG仿真器,不改變DSP上電啓動方式,實現TMS320C2000系列DSP應用程序在線更新的方法。
1 在線升級的基本思想
    一般的基於DSP的軟件程序更新是在CCS環境下通過JTAG接口操作來實現的。基於JTAG接口的方法雖然易於操作,而且調試方便,但經常受空間以及傳輸距離的限制。例如一台DSP系統安裝在複雜、封閉的環境下,當程序需要更新或升級時,利用JTAG接口難以實現程序的在線升級[3]。而基於串口通信的在線升級技術是通過用底層程序燒寫應用程序的方法來達到程序升級的目的,該方法則不受複雜系統和複雜環境的限制。另外,在線升級方法不需要改變DSP的啓動方式,直接採用DSP默認的內部Flash方式啓動[5],從而省去了要對DSP的一些引腳進行硬件設置的麻煩。底層程序指已經固化在DSP指定Flash空間中的程序,不允許用户修改和擦除,主要用於實現在線升級的時機判斷、數據接收及代碼燒寫等功能,該程序中使用了Flash2803x_API庫存函數(詳見2.2節);應用程序即為用户的升級程序[3]。
    F28035 DSP每次上電覆位,先運行底層程序,與PC機建立聯繫,然後根據PC機的指令來判斷是否需要升級應用程序。若需要,則將通過串口發送來的應用程序代碼燒寫至F28035片內Flash指定扇區;否則將繼續執行原有的應用程序。當應用程序很大或DSP的RAM空間比較小時,可採用將應用程序代碼分批發給DSP,DSP接收並燒寫完一批代碼後,再進行下一批代碼的接收和燒寫工作,直到所有的應用程序代碼都燒寫完畢。
2 在線升級的具體實現
2.1 應用程序

    用户的應用程序經過CCS編譯連接生成具有模塊化格式的目標文件(.out),該文件中的代碼和數據分別存放在不同的段中,因而不能直接用來燒寫Flash,需將其轉換為Flash能識別的數據格式——二進制文件  (.bin)。本文采用hex2000.exe和FileOshell.exe工具來實現文件轉換。首先,應用程序經過編譯連接生成.out文件,然後通過hex2000.exe把.out文件轉換成.hex文件,再通過FileOshell.exe將文件轉換成.bin文件。先做一個批處理文件,內容如下:

Example_2803xAdcSoc.out
    -map Example_2803xAdcSoc.map
    -o Example_2803xAdcSoc.hex
    -m
    -memwidth 16
    -image
    ROMS
    {
       Flash28035:  origin = 0x3e8000, len= 0x1000, romwidth=
        16, fill=0xFFFF
    }


其中,Example_2803xAdcSoc.out 是應用程序經過CCS生成的文件;-map是生成map文件;-o是生成hex文件;-m是Motorola-S 格式;-memwidth 16指存儲器位數為16 bit;-image指選擇映像文件;ROMS 是所需要轉換的起始地址、長度、位數及填充。本文選擇從0x3e8000開始,長度是4 KB,即FlashH,FlashH中未用的部分用0xFFFF填充,本文把這個批處理文件命名為:Example_2803xAdcSoc.cmd。接下來要生成.bin文件,先做一個MS-DOS型批處理文件,其內容如下:     IFileIOShell.exe -i Example_2803xAdcSoc.hex -o Example_2803xAdcSoc.bin
    注意要把Example_2803xAdcSoc.out、hex2000.exe、FileIO
Shell.exe、Example_2803xAdcSoc.cmd和MS-DOS型批處理文件放在同一目錄下,然後雙擊MS-DOS型批處理文件,即生成所需要的Example_2803xAdcSoc.bin文件。
2.2 底層程序
    底層程序用於實現將串口發送的數據燒寫至Flash的指定部分,涉及到應用程序的正確定位和復位後的啓動過程,是實現整個在線升級的重點。底層程序流程圖如圖1所示。底層程序主要實現以下功能[3]:

    (1)上電覆位查詢功能。上電覆位後通過接收上位機發送的命令判斷是否升級。若上位機發送的是升級命令,則跳轉到底層程序中升級部分執行;否則,跳轉到原有的應用程序處執行。
    (2)搬移燒寫程序的功能。由於F28035片上Flash不支持在其中一個扇區運行程序去擦除或燒寫其他扇區,故完成接收數據和燒寫Flash工作的這部分程序(即底層程序中的升級部分程序)需搬移至片內RAM或片外RAM上運行。實現程序搬移的函數為:

void MemCopy (Uint16 *SourceAddr,  Uint16 *Source End
        Addr, Uint16 *DestAddr)
       {
      while (SourceAddr < SourceEndAddr)
          {
                *DestAddr++ = *SourceAddr++;
            }
             return ;
    }


其中,SourceAddr為Flash中升級程序的起始地址,SourceEndAddr為Flash中升級程序的結束地址;DestAddr為搬移至內存的首地址。
    (3)接收上位機發送的應用程序代碼並保存到DSP
指定的內存中(一般為RAM區)。這是通過串口RS232來實現的。並確定用於數據保存的這部分內存未被佔用。例如,若需要將應用程序代碼暫存到F28035的L0 SARAM區域(地址空間0x3F8000-0x3F8800)。定義數組Uint16 BlockBuffer[2048]用於存儲應用程序代碼,在底層程序中採用存儲器定位語句,將上面的緩衝數組定位到相應的存儲空間:
  #pragma DATA_SECTION(BlockBuffer,“BlockTransferbuffer”);
    在底層程序CMD文件中,採用定位語句,將BlockTransferbuffer定位到DSP的L0 SRAM空間:
    BlockTransferBuffer:> L0 SARAM  PAGE=2  
                          //地址空間:0x3F8000~0x3F8800
    通過以上底層程序的設置,可將應用程序緩存到指定的RAM區域中。
    (4)代碼接收結束後,將內存中的代碼燒寫至指定Flash扇區,該步驟通過調用Flash2803x_API庫函數完成。底層程序中所用到的Flash2803x_API庫函數如下[6]:
    ①擦除扇區的函數為Uint16 Flash28035_Erase(Uint16 SectorMask,&Fstatus),其中,SectorMask為即將被擦除的扇區;&Fstatus為執行擦除操作後返回的狀態值,用來判斷擦除操作是否成功。②將程序燒寫到Flash扇區的函數為Uintl6 Flash28035_Program(&FlashAddr, &BuffAddr,Length,&Fstatus),其中,&FlashAddr為即將被燒寫的Flash扇區的起始地址;&BuffAddr為即將準備燒寫的程序當前存放在內存空間的首地址;Length為程序長度;&Fstatus為執行燒寫操作後返回的狀態值,用來判斷燒寫操作是否成功。③校驗燒寫到Flash中的程序為Uint16 Flash28035_Verify(&FlashAddr,&BuffAddr,Length,&Fstatus),其中,&FlashAddr指定從Flash內開始比較的首地址;&BuffAddr為被比較文件的存儲首地址;Length是需要比較的16 bit字的個數,程序長度;&Fstatus是執行校驗操作後返回的狀態值,用來判斷校驗操作是否成功。
2.3 底層程序和應用程序的定位
    DSP F28035上電覆位後,CPU將從內部Boot Rom獲得復位向量。復位向量指向Boot Rom並執行其內部的Bootloader程序,執行完畢後確定從內部Flash啓動。程序指針跳轉到Flash的0x3F7FF6處。由於這個地址是固定的,因此底層程序必須燒寫在以這個地址為起始地址的空間內。DSP進入底層軟件程序中運行,首先通過接收上位機的命令來判斷是否進行在線升級,如果進行在線升級,則跳轉到相應升級程序中執行;否則,跳轉到原有的應用程序處執行。由底層程序跳轉到原有的應用程序處執行時,採用絕對地址跳轉。部分程序如下所示:

#define Jumpgxcx (void (*)(void))0x3E8FFE 
                            //定義應用程序的跳轉地址
     SCI_SendStatus(“upgrade program? (y/n):”)  
                                 //向上位機詢問是否升級
     temp = SCIA_GetByteData_app();   
                        //接收上位機發送來的是否升級命令
           if (temp==’y’)                  
           {
                 main2();   //如果升級,則跳轉到升級程序中執行
           }
           Else     
           {
                  (*Jumpgxcx)();   
    //如果不升級,則採用絕對地址跳轉到應用程序中執行
       }
}


    底層程序的cmd配置與應用程序的cmd配置要保持一致,不能產生地址衝突。同時,要注意底層程序和應用程序的跳轉地址配置。
    底層程序cmd文件的部分配置如下:

BEGIN : origin = 0x3F7FF6, length = 0x000002 
    RESET : origin = 0x3FFFC0, length = 0x000002 /*
    codestart : > BEGIN           PAGE = 0
    應用程序cmd文件的部分配置如下:
    BEGIN : origin = 0x3E8FFE, length = 0x000002 
    codestart : > BEGIN         PAGE = 0


3 燒寫步驟
    首先把底層程序通過JTAG接口燒寫到F28035中,然後再進行應用程序的燒寫。應用程序的燒寫步驟為:先把串口調試工具的參數配置為波特率9 600 bit/s、8 bit數據位、1 bit停止位、沒有奇偶校驗位;選擇發送文本文件方式,發送應用程序的.bin文件到DSP。由於F28035的RAM區比較小,可以採取把應用程序代碼分為多次發送的方式。燒寫過程如圖2所示。

 

 

    本文介紹了一種基於串口通信的DSP應用程序在線升級技術,可以在不打開機箱的條件下實現模塊軟件的更新升級。經過實驗發現,採用在線升級技術來更新程序所耗費的時間比採用JTAG口燒寫程序所耗費的時間要長一些,但解決了複雜情況下程序升級困難的問題。總之,該方法簡單可靠,可應用於嵌入式設備的軟件程序更新升級中。