• 3线SPI驱动 HX8347 TFT屏


    老五家2.8寸屏,3线GPIO模拟SPI驱动

    前言

          面对没有定义的屏幕应该如何应对?查丝印吗?屏幕的驱动芯片都小的惊人,想必是不会打上丝印的,所以即使用显微镜观察,也不能直接看到芯片的类型。没有资料的屏幕,应该只能人工从几百个引脚中判断哪个是哪个,进而推断芯片类型,想想就晕。  但大佬们都太厉害了,看看FPC就知道屏幕的接线定义,也许还有其他巧妙的办法吧。一直好奇这种神技是怎么练成的,也尝试自己来猜定义,但发现实在太难,还是拿来主义,做个等等党(等着引脚定义)比较靠谱。

    a81c5161c51b4678bbb7adaeaebc35ee.jpg

           本文用的屏是2.8寸TFT屏,驱动芯片是HX8347,定义早被大佬分析出来:

    b7696496f63f4e89b69a8e1a48370bc2.png

            定义有了。但不巧[IM3~0]=1100,只能支持3线SPI+RGB。驱动RGB屏不是单片机的强项。好像STM32F7XX和ESP32 S3是支持RGB,但也只是好像,这两个都价格不菲,手头也没有。

            还是想办法用3线SPI来驱动吧。


    一、源码

    HX8347.h

    1. #ifndef USER_HX8347_H_
    2. #define USER_HX8347_H_
    3. #define X_MAX_PIXEL 240
    4. #define Y_MAX_PIXEL 320
    5. #define RED 0xf800
    6. #define GREEN 0x07e0
    7. #define BLUE 0x001f
    8. #define WHITE 0xffff
    9. #define BLACK 0x0000
    10. #define YELLOW 0xFFE0
    11. #define GRAY0 0xEF7D //灰色0 3165 00110 001011 00101
    12. #define GRAY1 0x8410 //灰色1 00000 000000 00000
    13. #define GRAY2 0x4208 //灰色2 1111111111011111
    14. #define LCD_CS GPIO_Pin_0 // CS:PA0
    15. #define LCD_SDA GPIO_Pin_1 // SDA:PA1
    16. #define LCD_SCL GPIO_Pin_3 // SCL:PA3
    17. #define LCD_RST GPIO_Pin_4 // RST:PA4
    18. #define LCD_SCL_SET GPIO_WriteBit(GPIOA, LCD_SCL,Bit_SET)
    19. #define LCD_SDA_SET GPIO_WriteBit(GPIOA, LCD_SDA,Bit_SET)
    20. #define LCD_CS_SET GPIO_WriteBit(GPIOA, LCD_CS,Bit_SET)
    21. #define LCD_RST_SET GPIO_WriteBit(GPIOA, LCD_RST,Bit_SET)
    22. #define LCD_SCL_CLR GPIO_WriteBit(GPIOA, LCD_SCL,Bit_RESET)
    23. #define LCD_SDA_CLR GPIO_WriteBit(GPIOA, LCD_SDA,Bit_RESET)
    24. #define LCD_CS_CLR GPIO_WriteBit(GPIOA, LCD_CS,Bit_RESET)
    25. #define LCD_RST_CLR GPIO_WriteBit(GPIOA, LCD_RST,Bit_RESET)
    26. void LCD_GPIO_Init(void);
    27. void Lcd_WriteIndex(unsigned char Index);
    28. void Lcd_WriteData(unsigned char Data);
    29. void LCD_WriteData_16Bit(unsigned int Data);
    30. void Lcd_Write_REG(unsigned char Index,unsigned char Data);
    31. void LCD_Init(void);
    32. void Lcd_Clear(unsigned int Color);
    33. void FillRect(u16 x1, u16 y1, u16 x2, u16 y2, u16 color);
    34. #endif /* USER_HX8347_H_ */

    HX8347.c

    1. #include "debug.h"
    2. #include "HX8347.h"
    3. void LCD_GPIO_Init(void)
    4. {
    5. GPIO_InitTypeDef GPIO_InitStructure = {0};
    6. RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);
    7. GPIO_InitStructure.GPIO_Pin = LCD_SCL|LCD_SDA|LCD_CS|LCD_RST;
    8. GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;
    9. GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    10. GPIO_Init(GPIOA, &GPIO_InitStructure);
    11. }
    12. //向SPI总线传输一个8位数据
    13. void SPI_WriteData(unsigned char Data)
    14. {
    15. unsigned char i=0;
    16. for(i=8;i>0;i--)
    17. {
    18. if(Data&0x80)
    19. LCD_SDA_SET; //输出数据
    20. else LCD_SDA_CLR;
    21. LCD_SCL_CLR;
    22. LCD_SCL_SET;
    23. Data<<=1;
    24. }
    25. }
    26. //向液晶屏写一个8位指令
    27. void Lcd_WriteIndex(unsigned char Index)
    28. {
    29. //SPI 写命令时序开始
    30. LCD_CS_CLR;
    31. SPI_WriteData(0x70);
    32. SPI_WriteData(Index);
    33. LCD_CS_SET;
    34. }
    35. //向液晶屏写一个8位数据
    36. void Lcd_WriteData(unsigned char Data)
    37. {
    38. LCD_CS_CLR;
    39. SPI_WriteData(0x72);
    40. SPI_WriteData(Data);
    41. LCD_CS_SET;
    42. }
    43. //向液晶屏写一个16位数据
    44. void LCD_WriteData_16Bit(unsigned int Data)
    45. {
    46. LCD_CS_CLR;
    47. SPI_WriteData(0x72);
    48. SPI_WriteData(Data>>8);
    49. SPI_WriteData(Data);
    50. LCD_CS_SET;
    51. }
    52. void Lcd_Write_REG(unsigned char Index,unsigned char Data)
    53. {
    54. Lcd_WriteIndex(Index);
    55. Lcd_WriteData(Data);
    56. }
    57. void Lcd_Reset(void)
    58. {
    59. unsigned int i;
    60. LCD_RST_CLR;
    61. Delay_Ms(50);
    62. LCD_RST_SET;
    63. Delay_Ms(50);
    64. }
    65. // set region to paint
    66. void LCD_SetWindow(unsigned int x1,unsigned int y1,unsigned int x2,unsigned int y2)
    67. {
    68. //SC
    69. Lcd_Write_REG(0x02,x1>>8); // Column address start2
    70. Lcd_Write_REG(0x03,(u8)x1); // Column address start1
    71. //EC
    72. Lcd_Write_REG(0x04,x2>>8); // Column address end2
    73. Lcd_Write_REG(0x05,(u8)x2); // Column address end1
    74. //SP
    75. Lcd_Write_REG(0x06,y1>>8); // Row address start2
    76. Lcd_Write_REG(0x07,(u8)y1); // Row address start1
    77. //EP
    78. Lcd_Write_REG(0x08,y2>>8); // Row address end2
    79. Lcd_Write_REG(0x09,(u8)y2); // Row address end1
    80. //写0x22到index register,那么下次send data就会直接被写到graphic ram
    81. Lcd_WriteIndex(0x22);
    82. }
    83. void FillRect(u16 x1, u16 y1, u16 x2, u16 y2, u16 color)
    84. {
    85. LCD_SetWindow(x1, y1,x2,y2);
    86. x2 = x2 - x1 + 1;
    87. y2 = y2 - y1 + 1;
    88. for(x1 = x2; x1 != 0 ; x1--)
    89. {
    90. for (y1 = y2;y1 != 0 ;y1--)
    91. {
    92. LCD_WriteData_16Bit(color);
    93. }
    94. }
    95. }
    96. void LCD_Init(void)
    97. {
    98. LCD_GPIO_Init();
    99. Lcd_Reset();
    100. Lcd_Write_REG(0x18,0x88); //UADJ 75Hz
    101. Lcd_Write_REG(0x19,0x01); //OSC_EN='1', start Osc
    102. //Power Voltage Setting
    103. Lcd_Write_REG(0x1B,0x1E); //VRH=4.60V
    104. Lcd_Write_REG(0x1C,0x07); //AP Crosstalk 04
    105. Lcd_Write_REG(0x1A,0x01); //BT (VGH~15V,VGL~-10V,DDVDH~5V)
    106. Lcd_Write_REG(0x24,0x38); //VMH 27
    107. Lcd_Write_REG(0x25,0x5F); //VML
    108. //VCOM offset
    109. Lcd_Write_REG(0x23,0x8C); //for Flicker adjust
    110. Lcd_Write_REG(0x1F,0x88);// GAS=1, VOMG=00, PON=0, DK=1, XDK=0, DVDH_TRI=0, STB=0
    111. Delay_Ms(5);
    112. Lcd_Write_REG(0x1F,0x80);// GAS=1, VOMG=00, PON=0, DK=0, XDK=0, DVDH_TRI=0, STB=0
    113. Delay_Ms(5);
    114. Lcd_Write_REG(0x1F,0x90);// GAS=1, VOMG=00, PON=1, DK=0, XDK=0, DVDH_TRI=0, STB=0
    115. Delay_Ms(5);
    116. Lcd_Write_REG(0x1F,0xD0);// GAS=1, VOMG=10, PON=1, DK=0, XDK=0, DDVDH_TRI=0, STB=0
    117. Delay_Ms(5);
    118. //Display ON Setting
    119. Lcd_Write_REG(0x28,0x38); //GON=1, DTE=1, D=1000
    120. Delay_Ms(40);
    121. Lcd_Write_REG(0x28,0x3C); //GON=1, DTE=1, D=1100
    122. Lcd_Write_REG(0x36,0x09); //REV, BGR
    123. Lcd_Write_REG(0x17,0x05); //16BIT/PIXEL
    124. //Gamma 2.2 Setting
    125. Lcd_Write_REG(0x40,0x00); //
    126. Lcd_Write_REG(0x41,0x00); //
    127. Lcd_Write_REG(0x42,0x00); //
    128. Lcd_Write_REG(0x43,0x11); //
    129. Lcd_Write_REG(0x44,0x0e); //
    130. Lcd_Write_REG(0x45,0x23); //
    131. Lcd_Write_REG(0x46,0x08); //
    132. Lcd_Write_REG(0x47,0x53); //
    133. Lcd_Write_REG(0x48,0x03); //
    134. Lcd_Write_REG(0x49,0x11); //
    135. Lcd_Write_REG(0x4A,0x18); //
    136. Lcd_Write_REG(0x4B,0x1a); //
    137. Lcd_Write_REG(0x4C,0x16); //
    138. Lcd_Write_REG(0x50,0x1c); //
    139. Lcd_Write_REG(0x51,0x31); //
    140. Lcd_Write_REG(0x52,0x2e); //
    141. Lcd_Write_REG(0x53,0x3f); //
    142. Lcd_Write_REG(0x54,0x3f); //
    143. Lcd_Write_REG(0x55,0x3f); //
    144. Lcd_Write_REG(0x56,0x2c); //
    145. Lcd_Write_REG(0x57,0x77); //
    146. Lcd_Write_REG(0x58,0x09); //
    147. Lcd_Write_REG(0x59,0x05); //
    148. Lcd_Write_REG(0x5A,0x07); //
    149. Lcd_Write_REG(0x5B,0x0e); //
    150. Lcd_Write_REG(0x5C,0x1c); //
    151. Lcd_Write_REG(0x5D,0x88); //
    152. Delay_Ms(100);
    153. FillRect(0, 160, 239, 239, 0xf800 );
    154. }

    二、说明

    1、LCD初始化是从GITHUB上拷的一段程序,具体是啥意思我也不清楚。试过官方数据手册里的初始化程序,一直不能成功。

    2、写寄存器的程序

    void Lcd_WriteIndex(unsigned char Index)
    {
       LCD_CS_CLR;
       SPI_WriteData(0x70);
       SPI_WriteData(Index);
       LCD_CS_SET;
    }

     3、写数据的程序

    void Lcd_WriteData(unsigned char Data)
    {
       LCD_CS_CLR;
       SPI_WriteData(0x72);
       SPI_WriteData(Data);

       LCD_CS_SET;
    }

            上面程序中出现SPI_WriteData(0x70); SPI_WriteData(0x72);统一解释下:

            因为是3线SPI,所以缺少CD(或RS)引线,解决的办法一般是SPI发送数据时第一个bit的值代表CD。这样因为多出了1bit,就需要处理9bit的数据。UC1601就是这样,但是HX8347不是这样。

            下面摘自HX8347的数据手册:

    fd17fcefa9e34c62a4180b2c2fa0f32b.png

    87780c2fc9a94fe089da65e3175e7d94.png

            大概意思就是在发数据前需要发“01110”[ID][RS][RW],一共是8bit。

            这里ID应该取0。RS取0是写指令,取1是写数据。RW取0,表示写。

            所以写指令就是:01110000=0x70;写数据就是:01110010=0x72。也因此就有了:

            SPI_WriteData(0x70); SPI_WriteData(0x72);

    三、总结

    1、主控是沁恒家申请的CH32V307的评估板,开发工具是MounRiver。不得不说评估板和开发工具都很完美。评估板留有arduino接口,可以和arduino扩展板链接。MounRiver编译很快,下载烧录更快,基本都是零点几秒。

    2、本源码可以直接添加到CH32V307的demo中CH32V307EVT\EVT\EXAM\GPIO\GPIO_Toggle例程中运行。

    int main(void)
    {
        NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);
        SystemCoreClockUpdate();
        Delay_Init();

     

        LCD_Init();
        while(1)
        {
            Delay_Ms(1000);
            FillRect(0, 0, 239, 319, 0x07c0 );
            Delay_Ms(1000);
            FillRect(0, 0, 239, 319, 0x0030 );
            Delay_Ms(1000);
            FillRect(0, 0, 239, 319, 0x0000 );
            Delay_Ms(1000);
            FillRect(0, 0, 239, 319, 0xf800 );
        }
    }

    3、用的是软件模拟3线SPI,所以帧数很低。以后考虑硬件SPI,或再加DMA,但DMA需要SRAM太大,可能玩不转。

    演示用视频

     

     

     

     

     

     

     

  • 相关阅读:
    gRPC之gRPC转换HTTP
    centos 上安装 kafka 与 python 调用
    怎么做手机App测试?app测试详细流程和方法介绍
    Android笔记:(最全)判断Item在RecyclerView中滑动到顶部或底部
    SSM处理过程
    java基于微信小程序的超市购物商城系统 小程序 uniapp
    quarkus依赖注入之一:创建bean
    Google Earth Engine APP——gee-ui geetemp 前端团队组件库
    JavaScript 解决dayjs在周日获取当前周第一天显示下一周第一天问题
    图像绘制-线段、矩形、圆形、椭圆等
  • 原文地址:https://blog.csdn.net/weixin_44067125/article/details/134274918