STM32 硬件SPI+DMA实现快速刷TFT屏

这篇具有很好参考价值的文章主要介绍了STM32 硬件SPI+DMA实现快速刷TFT屏。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

1.配置硬件SPI实现刷屏

首先在TB上找一块SPI驱动的彩屏,下载商家提供的示例

例如我买的一款2.8寸SPI的TFT彩屏,商家提供的资料很齐全,模拟SPI和硬件SPI驱动的程序都有

tft dma stm32,stm32,嵌入式硬件,单片机

打开硬件SPI驱动的工程,商家提供的代码是SPI2驱动,想换成其他的SPI可以到SPI.c文件中更改

void SPI2_Init(void)	
{
	SPI_InitTypeDef  SPI_InitStructure;
	GPIO_InitTypeDef GPIO_InitStructure;
	 
	//配置SPI2管脚
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_AFIO|RCC_APB2Periph_GPIOB, ENABLE);
	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_13 | GPIO_Pin_15;
	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;
	GPIO_Init(GPIOB, &GPIO_InitStructure);

	GPIO_InitStructure.GPIO_Pin =  GPIO_Pin_14;    
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPU; 
	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;  
	GPIO_Init(GPIOB, &GPIO_InitStructure);  
	
	//SPI2配置选项
	RCC_APB1PeriphClockCmd(RCC_APB1Periph_SPI2 ,ENABLE);
	   
	SPI_InitStructure.SPI_Direction = SPI_Direction_2Lines_FullDuplex;
	SPI_InitStructure.SPI_Mode = SPI_Mode_Master;
	SPI_InitStructure.SPI_DataSize = SPI_DataSize_8b;
	SPI_InitStructure.SPI_CPOL = SPI_CPOL_Low;
	SPI_InitStructure.SPI_CPHA = SPI_CPHA_1Edge;
	SPI_InitStructure.SPI_NSS = SPI_NSS_Soft;
	SPI_InitStructure.SPI_BaudRatePrescaler = SPI_BaudRatePrescaler_2;
	SPI_InitStructure.SPI_FirstBit = SPI_FirstBit_MSB;
	SPI_InitStructure.SPI_CRCPolynomial = 7;
	SPI_Init(SPI2, &SPI_InitStructure);

	//使能SPI2
	SPI_Cmd(SPI2, ENABLE);   
}

打开main.c,测试一下简单颜色填充刷屏,下载到单片机

int main(void)
{	
	SystemInit();//初始化RCC 设置系统主频为72MHZ
	delay_init(72);	     //延时初始化
	LCD_Init();	   //液晶屏初始化
  //循环测试
	while(1)
	{
//		main_test(); 		//测试主界面
		Test_Color();  		//简单刷屏填充测试
//		Test_FillRec();		//GUI矩形绘图测试
//		Test_Circle(); 		//GUI画圆测试
//		Test_Triangle();    //GUI三角形绘图测试
//		English_Font_test();//英文字体示例测试
//		Chinese_Font_test();//中文字体示例测试
//		Pic_test();			//图片显示示例测试
//		Rotate_Test();   //旋转显示测试
	}
}

硬件SPI刷屏

 相比于模拟SPI,刷屏速度还是很快的

2.配置硬件SPI+DMA实现快速刷屏

首先配置好DMA的初始化

#include "MyDMA.h"                  // Device header
#include "lcd.h"

u8 SendBuff[2*240];
DMA_InitTypeDef DMA_InitStructure;
u32 DMA1_MEM_LEN;

void MYDMA_Config(DMA_Channel_TypeDef* DMA_CHx,u32 cpar,u32 cmar,u16 cndtr)
{
		RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE);	//使能DMA传输

		DMA_DeInit(DMA_CHx);   															//将DMA的通道1寄存器重设为缺省值
		DMA1_MEM_LEN=cndtr;
		DMA_InitStructure.DMA_PeripheralBaseAddr = cpar;  	//DMA外设ADC基地址
		DMA_InitStructure.DMA_MemoryBaseAddr = cmar;  			//DMA内存基地址
		DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralDST;  //数据传输方向,从内存读取发送到外设
		DMA_InitStructure.DMA_BufferSize = cndtr;  					//DMA通道的DMA缓存的大小
		DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;  //外设地址寄存器不变
		DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Disable;  				//内存地址寄存器不变
		DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_HalfWord; //数据宽度为16位
		DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_HalfWord; //数据宽度为16位
		DMA_InitStructure.DMA_Mode = DMA_Mode_Normal;  	//工作在正常缓存模式
		DMA_InitStructure.DMA_Priority = DMA_Priority_High; 	//DMA通道 x拥有中优先级 
		DMA_InitStructure.DMA_M2M = DMA_M2M_Disable;  //DMA通道x没有设置为内存到内存传输
		DMA_Init(DMA_CHx, &DMA_InitStructure);  //根据DMA_InitStruct中指定的参数初始化DMA的通道USART1_Tx_DMA_Channel所标识的寄存器
} 


void MYDMA_Enable(DMA_Channel_TypeDef*DMA_CHx)
{
		DMA_Cmd(DMA_CHx, DISABLE);
		DMA_SetCurrDataCounter(DMA_CHx, 2*240);
		DMA_Cmd(DMA_CHx, ENABLE);
}

void DMA_Start(void)
{
		LCD_CS_CLR; //关闭片选
		LCD_RS_SET; //写数据
		SPI_I2S_DMACmd(SPI2,SPI_I2S_DMAReq_Tx,ENABLE);
		MYDMA_Enable(DMA1_Channel5);
		while(DMA_GetFlagStatus(DMA1_FLAG_TC5)==RESET);
		DMA_ClearFlag(DMA1_FLAG_TC5);//清除通道5传输完成标志
}

由于用的是SPI2,所以选择的DMA1的通道5,具体的各个外设对应的DMA通道可以查看数据手册(SPI1对应DMA1的通道3),于是我们在LCD_Init()中加入DMA的初始化

	MYDMA_Config(DMA1_Channel5,(u32)&(SPI2->DR),(u32)SendBuff,2*lcddev.width);

注意:SendBuff是DMA通道的缓存大小,SendBuff[2*lcddev.width],lcddev.width为屏幕的长,具体长度要看你的屏幕设置的横向还是纵向

接着修改LCD的填充和清屏函数

void LCD_Fill(u16 sx,u16 sy,u16 ex,u16 ey,u16 color)
{  	
	u16 i,j;			
//	u16 width=ex-sx+1; 		//得到填充的宽度
//	u16 height=ey-sy+1;		//高度
	LCD_SetWindows(sx,sy,ex,ey);//设置显示窗口
//	for(i=0;i<height;i++)
//	{
//		for(j=0;j<width;j++)
//		Lcd_WriteData_16Bit(color);	//写入数据 	 
//	}
	for(j=0;j<ey;j++)
	{
		for(i=0;i<ex;i++)
		{
			SendBuff[2*i] = color>>8;
			SendBuff[2*i+1] = color;				
		}
		DMA_Start();	//启动传输	
	} 
	LCD_SetWindows(0,0,lcddev.width-1,lcddev.height-1);//恢复窗口设置为全屏
}

void LCD_Clear(u16 Color)
{
  unsigned int i,m;  
	DMA1_MEM_LEN = 2*lcddev.width;
	LCD_SetWindows(0,0,lcddev.width-1,lcddev.height-1);   
//	LCD_CS_CLR;
//	LCD_RS_SET;
//	for(i=0;i<lcddev.height;i++)
//	{
//    for(m=0;m<lcddev.width;m++)
//    {	
//			Lcd_WriteData_16Bit(Color);
//		}
//	}
	for(i=0;i<2*lcddev.width;)
	{
			SendBuff[i] = Color>>8;
			SendBuff[i+1] = Color;
			i+=2;
	}
	for(m=0;m<lcddev.height;m++)
	{
			DMA_Start();
	}
//	LCD_CS_SET;
} 

下载代码测试

硬件SPI+DMA刷屏

对比上文硬件SPI刷屏,加上DMA后刷屏速度肉眼可见的变快了

3.其他屏幕函数

其他的屏幕函数可以以此类推更改文章来源地址https://www.toymoban.com/news/detail-852362.html

void LCD_ShowChar(u16 x,u16 y,u8 num,u16 fc,u16 bc,u8 sizey,u8 mode)
{
	u8 temp,sizex,t,m=0;
	u16 i,TypefaceNum;//一个字符所占字节大小
	u16 x0=x;
	sizex=sizey/2;
	TypefaceNum=(sizex/8+((sizex%8)?1:0))*sizey;
	DMA1_MEM_LEN=2*sizex;
	num=num-' ';    //得到偏移后的值
	LCD_Address_Set(x,y,x+sizex-1,y+sizey-1);  //设置光标位置 	

	for(i=0;i<TypefaceNum;i++)
		{ 
			if(sizey==12)temp=ascii_1206[num][i];		       //调用6x12字体
			else if(sizey==16)temp=ascii_1608[num][i];		 //调用8x16字体
			else if(sizey==24)temp=ascii_2412[num][i];		 //调用12x24字体
			else if(sizey==32)temp=ascii_3216[num][i];		 //调用16x32字体
			else return;
			for(t=0;t<8;t++)
			{
				if(!mode)//非叠加模式
				{
					if(temp&(0x01<<t))
						{
							SendBuff[2*m] = fc>>8;
							SendBuff[2*m+1] = fc;
						}
					else
						{
							SendBuff[2*m] = bc>>8;
							SendBuff[2*m+1] = bc;
						}
						
					m++;
					if(m%sizex==0)
						{     
							DMA_Start(); //启动传输
							m=0;
							break;
						}
				}
				else//叠加模式
				{
					if(temp&(0x01<<t))LCD_Fill(x,y,1,1,fc);//以填充方式画点
					x++;
					if((x-x0)==sizex)
					{	
						x=x0;
						y++;
						break;
					}
				}				
		 }			
	 }
}

void LCD_ShowPicture(u16 x,u16 y,u16 length,u16 width,const u8 pic[])
{
	u16 i,j;
	u32 k=0;
	DMA1_MEM_LEN=2*width;
	LCD_Address_Set(x,y,x+length-1,y+width-1);
	for(i=0;i<length;i++)
	{	
		for(j=0;j<width;j++)
		{
			SendBuff[2*j+1]=pic[k];
			SendBuff[2*j]=pic[k+1];
			k+=2;	
		}
		DMA_Start();	//启动DMA		
	}		
}

到了这里,关于STM32 硬件SPI+DMA实现快速刷TFT屏的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • U8g2库的STM32硬件SPI(DMA)移植教程(HAL、OLED显示、四线SPI、DMA)

    本文教你把U8g2图形库移植到STM32上,基于STM32的硬件SPI、CubeMX U8g2库Github网址:https://github.com/olikraus/u8g2 U8g2库CSDN镜像网址:https://gitcode.net/mirrors/olikraus/u8g2?utm_source=csdn_github_accelerator 硬件准备:STM32C8T6(STM32系列芯片)、0.96寸OLED(128×64)、J-Link(或其他) 引脚连接: 出自此

    2024年02月09日
    浏览(25)
  • 【正点原子STM32】LTDC LCD(RGB屏)实验(TFT LCD、RGB LCD信号线、驱动模式、LTDC驱动LCD-TFT显示器、LTDC控制器、DMA2D-GPU、DMA2D颜色填充步骤)

    一、RGB LCD简介 二、LTDC介绍 2.1、LTDC简介 2.2、LTDC控制器框图介绍 2.3、LTDC相关寄存器介绍 2.4、LTDC相关HAL库驱动介绍 三、RGB屏基本驱动步骤 四、编程实战1 五、DMA2D介绍 5.1、DMA2D简介 5.2、DMA2D框图介绍 5.3、DMA2D相关寄存器介绍 六、DMA2D颜色填充的具体步骤 七、编程实战2 八、

    2024年04月16日
    浏览(26)
  • STM32使用软件SPI协议操作TFT18彩屏

    时间记录:2024/2/20 (1)SPI设备通过4根线进行通信,CS片选线,选择从设备,SCK时钟线,由主设备产生时钟,主机MOSI线连从机MISO线,由主机向从机发送信息,主机MISO线连接从机MOSI线,由从机向主机发送信息 (2)SPI分为4种通信方式 模式0:CPOL=0,CPHA=0,空闲时SCK时钟线为低

    2024年02月21日
    浏览(13)
  • STM32H5开发(6)----SPI驱动TFT-LCD屏

    在嵌入式领域,TFT-LCD屏是最常用的显示解决方案之一,因为它们提供了丰富的颜色和高分辨率的图像显示能力。STM32H5作为ST的高性能微控制器系列,具备了强大的处理能力和多种通信接口,非常适合于驱动TFT-LCD显示屏,该液晶屏st7796或者ILI9488驱动芯片,这两个屏幕都是兼容

    2024年02月05日
    浏览(21)
  • STM32CubeMX配置HAL库实现SPI-DMA的递归调用

    核心: STM32F407ZET6   外设ADC: ADS1258   数量:3个   ※ 核心与3个ADC使用SPI总线 “一主多从” 方式连接,PCB布线的方式与下图一致。 ※ 在电路板上STM32与三个ADS1258在同一直线上分布,STM32在一端,三个ADC依次排布。 ※ 离STM32最远ADC的DRDY硬件管脚与STM32的EXTI line4 interrupt连接

    2024年02月15日
    浏览(22)
  • 讲解STM32驱动WS281x灯珠的多种实现方式:普通IO、SPI+DMA、以及PWM+DMA驱动方法

    STM32作为一款高性能、功能丰富的单片机,其丰富的外设和强大的性能,使其在嵌入式领域得到了广泛的应用。本篇文章将介绍如何利用STM32驱动WS281x系列的LED灯珠。我们会使用三种不同的驱动方式进行实现:一种是普通IO方式驱动,一种是SPI+DMA方式驱动,最后一种是PWM+DMA方

    2024年02月11日
    浏览(15)
  • STM32使用SPI+DMA(标准库)

    DMA DMA原理 DMA通道资源分配 SPI+DMA配置 主函数编写如下 1.CPU配置好DMA。 2.SPI发出DMA请求。(在DMA_Mode_Normal模式下,该请求实际上需要CPU命令SPI发出请求) 3.若该通道有多个请求,DMA控制器通过仲裁器判断,根据配置的优先级,选择先回应该通道高优先级的请求,再回应低优先

    2024年02月11日
    浏览(24)
  • STM32-DMA(软件出发、硬件触发)

    直接存储器存取 (DMA)用来提供在 外设 和 存储器 之间 或者 存储器和存储器 之间的 高速数据传输 。 无须CPU干预,数据可以通过DMA快速地移动,这就节省了CPU的资源来做其他操作 。 两个DMA控制器有12个通道 (DMA1有7个通道,DMA2有5个通道),每个通道专门用来管理来自于一个或

    2024年04月28日
    浏览(8)
  • [HAL]STM32 SPI+DMA驱动WS2812

    该程序是纯手敲,非Cube生成!所有代码均注释。 源码在文章后面获取 Keyword: 单线通讯、归零码、Reset、RGB顺序 RGB一共有24bit位 -相当于驱动一个灯要24bit位 -驱动若干个灯要24* n bit位,通过Reset码决定数据终止(保持) 24bit位应该如何发送? 可见: 表示低电平需要 T0H和T0L的配

    2024年02月09日
    浏览(20)
  • DMA技术在STM32中优化UART、SPI和I2C通信性能的研究与实现

    DMA(Direct Memory Access,直接存储器访问)技术可以在STM32微控制器上优化UART、SPI和I2C等通信性能。 DMA可以实现数据的高速传输,减轻CPU的负担,提高系统性能。在本篇文章中,我将探讨DMA技术在STM32中优化这些通信协议的研究和实现。 一、DMA工作原理 DMA可以实现外设与存储器

    2024年01月20日
    浏览(21)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包