1.8 1.8 寸彩色触摸屏
1.8.1 模块来源
1.8 寸液晶屏 1.8 寸 TFT LCD SPI 串口模块 TFT 彩屏触摸屏 st7735
资料下载链接:
https://pan.baidu.com/s/1n_vp38V7ij88PUGpbJPd7Q
资料提取码:8888
1.8.2 规格参数
工作电压: 3.3V
工作电流: 30MA
模块尺寸: 35(H) x 56(V) MM
像素大小: 128(H) x 160(V)RGB
驱动芯片: ST7735S
通信协议: SPI
管脚数量: 12 Pin(2.54mm 间距排针)
带电阻触摸芯片: XPT2046
1.8.3 移植过程
我们的目标是将例程移植至梁山派 GD32F470 上。按照以下步骤,即可完成移植。
- 将源码导入工程;
- 根据编译报错处进行粗改;
- 修改引脚配置;
- 修改时序配置;
- 移植验证。
1.8.3.1 查看资料
打开厂家资料例程。具体路径见 图 1.8.3.1 例程路径
在 main.c 文件发现对屏幕的初始化配置函数为 LCD_Init(); ,该函数里包含了对屏幕引脚的配置、对屏幕显示方式的配置和对触摸功能引脚的配置。
我们主要修改的是引脚的初始化与时序的延时修改。
1.8.3.2 移植至工程
将厂家资料路径下的【LCD】和文件夹,复制到自己的工程中。自己的工程至少需要有毫秒级延时函数。
打开自己的工程,将我们刚刚复制过来的文件导入.c 和.h 文件。
选择导入复制过来的文件夹里面的全部.c 文件。
导入完成后,工程目录下会出现新导入的文件。(原工程目录见图 1.8.3.5 C 文件导入步骤)
添加.h 路径
在弹出的路径窗口,选择添加新路径
选择 LCD 和 TOUCH 文件夹的路径,我们的.h 就在里面。因为是选择导入路径,故不会显示文件夹下有什么文件。
添加完成
尝试编译,发现有错误。错误内容为找不到 sys.h 这个文件。
因为后续触摸功能需要位带操作,故这里提供 sys.c 与 sys.h 文件。将文件导入自己的工程,包括.c 和.h。
再编译发现还有错误,错误内容分别为标识符“u8”、“u16”、“u32”未定义和找不到 delay.h 这个文件。
分别在**lcd_init.h、lcd.h 和 touch.h **文件中定义三个宏,u32、u16 与 u8。
#ifndef u8
#define u8 uint8_t
#endif
#ifndef u16
#define u16 uint16_t
#endif
#ifndef u32
#define u32 uint32_t
#endif
2
3
4
5
6
7
8
9
10
11
在源代码中使用 delay.h 的主要作用是给屏幕初始化时提供一定的延时,确保初始化成功。主要使用的函数是 delay_ms,进行毫秒级延时。
将 lcd_init.c、lcd.c和touch.c文件中的头文件delay.h改为自己的延时函数头文件,这里我使用的是 systick.h;再将文件中使用到的延时函数 delay_ms 替换为自己的毫秒级延时,这里替换为 delay_1ms。
#include "systick.h"
#define delay_ms delay_1ms
2
再编译发现只剩下 LCD 引脚初始化的内容报错,接下来我们要进行引脚选择。
1.8.3.3 引脚选择
该屏幕需要设置 12 个接口,具体接口说明见 表 1.8.3.1 各引脚说明。
下面分为软件 SPI 移植与硬件 SPI 移植进行讲解。
1.8.3.3.1 软件 SPI 移植
当前厂家源码使用的是软件 SPI 接口,SPI 时序部分厂家已经完成,我们只需要将引脚和延时配置好即可。所以对应接入的屏幕引脚请按照你的需要。这里选择的引脚见表 1.8.3.2 软件 SPI 接线
选择好引脚后,进入工程开始编写屏幕引脚初始化代码。
为了方便后续移植,我在 lcd_init.h 处宏定义了每一个引脚,后续根据需要进行修改即可。
//-----------------LCD端口移植----------------
//GND - GND
//VCC - 3.3V
//CLK - PB13 SPI1_SCK
//MOS - PB15 SPI1_MOSI
//RES - PD0(可以接入复位)
//DC - PC6
//BLK - PC7- TIMER7_CH0
//MIS - PB14 SPI1_MISO
//CS1 - PB12
//CS2 - PG6
//PEN - PG7
//NC - 不用接
#define RCU_LCD_CLK RCU_GPIOB//SCK
#define PORT_LCD_CLK GPIOB
#define GPIO_LCD_CLK GPIO_PIN_13
#define RCU_LCD_MOS RCU_GPIOB//MOSI
#define PORT_LCD_MOS GPIOB
#define GPIO_LCD_MOS GPIO_PIN_15
#define RCU_LCD_MIS RCU_GPIOB//MIS0
#define PORT_LCD_MIS GPIOB
#define GPIO_LCD_MIS GPIO_PIN_14
#define RCU_LCD_CS1 RCU_GPIOB//NSS
#define PORT_LCD_CS1 GPIOB
#define GPIO_LCD_CS1 GPIO_PIN_12
#define RCU_LCD_CS2 RCU_GPIOG//CS2
#define PORT_LCD_CS2 GPIOG
#define GPIO_LCD_CS2 GPIO_PIN_6
#define RCU_LCD_PEN RCU_GPIOG //PEN
#define PORT_LCD_PEN GPIOG
#define GPIO_LCD_PEN GPIO_PIN_7
#define RCU_LCD_DC RCU_GPIOC //DC
#define PORT_LCD_DC GPIOC
#define GPIO_LCD_DC GPIO_PIN_6
#define RCU_LCD_RES RCU_GPIOD//RES
#define PORT_LCD_RES GPIOD
#define GPIO_LCD_RES GPIO_PIN_0
#define RCU_LCD_BLK RCU_GPIOC//BLK
#define PORT_LCD_BLK GPIOC
#define GPIO_LCD_BLK GPIO_PIN_7
/******** 硬件SPI修改此次 ********/
#define RCU_SPI_HARDWARE RCU_SPI1
#define PORT_SPI SPI1
#define LINE_AF_SPI GPIO_AF_5
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
将 lcd_init.c 源代码中的**void LCD_GPIO_Init(void)**修改为如下代码。
void LCD_GPIO_Init(void)
{
/* 使能时钟 */
rcu_periph_clock_enable(RCU_LCD_CLK);
rcu_periph_clock_enable(RCU_LCD_MOS);
rcu_periph_clock_enable(RCU_LCD_MIS);
rcu_periph_clock_enable(RCU_LCD_CS1);
rcu_periph_clock_enable(RCU_LCD_CS2);
rcu_periph_clock_enable(RCU_LCD_PEN);
rcu_periph_clock_enable(RCU_LCD_DC);
rcu_periph_clock_enable(RCU_LCD_RES);
rcu_periph_clock_enable(RCU_LCD_BLK);
/* 配置CLK */
gpio_mode_set(PORT_LCD_CLK,GPIO_MODE_OUTPUT,GPIO_PUPD_PULLUP,GPIO_LCD_CLK);
gpio_output_options_set(PORT_LCD_CLK,GPIO_OTYPE_PP,GPIO_OSPEED_50MHZ,GPIO_LCD_CLK);
gpio_bit_write(PORT_LCD_CLK, GPIO_LCD_CLK, SET);
/* 配置MOS */
gpio_mode_set(PORT_LCD_MOS,GPIO_MODE_OUTPUT,GPIO_PUPD_PULLUP,GPIO_LCD_MOS);
gpio_output_options_set(PORT_LCD_MOS,GPIO_OTYPE_PP,GPIO_OSPEED_50MHZ,GPIO_LCD_MOS);
gpio_bit_write(PORT_LCD_MOS, GPIO_LCD_MOS, SET);
/* 配置MIS */
gpio_mode_set(PORT_LCD_MIS,GPIO_MODE_INPUT,GPIO_PUPD_PULLUP,GPIO_LCD_MIS);
gpio_bit_write(PORT_LCD_MIS, GPIO_LCD_MIS, SET);
/* 配置DC */
gpio_mode_set(PORT_LCD_DC,GPIO_MODE_OUTPUT,GPIO_PUPD_PULLUP,GPIO_LCD_DC);
gpio_output_options_set(PORT_LCD_DC,GPIO_OTYPE_PP,GPIO_OSPEED_50MHZ,GPIO_LCD_DC);
gpio_bit_write(PORT_LCD_DC, GPIO_LCD_DC, SET);
/* 配置CS1 */
gpio_mode_set(PORT_LCD_CS1,GPIO_MODE_OUTPUT,GPIO_PUPD_PULLUP,GPIO_LCD_CS1);
gpio_output_options_set(PORT_LCD_CS1,GPIO_OTYPE_PP,GPIO_OSPEED_50MHZ,GPIO_LCD_CS1);
gpio_bit_write(PORT_LCD_CS1, GPIO_LCD_CS1, SET);
/* 配置CS2 */
gpio_mode_set(PORT_LCD_CS2,GPIO_MODE_OUTPUT,GPIO_PUPD_PULLUP,GPIO_LCD_CS2);
gpio_output_options_set(PORT_LCD_CS2,GPIO_OTYPE_PP,GPIO_OSPEED_50MHZ,GPIO_LCD_CS2);
gpio_bit_write(PORT_LCD_CS2, GPIO_LCD_CS2, SET);
/* 配置RES */
gpio_mode_set(PORT_LCD_RES,GPIO_MODE_OUTPUT,GPIO_PUPD_PULLUP,GPIO_LCD_RES);
gpio_output_options_set(PORT_LCD_RES,GPIO_OTYPE_PP,GPIO_OSPEED_50MHZ,GPIO_LCD_RES);
gpio_bit_write(PORT_LCD_RES, GPIO_LCD_RES, SET);
/* 配置BLK */
gpio_mode_set(PORT_LCD_BLK, GPIO_MODE_OUTPUT, GPIO_PUPD_PULLUP, GPIO_LCD_BLK);
gpio_output_options_set(PORT_LCD_BLK, GPIO_OTYPE_PP, GPIO_OSPEED_50MHZ, GPIO_LCD_BLK);
gpio_bit_write(PORT_LCD_BLK, GPIO_LCD_BLK, SET);
/* 配置PEN */
gpio_mode_set(PORT_LCD_PEN,GPIO_MODE_INPUT,GPIO_PUPD_PULLUP,GPIO_LCD_PEN);
gpio_bit_write(PORT_LCD_PEN, GPIO_LCD_PEN, SET);
}
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
将 lcd_init.h 中的LCD 端口定义 宏,修改为图 1.8.3.25 样式。
//-----------------LCD端口定义----------------
#define LCD_SCLK_Clr() gpio_bit_write(PORT_LCD_CLK, GPIO_LCD_CLK, RESET)//SCL=SCLK
#define LCD_SCLK_Set() gpio_bit_write(PORT_LCD_CLK, GPIO_LCD_CLK, SET)
#define LCD_MOSI_Clr() gpio_bit_write(PORT_LCD_MOS, GPIO_LCD_MOS, RESET)//SDA=MOSI
#define LCD_MOSI_Set() gpio_bit_write(PORT_LCD_MOS, GPIO_LCD_MOS, SET)
#define LCD_RES_Clr() gpio_bit_write(PORT_LCD_RES, GPIO_LCD_RES, RESET)//RES
#define LCD_RES_Set() gpio_bit_write(PORT_LCD_RES, GPIO_LCD_RES, SET)
#define LCD_DC_Clr() gpio_bit_write(PORT_LCD_DC, GPIO_LCD_DC, RESET)//DC
#define LCD_DC_Set() gpio_bit_write(PORT_LCD_DC, GPIO_LCD_DC, SET)
#define LCD_CS_Clr() gpio_bit_write(PORT_LCD_CS1, GPIO_LCD_CS1, RESET)//CS
#define LCD_CS_Set() gpio_bit_write(PORT_LCD_CS1, GPIO_LCD_CS1, SET)
#define LCD_BLK_Clr() gpio_bit_write(PORT_LCD_BLK, GPIO_LCD_BLK, RESET)//BLK
#define LCD_BLK_Set() gpio_bit_write(PORT_LCD_BLK, GPIO_LCD_BLK, SET)
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
将 lcd_init.h 处的触摸功能引脚位带操作宏进行修改(需要导入 sys.h,否则报错)。
修改为如图 1.8.3.27 修改后触摸功能引脚所示
#define TCLK PBout(13) //PB13 SCLK--
#define TDIN PBout(15) //PB15 MOSI--
#define DOUT PBin(14) //PB14 MISO--
#define TCS PGout(6) //PG6 CS2--
#define PEN PGin(7) //PG7 INT--
2
3
4
5
到这里软件 SPI 就移植完成了,请移步到 1.8.4 节进行移植验证。
1.8.3.3.2 硬件 SPI 移植
硬件 SPI 与软件 SPI 相比,硬件 SPI 是靠硬件上面的 SPI 控制器,所有的时钟边缘采样,时钟发生,还有时序控制,都是由硬件完成的。它降低了 CPU 的使用率,提高了运行速度。软件 SPI 就是用代码控制 IO 输出高低电平,模拟 SPI 的时序,这种方法通信速度较慢,且不可靠。
梁山派的主控为 GD32F470ZGT6,其带有 6 个 SPI 外设,可以通过简单的库函数调用,实现硬件 SPI 的通信。想要使用硬件 SPI 驱动屏幕,需要确定使用的引脚是否有 SPI 外设功能。可以通过数据手册【GD32F450xx_Datasheet_Rev2.2.pdf】进行查看。
在数据手册的第 28 页结尾,是关于 GD32F450Zx 系列芯片引脚的功能定义示意图。
当前使用的是硬件 SPI 接口,而屏幕我们只需要控制它,而不需要读取屏幕的数据,故使用的是 3 线的 SPI,只使用到了时钟线 SCK、主机输出从机输入线 MOSI 和软件控制的片选线 NSS。所以除了 SCL/SDA 引脚需要使用硬件 SPI 功能的引脚外,其他引脚都可以使用开发板上其他的 GPIO。这里选择使用 PB13/PB15 的 SPI 复用功能。其他对应接入的屏幕引脚请按照你的需要。这里选择的引脚见表 1.8.3.3 硬件 SPI 接线
选择好引脚后,进入工程开始编写屏幕引脚初始化代码。为了方便后续移植,我在 lcd_init.h 处宏定义了每一个引脚,后续根据需要进行修改即可。
//-----------------LCD端口移植----------------
//GND - GND
//VCC - 3.3V
//CLK - PB13 SPI1_SCK
//MOS - PB15 SPI1_MOSI
//RES - PD0(可以接入复位)
//DC - PC6
//BLK - PC7- TIMER7_CH0
//MIS - PB14 SPI1_MISO
//CS1 - PB12
//CS2 - PG6
//PEN - PG7
//NC - 不用接
#define RCU_LCD_CLK RCU_GPIOB//SCK
#define PORT_LCD_CLK GPIOB
#define GPIO_LCD_CLK GPIO_PIN_13
#define RCU_LCD_MOS RCU_GPIOB//MOSI
#define PORT_LCD_MOS GPIOB
#define GPIO_LCD_MOS GPIO_PIN_15
#define RCU_LCD_MIS RCU_GPIOB//MIS0
#define PORT_LCD_MIS GPIOB
#define GPIO_LCD_MIS GPIO_PIN_14
#define RCU_LCD_CS1 RCU_GPIOB//NSS
#define PORT_LCD_CS1 GPIOB
#define GPIO_LCD_CS1 GPIO_PIN_12
#define RCU_LCD_CS2 RCU_GPIOG//CS2
#define PORT_LCD_CS2 GPIOG
#define GPIO_LCD_CS2 GPIO_PIN_6
#define RCU_LCD_PEN RCU_GPIOG //PEN
#define PORT_LCD_PEN GPIOG
#define GPIO_LCD_PEN GPIO_PIN_7
#define RCU_LCD_DC RCU_GPIOC //DC
#define PORT_LCD_DC GPIOC
#define GPIO_LCD_DC GPIO_PIN_6
#define RCU_LCD_RES RCU_GPIOD//RES
#define PORT_LCD_RES GPIOD
#define GPIO_LCD_RES GPIO_PIN_0
#define RCU_LCD_BLK RCU_GPIOC//BLK
#define PORT_LCD_BLK GPIOC
#define GPIO_LCD_BLK GPIO_PIN_7
/******** 硬件SPI修改此次 ********/
#define RCU_SPI_HARDWARE RCU_SPI1
#define PORT_SPI SPI1
#define LINE_AF_SPI GPIO_AF_5
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
引脚初始化配置见如下代码。
void LCD_GPIO_Init(void)
{
spi_parameter_struct spi_init_struct;
rcu_periph_clock_enable(RCU_LCD_CLK);
rcu_periph_clock_enable(RCU_LCD_MOS);
rcu_periph_clock_enable(RCU_LCD_MIS);
rcu_periph_clock_enable(RCU_LCD_CS1);
rcu_periph_clock_enable(RCU_LCD_CS2);
rcu_periph_clock_enable(RCU_LCD_PEN);
rcu_periph_clock_enable(RCU_LCD_DC);
rcu_periph_clock_enable(RCU_LCD_RES);
rcu_periph_clock_enable(RCU_LCD_BLK);
rcu_periph_clock_enable(RCU_SPI_HARDWARE); // 使能SPI
/* 配置 SPI的CLK GPIO */
gpio_af_set(PORT_LCD_CLK, LINE_AF_SPI, GPIO_LCD_CLK);
gpio_mode_set(PORT_LCD_CLK, GPIO_MODE_AF, GPIO_PUPD_NONE, GPIO_LCD_CLK);
gpio_output_options_set(PORT_LCD_CLK, GPIO_OTYPE_PP, GPIO_OSPEED_50MHZ, GPIO_LCD_CLK);
gpio_bit_set(PORT_LCD_CLK,GPIO_LCD_CLK);
/* 配置 SPI的MOSI GPIO */
gpio_af_set(PORT_LCD_MOS, LINE_AF_SPI, GPIO_LCD_MOS);
gpio_mode_set(PORT_LCD_MOS, GPIO_MODE_AF, GPIO_PUPD_NONE, GPIO_LCD_MOS);
gpio_output_options_set(PORT_LCD_MOS, GPIO_OTYPE_PP, GPIO_OSPEED_50MHZ, GPIO_LCD_MOS);
gpio_bit_set(PORT_LCD_MOS, GPIO_LCD_MOS);
/* 配置 SPI的MIS0 GPIO */
gpio_af_set(PORT_LCD_MIS, LINE_AF_SPI, GPIO_LCD_MIS);
gpio_mode_set(PORT_LCD_MIS, GPIO_MODE_AF, GPIO_PUPD_NONE, GPIO_LCD_MIS);
gpio_output_options_set(PORT_LCD_MIS, GPIO_OTYPE_PP, GPIO_OSPEED_50MHZ, GPIO_LCD_MIS);
gpio_bit_set(PORT_LCD_MIS, GPIO_LCD_MIS);
/* 配置DC */
gpio_mode_set(PORT_LCD_DC,GPIO_MODE_OUTPUT,GPIO_PUPD_PULLUP,GPIO_LCD_DC);
gpio_output_options_set(PORT_LCD_DC,GPIO_OTYPE_PP,GPIO_OSPEED_50MHZ,GPIO_LCD_DC);
gpio_bit_write(PORT_LCD_DC, GPIO_LCD_DC, SET);
/* 配置RES */
gpio_mode_set(PORT_LCD_RES,GPIO_MODE_OUTPUT,GPIO_PUPD_PULLUP,GPIO_LCD_RES);
gpio_output_options_set(PORT_LCD_RES,GPIO_OTYPE_PP,GPIO_OSPEED_50MHZ,GPIO_LCD_RES);
gpio_bit_write(PORT_LCD_RES, GPIO_LCD_RES, SET);
/* 配置BLK */
gpio_mode_set(PORT_LCD_BLK, GPIO_MODE_OUTPUT, GPIO_PUPD_PULLUP, GPIO_LCD_BLK);
gpio_output_options_set(PORT_LCD_BLK, GPIO_OTYPE_PP, GPIO_OSPEED_50MHZ, GPIO_LCD_BLK);
gpio_bit_write(PORT_LCD_BLK, GPIO_LCD_BLK, SET);
/* 配置CS1 */
gpio_mode_set(PORT_LCD_CS1,GPIO_MODE_OUTPUT,GPIO_PUPD_NONE,GPIO_LCD_CS1);
gpio_output_options_set(PORT_LCD_CS1,GPIO_OTYPE_PP,GPIO_OSPEED_50MHZ,GPIO_LCD_CS1);
gpio_bit_write(PORT_LCD_CS1, GPIO_LCD_CS1, SET);
/* 配置CS2 */
gpio_mode_set(PORT_LCD_CS2,GPIO_MODE_OUTPUT,GPIO_PUPD_NONE,GPIO_LCD_CS2);
gpio_output_options_set(PORT_LCD_CS2,GPIO_OTYPE_PP,GPIO_OSPEED_50MHZ,GPIO_LCD_CS2);
gpio_bit_write(PORT_LCD_CS2, GPIO_LCD_CS2, SET);
/* 配置PEN */
gpio_mode_set(PORT_LCD_PEN,GPIO_MODE_INPUT,GPIO_PUPD_PULLUP,GPIO_LCD_PEN);
gpio_bit_write(PORT_LCD_PEN, GPIO_LCD_PEN, SET);
/* 配置 SPI 参数 */
spi_init_struct.trans_mode = SPI_TRANSMODE_FULLDUPLEX; // 传输模式全双工
spi_init_struct.device_mode = SPI_MASTER; // 配置为主机
spi_init_struct.frame_size = SPI_FRAMESIZE_8BIT; // 8位数据
spi_init_struct.clock_polarity_phase = SPI_CK_PL_HIGH_PH_2EDGE;
spi_init_struct.nss = SPI_NSS_SOFT; // 软件cs
spi_init_struct.prescale = SPI_PSC_32;//2分频
spi_init_struct.endian = SPI_ENDIAN_MSB;
spi_init(PORT_SPI, &spi_init_struct);
/* 使能 SPI */
spi_enable(PORT_SPI);
}
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
将 lcd_init.h 中的 LCD 端口定义 宏,修改为图 1.8.3.33 样式。
//-----------------LCD端口定义----------------
#define LCD_SCLK_Clr() gpio_bit_write(PORT_LCD_CLK, GPIO_LCD_CLK, RESET)//SCL=SCLK
#define LCD_SCLK_Set() gpio_bit_write(PORT_LCD_CLK, GPIO_LCD_CLK, SET)
#define LCD_MOSI_Clr() gpio_bit_write(PORT_LCD_MOS, GPIO_LCD_MOS, RESET)//SDA=MOSI
#define LCD_MOSI_Set() gpio_bit_write(PORT_LCD_MOS, GPIO_LCD_MOS, SET)
#define LCD_RES_Clr() gpio_bit_write(PORT_LCD_RES, GPIO_LCD_RES, RESET)//RES
#define LCD_RES_Set() gpio_bit_write(PORT_LCD_RES, GPIO_LCD_RES, SET)
#define LCD_DC_Clr() gpio_bit_write(PORT_LCD_DC, GPIO_LCD_DC, RESET)//DC
#define LCD_DC_Set() gpio_bit_write(PORT_LCD_DC, GPIO_LCD_DC, SET)
#define LCD_CS_Clr() gpio_bit_write(PORT_LCD_CS1, GPIO_LCD_CS1, RESET)//CS
#define LCD_CS_Set() gpio_bit_write(PORT_LCD_CS1, GPIO_LCD_CS1, SET)
#define LCD_BLK_Clr() gpio_bit_write(PORT_LCD_BLK, GPIO_LCD_BLK, RESET)//BLK
#define LCD_BLK_Set() gpio_bit_write(PORT_LCD_BLK, GPIO_LCD_BLK, SET)
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
将 lcd_init.h 处的触摸功能引脚位带操作宏进行修改(需要导入 sys.h,否则报错)。
修改为如图 1.8.3.35 修改后触摸功能引脚所示
#define TCLK PBout(13) //PB13 SCLK--
#define TDIN PBout(15) //PB15 MOSI--
#define DOUT PBin(14) //PB14 MISO--
#define TCS PGout(6) //PG6 CS2--
#define PEN PGin(7) //PG7 INT--
2
3
4
5
初始化部分完完成,还需要修改发送数据部分。源代码中使用的是软件 SPI,时序是由厂家编写完成的。我们使用硬件 SPI 则需要对其进行修改。
在 lcd_init.c 文件中,将源代码的 void LCD_Writ_Bus(u8 dat)函数修改为图 1.8.3.37样式。
/******************************************************************************
函数说明:LCD串行数据写入函数
入口数据:dat 要写入的串行数据
返回值: 无
******************************************************************************/
void LCD_Writ_Bus(u8 dat)
{
LCD_CS_Clr();
while(RESET == spi_i2s_flag_get(PORT_SPI, SPI_FLAG_TBE));
spi_i2s_data_transmit(PORT_SPI, dat);
while(RESET == spi_i2s_flag_get(PORT_SPI, SPI_FLAG_RBNE));
spi_i2s_data_receive(PORT_SPI);
LCD_CS_Set();
}
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
将 touch.c 文件中的 void TP_Write_Byte(u8 num) 函数修改为图 1.8.3.39 样式。
u8 TP_Write_Byte(u8 num)
{
u8 count=0;
while(RESET == spi_i2s_flag_get(PORT_SPI, SPI_FLAG_TBE));
spi_i2s_data_transmit(PORT_SPI, num);
while(RESET == spi_i2s_flag_get(PORT_SPI, SPI_FLAG_RBNE));
count = spi_i2s_data_receive(PORT_SPI);
return count;
}
2
3
4
5
6
7
8
9
10
这里因为 TP_Write_Byte(u8 num)函数增加了返回类型为 u8 的返回值,故touch.h处关于**TP_Write_Byte(u8 num)**的定义也要改为返回 u8 类型的返回值。
再将 touch.c 文件中的u16 TP_Read_AD(u8 CMD) 函数修改为图 1.8.3.41 样式。
u16 TP_Read_AD(u8 CMD)
{
u8 count = 0;
u16 Num=0;
TCS=0; //选中触摸屏IC
TP_Write_Byte(CMD);//发送命令字
Num=TP_Write_Byte(0xff)<<8;
Num |= TP_Write_Byte(0xff);
Num=Num>>4;
TCS=1; //释放片选
return(Num);
}
2
3
4
5
6
7
8
9
10
11
12
到这里硬件 SPI 就移植完成了,请移步到 5.4 节进行移植验证。
1.8.4 移植验证
在 main.c 中输入代码如下
#include "gd32f4xx.h"
#include "systick.h"
#include "lcd.h"
#include "lcd_init.h"
#include "touch.h"
#include "pic.h"
int main(void)
{
float t=0;
u16 lastpos[2];//最后一次的数据
nvic_priority_group_set(NVIC_PRIGROUP_PRE2_SUB2); // 优先级分组
systick_config(); //滴答定时器初始化 1ms
LCD_Init();//LCD初始化
LCD_Fill(0,0,LCD_W,LCD_H,WHITE);
lastpos[0]=0XFFFF;
LCD_ShowString(24,30,"LCD_W:",RED,WHITE,16,0);
LCD_ShowIntNum(72,30,LCD_W,3,RED,WHITE,16);
LCD_ShowString(24,50,"LCD_H:",RED,WHITE,16,0);
LCD_ShowIntNum(72,50,LCD_H,3,RED,WHITE,16);
LCD_ShowFloatNum1(20,80,t,4,RED,WHITE,16);
t+=0.11;
LCD_ShowPicture(65,80,40,40,gImage_1);
delay_1ms(1000);
LCD_Fill(0,0,LCD_W,LCD_H,WHITE);
TP_Init();
LCD_ShowString(10,LCD_H-40,"X:",RED,WHITE,16,0);
LCD_ShowIntNum(26,LCD_H-40,0,3,RED,WHITE,16);
LCD_ShowString(10,LCD_H-20,"Y:",RED,WHITE,16,0);
LCD_ShowIntNum(26,LCD_H-20,0,3,RED,WHITE,16);
while(1)
{
tp_dev.scan(0);//扫描
if(tp_dev.sta&TP_PRES_DOWN)//有按键被按下
{
delay_1ms(1);//必要的延时,否则老认为有按键按下.
if((tp_dev.x[0]<(LCD_W-1)&&tp_dev.x[0]>=1)&&(tp_dev.y[0]<(LCD_H-1)&&tp_dev.y[0]>=1))
{
if(lastpos[0]==0XFFFF)
{
lastpos[0]=tp_dev.x[0];
lastpos[1]=tp_dev.y[0];
}
//给触摸过的地方画线
LCD_DrawRoughLine(lastpos[0],lastpos[1],tp_dev.x[0],tp_dev.y[0],BLUE);
lastpos[0]=tp_dev.x[0];
lastpos[1]=tp_dev.y[0];
//显示当前触摸位置的X轴坐标
LCD_ShowString(10,LCD_H-40,"X:",RED,WHITE,16,0);
LCD_ShowIntNum(26,LCD_H-40,tp_dev.x[0],3,RED,WHITE,16);
//显示当前触摸位置的Y轴坐标
LCD_ShowString(10,LCD_H-20,"Y:",RED,WHITE,16,0);
LCD_ShowIntNum(26,LCD_H-20,tp_dev.y[0],3,RED,WHITE,16);
}
}
}
}
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
上电效果
移植成功案例
注意事项:
如果使用硬件 SPI 时,使用厂家预设的校准值,可能会导致触摸区域变小。解决问题的方法是进行手动校准。
手动校准设置:其他不变,将 touch.h 中的宏Adujust 置 0。