news 2026/6/26 5:06:04

[实例] SPI接口的ADC芯片全通道纯硬件驱动——基于HAL库和TLA2518芯片0.概述

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
[实例] SPI接口的ADC芯片全通道纯硬件驱动——基于HAL库和TLA2518芯片0.概述

寄存器读写

该芯片虽然是SPI接口,但是数据帧格式没有完全遵守SPI的标准格式,因此配置主机的SPI时,CS必须选择软件控制。

以上是其读写的时序,下面是我选用的SPI配置,这是从某开发板的例程上抄的。至于SPI的时间频率,建议选大一点,因为TL2518芯片SPI接口最快可以接受30MHz的SPI_CLK。

void SPI2_Init(u32 datasize) { SPI2_Handler.Instance=SPI2; //SPI2 SPI2_Handler.Init.Mode=SPI_MODE_MASTER; //设置SPI工作模式,设置为主模式 SPI2_Handler.Init.Direction=SPI_DIRECTION_2LINES; //设置SPI单向或者双向的数据模式:SPI设置为双线模式 SPI2_Handler.Init.DataSize=datasize; //设置SPI的数据大小:寄存器读写时8bit;读数据时16bit SPI2_Handler.Init.CLKPolarity=SPI_POLARITY_LOW; //串行同步时钟的空闲状态为高电平 SPI2_Handler.Init.CLKPhase=SPI_PHASE_1EDGE; //串行同步时钟的第二个跳变沿(上升或下降)数据被采样 SPI2_Handler.Init.NSS=SPI_NSS_SOFT; //NSS信号由硬件(NSS管脚)还是软件(使用SSI位)管理:内部NSS信号有SSI位控制 SPI2_Handler.Init.BaudRatePrescaler=SPI_BAUDRATEPRESCALER_256;//定义波特率预分频的值:波特率预分频值为256 SPI2_Handler.Init.FirstBit=SPI_FIRSTBIT_MSB; //指定数据传输从MSB位还是LSB位开始:数据传输从MSB位开始 SPI2_Handler.Init.TIMode=SPI_TIMODE_DISABLE; //关闭TI模式 SPI2_Handler.Init.CRCCalculation=SPI_CRCCALCULATION_DISABLE;//关闭硬件CRC校验 SPI2_Handler.Init.CRCPolynomial=7; //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.Pin=GPIO_PIN_13|GPIO_PIN_14|GPIO_PIN_15; GPIO_Initure.Mode=GPIO_MODE_AF_PP; //复用推挽输出 GPIO_Initure.Pull=GPIO_PULLUP; //上拉 GPIO_Initure.Speed=GPIO_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->CR1&=0XFFC7; //位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; *data=TLA_SPIReadWriteByte(DUMMY); TLA_CS = 1; // 解码完成后,读回数据 // *data=TLA_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.Instance=TIM3; //定时器3 TIM3_Handler.Init.Prescaler=psc; //定时器分频 TIM3_Handler.Init.CounterMode=TIM_COUNTERMODE_UP;//向上计数模式 TIM3_Handler.Init.Period=arr; //自动重装载值 TIM3_Handler.Init.ClockDivision=TIM_CLOCKDIVISION_DIV1; HAL_TIM_PWM_Init(&TIM3_Handler); //初始化PWM TIM3_CH4Handler.OCMode=TIM_OCMODE_PWM1; //模式选择PWM1 TIM3_CH4Handler.Pulse=arr/2; //设置比较值,此值用来确定占空比,默认比较值为自动重装载值的一半,即占空比为50% TIM3_CH4Handler.OCPolarity=TIM_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->Instance==TIM3) { __HAL_RCC_TIM3_CLK_ENABLE(); //使能定时器3 // __HAL_AFIO_REMAP_TIM3_PARTIAL(); //TIM3通道引脚部分重映射使能 __HAL_RCC_GPIOB_CLK_ENABLE(); //开启GPIOB时钟 GPIO_Initure.Pin=GPIO_PIN_1; //PB1 GPIO_Initure.Mode=GPIO_MODE_AF_PP; //复用推挽输出 GPIO_Initure.Pull=GPIO_PULLUP; //上拉 GPIO_Initure.Speed=GPIO_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.Instance=DMA1_Channel4; //通道选择 SPIxDMA_Handler.Init.Direction=DMA_PERIPH_TO_MEMORY; //存储器到外设 SPIxDMA_Handler.Init.PeriphInc=DMA_PINC_DISABLE; //外设非增量模式 SPIxDMA_Handler.Init.MemInc=DMA_MINC_ENABLE; //存储器增量模式 SPIxDMA_Handler.Init.PeriphDataAlignment=DMA_PDATAALIGN_HALFWORD; //外设数据长度:8位 SPIxDMA_Handler.Init.MemDataAlignment=DMA_MDATAALIGN_HALFWORD; //存储器数据长度:8位 SPIxDMA_Handler.Init.Mode=DMA_CIRCULAR; //外设循环模式 SPIxDMA_Handler.Init.Priority=DMA_PRIORITY_HIGH; //中等优先级 HAL_DMA_DeInit(&SPIxDMA_Handler); HAL_DMA_Init(&SPIxDMA_Handler); __HAL_DMA_ENABLE(&SPIxDMA_Handler); // TX SPIxDMA_HandlerTX.Instance=DMA1_Channel3; //通道选择 SPIxDMA_HandlerTX.Init.Direction=DMA_MEMORY_TO_PERIPH; //存储器到外设 SPIxDMA_HandlerTX.Init.PeriphInc=DMA_PINC_DISABLE; //外设非增量模式 SPIxDMA_HandlerTX.Init.MemInc=DMA_MINC_ENABLE; //存储器增量模式 SPIxDMA_HandlerTX.Init.PeriphDataAlignment=DMA_PDATAALIGN_HALFWORD; //外设数据长度:8位 SPIxDMA_HandlerTX.Init.MemDataAlignment=DMA_MDATAALIGN_HALFWORD; //存储器数据长度:8位 SPIxDMA_HandlerTX.Init.Mode=DMA_CIRCULAR; //外设循环模式 SPIxDMA_HandlerTX.Init.Priority=DMA_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"); } }

在完成这些配置后,只需要控制TIM3输出的PWM的频率和占空比即可完成全自动的收发。考虑芯片的采样时钟特性,建议每周期1us以下,占空比进行调整,让每周期的CS低电平时间在50-100ns,高电平稍微占比多一点。

分类: 嵌入式 / 实例

免责声明:本内容来自平台创作者,博客园系信息发布平台,仅提供信息存储空间服务。

好文要顶 关注我 收藏该文 微信分享

Chanlin
粉丝 - 1 关注 - 0

+加关注

0

0

升级成为会员

« 上一篇: 【EXMC】简介

posted on 2026-05-11 23:20 Chanlin 阅读(106) 评论(0) 收藏 举报

刷新页面返回顶部

登录后才能查看或发表评论,立即 登录 或者 逛逛 博客园首页

【推荐】 凌霞 618 年中大促,Halo 与 1Panel 产品全线半价,叠加满减!
【推荐】HarmonyOS 6.1.0 创新特性“悬浮页签+沉浸光感”精品文章专题
【推荐】科研领域的连接者艾思科蓝,一站式科研学术服务数字化平台

编辑推荐:

导航

<2026年6月>
31123456
78910111213
14151617181920
21222324252627
2829301234
567891011

公告

昵称: Chanlin
园龄: 2年9个月
粉丝: 1
关注: 0

+加关注

搜索

常用链接

随笔分类

随笔档案

阅读排行榜

博客园 © 2004-2026

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/6/26 5:05:57

0.8VIN,外置MOS,XZ5120,升压LED驱动芯片

概述 这是一款异步升压驱动LED的DC-DC转换器&#xff0c;输入电压范围宽&#xff1a;0.8-5.5V&#xff0c;可以设置可调恒定电流或者限定电流两种模式。适用于单节或者双节干电池驱动单颗大功率LED&#xff0c;或者是单节锂电驱动2颗&#xff0c;3颗多颗WLED串联。LED电流由外部…

作者头像 李华
网站建设 2026/6/26 5:01:29

AI Agent 在甘特计划变更场景中的动态响应工程实践

计划赶不上变化&#xff0c;这句话在项目管理里几乎是公理。一个在启动会上被所有人认可的甘特图&#xff0c;往往在项目进行到 30% 的时候就已经面目全非。任务延期、需求插入、人员变动、外部依赖未就绪——每一类变更都可能引发连锁反应&#xff0c;影响远比直觉判断的要深。…

作者头像 李华
网站建设 2026/6/26 5:00:32

优学宝一体化知识付费与在线刷题系统全新上线,兼容 Word、Excel 题库一键导入,内置 AI 大模型实现试题智能识别分类。系统对注册用户免费开放,涵盖题库共享、协同刷题、全真模拟测验、多题型分类练

历经数月开发、功能打磨与多轮压力测试&#xff0c;优学宝一体化知识付费与在线刷题网页系统现已正式上线运营&#xff01; 本系统是专为个人讲师、中小教培机构、知识博主量身打造的全功能知识付费解决方案&#xff0c;依托网页端架构搭建&#xff0c;无需下载、无需安装、跨…

作者头像 李华
网站建设 2026/6/26 4:56:36

提前打磨答辩PPT,全力冲刺国家科学技术奖

国家科学技术奖是我国科技领域最高层级荣誉&#xff0c;评审答辩是决定申报成败的关键环节&#xff0c;答辩 PPT 作为成果可视化核心载体&#xff0c;绝非临时拼凑、仓促制作即可达标&#xff0c;必须提前统筹、精工打磨&#xff0c;核心原因分为五大维度&#xff1a; 1、评审标…

作者头像 李华