会。sscom是一款用于串口调试的工具软件,串口数据传输过快时,接收数据软件出现“卡顿”情况,会继续接受数据,只要软件未关闭,也可以调整为DMA接收,在主循环或者定时任务里进行串口接收DMA数据的处理。
DMA,全称为:Direct Memory Access,即直接存储器访问。DMA传输方式无需CPU 直接控制传输,也没有中断处理方式那样保留现场和恢复现场的过程,通过硬件为RAM 与I/O设备开辟一条直接传送数据的通路,能使CPU 的效率大为提高。
STM32中 DMA1有7个通道,DMA2有5个通道(DMA2 仅存在大容量产品中)。DMA挂载的时钟为AHB总线,其时钟为72Mhz,所以可以实现高速数据搬运。
STM32F103RBT6 只有1 个DMA控制器,DMA1 ,下面我们就针对DMA1 进行介绍。
从外设(TIMx、ADC、SPIx 、I2Cx 和USARTx )产生的DMA请求,通过逻辑或输入到DMA控制器,这就意味着同时只能有一个请求有效。外设的DMA请求,可以通过设置相应的外设寄存器中的控制位,被独立地开启或关闭。
DMA1各通道一览:
这里我们要使用的是串口 1 的 DMA 传送,也就是要用到通道 4。
DMA1通道4的配置方法如下:
dmac主要代码:
[cpp] view plain copy
u16 DMA1_MEM_LEN;//保存DMA每次数据传送的长度
//DMA1的各通道配置
//这里的传输形式是固定的,这点要根据不同的情况来修改
//从存储器->外设模式/8位数据宽度/存储器增量模式
//DMA_CHx:DMA通道CHx
//cpar:外设地址
//cmar:存储器地址
//cndtr:数据传输量
void MYDMA_Config(DMA_Channel_TypeDefDMA_CHx,u32 cpar,u32 cmar,u16 cndtr)
{
RCC->AHBENR|=1<<0; //开启DMA1时钟
delay_ms(1); //等待DMA时钟稳定
DMA_CHx->CPAR=cpar; //DMA1 外设地址
DMA_CHx->CMAR=(u32)cmar;//DMA1,存储器地址
DMA1_MEM_LEN=cndtr; //保存DMA传输数据量
DMA_CHx->CNDTR=cndtr; //DMA1,传输数据量
DMA_CHx->CCR=0X00000000;//复位
DMA_CHx->CCR|=1<<4; //从存储器读
DMA_CHx->CCR|=0<<5; //普通模式
DMA_CHx->CCR|=0<<6; //外设地址非增量模式
DMA_CHx->CCR|=1<<7; //存储器增量模式
DMA_CHx->CCR|=0<<8; //外设数据宽度为8位
DMA_CHx->CCR|=0<<10; //存储器数据宽度8位
DMA_CHx->CCR|=1<<12; //中等优先级
DMA_CHx->CCR|=0<<14; //非存储器到存储器模式
}
//开启一次DMA传输
void MYDMA_Enable(DMA_Channel_TypeDefDMA_CHx)
{
DMA_CHx->CCR&=~(1<<0); //关闭DMA传输
DMA_CHx->CNDTR=DMA1_MEM_LEN; //DMA1,传输数据量
DMA_CHx->CCR|=1<<0; //开启DMA传输
}
}
在主函数里主要有这几个语句完成DMA传输:
1首先配置DMA1通道4相关参数
MYDMA_Config(DMA1_Channel4,(u32)&USART1->DR,(u32)SendBuff,5200);//DMA1通道4,外设为串口1,存储器为SendBuff,长度5200
2然后将待发送内容装入存储器
SendBuff[i]=TEXT_TO_SEND[t];
3然后开启一次DMA传输
MYDMA_Enable(DMA1_Channel4);//开始一次DMA传输!
4监控传送进度。
pro=DMA1_Channel4->CNDTR;//得到当前还剩余多少个数据
1串口轮询部分。通过如下代码调串口回调函数。if (cnt >= HAL_UART_DMA_FULL){evt = HAL_UART_RX_FULL;}else if (cnt >= HAL_UART_DMA_HIGH){evt = HAL_UART_RX_ABOUT_FULL;PxOUT |= HAL_UART_Px_RTS;}else if (cnt && !dmaCfgrxTick){evt = HAL_UART_RX_TIMEOUT;但是我们希望当收到一条指令,大概8~10个字节,接收完就调回调函数。如果设置if (cnt >= 8){evt = HAL_UART_RX_IND;}则会出现DMA未接收完就调用回调函数的情况,则会出现接收错误。如果将cnt>24,则势必会在缓冲区堵塞一条指令。想问一下,如何判断DMA串口接收完成?我理解的是根据DMAIRQ寄存器中,DMAIF的状态位来判断,但硬件调试的过程中,一直没发现变为1的情况,所以,无法用来判断接收完成。2程序中,电源电压检测部分,BattMeasure函数中,HalAdcSetReference( HAL_ADC_REF_125V );adc = HalAdcRead( HAL_ADC_CHANNEL_VDD, HAL_ADC_RESOLUTION_10 );adc测量的数,换算成电压值后,和用稳压电源输出的电压值(搭配着万用表测量电压),有大致01V的误差,当电压逼近20V时,误差会减小。很想问一下,这是什么原因?是CC2540adc检测过程中不可避免的误差还是什么原因造成的?如果恒定误差有01V的话,在程序中电量检测部分,可以做出调整,否则低电量报警也会导致误差。希望能有一个官方的说法。非常希望技术支持给予答复,非常感谢您的帮助。
串口idle中断连续接收出错解决方法:
1、清除中断标志。
2、DMA传输停止。
3、检测到有效长度后,算出接收到的有效数据长度。
4、修改标志位,方便主函数里面的函数处理接收到的数据。
5、再次打开DMA接收。
查询就是一直在查看标志位,是不是被置1了,如果是就去读或者其他 *** 作
中断就是平时不用管,一单有东西来就会进入中断服务程序,你再去 *** 作
DMA是你初始化的时候把串口地址和需要传输的地址写上,来东西他就自己把数据存到你初始化的地址上,可以连续,或者传输完中断,丫的,不采纳我真是对不起我。都是手打的
以上就是关于sscom串口调试记录gga数据。如果仪器卡顿了,还会接收数全部的内容,包括:sscom串口调试记录gga数据。如果仪器卡顿了,还会接收数、stm32f103 dma是怎么实现的、CC2540两个问题,DMA传送结束的标志位是哪个以及程序中电量检测的电压值和实际值得误差是什么原因等相关内容解答,如果想了解更多相关内容,可以关注我们,你们的支持是我们更新的动力!
欢迎分享,转载请注明来源:内存溢出
评论列表(0条)