分享

Android中的FrameBuffer

 James130 2015-10-12

FrameBuffer 在Android中并不像在其它GUI那样直观,抽象的层次比较多,加上GUI的更新是通过OpenGLES来做的。所以让人很难搞清GUI更新的整个流 程,最近要准备一个讲稿,所以花了一些去研究,这里做点笔记供大家参考,源代码是基于高通平台的,这些代码在网上都可以下载。

FrameBuffer相关的组件
fb
1.SurfaceFlinger是一个服务,主要是负责合成各窗口的Surface,然后通过OpenGLES显示到FrameBuffer上。SurfaceFlinger本身比较重要而且比较复杂,以后专门写一篇吧。

2.DisplayHardware是对显示设备的抽象,包括FrameBuffer和Overlay。它加载FrameBuffer和Overlay插件,并初始化OpenGLES:

  1.     mNativeWindow <span style="color: #339933;">=</span>  
  2.  new FramebufferNativeWindow<span style="color: #009900;">(</span>  
  3. <span style="color: #009900;">)</span>  
  4. <span style="color: #339933;">;</span>  
  5.   
  6.     framebuffer_device_t <span style="color: #993333;">const</span>  
  7.  <span style="color: #339933;">*</span>  
  8.  fbDev <span style="color: #339933;">=</span>  
  9.  mNativeWindow<span style="color: #339933;">-></span>  
  10. getDevice<span style="color: #009900;">(</span>  
  11. <span style="color: #009900;">)</span>  
  12. <span style="color: #339933;">;</span>  
  13.   
  14.     <span style="color: #b1b100;">if</span>  
  15.  <span style="color: #009900;">(</span>  
  16. hw_get_module<span style="color: #009900;">(</span>  
  17. OVERLAY_HARDWARE_MODULE_ID<span style="color: #339933;">,</span>  
  18.  <span style="color: #339933;">&</span>  
  19. module<span style="color: #009900;">)</span>  
  20.  <span style="color: #339933;">==</span>  
  21.  <span style="color: #0000dd;">0</span>  
  22. <span style="color: #009900;">)</span>  
  23.  <span style="color: #009900;">{</span>  
  24.   
  25.        overlay_control_open<span style="color: #009900;">(</span>  
  26. module<span style="color: #339933;">,</span>  
  27.  <span style="color: #339933;">&</span>  
  28. mOverlayEngine<span style="color: #009900;">)</span>  
  29. <span style="color: #339933;">;</span>  
  30.   
  31.     <span style="color: #009900;">}</span>  
  32.   
  33.     surface <span style="color: #339933;">=</span>  
  34.  eglCreateWindowSurface<span style="color: #009900;">(</span>  
  35. display<span style="color: #339933;">,</span>  
  36.  config<span style="color: #339933;">,</span>  
  37.  mNativeWindow.<span style="color: #202020;">get</span>  
  38. <span style="color: #009900;">(</span>  
  39. <span style="color: #009900;">)</span>  
  40. <span style="color: #339933;">,</span>  
  41.  NULL<span style="color: #009900;">)</span>  
  42. <span style="color: #339933;">;</span>  
  43.   
  44.     eglMakeCurrent<span style="color: #009900;">(</span>  
  45. display<span style="color: #339933;">,</span>  
  46.  surface<span style="color: #339933;">,</span>  
  47.  surface<span style="color: #339933;">,</span>  
  48.  context<span style="color: #009900;">)</span>  
  49. <span style="color: #339933;">;</span>  

3.FramebufferNativeWindow 是framebuffer 的抽象,它负责加载libgralloc,并打开framebuffer设备。FramebufferNativeWindow并不直接使用 framebuffer,而是自己创建了两个Buffer:

queueBuffer负责显示一个Buffer到屏幕上,它调用fb->post去显示。
dequeueBuffer获取一个空闲的Buffer,用来在后台绘制。

这两个函数由eglSwapBuffers调过来,调到

  1. egl_window_surface_v2_t<span style="color: #339933;">::</span>  
  2. <span style="color: #202020;">swapBuffers</span>  
  3. :  
  4.     nativeWindow<span style="color: #339933;">-></span>  
  5. queueBuffer<span style="color: #009900;">(</span>  
  6. nativeWindow<span style="color: #339933;">,</span>  
  7.  buffer<span style="color: #009900;">)</span>  
  8. <span style="color: #339933;">;</span>  
  9.   
  10.     nativeWindow<span style="color: #339933;">-></span>  
  11. dequeueBuffer<span style="color: #009900;">(</span>  
  12. nativeWindow<span style="color: #339933;">,</span>  
  13.  <span style="color: #339933;">&</span>  
  14. buffer<span style="color: #009900;">)</span>  
  15. <span style="color: #339933;">;</span>  

4.msm7k/liboverlay是Overlay的实现,与其它平台不同的是,高通平台上的Overlay并不是提供一个framebuffer设备,而通过fb0的ioctl来实现的,ioctl分为两类操作:

OverlayControlChannel用于设置参数,比如设置Overlay的位置,宽度和高度:

  1. bool OverlayControlChannel<span style="color: #339933;">::</span>  
  2. <span style="color: #202020;">setPosition</span>  
  3. <span style="color: #009900;">(</span>  
  4. <span style="color: #993333;">int</span>  
  5.  x<span style="color: #339933;">,</span>  
  6.  <span style="color: #993333;">int</span>  
  7.  y<span style="color: #339933;">,</span>  
  8.  uint32_t w<span style="color: #339933;">,</span>  
  9.  uint32_t h<span style="color: #009900;">)</span>  
  10.  <span style="color: #009900;">{</span>  
  11.   
  12.     ov.<span style="color: #202020;">dst_rect</span>  
  13. .<span style="color: #202020;">x</span>  
  14.  <span style="color: #339933;">=</span>  
  15.  x<span style="color: #339933;">;</span>  
  16.   
  17.     ov.<span style="color: #202020;">dst_rect</span>  
  18. .<span style="color: #202020;">y</span>  
  19.  <span style="color: #339933;">=</span>  
  20.  y<span style="color: #339933;">;</span>  
  21.   
  22.     ov.<span style="color: #202020;">dst_rect</span>  
  23. .<span style="color: #202020;">w</span>  
  24.  <span style="color: #339933;">=</span>  
  25.  w<span style="color: #339933;">;</span>  
  26.   
  27.     ov.<span style="color: #202020;">dst_rect</span>  
  28. .<span style="color: #202020;">h</span>  
  29.  <span style="color: #339933;">=</span>  
  30.  h<span style="color: #339933;">;</span>  
  31.   
  32.    
  33.     ioctl<span style="color: #009900;">(</span>  
  34. mFD<span style="color: #339933;">,</span>  
  35.  MSMFB_OVERLAY_SET<span style="color: #339933;">,</span>  
  36.  <span style="color: #339933;">&</span>  
  37. ov<span style="color: #009900;">)</span>  
  38. <span style="color: #339933;">;</span>  
  39.   
  40. <span style="color: #009900;">}</span>  

OverlayDataChannel用于显示Overlay,其中最重要的函数就是queueBuffer:
bool OverlayDataChannel::queueBuffer(uint32_t offset) {

  1. mOvData.<span style="color: #202020;">data</span>  
  2. .<span style="color: #202020;">offset</span>  
  3.  <span style="color: #339933;">=</span>  
  4.  offset<span style="color: #339933;">;</span>  
  5.      
  6.    
  7. ioctl<span style="color: #009900;">(</span>  
  8. mFD<span style="color: #339933;">,</span>  
  9.  MSMFB_OVERLAY_PLAY<span style="color: #339933;">,</span>  
  10.  odPtr<span style="color: #009900;">)</span>  
  11. <span style="color: #009900;">)</span>  
  12.   
  13. <span style="color: #009900;">}</span>  

5.msm7k/libgralloc 是显示缓存的抽象,包括framebuffer和普通Surface的Buffer。framebuffer只是/dev/graphic/fb0的包 装,Surface的Buffer则是对/dev/pmem、ashmem和GPU内存(msm_hw3dm)的包装,它的目标主要是方便硬件加速,因为 DMA传输使用物理地址,要求内存在物理地址上连续。

6.msm7k/libcopybit这是2D加速库,主要负责Surface的拉伸、旋转和合成等操作。它有两种实现方式:
copybit.cpp: 基于fb0的ioctl(MSMFB_BLIT)的实现。
copybit_c2d.cpp: 基于kgsl的实现,只是对libC2D2.so的包装,libC2D2.so应该是不开源的。

7.pmem
misc/pmem.c: 对物理内存的管理,算法和用户空间的接口。
board-msm7x27.c定义了物理内存的缺省大小:

  1. <span style="color: #339933;">#define MSM_PMEM_MDP_SIZE   0x1B76000</span>  
  2.   
  3. <span style="color: #339933;">#define MSM_PMEM_ADSP_SIZE  0xB71000</span>  
  4.   
  5. <span style="color: #339933;">#define MSM_PMEM_AUDIO_SIZE 0x5B000</span>  
  6.   
  7. <span style="color: #339933;">#define MSM_FB_SIZE     0x177000</span>  
  8.   
  9. <span style="color: #339933;">#define MSM_GPU_PHYS_SIZE   SZ_2M</span>  
  10.   
  11. <span style="color: #339933;">#define PMEM_KERNEL_EBI1_SIZE   0x1C000</span>  

msm_msm7x2x_allocate_memory_regions分配几大块内存用于给pmem做二次分配。

8.KGSL
Kernel Graphics System Layer (KGSL),3D图形加速驱动程序,源代码drivers/gpu/msm目录下,它是对GPU的包装,给OpenGLES 2.0提供抽象的接口。

9.msm_hw3dm
这个我在内核中没有找到相关代码。

10.msm_fb
msm_fb.c: framebuffer, overlay和blit的用户接口。
mdp_dma.c: 对具体显示设备的包装,提供两种framebuffer更新的方式:

  • mdp_refresh_screen: 定时更新。
  • mdp_dma_pan_update: 通过pan display主动更新。

mdp_dma_lcdc.c:针对LCD实现的显示设备,mdp_lcdc_update用更新framebuffer。

    本站是提供个人知识管理的网络存储空间,所有内容均由用户发布,不代表本站观点。请注意甄别内容中的联系方式、诱导购买等信息,谨防诈骗。如发现有害或侵权内容,请点击一键举报。
    转藏 分享 献花(0

    0条评论

    发表

    请遵守用户 评论公约

    类似文章 更多