当前位置: 首页 > news >正文

DMA发送全部历史记录数据到串口

背景

博主参与的项目中,有个读取全部历史记录的功能,如果下位机在主程序中将全部历史记录单纯地通过串口传输会比较占用cpu资源,影响主程序中别的功能。最后商量得出以下实现方案:

定义两个发送缓冲区DMATxbuf1和DMATxbuf2,以及这两个发送缓冲区的标志DMATxbuf1Flag和DMATxbuf2Flag,发送缓冲区的标志取值有三种:可搬运、可发送和发送中。主main中负责搬运数据到这两个发送缓冲区中以及如果存在可发送的缓冲区且不存在发送中的缓冲区,那么主main开启一下这个DMA的通道使能。在DMA传输完成中断中,需要更新刚刚发送缓冲区的标志即可。

相关代码

初始化串口和DMA

void InitDebug_Sub(void)
{
//  stc_gpio_cfg_t stcGpioCfg;
//  stc_uart_cfg_t    stcCfg;/***********************RAM口初始化******************************************/Rxput_232 = 0;Rxget_232 = 0;Txput_232 = 0;Txget_232 = 0;TXing_232 = 0;/***********************GPIO口初始化**********************************************/      stc_gpio_cfg_t stcGpioCfg;DDL_ZERO_STRUCT(stcGpioCfg);Sysctrl_SetPeripheralGate(SysctrlPeripheralGpio,TRUE);/***********TX************/stcGpioCfg.enDir =  GpioDirOut;Gpio_Init(GpioPortA,GpioPin0,&stcGpioCfg);Gpio_SetAfMode(GpioPortA,GpioPin0,GpioAf2); //配置PC10为LPUART1_TX/**********RX**************/stcGpioCfg.enDir =  GpioDirIn;Gpio_Init(GpioPortA,GpioPin1,&stcGpioCfg);Gpio_SetAfMode(GpioPortA,GpioPin1,GpioAf2); //配置PC11为LPUART1_RX/***********************串口初始化**********************************************/    stc_lpuart_cfg_t  stcCfg;DDL_ZERO_STRUCT(stcCfg);Sysctrl_SetPeripheralGate(SysctrlPeripheralLpUart1,TRUE);  //<外设模块时钟使能  stcCfg.enStopBit = LPUart1bit;                   ///<1停止位    stcCfg.enMmdorCk = LPUartEven;                   ///<偶校验stcCfg.stcBaud.enSclkSel = LPUartMskPclk;        ///<传输时钟源stcCfg.stcBaud.u32Sclk = Sysctrl_GetPClkFreq();  ///<PCLK获取stcCfg.stcBaud.enSclkDiv = LPUartMsk4Or8Div;     ///<采样分频stcCfg.stcBaud.u32Baud = 38400;                   ///<波特率stcCfg.enRunMode = LPUartMskMode1;               ///<工作模式1,异步模式全双工LPUart_Init(M0P_LPUART1, &stcCfg);/***********************DMA初始化**********************************************/    Sysctrl_SetPeripheralGate(SysctrlPeripheralDma,TRUE); //开启DMA外设时钟stc_dma_cfg_t dmaCfg;//CONFB配置dmaCfg.enMode = DmaMskBlock;  //Block传输dmaCfg.enTransferWidth = DmaMsk8Bit; // 传输宽度 8bit(适用于 UART)dmaCfg.enSrcAddrMode = DmaMskSrcAddrInc;  // 源地址递增dmaCfg.enDstAddrMode = DmaMskDstAddrFix;  // 目标地址固定dmaCfg.enSrcAddrReloadCtl = DmaMskSrcAddrReloadEnable;//使能源地址重载dmaCfg.enDestAddrReloadCtl = DmaMskDstAddrReloadEnable;//使能目标地址重载dmaCfg.enSrcBcTcReloadCtl = DmaMskBcTcReloadEnable; //使能BC/TC重载dmaCfg.enTransferMode = DmaMskOneTransfer;  // 单次传输模式//CONFA配置dmaCfg.u16BlockSize = 1;  // 每次传输 1 字节dmaCfg.u16TransferCnt = 131;  //实际传输的时候会设置长度dmaCfg.enRequestNum = DmaLpUart1TxTrig;                //触发源配置:LPUART1 SBUF为空dmaCfg.u32SrcAddress = (uint32_t)&DMATxbuf1.DATA[0];      // 源地址:发送缓冲区dmaCfg.u32DstAddress = (uint32_t)&M0P_LPUART1->SBUF; // 目标地址:LPUART1 发送寄存器dmaCfg.enPriority = DmaMskPriorityFix;  // 固定优先级Dma_InitChannel(DmaCh1, &dmaCfg);     //初始化DMA通道Dma_EnableChannelIrq(DmaCh1); //使能DMA传输完成中断NVIC_EnableIRQ(DMAC_IRQn);    //使能NVIC DMA中断sendingHistoryRemaingTime = 0; //初始化发送历史记录剩余时间为0carryHistoryFinishFlag = 1;   //初始化不需要搬运历史记录DMATxbuf1Flag = 1;            //初始化DMA缓冲区1可搬运DMATxbuf2Flag = 1;            //初始化DMA缓冲区2可搬运/**********************LPUART 中断使能*********************************/LPUart_ClrStatus(M0P_LPUART1,LPUartRC);          //<清接收中断请求LPUart_ClrStatus(M0P_LPUART1,LPUartTC);          //<清发送中断请求LPUart_EnableIrq(M0P_LPUART1,LPUartRxIrq);       //<使能接收中断LPUart_DisableIrq(M0P_LPUART1,LPUartTxIrq);      //发送完成之后关闭发送中断//  LPUart_EnableIrq(M0P_LPUART1,LPUartTxIrq);       //<不使能发送中断,EnableNvic(LPUART1_IRQn,IrqLevel3,TRUE);         //<系统中断使能LPUart_EnableFunc(M0P_LPUART1,LPUartDmaTxFunc);  //DMA发送LPUART使能P_RD_DBG = 0;//初始化调试口485方向为接收

收到获取全部指令时

  case 12:      //读取全部历史记录sendingHistoryRemaingTime = 2000; //设置发送超时时间,同时也标记着此时在传输历史及记录FlashReadSP = W25QX_RECSTART_ADDR; //初始化历史记录读取地址为开始地址//重置两个缓冲区状态为可搬运DMATxbuf1Flag = 1;DMATxbuf2Flag = 1;carryHistoryFinishFlag = 0;//重置历史记录没有搬运完break;

主main中的搬运、发送历史数据

  //发送历史记录期间且历史记录没有搬运完if(sendingHistoryRemaingTime){//尝试搬运历史记录到缓冲区if(!carryHistoryFinishFlag){carryHistory();}//尝试并触发DMA发送dmaSendHistoryRecord();}

搬运、发送历史数据方法详情

void carryHistory()
{typedef_DMATXBUF* dmaTxbuf;u8 carryBufType = 0; //记录搬的是哪个缓冲区//选择可搬运的缓冲区if(DMATxbuf1Flag == 1){dmaTxbuf = &DMATxbuf1;carryBufType = 1;}else if(DMATxbuf2Flag == 1){dmaTxbuf = &DMATxbuf2;carryBufType = 2;}else{//目前没有可搬运的return;}u8 count = 0;dmaTxbuf->LEN = 0;while(count < 3 && !carryHistoryFinishFlag){u8 cmd2=0;u8 buf[300];structHistory_TypeDef* ptr = (structHistory_TypeDef*)buf;//历史记录已经读完了if(FlashReadSP>=W25QX_RECEND_ADDR){cmd2 = 0x01;FlashReadSP = W25QX_RECSTART_ADDR;     //首条记录carryHistoryFinishFlag = 1;}W25QX_BufferRead((u8*)&ptr->HistoryData,FlashReadSP,sizeof(structEEP_GET_HISTOR_TypeDef)); //读取记录if(ptr->HistoryData.FLAG != 0x7e){FlashReadSP += 4096;FlashReadSP &= 0xfffff000;W25QX_BufferRead((u8*)&ptr->HistoryData,FlashReadSP,sizeof(structEEP_GET_HISTOR_TypeDef)); //读取记录索引if(ptr->HistoryData.FLAG != 0x7e) {cmd2 = 0x01;FlashReadSP = W25QX_RECSTART_ADDR;     //首条记录carryHistoryFinishFlag = 1;}}ptr->CMD1 = 0x8c;ptr->CMD2 = cmd2;ptr->ADDR = 0x01;ptr->SoftVer = SOFTVER;      ptr->checksum = CRC16(buf,sizeof(structHistory_TypeDef)-2);dmaTxbuf->DATA[dmaTxbuf->LEN++] = 0x10;dmaTxbuf->DATA[dmaTxbuf->LEN++] = 0x02;for(u16 i = 0; i < sizeof(structHistory_TypeDef); i++){u8 qq = buf[i];dmaTxbuf->DATA[dmaTxbuf->LEN++] = qq;/* -----0x10加发一个 -----*/if(qq == DLE) dmaTxbuf->DATA[dmaTxbuf->LEN++] = DLE;}dmaTxbuf->DATA[dmaTxbuf->LEN++] = DLE;dmaTxbuf->DATA[dmaTxbuf->LEN++] = ETX;dmaTxbuf->DATA[dmaTxbuf->LEN++] = 0x16; count++;FlashReadSP += 128;}//说明搬运了一些历史记录,需要发送,更新标记if(count > 0){//更新缓冲区1的标志if(carryBufType == 1){DMATxbuf1Flag = 2;//缓冲区1可发送}if(carryBufType == 2){DMATxbuf2Flag = 2;//缓冲区2可发送}//说明还有数据待发送,更新过期时间sendingHistoryRemaingTime = 2000;}
}void dmaSendHistoryRecord()
{//存在发送中的数据if(DMATxbuf1Flag == 3 || DMATxbuf2Flag == 3){return;}if(DMATxbuf1Flag == 2){UART_DMA_Send_History(1);}else if(DMATxbuf2Flag == 2){UART_DMA_Send_History(2);}
}

DMA发送中断处理函数

void DMAC_IRQHandler(void)
{
#if (INT_CALLBACK_ON == INT_CALLBACK_DMAC)    Dmac_IRQHandler();
#endif // DMA 中断处理代码if (M0P_DMAC->CONFB1_f.STAT == 5)  // 传输完成标志101{// 清除传输完成标志M0P_DMAC->CONFB1_f.STAT = 0;  //更新缓冲区标志if(DMATxbuf1Flag == 3){DMATxbuf1Flag = 1;}else if (DMATxbuf2Flag == 3){DMATxbuf2Flag = 1;}}M0P_DMAC->CONFB1_f.STAT = 0; 
}

相关文章:

DMA发送全部历史记录数据到串口

背景 博主参与的项目中&#xff0c;有个读取全部历史记录的功能&#xff0c;如果下位机在主程序中将全部历史记录单纯地通过串口传输会比较占用cpu资源&#xff0c;影响主程序中别的功能。最后商量得出以下实现方案&#xff1a; 定义两个发送缓冲区DMATxbuf1和DMATxbuf2&…...

蓝桥杯好题推荐-----高精度减法

&#x1f308;个人主页&#xff1a;羽晨同学 &#x1f4ab;个人格言:“成为自己未来的主人~” 题目链接 记录详情 - 洛谷 | 计算机科学教育新生态https://www.luogu.com.cn/record/205122671 思路讲解 这个题目的解题思路&#xff0c;其实是和高精度加法是非常像的。怎么说…...

SpringMVC (3)

目录 1. 传递对象 2. 后端参数重命名&#xff08;后端参数映射&#xff09; 3. 传递数组 4. 传递集合 5. 传递JSON数据 5.1 JSON概念 5.2 JSON语法 5.3 JSON字符串和Java对象互转 5.4 JSON优点 5.5 传递JSON对象 6. 获取URL中参数PathVariable 7. 上传文件RequestP…...

vscode使用豆包MARSCode----集成doubao1.5 DeepSeekR1 DeepseekV3模型的ai编程插件

引入扩展 打开VSCode扩展窗口&#xff0c;在搜索窗口搜索MarsCode&#xff0c;找到MarsCode 插件单击「install」&#xff0c;完成安装&#xff0c;登录即可使用MarsCode 编程助手。 主要功能 主要快捷键 / explain 解释项目代码&#xff0c;AI 返回的内容有结构分类&#…...

Ubuntu 下 nginx-1.24.0 源码分析 - ngx_buf_t

ngx_buf_t 定义在 src/core/ngx_buf.h typedef struct ngx_buf_s ngx_buf_t;struct ngx_buf_s {u_char *pos;u_char *last;off_t file_pos;off_t file_last;u_char *start; /* start of buffer */u_char …...

分布式开源协调服务之zookeeper

Zookeeper简介 Zookeeper是什么&#xff1f; Zookeeper官网中对Zookeeper的定义还是比较明确的&#xff1a; ZooKeeper is a centralized service for maintaining configuration information, naming, providing distributed synchronization, and providing group services…...

ubuntu系统安装playhouse三方库

ubuntu系统python3.10安装playhouse三方库 问题描述 问题描述 虚拟环境中使用pip install playhouse&#xff0c;返回安装成功 用pip list查看&#xff0c;能看到playhouse及版本号 导包时提示没有找到playhouse我那件目录&#xff0c;能发现 检查site-package发现问题&#x…...

【星云 Orbit-F4 开发板】04.一触即发:GPIO 外部中断

【星云 Orbit-F4 开发板】04. 一触即发&#xff1a;外部中断控制 摘要 本文详细介绍了如何使用STM32F407微控制器的HAL库实现外部中断功能。通过配置GPIO引脚作为外部中断源&#xff0c;并在中断回调函数中处理按键事件&#xff0c;实现了按键控制LED状态翻转的功能。本文旨在…...

笔记二:整数和浮点数在内存中存储

目录 一、数据类型介绍 二、类型的基本归类 1.整形家族&#xff1a; 2.浮点数家族&#xff1a; 3.构造类型&#xff1a; 4.指针类型 5.空类型&#xff1a; 三、整形在内存中的存储 3.1 原码&#xff0c;反码、补码 3.2 大小端介绍 四、浮点数在内存中的存储 ​编辑 4.…...

PyQT(PySide)的上下文菜单策略设置setContextMenuPolicy()

在 Qt 中&#xff0c;QWidget 类提供了几种不同的上下文菜单策略&#xff0c;这些策略通过 Qt::ContextMenuPolicy 枚举类型来定义&#xff0c;用于控制控件&#xff08;如按钮、文本框等&#xff09;在用户右键点击时如何显示上下文菜单。 以下是 Qt::ContextMenuPolicy 枚举中…...

BladeX框架接口请求跨域

前端使用代理请求接口&#xff0c;接口可以正常访问。如果换全路径请求就跨域。 除了后端要配置跨域 还需要修改配置文件对OPTIONS请求的限制...

如何在Apple不再支持的MacOS上安装Homebrew

手头有一台2012年产的Macbook Pro&#xff0c;系统版本停留在了10.15.7&#xff08;2020年9月24日发布的&#xff09;。MacOS 11及后续的版本都无法安装到这台老旧的电脑上。想通过pkg安装Homebrew&#xff0c;发现Homebrew releases里最新的pkg安装包不支持MacOS 10.15.7&…...

本地大模型编程实战(26)用langgraph实现基于SQL数据构建的问答系统(5)

本文将将扩展上一篇文章完成的 langgraph 链&#xff0c;继续使用基于 langgraph 链 &#xff0c;对结构化数据库 SQlite 进行查询的方法。该系统建立以后&#xff0c;我们不需要掌握专业的 SQL 技能&#xff0c;可以用自然语言询问有关数据库中数据的问题并返回答案。主要完善…...

数据结构与算法:滑动窗口

前言 滑动窗口一般主要用于解决子数组或子串问题&#xff0c;这类的题目更看重对题目的分析和转化。 一、原理 在整个数组上&#xff0c;用l和r分别控制窗口的左右边界&#xff0c;r就扩大&#xff0c;l就减小。 当窗口的范围和题目中某个指标间存在单调关系时&#xff0c;…...

江协科技/江科大-51单片机入门教程——P[2-1] 点亮一个LED

本节将向大家介绍如何用 51 单片机去控制开发板上的 LED。开发板上的 LED 位置标注有 “LED 模块”。 第二章要写 3 个程序代码&#xff1a;第一个代码实现点亮开发板上的第一个 LED&#xff1b;第二个代码让第一个 LED 以 1 秒为周期闪烁&#xff1b;第三个代码使 8 个 LED 以…...

leetcode hot 100 41. 缺失的第一个正数

代码 测试用例 测试用例 测试结果 41. 缺失的第一个正数 已解答 困难 相关标签 相关企业 提示 给你一个未排序的整数数组 nums &#xff0c;请你找出其中没有出现的最小的正整数。 请你实现时间复杂度为 O(n) 并且只使用常数级别额外空间的解决方案。 示例 1&#xf…...

UniApp 使用 u-loadmore 完整步骤

文章目录 一、前期准备1. 安装 uView - UI 二、使用 u-loadmore组件1. 创建页面2. 编写页面代码模板部分&#xff08;loadmore-demo.vue&#xff09;样式部分脚本部分 三、要点补充1. u-loadmore 状态说明2. 数据请求优化3. 性能优化4. 兼容性问题 在 UniApp 开发中&#xff0c…...

设置电脑一接通电源就主动开机

文章目录 1、进入BIOS2、设置4、功能弊端5、电脑自动开机的设置 1、进入BIOS 在电脑重启时&#xff0c;这时屏幕上会显示按XXX键到BIOS界面 没有进入BIOS提示的&#xff0c;按下面方法操作&#xff1a; 方法一 在开机显示logo的时候&#xff0c;立即按下面这几个按键&#xf…...

优艾智合机器人日本子公司成立,加速推进国际化布局

2月27日&#xff0c;工业移动机器人解决方案商优艾智合宣布日本子公司Youibot Robotics Japan株式会社&#xff08;以下简称“Youibot Japan”&#xff09;成立&#xff0c;并于东京举行开业典礼。此举标志着优艾智合在日本市场的现地服务能力进一步深化&#xff0c;是其全球化…...

自然语言处理NLP入门 -- 第七节预训练语言模型

1 什么是预训练模型&#xff1f; 在自然语言处理&#xff08;NLP&#xff09;里&#xff0c;训练一个好模型通常需要很多数据和计算资源。为了解决这个难题&#xff0c;就出现了“预训练模型”。 预训练模型 是指我们先在海量文本&#xff08;比如网络上爬到的大量文章、对话…...

OpenClaw环境隔离方案:用Docker部署Qwen3-4B-Thinking-2507-GPT-5-Codex-Distill-GGUF避免依赖冲突

OpenClaw环境隔离方案&#xff1a;用Docker部署Qwen3-4B-Thinking-2507-GPT-5-Codex-Distill-GGUF避免依赖冲突 1. 为什么需要Docker环境隔离 去年我在本地尝试部署OpenClaw时&#xff0c;最头疼的问题就是Python依赖冲突。当时为了同时运行OpenClaw和一个本地大模型&#xf…...

Ostrakon-VL集成VSCode Codex:智能代码辅助下的视觉应用开发

Ostrakon-VL集成VSCode Codex&#xff1a;智能代码辅助下的视觉应用开发 1. 开篇&#xff1a;当视觉AI遇上智能编程助手 想象一下这样的开发场景&#xff1a;你正在构建一个基于Ostrakon-VL的视觉分析应用&#xff0c;需要处理摄像头采集的图像数据。传统方式下&#xff0c;你…...

Ostrakon-VL-8B在教育领域的应用:实现AI驱动的自动化作业批改与反馈

Ostrakon-VL-8B在教育领域的应用&#xff1a;实现AI驱动的自动化作业批改与反馈 1. 引言 想象一下&#xff0c;一位中学数学老师&#xff0c;晚上十点还在台灯下批改着两个班级、近百份的作业。每一份作业都需要仔细检查解题步骤是否正确、逻辑是否清晰、答案是否准确。这不仅…...

OpenClaw技能扩展指南:为Qwen3-4B-Thinking添加公众号发布模块

OpenClaw技能扩展指南&#xff1a;为Qwen3-4B-Thinking添加公众号发布模块 1. 为什么需要公众号发布技能 上周我尝试用OpenClaw自动整理技术文档时&#xff0c;突然想到个痛点&#xff1a;每次写完文章都要手动复制到公众号后台&#xff0c;调整格式、上传封面、设置摘要&…...

OpenClaw+百川2-13B量化模型:个人知识库自动整理实战指南

OpenClaw百川2-13B量化模型&#xff1a;个人知识库自动整理实战指南 1. 为什么需要自动化知识管理 作为一名独立研究者&#xff0c;我常年被两个问题困扰&#xff1a;一是收集的文献资料散落在不同文件夹&#xff0c;每次找文件都要经历"考古式搜索"&#xff1b;二…...

【工业C# OPC UA配置黄金法则】:20年资深工程师亲授5大避坑指南与一键式配置模板

第一章&#xff1a;工业C# OPC UA配置黄金法则总览在工业自动化系统中&#xff0c;C# 与 OPC UA 的集成必须兼顾安全性、可维护性与实时性。配置不当不仅导致通信中断&#xff0c;更可能引发证书信任链失效、节点访问越权或会话超时风暴。以下核心原则构成稳定部署的基石。强制…...

学术PDF处理术:OpenClaw+Qwen3-32B实现论文关键图表提取

学术PDF处理术&#xff1a;OpenClawQwen3-32B实现论文关键图表提取 1. 为什么需要自动化PDF图表提取 作为一名经常需要阅读大量学术论文的研究者&#xff0c;我长期被一个问题困扰&#xff1a;如何高效地从PDF论文中提取关键图表和数据。传统方法要么依赖手动截图和转录&…...

stock-sdk-mcp 的实践整理侗

一、什么是urllib3&#xff1f; urllib3 是一个用于处理 HTTP 请求和连接池的强大、用户友好的 Python 库。 它可以帮助你&#xff1a; 发送各种 HTTP 请求&#xff08;GET, POST, PUT, DELETE等&#xff09;。 管理连接池&#xff0c;提高网络请求效率。 处理重试和重定向。 支…...

双轴卷取分切机程序,PLC和触摸屏使用西门子smart200系列。 前后卷取双轴张力控制计算

双轴卷取分切机程序&#xff0c;PLC和触摸屏使用西门子smart200系列。 前后卷取双轴张力控制计算。 利用变频器模拟量输出控制张力。 卷取版型较好。 内部张力梯度算法理解后可用于恒张力卷取设备。 程序有完整注释&#xff0c;完整的设备图纸&#xff0c;方便理解阅读。 只包含…...

【2026技术实战】Claude Code编程神器:weelinking中转站部署完全指南

引言&#xff1a;为什么Claude Code成为开发者新宠&#xff1f; 随着AI技术的快速发展&#xff0c;国内开发者对AI工具的关注已不再局限于ChatGPT。Anthropic公司推出的Claude系列模型&#xff0c;特别是其编程增强版本Claude Code&#xff0c;正凭借卓越的逻辑推理和代码生成…...