[实例] SPI接口的ADC芯片全通道纯硬件驱动——基于HAL库和TLA2518芯片
本次需要通过TI的TL2518芯片进行ADC采样。该芯片为SPI接口具有八个通道可以全部配置成AIN进行采样本次需要探究如何该如何配置才能将芯片的采样率达到最大。1.TLA2158首先要陈列一下该芯片的一些特性为节省篇幅此处只罗列最关键的特性该芯片的详细描述请查看其手册。1.1.1寄存器读写该芯片虽然是SPI接口但是数据帧格式没有完全遵守SPI的标准格式因此配置主机的SPI时CS必须选择软件控制。以上是其读写的时序下面是我选用的SPI配置这是从某开发板的例程上抄的。至于SPI的时间频率建议选大一点因为TL2518芯片SPI接口最快可以接受30MHz的SPI_CLK。void SPI2_Init(u32 datasize) { SPI2_Handler.InstanceSPI2; //SPI2 SPI2_Handler.Init.ModeSPI_MODE_MASTER; //设置SPI工作模式设置为主模式 SPI2_Handler.Init.DirectionSPI_DIRECTION_2LINES; //设置SPI单向或者双向的数据模式:SPI设置为双线模式 SPI2_Handler.Init.DataSizedatasize; //设置SPI的数据大小:寄存器读写时8bit读数据时16bit SPI2_Handler.Init.CLKPolaritySPI_POLARITY_LOW; //串行同步时钟的空闲状态为高电平 SPI2_Handler.Init.CLKPhaseSPI_PHASE_1EDGE; //串行同步时钟的第二个跳变沿上升或下降数据被采样 SPI2_Handler.Init.NSSSPI_NSS_SOFT; //NSS信号由硬件NSS管脚还是软件使用SSI位管理:内部NSS信号有SSI位控制 SPI2_Handler.Init.BaudRatePrescalerSPI_BAUDRATEPRESCALER_256;//定义波特率预分频的值:波特率预分频值为256 SPI2_Handler.Init.FirstBitSPI_FIRSTBIT_MSB; //指定数据传输从MSB位还是LSB位开始:数据传输从MSB位开始 SPI2_Handler.Init.TIModeSPI_TIMODE_DISABLE; //关闭TI模式 SPI2_Handler.Init.CRCCalculationSPI_CRCCALCULATION_DISABLE;//关闭硬件CRC校验 SPI2_Handler.Init.CRCPolynomial7; //CRC值计算的多项式 HAL_SPI_Init(SPI2_Handler);//初始化 __HAL_SPI_ENABLE(SPI2_Handler); //使能SPI2 // SPI2_ReadWriteByte(0Xff); //启动传输 } //SPI5底层驱动时钟使能引脚配置 //此函数会被HAL_SPI_Init()调用 //hspi:SPI句柄 void HAL_SPI_MspInit(SPI_HandleTypeDef *hspi) { GPIO_InitTypeDef GPIO_Initure; __HAL_RCC_GPIOB_CLK_ENABLE(); //使能GPIOB时钟 __HAL_RCC_SPI2_CLK_ENABLE(); //使能SPI2时钟 //PB13,14,15 GPIO_Initure.PinGPIO_PIN_13|GPIO_PIN_14|GPIO_PIN_15; GPIO_Initure.ModeGPIO_MODE_AF_PP; //复用推挽输出 GPIO_Initure.PullGPIO_PULLUP; //上拉 GPIO_Initure.SpeedGPIO_SPEED_FREQ_HIGH; //快速 HAL_GPIO_Init(GPIOB,GPIO_Initure); } void SPI2_SetSpeed(u8 SPI_BaudRatePrescaler) { assert_param(IS_SPI_BAUDRATE_PRESCALER(SPI_BaudRatePrescaler));//判断有效性 __HAL_SPI_DISABLE(SPI2_Handler); //关闭SPI SPI2_Handler.Instance-CR10XFFC7; //位3-5清零用来设置波特率 SPI2_Handler.Instance-CR1|SPI_BaudRatePrescaler;//设置SPI速度 __HAL_SPI_ENABLE(SPI2_Handler); //使能SPI }1.1.2数据帧格式TL2518的ADC分辨率为12bit这意味着每次仅读回一字节数据是根本不够的你必须按照半字读回但多出来的四位也不会浪费因为该芯片可以启用ID APPEND模式在每帧数据的末尾附上所采样的通道ID。至于那个16bit的数据帧则是开启了芯片过采样这会降低你的总采样率但是却能提高单次的采样分辨率。利用ID APPEND模式我们可以在不启用CRC的前提下也能保证每次数据帧的正确性你只需要解码ID即可。以下展示一下我的芯片寄存器是如何配置的。里面的一些宏定义没有完整展示但你只要看芯片手册就能理解了建议找一下官方写的TLA2528.h头文件这样你就不要自己去定义每个寄存器了。本随笔的重点在于后面如何配置来完成纯硬件驱动SPI来达到最高采样率的ADC采样。/************************************************* * 写入一串字符 * * param void * return void * author Chanlin **************************************************/ static void TLA_WriteBytes(uint8_t bytes[],uint32_t size){ TLA_CS 0; while(size -- 0){ // printf(byte:%x\t,*bytes); TLA_SPIReadWriteByte(*(bytes)); // bytes; } TLA_CS 1; // printf(\r\n); } /************************************************* * 完成一次寄存器写入操作 * * param void * return void * author Chanlin **************************************************/ static void TLA_WriteReg(Reg addr,Data data){ // 先简单实现一下 uint8_t bytes[3]; // 设置spi frame {WR_REG,addr,data} bytes[0] WR_REG; bytes[1] addr; bytes[2] data; TLA_WriteBytes(bytes,3); // delay_us(2); } /************************************************* * 完成一次寄存器读取操作 * * param void * return void * author Chanlin **************************************************/ static void TLA_ReadReg(Reg addr,Data* data){ // 先简单实现一下 uint8_t bytes[3]; // 读取数据帧 {RD_REG,addr,DUMMY}; bytes[0] RD_REG; bytes[1] addr; bytes[2] DUMMY; // 写入读取帧 TLA_WriteBytes(bytes,3); // 读出数据 TLA_CS 0; *dataTLA_SPIReadWriteByte(DUMMY); TLA_CS 1; // 解码完成后读回数据 // *dataTLA_SPIReadWriteByte(DUMMY); } // 以下是对寄存器的配置 // 读写检查 TLA_WriteReg(GENERAL_CFG,0x01); // soft reset delay_ms(20); // wait for the reset completing TLA_ReadReg(GENERAL_CFG,data); // soft reset printf(GENERAL_CFG:%x\r\n,data); TLA_ReadReg(OSR_CFG,data); // soft reset printf(OSR_CFG:%x\r\n,data); TLA_ReadReg(SYSTEM_STATUS,data); printf(chip sys status:%x\r\n,data); if(data ! 0x81){ if(data 0xc1) printf(chip sequence is ongoing\r\n); else printf(Cannot access the chip\r\n); } // timing // TLA_WriteReg(OPMODE_CFG,0x0); // 默认高速时钟源如果你发现时钟不对或者想要修改 // pin // TLA_WriteReg(PIN_CFG,0x00); // 全部设置为 AIN(默认) // TLA_ReadReg(PIN_CFG,data); // printf(PIN_CFG:%x\r\n,data); // DATA TLA_WriteReg(DATA_CFG,0x10); // 默认无debug有ID APPEND请检查此处时序设置是否正确 TLA_ReadReg(DATA_CFG,data); printf(DATA_CFG:%x\r\n,data); // mode TLA_WriteReg(AUTO_SEQ_CH_SEL,0xFF); // 默认通道全选 TLA_ReadReg(AUTO_SEQ_CH_SEL,data); printf(SEQ_CH:%x\r\n,data); TLA_WriteReg(SEQUENCE_CFG,0x11); // 默认使用auto-sequence mode且打开 TLA_ReadReg(SEQUENCE_CFG,data); printf(SEQUENCE_CFG:%x\r\n,data); TLA_CS 1; // TLA_ReadReg(PIN_CFG,data); printf(PIN_CFG:%x\r\n,data); // 使用manual试下 // TLA_WriteReg(CHANNEL_SEL,1); // ADC offset Calib while(1){ TLA_ReadReg(GENERAL_CFG,data); // printf(ADC offset Calib:%x\r\n,data); if((data 1 0x1) 0 (data 2 0x01) 1) break; // 非常重要的一点是配完TLA2518的寄存器后不要忘记把主机的SPI改成16bit的数据帧格式 __HAL_SPI_DISABLE(SPI2_Handler); SPI2_Handler.Init.DataSize SPI_DATASIZE_16BIT; HAL_SPI_Init(SPI2_Handler);//初始化 __HAL_SPI_ENABLE(SPI2_Handler); SPI2_SetSpeed(SPI_BAUDRATEPRESCALER_2); //设置为42M时钟,高速模式1.2.1采样时间该芯片可选时钟但一般也不会在慢时钟源下运行尤其是在用于ADC模式下采样率越高越好。而该芯片最快采样率为1MHz但考虑到其有八个通道如果全开的话分配到每个通道上最快也就125KHz。1.2.2采样通道切换模式TLA2518提供了三种通道切换模式分别是Mannual、On-the-fly和Auto-Sequence模式这里仅介绍之后会用的Auto-Sequence模式其实用on-the-fly模式也能实现。在使用这一模式时你只需在最开始往寄存器中写好你要采样的通道在上面展示的配置中我把八个通道全开了。然后你需要达到三个条件才能让整个时序动起来并读到你想要的数据。1.控制CS引脚生成上升沿和下降沿2.控制SPI生成时钟如果你是主机的话3.从SPI-DR寄存器中读取数据到内存这样才能使用这三个条件放在一起时很容易联想到采用PWM控制CS引脚采用DMA来让SPI进行自动的收发最终实现整个时序。2.实现毫无疑问这里需要用的的片上外设资源包括一个定时器的通道要被配置成PWM两个DMA(一个触发源为TIM_CH一个触发源为SPI_RX)。以下是TIM的配置当然也是抄的例程。/************************************************* * * * param void * return void * author Chanlin **************************************************/ void TIM3_PWM_Init(u16 arr,u16 psc) { TIM3_Handler.InstanceTIM3; //定时器3 TIM3_Handler.Init.Prescalerpsc; //定时器分频 TIM3_Handler.Init.CounterModeTIM_COUNTERMODE_UP;//向上计数模式 TIM3_Handler.Init.Periodarr; //自动重装载值 TIM3_Handler.Init.ClockDivisionTIM_CLOCKDIVISION_DIV1; HAL_TIM_PWM_Init(TIM3_Handler); //初始化PWM TIM3_CH4Handler.OCModeTIM_OCMODE_PWM1; //模式选择PWM1 TIM3_CH4Handler.Pulsearr/2; //设置比较值,此值用来确定占空比默认比较值为自动重装载值的一半,即占空比为50% TIM3_CH4Handler.OCPolarityTIM_OCPOLARITY_HIGH; //输出比较极性为低 HAL_TIM_PWM_ConfigChannel(TIM3_Handler,TIM3_CH4Handler,TIM_CHANNEL_4);//配置TIM3通道2 SET_BIT(TIM3_Handler.Instance-DIER,TIM_DIER_CC4DE_Msk); HAL_TIM_PWM_Start(TIM3_Handler,TIM_CHANNEL_4);//开启PWM通道2 } /************************************************* * * * param void * return void * author Chanlin **************************************************/ void HAL_TIM_PWM_MspInit(TIM_HandleTypeDef *htim) { GPIO_InitTypeDef GPIO_Initure; if(htim-InstanceTIM3) { __HAL_RCC_TIM3_CLK_ENABLE(); //使能定时器3 // __HAL_AFIO_REMAP_TIM3_PARTIAL(); //TIM3通道引脚部分重映射使能 __HAL_RCC_GPIOB_CLK_ENABLE(); //开启GPIOB时钟 GPIO_Initure.PinGPIO_PIN_1; //PB1 GPIO_Initure.ModeGPIO_MODE_AF_PP; //复用推挽输出 GPIO_Initure.PullGPIO_PULLUP; //上拉 GPIO_Initure.SpeedGPIO_SPEED_FREQ_HIGH;//高速 HAL_GPIO_Init(GPIOB,GPIO_Initure); } }以下是DMA的配置这个真是我自己写的/************************************************* * 完成一次寄存器读取操作 * * param void * return void * author Chanlin **************************************************/ static void ConfigDMA(){ __HAL_RCC_DMA1_CLK_ENABLE(); //DMA1时钟使能 __HAL_LINKDMA(SPI2_Handler,hdmarx,SPIxDMA_Handler); //将DMA与SPI联系起来(发送DMA) __HAL_LINKDMA(SPI2_Handler,hdmatx,SPIxDMA_HandlerTX); //将DMA与SPI联系起来(发送DMA) //Rx DMA配置 SPIxDMA_Handler.InstanceDMA1_Channel4; //通道选择 SPIxDMA_Handler.Init.DirectionDMA_PERIPH_TO_MEMORY; //存储器到外设 SPIxDMA_Handler.Init.PeriphIncDMA_PINC_DISABLE; //外设非增量模式 SPIxDMA_Handler.Init.MemIncDMA_MINC_ENABLE; //存储器增量模式 SPIxDMA_Handler.Init.PeriphDataAlignmentDMA_PDATAALIGN_HALFWORD; //外设数据长度:8位 SPIxDMA_Handler.Init.MemDataAlignmentDMA_MDATAALIGN_HALFWORD; //存储器数据长度:8位 SPIxDMA_Handler.Init.ModeDMA_CIRCULAR; //外设循环模式 SPIxDMA_Handler.Init.PriorityDMA_PRIORITY_HIGH; //中等优先级 HAL_DMA_DeInit(SPIxDMA_Handler); HAL_DMA_Init(SPIxDMA_Handler); __HAL_DMA_ENABLE(SPIxDMA_Handler); // TX SPIxDMA_HandlerTX.InstanceDMA1_Channel3; //通道选择 SPIxDMA_HandlerTX.Init.DirectionDMA_MEMORY_TO_PERIPH; //存储器到外设 SPIxDMA_HandlerTX.Init.PeriphIncDMA_PINC_DISABLE; //外设非增量模式 SPIxDMA_HandlerTX.Init.MemIncDMA_MINC_ENABLE; //存储器增量模式 SPIxDMA_HandlerTX.Init.PeriphDataAlignmentDMA_PDATAALIGN_HALFWORD; //外设数据长度:8位 SPIxDMA_HandlerTX.Init.MemDataAlignmentDMA_MDATAALIGN_HALFWORD; //存储器数据长度:8位 SPIxDMA_HandlerTX.Init.ModeDMA_CIRCULAR; //外设循环模式 SPIxDMA_HandlerTX.Init.PriorityDMA_PRIORITY_MEDIUM; //中等优先级 HAL_DMA_DeInit(SPIxDMA_HandlerTX); HAL_DMA_Init(SPIxDMA_HandlerTX); __HAL_DMA_ENABLE(SPIxDMA_HandlerTX); if (HAL_SPI_TransmitReceive_DMA(SPI2_Handler, (uint8_t*)dummy_data, // 发送缓冲区 (uint8_t*)s_arrAINChannelVal, // 接收缓冲区 TLA2518_CHANNEL_MAX) ! HAL_OK) { // 启动失败处理 printf(SPI DMA start failed!\r\n); }

相关新闻

【全文系列目录】风控PM记

【全文系列目录】风控PM记

风控PM记 一:风险认知与识别(入门篇) ① 入门第一课:认识风险,了解风控 ② 入门第二课:业务催生风险,常见的业务风险有哪些? ③ 《电商风控入门:我们到底在“防”什…

2026/7/6 3:48:11 阅读更多 →
基于Databricks的企业级AI Agent生产实践:从架构设计到部署运维

基于Databricks的企业级AI Agent生产实践:从架构设计到部署运维

🚀 30款热门AI模型一站整合,DeepSeek/GLM/Qwen 随心用,限时 5 折。 👉 点击领海量免费额度 如果你正在考虑将AI Agent引入企业生产环境,可能会面临这样的困境:在本地开发环境中跑得飞快的Agent原型&…

2026/7/6 3:42:09 阅读更多 →
飞书卡片表格渲染踩坑记:从 Markdown 到原生 table 组件的迁移实战

飞书卡片表格渲染踩坑记:从 Markdown 到原生 table 组件的迁移实战

背景 团队每日通过飞书推送项目晨报和日报,内容从项目管理平台实时拉取,包含任务统计、进度列表、风险项等多维数据,天然需要表格来承载。 最初的实现方案是飞书消息推送 纯文本,格式简陋,阅读体验差。于是决定升级为…

2026/7/6 3:40:09 阅读更多 →

最新新闻

FinalBurn Neo:打造完美复古街机游戏体验的终极指南

FinalBurn Neo:打造完美复古街机游戏体验的终极指南

FinalBurn Neo:打造完美复古街机游戏体验的终极指南 【免费下载链接】FBNeo FinalBurn Neo - We are Team FBNeo. 项目地址: https://gitcode.com/gh_mirrors/fb/FBNeo FinalBurn Neo(简称FBNeo)是一款开源的街机游戏模拟器&#xff0…

2026/7/6 4:44:23 阅读更多 →
3个关键问题:如何通过WSC API安全管理Windows Defender?

3个关键问题:如何通过WSC API安全管理Windows Defender?

3个关键问题:如何通过WSC API安全管理Windows Defender? 【免费下载链接】no-defender A slightly more fun way to disable windows defender firewall. (through the WSC api) 项目地址: https://gitcode.com/GitHub_Trending/no/no-defender …

2026/7/6 4:44:23 阅读更多 →
珀斯与袋鼠岛之旅:波浪岩与野生海鲜市场探访

珀斯与袋鼠岛之旅:波浪岩与野生海鲜市场探访

珀斯与袋鼠岛之旅:波浪岩与野生海鲜市场探访从西澳大利亚州的首府珀斯出发,向东驱车约340公里,可抵达海登附近的波浪岩。这块巨大的花岗岩体高约15米,长度约110米,其岩石表面因长期的风化与水蚀作用,形成了…

2026/7/6 4:42:23 阅读更多 →
叶兴阳双语音标,英语发音工具断层级天花板

叶兴阳双语音标,英语发音工具断层级天花板

功能向实测评价:叶兴阳双语音标,英语发音工具断层级天花板 深耕英语学习多年,试过市面各类音标教辅、发音软件、双语读物,唯有叶兴阳双语音标在功能性上做到全方位无短板,每一项核心功能都精准戳中自学、教学、精读全场…

2026/7/6 4:38:22 阅读更多 →
Python+OpenCV 4.8 与 Tesseract OCR 5.3 车牌识别方案对比评测

Python+OpenCV 4.8 与 Tesseract OCR 5.3 车牌识别方案对比评测

PythonOpenCV 4.8 与 Tesseract OCR 5.3 车牌识别方案深度评测车牌识别技术作为计算机视觉领域的重要应用,在智能交通、停车场管理等领域发挥着关键作用。本文将深入对比两种主流车牌识别方案:基于OpenCV 4.8的传统图像处理方案和基于Tesseract OCR 5.3的…

2026/7/6 4:38:22 阅读更多 →
3分钟掌握免费Android投屏神器:scrcpy终极使用指南

3分钟掌握免费Android投屏神器:scrcpy终极使用指南

3分钟掌握免费Android投屏神器:scrcpy终极使用指南 【免费下载链接】scrcpy Display and control your Android device 项目地址: https://gitcode.com/GitHub_Trending/sc/scrcpy 还在为手机屏幕太小而烦恼?想要在电脑大屏幕上操作手机应用&…

2026/7/6 4:36:22 阅读更多 →

日新闻

H2 与 MySQL 单元测试兼容性:5 个关键 SQL 语句差异与规避方案

H2 与 MySQL 单元测试兼容性:5 个关键 SQL 语句差异与规避方案

H2与MySQL单元测试兼容性:5个关键SQL语句差异与规避方案1. 单元测试中的数据库兼容性挑战在Java开发领域,单元测试是保证代码质量的重要环节。当应用涉及数据库操作时,测试环境的搭建往往成为开发者的痛点。H2数据库因其轻量级、内存模式和快…

2026/7/6 0:01:17 阅读更多 →
Windows任务栏终极清理指南:用RBTray一键隐藏窗口到系统托盘

Windows任务栏终极清理指南:用RBTray一键隐藏窗口到系统托盘

Windows任务栏终极清理指南:用RBTray一键隐藏窗口到系统托盘 【免费下载链接】rbtray A fork of RBTray from http://sourceforge.net/p/rbtray/code/. 项目地址: https://gitcode.com/gh_mirrors/rb/rbtray 你是否厌倦了Windows任务栏上密密麻麻的图标&…

2026/7/6 0:01:17 阅读更多 →
Visual C++ 运行时库一键安装终极指南:告别DLL缺失烦恼

Visual C++ 运行时库一键安装终极指南:告别DLL缺失烦恼

Visual C 运行时库一键安装终极指南:告别DLL缺失烦恼 【免费下载链接】vcredist AIO Repack for latest Microsoft Visual C Redistributable Runtimes 项目地址: https://gitcode.com/gh_mirrors/vc/vcredist 你是否曾经遇到过这样的情况:下载了…

2026/7/6 0:05:19 阅读更多 →

周新闻

B站视频下载神器BiliTools:5分钟学会轻松保存任何B站内容

B站视频下载神器BiliTools:5分钟学会轻松保存任何B站内容

B站视频下载神器BiliTools:5分钟学会轻松保存任何B站内容 【免费下载链接】BiliTools A cross-platform bilibili toolbox. 跨平台哔哩哔哩工具箱,支持下载视频、番剧等等各类资源 项目地址: https://gitcode.com/GitHub_Trending/bilit/BiliTools …

2026/7/5 0:03:34 阅读更多 →
威胁模型全解析:从新手入门到实战应用,助你构建安全产品!

威胁模型全解析:从新手入门到实战应用,助你构建安全产品!

威胁模型的陌生现状在忙碌疲惫的一天里,参与了关于混合后量子密码学的讨论,应付端点攻击找茬的人,还参与留言板讨论后,发现“威胁模型”对多数人仍是陌生概念,且多被当作时髦用语。有趣的相关画作有一幅由 Embyr 创作的…

2026/7/5 0:03:34 阅读更多 →
渗透测试入门指南:从零基础到实战环境搭建

渗透测试入门指南:从零基础到实战环境搭建

1. 从“看热闹”到“入门”:我理解的渗透测试到底是什么?每次看到新闻里说某个大公司的数据被“黑”了,或者某个网站被攻击导致服务瘫痪,你是不是和我一样,心里会冒出两个念头:一是“这黑客真厉害”&#x…

2026/7/5 0:07:38 阅读更多 →

月新闻