STM32CubeMX学习笔记17--- FSMC

1.1 TFTLCD简介

        TFT-LCD(thin film transistor-liquid crystal display)即薄膜晶体管液晶显示器。液晶显示屏的每一个像素上都设置有一个薄膜晶体管(TFT),每个像素都可以通过点脉冲直接控制,因而每个节点都相对独立,并可以连续控制,不仅提高了显示屏的反应速度,同时可以精确控制显示色阶,所以TFT液晶的色彩更真,因此TFT-LCD也被叫做真彩液晶显示器 常用的TFT液晶屏接口有8位、9位、16位、18位,这里的位数表示的是彩屏数据线的数量。常用的通信模式有6800模式和8080模式,本例程使用8080并口模式(简称80并口),8080接口有5条基本的控制线和多条数据线(8/9/16/18位),它们的功能如下表:

        8080接口模式的时序如下图所示:

  • 在WR跳变为低电平后,液晶屏开始读取总线上面的数据
  • 在RD跳变为低电平后,液晶屏开始放置数据到总线上面

        本例程使用3.0寸的TFT-LCD模块,驱动芯片为R61509VN,分辨率240*400,接口为16位的80并口,自带触摸功能,该模块原理图如下所示:

该模块的80并口有如下信号线

信号线作用
CSTFTLCD片选信号
WR向TFTLCD写入数据
RD从TFTLCD读取数据
RS命令/数据选择(0:读写命令,1:读写数据)
DB[15:0]16位双向数据线
RSTTFTLCD复位

任何LCD的使用流程都可以简单的用以下流程图来表示,其中硬复位和初始化序列只需要执行一次即可。画点流程:设置坐标 --> 写GRAM指令 --> 写入颜色数据,然后在LCD上就可以看到对应的点显示写入的颜色了;读点流程:设置坐标 --> 读GRAM指令 --> 读取颜色数据,这样就可以获取到对应点的颜色数据了

以上是最简单也是最常用的操作,有了这些操作一般就可以正常使用TFTLCD了。设置TFT液晶显示通常需要以下几个步骤:

- 设置STM32F1与TFTLCD模块箱连接的IO口,将与TFTLCD模块相连的IO口进行初始化,以便驱动LCD,这里使用的是STM32F1的FSMC
- 初始化TFTLCD模块(写入一系列设置值),本例程中LCD模块的复位引脚是接在STM32F1的复位上,所以直接按下开发板复位键即可;然后就是初始化序列,即向LCD控制器写入一系列的设置值(比如RGB格式、LCD显示方向、伽马校准等),这部分代码一般LCD厂商会提供。初始化之后LCD才可以正常使用
- 通过函数将字符和数字显示到TFTLCD模块上,画点流程只是一个点的处理,因此需要设计一个函数来多次使用这个步骤以实现字符和数字的显示

1.2 FSMC简介

        FSMC(Flexible Static Memory Controller即灵活的静态存储控制器)是STM32系列采用的一种新型存储器扩展技术,能够连接同步、异步存储器和16位PC存储卡,STM32通过FSMC可以与SRAM、ROM、PSRAM、NOR Flash和NAND Flash等存储器的引脚直接相连。STM32F1的FSMC内部框图如下图示:

- 时钟输入:FSMC的时钟来自时钟控制器HCLK
- AHB接口:CPU和其他AHB总线主设备可通过该AHB从设备接口访问外部静态存储器
- 外部设备:FSMC将外部设备分为2类,NOR/PSRAM设备和NAND/PC卡设备。它们共用地址数据总线等信号,但具有不同的CS以区分不同的设备

        本例程中使用的是FSMC的NOR/PSRAM存储器控制器部分,即把TFTLCD当成SRAM设备使用。NOR/PSRAM存储器控制器的接口信号功能如下图示:

        从上图中可以看出外部SRAM的控制一般有:A0 ~ A25、D0 ~ D15、NWE、NOE、NE[x],如果支持字节控制还有UB/LB信号。而TFTLCD的信号线包括:RS、DB0 ~ DB15、WR、RD、CS、RST等。由此可见它们的操作接口信号完全类似,唯一不同就是TFTLCD有RS信号没有地址信号

        TFTLCD通过RS信号来决定传送的数据是数据还是命令,可以理解为一个地址信号。比如我们把 RS 接在A0上面,当FSMC控制器写地址0的时候,会使得A0变为0,对TFTLCD来说就是写命令;而FSMC写地址1的时候,A0将会变为1,对TFTLCD来说就是写数据了。这样就把数据和命令区分开了,其实就是对应SRAM操作的两个连续地址。 当然RS也可以接在其他地址线上,本例程中RS是连接在 A10 上面的

        下面介绍一下FSMC的外部设备地址映射,从FSMC的角度外部存储器被划分为4个固定大小的存储区域(Bank),每个存储区域的大小为256MB,共1GB空间,详见下图:

        Bank1可连接多达4个NOR Flash或PSRAM存储器器件,此存储区域被划分为4个NOR/PSRAM区域,带4个专用片选信号;Bank2和Bank3用于连接NAND Flash器件;Bank4用于连接PC卡设备;对于每个存储区域,所要使用的存储器类型有用户在配置寄存器中定义。FSMC各Bank配置寄存器如下图示:

        这里使用的是Bank1,Bank1又被分为4个区,每个区管理64MB空间,每个区都有独立的寄存器对所连接的存储器进行配置。Bank1的256MB空间由28根地址线(HADDR[27:0])寻址,HADDR是内部AHB地址总线,其中HADDR[25:0]来自外部存储器地址FSMC_A[25:0],HADDR[27:26]用于对4个区寻址:

        本例程使用的是Bank1的第4区,即起始地址为0x6C000000,这里需要特别注意HADDR[25:0],HADDR[25:0]包含外部存储器地址,由于HADDR为字节地址,而存储器按字寻址,所以根据存储器数据宽度不同,实际向存储器发送的地址也有所不同,如下图示:

        不论外部接8位/16位宽设备,FSMC_A[0]始终接在外部设备地址A[0]。本例程中TFTLCD使用的是16位数据宽度,所以HADDR[0]并没有用到,只有HADDR[25:1]是有效的,对应关系变为:HADDR[25:1] --> FSMC_A[24:0],相当于右移了一位。比如地址0x7E对应的二进制是01111110,此时FSMC_A6是0而不是1,因为要右移一位。另外HADDR[27:26]是不需要我们干预的。

        FSMC的NOR Flash控制器支持同步和异步突发两种访问方式,选用不同的时序模型时,需要设置不同的时序参数:

  • 同步突发访问方式:需要设置CLKDIV(分频系数)、DATLAT(获得第一个数据所需要的等待延迟)
  • 异步突发访问方式:需要设置DATAST(数据建立时间)、ADDSET(地址建立时间)、ADDHLD(地址保持时间)

        在实际扩展时根据选用存储器的特征确定时序模型,从而确定个时间参数与存储器读写周期参数指标之间的计算关系;利用该计算关系和存储芯片数据手册中给定的参数指标,可计算出FSMC所需要的各时间参数,从而对时间参数寄存器进行合理的配置

        这里使用异步模式A(ModeA)方式来控制TFTLCD,模式A的读操作时序图如下所示:

        模式A的写操作时序图如下所示:

2. 硬件设计

LED2指示灯用来提示系统运行状态,TFTLCD的ID通过串口1打印输出

  • LED2指示灯
  • USART1
  • FSMC
  • TFTLCD模块

3、 STM32CubeMX设置

  • RCC设置外接HSE,时钟设置为72M
  • PE5设置为GPIO推挽输出模式、上拉、高速、默认输出电平为高电平
  • USART1选择为异步通讯方式,波特率设置为115200Bits/s,传输数据长度为8Bit,无奇偶校验,1位停止位

FSMC参数配置

(1)、选择FSMC,选中Bank1,片选为NE4(即Bank1的第4区),存储器类型选为LCD Interface,LCD寄存器选择(RS)设置为A10,数据宽度设为16位

在 Connectivity 中选择 FSMC 设置,并在 NOR Flash/PSRAM/SRAM/ROM/LCD 1 中选择 NE1 Chip Select 片选选择原理图中的片选引脚NE1【选择Bank1的第一区,是根据原理图的映射管脚进行选择的,这里选择不同区对应的引脚是不同的】

FSMC_NE 是用于控制存储器芯片的片选控制信号线,STM32 具有 FSMC_NE1/2/3/4 号引脚,不同的引脚对应 STM32 内部不同的地址区域。例如,当 STM32 访问 0x68000000-0x6BFFFFFF 地址空间时,FSMC_NE3 引脚会自动设置为低电平,由于它一般连接到外部存储器的片选引脚且低电平有效,所以外部存储器的片选被使能,而访问 0x60000000-
0x63FFFFFF 地址时,FSMC_NE1 会输出低电平。当使用不同的 FSMC_NE 引脚连接外部存储器时,STM32 访问外部存储的地址不一样,从而达到控制多个外部存储器芯片的目的。

  • Memory type(设置要控制的存储器类型): 选择 LCD Interface LCD接口
  • LCD Register Select(RS引脚): 选择 A16 ,RS脚也就是命令/数据选择位,同样是根据原理图得知这里应该选择A10

             Data(设置要控制的存储器的数据宽度): 选择 16 bits 很明显从原理图看出有16个数据引脚,这里选择16bits就好

          

(2)、在配置菜单中,使能存储器写,根据LCD驱动芯片的数据参数设置地址建立时间、数据保存时间、总线周转阶段持续时间(设置为0),模式设置为A

NOR/PSRAM control:

  • Write operation(设置是否写使能): 选择 Enabled ,禁止写使能的话 FSMC 只能从存储器中读取数据,不能写入。
  • Extended mode(设置是否使用扩展模式): 选择 Enabled ,在非扩展模式下,对存储器读写的时序都只使用 FSMC_BCR 寄存器中的配;在扩展模式下,对存储器的读写时序可以分开配置,读时序使用 FSMC_BCR 寄存器,写时序使用 FSMC_BWTR 寄存器的配置。
FSMC读时序配置

这里引入一个基本概念:

HCLK周期:
按STM32F103的默认配置,HCLK的时钟频率为72MHz,即一个 THCLK 为 1/72us=0.0138us=13.8us。

NOR/PSRAM timing(FSMC读时序配置):

  • Address setup time in HCLK clock cycles(地址建立时间): 填 0
  • Data setup time in HCLK clock cycles(数据建立时间): 填 26

 根据ILI9341时序配置FSMC读时序

 NEx片选后,NOE要保持一段时间的高电平,这个时间就是ADDSET地址建立时间(通过寄存器FMC_BTRx可配置)
之后NOE变为低电平,读使能。低电平保持的时间由DATAST数据建立时间(通过寄存器FMC_BTRx可配置)决定。

tast:

tast 表示地址建立时间,最小为0ns
由时序图可以知道,FSMC在ADDSET周期之后,进入DATAST周期之后将会进行数据采样。

所以我们设置(ADDSET)HCLK的时间要大于等于tast地址建立时间。
(ADDSET)HCLK >= 0ns,(0)·13.8 = 0ns,所以ADDSET可以设置为0就可保证满足最小tast地址建立时间

trdlfm:
trdlfm 表示读取数据低电平的时间,最小为355ns
ILI9341时序图没有给出ILI9341操作数据线传输被读取的数据时的相关信息,,我们最好做到满足其读取数据低电平的最小时间。
当然不做到也行,影响不大,只要在FSMC在DATAST的这个周期的数据采样中获取所有的要访问的数据就行。

(DATAST)HCLK >355ns,(26) * 13.8 = 358.8ns>355ns,所以DATAST设置为26。

  • Bus turn around time in HCLK clock cycles(总线转换周期): 填 0 ,仅适用于总线复用模式的NOR Flash操作,所以这里设0。
  • Access mode(存储器访问模式): 选 A ,LCD控制器使用 Mode A ,该模式用来控制SRAM/PSRAM且OE会翻转。控制异步 NOR FLASH 时使用 B 模式。

FSMC写时序配置

NOR/PSRAM timing for write accesses(FSMC写时序配置):

  • Extended address setup time(地址建立时间): 填 0
  • Extended data setup time(数据建立时间): 填 1

根据ILI9341时序配置FSMC写时序

NEx片选后,NWE要保持一段时间的高电平,这个时间就是ADDSET地址建立时间(通过寄存器FMC_BTRx可配置)。
之后NWE变为低电平,写使能。低电平保持的时间由DATAST数据建立时间(通过寄存器FMC_BTRx可配置)决定。

tast:

tast 表示地址建立时间,最小为0ns
由时序图可以知道,FSMC在ADDSET周期之后,进入DATAST周期之后将会进行数据采样。

  • 所以我们设置(ADDSET)HCLK的时间要大于等于tast地址建立时间。
    (ADDSET)HCLK >= 0ns,(0)·13.8 = 0ns,所以ADDSET可以设置为0就可保证满足最小tast地址建立时间。

  • tdst、tdht:

tdst:数据设置时间最小是10ns,在这个周期内WRX线处于低电平。
tdht:数据保持时间,与 twrh写控制高电平的最小时间相同,是10ns,在这个周期内WRX线处于高电平。

观察时序图,我们设置 tdst数据设置时间 为1HCLK(13.8>10)就能满足数据设置最小时间的要求,我们不需要考虑tdht数据保持时间(看上面模式B时序图,NWE变成高电平后,会持续1HCLK=13.8ns,默认满足tdht了)。

故我们只需考虑数据建立周期 DATAST 要大于10ns就行。
(DATAST)HCLK > 10ns,13.8>10 ,故DATAST 至少设置为1

  • Extended bus turn around time(总线转换周期): 填 0
  • Extended access mode(存储器访问模式): 选 A
(3)、设置背光和复位引脚

根据原理图,复位脚已经跟单片机的复位脚连接在一起了,因此只需要设置背光引脚BL_EN就行了。

 将PB0设置为GPIO推挽输出模式、上拉、高速、默认输出电平为低电平,重命名为BL_EN。

(4)、输入工程名,选择工程路径(不要有中文),选择MDK-ARM V5;勾选Generated periphera initialization as a pair of ‘.c/.h’ files per IP ;点击GENERATE CODE,生成工程代码

4、 MDK-ARM编程

  • 在fsmc.c文件下可以看到FSMC的初始化函数
void MX_FSMC_Init(void){
  FSMC_NORSRAM_TimingTypeDef Timing = {0};
  FSMC_NORSRAM_TimingTypeDef ExtTiming = {0};
  hsram1.Instance = FSMC_NORSRAM_DEVICE;
  hsram1.Extended = FSMC_NORSRAM_EXTENDED_DEVICE;
  /* hsram1.Init */
  hsram1.Init.NSBank = FSMC_NORSRAM_BANK4;
  hsram1.Init.DataAddressMux = FSMC_DATA_ADDRESS_MUX_DISABLE;
  hsram1.Init.MemoryType = FSMC_MEMORY_TYPE_SRAM;
  hsram1.Init.MemoryDataWidth = FSMC_NORSRAM_MEM_BUS_WIDTH_16;
  hsram1.Init.BurstAccessMode = FSMC_BURST_ACCESS_MODE_DISABLE;
  hsram1.Init.WaitSignalPolarity = FSMC_WAIT_SIGNAL_POLARITY_LOW;
  hsram1.Init.WrapMode = FSMC_WRAP_MODE_DISABLE;
  hsram1.Init.WaitSignalActive = FSMC_WAIT_TIMING_BEFORE_WS;
  hsram1.Init.WriteOperation = FSMC_WRITE_OPERATION_ENABLE;
  hsram1.Init.WaitSignal = FSMC_WAIT_SIGNAL_DISABLE;
  hsram1.Init.ExtendedMode = FSMC_EXTENDED_MODE_ENABLE;
  hsram1.Init.AsynchronousWait = FSMC_ASYNCHRONOUS_WAIT_DISABLE;
  hsram1.Init.WriteBurst = FSMC_WRITE_BURST_DISABLE;
  /* Timing */
  Timing.AddressSetupTime = 0x01;   //地址建立时间(ADDSET)为2个HCLK,即2/72M = 27.8ns
  Timing.AddressHoldTime = 15;  //默认值,地址保持时间(ADDHLD)模式A未用到
  Timing.DataSetupTime = 0x0f;  //数据建立时间(DATST)为16个HCLK,即16/72M = 222.2ns
  Timing.BusTurnAroundDuration = 0x00;  //总线恢复时间
  Timing.CLKDivision = 16;  //默认值(时钟分频),同步突发访问方式才需要设置此值
  Timing.DataLatency = 17;  //默认值(数据保持时间),同步突发访问方式才需要设置此值
  Timing.AccessMode = FSMC_ACCESS_MODE_A;
  /* ExtTiming */
  ExtTiming.AddressSetupTime = 0x0f;//地址建立时间(ADDSET)为16个HCLK,即16/72M = 222.2ns
  ExtTiming.AddressHoldTime = 15;   //默认值,地址保持时间(ADDHLD)模式A未用到
  ExtTiming.DataSetupTime = 0x05;   //数据建立时间(DATST)为6个HCLK,即6/72M = 83.3ns
  ExtTiming.BusTurnAroundDuration = 0x00;   //总线恢复时间
  ExtTiming.CLKDivision = 16;   //默认值(时钟分频),同步突发访问方式才需要设置此值
  ExtTiming.DataLatency = 17;   //默认值(数据保持时间),同步突发访问方式才需要设置此值
  ExtTiming.AccessMode = FSMC_ACCESS_MODE_A;
  if (HAL_SRAM_Init(&hsram1, &Timing, &ExtTiming) != HAL_OK){
    Error_Handler( );
  }
  __HAL_AFIO_FSMCNADV_DISCONNECTED();
}
  • 创建TFTLCD驱动文件tftlcd.c 和相关头文件tftlcd.h 本例程使用FSMC存储器1的第4区驱动LCD,TFTLCD的RS接在FSMC_A10上,CS接在FSMC_NE4上,并且是16位数据总线,可以在tftlcd.h里定义如下LCD操作结构体:
//TFTLCD操作结构体
typedef struct{
    uint16_t LCD_CMD;
    uint16_t LCD_DATA;
}TFTLCD_TypeDef;
//HADDR[27,26]=11,A10作为数据命令区分线
//设置时STM32内部会右移以为对齐
#define TFTLCD_BASE     ((uint32_t)(0x6C000000 | 0x000007FE))
#define TFTLCD          ((TFTLCD_TypeDef *) TFTLCD_BASE)
//0x6C000000是Bank1.sector4的起始地址;0x000007FE是A10的偏移量
//0x7FE即11111111110,16位数据宽度时地址需要右移一位对齐
//因此实际对应到引脚时A10:A0=01111111111,此时A10是0
//如果16位地址再加1,则A10:A0=10000000000,此时A10是1
//有了以上定义,LCD写命令/数据极为方便
TFTLCD->LCD_CMD = CMD;      //写命令
TFTLCD->LCD_DATA = DATA;    //写数据   
//读的时候反过来操作即可
CMD = TFTLCD->LCD_CMD       //读LCD寄存器
DATA = TFTLCD->LCD_DATA     //读LCD数据
tftlcd.h中另一个重要结构体:管理LCD重要参数
//LCD重要参数集
typedef struct{                                         
    uint16_t width;     //LCD宽度
    uint16_t height;    //LCD高度
    uint16_t id;        //LCD ID
    uint8_t  dir;       //LCD显示方向
}_tftlcd_data;
extern _tftlcd_data tftlcd_data;    //管理LCD重要参数
几个简单但是很重要的函数:
#define TFTLCD_R61509VN     //本例程的TFTLCD屏驱动类型为R61509VN
//写寄存器函数,cmd:寄存器地址
void LCD_WriteCmd(uint16_t cmd){
    TFTLCD->LCD_CMD=(cmd>>8)<<1;
    TFTLCD->LCD_CMD=(cmd&0xff)<<1;  //写入要写的寄存器序号
}
//写LCD数据,data:要写入的值
void LCD_WriteData(uint16_t data){
    TFTLCD->LCD_DATA=(data>>8)<<1;
    TFTLCD->LCD_DATA=(data&0xff)<<1;    
}
//写寄存器,cmd:寄存器地址;data:要写入的数据
void LCD_WriteCmdData(uint16_t cmd,uint16_t data){
    LCD_WriteCmd(cmd);
    LCD_WriteData(data);
}
//读LCD数据,返回读到的值
uint16_t LCD_ReadData(void){    
    uint16_t ram;       
    ram=(TFTLCD->LCD_DATA)>>1;
    ram=ram<<8;
    ram|=(TFTLCD->LCD_DATA)>>1; 
    return ram; 
}
链接: 百度网盘 请输入提取码 提取码:tvil
链接: https://pan.baidu.com/s/1Zb8qI-yZ795eyjGyXpgocQ?pwd=fnro 提取码:fnro
void LCD_Display_Dir(uint8_t dir);  //设置LCD显示方向
void LCD_Set_Window(uint16_t sx,uint16_t sy,uint16_t width,uint16_t height);    //设置窗口
void LCD_Clear(uint16_t Color);     //清屏                    
void LCD_Fill(uint16_t xState,uint16_t yState,uint16_t xEnd,uint16_t yEnd,uint16_t color);  //在指定区域内填充颜色
void LCD_DrawPoint(uint16_t x,uint16_t y);  //画点
void LCD_DrawFRONT_COLOR(uint16_t x,uint16_t y,uint16_t color); //指定颜色画点
uint16_t LCD_ReadPoint(uint16_t x,uint16_t y);  //读取某点的颜色
void LCD_DrawLine(uint16_t x1, uint16_t y1, uint16_t x2, uint16_t y2);  //画线
void LCD_DrawLine_Color(uint16_t x1, uint16_t y1, uint16_t x2, uint16_t y2,uint16_t color); //指定颜色画线
void LCD_DrowSign(uint16_t x, uint16_t y, uint16_t color);  //画十字标记
void LCD_DrawRectangle(uint16_t x1, uint16_t y1, uint16_t x2, uint16_t y2); //画矩形
void LCD_Draw_Circle(uint16_t x0,uint16_t y0,uint8_t r);    //指定位置画指定大小的圆
void LCD_ShowChar(uint16_t x,uint16_t y,uint8_t num,uint8_t size,uint8_t mode); //在指定位置显示一个字符
void LCD_ShowNum(uint16_t x,uint16_t y,uint32_t num,uint8_t len,uint8_t size);  //显示数字,不显示高位的0
void LCD_ShowxNum(uint16_t x,uint16_t y,uint32_t num,uint8_t len,uint8_t size,uint8_t mode);    //显示数字,显示高位的0
void LCD_ShowString(uint16_t x,uint16_t y,uint16_t width,uint16_t height,uint8_t size,uint8_t *p);  //显示字符串
void LCD_ShowFontHZ(uint16_t x, uint16_t y, uint8_t *cn);   //显示汉字                                                                                                                      
void LCD_ShowPicture(uint16_t x, uint16_t y, uint16_t wide, uint16_t high,uint8_t *pic);    //显示图片
  • 在main.c文件下编写TFTLCD测试代码
int main(void){
  HAL_Init();
  SystemClock_Config();
  MX_GPIO_Init();
  MX_FSMC_Init();
  MX_USART1_UART_Init();
  /* USER CODE BEGIN 2 */
  TFTLCD_Init();    
  FRONT_COLOR=BLACK;
  LCD_ShowString(10,10,240,400,12,(uint8_t*)"TFTLCD Testing...!");
  FRONT_COLOR=RED;
  LCD_ShowString(10,30,240,400,16,(uint8_t*)"TFTLCD Testing...!");
  FRONT_COLOR=GREEN;
  LCD_ShowString(10,50,240,400,24,(uint8_t*)"TFTLCD Testing...!");  
  /* USER CODE END 2 */
  while (1){
    HAL_GPIO_TogglePin(GPIOE,GPIO_PIN_5);
    HAL_Delay(200);
  }
}

5. 下载验证

STM32CubeMX生成的代码默认优化级别为Level 3,在此优化级别下编译无误下载到开发板后,可以看到D1指示灯不断闪烁,但是LCD屏不能被点亮;将优化级别调整到Level 0编译下载后,LCD屏能够被点亮

KEIL5中C/C++优化等级介绍: -O0:最少的优化,可以最大程度上配合产生代码调试信息,可以在任何代码行打断点,特别是死代码处。 -O1:有限的优化,去除无用的inline和无用的static函数、死代码消除等,在影响到调试信息的地方均不进行优化。在适当的代码体积和充分的调试之间平衡,代码编写阶段最常用的优化等级。 -O2:高度优化,调试信息不友好,有可能会修改代码和函数调用执行流程,自动对函数进行内联等。 -O3:最大程度优化,产生极少量的调试信息。会进行更多代码优化,例如循环展开,更激进的函数内联等。

编译无误下载到开发板后,可以看到D1指示灯不断闪烁,LCD屏显示相应的文字,串口打印出LCD的ID信息

6、参考文献

 STM32CubeMX学习笔记(38)——FSMC接口使用(TFT-LCD屏显示)_cubemx配置lcd时复位引脚-CSDN博客

STM32CubeMX系列 | TFTLCD显示 - 知乎 (zhihu.com)

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/439837.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

JavaScript 二分查找(迭代与递归)

二分搜索被定义为一种在排序数组中使用的搜索算法&#xff0c;通过重复将搜索间隔一分为二。二分查找的思想是利用数组已排序的信息&#xff0c;将时间复杂度降低到O(log N)。 二分查找算法示例 何时在数据结构中应用二分查找的条件&#xff1a; 应用二分查找算法&#xff1a…

Langchain-Chatchat本地搭建ChatGLM3模型和提取PDF内容

文章目录 1、软件要求2、安装CUDA2.1、安装gcc2.2、安装CUDA 3、安装Anaconda33.1、下载Anaconda33.2、创建python虚拟环境 4、部署系统4.1、下载源码4.2、安装依赖4.3、下载模型4.4、初始化配置和知识库4.4.1、初始化配置4.4.2、初始化知识库 4.5、运行4.6、运行4.6.1、启动4.…

Python学习笔记-Flask实现简单的抽奖程序

1.导入flask包和randint包 from flask import Flask,render_template from random import randint 2.初始化 Flask 应用: app Flask(__name__) 3. 定义英雄列表 hero [黑暗之女,狂战士,正义巨像,卡牌大师,德邦总管,无畏战车,诡术妖姬,猩红收割者,远古恐惧,正义天使,无极剑…

Clickhouse表引擎介绍

作者&#xff1a;俊达 1 引擎分类 ClickHouse表引擎一共分为四个系列&#xff0c;分别是Log、MergeTree、Integration、Special。其中包含了两种特殊的表引擎Replicated、Distributed&#xff0c;功能上与其他表引擎正交&#xff0c;根据场景组合使用。 2 Log系列 Log系列…

高阶提示词工程、幻觉综述

提示词工程技术 类比提示 “类比提示”利用类比推理的概念&#xff0c;鼓励模型生成自己的例子和知识&#xff0c;从而实现更灵活和高效的解决问题。 后退提示 “后退提示”专注于抽象&#xff0c;引导模型推导出高级概念和原理&#xff0c;进而提高其推理能力。 使用一个…

mysql学习笔记8——常用5个内置方法

1count 对查询内容进行计数&#xff0c;并返回结果 2as as可以将查询出来结果赋予新名字 3sum sum可以查询某字段特定条件下的和值 4concat concat可以将多列数据合并成一列&#xff0c;只要concat&#xff08;&#xff09;即可 5group_concat group_concat可以把多列…

SpringBoot+Ajax+redis实现隐藏重要接口地址

&#x1f3e1;浩泽学编程&#xff1a;个人主页 &#x1f525; 推荐专栏&#xff1a;《深入浅出SpringBoot》《java对AI的调用开发》 《RabbitMQ》《Spring》《SpringMVC》《项目实战》 &#x1f6f8;学无止境&#xff0c;不骄不躁&#xff0c;知行合一 文章目录 …

预付费电表的应用和预付费平台的操作方式

*、智能预付费电能表的应用分析 1应用功能的分析 这里主要讲的是与远程抄表系统的结合&#xff0e;如图2所示&#xff0e;为系统工作的程序&#xff0e;在远程抄表中&#xff0c;通信方式多种多样&#xff0e;主要有互联网、电话线通信、有线电视通信、光纤通信、GPRS、卫星通…

关于esp8266的一些经验汇总,新手必看

说实话&#xff0c;esp8266的nodemcu 已经使用了2年多了&#xff0c;各种问题遇到过&#xff0c;就尝试各种解决&#xff0c;而现在回头来看真的是稀里糊涂的在用&#xff0c;当然这个问题也同样涉及到esp32. 因为最近打算自己打一块esp8266的板&#xff0c;之前打的比较多的是…

Hi3516DV500+SC2210 AIISP 黑光相机

1. Hi3516DV500 Hi3516DV500是一颗面向行业市场推出的高清智能网络摄像头SoC。该芯片最高支持2路sensor输入&#xff0c;支持最高5M30fps的ISP图像处理能力&#xff0c;支持2F WDR、多级降噪、六轴防抖、多光谱融合等多种传统图像增强和处理算法&#xff0c;支持通过AI算法对输…

ABAP 内表排序总结

目录 ABAP 内表排序总结需求的场景二分法查找SAP 二分法查找SAP SORT排序 ABAP 内表排序总结 ABAP 内表排序SORT总结&#xff1a; 在创建完内表之后&#xff0c;最好使用sort去排序一下使用read读取内表&#xff0c;如果没有排序的话&#xff0c;可能会读取失败read内表只能读…

Fortran语法介绍(一)

个人专栏—ABAQUS专栏 Abaqus2023的用法教程——与VS2022、oneAPI 2024子程序的关联方法 Abaqus2023的用法教程——与VS2022、oneAPI 2024子程序的关联方法Abaqus有限元分析——有限元网格划分基本原则 Abaqus有限元分析——有限元网格划分基本原则各向同性线弹性材料本构模型…

创维汽车SKYHOME获德国设计奖,中国红设计闪耀世界

祝贺&#xff01;创维汽车SKYHOME以卓越的国潮设计理念和突破性的设计语言强势出圈&#xff0c;荣获被誉为设计界“奥斯卡”德国iF设计奖&#xff01; 创维汽车SYHOME是一款集完美设计理念、出色用户体验及创新实用功能为一体的优秀设计产品。SKYHOME的设计灵感来源于中式亭台楼…

【掌握数学公式的魔法】LatexEasy:让你的数学写作不再是难题!

内容摘要&#xff1a;在学术和研究领域&#xff0c;数学公式的准确表达至关重要。然而&#xff0c;传统的LaTeX编辑过程往往复杂且耗时。幸运的是&#xff0c;有了LatexEasy&#xff0c;一切都变得简单起来。这款工具不仅简化了数学公式的编辑流程&#xff0c;还大大提高了工作…

【梳理】k8s使用Operator搭建Flink集群

文章目录 架构图安装cert-manager依赖helm 安装operator运行集群实例k8s上的两种模式&#xff1a;Native和Standalone两种CRDemo1&#xff1a;Application 单任务Demo2&#xff1a;Session 多任务创建ingress 总结 架构图 参考&#xff1a;部署验证demo 安装cert-manager依赖 …

面试高频 牛群的位置排序---搜索插入位置

题目描述 农场里有一群牛&#xff0c;每头牛都有一个标签值&#xff0c;这些标签值组成一个升序排列的数组 labels。现在农场主想知道&#xff0c;给定一个目标标签值 target&#xff0c;如果在牛群中存在这个标签&#xff0c;返回它的位置&#xff0c;如果不存在&#xff0c;…

NSSCTF Round#13 WEB

1.flask?jwt? 在忘记密码下面有提示secretkey,那么就可以jwt伪造 自己注册个账号然后登录 点击拿flag提示你不是admin&#xff0c;并且cookie里面有个session,用工具解密一下 python flask_session_cookie_manager3.py decode -s th3f1askisfunny -c .eJwlzjsOAyEMANG7UK…

JavaScript实现小球移动(二)

这次采用了封装函数的方法&#xff0c;将小球向左向右移动封装在同一个函数内。 代码&#xff1a; <!DOCTYPE html> <html lang"en"><head><meta charset"UTF-8"><meta name"viewport" content"widthdevice-wi…

高效办公-浏览器基本操作

日常我们使用电脑&#xff0c;其实很大部分是用于网络功能&#xff0c;这里面除了客户端程序剩余的就是通过我们的浏览器获取信息或者使用业务系统了&#xff0c;这里就简单学习下浏览器基本常识与操作。 一、浏览器是什么&#xff1f; 白话讲浏览器就是一个软件&#xff0c;我…

springboot3.x集成nacos踩坑,并实现多环境配置

一、nacos安装部署 springboot3.x集成Nacos首先需要将Nacos从1.x升级到2.x&#xff0c;建议直接安装2.x版本&#xff0c;手动将1.x的配置信息迁移到2.x中&#xff0c;先并行一段时间&#xff0c;待全部迁移完成稳定运行之后再停掉1.x&#xff0c;升级和安装、操作请查看官方文…