ADS1246是TI公司大致在2009年中期推出的24位ADC,最高采样速率可达2Ksps,其为单通道器材,与之相对应的还有ADS1247和ADS1248三通道器材,但特性并非彻底一致。据TI材料介绍,ADS1246在ADS1247/ADS1248功用上做出简化,保留了其部分特性。本次规划,需求用到24位单通道转化器材,所以考虑用到ADS1246,主操控器用STM32L系列。以下为ADS1246的引脚图
上图显现ADS1246引脚图,其CS/SCLK/DIN/DUT为SPI通讯接口,RESET/START/DRDY为操控与状况脚,AVDD/AVSS以及DVDD/DGND别离为模仿/数字电源供电端,REFP/REFN为基准源输入脚,AINP/AINN为模仿信号输入端。其间,DRDY忙信号指示功用能够附加到DOUT引脚上,这样DRDY脚能够留空。在实际使用中发现,START脚做为ADC的发动脚,还必须得接出来,因我还未找到有经过软件能发动ADS1246转化的办法,但其DS中有说到START信号和SLEEP/WAKEUP相类似的功用,暂未深研讨。趁便 提一下,TI关于ADS1246的文档是改自于另一颗ADC器材的文档,所以极端烂……
ADS1246的SPI时序,这个是需求提一下的,一般来说,SPI协议在上升沿锁存数据,下降沿更新数据,这是一般SPI协议的作法。但ADS1246需求在下降沿锁存数据,上升沿更新数据,在设置SPI寄存器的时分需求留意一下,当我选用一般性设置的时分,发现通讯不正常。以下是STM32L的SPI设置,用的是SPI2。
//SPI2装备
RCC-》APB1ENR|=RCC_APB1ENR_SPI2EN;
SPI2-》CR1=SPI_CR1_MSTR|SPI_CR1_BR|SPI_CR1_SSM|SPI_CR1_SSI|SPI_CR1_CPHA;//8位形式
SPI2-》CR1|=SPI_CR1_SPE;
SPI2的驱动:
//SPI2写数据
voidSPI2_WriteBytes(uint8*TxBuffer,uint16TxLenth)
{
uint8i;
while(TxLenth–){
while((SPI2-》SR&SPI_SR_TXE)==0);
SPI2-》DR=*TxBuffer++;
while((SPI2-》SR&SPI_SR_RXNE)==0);
i=SPI2-》DR;
}
i++;
}
上程序中i++的引进在于防止keil-MDK发生编译正告。
//SPI2读数据
voidSPI2_ReadBytes(uint8*RxBuffer,uint16RxLenth)
{
while(RxLenth–){
while((SPI2-》SR&SPI_SR_TXE)==0);
SPI2-》DR=*RxBuffer;
while((SPI2-》SR&SPI_SR_RXNE)==0);
*RxBuffer++=SPI2-》DR;
}
}
以上驱动代码,能确保SPI在最终一个字节彻底发送完结之后退出,假如没有等候SPI_SR_RXNE,则仅仅只是把数据转移到SPI移位寄存器,并未彻底送出,不胪陈。
以下介绍我的驱动进程,在驱动ADS1246的时分,首要参阅那个网方的58页的极烂文档,上面没有清晰说到整个上电进程以及初始化进程,至于那个适当重要的自校准进程及操作办法也没有说到,所以自己探索了一整天时刻,在此收拾。
ADS1246选用SPI通讯 ,其一切通讯引脚(SCK/DIN/DOUT)都在CS脚为低电平的时分有用,在CS为高时均为三态,当DRDY绑定到DOUT脚时,只要在CS为低时才干正确的指示忙状况,若DRDY选用独自的引脚,则不受CS操控。ADS1246的一切通讯进程被分为若干个指令组,有的需求带参数,有的不需求带参数,其实我也不明白它为什么要搞那么费事,感觉原本能够很简单的处理,成果弄的很乱。以下为其指令分组:
大致分为指令类(不带参数),读寄存器,写寄存器三类,以下别离分其完成:
//ADS1246指令码列表
#defineADC_CMD_WAKEUP0x00//退出睡觉形式
#defineADC_CMD_SLEEP0x02//进入睡觉形式
#defineADC_CMD_SYNC0x04//同步ADC转化
#defineADC_CMD_RESET0x06//芯片复位
#defineADC_CMD_NOP0xFF//空操作
#defineADC_CMD_RDATA0x12//单次读取数据
#defineADC_CMD_RDATAC0x14//接连读取数据
#defineADC_CMD_SDATAC0x16//中止接连读取
#defineADC_CMD_RREG0x20//读寄存器
#defineADC_CMD_WREG0x40//写寄存器
#defineADC_CMD_SYSOCAL0x60//体系偏移校准
#defineADC_CMD_SYSGCAL0x61//体系增益校准
#defineADC_CMD_SELFOCAL0x62//体系自校准
#defineADC_CMD_RESTRICTED0xF1//
/*———————————————————
写指令
———————————————————*/
voidADS1246_WriteCmd(uint8Cmd)
{
ADC_SPI_CS_CLR
ADC_WriteBytes(&Cmd,1);
ADC_SPI_CS_SET
}
/*———————————————————
读寄存器
———————————————————*/
voidADS1246_ReadReg(uint8RegAddr,uint8*Buffer,uint8Length)
{
uint8Cmd[2];
ADC_SPI_CS_CLR
Cmd[0]=ADC_CMD_RREG|RegAddr;
Cmd[1]=Length-1;
ADC_WriteBytes(Cmd,2);
ADC_ReadBytes(Buffer,Length);
Cmd[0]=ADC_CMD_NOP;
ADC_WriteBytes(Cmd,1);
ADC_SPI_CS_SET
}
/*———————————————————
写寄存器
———————————————————*/
voidADS1246_WriteReg(uint8RegAddr,uint8*Buffer,uint8Length)
{
uint8Cmd[2];
ADC_SPI_CS_CLR
Cmd[0]=ADC_CMD_WREG|RegAddr;
Cmd[1]=Length-1;
ADC_WriteBytes(Cmd,2);
ADC_WriteBytes(Buffer,Length);
ADC_SPI_CS_SET
}
在写读存器时,必定要留意,依据其DS文档第32页阐明,这以后发一个NOP指令能够强制DOUT引脚输出高电平,这样能够随后判别DOUT是否为低从而知道是否处于忙状况,不然会得到一个脉冲。其实在任何的读操作完结后,发一个字节的NOP指令即可将DOUT强制输出高电平。当DRDY绑定到DOUT的时分,这个是非常重要的。
在弄清楚以上指令读写办法之后,需求完成其忙状况判别,这个在许多芯片驱动时都会遇到,它直接提示了其内部的作业状况,只要在不忙时才干持续履行下一条指令。
/*———————————————————
忙状况判别,最长等候时刻,200X10ms=2S
————————-