描述
本文来源电子发烧友社区,作者:cszzlsw, 帖子地址:
https://bbs.elecfans.com/jishu_2287667_1_1.html1.前言前面两篇帖子咱们回顾了从头制作系统并运行homeassistant,其中不可谓不辛酸,然后我就在想再做个啥,找了一下自己库存的外设,有一个tft屏带触摸芯片的,一想,这不正好的嘛,可以玩玩图形编程了,老听说linux开发板的qt编程咱也没见过啊,说干就干,这就开始了.2.移植驱动并首先尝试首先我的屏主控是lil9488的,分辨率320*480,据说再往上就得上rgb屏了,这咱没玩过也不懂,把手头上这个整明白先.这个在git仓库:https://github.com/starfive-tech/freelight-u-sdk,给的linux内核里面找了一圈,也没见,然后发现有一个芯片是ili9486的,跟我这差一位数,一想差不多,这个开始弄吧,把文件拷过来,然后改吧:
然后把所有涉及9486的地方都改成9488,然后是设备树里面改吧改吧(改了好多次,也是第一次写设备树):
这款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里这么改:
2.ili支持888颜色而ili9486是565模式这里需要改好几个地方,首先是ili9488文件里初始化序列里面,把这个值由0x55改成0x66,
然后在ser_var里面添加上888色彩的信息,
然后是display的值,增加一个函数和bbp值:
这里面涉及的这个函数:fbtft_write_vmem24_bus8就是写入像素信息的这个函数需要放到fbtft-core.c里,放文件最后也行fbtft-core.c文件还需要修改fbtft_framebuffer_alloc函数的几个地方:
3.修改fbtft-core.c里面的几个函数,首先是:fbtft_request_one_gpio:
-
-
#include "linux/gpio.h"
-
#include "linux/of_gpio.h"
-
-
static int fbtft_request_one_gpio(struct fbtft_par *par,
-
const char *name, int index,
-
struct gpio_desc **gpiop)
-
{
-
struct device *dev = par->info->device;
-
struct device_node *node = dev->of_node;
-
int gpio, flags, ret = 0;
-
enum of_gpio_flags of_flags;
-
if (of_find_property(node, name, NULL)) {
-
gpio = of_get_named_gpio_flags(node, name, index, &of_flags);
-
if (gpio == -ENOENT)
-
return 0;
-
if (gpio == -EPROBE_DEFER)
-
return gpio;
-
if (gpio < 0) {
-
dev_err(dev,
-
"failed to get '%s' from DTn", name);
-
return gpio;
-
}
-
//active low translates to initially low
-
flags = (of_flags & OF_GPIO_ACTIVE_LOW) ? GPIOF_OUT_INIT_LOW :
-
GPIOF_OUT_INIT_HIGH;
-
ret = devm_gpio_request_one(dev, gpio, flags,
-
dev->driver->name);
-
if (ret) {
-
dev_err(dev,
-
"gpio_request_one('%s'=%d) failed with %dn",
-
name, gpio, ret);
-
return ret;
-
}
-
-
*gpiop = gpio_to_desc(gpio);
-
fbtft_par_dbg(DEBUG_REQUEST_GPIOS, par, "%s: '%s' = GPIO%dn",
-
__func__, name, gpio);
-
}
-
-
return ret;
-
}
-
复制代码
然后是fbtft_request_gpios:
-
-
static int fbtft_request_gpios(struct fbtft_par *par)
-
{
-
int i;
-
int ret;
-
-
ret = fbtft_request_one_gpio(par, "reset-gpios", 0, &par->gpio.reset);
-
if (ret)
-
return ret;
-
ret = fbtft_request_one_gpio(par, "dc-gpios", 0, &par->gpio.dc);
-
if (ret)
-
return ret;
-
ret = fbtft_request_one_gpio(par, "rd-gpios", 0, &par->gpio.rd);
-
if (ret)
-
return ret;
-
ret = fbtft_request_one_gpio(par, "wr-gpios", 0, &par->gpio.wr);
-
if (ret)
-
return ret;
-
ret = fbtft_request_one_gpio(par, "cs-gpios", 0, &par->gpio.cs);
-
if (ret)
-
return ret;
-
ret = fbtft_request_one_gpio(par, "latch-gpios", 0, &par->gpio.latch);
-
if (ret)
-
return ret;
-
for (i = 0; i < 16; i++) {
-
ret = fbtft_request_one_gpio(par, "db-gpios", i,
-
&par->gpio.db[i]);
-
if (ret)
-
return ret;
-
ret = fbtft_request_one_gpio(par, "led-gpios", i,
-
&par->gpio.led[i]);
-
if (ret)
-
return ret;
-
ret = fbtft_request_one_gpio(par, "aux-gpios", i,
-
&par->gpio.aux[i]);
-
if (ret)
-
return ret;
-
}
-
-
return 0;
-
}
复制代码
最后是fbtft_reset:
-
static void fbtft_reset(struct fbtft_par *par)
-
{
-
if (!par->gpio.reset)
-
return;
-
-
fbtft_par_dbg(DEBUG_RESET, par, "%s()n", __func__);
-
-
gpiod_set_value_cansleep(par->gpio.reset, 1);
-
usleep_range(20, 40);
-
gpiod_set_value_cansleep(par->gpio.reset, 0);
-
msleep(120);
-
gpiod_set_value_cansleep(par->gpio.reset, 1);
-
// gpiod_set_value_cansleep(par->gpio.cs, 1); /* Activate chip */
-
}
复制代码
这里主要是那个复位脚重启一下,原来的是把cs脚使能,不知道为什么是这样烧录之后如果不能显示,可以尝试把系统原来的显示去掉(不知道有没有作用,可以试下,我实验过程中去掉了):
大概主要碰到的问题就是这些吧,可能有些忘记了,想起来的时候再添加.三.显示logo,这里需要修改linux配置,配置这个参数,这个就是显示logo的-> Device Drivers │
│ -> Graphics support │
│ -> Bootup logo (LOGO [=y])
效果看一看(后面有时间换成咱们赛昉的logo试试):
如果还想把终端的打印信息在屏幕上显示出来,就配置这个│ -> Device Drivers │
│ -> Graphics support │
│ -> Console display driver support
但是这还不够,还需要把console额外赋个值,在bootarg那里增加一个参数(这个功能很酷炫,但实际没什么用,建议不要搞):
这里我们把固件编译出来试一下是可以将字符显示在屏幕上的,在linux终端上输入:
echo hello >/dev/tty0
可以看到hello字符出现在屏幕上了.
四.总结
到这里一款LCD屏幕的移植过程大致就完了,整个步骤其实不是很繁琐,其实最有意义的是这中间找问题的过程,借此机会大致弄明白了设备数的写法
也知道了一些驱动方面的知识,后续等tf卡刷方式速度快点再玩了,准备的实验有触摸,QT写界面等等,拭目以待吧.
再次吐槽发烧友的发帖机制,太难受,要赶紧把图片粘贴功能弄出来
打开APP阅读更多精彩内容