ADS1298 single-shot模式数据未转换,求助
时间:10-02
整理:3721RD
点击:
采用STM32和ADS1298开发12导心电测量仪。ADS1298配置为continuous模式已经工作正常,现在要改为single-shot模式。在single-shot模式下,能正确读取芯片ID,也能正确判断导联是否脱落,但ADC的数据明显不正确。主要代码如下:
1、ADS1298寄存器配置
uint8_t RegisterData[25] = {0xC2, //0x82, 0x03, /* CONFIG1 */ //8kSPS 0x10, /* CONFIG2 */ //RLD 0xDC,/*0xD4*/ /* CONFIG3 */ 0x07/*0x0F*/, /* LOFF */// 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,/*CHnSET*/ 0xFF, /* RLD_SENSP */// 0xFF, /* RLD_SENSN */// 0xFF, /* LOFF_SENSP *///所有通道的正端均进行脱落检测 0xFF, /* LOFF_SENSN *///所有通道的负端均进行脱落检测 0x00, /* LOFF_FLIP */ 0, /* LOFF_STATP,导联正端连接状态,只读,不需要设*/ 0, /* LOFF_STATN,导联负端连接状态,只读,不需要设*/ 0, /* GPIO */ 0, /* PACE */ 0xF0, /* RESP */ 0x2E,//0x2A/*0x22*/, /* CONFIG4 */ //采用0x2A或0x2E为单次模式,采用0x22为连续AD模式 0x0B, /* WCT1 *///WCT配置管脚 0xE2 /* WCT2 */ };
2、ADS1298初始化部分程序
void ADS1298_Init( void ) { uint8_t device_id = 0; ADS1298_IO_Init();//硬件端口初始化 ADS1298_PIN_CS = LOW; ADS1298_RESET(); device_id = ADS1298_GetID(); if( device_id == 0x92 ) //ID正确 { ADS1298_WriteMultiReg( ADS1298_CONFIG1_REG, RegisterData, 25 ); //配置ADS1298 } else { } }
3、读取数据子程序
void ADS1298_ReadDataOneShot() // use in Stop Read Continuous mode when DRDY goes low { uint8_t inByte, inByte1, inByte2, inByte3; int i; uint32_t stat_1 = 0; for( i = 0; i < 8; i++ ) { ADSChannelData[i] = 0; } ADS1298_WriteCmd( ADS1298_CMD_RDATA ); // READ CHANNEL DATA FROM FIRST ADS IN DAISY LINE for( i = 0; i < 3; i++ ) // read 3 byte status register (1100+LOFF_STATP+LOFF_STATN+GPIO[7:4]) { inByte = ADS1298_ReadByte( ); stat_1 = ( stat_1 << 8 ) | inByte; } /*输出电极脱落状态*/ for( i = 0; i < 16; i++ ) { uint8_t tmp = 0; tmp = ( stat_1 >> ( 19 - i ) ) & 0x01; if( tmp == 0x01 ) //导联脱落检测正常 { if( i <= 7 ) { printf( "第%d路正端脱落\r\n", 8 - i ); } else { printf( "第%d路负端脱落\r\n", 16 - i ); } } } for( i = 0; i < 8; i++ ) { inByte1 = ADS1298_ReadByte( ); inByte2 = ADS1298_ReadByte( ); inByte3 = ADS1298_ReadByte( ); ADSChannelData[i] = ( inByte1 << 16 ) | ( inByte2 << 8 ) | inByte3; } printf( "%d\r\n", ADSChannelData[0] ); //数据输出不正常 }
4.主函数
int main( void ) { NVIC_PriorityGroupConfig( NVIC_PriorityGroup_2 ); //设置系统中断优先级分组2 delay_init( 168 ); //初始化延时函数 uart_init(115200); ADS1298_Init();// /*--1--测试读ID号和寄存器,测试通过*/ // while( 1 ) // { // uint8_t IDtmp = 0; // IDtmp = ADS1298_GetID();//正确应该是0x92 // IDtmp = IDtmp; // delay_ms( 500 ); // ADS1298_WriteReg( 0x10, 0xff ); // delay_ms( 500 ); // IDtmp = ADS1298_ReadReg( 0x10 ); //应该是前面写的0xff // IDtmp = IDtmp; // } ADS1298_WriteCmd(ADS1298_CMD_SDATAC); while( 1 ) {uint32_t tmp = 0;ADS1298_WriteCmd(ADS1298_CMD_START);while(ADS1298_IsReady()!=1); ADS1298_ReadDataOneShot();delay_ms(10);tmp = ADSChannelData[0]; tmp = ADSChannelData[1];tmp = ADSChannelData[2];tmp = ADSChannelData[3];tmp = ADSChannelData[4];tmp = ADSChannelData[5];tmp = ADSChannelData[6];tmp = ADSChannelData[7]; } }
5、判断是否转换结束函数,主函数中调用
uint8_t ADS1298_IsReady( void ) { return ( ADS1298_PIN_DRDY == 0 ); }
现在无论是否加心电信号源,通过printf函数串口输出的数据基本都是如下:
8388607
0
8388607
8388607
8388607
0
8388607
0
0
0
8388607
8388607
0
8388607
0
8388607
8388607
8388607
0
8388607
0
0
0
8388607
8388607
求助~
补充下今天测试的结果,在函数void ADS1298_ReadDataOneShot() 中每次读取第1个字节,都是datasheet中说的1100,ADSChannelData[1]和ADSChannelData[2]的数据偶尔正确,但夹杂很多0,结果见下图。其他通道的结果仍然如上帖。
再次补充下今天调试的结果。
1、其他通道输出数据为8388607是因为增益太大饱和,在寄存器CHnSET中将增益改为1即可。
2、经过多次调试,发现在主函数的while循环中每次START之前重新对ADS1298初始化,采集即可正常,即:
...... ADS1298_Init(); ADS1298_WriteCmd( ADS1298_CMD_START ); ......
另外,在每次START之后,等待几个时钟周期后再去等待DRDY/,也可以,即:
…… uint32_t i=10; ADS1298_WriteCmd( ADS1298_CMD_START ); while(i!=0) //或改为delay_us(10); { i--; } while( ADS1298_IsReady() != 1 ); ……
系统时钟是168MHz。
这就说明,在ADS1298初始化为Single-shot模式后,第1次发送START指令采集时,
直接等待DRDY/准备好即可。后面各次发送START指令后,
都要先延迟几us,然后再去等待DRDY/。不知道为什么?
期待答复。谢谢。
使用连续模式,在DRDY的下降沿去读取数据即可。