STemWin如何启用Multiple Buffering功能

STemWin如何启用Multiple Buffering功能

1. Multiple Buffering的工作原理

多缓冲是一种使用多个帧缓冲器的方法。其基本原理如下:在启用多个缓冲器的情况下,由显示控制 器所使用的前置缓冲器(front buffer)会在屏幕上产生图像,同时,一个或多个后置缓冲器(back buffers)则用于绘图操作。绘图操 作完成后,后置缓冲器成为可见的前置缓冲器。

如果使用两个缓冲器 (即一个前置缓冲器和一个后置缓冲器),通常称之为 “双缓冲”;如果使用两 个后置缓冲器和一个前置缓冲器,则称之为 “三缓冲”。

由于多缓冲方法使用多个帧缓冲器,因此,即便绘图操作仍在进行中,屏幕画面也是完全渲染的结果。 启动绘图过程时,前置缓冲器的当前内容会被复制到一个后置缓冲器中。在该操作完成后,所有绘图 操作只对该后置缓冲器起作用。绘图操作完成后,后置缓冲器成为前置缓冲器。 如果要使后置缓冲器 成为可见的前置缓冲器,通常只需修改显示控制器的帧缓冲器起始地址寄存器即可。

可以认为,显示器的持续刷新是通过显示控制器的应用程序得以实现的。每秒60次。每个周期完成 之后有一个垂直同步信号,通常称之为VSYNC信号。使后置缓冲器成为前置缓冲器的最佳时机是该 信号出现之时。如果不考虑VSYNC信号,则可能产生撕裂效果(如图1.1所示):
 STemWin如何启用Multiple Buffering功能_第1张图片

图1.1 撕裂效果

2. 开发环境

硬件:STM32F429Discovery开发板,主板芯片是STM32F429ZI,外部64Mb的SDRAM,LCD是LG的LD050WV1。

软件:基于ST官方提供的STemWin_Library_V1.0.0固件库和例程进行移植,操作系统是FreeRTOS,emWin版本为5.22。

3. 如何启用 Multiple Buffering

3.1Multiple Buffering功能的工作流程


图3.1 Multiple Buffering的工作流程

如图3.1所示, Multiple Buffering机制的工作顺序为:在 void   LCD_X_Config ( void )时进行配置;调用GUI_MULTIBUF_Begin()或GUI_MULTIBUF_BeginEx()对绘图进行缓冲 ,此时emWin会将front buffer的内容拷贝到back buffer中 ;进行绘图工作,此时所有的绘图工作都将在back buffer中进行;调用GUI_MULTIBUF_End()或GUI_MULTIBUF_EndEX()结束缓冲,此时emWin会将back buffer切换到front。

3.2 官方代码修改、实现以及实现原理解说

​主要修改STemWinLibrary555_4x9i目录下的 GUIDRV_stm32f429i_discovery.c文件。由于该文件已经对Multiple Buffering功能进行了支持,所以重点修改此处即可:

//
//   Buffers   /   VScreens
//
#define   NUM_BUFFERS   2   //   Number   of   multiple   buffers   to   be   used

即使用两个buffer,也就是双缓冲。后续的内容官方代码已经写好了,这里指进行介绍说明,同时捋清emWin进行双缓冲功能的工作流程。

开启双缓冲后,在 void   LCD_X_Config ( void )函数中,预编译条件被打开:

#if  ( NUM_BUFFERS  >   1)
for  (i  =   0;  i  <   GUI_NUM_LAYERS;  i++)  {
    GUI_MULTIBUF_ConfigEx(i,   NUM_BUFFERS);
}
#endif

这是对多缓冲功能进行配置,由于 NUM_BUFFERS 的值是 2 ,所以多缓冲功能被打开。

前面提及开启双缓冲后,所有的绘图操作将在back buffer里进行,为了确保绘图的正确性,emWin需要一个copy函数将front buffer中的数据首先拷贝到back buffer中,然后再进行绘图。ST官方代码使用了用户自定义的buffer copy程序,这样可使使用DMA进行内存拷贝,以提高工作效率。使用自定义buffer copy程序,需要在 void   LCD_X_Config ( void )函数中添加下面内容:

LCD_SetDevFunc(i,   LCD_DEVFUNC_COPYBUFFER,  ( void(*)( void))_LCD_CopyBuffer);

下面是官方给出的“ _LCD_CopyBuffer”函数 代码:

/*********************************************************************
*
*   _LCD_CopyBuffer
*/
static   void  _LCD_CopyBuffer( int  LayerIndex,   int  IndexSrc,   int  IndexDst)  {
     U32  BufferSize,  AddrSrc,  AddrDst;

    BufferSize  =  _GetBufferSize(LayerIndex);
    AddrSrc  =  _aAddr[LayerIndex]  +  BufferSize  *  IndexSrc;
    AddrDst  =  _aAddr[LayerIndex]  +  BufferSize  *  IndexDst;
    _DMA_Copy(LayerIndex,  ( void  *)AddrSrc,  ( void  *)AddrDst,  _axSize[LayerIndex],   _aySize [ LayerIndex ],   0 );
}

在绘制完图形后,用户调用 GUI_MULTIBUF_End()函数,此时emWin将back buffer切换到前台显示出来。这个切换动作最好在LCD进行第一行的行扫描时进行,否则容易出现撕裂效果。因此这段代码被写在LTDC的中断处理程序中。
/*********************************************************************
*
*   LTDC_ISR_Handler
*
*   Purpose:
*   End-Of-Frame-Interrupt   for   managing   multiple   buffering
*/
void  LTDC_ISR_Handler( void)  {
     U32  Addr;
     int  i;

     LTDC-> ICR  =  ( U32) LTDC_IER_LIE;
     for  (i  =   0;  i  <   GUI_NUM_LAYERS;  i++)  {
          if   ( _aPendingBuffer [ i ]   >=   0 )   {
               //
               //   Calculate   address   of   buffer   to   be   used   as   visible   frame   buffer
               //
               Addr   =   _aAddr [ i ]   +   _axSize [ i ]   *   _aySize [ i ]   *   _aPendingBuffer [ i ]   *   _aBytesPerPixels [ i ];
               //
               //   Store   address   into   SFR
               //
               _apLayer [ i ]-> CFBAR   &=   ~( LTDC_LxCFBAR_CFBADD );
               _apLayer [ i ]-> CFBAR   =   Addr ;
               //
               //   Reload   configuration
               //
               LTDC_ReloadConfig ( LTDC_SRCR_IMR );
               //
               //   Tell   emWin   that   buffer   is   used
               //
               GUI_MULTIBUF_ConfirmEx ( i ,   _aPendingBuffer [ i ]);
               //
               //   Clear   pending   buffer   flag   of   layer
               //
               _aBufferIndex [ i ]   =   _aPendingBuffer [ i ];
               _aPendingBuffer [ i ]   =   - 1 ;
          }
    }
}

从以上程序可以看出,back buffer到front buffer的切换,只是将back buffer的首地址赋值给了LTDC的CFBAR寄存器。这样的切换时很迅速的,因此用户看不到图像的绘制过程,避免了由绘图工作带来的屏幕闪烁问题。

那么,emWin如何知道何时需要进行buffer的切换工作呢?实际上,当用户调用 GUI_MULTIBUF_End()等函数的时候,会发出 LCD_X_SHOWBUFFER 信号给“ LCD_X_DisplayDriver”函数。

int  LCD_X_DisplayDriver( unsigned  LayerIndex,   unsigned  Cmd,   void  *  pData)
{
switch  (Cmd)  {
     case   LCD_X_SHOWBUFFER :   {
     //
     //   Required   if   multiple   buffers   are   used.   The   'Index'   element   of   p   contains   the   buffer   index.
     //
     LCD_X_SHOWBUFFER_INFO  *  p;

    p  =  ( LCD_X_SHOWBUFFER_INFO  *)pData;
    _aPendingBuffer[LayerIndex]  =  p-> Index;
     break;
}

在需要进行buffer切换时,传入的p->Index为1,这样,在“LTDC_ISR_Handler”函数里就满足了“_aPendingBuffer[i] >= 0”的条件。

3.3 官方代码中存在的bug

如果按上述方法进行修改和配置,并启用 Multiple Buffering功能,那你可能发现,屏幕上能够出现文字,但是却没有方块、水平直线和竖直直线。如果仿真调试,会发现,其实矩形 、水平直线和竖直直线被画在了front buffer里。

这是由于官方示例代码中存在一个bug,导致用户自定义的“ LCD_DEVFUNC_FILLRECT ”功能函数使用了错误地址。该示例代码中配置使用了自定义“ LCD_DEVFUNC_FILLRECT ”函数:

if  (_GetPixelformat(i)  <=   LTDC_Pixelformat_ARGB4444)  {
    LCD_SetDevFunc(i,   LCD_DEVFUNC_FILLRECT,  ( void(*)( void))_LCD_FillRect);
}

而“  _LCD_FillRect" 函数的实现如下:

/*********************************************************************
*
*   _LCD_FillRect
*/
static   void  _LCD_FillRect( int  LayerIndex,   int  x0,   int  y0,   int  x1,   int  y1,   U32  PixelIndex)  {
     U32  BufferSize,  AddrDst;
     int  xSize,  ySize;

     if  (GUI_GetDrawMode()  ==   GUI_DM_XOR)  {
          LCD_SetDevFunc ( LayerIndex ,   LCD_DEVFUNC_FILLRECT ,   NULL );
          LCD_FillRect ( x0 ,   y0 ,   x1 ,   y1 );
         LCD_SetDevFunc ( LayerIndex ,   LCD_DEVFUNC_FILLRECT ,   ( void (*)   ( void )) _LCD_FillRect );
    }   else  {
          xSize   =   x1   -   x0   +   1 ;
          ySize   =   y1   -   y0   +   1 ;
          BufferSize   =   _GetBufferSize ( LayerIndex );
          AddrDst   =   _aAddr [ LayerIndex ]   +   BufferSize   *   _aBufferIndex [ LayerIndex ]   +   ( y0   *   _axSize [ LayerIndex ]   +   x0 )   *   _aBytesPerPixels [ LayerIndex ];
          _DMA_Fill ( LayerIndex ,   ( void   *) AddrDst ,   xSize ,   ySize ,   _axSize [ LayerIndex ]   -   xSize ,   PixelIndex );
    }
}

该代码将在绘制矩形、水平直线和竖直直线时进行调用,将绘制的图形拷贝到相应的buffer中。 如果对该处代码进行调试,会发现“AddrDst的值始终都是front buffer的值。而在启用双缓冲时,AddrDst”应该指向back buffer。进一步会发现“_aBufferIndex[LayerIndex]”的值是错误的。这是因为不应该在LTDC的中断服务程序中对"_aBufferIndex[LayerIndex]"进行赋值。因为中断服务程序中的赋值只有在用户调用GUI_MULTIBUF_End()时才执行,这个时候绘图工作都已经完成了。正确的方式应该在GUI_MULTIBUF_Begin()后立即对"_aBufferIndex[LayerIndex]"进行赋值。
那么,应该在代码的何处进行赋值呢?答案是“_LCD_CopyBuffer”函数,因为调用GUI_MULTIBUF_Begin()后,emWin会调用该函数将front buffer中的内容拷贝的back buffer,然后才绘图。
因此,修改后的代码为:

/*********************************************************************
*
*   _LCD_CopyBuffer
*/
static   void  _LCD_CopyBuffer( int  LayerIndex,   int  IndexSrc,   int  IndexDst)  {
     U32  BufferSize,  AddrSrc,  AddrDst;

    BufferSize  =  _GetBufferSize(LayerIndex);
    AddrSrc  =  _aAddr[LayerIndex]  +  BufferSize  *  IndexSrc;
    AddrDst  =  _aAddr[LayerIndex]  +  BufferSize  *  IndexDst;

    _aBufferIndex[LayerIndex]  =  IndexDst;

    _DMA_Copy(LayerIndex,  ( void  *)AddrSrc,  ( void  *)AddrDst,  _axSize[LayerIndex],  _aySize[LayerIndex],   0);
}

/*********************************************************************
*
*   LTDC_ISR_Handler
*
*   Purpose:
*   End-Of-Frame-Interrupt   for   managing   multiple   buffering
*/
void  LTDC_ISR_Handler( void)  {
     U32  Addr;
     int  i;

     LTDC-> ICR  =  ( U32) LTDC_IER_LIE;
     for  (i  =   0;  i  <   GUI_NUM_LAYERS;  i++)  {
          if   ( _aPendingBuffer [ i ]   >=   0 )   {
               //
               //   Calculate   address   of   buffer   to   be   used   as   visible   frame   buffer
               //
              Addr  =  _aAddr[i]  +  _axSize[i]  *  _aySize[i]  *  _aPendingBuffer[i]  *  _aBytesPerPixels[i];
               //
               //   Store   address   into   SFR
               //
               //   _apLayer[i]->CFBAR   &=   ~(LTDC_LxCFBAR_CFBADD);     // 没发现这句有什么用,一起注掉了吧!
               _apLayer [ i ]-> CFBAR   =   Addr ;
               //
               //   Reload   configuration
               //
               LTDC_ReloadConfig ( LTDC_SRCR_IMR );
               //
               //   Tell   emWin   that   buffer   is   used
               //
               GUI_MULTIBUF_ConfirmEx ( i ,   _aPendingBuffer [ i ]);
               //
               //   Clear   pending   buffer   flag   of   layer
               //
               _aPendingBuffer [ i ]   =   - 1 ;
         }
    }
}

这个bug在 STemWin_Library_V1.0.1中仍然存在。但是1.0.1的官方例程中没有使用Multiple Buffering功能,而1.0.0中的demo明显使用了该功能(如图3.2):

STemWin如何启用Multiple Buffering功能_第2张图片
图3.2 官方例程明显开启了Multiple Buffering功能

看起来似乎ST方面发现了Multiple Buffering功能不好用,但是没有从根本上解决问题,所以官方例程就不使用该功能了。这种治标不治本的处理方式,让我对st的工作态度表示怀疑。

你可能感兴趣的:(multiple,Buffering,EMWIN,STemWin,STM32F429)