本應(yīng)用筆記闡述了如何利用MxTNI? JTAG庫以及串行向量格式(SVF)文件來編程Xilinx? PROM器件。假定讀者已經(jīng)對(duì)JTAG和可編程邏輯器件有了一定認(rèn)識(shí)。
介紹
Maxim微型網(wǎng)絡(luò)接口(MxTNI)是Dallas Semiconductor (Maxim Integrated的全資子公司)開發(fā)的一個(gè)平臺(tái)。它包含了一套小型卻功能強(qiáng)大的芯片組以及Java?可編程虛擬機(jī)。芯片組具有處理、控制、器件級(jí)通信和網(wǎng)絡(luò)互連的能力。為了和任何JTAG器件通信,TINIs400適配板在J21端具有4引腳JTAG輸出。 這些引腳將直接連到JTAG器件上標(biāo)準(zhǔn)JTAG引腳TDI、TDO、TMS和TCK。
在系統(tǒng)編程PROM可以進(jìn)行單獨(dú)編程,或者級(jí)連編程。鏈中的所有器件共享TCK和TMS信號(hào)。MxTNI的TDI信號(hào)接到邊界掃描鏈中第一個(gè)器件的TDI輸入端。第一個(gè)器件的TDO信號(hào)接到鏈中第二個(gè)器件的TDI輸入上,如此連接下去。鏈中的最后一個(gè)器件的TDO輸出接到MxTNI的TDO引腳上,見圖1所示。
圖1. 所有JTAG操作都通過器件的測試訪問端口進(jìn)行控制
所有JTAG操作都是由器件的測試訪問端口(TAP)控制的。TAP包括四個(gè)信號(hào):TMS、TDI、TDO和TCK。 這些信號(hào)通過TAP控制器,即16狀態(tài)有限狀態(tài)機(jī)與器件相互作用。JTAG的TMS信號(hào)控制狀態(tài)間的轉(zhuǎn)換。指令和數(shù)據(jù)由TDI引腳移入器件,并由TDO引腳移出。TDI和TDO信號(hào)的所有狀態(tài)轉(zhuǎn)換和行為都與TCK同步。見 圖2 。
圖2.
所有JTAG操作都是將數(shù)據(jù)移入或移出JTAG指令和數(shù)據(jù)寄存器。TAP控制器可對(duì)所有這些寄存器直接訪問。有兩類JTAG寄存器:指令寄存器(IR)和數(shù)據(jù)寄存器(DR)。訪問IR通過移位-IR (Shift-IR)狀態(tài)實(shí)現(xiàn),而訪問DR通過移位-DR (Shift-IR)狀態(tài)實(shí)現(xiàn)。IR長度通常是大于2位的任意長度。除了由IEEE? Std. 1149.1定義的BYPASS指令為全1以外,生產(chǎn)商定義所有其它的指令位碼。
在本應(yīng)用筆記中,JAVA樣例代碼將解釋說明串行向量格式(SVF)文件來進(jìn)行編程。這里所用的SVF是描述高層IEEE 1149.1 (JTAG)總線操作的語法規(guī)范。JTAG設(shè)備和軟件提供商已經(jīng)將SVF作為標(biāo)準(zhǔn)用于數(shù)據(jù)交換。SVF以緊湊和可移植的形式描述JTAG鏈操作。SVF文件通過描述需要移入器件鏈的信息,記錄JTAG操作。通過Xilinx iMPACT (詳細(xì)信息見下面)軟件,將JTAG操作記錄在SVF文件中。SVF文件寫成ASCII文本形式,因此可以在任何文本編輯器中人工讀、修改和寫?!霸S多第三方編程工具使用SVF文件作為輸入,這樣利用包含在SVF文件中的信息可以對(duì)JTAG鏈中的Xilinx器件編程?!?/p>
JTAG庫簡要說明
所開發(fā)的JTAG類庫可幫助用戶使用MxTNI與JTAG器件進(jìn)行通信。一個(gè)可能的應(yīng)用是:從遙遠(yuǎn)的位置對(duì)可編程邏輯進(jìn)行動(dòng)態(tài)和在系統(tǒng)更新。用戶能夠?qū)AP控制器初始化到初始狀態(tài)、瀏覽16個(gè)狀態(tài)、獲得或設(shè)置TAP狀態(tài)、產(chǎn)生時(shí)鐘信號(hào)、發(fā)送命令和數(shù)據(jù)等等。以下為JTAG類方法的簡要說明:
- public jtag(): 加載jtag庫。
- public String getVersion(): 返回jtag類的版本。
- public int runClock( int numticks ): 以numticks指定的數(shù)量運(yùn)行時(shí)鐘。
- public byte initialize(): 先將TMS置高達(dá)五個(gè)TCK時(shí)鐘脈沖,然后TMS置低1個(gè)時(shí)鐘并進(jìn)入缺省狀態(tài)Run-Test-Idle。
- public byte setState( byte aState ): 將目標(biāo)狀態(tài)設(shè)置為aState指定的狀態(tài)。
- public byte getState(): 返回目標(biāo)狀態(tài)。
- public byte scanState( byte state ): 將TAP控制器的狀態(tài)機(jī)轉(zhuǎn)換到參數(shù)state指定的狀態(tài)。
- public String displayState(byte state): 顯示TAP控制器的狀態(tài)。
- public byte waitState( int msecs ): 進(jìn)程延時(shí)一定毫秒數(shù)。
- public byte getTDO(): 獲得引腳TDO的當(dāng)前狀態(tài)。
- public void setTDI(byte logic): 引腳TDI設(shè)定為指定的邏輯狀態(tài)。
- public byte getTDI(): 獲得引腳TDI的當(dāng)前狀態(tài)。
- public void setTMS(byte logic): s引腳TMS設(shè)定為指定的邏輯狀態(tài)。
- public byte getTMS(): 獲得引腳TMS的當(dāng)前狀態(tài)。
- public void setTCK(byte logic): 引腳TCK設(shè)定為指定的邏輯狀態(tài)。
- public byte getTCK(): 獲得引腳TCK的當(dāng)前狀態(tài)。
- public byte sendNrcv(byte[] data, int offset, int size, byte numberOfBits, boolean state, boolean update, byte extraHeaderClock, byte HeaderBitVal,byte extraTrailerClock, byte TrailerBitVal): 發(fā)送數(shù)據(jù)字節(jié)數(shù)組,并接收返回的字節(jié)數(shù)組。
- public byte sendNrcv(int[] data, int offset, int size, byte numberOfBits, boolean state, boolean update,byte extraHeaderClock, byte HeaderBitVal,byte extraTrailerClock, byte TrailerBitVal): 發(fā)送數(shù)據(jù)整形數(shù)組,并接收返回的整形數(shù)組。
硬件和軟件需求
以下為所需要的硬件和軟件:
- TINI400評(píng)估板,并配置有MxTNI OS 1.12或更高版本。
- JTAG器件(可編程邏輯)。
- 器件鏈中JTAG器件模型的SVF文件。
編程步驟
第1步: 將MxTNI板的4個(gè)JTAG引腳與JTAG器件的4個(gè)JTAG引腳相連。
第2步: 遵循SVF文件的命令并使用JTAG庫編寫JAVA應(yīng)用程序,來對(duì)JTAG器件進(jìn)行編程,編譯后加載到MxTNI。
(附錄A: 一個(gè)樣例Idcode.svf文件,實(shí)現(xiàn)從單獨(dú)的XC18V02 Xilinx器件讀取IDCODE。)
(附錄B: 樣例AppJtag.java程序,將SVF文件作為輸入并產(chǎn)生JTAG信號(hào)來讀取IDCODE。該樣例程序是用來處理單獨(dú)器件的(對(duì)于級(jí)連器件的詳細(xì)信息見附錄C)。)
(附錄C: 展示了由已有的SVF文件編程級(jí)連器件的步驟。)
第3步: 運(yùn)行JAVA程序??蓤?zhí)行AppJtag.tini將加載到MxTNI板。在MxTNI提示符下錄入“java AppJtag.tini Idcode.svf”來運(yùn)行程序。程序?qū)a(chǎn)生JTAG信號(hào)與JTAG器件進(jìn)行通信。注意:
- 生成的編程JTAG器件SVF文件應(yīng)該與器件鏈中設(shè)計(jì)的確切模型一致。
- 所有操作碼,例如編程JTAG器件的READ、WRITE、ERASE及其它命令等,都是由生產(chǎn)商定義的。SVF文件應(yīng)該包含用戶需要的所有操作碼。
示例
以下示例說明了如何利用JTAG庫從Xilinx JTAG器件XC18V02中讀取Idcode。完整的樣例代碼,請(qǐng)查閱附錄B。
-
生成器件模型的SVF文件:
使用Xilinx WEB START來生成SVF文件。
- 執(zhí)行iMPACT并選擇單選按鈕“Prepare Configuration Files”選項(xiàng),然后點(diǎn)擊next。選擇單選按鈕“Boundary Scan file”并點(diǎn)擊next。選擇“SVF File”然后點(diǎn)擊finish。
- 在對(duì)話框中錄入SVF文件名,例如example,并點(diǎn)擊OK。
- 選擇要加載的name_of_mcs_file.mcs文件,并從列表中選擇PROM器件(XC18V02_vq44)。
- XC18V02已加到模型中。
- 點(diǎn)擊我們要編程的XC18V02器件型號(hào)。器件將為高亮顯示。
- 點(diǎn)擊鼠標(biāo)右鍵,并選擇編程選項(xiàng)。勾選“Get Idcode”。
- 在模型外點(diǎn)擊鼠標(biāo)以取消高亮顯示器件,右擊并選擇option以關(guān)閉SVF。這時(shí)就生成了模型的SVF文件。
-
讀取SVF文件并使用JTAG庫來編程XC18V02器件。
- 以下為SVF文件的部分樣例代碼:
// Created using Xilinx iMPACT Software [ISE WebPACK - 5.1i] TRST OFF; ENDIR IDLE; ENDDR IDLE; STATE RESET IDLE; TIR 0 ; HIR 0 ; TDR 0 ; HDR 0 ; // Validating chain... TIR 0 ; HIR 0 ; TDR 0 ; HDR 0 ; SIR 8 TDI (ff) SMASK (ff) ; TIR 0 ; HIR 5 TDI (1f) SMASK (1f) ; HDR 1 TDI (00) SMASK (01) ; TDR 0 ; //Loading device with 'idcode' instruction. SIR 8 TDI (fe) SMASK (ff) ; SDR 32 TDI (00000000) SMASK (ffffffff) TDO (05025093) MASK (ffffffff) ; //Loading device with 'conld' instruction. SIR 8 TDI (f0) ; RUNTEST 110000 TCK;
- 遵循SVF命令并使用庫向XC18V02發(fā)送命令。
- SVF規(guī)范提供了四個(gè)全局填充指令:頭指令寄存器(HIR)、尾部指令寄存器(TIR)、頭數(shù)據(jù)寄存器(HDR)和尾部數(shù)據(jù)寄存器(TDR)。這些全局命令規(guī)定了移位操作的開始和結(jié)尾處要填充的位數(shù),從而解決旁路器件問題,并為SVF文件壓縮提供了一種簡單的方法。一旦指定,這些位會(huì)出現(xiàn)在SIR或者SDR命令的每一組比特位的前面或者后面。
- 以下兩個(gè)示例說明如何把SVF命令解釋為JTAG庫命令。
示例1: 帶有全局填充指令的SVF語法結(jié)構(gòu)
a/ SVF file commands:
STATE RESET IDLE;
TIR 0 ;
HIR 5 TDI (1f) SMASK (1f) ;
HDR 1 TDI (00) SMASK (01) ;
TDR 0 ;
//Loading device with 'idcode' instruction.
SIR 8 TDI (fe) SMASK (ff) ;
SDR 32 TDI (00000000) SMASK (ffffffff) TDO (05025093) MASK (ffffffff) ;
//Loading device with 'conld' instruction.
SIR 8 TDI (f0) ;
RUNTEST 110000 TCK;
//Check for Read/Write Protect.
SIR 8 TDI (ff) TDO (01) MASK (ff) ;
//Loading device with 'idcode' instruction.
SIR 8 TDI (fe) ;
SDR 32 TDI (00000000) TDO (05025093) ;
//Loading device with 'conld' instruction.
SIR 8 TDI (f0) ;
RUNTEST 110000 TCK;
//Check for Read/Write Protect.
SIR 8 TDI (ff) TDO (01) ;
TIR 0 ;
HIR 0 ;
TDR 0 ;
HDR 0 ;
b/ Sample JAVA program using JTAG library:
import java.io.*;
import javax.comm.*;
import com.dalsemi.comm.*
public static void main(String[] args)
{
myJtag = new jtag();
int SIZE = 0x1000;
byte[] byteArray = new byte[SIZE];
byte HeaderInstBitVal = (byte)0x00;
byte TrailerInstBitVal = (byte)0x00;
byte HeaderDataBitVal= (byte)0x00;
byte TrailerDataBitVal= (byte)0x00;
// STATE RESET IDLE;
myJtag.initialize();//This JTAG library method will initialize
// XC18V02 TAP controller and stay at
// Run-Test/Idle
// TIR 0 ;
byte TIR = (byte)0x00;
// HIR 5 TDI (1f) SMASK (1f) ;
byte HIR = (byte)0x05;
HeaderInstBitVal = (byte)0x01;
// HDR 1 TDI (00) SMASK (01) ;
byte HDR = (byte)0x01;
HeaderDataBitVal = (byte)0x00;
// TDR 0
byte TDR = (byte)0x00;
// SIR 8 TDI (fe) SMASK (ff) ;
byteArray[0] = (byte)0xfe;
myJtag.sendNrcv(byteArray,0,1,(byte)8,true,false,(byte)HIR,
(byte)HeaderInstBitVal,(byte)TIR,(byte)TrailerInstBitVal);
// SDR 32 TDI (00000000) SMASK (ffffffff) TDO (05025093) MASK
// (ffffffff) ;
byteArray[0] = (byte)0x00;
byteArray[1] = (byte)0x00;
byteArray[2] = (byte)0x00;
byteArray[3] = (byte)0x00;
myJtag.sendNrcv(byteArray,0,4,(byte)8,false,false,(byte)HDR,
(byte)HeaderDataBitVal,(byte)TDR,(byte)TrailerDataBitVal);
// SIR 8 TDI (f0) ;