ADS1118 作为常用温度测量芯片被越来越多的开发者熟知,TI官方给出的是基于 MSP430 的驱动测试程序,由于 STM32 的普及,闲暇中移植了 MSP430 的 ADS1118 驱动程序到 STM32F103 平台下,并进行了测试,特在此记录,以飨读者。
使用 STM32F103 的 SPI2 接口连接 ADS1118 的通信接口:
STM32F103|ADS1118
—|—|—|
PB12|CS
PB13|SCLK
PB14|DOUT
PB15|DIN
ADS1118 手册建议数据线之间接一个50欧电阻,实际测试可不接。ADS1118 电源接3.3V并加滤波电容。
SPI 配置
ADS1118 接口的时序要求如图:

DIN 接口接收控制器送过来的配置数据,并且在 SCLK 的下降沿将数据锁存读入 ADS1118 内部,并且在 SCLK 的上升沿中将数据送出到 DOUT。基于此将 STM32F103 的 SPI 接口做如下配置,CPOL=0,CPHA = 1;
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16
|
void SPI_config(void){ SPI_Cmd(SPI_MASTER, DISABLE);//配置之前先关闭SPI接口
SPI_InitStructure.SPI_Direction = SPI_Direction_2Lines_FullDuplex; SPI_InitStructure.SPI_Mode = SPI_Mode_Master;//主模式 SPI_InitStructure.SPI_DataSize = SPI_DataSize_8b;//8bits SPI_InitStructure.SPI_CPOL = SPI_CPOL_Low;// CPOL=0 SPI_InitStructure.SPI_CPHA = SPI_CPHA_2Edge;//CPHA=1 SPI_InitStructure.SPI_NSS = SPI_NSS_Soft;//CS设置为软件配置 SPI_InitStructure.SPI_BaudRatePrescaler = SPI_BaudRatePrescaler_64;//通信速率 SPI_InitStructure.SPI_FirstBit = SPI_FirstBit_MSB;//高位在前 SPI_InitStructure.SPI_CRCPolynomial = 7; SPI_Init(SPI_MASTER, &SPI_InitStructure);
SPI_Cmd(SPI_MASTER, ENABLE);//配置完成使能SPI接口 }
|
对应 GPIO 的配置:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34
|
void SPI_GPIO_Configuration(void) { GPIO_InitTypeDef GPIO_InitStructure;
/* Configure SPI_MASTER pins-*/
// Pin PB13 (SCLK) must be configured as as 50MHz push pull GPIO_StructInit( GPIO_InitStructure.GPIO_Pin = SPI_MASTER_PIN_SCK; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP; GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; GPIO_Init(SPI_MASTER_GPIO,
// Pin PB14 (MISO) must be configured as as input pull-up GPIO_StructInit( GPIO_InitStructure.GPIO_Pin = SPI_MASTER_PIN_MISO; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING; GPIO_Init(SPI_MASTER_GPIO,
// Pin PB15 (MOSI) must be configured as as 50MHz push pull GPIO_StructInit( GPIO_InitStructure.GPIO_Pin = SPI_MASTER_PIN_MOSI; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP; GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; GPIO_Init(SPI_MASTER_GPIO, &GPIO_InitStructure);
//SPI1 NSS GPIO_StructInit(&GPIO_InitStructure); GPIO_InitStructure.GPIO_Pin = SPI_MASTER_PIN_NSS; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; GPIO_Init(SPI_MASTER_GPIO, &GPIO_InitStructure);
GPIO_SetBits(SPI_MASTER_GPIO, SPI_MASTER_PIN_NSS); }
|
另外记得使能端口时钟:
1 2 3 4 5 6
|
void (void){ /* Enable GPIO clock for SPI_MASTER */ RCC_APB2PeriphClockCmd(SPI_MASTER_GPIO_CLK | RCC_APB2Periph_AFIO, ENABLE); /* Enable SPI_MASTER Periph clock */ RCC_APB1PeriphClockCmd(SPI_MASTER_CLK, ENABLE); }
|
读写实现
ADS1118 在发送寄存器配置的同时会传输转换结果,手册中提到既支持16bits 模式也支持32bits模式。
16bits模式:

32bits模式:

DIN 接口接收寄存器配置的同时DOUT接口输出转换结果,在32bits模式下发送完寄存器配置后第二次可以发送数据0,具体可参考手册。基于此设计数据发送读取函数:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22
|
//读写寄存器16bits模式 uint16_t SPI_read_write_Reg(uint16_t CofigReg) {
getdata1=SPI_send_Byte((uint8_t)(CofigReg>>8)); getdata2=SPI_send_Byte((uint8_t)CofigReg);
getdata= (uint16_t)getdata2|((uint16_t)getdata1<<8);
return getdata; } uint8_t SPI_send_Byte(uint8_t byte) {
while(SPI_I2S_GetFlagStatus(SPI2, SPI_I2S_FLAG_TXE) == RESET){}
SPI_I2S_SendData(SPI2, byte);
while(SPI_I2S_GetFlagStatus(SPI2, SPI_I2S_FLAG_RXNE) == RESET){}
return SPI_I2S_ReceiveData(SPI2); }
|
获取 ADS1118 的片内温度需要将 ADS1118 的寄存器的第4bit修改为1,启动单次转换将第15bit写1即可:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 大专栏 STM32F103驱动ADS1118/pre> |
float ads1118_get_temperature(void) { uint16_t adc=0; float value=0; adsConfigReg.stru.NOP = DATA_VALID; adsConfigReg.stru.TS_MODE = TEMPERATURE_MODE; adsConfigReg.stru.DR = DR_8_SPS; adsConfigReg.stru.MODE = SIGNLE_SHOT; adsConfigReg.stru.OS = SINGLE_CONVER_START;
ADS1118_ENABLE;
adc = SPI_read_write_Reg(adsConfigReg.word);
//conver to temperture if(adc&0x8000) { //-xx.xxx c adc>>=2; value=(0x3fff-adc+1)*(-0.03125); } else { //+xx.xxx c adc>>=2; value=adc*0.03125; } ADS1118_DISABLE; return value; }
|
ADS1118默认开启ADC模式,通过配置寄存器的12-14bit可以选择开启哪个通道,具体可参考手册。另外关于片内温度模式时,使用的是14bit左对齐模式。而且ADS1118的转换结果中,负数使用二进制补码格式,因此需要做一个转换,每个值代表0.03125℃,测试中的SPI通信数据:

热电偶
ADS1118 可以使用差分方式连接两路热电偶温度传感器,本篇也是参考了TI官方文档 使用ADS1118进行精密热电偶测量。冷端温度读取片内温度即可,将片内温度转换成热电偶对应的电压,然后加上热端获取的电压值,反向查表即可获取热电偶温度值。实际测试时使用的是K型热电偶,从网上查到K型热电偶的温度电压对应表,进行查表:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18
|
type_k_thermo_lookup_table type_k_thermo_lookup[16] = { {-200,-5.891}, {-100,-3.554}, {0,0}, {100,4.096}, {200,8.138}, {300,12.209}, {400,16.397}, {500,20.644}, {600,24.905}, {700,29.129}, {800,33.275}, {900,37.326}, {1000,41.276}, {1100,45.119}, {1200,48.838}, {1300,52.410} };
|
测试时冷端温度一般高于室内温度2摄氏度左右,因为芯片内部发热所致,获取差分输入的代码设置如下:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40
|
uint16_t ads1118_get_differential_0_1(uint8_t PGA) { uint16_t adc=0;
ADS_InitTypeDef ConfigReg;
ConfigReg.stru.NOP = DATA_VALID; ConfigReg.stru.TS_MODE = ADC_MODE; ConfigReg.stru.DR = DR_860_SPS; ConfigReg.stru.PGA = PGA; ConfigReg.stru.MODE = SIGNLE_SHOT; ConfigReg.stru.OS = SINGLE_CONVER_START; //high ConfigReg.stru.MUX = AINPN_0_1; ConfigReg.stru.PULLUP = PULL_UP_EN; ConfigReg.stru.RESV = CONFIG_BIT_RESV; ADS1118_ENABLE; delay_us((uint32_t)1); adc = SPI_read_write_Reg(ConfigReg.word);
ADS1118_DISABLE; delay_ms(1); ADS1118_ENABLE; ConfigReg.stru.NOP = DATA_INVALID; ConfigReg.stru.TS_MODE = ADC_MODE; ConfigReg.stru.DR = DR_860_SPS; ConfigReg.stru.PGA = PGA; ConfigReg.stru.MODE = SIGNLE_SHOT; ConfigReg.stru.OS = SINGLE_CONVER_START; //high ConfigReg.stru.MUX = AINPN_0_1; ConfigReg.stru.PULLUP = PULL_UP_EN; ConfigReg.stru.RESV = CONFIG_BIT_RESV; delay_us((uint32_t)1);
//等待Dout给出数据有效信号--从高变低 while(GPIO_ReadInputDataBit(SPI_MASTER_GPIO,SPI_MASTER_PIN_MISO)); adc = SPI_read_write_Reg(ConfigReg.word); delay_us(1); ADS1118_DISABLE; return adc; }
|
实际测试时,温度精度在2摄氏度左右,并没有达到TI官方公布的精度,在硬件电路方面有待进一步提高。全部源码后续会放出。
- STM32F103驱动GT911
0x00 引脚连接: // SCL-------PB10 // SDA-------PB11 // INT--------PB1 // RST--------PB2 IIC的SCL与SDA需要接上拉电 ...
- 电赛总结(四)——波形发生芯片总结之AD9834
一.特性参数 1.2.3V~5.5V供电 2.输出频率高达37.5MHz 3.正弦波.三角波输出 4.提供相位调制和频率调制功能 5.除非另有说明,VDD = 2.3 V至5.5 V,AGND = D ...
- FLASH 存储学习-串行SPI NOR FLASH
1.1 SST25VF080B简介1.1.1 主要特性 关键点:容量.速度(时钟速度.读写速度).功耗. l 容量:8MBit: l 最高SPI时钟频率:50MHz: l 低功耗模式下电流消耗:5uA ...
- FLASH 存储学习-串行SPI nor
1.1 SST25VF080B简介 1.1.1 主要特性 关键点:容量.速度(时钟速度.读写速度).功耗. 容量:8MBit: 最高SPI时钟频率:50MHz: 低功耗模式下电流消耗:5uA,正常读模 ...
- STM32-串行SPI nor
源:FLASH 存储学习-串行SPI nor 1.1 SST25VF080B简介1.1.1 主要特性 关键点:容量.速度(时钟速度.读写速度).功耗. l 容量:8MBit: l 最高SPI时钟频率: ...
- 从FPGA搞定OV7670 VGA显示 移植到 STM32F10x TFT显示 总结及疑问(高手请进)
OV7670不愧是最便宜的摄像头了最大显示像素:640*480(在VGA显示器上显示效果还不赖,用usb模块采集显示依然显著) 第一步:VGA显示 视频图像(实时)FPGA+SDRAM+OV7670= ...
- STM32F103 USB虚拟串口 驱动例程移植
1)驱动下载及安装.目前ST公司支持WIN7版本号为:VCP_V1.3.1_Setup.exe (在官网上搜索stsw-stm32102即是了):先安装驱动后再插入USB不然安装不成功. 2)固件下载 ...
- 如何为编程爱好者设计一款好玩的智能硬件(八)——LCD1602点阵字符型液晶显示模块驱动封装(中)
六.温湿度传感器DHT11驱动封装(下):如何为编程爱好者设计一款好玩的智能硬件(六)——初尝试·把温湿度给收集了(下)! 七.点阵字符型液晶显示模块LCD1602驱动封装(上):如何为编程爱好者设计 ...
- STM32F103的11个定时器详解(转)
源:STM32F103的11个定时器详解 STM32F103系列的单片机一共有11个定时器,其中:2个高级定时器4个普通定时器2个基本定时器2个看门狗定时器1个系统嘀嗒定时器 出去看门狗定时器和系统滴 ...
随机推荐
- 简述cookies 和 session
session HTTP协议本身是无状态的,本身并不能支持服务端保存客户端的状态信息,于是,Web Server中引入了session的概念,用来保存客户端的状态信息. 1)当一个用户向服务器发送第一 ...
- C#在listview控件中显示数据库数据
一.了解listview控件的属性 view:设置为details columns:设置列 items:设置行 1.将listview的view设置为details 2.设置列属性 点击添加,添加一列 ...
- 14 微服务电商【黑马乐优商城】:day03-springcloud(Hystix,Feign)
本项目的笔记和资料的Download,请点击这一句话自行获取. day01-springboot(理论篇) :day01-springboot(实践篇) day02-springcloud(理论篇一) ...
- Spring Boot 默认的指标数据从哪来的?
了解有关 Spring Boot 默认指标及其来源的更多信息. 您是否注意到 Spring Boot 和 Micrometer 为您的应用生成的所有默认指标?如果没有 - 您可以将 actuator ...
- GSON解译Json为DTO
除了用okhttp网络库外,还用到google的gson库. 1. uti类的对象一般都用懒汉模式.这次gson也是用懒汉模式. public class GsonTools { private st ...
- HDU 6126 Give out candies(网络流)
题目给出n,m,k 然后给出n*m的矩阵a[i][j]代表第i个人在获得j 颗糖果能得到的满足值, 然后k是k行每行输入三个整数x,y,z ,x,y,z表示一组限制表示第x个人分到的糖数减去第 ...
- [原]调试实战——使用windbg调试excel启动时死锁
原调试debugwindbg死锁deadlock 前言 这是几年前在项目中遇到的一个死锁问题,在博客园发布过.我对之前的笔记进行了整理重新发布于此. 本文假设小伙伴们知道一些基本概念,比如什么是.du ...
- 玩转SpringBoot用好条件相关注解,开启自...
官方提供的常用条件注解 因为Spring的核心是基于bean的,所以这些条件注解主要是影响bean的注册. 因为注册的bean不同了,最后对外呈现的行为就不同了.不就是自动配置了. 一.最常用的应该是 ...
- C++头文件和std命名空间
C++ 是在C语言的基础上开发的,早期的 C++ 还不完善,不支持命名空间,没有自己的编译器,而是将 C++ 代码翻译成C代码,再通过C编译器完成编译.这个时候的 C++ 仍然在使用C语言的库,std ...
- Derby 命令
SHOW [ TABLES | VIEWS | PROCEDURES | FUNCTIONS | SYNONYMS ] { IN sche -- 列出表.视图.过程.函数或同义词 SHOW INDEX ...