初级会员
- 积分
- 103
- 金钱
- 103
- 注册时间
- 2020-5-1
- 在线时间
- 34 小时
|
我从LVGL的GIT仓库里面拉取了官方源码,按照网上资料进行移植。用正点原子的打点函数LCD显示是正常的,但是会有拉帘的感觉(刷新不过来),然后我就使用了DMA来优化,发现会出现如图所示的现象。
图1是刚刚烧录程序后的(显示正常),图2是我点击了左上角的开关后出现该现象,图3是我点击了按钮后出现的。
这个程序基于FreeRTOS,移植LVGL时发现的现象,如果不使用DMA而使用普通打点他就没有花屏。@ 正点原子
这是【DMA部分的初始化】
void MYDMA_Config(DMA_Stream_TypeDef *DMA_Streamx,u32 chx,u32 par,u32 mar,u16 ndtr)
{
NVIC_InitTypeDef NVIC_InitStructure;
DMA_InitTypeDef DMA_InitStructure;
if((u32)DMA_Streamx>(u32)DMA2)//得到当前stream是属于DMA2还是DMA1
{
RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_DMA2,ENABLE);//DMA2时钟使能
}else
{
RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_DMA1,ENABLE);//DMA1时钟使能
}
DMA_DeInit(DMA_Streamx);
while (DMA_GetCmdStatus(DMA_Streamx) != DISABLE){}//等待DMA可配置
/* 配置 DMA Stream */
DMA_InitStructure.DMA_Channel = chx; //通道选择
DMA_InitStructure.DMA_PeripheralBaseAddr = par;//DMA外设地址
DMA_InitStructure.DMA_Memory0BaseAddr = mar;//DMA 存储器0地址
DMA_InitStructure.DMA_DIR = DMA_DIR_MemoryToMemory;//存储器到外设模式
DMA_InitStructure.DMA_BufferSize = ndtr;//数据传输量
DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Enable;//外设增量模式
DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Disable;//存储器增量模式
DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_HalfWord;//外设数据长度:8位
DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_Word;//存储器数据长度:16位
DMA_InitStructure.DMA_Mode = DMA_Mode_Normal; // 使用普通模式
DMA_InitStructure.DMA_Priority = DMA_Priority_VeryHigh;//中等优先级
DMA_InitStructure.DMA_FIFOMode = DMA_FIFOMode_Enable; // DMA_FIFOMode_Enable DMA_FIFOMode_Disable
DMA_InitStructure.DMA_FIFOThreshold = DMA_FIFOThreshold_Full;
DMA_InitStructure.DMA_PeripheralBurst = DMA_PeripheralBurst_Single;//外设突发单次传输
DMA_InitStructure.DMA_MemoryBurst = DMA_MemoryBurst_Single;//存储器突发单次传输
DMA_Init(DMA_Streamx, &DMA_InitStructure);//初始化DMA Stream
DMA_ITConfig(DMA_Streamx, DMA_IT_TC, ENABLE);
//DMA NVIC 配置
NVIC_InitStructure.NVIC_IRQChannel = DMA2_Stream1_IRQn;
NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority=7;
NVIC_InitStructure.NVIC_IRQChannelSubPriority =0;
NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
NVIC_Init(&NVIC_InitStructure);
DMA_Cmd(DMA_Streamx, DISABLE);
while (DMA_GetCmdStatus(DMA_Streamx) != DISABLE){} //确保DMA可以被设置
DMA_SetCurrDataCounter(DMA_Streamx,ndtr); //数据传输量
DMA_Cmd(DMA_Streamx, ENABLE);
}
这是【显示函数部分的修改】
//刷新
static void disp_flush(lv_disp_drv_t * disp_drv, const lv_area_t * area, lv_color_t * color_p)
{
if(disp_flush_enabled)
{
distmp1 = disp_drv->draw_buf->buf_act;
u32 w = area->x2 - area->x1+1;
u32 h = area->y2 - area->y1+1;
LCD_Set_Window(area->x1,area->y1,(w),(h));
MYDMA_Config(DMA2_Stream1,DMA_Channel_0,(u32)distmp1,(u32)&LCD->LCD_RAM,disp_drv->draw_buf->size);
LCD_WriteRAM_Prepare(); //开始写入GRAM
}
/*IMPORTANT!!!
*Inform the graphics library that you are ready with the flushing*/
// lv_disp_flush_ready(disp_drv);
}
//DMA中断
void DMA2_Stream1_IRQHandler(void)
{ // DMA_IT_TCIF1
if(DMA_GetITStatus(DMA2_Stream1,DMA_IT_TCIF1)!=RESET)
{
DMA_ClearITPendingBit(DMA2_Stream1,DMA_IT_TCIF1);
// lv_disp_flush_ready(disp_drv_tmp);
disp_drv_tmp->draw_buf->flushing = 0;
disp_drv_tmp->draw_buf->flushing_last = 0;
}
}
|
|