RISC-V技术论坛
直播中

cszzlsw

9年用户 202经验值
擅长:嵌入式技术
私信 关注
[经验]

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

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

2.移植驱动并首先尝试
首先我的屏主控是lil9488的,分辨率320*480,据说再往上就得上rgb屏了,这咱没玩过也不懂,把手头上这个整明白先.
这个在git仓库:https://github.com/starfive-tech/freelight-u-sdk,给的linux内核里面找了一圈,也没见,
然后发现有一个芯片是ili9486的,跟我这差一位数,一想差不多,这个开始弄吧,把文件拷过来,然后改吧:
QQ浏览器截图20220617232251.png
然后把所有涉及9486的地方都改成9488,
然后是设备树里面改吧改吧(改了好多次,也是第一次写设备树):
微信截图_20220619225741.png
这款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里这么改:
微信截图_20220619230751.png
2.ili支持888颜色而ili9486是565模式
这里需要改好几个地方,
首先是ili9488文件里初始化序列里面,把这个值由0x55改成0x66,
微信截图_20220619230724.png
然后在ser_var里面添加上888色彩的信息,
微信截图_20220619230230.png
然后是display的值,增加一个函数和bbp值:
微信截图_20220619230918.png
这里面涉及的这个函数:fbtft_write_vmem24_bus8就是写入像素信息的
这个函数需要放到fbtft-core.c里,放文件最后也行
fbtft-core.c文件还需要修改fbtft_framebuffer_alloc函数的几个地方:
微信截图_20220619230034.png
微信截图_20220619230102.png


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

  1. #include "linux/gpio.h"
  2. #include "linux/of_gpio.h"

  3. static int fbtft_request_one_gpio(struct fbtft_par *par,
  4.                   const char *name, int index,
  5.                   struct gpio_desc **gpiop)
  6. {
  7.     struct device *dev = par->info->device;
  8.     struct device_node *node = dev->of_node;
  9.     int gpio, flags, ret = 0;
  10.     enum of_gpio_flags of_flags;
  11.     if (of_find_property(node, name, NULL)) {
  12.         gpio = of_get_named_gpio_flags(node, name, index, &of_flags);
  13.         if (gpio == -ENOENT)
  14.             return 0;
  15.         if (gpio == -EPROBE_DEFER)
  16.             return gpio;
  17.         if (gpio < 0) {
  18.             dev_err(dev,
  19.                 "failed to get '%s' from DTn", name);
  20.             return gpio;
  21.         }
  22.          //active low translates to initially low
  23.         flags = (of_flags & OF_GPIO_ACTIVE_LOW) ? GPIOF_OUT_INIT_LOW :
  24.                             GPIOF_OUT_INIT_HIGH;
  25.         ret = devm_gpio_request_one(dev, gpio, flags,
  26.                         dev->driver->name);
  27.         if (ret) {
  28.             dev_err(dev,
  29.                 "gpio_request_one('%s'=%d) failed with %dn",
  30.                 name, gpio, ret);
  31.             return ret;
  32.         }

  33.         *gpiop = gpio_to_desc(gpio);
  34.         fbtft_par_dbg(DEBUG_REQUEST_GPIOS, par, "%s: '%s' = GPIO%dn",
  35.                             __func__, name, gpio);
  36.     }

  37.     return ret;
  38. }
然后是
fbtft_request_gpios:

  1. static int fbtft_request_gpios(struct fbtft_par *par)
  2. {
  3.     int i;
  4.     int ret;

  5.     ret = fbtft_request_one_gpio(par, "reset-gpios", 0, &par->gpio.reset);
  6.     if (ret)
  7.         return ret;
  8.     ret = fbtft_request_one_gpio(par, "dc-gpios", 0, &par->gpio.dc);
  9.     if (ret)
  10.         return ret;
  11.     ret = fbtft_request_one_gpio(par, "rd-gpios", 0, &par->gpio.rd);
  12.     if (ret)
  13.         return ret;
  14.     ret = fbtft_request_one_gpio(par, "wr-gpios", 0, &par->gpio.wr);
  15.     if (ret)
  16.         return ret;
  17.     ret = fbtft_request_one_gpio(par, "cs-gpios", 0, &par->gpio.cs);
  18.     if (ret)
  19.         return ret;
  20.     ret = fbtft_request_one_gpio(par, "latch-gpios", 0, &par->gpio.latch);
  21.     if (ret)
  22.         return ret;
  23.     for (i = 0; i < 16; i++) {
  24.         ret = fbtft_request_one_gpio(par, "db-gpios", i,
  25.                          &par->gpio.db[i]);
  26.         if (ret)
  27.             return ret;
  28.         ret = fbtft_request_one_gpio(par, "led-gpios", i,
  29.                          &par->gpio.led[i]);
  30.         if (ret)
  31.             return ret;
  32.         ret = fbtft_request_one_gpio(par, "aux-gpios", i,
  33.                          &par->gpio.aux[i]);
  34.         if (ret)
  35.             return ret;
  36.     }

  37.     return 0;
  38. }
最后是
fbtft_reset:
  1. static void fbtft_reset(struct fbtft_par *par)
  2. {
  3.         if (!par->gpio.reset)
  4.                 return;

  5.         fbtft_par_dbg(DEBUG_RESET, par, "%s()n", __func__);

  6.         gpiod_set_value_cansleep(par->gpio.reset, 1);
  7.         usleep_range(20, 40);
  8.         gpiod_set_value_cansleep(par->gpio.reset, 0);
  9.         msleep(120);
  10.         gpiod_set_value_cansleep(par->gpio.reset, 1);
  11. //        gpiod_set_value_cansleep(par->gpio.cs, 1);  /* Activate chip */
  12. }
这里主要是那个复位脚重启一下,原来的是把cs脚使能,不知道为什么是这样


烧录之后如果不能显示,可以尝试把系统原来的显示去掉(不知道有没有作用,可以试下,我实验过程中去掉了):
微信截图_20220619231148.png
大概主要碰到的问题就是这些吧,可能有些忘记了,想起来的时候再添加.

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

效果看一看(后面有时间换成咱们赛昉的logo试试):
微信图片_20220619231854.jpg
如果还想把终端的打印信息在屏幕上显示出来,就配置这个
│     -> Device Drivers                                                                                                                                                          │  
  │       -> Graphics support                                                                                                                                                      │  
  │         -> Console display driver support   
微信截图_20220619231827.png

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

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

回帖(1)

华仔stm32

2022-6-20 22:41:10
牛呀,恭喜成功!
举报

更多回帖

发帖
×
20
完善资料,
赚取积分