你的浏览器版本过低,可能导致网站不能正常访问!
为了你能正常使用网站功能,请使用这些浏览器。

【经验分享】STM32驱动SPI接口的ST7798

[复制链接]
STMCU小助手 发布时间:2022-4-23 16:44
lcd.c

  1. void lcd_init()
  2. {
  3.          GPIO_InitTypeDef GPIO_InitStructure;
  4.     SPI_InitTypeDef  SPI_InitStructure;

  5.         RCC_APB2PeriphClockCmd(        RCC_APB2Periph_GPIOA|RCC_APB2Periph_GPIOB|RCC_APB2Periph_AFIO, ENABLE );//PORTA时钟使能
  6.         RCC_APB2PeriphClockCmd(        RCC_APB2Periph_SPI1,  ENABLE );//SPI1时钟使能         

  7.         GPIO_PinRemapConfig(GPIO_Remap_SWJ_JTAGDisable, ENABLE);
  8.         
  9.         
  10.         GPIO_InitStructure.GPIO_Pin = GPIO_Pin_4|GPIO_Pin_8|GPIO_Pin_15;
  11.     GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
  12.     GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;         
  13.     GPIO_Init(GPIOA, &GPIO_InitStructure);
  14.         GPIO_SetBits(GPIOA,GPIO_Pin_4|GPIO_Pin_8|GPIO_Pin_15);

  15.         GPIO_InitStructure.GPIO_Pin = GPIO_Pin_3;
  16.     GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
  17.     GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;         
  18.     GPIO_Init(GPIOB, &GPIO_InitStructure);
  19.         GPIO_SetBits(GPIOB,GPIO_Pin_3);        
  20.         
  21.         
  22.         GPIO_InitStructure.GPIO_Pin = GPIO_Pin_5 | GPIO_Pin_6| GPIO_Pin_7;
  23.         GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;  
  24.         GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
  25.         GPIO_Init(GPIOA, &GPIO_InitStructure);//初始化GPIOA

  26.          GPIO_SetBits(GPIOA,GPIO_Pin_5 | GPIO_Pin_6| GPIO_Pin_7);  //PB13/14/15上拉

  27.         SPI_InitStructure.SPI_Direction = SPI_Direction_2Lines_FullDuplex;  //设置SPI单向或者双向的数据模式:SPI设置为双线双向全双工
  28.         SPI_InitStructure.SPI_Mode = SPI_Mode_Master;                //设置SPI工作模式:设置为主SPI
  29.         SPI_InitStructure.SPI_DataSize = SPI_DataSize_8b;                //设置SPI的数据大小:SPI发送接收8位帧结构
  30.         SPI_InitStructure.SPI_CPOL = SPI_CPOL_High;                //串行同步时钟的空闲状态为高电平
  31.         SPI_InitStructure.SPI_CPHA = SPI_CPHA_2Edge;        //串行同步时钟的第二个跳变沿(上升或下降)数据被采样
  32.         SPI_InitStructure.SPI_NSS = SPI_NSS_Soft;                //NSS信号由硬件(NSS管脚)还是软件(使用SSI位)管理:内部NSS信号有SSI位控制
  33.         SPI_InitStructure.SPI_BaudRatePrescaler = SPI_BaudRatePrescaler_256;                //定义波特率预分频的值:波特率预分频值为256
  34.         SPI_InitStructure.SPI_FirstBit = SPI_FirstBit_MSB;        //指定数据传输从MSB位还是LSB位开始:数据传输从MSB位开始
  35.         SPI_InitStructure.SPI_CRCPolynomial = 7;        //CRC值计算的多项式
  36.         SPI_Init(SPI1, &SPI_InitStructure);  //根据SPI_InitStruct中指定的参数初始化外设SPIx寄存器

  37.         SPI_Cmd(SPI1, ENABLE); //使能SPI外设
  38.         
  39. //        lcd_read_write_byte(0xff);//启动传输        
  40.         
  41.         
  42.         lcd_rst=1;
  43.         delay_ms(120);
  44.         lcd_rst=0;
  45.         delay_ms(120);
  46.         lcd_rst=1;
  47.     delay_ms(120);
  48.     /* Sleep Out */
  49.     lcd_write_cmd(0x11);
  50.     /* wait for power stability */
  51.     delay_ms(120);

  52.         lcd_write_cmd(0x36);
  53.         lcd_write_data(0x00);
  54.         lcd_write_cmd(0x3a);
  55.         lcd_write_data(0x05);
  56.         //--------------------------------ST7789V Frame rate setting----------------------------------//
  57.         lcd_write_cmd(0xb2);
  58.         lcd_write_data(0x0c);
  59.         lcd_write_data(0x0c);
  60.         lcd_write_data(0x00);
  61.         lcd_write_data(0x33);
  62.         lcd_write_data(0x33);
  63.         lcd_write_cmd(0xb7);
  64.         lcd_write_data(0x35);
  65.         //---------------------------------ST7789V Power setting--------------------------------------//
  66.         lcd_write_cmd(0xbb);
  67.         lcd_write_data(0x1c);
  68.         lcd_write_cmd(0xc0);
  69.         lcd_write_data(0x2c);
  70.         lcd_write_cmd(0xc2);
  71.         lcd_write_data(0x01);
  72.         lcd_write_cmd(0xc3);
  73.         lcd_write_data(0x0b);
  74.         lcd_write_cmd(0xc4);
  75.         lcd_write_data(0x20);
  76.         lcd_write_cmd(0xc6);
  77.         lcd_write_data(0x0f);
  78.         lcd_write_cmd(0xd0);
  79.         lcd_write_data(0xa4);
  80.         lcd_write_data(0xa1);
  81.         //--------------------------------ST7789V gamma setting---------------------------------------//
  82.         lcd_write_cmd(0xe0);
  83.         lcd_write_data(0xd0);
  84.         lcd_write_data(0x00);
  85.         lcd_write_data(0x03);
  86.         lcd_write_data(0x09);
  87.         lcd_write_data(0x13);
  88.         lcd_write_data(0x1c);
  89.         lcd_write_data(0x3a);
  90.         lcd_write_data(0x55);
  91.         lcd_write_data(0x48);
  92.         lcd_write_data(0x18);
  93.         lcd_write_data(0x12);
  94.         lcd_write_data(0x0e);
  95.         lcd_write_data(0x19);
  96.         lcd_write_data(0x1e);
  97.         lcd_write_cmd(0xe1);
  98.         lcd_write_data(0xd0);
  99.         lcd_write_data(0x00);
  100.         lcd_write_data(0x03);
  101.         lcd_write_data(0x09);
  102.         lcd_write_data(0x05);
  103.         lcd_write_data(0x25);
  104.         lcd_write_data(0x3a);
  105.         lcd_write_data(0x55);
  106.         lcd_write_data(0x50);
  107.         lcd_write_data(0x3d);
  108.         lcd_write_data(0x1c);
  109.         lcd_write_data(0x1d);
  110.         lcd_write_data(0x1d);
  111.         lcd_write_data(0x1e);
  112.         lcd_write_cmd(0x29);

  113.     lcd_address_set(0, 0, LCD_Width - 1, LCD_Height - 1);
  114.         lcd_set_speed(SPI_BaudRatePrescaler_2);
  115.         open_lcd();        


  116.         
  117. }

  118. //SPI 速度设置函数
  119. //SpeedSet:
  120. //SPI_BaudRatePrescaler_2   2分频   
  121. //SPI_BaudRatePrescaler_8   8分频   
  122. //SPI_BaudRatePrescaler_16  16分频  
  123. //SPI_BaudRatePrescaler_256 256分频

  124. void lcd_set_speed(u8 SPI_BaudRatePrescaler)
  125. {
  126.         assert_param(IS_SPI_BAUDRATE_PRESCALER(SPI_BaudRatePrescaler));
  127.         SPI1->CR1&=0XFFC7;
  128.         SPI1->CR1|=SPI_BaudRatePrescaler;        //设置SPI1速度
  129.         SPI_Cmd(SPI1,ENABLE);

  130. }
  131. //SPIx 读写一个字节
  132. //TxData:要写入的字节
  133. //返回值:读取到的字节
  134. u8 lcd_read_write_byte(u8 TxData)
  135. {               
  136.         u8 retry=0;                                         
  137.         while (SPI_I2S_GetFlagStatus(SPI1, SPI_I2S_FLAG_TXE) == RESET) //检查指定的SPI标志位设置与否:发送缓存空标志位
  138.         {
  139.                 retry++;
  140.                 if(retry>200)return 0;
  141.         }                          
  142.         SPI_I2S_SendData(SPI1, TxData); //通过外设SPIx发送一个数据
  143.         retry=0;

  144.         while (SPI_I2S_GetFlagStatus(SPI1, SPI_I2S_FLAG_RXNE) == RESET) //检查指定的SPI标志位设置与否:接受缓存非空标志位
  145.         {
  146.                 retry++;
  147.                 if(retry>200)return 0;
  148.         }                                                              
  149.         return SPI_I2S_ReceiveData(SPI1); //返回通过SPIx最近接收的数据                                            
  150. }

  151. /**
  152. * @brief        写命令到LCD
  153. *
  154. * @param   cmd                需要发送的命令
  155. *
  156. * @return  void
  157. */
  158. void lcd_write_cmd(u8 cmd)
  159. {
  160.     lcd_dc=0;
  161.         lcd_cs=0;

  162.     lcd_read_write_byte(cmd);
  163.         lcd_cs=1;
  164. }
  165. /**
  166. * @brief        写数据到LCD
  167. *
  168. * @param   cmd                需要发送的数据
  169. *
  170. * @return  void
  171. */
  172. void lcd_write_data(u8 data)
  173. {
  174.    lcd_dc=1;
  175.         lcd_cs=0;
  176.    lcd_read_write_byte(data);
  177.         lcd_cs=1;
  178. }
  179. /**
  180. * @brief        写半个字的数据到LCD
  181. *
  182. * @param   cmd                需要发送的数据
  183. *
  184. * @return  void
  185. */
  186. void lcd_write_halfword(const u16 da)
  187. {
  188.         u8 data[2] = {0},i=0;

  189.         data[0] = da >> 8;
  190.         data[1] = da;

  191.         lcd_dc=1;
  192.         lcd_cs=0;
  193.         for(i=0;i<2;i++)
  194.         {
  195.                 lcd_read_write_byte(data<i>);        
  196.         }
  197.         lcd_cs=1;
  198.                
  199. }

  200. /**
  201. * 设置数据写入LCD缓存区域
  202. *
  203. * @param   x1,y1        起点坐标
  204. * @param   x2,y2        终点坐标
  205. *
  206. * @return  void
  207. */
  208. void lcd_address_set(u16 x1, u16 y1, u16 x2, u16 y2)
  209. {
  210.     lcd_write_cmd(0x2a);
  211.     lcd_write_data(x1 >> 8);
  212.     lcd_write_data(x1);
  213.     lcd_write_data(x2 >> 8);
  214.     lcd_write_data(x2);

  215.     lcd_write_cmd(0x2b);
  216.     lcd_write_data(y1 >> 8);
  217.     lcd_write_data(y1);
  218.     lcd_write_data(y2 >> 8);
  219.     lcd_write_data(y2);

  220.     lcd_write_cmd(0x2C);
  221. }

  222. /**
  223. * 以一种颜色清空LCD屏
  224. *
  225. * @param   color        清屏颜色
  226. *
  227. * @return  void
  228. */
  229. void lcd_clear(u16 color)
  230. {
  231.     unsigned int i;
  232.         u8 data[2] = {0};

  233.     data[0] = color >> 8;
  234.     data[1] = color;

  235.         lcd_address_set(0, 0, LCD_Width - 1, LCD_Height - 1);
  236.         
  237.         lcd_dc=1;
  238.         lcd_cs=0;
  239.     for(i = 0; i < (LCD_Width*LCD_Height); i++)
  240.     {
  241.                 lcd_read_write_byte(data[0]);
  242.                 lcd_read_write_byte(data[1]);
  243.     }
  244.         lcd_cs=0;
  245. }

  246. /**
  247. * 画点函数
  248. *
  249. * @param   x,y                画点坐标
  250. *
  251. * @return  void
  252. */
  253. void lcd_draw_point(u16 x, u16 y,u16 pointColor)
  254. {
  255.     lcd_address_set(x, y, x, y);
  256.     lcd_write_halfword(pointColor);
  257. }

  258. /**
  259. * @brief        画一个圆
  260. *
  261. * @param   x0,y0        圆心坐标
  262. * @param   r       圆半径
  263. *
  264. * @return  void
  265. */
  266. void lcd_draw_circle(u16 x0, u16 y0, u8 r,u16 pointColor)
  267. {
  268.     int a, b;
  269.     int di;
  270.     a = 0;
  271.     b = r;
  272.     di = 3 - (r << 1);

  273.     while(a <= b)
  274.     {
  275.         lcd_draw_point(x0 - b, y0 - a,pointColor);
  276.         lcd_draw_point(x0 + b, y0 - a,pointColor);
  277.         lcd_draw_point(x0 - a, y0 + b,pointColor);
  278.         lcd_draw_point(x0 - b, y0 - a,pointColor);
  279.         lcd_draw_point(x0 - a, y0 - b,pointColor);
  280.         lcd_draw_point(x0 + b, y0 + a,pointColor);
  281.         lcd_draw_point(x0 + a, y0 - b,pointColor);
  282.         lcd_draw_point(x0 + a, y0 + b,pointColor);
  283.         lcd_draw_point(x0 - b, y0 + a,pointColor);
  284.         a++;

  285.         if(di < 0)di += 4 * a + 6;
  286.         else
  287.         {
  288.             di += 10 + 4 * (a - b);
  289.             b--;
  290.         }

  291.         lcd_draw_point(x0 + a, y0 + b,pointColor);
  292.     }
  293. }

  294. /**
  295. * @brief        显示一个ASCII码字符
  296. *
  297. * @param   x,y                显示起始坐标
  298. * @param   chr                需要显示的字符
  299. * @param   size        字体大小(支持16/24/32号字体)
  300. *
  301. * @return  void
  302. */
  303. void lcd_show_char(u16 x, u16 y, char chr, u8 size,u16 pointColor,u16 backColor)
  304. {
  305.     u8 temp, t1, t;
  306.     u8 csize;                //得到字体一个字符对应点阵集所占的字节数
  307.     u16 colortemp;
  308.     u8 sta;

  309.     chr = chr - ' '; //得到偏移后的值(ASCII字库是从空格开始取模,所以-' '就是对应字符的字库)

  310.     if((x > (LCD_Width - size / 2)) || (y > (LCD_Height - size)))        return;

  311.     lcd_address_set(x, y, x + size / 2 - 1, y + size - 1);//(x,y,x+8-1,y+16-1)

  312.     if((size == 16) || (size == 32) )        //16和32号字体
  313.     {
  314.         csize = (size / 8 + ((size % 8) ? 1 : 0)) * (size / 2);

  315.         for(t = 0; t < csize; t++)
  316.         {
  317.             if(size == 16)temp = asc2_1608[chr][t];        //调用1608字体
  318.             else if(size == 32)temp = asc2_3216[chr][t];        //调用3216字体
  319.             else return;                        //没有的字库

  320.             for(t1 = 0; t1 < 8; t1++)
  321.             {
  322.                 if(temp & 0x80) colortemp = pointColor;
  323.                 else colortemp = backColor;

  324.                 lcd_write_halfword(colortemp);
  325.                 temp <<= 1;
  326.             }
  327.         }
  328.     }

  329.         else if  (size == 12)        //12号字体
  330.         {
  331.         csize = (size / 8 + ((size % 8) ? 1 : 0)) * (size / 2);

  332.         for(t = 0; t < csize; t++)
  333.         {
  334.             temp = asc2_1206[chr][t];

  335.             for(t1 = 0; t1 < 6; t1++)
  336.             {
  337.                 if(temp & 0x80) colortemp = pointColor;
  338.                 else colortemp = backColor;

  339.                 lcd_write_halfword(colortemp);
  340.                 temp <<= 1;
  341.             }
  342.         }
  343.     }
  344.         
  345.     else if(size == 24)                //24号字体
  346.     {
  347.         csize = (size * 16) / 8;

  348.         for(t = 0; t < csize; t++)
  349.         {
  350.             temp = asc2_2412[chr][t];

  351.             if(t % 2 == 0)sta = 8;
  352.             else sta = 4;

  353.             for(t1 = 0; t1 < sta; t1++)
  354.             {
  355.                 if(temp & 0x80) colortemp = pointColor;
  356.                 else colortemp = backColor;

  357.                 lcd_write_halfword(colortemp);
  358.                 temp <<= 1;
  359.             }
  360.         }
  361.     }
  362. }


  363. /**
  364. * @brief        显示字符串
  365. *
  366. * @param   x,y                起点坐标
  367. * @param   width        字符显示区域宽度
  368. * @param   height        字符显示区域高度
  369. * @param   size        字体大小
  370. * @param   p                字符串起始地址
  371. *
  372. * @return  void
  373. */
  374. void lcd_show_string(u16 x, u16 y, u16 width, u16 height, u8 size, char *p,u16 pointColor,u16 backColor)
  375. {
  376.     u8 x0 = x;
  377.     width += x;
  378.     height += y;

  379.     while((*p <= '~') && (*p >= ' ')) //判断是不是非法字符!
  380.     {
  381.         if(x >= width)
  382.         {
  383.             x = x0;
  384.             y += size;
  385.         }

  386.         if(y >= height)break; //退出

  387.         lcd_show_char(x, y, *p, size,pointColor,backColor);
  388.         x += size / 2;
  389.         p++;
  390.     }
  391. }

  392. /**
  393. * @brief        显示图片
  394. *
  395. * @remark        Image2Lcd取模方式:        C语言数据/水平扫描/16位真彩色(RGB565)/高位在前                其他的不要选
  396. *
  397. * @param   x,y                起点坐标
  398. * @param   width        图片宽度
  399. * @param   height        图片高度
  400. * @param   p                图片缓存数据起始地址
  401. *
  402. * @return  void
  403. */
  404. void lcd_show_image(u16 x, u16 y, u16 width, u16 height, const u8 *p)
  405. {
  406.         u16 i;
  407.     if(x + width > LCD_Width || y + height > LCD_Height)
  408.     {
  409.         return;
  410.     }

  411.     lcd_address_set(x, y, x + width - 1, y + height - 1);

  412.     lcd_dc=1;

  413.         for(i=0;i<width * height * 2;i++)
  414.         {
  415.                 lcd_read_write_byte(p<i>);
  416.         }
  417. }
  418. </i></i>
复制代码

lcd.h

  1. #define lcd_cs PAout(4)
  2. #define lcd_dc PAout(8)
  3. #define lcd_rst PAout(15)
  4. #define lcd_background PBout(3)


  5. #define open_lcd() lcd_background=1
  6. #define close_lcd() lcd_background=0


  7. //LCD的宽和高定义
  8. #define LCD_Width         240
  9. #define LCD_Height         320

  10. //画笔颜色
  11. #define WHITE                  0xFFFF
  12. #define BLACK                  0x0000         
  13. #define BLUE                  0x001F  
  14. #define BRED             0XF81F
  15. #define GRED                          0XFFE0
  16. #define GBLUE                         0X07FF
  17. #define RED                    0xF800
  18. #define MAGENTA                0xF81F
  19. #define GREEN                  0x07E0
  20. #define CYAN                   0x7FFF
  21. #define YELLOW                 0xFFE0
  22. #define BROWN                          0XBC40 //棕色
  23. #define BRRED                          0XFC07 //棕红色
  24. #define GRAY                           0X8430 //灰色
  25. //GUI颜色

  26. #define DARKBLUE               0X01CF        //深蓝色
  27. #define LIGHTBLUE               0X7D7C        //浅蓝色  
  28. #define GRAYBLUE                0X5458 //灰蓝色
  29. //以上三色为PANEL的颜色

  30. #define LIGHTGREEN              0X841F //浅绿色
  31. //#define LIGHTGRAY        0XEF5B //浅灰色(PANNEL)
  32. #define LGRAY                          0XC618 //浅灰色(PANNEL),窗体背景色

  33. #define LGRAYBLUE        0XA651 //浅灰蓝色(中间层颜色)
  34. #define LBBLUE           0X2B12 //浅棕蓝色(选择条目的反色)



  35. void lcd_init(void);
  36. void lcd_set_speed(u8 SPI_BaudRatePrescaler);
  37. u8 lcd_read_write_byte(u8 TxData);
  38. void lcd_clear(u16 color);
  39. u16 lcd_read_id(void);
  40. void lcd_draw_point(u16 x, u16 y,u16 pointColor);
  41. void lcd_show_image(u16 x, u16 y, u16 width, u16 height, const u8 *p);
  42. void lcd_show_string(u16 x, u16 y, u16 width, u16 height, u8 size, char *p,u16 pointColor,u16 backColor);
  43. void lcd_show_char(u16 x, u16 y, char chr, u8 size,u16 pointColor,u16 backColor);
  44. void lcd_draw_circle(u16 x0, u16 y0, u8 r,u16 pointColor);
  45. void lcd_address_set(u16 x1, u16 y1, u16 x2, u16 y2);
  46. void lcd_write_data(u8 data);
  47. void lcd_write_cmd(u8 cmd);


复制代码

收藏 评论0 发布时间:2022-4-23 16:44

举报

0个回答
关于
我们是谁
投资者关系
意法半导体可持续发展举措
创新与技术
意法半导体官网
联系我们
联系ST分支机构
寻找销售人员和分销渠道
社区
媒体中心
活动与培训
隐私策略
隐私策略
Cookies管理
行使您的权利
官方最新发布
STM32N6 AI生态系统
STM32MCU,MPU高性能GUI
ST ACEPACK电源模块
意法半导体生物传感器
STM32Cube扩展软件包
关注我们
st-img 微信公众号
st-img 手机版