【赛昉科技昉·星光RISC-V单板计算机试用体验】添加tft触摸屏(ili9488)并显示开机logo,为qt做准备

描述

本文来源电子发烧友社区,作者:cszzlsw, 帖子地址:https://bbs.elecfans.com/jishu_2287667_1_1.html


1.前言
前面两篇帖子咱们回顾了从头制作系统并运行homeassistant,其中不可谓不辛酸,然后我就在想再做个啥,找了一下自己库存的外设,有一个tft屏带触摸芯片的,一想,这不正好的嘛,可以玩玩图形编程了,老听说linux开发板的qt编程咱也没见过啊,说干就干,这就开始了.

2.移植驱动并首先尝试
首先我的屏主控是lil9488的,分辨率320*480,据说再往上就得上rgb屏了,这咱没玩过也不懂,把手头上这个整明白先.
这个在git仓库:https://github.com/starfive-tech/freelight-u-sdk,给的linux内核里面找了一圈,也没见,
然后发现有一个芯片是ili9486的,跟我这差一位数,一想差不多,这个开始弄吧,把文件拷过来,然后改吧:
RISC-V

然后把所有涉及9486的地方都改成9488,
然后是设备树里面改吧改吧(改了好多次,也是第一次写设备树):
RISC-V

这款tft屏除了常规的spi线之外,还有BL(背光BackLight),Reset(复位),DC(Data-Control)线需要接,这里我用的gpio1,2,3脚,
另外最后的那个debug如果不需要就置0,需要就置1.

然后在linux内核里面配置一下,把ili9488加进来:

-> Device Drivers                                                                                                                                                          │  
  │       -> Staging drivers (STAGING [=y])                                                                                                                                        │  
  │         -> Support for small TFT LCD display modules (FB_TFT [=y])   



然后就是编译内核,再删除rootfs,然后再生成rootfs,这都老熟人了,
这些都整完之后,就把卡插上准备烧录,
然后漫长的等待过程之后,终于烧完了,

紧跟着就把屏接上,按照文档里介绍的40pin引脚对应接好,
上电,然而,并没有出现期待中的点亮屏幕,

二.各种解决问题
1.spi没有设备id问题:

出现这个问题的原因是因为sdk的系统是5.15,属于是很新的系统了,但是linux里面的驱动几年都不更新,所以过期了,需要在ili9844里这么改:
RISC-V

2.ili支持888颜色而ili9486是565模式
这里需要改好几个地方,
首先是ili9488文件里初始化序列里面,把这个值由0x55改成0x66,
RISC-V

然后在ser_var里面添加上888色彩的信息,
RISC-V

然后是display的值,增加一个函数和bbp值:
RISC-V

这里面涉及的这个函数:fbtft_write_vmem24_bus8就是写入像素信息的
这个函数需要放到fbtft-core.c里,放文件最后也行
fbtft-core.c文件还需要修改fbtft_framebuffer_alloc函数的几个地方:
RISC-V

RISC-V



3.修改fbtft-core.c里面的几个函数,
首先是:fbtft_request_one_gpio:


  1.  
  2. #include "linux/gpio.h"
  3. #include "linux/of_gpio.h"
  4.  
  5. static int fbtft_request_one_gpio(struct fbtft_par *par,
  6.                   const char *name, int index,
  7.                   struct gpio_desc **gpiop)
  8. {
  9.     struct device *dev = par->info->device;
  10.     struct device_node *node = dev->of_node;
  11.     int gpio, flags, ret = 0;
  12.     enum of_gpio_flags of_flags;
  13.     if (of_find_property(node, name, NULL)) {
  14.         gpio = of_get_named_gpio_flags(node, name, index, &of_flags);
  15.         if (gpio == -ENOENT)
  16.             return 0;
  17.         if (gpio == -EPROBE_DEFER)
  18.             return gpio;
  19.         if (gpio < 0) {
  20.             dev_err(dev,
  21.                 "failed to get '%s' from DTn", name);
  22.             return gpio;
  23.         }
  24.          //active low translates to initially low
  25.         flags = (of_flags & OF_GPIO_ACTIVE_LOW) ? GPIOF_OUT_INIT_LOW :
  26.                             GPIOF_OUT_INIT_HIGH;
  27.         ret = devm_gpio_request_one(dev, gpio, flags,
  28.                         dev->driver->name);
  29.         if (ret) {
  30.             dev_err(dev,
  31.                 "gpio_request_one('%s'=%d) failed with %dn",
  32.                 name, gpio, ret);
  33.             return ret;
  34.         }
  35.  
  36.         *gpiop = gpio_to_desc(gpio);
  37.         fbtft_par_dbg(DEBUG_REQUEST_GPIOS, par, "%s: '%s' = GPIO%dn",
  38.                             __func__, name, gpio);
  39.     }
  40.  
  41.     return ret;
  42. }
  43.  
复制代码

然后是fbtft_request_gpios:


  1.  
  2. static int fbtft_request_gpios(struct fbtft_par *par)
  3. {
  4.     int i;
  5.     int ret;
  6.  
  7.     ret = fbtft_request_one_gpio(par, "reset-gpios", 0, &par->gpio.reset);
  8.     if (ret)
  9.         return ret;
  10.     ret = fbtft_request_one_gpio(par, "dc-gpios", 0, &par->gpio.dc);
  11.     if (ret)
  12.         return ret;
  13.     ret = fbtft_request_one_gpio(par, "rd-gpios", 0, &par->gpio.rd);
  14.     if (ret)
  15.         return ret;
  16.     ret = fbtft_request_one_gpio(par, "wr-gpios", 0, &par->gpio.wr);
  17.     if (ret)
  18.         return ret;
  19.     ret = fbtft_request_one_gpio(par, "cs-gpios", 0, &par->gpio.cs);
  20.     if (ret)
  21.         return ret;
  22.     ret = fbtft_request_one_gpio(par, "latch-gpios", 0, &par->gpio.latch);
  23.     if (ret)
  24.         return ret;
  25.     for (i = 0; i < 16; i++) {
  26.         ret = fbtft_request_one_gpio(par, "db-gpios", i,
  27.                          &par->gpio.db[i]);
  28.         if (ret)
  29.             return ret;
  30.         ret = fbtft_request_one_gpio(par, "led-gpios", i,
  31.                          &par->gpio.led[i]);
  32.         if (ret)
  33.             return ret;
  34.         ret = fbtft_request_one_gpio(par, "aux-gpios", i,
  35.                          &par->gpio.aux[i]);
  36.         if (ret)
  37.             return ret;
  38.     }
  39.  
  40.     return 0;
  41. }
复制代码

最后是fbtft_reset:


  1. static void fbtft_reset(struct fbtft_par *par)
  2. {
  3.         if (!par->gpio.reset)
  4.                 return;
  5.  
  6.         fbtft_par_dbg(DEBUG_RESET, par, "%s()n", __func__);
  7.  
  8.         gpiod_set_value_cansleep(par->gpio.reset, 1);
  9.         usleep_range(20, 40);
  10.         gpiod_set_value_cansleep(par->gpio.reset, 0);
  11.         msleep(120);
  12.         gpiod_set_value_cansleep(par->gpio.reset, 1);
  13. //        gpiod_set_value_cansleep(par->gpio.cs, 1);  /* Activate chip */
  14. }
复制代码

这里主要是那个复位脚重启一下,原来的是把cs脚使能,不知道为什么是这样

烧录之后如果不能显示,可以尝试把系统原来的显示去掉(不知道有没有作用,可以试下,我实验过程中去掉了):
RISC-V

大概主要碰到的问题就是这些吧,可能有些忘记了,想起来的时候再添加.

三.显示logo,
这里需要修改linux配置,配置这个参数,这个就是显示logo的
-> Device Drivers                                                                                                                                                          │  
  │       -> Graphics support                                                                                                                                                      │  
  │         -> Bootup logo (LOGO [=y])   
RISC-V


效果看一看(后面有时间换成咱们赛昉的logo试试):
RISC-V

如果还想把终端的打印信息在屏幕上显示出来,就配置这个
│     -> Device Drivers                                                                                                                                                          │  
  │       -> Graphics support                                                                                                                                                      │  
  │         -> Console display driver support   
RISC-V


但是这还不够,还需要把console额外赋个值,在bootarg那里增加一个参数(这个功能很酷炫,但实际没什么用,建议不要搞):
这里我们把固件编译出来试一下是可以将字符显示在屏幕上的,在linux终端上输入:
echo hello >/dev/tty0
可以看到hello字符出现在屏幕上了.

四.总结
到这里一款LCD屏幕的移植过程大致就完了,整个步骤其实不是很繁琐,其实最有意义的是这中间找问题的过程,借此机会大致弄明白了设备数的写法
也知道了一些驱动方面的知识,后续等tf卡刷方式速度快点再玩了,准备的实验有触摸,QT写界面等等,拭目以待吧.
再次吐槽发烧友的发帖机制,太难受,要赶紧把图片粘贴功能弄出来

打开APP阅读更多精彩内容
声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉

全部0条评论

快来发表一下你的评论吧 !

×
20
完善资料,
赚取积分