我在用扇區(qū)讀寫的時(shí)候每次執(zhí)行寫操作的時(shí)候好像要花很長(zhǎng)時(shí)間,這樣的話就會(huì)有數(shù)據(jù)丟失,程序如下: SIGNAL(SIG_INTERRUPT2) { UINT8 i; WriteToReg_ADC(0X38); CS_0; FILE_DATA_BUF[h]=Read_Byte_SPI(); FILE_DATA_BUF[h+1]=Read_Byte_SPI(); CS_1; CS_A; FILE_DATA_BUF[h+2]=Read_Byte_SPI(); FILE_DATA_BUF[h+3]=Read_Byte_SPI(); CS_B; h+=4; if(h==512) { h=0; mCmdParam.Write.mSectorCount=1; i = CH375FileWrite( ); /* 向文件寫入數(shù)據(jù) */ mStopIfError( i ); mCmdParam.Write.mSectorCount=0; i = CH375FileWrite( ); /* 向文件寫入數(shù)據(jù) */ mStopIfError( i ); } }
這是用labview回放的數(shù)據(jù)。
請(qǐng)問哪個(gè)例子是用雙緩沖寫的?
請(qǐng)問怎么應(yīng)用CH375FileWriteX這個(gè)函數(shù),能否告知一個(gè)用雙緩沖寫的例子?
375沒有雙緩沖的例子,你可以使用374芯片,這個(gè)芯片有128字節(jié)的緩沖區(qū),可以用來做雙緩沖使用,速度比375要快
那么我應(yīng)該怎樣應(yīng)用CH375FileWriteX()這個(gè)函數(shù)呢?怎樣將數(shù)據(jù)先放到(例如0x40位起始地址)的緩沖區(qū)呢,有沒有用CH375FileWriteX()寫的例子呢?
使用雙緩沖區(qū)交替讀寫,那么不要定義FILE_DATA_BUF_LEN,而是在參數(shù)中指定緩沖區(qū)起址,用CH375FileReadX代替CH375FileRead,用CH375FileWriteX代替CH375FileWrite。 在調(diào)用之前指定緩沖區(qū)mCmdParam.WriteX.mDataBuffer的位子。CH375EVT\PUB\MCS51C\FILELIB4\EXAM1中的注釋掉的地方有使用說明。
那就是說FILE_DATA_BUF在這里就沒有用了是嗎?那我采集的數(shù)據(jù)怎樣放到緩沖區(qū)呢?
FILE_DATA_BUF在這里就沒有用了
將你的數(shù)據(jù)緩沖區(qū)的地址賦給mCmdParam.WriteX.mDataBuffer
我可不可以這樣: data[i]=get_adc(); mCmdParam.WriteX.mDataBuffer=&data[0];
我用了CH375FileWriteX之后,為什么會(huì)有下面的錯(cuò)誤?。? libCH375HFD.A(CH375SEC.o)(.text+0x166): In function `CH375ReadBlock': : undefined reference to `xWriteToExtBuf' libCH375HFD.A(CH375SEC.o)(.text+0x1be): In function `CH375ReadBlock': : undefined reference to `xWriteToExtBuf' libCH375HFD.A(CH375SEC.o)(.text+0x2d4): In function `CH375WriteBlock': : undefined reference to `xReadFromExtBuf' libCH375HFD.A(CH375SEC.o)(.text+0x32c): In function `CH375WriteBlock': : undefined reference to `xReadFromExtBuf' make.exe: *** [ch375.elf] Error 1
象你這樣功能的程序?qū)r(shí)間要求比較高的不適合用這種結(jié)構(gòu),需要改成實(shí)時(shí)系統(tǒng)結(jié)構(gòu)。即你不可能等待優(yōu)盤寫完再繼續(xù)處理,應(yīng)該寫入命令和64字節(jié)數(shù)據(jù)后就繼續(xù)去采集數(shù)據(jù),有中斷后繼續(xù)傳輸下一個(gè)64字節(jié)。
to Skunk { M }: 你有qq嗎?想和你探討一下!
你可以將你的程序帖出來看下,這個(gè)更象程序的問題
/* 2004.06.05 **************************************** ** Copyright (C) W.ch 1999-2004 ** ** Web: http://www.winchiphead.com ** **************************************** ** USB Host File Interface for CH375 ** ** TC2.0@PC, ICCAVR_6.31@AVR ** **************************************** */ /* CH375 主機(jī)文件系統(tǒng)接口 */ /* 支持: FAT12/FAT16/FAT32 */
/* AVR單片機(jī)C語言的U盤文件讀寫示例程序 */ /* 該程序?qū)盤中的/C51/CH375HFT.C文件中的小寫字母轉(zhuǎn)成大寫字母后, 寫到新建的文件NEWFILE.TXT中, 如果找不到原文件CH375HFT.C, 那么該程序?qū)@示C51子目錄下所有以CH375開頭的文件名, 并新建NEWFILE.TXT文件并寫入提示信息, 如果找不到C51子目錄, 那么該程序?qū)@示根目錄下的所有文件名, 并新建NEWFILE.TXT文件并寫入提示信息 */ /* CH375的INT#引腳采用查詢方式處理, 數(shù)據(jù)復(fù)制方式為"內(nèi)部復(fù)制", 本程序適用于ATmega128單片機(jī), 串口0輸出監(jiān)控信息,9600bps */
/* ICCAVR -v -Wp -I\ICC\INCLUDE -Wf -Mavr_enhanced -Wl -L\ICC\LIB -m -bfunc_lit:0x8C.0x20000 -bdata:0x0100.0x0FFF -dram_end:0x0FFF -dhwstk_size:40 -ucrtatmega.o CH375HFB.C CH375HFB.A */
#include #include #include #include #include #include #define uchar unsigned char #define uint unsigned int #define ulonglong unsigned long long #define has_volatile 1 #if has_volatile volatile unsigned int h=0; //全局變量,會(huì)在中斷服務(wù)程序中被修改,須加volatile限定 #else unsigned int h=0; //全局變量. #endif
/* 以下定義的詳細(xì)說明請(qǐng)看CH375HF9.H文件 */ #define LIB_CFG_FILE_IO 1 /* 文件讀寫的數(shù)據(jù)的復(fù)制方式,0為"外部子程序",1為"內(nèi)部復(fù)制" */ #define LIB_CFG_INT_EN 0 /* CH375的INT#引腳連接方式,0為"查詢方式",1為"中斷方式" */
/* 單片機(jī)的RAM有限,其中CH375子程序用512字節(jié),剩余RAM部分可以用于文件讀寫緩沖 */ #define FILE_DATA_BUF_LEN 0x0200 /* 外部RAM的文件數(shù)據(jù)緩沖區(qū),緩沖區(qū)長(zhǎng)度不小于一次讀寫的數(shù)據(jù)長(zhǎng)度 */ /* 如果準(zhǔn)備使用雙緩沖區(qū)交替讀寫,那么不要定義FILE_DATA_BUF_LEN,而是在參數(shù)中指定緩沖區(qū)起址,用CH375FileReadX代替CH375FileRead,用CH375FileWriteX代替CH375FileWrite */
#define CH375_INT_WIRE ( PIND & 0x04 ) /* PINB.4, CH375的中斷線INT#引腳,連接CH375的INT#引腳,用于查詢中斷狀態(tài) */
#define NO_DEFAULT_CH375_F_ENUM 1 /* 未調(diào)用CH375FileEnumer程序故禁止以節(jié)約代碼 */ #define NO_DEFAULT_CH375_F_QUERY 1 /* 未調(diào)用CH375FileQuery程序故禁止以節(jié)約代碼 */
#include "CH375HFB.H" #define CS_1 (PORTB|= 0X08) #define CS_0 (PORTB&= 0XF7)
/* 有些AVR單片機(jī)提供開放系統(tǒng)總線,那么直接將CH375掛在其系統(tǒng)總線上,以8位I/O方式進(jìn)行讀寫 */ /* 雖然Atmega128提供系統(tǒng)總線,不過本例假定不開放系統(tǒng)總線,所以用I/O引腳模擬產(chǎn)生CH375的并口讀寫時(shí)序 */ /* 本例中的硬件連接方式如下(實(shí)際應(yīng)用電路可以參照修改下述3個(gè)并口讀寫子程序) */ /* 單片機(jī)的引腳 CH375芯片的引腳 PIND.2 INT# PORTD.7 A0 PORTD.6 CS# PORTD.5 WR# PORTD.4 RD# PORTA(8位端口) D7-D0 */
void mDelay1uS(void ) /* 至少延時(shí)1uS,根據(jù)單片機(jī)主頻調(diào)整 */ { UINT8 i; for ( i = 5; i != 0; i -- ); }
void CH375_PORT_INIT(void ) /* 由于使用通用I/O模塊并口讀寫時(shí)序,所以進(jìn)行初始化 */ { DDRA = 0x00; /* 設(shè)置8位并口為輸入 */ PORTD = 0x70; /* 設(shè)置CS,WR,RD默認(rèn)為高電平 */ DDRD = 0xF0; /* 設(shè)置CS,WR,RD,A0為輸出,設(shè)置INT#為輸入 */ }
void xWriteCH375Cmd( UINT8 mCmd ) /* 外部定義的被CH375程序庫調(diào)用的子程序,向CH375寫命令 */ { mDelay1uS( ); mDelay1uS( ); /* 至少延時(shí)1uS */ /* *(volatile unsigned char *)CH375_CMD_PORT_ADDR = mCmd; 通過并口直接讀寫CH375而非普通I/O模擬 */ PORTD |= 0x80; /* 輸出A0=1 */ PORTA = mCmd; /* 向CH375的并口輸出數(shù)據(jù) */ DDRA = 0xFF; /* 并口D0-D7輸出 */ PORTD &= 0x9F; /* 輸出有效寫控制信號(hào), 寫CH375芯片的命令端口, A0=1; CS=0; WR=0; RD=1; */ DDRA = 0xFF; /* 該操作無意義,僅作延時(shí),CH375要求讀寫脈沖寬度大于100nS */ PORTD |= 0xF0; /* 輸出無效的控制信號(hào), 完成操作CH375芯片, A0=1; CS=1; WR=1; RD=1; */ DDRA = 0x00; /* 禁止數(shù)據(jù)輸出 */ PORTD &= 0x7F; /* 輸出A0=0; 可選操作 */ mDelay1uS( ); mDelay1uS( ); /* 至少延時(shí)2uS */ }
void xWriteCH375Data( UINT8 mData ) /* 外部定義的被CH375程序庫調(diào)用的子程序,向CH375寫數(shù)據(jù) */ { /* *(volatile unsigned char *)CH375_DAT_PORT_ADDR = mData; 通過并口直接讀寫CH375而非普通I/O模擬 */ PORTA = mData; /* 向CH375的并口輸出數(shù)據(jù) */ DDRA = 0xFF; /* 并口D0-D7輸出 */ PORTD &= 0x1F; /* 輸出有效寫控制信號(hào), 寫CH375芯片的數(shù)據(jù)端口, A0=0; CS=0; WR=0; RD=1; */ DDRA = 0xFF; /* 該操作無意義,僅作延時(shí),CH375要求讀寫脈沖寬度大于100nS */ PORTD |= 0x70; /* 輸出無效的控制信號(hào), 完成操作CH375芯片, A0=0; CS=1; WR=1; RD=1; */ DDRA = 0x00; /* 禁止數(shù)據(jù)輸出 */ mDelay1uS( ); /* 至少延時(shí)1.2uS */ }
UINT8 xReadCH375Data( void ) /* 外部定義的被CH375程序庫調(diào)用的子程序,從CH375讀數(shù)據(jù) */ { UINT8 mData; /* mData = *(volatile unsigned char *)CH375_DAT_PORT_ADDR; 通過并口直接讀寫CH375而非普通I/O模擬 */ mDelay1uS( ); /* 至少延時(shí)1.2uS */ DDRA = 0x00; /* 數(shù)據(jù)輸入 */ PORTD &= 0x2F; /* 輸出有效讀控制信號(hào), 讀CH375芯片的數(shù)據(jù)端口, A0=0; CS=0; WR=1; RD=0; */ DDRA = 0x00; /* 該操作無意義,僅作延時(shí),CH375要求讀寫脈沖寬度大于100nS */ mData = PINA; /* 從CH375的并口PA輸入數(shù)據(jù) */ PORTD |= 0x70; /* 輸出無效的控制信號(hào), 完成操作CH375芯片, A0=0; CS=1; WR=1; RD=1; */ return( mData ); }
/* 在P0.2連接一個(gè)LED用于監(jiān)控演示程序的進(jìn)度,低電平LED亮 */ //#define LED_OUT_INIT( ) { PORTB |= 0x80; DDRB |= 0x80; } /* PORTB.7 高電平為輸出方向 */ //#define LED_OUT_ACT( ) { PORTB &= 0x7F; } /* PORTB.7 低電平驅(qū)動(dòng)LED顯示 */ //#define LED_OUT_INACT( ) { PORTB |= 0x80; } /* PORTB.7 低電平驅(qū)動(dòng)LED顯示 */
/* 延時(shí)指定毫秒時(shí)間,根據(jù)單片機(jī)主頻調(diào)整,不精確 */ void mDelaymS( UINT8 ms ) { UINT16 i; while ( ms -- ) for ( i = 2600; i != 0; i -- ); }
/* 檢查操作狀態(tài),如果錯(cuò)誤則顯示錯(cuò)誤代碼并停機(jī) */ void mStopIfError( UINT8 iError ) { if ( iError == ERR_SUCCESS ) return; /* 操作成功 */ printf( "Error: %02X\n", (UINT16)iError ); /* 顯示錯(cuò)誤 */ while ( 1 ) { //LED_OUT_ACT( ); /* LED閃爍 */ //mDelaymS( 100 ); //LED_OUT_INACT( ); //mDelaymS( 100 ); } }
/* 為printf和getkey輸入輸出初始化串口 */ //extern int _textmode; //void mInitSTDIO( ) //{ // UBRR0H = 0; // UBRR0L = 103; /* 9600bps@16MHz */ // UCSR0B = 0x18; /* BIT(RXEN) | BIT(TXEN); */ // UCSR0C = 0x06; /* BIT(UCSZ1) | BIT(UCSZ0); */ // _textmode = 1; //} //**********************短延時(shí)程序50us**************************// void delay50us(uint t) { uint j; for(;t>0;t--) for(j=0;j<70;j++) ; } //**********************短延時(shí)程序5us**************************// void delay5us(uint t) { uint j; for(;t>0;t--) for(j=0;j<7;j++) ; }
void spi_init(void) { DDRB=0XBB; //0b10111011; //PB3\PB6輸入 PORTB=0X67;//0b01100111; //MISO輸入加上拉電阻 SPCR=0X5F;//SPI 設(shè)置//SPCR = (1< SPSR = 0X00; //setup SPI,主機(jī)倍頻 //CS_0; //使能SPI器件 }
//******************SPI寫寄存器函數(shù) 8BIT*******************************// void WriteToReg_ADC(uint byteword) { CS_0;//PORTB&= 0XFB; //使能SPI器件 //SPE_1; SPDR=byteword; //發(fā)送數(shù)據(jù) while(!(SPSR & (1< CS_1;//PORTB|= 0X04; //禁止SPI器件 delay5us(10); } //******************SPI讀函數(shù) 8BIT*******************************// unsigned char Read_Byte_SPI(void) { SPDR = 0Xff; //發(fā)送數(shù)據(jù),給AD7705提供脈沖 while (!(SPSR &(1< return SPDR; } //******************SPI讀數(shù)據(jù)寄存器函數(shù) 16BIT****************************// uint Read_ADC16BitValue(void) { uchar temp1=0; uint temp=0; CS_0;//PORTB&= 0XFB; //使能SPI器件 temp=Read_Byte_SPI(); temp=temp<<8; temp1=Read_Byte_SPI(); temp=temp|temp1; CS_1;//PORTB|= 0X04; //禁止SPI器件 return temp; } //**************************AD初始化*************************//
在主函數(shù)里面只有許多器件的初始化和打開文件,然后進(jìn)入中斷。