AT91SAM9260添加Framebuff驱动

出处:21ic 发布于:2018-07-23 14:28:46

  原有的LCD操作方法实现了画矩形、ASCII字符、汉字。终是根据字符或汉字的点阵信息在屏幕上打点!如此而已。原LCD驱动做了个双缓冲显示区,根据缓冲区的变化改写LCD设备的显示区,新的framebuff驱动思想是:直接操作显示区域,需要自己写的framebuff驱动里没有画点、画圆、显示字符、显示汉字等的具体操作。这些操作在framebuff驱动框架里已经实现,无需自己编写。下面记录下framebuff驱动的编写过程,LCD硬件部分仅保留修改LCD显示区的IO映射和数据写入即可。
  手上这款LCD自带控制器,只能通过读写其提供寄存器和他交互数据,不能直接映射他的显示区域。所以我在驱动里申请了2个和LCD显示缓冲区一样大小的内存,一个用于模拟framebuff驱动需要的共享内存区域,另一个用来保存这个模拟共享区域的快照,用于比对共享区域的变化。当检测到共享内存区域的变化后,将这个变化通过LCD的寄存器写给LCD,这样就能实现共享区域的变化能被同步反映到LCD设备上。
  在内核的drivers/video/目录下有很多fb设备的驱动,我找了个简单的dnfb.c作为参考,以他为蓝本实现我的驱动。首先修改drivers/video下Kconfig,添加:
  config FB_DISPLAY
  tristate"WHZYDZ LCD support"
  depends on FB && ARM
  select FB_CFB_FILLRECT
  select FB_CFB_COPYAREA
  select FB_CFB_IMAGEBLIT
  接着修改Makefile,添加:
  obj-$(CONFIG_FB_DISPLAY) += zydz_fb.o
  我们在zydz_fb.c中来写驱动代码,首先要完成显示区域的变化如何写入到设备,这个虽不是framebuff驱动本身特有的,但其作为基本的一环,必须先实现。原系统平台的相关驱动可以借鉴。原来的驱动代码是先定位到LCD显示缓冲的行首,然后一个字节一个字节的写,直到写完一行的数据,其中位置光标自动右移。但在我这,一行点位根本显示不全,我们用的是RA9935A,我怀疑它在控制自动移位方面可能存在问题。后来我改变写数据的方式:自己控制位置光标,然后写一个字节!这样能正常显示了。
  接先来就是和MiniGUI联调,边调边修改我的驱动。MinGUI得使用shadow引擎才能支持8bpp以下的。重新编译minigui,configure 时加上--enable-newgal \
  --enable-videoshadow \
  --with-targetname=fbcon \
  MiniGUI.cfg配置文件修改如下:
  [system]
  # GAL engine and default options
  gal_engine=shadow
  defaultmode=320x240-1bpp
  [shadow]
  real_engine=fbcon
  经过n次的测试,主要方法是在MinGUI中增加打印信息,根据输出信息判断出错的位置,然后修改驱动。跟到了src/newgal/video.c的int GAL_VideoModeOK (int width, int height, int bpp, Uint32 flags)函数,
  里面有段注释和代码看了,让人心凉了一大节!
  /* Currently 1 and 4 bpp are not supported */
  if ( bpp < 8 || bpp > 32 ) {
  return(0);
  }
  看来MinGUI1.6.10是不支持位深小于8的屏了。我尝试着注释掉了这段代码,以便让MinGUI能完成初始化的工作。接着出现下面的错误:
  Linux_fbcon fb_fix.line_length=40
  Linux_fbcon fbcon_info.yres=240
  Linux_fbcon fbcon_info.fb_size=12288
  Linux_fbcon fbcon_info.fb=40021000
  Linux_fbcon fbcon_info.bpp=1
  GAL_GetVideoMode 1
  width=320
  height=240
  bpp=1
  Unhandled fault: external abort on non-linefetch (0x008) at 0x40021000
  Bus error
  查看linux_fbcon.c:
  fbcon_info.fb =
  #ifdef _FXRM9200_IAL /* workaround for Fuxu RM9200 */
  mmap (NULL, fbcon_info.fb_size, PROT_READ | PROT_WRITE, MAP_SHARED,
  fbcon_info.fd_fb, 0);
  #elif defined (__uClinux__)
  mmap (NULL, fbcon_info.fb_size, PROT_READ | PROT_WRITE, 0,
  fbcon_info.fd_fb, 0);
  #else
  mmap (NULL, fbcon_info.fb_size, PROT_READ | PROT_WRITE, MAP_SHARED,
  fbcon_info.fd_fb, 0);
  #endif
  这个使用到了framebuff驱动的mmap调用,再查看drivers/video/Fbmem.c默认的fb_mmap函数:
  /* frame buffer memory */
  start = info->fix.smem_start;
  len = PAGE_ALIGN((start & ~PAGE_MASK) + info->fix.smem_len);
  他会将info->fix.smem_start这个物理地址进行映射。好了,framebuff驱动里面我们可以使用virt_to_phys获取共享内存区域的物理地址!
  自此,edit例程总算运行起来了!显示效果见下图:


  AT91SAM9260添加Framebuff驱动 - yu_hongchang - 中原大鱼的博客
  显示效果不理想,MiniGUI还是用在8bpp以上屏上合适!,下面贴上主要的代码:
  * linux/drivers/video/zydzfb.c -- ZYDZ graphics adaptor frame buffer device
  *
  * Created 16 Sep2011 by hongchang.yu  * Based on dnfb.c
  *
  * History:
  *
  * This file is subject to the terms and conditions of the GNU General Public
  * License. See the file COPYING in the main directory of this archive
  * for more details.
  */
  #include
  #include
  #include
  #include
  #include
  #include
  #include
  #include
  #include
  #include
  #include
  #include
  #include
  #include
  #define LCD_WIDTH 320
  #define LCD_HEIGHT 240
  #define DISPRAMBUFLSZ (LCD_WIDTH/8)
  #define DISPRAMBUFSIZE (DISPRAMBUFLSZ*LCD_HEIGHT)
  /* display_ video definitions */
  static void __iomem *io_data=NULL;
  static void __iomem *io_cmd=NULL;
  static void __iomem *io_ctrl=NULL;
  static unsigned long ioo_data=0;
  static unsigned char *rambuf_org = NULL;
  static unsigned char *rambuf_cur = NULL;
  /* frame buffer operations */
  // zydzfb_blank控制屏幕开关
  static int zydzfb_blank(int blank, struct fb_info *info);
  static struct fb_ops zydzfb_ops = {
  .owner = THIS_MODULE,
  //.fb_blank = zydzfb_blank,
  .fb_fillrect = cfb_fillrect,
  .fb_copyarea = cfb_copyarea,
  .fb_imageblit = cfb_imageblit,
  };
  struct fb_var_screeninfo zydzfb_var __devinitdata = {
  .xres = 320,//实际x轴分辨率
  .yres = 240,//实际y轴分辨率
  .xres_virtual = 320,//虚拟x轴分辨率
  .yres_virtual = 240,//虚拟y轴分辨率
  .bits_per_pixel= 1, //定义每个点用多少位表示
  .height = -1,
  .width = -1,
  //.vmode = FB_VMODE_NONINTERLACED,
  };
  static struct fb_fix_screeninfo zydzfb_fix __devinitdata = {
  .id = "zydzfb",//设备名称
  .type = FB_TYPE_PACKED_PIXELS,
  .visual = FB_VISUAL_MONO01 ,/* Monochr. 1=Black 0=White */
  .line_length = DISPRAMBUFLSZ,
  };
  /*
  * Initialization
  */
  static int __devinit zydzfb_probe(struct platform_device *dev)
  {
  struct fb_info *info;
  int err = 0;
  info = framebuffer_alloc(0, &dev->dev);
  if (!info)
  return -ENOMEM;
  info->fbops = &zydzfb_ops;
  info->fix = zydzfb_fix;
  info->fix.smem_start = virt_to_phys(rambuf_cur);
  info->fix.smem_len = DISPRAMBUFSIZE;
  info->var = zydzfb_var;
  /* Virtual address */
  info->screen_base = rambuf_cur;
  info->screen_size = DISPRAMBUFSIZE;
  err = fb_alloc_cmap(&info->cmap, 2, 0);
  if (err < 0) {
  framebuffer_release(info);
  return err;
  }
  err = register_framebuffer(info);
  if (err < 0) {
  fb_dealloc_cmap(&info->cmap);
  framebuffer_release(info);
  return err;
  }
  platform_set_drvdata(dev, info);
  /* now we have registered we can safely setup the hardware */
  printk("display_ frame buffer alive and kicking !\n");
  retu

关键词:驱动

版权与免责声明

凡本网注明“出处:维库电子市场网”的所有作品,版权均属于维库电子市场网,转载请必须注明维库电子市场网,https://www.dzsc.com,违反者本网将追究相关法律责任。

本网转载并注明自其它出处的作品,目的在于传递更多信息,并不代表本网赞同其观点或证实其内容的真实性,不承担此类作品侵权行为的直接责任及连带责任。其他媒体、网站或个人从本网转载时,必须保留本网注明的作品出处,并自负版权等法律责任。

如涉及作品内容、版权等问题,请在作品发表之日起一周内与本网联系,否则视为放弃相关权利。

相关技术资料
上传BOM文件: BOM文件
*公司名:
*联系人:
*手机号码:
QQ:
应用领域:

有效期:
OEM清单文件: OEM清单文件
*公司名:
*联系人:
*手机号码:
QQ:
有效期:

扫码下载APP,
一键连接广大的电子世界。

在线人工客服

买家服务:
卖家服务:

0571-85317607

客服在线时间周一至周五
9:00-17:30

关注官方微信号,
第一时间获取资讯。

建议反馈

联系人:

联系方式:

按住滑块,拖拽到最右边
>>
感谢您向阿库提出的宝贵意见,您的参与是维库提升服务的动力!意见一经采纳,将有感恩红包奉上哦!