【spi】Milk-V Duo点亮st7735

前言

本文介绍milkv-duo加载st7735的lcd屏幕,以及屏幕显示log。

参考文章:
记录为Linux配置spi屏幕(st7735s)
https://community.milkv.io/t/milk-v-duo-spi-st7789/131

一、电路图

1.1 pin设置

打开spi2的引脚
duo-buildroot-sdk\build\boards\cv180x\cv1800b_milkv_duo_sd\u-boot\cvi_board_init.c

	//spi2 -- st7735
	pinmux_config(PINMUX_SPI2);
	PINMUX_CONFIG(SPINOR_MISO, XGPIOA_23);	//DC
	PINMUX_CONFIG(SD0_PWR_EN, XGPIOA_14);	//BL
	PINMUX_CONFIG(SPK_EN, XGPIOA_15);		//RES

注意这里的DC,属于gpioA

这里的pinmux_config(PINMUX_SPI2);可以在源码中找到对应的配置

		case PINMUX_SPI2:
			PINMUX_CONFIG(SD1_CMD, SPI2_SDO);
			PINMUX_CONFIG(SD1_CLK, SPI2_SCK);
			PINMUX_CONFIG(SD1_D0, SPI2_SDI);
			PINMUX_CONFIG(SD1_D3, SPI2_CS_X);
			break;

1.2 dtsi配置

build\boards\cv180x\cv1800b_milkv_duo_sd\dts_riscv\cv1800b_milkv_duo_sd.dts

&spi2 {
	status = "okay";

    /delete-node/ spidev@0;

	st7789v: st7789v@0{
		compatible = "sitronix,st7789v";
		reg = <0>;
		status = "okay";
		spi-max-frequency = <48000000>;
		spi-cpol;
		spi-cpha;
		rotate = <90>;
		fps = <30>;
		rgb;
		buswidth = <8>;

		dc-gpios = <&porta 23 GPIO_ACTIVE_HIGH>;	//DC
		reset-gpios = <&porta 15 GPIO_ACTIVE_HIGH>; //RES
		led-gpios = <&porta 14 GPIO_ACTIVE_HIGH>; //BL

		debug = <0x0>;
	};
};

这里的gpio要结合你的引脚实现。

二、加载驱动

仿照参考文章,使用st7789v代码改动到st7735上。
记录为Linux配置spi屏幕(st7735s)

编译条件
duo-buildroot-sdk\linux_5.10\drivers\staging\fbtft\Makefile

obj-$(CONFIG_FB_TFT_ST7735R)     += fb_st7735r.o
obj-$(CONFIG_FB_TFT_ST7789V)     += fb_st7789v.o

借用st7789v的代码

st7735s和st7735r一致

引用:https://www.cnblogs.com/milton/p/15614304.html#St7735%E4%BB%8B%E7%BB%8D 
ST7735的配置
因为ST7735存在多个型号, 对于ST7735R和ST7735S, 使用默认的初始化方法

duo-buildroot-sdk\build\boards\cv180x\cv1800b_milkv_duo_sd\linux\ cvitek_cv1800b_milkv_duo_sd_defconfig

# spi lcd -- st7735s
CONFIG_SPI=y
CONFIG_SPI_MASTER=y
CONFIG_SPI_DESIGNWARE=y
CONFIG_SPI_DW_MMIO=y
CONFIG_SPI_SPIDEV=y
CONFIG_FB=y
CONFIG_FB_TFT=y
# CONFIG_FB_TFT_ST7735R=y
CONFIG_FB_TFT_ST7789V=y

编译过程提示

  CC      drivers/staging/fbtft/fb_st7789v.o

三、更改驱动

3.1 st7789v驱动

duo-buildroot-sdk\linux_5.10\drivers\staging\fbtft\fb_st7789v.c

参考来源:https://blog.csdn.net/qq_46604211/article/details/116449891

3.1.2 接口更新

由于驱动是st7789v,我的屏幕是st7735,需要更改一些初始化接口。

注意gamma需要注释掉for循环。

// st7735
static int init_display(struct fbtft_par *par)
{
	par->fbtftops.reset(par);//硬复位

	mdelay(50);
	write_reg(par,0x11);//软复位
	mdelay(100);
	//下面添加初始化函数write_reg 参数分别为:结构体指针,写命令,写数据....(后都为数据)
	//ST7735s Frame Rate
	write_reg(par,0xB1,0x05,0x3c,0x3c); 

	write_reg(par,0xB2,0x05,0x3c,0x3c);  

	write_reg(par,0xB3,0x05,0x3c,0x3c,0x05,0x3c,0x3c); 
 
	
	write_reg(par,0xB4,0x03); //Column inversion 
	
	//ST7735s Power Sequence
	write_reg(par,0xC0,0x28,0x08,0x04); 

	write_reg(par,0xC1,0xc0); 

	write_reg(par,0xC2,0x0d,0x00); 
	
	write_reg(par,0xC3,0x8d,0x2a); //VCOM 
	
	write_reg(par,0xc4,0x8d,0xee); //MX, MY, RGB mode  
	write_reg(par,0xc5,0x1a);
	write_reg(par,0x36,0xc0);
	//ST7735s Gamma Sequence
	write_reg(par,0xe0,0x04,0x22,0x07,0x0a,0x2e,0x30,0x25,0x2a,0x28,0x26,0x2e,0x3a,0x00,0x01,0x03,0x13); 

	write_reg(par,0xe1,0x04,0x16,0x06,0x0d,0x2d,0x26,0x23,0x27,0x27,0x25,0x2d,0x3b,0x00,0x01,0x04,0x13);  
	
	write_reg(par,0x3A,0x05); //65k mode  
	
	write_reg(par,0x29);//Display on
	mdelay(100);

	return 0;
}

static int set_gamma(struct fbtft_par *par, u32 *curves)
{
	int i;
	int j;
	int c; /* curve index offset */

	/*
	 * Bitmasks for gamma curve command parameters.
	 * The masks are the same for both positive and negative voltage
	 * gamma curves.
	 */
	static const u8 gamma_par_mask[] = {
		0xFF, /* V63[3:0], V0[3:0]*/
		0x3F, /* V1[5:0] */
		0x3F, /* V2[5:0] */
		0x1F, /* V4[4:0] */
		0x1F, /* V6[4:0] */
		0x3F, /* J0[1:0], V13[3:0] */
		0x7F, /* V20[6:0] */
		0x77, /* V36[2:0], V27[2:0] */
		0x7F, /* V43[6:0] */
		0x3F, /* J1[1:0], V50[3:0] */
		0x1F, /* V57[4:0] */
		0x1F, /* V59[4:0] */
		0x3F, /* V61[5:0] */
		0x3F, /* V62[5:0] */
	};

	for (i = 0; i < par->gamma.num_curves; i++) {
		c = i * par->gamma.num_values;
		// for (j = 0; j < par->gamma.num_values; j++)
		// 	curves[c + j] &= gamma_par_mask[j];
		write_reg(par, PVGAMCTRL + i,
			  curves[c + 0],  curves[c + 1],  curves[c + 2],
			  curves[c + 3],  curves[c + 4],  curves[c + 5],
			  curves[c + 6],  curves[c + 7],  curves[c + 8],
			  curves[c + 9],  curves[c + 10], curves[c + 11],
			  curves[c + 12], curves[c + 13]);
	}
	return 0;
}

3.1.3 更改宽高

st7735的屏幕宽高为128*160。

duo-buildroot-sdk\linux_5.10\drivers\staging\fbtft\fb_st7789v.c

static struct fbtft_display display = {
	.regwidth = 8,
	.width = 128,//240,
	.height = 160,//320,
	.gamma_num = 2,
	.gamma_len = 14,
	.gamma = HSD20_IPS_GAMMA,
	.fbtftops = {
		.init_display = init_display,
		.set_var = set_var,
		.set_gamma = set_gamma,
		.blank = blank,
	},
};

3.2 fbtft-core

duo-buildroot-sdk\linux_5.10\drivers\staging\fbtft\fbtft-core.c

3.2.1增加头文件

#include <linux/gpio.h> //add
#include <linux/of_gpio.h> //add

3.2.2 更改接口

这是由于内核版本更新导致的,一些接口可能需要改动。
dtsi中的属性加上-gpio可以自行选择。
reset重新实现。

//https://blog.csdn.net/qq_46604211/article/details/116449891
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 DT\n", 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 %d\n",
                name, gpio, ret);
            return ret;
        }

        *gpiop = gpio_to_desc(gpio);
        fbtft_par_dbg(DEBUG_REQUEST_GPIOS, par, "%s: '%s' = GPIO%d\n",
                            __func__, name, gpio);
    }

    return ret;
}

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;
}

//作者:Leesans https://www.bilibili.com/read/cv9947785/ 出处:bilibili
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);
    msleep(10);
    gpiod_set_value_cansleep(par->gpio.reset, 0);
    msleep(200);
    gpiod_set_value_cansleep(par->gpio.reset, 1);
    msleep(10);
}

四、屏幕显示结果

驱动正常加载。

通过如下两个命令可以测试屏幕功能。

cat /dev/urandom > /dev/fb0 	//显示花屏
cat /dev/zero > /dev/fb0  		//清空屏幕
echo xxxx > /dev/fb0 				//显示点个数

仔细看右上角,有一条颜色不同的线,很短,你可以echo后面跟很长的x,x越多,线越长。
这是手动赋值给像素点的。

五、用屏幕显示log

参考来源:

5.1 添加编译模块

duo-buildroot-sdk\build\boards\cv180x\cv1800b_milkv_duo_sd\linux\cvitek_cv1800b_milkv_duo_sd_defconfig

# lcd display kernel log
CONFIG_TTY=y
CONFIG_VT=y
CONFIG_CONSOLE_TRANSLATIONS=y
CONFIG_VT_CONSOLE=y
CONFIG_HW_CONSOLE=y
CONFIG_VT_HW_CONSOLE_BINDING=y
CONFIG_FB=y
CONFIG_FB_CMDLINE=y
CONFIG_FB_NOTIFY=y
CONFIG_FONT_SUPPORT=y
CONFIG_FONTS=y
CONFIG_FONT_8x16=y
CONFIG_VGA_CONSOLE=y
CONFIG_DUMMY_CONSOLE=y
CONFIG_DUMMY_CONSOLE_COLUMNS=80
CONFIG_DUMMY_CONSOLE_ROWS=25
CONFIG_FRAMEBUFFER_CONSOLE=y
CONFIG_FRAMEBUFFER_CONSOLE_DETECT_PRIMARY=y

5.2 添加tty0用于显示log

duo-buildroot-sdk\u-boot-2021.10\include\configs\cv180x-asic.h

#define SET_BOOTARGS "setenv bootargs ${root} ${mtdparts} " \
		"console=tty0 console=$consoledev,$baudrate $othbootargs;"

5.3 显示

git diff
可以将如下内容存到txt,直接通过git导入。

diff --git a/build/boards/cv180x/cv1800b_milkv_duo_sd/dts_riscv/cv1800b_milkv_duo_sd.dts b/build/boards/cv180x/cv1800b_milkv_duo_sd/dts_riscv/cv1800b_milkv_duo_sd.dts
index 8f66e40d8..be9c27abd 100644
--- a/build/boards/cv180x/cv1800b_milkv_duo_sd/dts_riscv/cv1800b_milkv_duo_sd.dts
+++ b/build/boards/cv180x/cv1800b_milkv_duo_sd/dts_riscv/cv1800b_milkv_duo_sd.dts
@@ -8,6 +8,34 @@
 	snsr-reset = <&portc 8 GPIO_ACTIVE_LOW>, <&portc 8 GPIO_ACTIVE_LOW>, <&portc 8 GPIO_ACTIVE_LOW>;
 };
 
+&spi2 {
+	status = "okay";
+
+    /delete-node/ spidev@0;
+
+
+	st7789v: st7789v@0{
+		compatible = "sitronix,st7789v";
+		reg = <0>;
+		status = "okay";
+		spi-max-frequency = <48000000>;
+		spi-cpol;
+		spi-cpha;
+		rotate = <90>;
+		fps = <30>;
+		rgb;
+		buswidth = <8>;
+
+		//dc-gpios = <&port 21 GPIO_ACTIVE_HIGH>;	//DC
+		dc-gpios = <&porta 23 GPIO_ACTIVE_HIGH>;	//DC
+		reset-gpios = <&porta 15 GPIO_ACTIVE_HIGH>; //RES
+		led-gpios = <&porta 14 GPIO_ACTIVE_HIGH>; //BL
+
+		debug = <0x0>;
+	};
+
+};
+
 / {
 
 };
diff --git a/build/boards/cv180x/cv1800b_milkv_duo_sd/linux/cvitek_cv1800b_milkv_duo_sd_defconfig b/build/boards/cv180x/cv1800b_milkv_duo_sd/linux/cvitek_cv1800b_milkv_duo_sd_defconfig
index 6c05a05df..82fcb0298 100644
--- a/build/boards/cv180x/cv1800b_milkv_duo_sd/linux/cvitek_cv1800b_milkv_duo_sd_defconfig
+++ b/build/boards/cv180x/cv1800b_milkv_duo_sd/linux/cvitek_cv1800b_milkv_duo_sd_defconfig
@@ -260,3 +260,56 @@ CONFIG_ADVISE_SYSCALLS=n
 CONFIG_SIGNALFD=n
 CONFIG_TIMERFD=n
 CONFIG_EPOLL=n
+
+# -----------------------------------------------------------
+# youkai add
+# CONFIG_IIO=y
+# CONFIG_BH1750=y
+# CONFIG_TCS3472=y
+# CONFIG_INV_MPU6050_IIO=y
+# CONFIG_INV_MPU6050_I2C=y
+# CONFIG_SENSORS_HMC5843=y
+# CONFIG_SENSORS_HMC5843_I2C=y
+
+# sensor bmp180
+# CONFIG_BMP280=y
+# CONFIG_BMP280_I2C=y
+
+# spi lcd -- st7735
+CONFIG_SPI=y
+CONFIG_SPI_MASTER=y
+CONFIG_SPI_DESIGNWARE=y
+CONFIG_SPI_DW_MMIO=y
+CONFIG_SPI_SPIDEV=y
+CONFIG_FB=y
+CONFIG_FB_TFT=y
+# CONFIG_FB_TFT_ST7735R=y
+CONFIG_FB_TFT_ST7789V=y
+
+# CONFIG_TINYDRM_ST7735R=y
+
+# lcd display kernel log
+CONFIG_TTY=y
+CONFIG_VT=y
+CONFIG_CONSOLE_TRANSLATIONS=y
+CONFIG_VT_CONSOLE=y
+CONFIG_HW_CONSOLE=y
+CONFIG_VT_HW_CONSOLE_BINDING=y
+CONFIG_FB=y
+CONFIG_FB_CMDLINE=y
+CONFIG_FB_NOTIFY=y
+CONFIG_FONT_SUPPORT=y
+CONFIG_FONTS=y
+CONFIG_FONT_8x16=y
+CONFIG_VGA_CONSOLE=y
+CONFIG_DUMMY_CONSOLE=y
+CONFIG_DUMMY_CONSOLE_COLUMNS=80
+CONFIG_DUMMY_CONSOLE_ROWS=25
+CONFIG_FRAMEBUFFER_CONSOLE=y
+CONFIG_FRAMEBUFFER_CONSOLE_DETECT_PRIMARY=y
+CONFIG_VGA_CONSOLE=y
+
+# # no compatible
+# CONFIG_TCS3472=y
+
+# -----------------------------------------------------------
\ No newline at end of file
diff --git a/build/boards/cv180x/cv1800b_milkv_duo_sd/u-boot/cvi_board_init.c b/build/boards/cv180x/cv1800b_milkv_duo_sd/u-boot/cvi_board_init.c
index 25c95da75..cf14e1a0f 100644
--- a/build/boards/cv180x/cv1800b_milkv_duo_sd/u-boot/cvi_board_init.c
+++ b/build/boards/cv180x/cv1800b_milkv_duo_sd/u-boot/cvi_board_init.c
@@ -15,18 +15,30 @@ int cvi_board_init(void)
 	PINMUX_CONFIG(SPINOR_CS_X, XGPIOA_24);
 	PINMUX_CONFIG(IIC0_SDA, XGPIOA_29);
 	PINMUX_CONFIG(IIC0_SCL, XGPIOA_28);
-	// sd1
-	PINMUX_CONFIG(SD1_D3, PWR_GPIO_18);
-	PINMUX_CONFIG(SD1_D2, PWR_GPIO_19);
-	PINMUX_CONFIG(SD1_D1, PWR_GPIO_20);
-	PINMUX_CONFIG(SD1_D0, PWR_GPIO_21);
-	PINMUX_CONFIG(SD1_CMD, PWR_GPIO_22);
-	PINMUX_CONFIG(SD1_CLK, PWR_GPIO_23);
+	// // sd1
+	// PINMUX_CONFIG(SD1_D3, PWR_GPIO_18);
+	// PINMUX_CONFIG(SD1_D2, PWR_GPIO_19);
+	// PINMUX_CONFIG(SD1_D1, PWR_GPIO_20);
+	// PINMUX_CONFIG(SD1_D0, PWR_GPIO_21);
+	// PINMUX_CONFIG(SD1_CMD, PWR_GPIO_22);
+	// PINMUX_CONFIG(SD1_CLK, PWR_GPIO_23);
 
 	//default to gpio
 	PINMUX_CONFIG(PAD_MIPIRX1P, XGPIOC_9);
 	PINMUX_CONFIG(PAD_MIPIRX0N, XGPIOC_10);
 	PINMUX_CONFIG(USB_VBUS_DET, XGPIOB_6);
 	PINMUX_CONFIG(PWR_SEQ2, PWR_GPIO_4);
+
+	//spi2 -- st7735
+	// PINMUX_CONFIG(SD1_D3, SPI2_CS_X);		//CS
+	// PINMUX_CONFIG(SD1_D0, SPI2_SDI);			
+	// PINMUX_CONFIG(SD1_CMD, SPI2_SDO);		//SDA
+	// PINMUX_CONFIG(SD1_CLK, SPI2_SCK);		//SCL
+	pinmux_config(PINMUX_SPI2);
+	//PINMUX_CONFIG(SD1_D0, PWR_GPIO_21);		//DC
+	PINMUX_CONFIG(SPINOR_MISO, XGPIOA_23);		//DC
+	PINMUX_CONFIG(SD0_PWR_EN, XGPIOA_14);	//BL
+	PINMUX_CONFIG(SPK_EN, XGPIOA_15);		//RES
+
 	return 0;
 }
diff --git a/build/boards/default/dts/cv180x/cv180x_base.dtsi b/build/boards/default/dts/cv180x/cv180x_base.dtsi
index fcc07d2ff..b05f088a8 100644
--- a/build/boards/default/dts/cv180x/cv180x_base.dtsi
+++ b/build/boards/default/dts/cv180x/cv180x_base.dtsi
@@ -232,6 +232,7 @@
 		clocks = <&clk CV180X_CLK_SPI>;
 		#address-cells = <1>;
 		#size-cells = <0>;
+		bias-pull-up;	//youkai add
 	};
 
 	spi3:spi3@041B0000 {
diff --git a/linux_5.10/drivers/staging/fbtft/fb_st7789v.c b/linux_5.10/drivers/staging/fbtft/fb_st7789v.c
index 3a280cc18..58f5752b5 100644
--- a/linux_5.10/drivers/staging/fbtft/fb_st7789v.c
+++ b/linux_5.10/drivers/staging/fbtft/fb_st7789v.c
@@ -80,67 +80,112 @@ enum st7789v_command {
  *
  * Return: 0 on success, < 0 if error occurred.
  */
-static int init_display(struct fbtft_par *par)
-{
-	/* turn off sleep mode */
-	write_reg(par, MIPI_DCS_EXIT_SLEEP_MODE);
-	mdelay(120);
+// static int init_display(struct fbtft_par *par)
+// {
+// 	/* turn off sleep mode */
+// 	write_reg(par, MIPI_DCS_EXIT_SLEEP_MODE);
+// 	mdelay(120);
 
-	/* set pixel format to RGB-565 */
-	write_reg(par, MIPI_DCS_SET_PIXEL_FORMAT, MIPI_DCS_PIXEL_FMT_16BIT);
-	if (HSD20_IPS)
-		write_reg(par, PORCTRL, 0x05, 0x05, 0x00, 0x33, 0x33);
+// 	/* set pixel format to RGB-565 */
+// 	write_reg(par, MIPI_DCS_SET_PIXEL_FORMAT, MIPI_DCS_PIXEL_FMT_16BIT);
+// 	if (HSD20_IPS)
+// 		write_reg(par, PORCTRL, 0x05, 0x05, 0x00, 0x33, 0x33);
 
-	else
-		write_reg(par, PORCTRL, 0x08, 0x08, 0x00, 0x22, 0x22);
+// 	else
+// 		write_reg(par, PORCTRL, 0x08, 0x08, 0x00, 0x22, 0x22);
 
-	/*
-	 * VGH = 13.26V
-	 * VGL = -10.43V
-	 */
-	if (HSD20_IPS)
-		write_reg(par, GCTRL, 0x75);
-	else
-		write_reg(par, GCTRL, 0x35);
+// 	/*
+// 	 * VGH = 13.26V
+// 	 * VGL = -10.43V
+// 	 */
+// 	if (HSD20_IPS)
+// 		write_reg(par, GCTRL, 0x75);
+// 	else
+// 		write_reg(par, GCTRL, 0x35);
 
-	/*
-	 * VDV and VRH register values come from command write
-	 * (instead of NVM)
-	 */
-	write_reg(par, VDVVRHEN, 0x01, 0xFF);
+// 	/*
+// 	 * VDV and VRH register values come from command write
+// 	 * (instead of NVM)
+// 	 */
+// 	write_reg(par, VDVVRHEN, 0x01, 0xFF);
 
-	/*
-	 * VAP =  4.1V + (VCOM + VCOM offset + 0.5 * VDV)
-	 * VAN = -4.1V + (VCOM + VCOM offset + 0.5 * VDV)
-	 */
-	if (HSD20_IPS)
-		write_reg(par, VRHS, 0x13);
-	else
-		write_reg(par, VRHS, 0x0B);
+// 	/*
+// 	 * VAP =  4.1V + (VCOM + VCOM offset + 0.5 * VDV)
+// 	 * VAN = -4.1V + (VCOM + VCOM offset + 0.5 * VDV)
+// 	 */
+// 	if (HSD20_IPS)
+// 		write_reg(par, VRHS, 0x13);
+// 	else
+// 		write_reg(par, VRHS, 0x0B);
 
-	/* VDV = 0V */
-	write_reg(par, VDVS, 0x20);
+// 	/* VDV = 0V */
+// 	write_reg(par, VDVS, 0x20);
 
-	/* VCOM = 0.9V */
-	if (HSD20_IPS)
-		write_reg(par, VCOMS, 0x22);
-	else
-		write_reg(par, VCOMS, 0x20);
+// 	/* VCOM = 0.9V */
+// 	if (HSD20_IPS)
+// 		write_reg(par, VCOMS, 0x22);
+// 	else
+// 		write_reg(par, VCOMS, 0x20);
 
-	/* VCOM offset = 0V */
-	write_reg(par, VCMOFSET, 0x20);
+// 	/* VCOM offset = 0V */
+// 	write_reg(par, VCMOFSET, 0x20);
 
-	/*
-	 * AVDD = 6.8V
-	 * AVCL = -4.8V
-	 * VDS = 2.3V
-	 */
-	write_reg(par, PWCTRL1, 0xA4, 0xA1);
+// 	/*
+// 	 * AVDD = 6.8V
+// 	 * AVCL = -4.8V
+// 	 * VDS = 2.3V
+// 	 */
+// 	write_reg(par, PWCTRL1, 0xA4, 0xA1);
+
+// 	write_reg(par, MIPI_DCS_SET_DISPLAY_ON);
+
+// 	if (HSD20_IPS)
+// 		write_reg(par, MIPI_DCS_ENTER_INVERT_MODE);
+
+// 	return 0;
+// }
+
+// st7735
+//youkai add
+static int init_display(struct fbtft_par *par)
+{
+	par->fbtftops.reset(par);//硬复位
+
+	mdelay(50);
+	write_reg(par,0x11);//软复位
+	mdelay(100);
+	//下面添加初始化函数write_reg 参数分别为:结构体指针,写命令,写数据....(后都为数据)
+	//ST7735s Frame Rate
+	write_reg(par,0xB1,0x05,0x3c,0x3c); 
+
+	write_reg(par,0xB2,0x05,0x3c,0x3c);  
+
+	write_reg(par,0xB3,0x05,0x3c,0x3c,0x05,0x3c,0x3c); 
+ 
+	
+	write_reg(par,0xB4,0x03); //Column inversion 
+	
+	//ST7735s Power Sequence
+	write_reg(par,0xC0,0x28,0x08,0x04); 
+
+	write_reg(par,0xC1,0xc0); 
 
-	write_reg(par, MIPI_DCS_SET_DISPLAY_ON);
+	write_reg(par,0xC2,0x0d,0x00); 
+	
+	write_reg(par,0xC3,0x8d,0x2a); //VCOM 
+	
+	write_reg(par,0xc4,0x8d,0xee); //MX, MY, RGB mode  
+	write_reg(par,0xc5,0x1a);
+	write_reg(par,0x36,0xc0);
+	//ST7735s Gamma Sequence
+	write_reg(par,0xe0,0x04,0x22,0x07,0x0a,0x2e,0x30,0x25,0x2a,0x28,0x26,0x2e,0x3a,0x00,0x01,0x03,0x13); 
 
-	if (HSD20_IPS)
-		write_reg(par, MIPI_DCS_ENTER_INVERT_MODE);
+	write_reg(par,0xe1,0x04,0x16,0x06,0x0d,0x2d,0x26,0x23,0x27,0x27,0x25,0x2d,0x3b,0x00,0x01,0x04,0x13);  
+	
+	write_reg(par,0x3A,0x05); //65k mode  
+	
+	write_reg(par,0x29);//Display on
+	mdelay(100);
 
 	return 0;
 }
@@ -221,8 +266,8 @@ static int set_gamma(struct fbtft_par *par, u32 *curves)
 
 	for (i = 0; i < par->gamma.num_curves; i++) {
 		c = i * par->gamma.num_values;
-		for (j = 0; j < par->gamma.num_values; j++)
-			curves[c + j] &= gamma_par_mask[j];
+		// for (j = 0; j < par->gamma.num_values; j++)
+		// 	curves[c + j] &= gamma_par_mask[j];
 		write_reg(par, PVGAMCTRL + i,
 			  curves[c + 0],  curves[c + 1],  curves[c + 2],
 			  curves[c + 3],  curves[c + 4],  curves[c + 5],
@@ -252,8 +297,8 @@ static int blank(struct fbtft_par *par, bool on)
 
 static struct fbtft_display display = {
 	.regwidth = 8,
-	.width = 240,
-	.height = 320,
+	.width = 128,//240,
+	.height = 160,//320,
 	.gamma_num = 2,
 	.gamma_len = 14,
 	.gamma = HSD20_IPS_GAMMA,
diff --git a/linux_5.10/drivers/staging/fbtft/fbtft-core.c b/linux_5.10/drivers/staging/fbtft/fbtft-core.c
index 4f362dad4..97b139d46 100644
--- a/linux_5.10/drivers/staging/fbtft/fbtft-core.c
+++ b/linux_5.10/drivers/staging/fbtft/fbtft-core.c
@@ -30,6 +30,9 @@
 #include "fbtft.h"
 #include "internal.h"
 
+#include <linux/gpio.h> //add
+#include <linux/of_gpio.h> //add
+
 static unsigned long debug;
 module_param(debug, ulong, 0000);
 MODULE_PARM_DESC(debug, "override device debug level");
@@ -71,68 +74,151 @@ void fbtft_dbg_hex(const struct device *dev, int groupsize,
 }
 EXPORT_SYMBOL(fbtft_dbg_hex);
 
+// default
+// 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;
+// 	int ret = 0;
+
+// 	*gpiop = devm_gpiod_get_index_optional(dev, name, index,
+// 					       GPIOD_OUT_HIGH);
+// 	if (IS_ERR(*gpiop)) {
+// 		ret = PTR_ERR(*gpiop);
+// 		dev_err(dev,
+// 			"Failed to request %s GPIO: %d\n", name, ret);
+// 		return ret;
+// 	}
+// 	fbtft_par_dbg(DEBUG_REQUEST_GPIOS, par, "%s: '%s' GPIO\n",
+// 		      __func__, name);
+
+// 	return ret;
+// }
+
+// static int fbtft_request_gpios(struct fbtft_par *par)
+// {
+// 	int i;
+// 	int ret;
+
+// 	ret = fbtft_request_one_gpio(par, "reset", 0, &par->gpio.reset);
+// 	if (ret)
+// 		return ret;
+// 	ret = fbtft_request_one_gpio(par, "dc", 0, &par->gpio.dc);
+// 	if (ret)
+// 		return ret;
+// 	ret = fbtft_request_one_gpio(par, "rd", 0, &par->gpio.rd);
+// 	if (ret)
+// 		return ret;
+// 	ret = fbtft_request_one_gpio(par, "wr", 0, &par->gpio.wr);
+// 	if (ret)
+// 		return ret;
+// 	ret = fbtft_request_one_gpio(par, "cs", 0, &par->gpio.cs);
+// 	if (ret)
+// 		return ret;
+// 	ret = fbtft_request_one_gpio(par, "latch", 0, &par->gpio.latch);
+// 	if (ret)
+// 		return ret;
+// 	for (i = 0; i < 16; i++) {
+// 		ret = fbtft_request_one_gpio(par, "db", i,
+// 					     &par->gpio.db[i]);
+// 		if (ret)
+// 			return ret;
+// 		ret = fbtft_request_one_gpio(par, "led", i,
+// 					     &par->gpio.led[i]);
+// 		if (ret)
+// 			return ret;
+// 		ret = fbtft_request_one_gpio(par, "aux", i,
+// 					     &par->gpio.aux[i]);
+// 		if (ret)
+// 			return ret;
+// 	}
+
+// 	return 0;
+// }
+
+//https://blog.csdn.net/qq_46604211/article/details/116449891
 static int fbtft_request_one_gpio(struct fbtft_par *par,
-				  const char *name, int index,
-				  struct gpio_desc **gpiop)
+                  const char *name, int index,
+                  struct gpio_desc **gpiop)
 {
-	struct device *dev = par->info->device;
-	int ret = 0;
-
-	*gpiop = devm_gpiod_get_index_optional(dev, name, index,
-					       GPIOD_OUT_HIGH);
-	if (IS_ERR(*gpiop)) {
-		ret = PTR_ERR(*gpiop);
-		dev_err(dev,
-			"Failed to request %s GPIO: %d\n", name, ret);
-		return ret;
-	}
-	fbtft_par_dbg(DEBUG_REQUEST_GPIOS, par, "%s: '%s' GPIO\n",
-		      __func__, name);
-
-	return ret;
+    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 DT\n", 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 %d\n",
+                name, gpio, ret);
+            return ret;
+        }
+
+        *gpiop = gpio_to_desc(gpio);
+        fbtft_par_dbg(DEBUG_REQUEST_GPIOS, par, "%s: '%s' = GPIO%d\n",
+                            __func__, name, gpio);
+    }
+
+    return ret;
 }
 
 static int fbtft_request_gpios(struct fbtft_par *par)
 {
-	int i;
-	int ret;
-
-	ret = fbtft_request_one_gpio(par, "reset", 0, &par->gpio.reset);
-	if (ret)
-		return ret;
-	ret = fbtft_request_one_gpio(par, "dc", 0, &par->gpio.dc);
-	if (ret)
-		return ret;
-	ret = fbtft_request_one_gpio(par, "rd", 0, &par->gpio.rd);
-	if (ret)
-		return ret;
-	ret = fbtft_request_one_gpio(par, "wr", 0, &par->gpio.wr);
-	if (ret)
-		return ret;
-	ret = fbtft_request_one_gpio(par, "cs", 0, &par->gpio.cs);
-	if (ret)
-		return ret;
-	ret = fbtft_request_one_gpio(par, "latch", 0, &par->gpio.latch);
-	if (ret)
-		return ret;
-	for (i = 0; i < 16; i++) {
-		ret = fbtft_request_one_gpio(par, "db", i,
-					     &par->gpio.db[i]);
-		if (ret)
-			return ret;
-		ret = fbtft_request_one_gpio(par, "led", i,
-					     &par->gpio.led[i]);
-		if (ret)
-			return ret;
-		ret = fbtft_request_one_gpio(par, "aux", i,
-					     &par->gpio.aux[i]);
-		if (ret)
-			return ret;
-	}
-
-	return 0;
+    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;
 }
 
+
 #ifdef CONFIG_FB_BACKLIGHT
 static int fbtft_backlight_update_status(struct backlight_device *bd)
 {
@@ -222,17 +308,32 @@ static void fbtft_set_addr_win(struct fbtft_par *par, int xs, int ys, int xe,
 	write_reg(par, MIPI_DCS_WRITE_MEMORY_START);
 }
 
+//作者:Leesans https://www.bilibili.com/read/cv9947785/ 出处:bilibili
 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);
+    if (!par->gpio.reset)
+        return;
+    fbtft_par_dbg(DEBUG_RESET, par, "%s()\n", __func__);
+    gpiod_set_value_cansleep(par->gpio.reset, 1);
+    msleep(10);
+    gpiod_set_value_cansleep(par->gpio.reset, 0);
+    msleep(200);
+    gpiod_set_value_cansleep(par->gpio.reset, 1);
+    msleep(10);
 }
 
+// defalut
+// 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);
+// }
+
 static void fbtft_update_display(struct fbtft_par *par, unsigned int start_line,
 				 unsigned int end_line)
 {
diff --git a/u-boot-2021.10/include/configs/cv180x-asic.h b/u-boot-2021.10/include/configs/cv180x-asic.h
index 08b70193f..5d3404817 100644
--- a/u-boot-2021.10/include/configs/cv180x-asic.h
+++ b/u-boot-2021.10/include/configs/cv180x-asic.h
@@ -280,7 +280,7 @@
 	#endif
 
 	#define SET_BOOTARGS "setenv bootargs ${root} ${mtdparts} " \
-					"console=$consoledev,$baudrate $othbootargs;"
+					"console=tty0 console=$consoledev,$baudrate $othbootargs;"
 
 	#define SD_BOOTM_COMMAND \
 				SET_BOOTARGS \

感谢大佬的教程, 成功了 。。。。。。。