大家好:
有个问题困扰着我,有没有相关人员遇到这样的情况指点一下。
我想通过DMA来将数据拷贝到FSMC上,FSMC对应的LCD。
我DMA是这样配置的:
下面这段代码,我已经验证过了,如果将其外设地址改成自增的通过DMA——>FSCM——>SRAM可以知道,拷贝到SRAM的数据是我要拷贝到的内容,这说明DMA是OK的。
但将其拷贝到DMA——>FSCM——>LCD上就会有点问题。
用keil的断点看过了,DMA的数据量没有超过65535,一次就传输4800次的数据而已,每次2个字节(10点行,color 16bit),奇怪的是,我屏幕上显示出来的效果有点花屏。
如果将DMA拷贝换成普通的循环一个个赋值的话,屏幕就显示的很完好。
现在不明白为什么通过DMA——>FSCM——>LCD的方式就有点花屏(可以看清楚大概的界面,但有部分是花掉了。)
[mw_shl_code=c,true]
#include "dma.h"
DMA_InitTypeDef DMA_InitStructure;
u16 DmaMemLen;//保存DMA每次数据传送的长度
//DMA1的各通道配置
//这里的传输形式是固定的,这点要根据不同的情况来修改
//从存储器->外设模式/8位数据宽度/存储器增量模式
//DMA_CHx
MA通道CHx
/ar:外设地址
//cmar:存储器地址
//cndtr:数据传输量
void myDmaConfig(DMA_Channel_TypeDef* DMA_CHx,u32 cpar,u32 cmar,u16 cndtr)
{
NVIC_InitTypeDef NVIC_InitStructure;
RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE); //使能DMA传输
DMA_DeInit(DMA_CHx); //将DMA的通道1寄存器重设为缺省值
DmaMemLen=cndtr;
DMA_InitStructure.DMA_PeripheralBaseAddr = cpar; //DMA外设基地址
DMA_InitStructure.DMA_MemoryBaseAddr = cmar; //DMA内存基地址
DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralDST; //数据传输方向,从内存读取发送到外设
DMA_InitStructure.DMA_BufferSize = cndtr; //DMA通道的DMA缓存的大小
DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Enable; //外设地址寄存器不变
DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable; //内存地址寄存器递增
DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_HalfWord; //数据宽度为8位
DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_HalfWord; //数据宽度为8位
DMA_InitStructure.DMA_Mode = DMA_Mode_Normal; //工作在正常模式
DMA_InitStructure.DMA_Priority = DMA_Priority_Medium; //DMA通道 x拥有中优先级
DMA_InitStructure.DMA_M2M = DMA_M2M_Enable; //DMA通道x没有设置为内存到内存传输
DMA_Init(DMA_CHx, &DMA_InitStructure); //根据DMA_InitStruct中指定的参数初始化DMA的通道USART1_Tx_DMA_Channel所标识的寄存器
NVIC_InitStructure.NVIC_IRQChannel = DMA1_Channel4_IRQn;
NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0;
NVIC_InitStructure.NVIC_IRQChannelSubPriority = 3;
NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
NVIC_Init(&NVIC_InitStructure);
DMA_ITConfig(DMA_CHx,DMA_IT_TC|DMA_IT_TE,ENABLE);
}
extern int dmaTransFlag;
void DMA1_Channel4_IRQHandler(void)
{
if(DMA_GetITStatus(DMA1_IT_TC4) != RESET) //接收中断(接收到的数据必须是0x0d 0x0a结尾)
{
dmaTransFlag = 0;
printf("ok
");
}
if(DMA_GetITStatus(DMA1_IT_TE4) != RESET) //接收中断(接收到的数据必须是0x0d 0x0a结尾)
{
printf("error
");
}
DMA_ClearITPendingBit(DMA1_IT_TC4|DMA1_IT_TE4);
}
//开启一次DMA传输
void myDmaEnable(DMA_Channel_TypeDef*DMA_CHx)
{
DMA_Cmd(DMA_CHx, DISABLE ); //关闭USART1 TX DMA1 所指示的通道
DMA_SetCurrDataCounter(DMA_CHx,DmaMemLen); //DMA通道的DMA缓存的大小,一次传输完之后传输的长度会被减到0,因此这里在每次开始DMA传输前都先设置下它的传输长度
DMA_Cmd(DMA_CHx, ENABLE); //使能USART1 TX DMA1 所指示的通道
}[/mw_shl_code]
版主你好,在实际测试时我这个是设置成Disable的,上面这个函数是拷贝验证过的sram上面的代码,我在拷贝到屏上时外设地址寄存器是设置成不变的。
一周热门 更多>