【GD32F303红枫派使用手册】第二十六节 EXMC-液晶驱动实验

news2025/1/11 20:41:29

26.1 实验内容

通过本实验主要学习以下内容:

  • LCD显示原理
  • EXMC NOR/SRAM模式时序和8080并口时序
  • LCD显示控制

26.2 实验原理

使用MCU的EXMC外设实现8080并口时序,和TFT-LCD控制器进行通信,控制LCD显示图片、字符、色块等。

26.2.1 TFT-LCD介绍

薄膜晶体管液晶显示器(英语:Thin film transistor liquid crystal display,常简称为TFT-LCD)是多数液晶显示器的一种,它使用薄膜晶体管技术改善影象品质。虽然TFT-LCD被统称为LCD,不过它是种主动式矩阵LCD,被应用在电视、平面显示器及投影机上。

简单说,TFT-LCD面板可视为两片玻璃基板中间夹着一层液晶,上层的玻璃基板是彩色滤光片、而下层的玻璃则有晶体管镶嵌于上。当电流通过晶体管产生电场变化,造成液晶分子偏转,藉以改变光线的偏极性,再利用偏光片决定像素的明暗状态。此外,上层玻璃因与彩色滤光片贴合,形成每个像素各包含红蓝绿三颜色,这些发出红蓝绿色彩的像素便构成了面板上的视频画面。

为了对TFT-LCD的显示进行控制,需要通过接口和液晶屏通信,但所谓与液晶屏通信,实际上还是与液晶屏驱动控制芯片在通信,而主控制器需要按控制芯片支持的通信进行交互,通常有UART、IIC、SPI、8080、MIPI等各类接口。另外需要注意的一点是:一般支持普通MCU接口的LCD驱动芯片,都需要内置GRAM(Graphics RAM), 至少能存储一个屏幕的数据。

这这里,我们使用了8080接口通过并行总线传输控制命令和数据,并通过往LCD液晶模组自带的GRAM更新数据实现屏幕的刷新。

GD32F3红枫派开发板TFT-LCD如下图所示,采用了ILI9488 LCD驱动器,分辨率320*480,支持多种通信接口,在GD32F303xx上,适合使用16-bit Parallel MCU Interface接口进行通信,开发板配套的LCD模块也采用了该接口设计和开发板进行连接。

26.2.2 LCD 8080并口时序介绍(16-bit Parallel MCU Interface)

8080接口是由英特尔设计,是一种并行、异步、半双工通信协议,作用是用于外扩RAM、ROM,后面也用于LCD接口。并行接口又分为 8位/16位/24位 三种, 顾名思义,就是数据总线的位宽。

  • 如下图所示是16-bit Parallel MCU Interface的接口和MCU的连接信号:

  • 如下图所示是LCD驱动器 16BIT 8080并口读写时序:
  • CS拉低后,并口DATA IO在WR的上升沿被采样;
  • 可以理解为16线的SPI,而WR是写“CLK”,RD是读“CLK”;
  • 但这里还多了D/C引脚用于选择传输命令或数据

  • 8080接口的RGB颜色数据编码
  • 像素信息用RGB三原色表示,所以向液晶屏传输的数据帧主要也就是传输的RGB颜色数据。
  • 像素的颜色数据并不总是用 8R8G8B的24位真彩色 表示,共有下面几种表示情况:
  • 12-bits/pixel (R 4-bit, G 4-bit, B 4-bit), 4,096 Colors, 简称444;
  • 16-bits/pixel (R 5-bit, G 6-bit, B 5-bit), 65,536 Colors, 简称565;
  • 18-bits/pixel (R 6-bit, G 6-bit, B 6-bit), 262,144 Colors, 简称666;
  • 24-bits/pixel (R 8-bit, G 8-bit, B 8-bit), 16,777,216 Colors, 简称888;
    不同颜色表示方法和不同的总线位宽相组合,就会组合成多种RGB颜色数据编码。
  • 综合显示效果、内存资源开销等,我们采取了RGB565像素格式,这样16BIT 8080每次传输就是一个像素点的像素值,传输数据就为像素颜色值。

26.2.3 EXMC外设和EXMC NOR/SRAM模式实现8080时序

  • 这里我们使用EXMC中时序和接口类似的NOR/SRAM模式,来实现8080接口驱动TFT-LCD显示,EXMC外设可参考上文介绍东方红开发板使用手册 ,EXMC NOR/SRAM模式可参考上文介绍东方红开发板使用手册 
  • 这里我们使用SRAM模式异步模式A(扩展模式),时序如下图:

我们对比上面的时序和16-bit Data Bus 8080 LCD时序,发现一些信号的时序是类似的,我们可以将这些信号进行对应:

EXMC_NEx -> CSx

EXMC_NOE->RDx

EXMC_NWE->WR

EXMC_D[15:0]->DB[15:0]

EXMC_Ax->D/C

这里巧妙的是使用EXMC_Ax引脚实现D/C的数据/命令切换功能,所以我们只需要选择一个方便布线的EXMC_Ax引脚,然后在软件中对该引脚对应的EXMC逻辑地址进行操作就可以实现程序读写不同地址时,D/C引脚的状态切换,从而实现访问一个EXMC地址时是数据或命令类似,访问该地址位反向的任意地址就是另外一个类型。对于程序中逻辑地址的影响,除了Ax引脚的选择外还有NEx引脚的选择。NE[0]-NE[3]对应如下图的NOR/SRAM BANK下的Region0-Region3。

26.3 硬件设计

在红枫派开发板设计中,我们使用EXMC_NE1引脚作为CS,EXMC_A12引脚作为D/C,同时LCD触摸接口使用SPI,LCD_BL为背光控制引脚,这里的引脚选用了带PWM的引脚,可以实现LCD的背光亮度调节。

LCD在显示过程电源电流会有变化,为了稳定电源我们在3.3V和5V接口上使用了1uf电容。

26.4 代码解析

26.4.1 CSx、D/C、BL相关功能定义和注册;

在EXMC LCD驱动代码中存在和电路设计匹配的变更点,往往让开发者头大,需要详细阅读用户手册来进行配置调整、读写地址调整;而在我们的驱动文件bsp_lcd.c中定义注册背光引脚、Ax、NEx引脚,当硬件设计变更时只需要在这里调整,驱动就可以在新的硬件中正常使用。

C
//定义背光引脚的PWM通道和GPIO
TIMER_CH_DEF(LCD_BL,TIMER12,0,TIMER_CH_PWM_HIGH,F,8,AF_PP,GPIO_TIMER12_REMAP);

//定义使用的EXMC_Ax引脚
#define EXMC_Ax  12
GPIO_DEF(EXMC_Ax_GPIO,G,2,AF_PP,SET,NULL);

//定义使用的EXMC_NEx引脚
#define EXMC_NEx 1
GPIO_DEF(EXMC_NEx_GPIO,D,7,AF_PP,SET,NULL);

//LCD数据、命令地址转换
#define EXMC_LCD_D  REG16(((uint32_t)(EXMC_BANK0_NORSRAM_REGIONx_ADDR(EXMC_NEx)))|BIT(EXMC_Ax)*2)
#define EXMC_LCD_C  REG16(((uint32_t)(EXMC_BANK0_NORSRAM_REGIONx_ADDR(EXMC_NEx))))

26.4.2 gpio和exmc初始化:

exmc使用了扩展模式,这样读和写的时序可以单独配置,因为LCD对读和写的要求时间是不同的,读的时候速率不能太高,如果使用一种参数类型就会为了满足读的要求而降低写的速率,影响最终刷屏的性能。这里主要调整读和写时的地址建立、数据建立时间,通常和硬件设计也有较大关系,这里红枫派开发板和配套LCD的电路可以在120M主频下,设置读数据建立5个clk、地址建立1个clk,写数据建立2个clk,地址建立1个clk。

C
/*!
* 说明     emxc LCD模式通用gpio初始化
* 输入[1]  norsram_region: @EXMC_BANK0_NORSRAM_REGION0/EXMC_BANK0_NORSRAM_REGION1/EXMC_BANK0_NORSRAM_REGION2/EXMC_BANK0_NORSRAM_REGION3
* 返回值   无
*/
void driver_exmc_lcd_16bit_gpio_init(void)
{
    /* EXMC clock enable */
    rcu_periph_clock_enable(RCU_EXMC);

    /* GPIO clock enable */
    rcu_periph_clock_enable(RCU_GPIOD);
    rcu_periph_clock_enable(RCU_GPIOE);
    rcu_periph_clock_enable(RCU_GPIOG);
    rcu_periph_clock_enable(RCU_AF);    

    /* configure EXMC_D[0~15]*/
    /* PD14(EXMC_D0), PD15(EXMC_D1),PD0(EXMC_D2), PD1(EXMC_D3), PD8(EXMC_D13), PD9(EXMC_D14), PD10(EXMC_D15) */
    gpio_init(GPIOD, GPIO_MODE_AF_PP, GPIO_OSPEED_MAX, GPIO_PIN_0 | GPIO_PIN_1| GPIO_PIN_8 | GPIO_PIN_9 |
                                                         GPIO_PIN_10 | GPIO_PIN_14 | GPIO_PIN_15);

    /* PE7(EXMC_D4), PE8(EXMC_D5), PE9(EXMC_D6), PE10(EXMC_D7), PE11(EXMC_D8), PE12(EXMC_D9), 
       PE13(EXMC_D10), PE14(EXMC_D11), PE15(EXMC_D12) */
    gpio_init(GPIOE, GPIO_MODE_AF_PP, GPIO_OSPEED_MAX, GPIO_PIN_7 | GPIO_PIN_8 | GPIO_PIN_9 | 
                                                         GPIO_PIN_10 | GPIO_PIN_11 | GPIO_PIN_12 | 
                                                         GPIO_PIN_13 | GPIO_PIN_14 | GPIO_PIN_15);

//    /* configure PE2(EXMC_Ax) */ 
//    gpio_init(GPIOE, GPIO_MODE_AF_PP, GPIO_OSPEED_MAX, GPIO_PIN_2);
    
    /* configure NOE and NWE */
    gpio_init(GPIOD, GPIO_MODE_AF_PP, GPIO_OSPEED_MAX, GPIO_PIN_4 | GPIO_PIN_5);

//    /* configure EXMC NEx */
//    gpio_init(GPIOD, GPIO_MODE_AF_PP, GPIO_OSPEED_MAX, GPIO_PIN_7);

    gpio_compensation_config(GPIO_COMPENSATION_ENABLE);
    
    while(gpio_compensation_flag_get()==RESET);
}

/*!
* 说明     emxc LCD模式初始化
* 输入[1]  norsram_region: @EXMC_BANK0_NORSRAM_REGION0/EXMC_BANK0_NORSRAM_REGION1/EXMC_BANK0_NORSRAM_REGION2/EXMC_BANK0_NORSRAM_REGION3
* 返回值   无
*/
void driver_exmc_lcd_init(uint32_t norsram_region)
{
    exmc_norsram_parameter_struct nor_init_struct;
    exmc_norsram_timing_parameter_struct nor_timing_read_init_struct;
    exmc_norsram_timing_parameter_struct nor_timing_write_init_struct;

    /* EXMC clock enable */
    rcu_periph_clock_enable(RCU_EXMC);
    
    nor_init_struct.read_write_timing = &nor_timing_read_init_struct;
       
    nor_init_struct.write_timing = &nor_timing_write_init_struct;    
    
    exmc_norsram_struct_para_init(&nor_init_struct);
    
    /* config timing parameter */
    nor_timing_read_init_struct.asyn_access_mode = EXMC_ACCESS_MODE_A;
    nor_timing_read_init_struct.syn_data_latency = EXMC_DATALAT_2_CLK;
    nor_timing_read_init_struct.syn_clk_division = EXMC_SYN_CLOCK_RATIO_2_CLK;
    nor_timing_read_init_struct.bus_latency = 1;
    nor_timing_read_init_struct.asyn_data_setuptime = 5;
    nor_timing_read_init_struct.asyn_address_holdtime = 1;
    nor_timing_read_init_struct.asyn_address_setuptime = 1;
    
    /* config timing parameter */
    nor_timing_write_init_struct.asyn_access_mode = EXMC_ACCESS_MODE_A;
    nor_timing_write_init_struct.syn_data_latency = EXMC_DATALAT_2_CLK;
    nor_timing_write_init_struct.syn_clk_division = EXMC_SYN_CLOCK_RATIO_2_CLK;
    nor_timing_write_init_struct.bus_latency = 1;
    nor_timing_write_init_struct.asyn_data_setuptime = 2;
    nor_timing_write_init_struct.asyn_address_holdtime = 1;
    nor_timing_write_init_struct.asyn_address_setuptime = 1;    

    /* config EXMC bus parameters */
    nor_init_struct.norsram_region = norsram_region;
    nor_init_struct.write_mode = EXMC_ASYN_WRITE;
    nor_init_struct.extended_mode = ENABLE;
    nor_init_struct.asyn_wait = DISABLE;
    nor_init_struct.nwait_signal = DISABLE;
    nor_init_struct.memory_write = ENABLE;
    nor_init_struct.nwait_config = EXMC_NWAIT_CONFIG_BEFORE;
    nor_init_struct.wrap_burst_mode = DISABLE;
    nor_init_struct.nwait_polarity = EXMC_NWAIT_POLARITY_LOW;
    nor_init_struct.burst_mode = DISABLE;
    nor_init_struct.databus_width = EXMC_NOR_DATABUS_WIDTH_16B;
    nor_init_struct.memory_type = EXMC_MEMORY_TYPE_SRAM;
    nor_init_struct.address_data_mux = DISABLE;

    exmc_norsram_init(&nor_init_struct);

    /* enable the EXMC bank0 NORSRAM */
    exmc_norsram_enable(norsram_region);  
}

 26.4.3 LCD数据、命令读写:

LCD的命令、数据的读写都通过EXMC来实现,在读写EXMC的逻辑地址时对应的波形会发送到LCD上,命令/数据目前通过地址引脚控制,所以我们需要定义两个地址分别对应命令、数据地址,对这两个地址读和写就可以实现LCD读写数据、写命令功能。

C
/**
 * 说明       LCD写数据
 * 输入       data: 要写入的数据
 * 返回值      无
 */
void bsp_lcd_wr_data(__IO uint16_t data)
{
    //delay_sysclk(1);            
    EXMC_LCD_D = data;
}

/**
 * 说明       LCD写寄存器编号/地址函数
 * 输入       regno: 寄存器编号/地址
 * 返回值      无
 */
void bsp_lcd_wr_regno(__IO uint16_t regno)
{
    //delay_sysclk(1);        
    EXMC_LCD_C = regno;   /* 写入要写的寄存器序号 */
}

/**
 * 说明       LCD写寄存器
 * 输入       regno:寄存器编号/地址
 * 输入       data:要写入的数据
 * 返回值      无
 */
void bsp_lcd_write_reg(__IO uint16_t regno,__IO uint16_t data)
{
    //delay_sysclk(1);     
    EXMC_LCD_C = regno;   /* 写入要写的寄存器序号 */
    //delay_sysclk(1);     
    EXMC_LCD_D = data;    /* 写入数据 */
}

/**
 * 说明       LCD读数据
 * 输入       无
 * 返回值      读取到的数据
 */
static uint16_t bsp_lcd_read_data(void)
{
    //delay_sysclk(1);
    return EXMC_LCD_D;
}

26.4.4 LCD初始化

LCD初始化序列通常LCD驱动器厂家会提供相关寄存器配置,为了兼容不同的LCD,可以读取LCD ID后执行不同的驱动芯片初始化,初始化过程的寄存器配置就通过25.4.3章节实现的命令、数据读写接口实现:

C
/**
 * 说明       初始化LCD
 *   @note      该初始化函数可以初始化各种型号的LCD(详见本.c文件最前面的描述)
 *
 * 输入       无
 * 返回值      无
 */
uint32_t bsp_lcd_init(void)
{
    
    bsp_lcd_port_init();


    bsp_lcd_backlight_duty_set(BACK_LIGHT_DUTY);
    
    bsp_lcd_backlight_on();          /* 点亮背光 */    
    
    /* LCD的画笔颜色和背景色 */
    bsp_lcd_parameter.g_point_color = WHITE;    /* 画笔颜色 */
    bsp_lcd_parameter.g_back_color  = BLACK;    /* 背景色 */    
    
    delay_ms(1);     /* 初始化FSMC后,必须等待一定时间才能开始初始化 */

    /* 尝试9341 ID的读取 */
    bsp_lcd_wr_regno(0XD3);
    bsp_lcd_parameter.id = bsp_lcd_read_data();  /* dummy read */
    bsp_lcd_parameter.id = bsp_lcd_read_data();  /* 读到0X00 */
    bsp_lcd_parameter.id = bsp_lcd_read_data();  /* 读取0X93 */
    bsp_lcd_parameter.id <<= 8;
    bsp_lcd_parameter.id |= bsp_lcd_read_data(); /* 读取0X41 */


    if (bsp_lcd_parameter.id == 0X9488)
    {
        lcd_ex_ili9488_reginit();   /* 执行ILI9388初始化 */
    }     
    else if (bsp_lcd_parameter.id == 0X9341)
    {
        lcd_ex_ili9341_reginit();   /* 执行ILI9341初始化 */
    }else{
        return DRV_ERROR;
    }

    bsp_lcd_display_dir(0); /* 默认为竖屏 */
    bsp_lcd_clear(WHITE);
    
    return DRV_SUCCESS;
}

/**
 * @brief       ILI9488寄存器初始化代码
 * @param       无
 * @retval      无
 */
void lcd_ex_ili9488_reginit(void)
{
    //************* Start Initial Sequence **********//
    bsp_lcd_wr_regno(0XF7);            
    bsp_lcd_wr_data(0xA9);         
    bsp_lcd_wr_data(0x51);         
    bsp_lcd_wr_data(0x2C);         
    bsp_lcd_wr_data(0x82);

    bsp_lcd_wr_regno(0XEC);            
    bsp_lcd_wr_data(0x00);         
    bsp_lcd_wr_data(0x02);         
    bsp_lcd_wr_data(0x03);         
    bsp_lcd_wr_data(0x7A);

    bsp_lcd_wr_regno(0xC0);         
    bsp_lcd_wr_data(0x13);         
    bsp_lcd_wr_data(0x13);         
        
    bsp_lcd_wr_regno(0xC1);         
    bsp_lcd_wr_data(0x41);         
        
    bsp_lcd_wr_regno(0xC5);         
    bsp_lcd_wr_data(0x00);         
    bsp_lcd_wr_data(0x28);         
    bsp_lcd_wr_data(0x80);
        
    bsp_lcd_wr_regno(0xB1);   //Frame rate 70HZ          
    bsp_lcd_wr_data(0xB0);
    bsp_lcd_wr_data(0x11);        
        
    bsp_lcd_wr_regno(0xB4);         
    bsp_lcd_wr_data(0x02);           
        
    bsp_lcd_wr_regno(0xB6); //RGB/MCU Interface Control        
    bsp_lcd_wr_data(0x02);   //MCU        
    bsp_lcd_wr_data(0x22); 

    bsp_lcd_wr_regno(0xB7);         
    bsp_lcd_wr_data(0xc6); 

    bsp_lcd_wr_regno(0xBE);         
    bsp_lcd_wr_data(0x00); 
    bsp_lcd_wr_data(0x04);        
        
    bsp_lcd_wr_regno(0xE9);         
    bsp_lcd_wr_data(0x00);

    bsp_lcd_wr_regno(0xF4);         
    bsp_lcd_wr_data(0x00); 
    bsp_lcd_wr_data(0x00);
    bsp_lcd_wr_data(0x0f);        
        
    bsp_lcd_wr_regno(0xE0);         
    bsp_lcd_wr_data(0x00);         
    bsp_lcd_wr_data(0x04);         
    bsp_lcd_wr_data(0x0E);         
    bsp_lcd_wr_data(0x08);         
    bsp_lcd_wr_data(0x17);         
    bsp_lcd_wr_data(0x0A);         
    bsp_lcd_wr_data(0x40);         
    bsp_lcd_wr_data(0x79);         
    bsp_lcd_wr_data(0x4D);         
    bsp_lcd_wr_data(0x07);         
    bsp_lcd_wr_data(0x0E);         
    bsp_lcd_wr_data(0x0A);         
    bsp_lcd_wr_data(0x1A);         
    bsp_lcd_wr_data(0x1D);         
    bsp_lcd_wr_data(0x0F);          
        
    bsp_lcd_wr_regno(0xE1);         
    bsp_lcd_wr_data(0x00);         
    bsp_lcd_wr_data(0x1B);         
    bsp_lcd_wr_data(0x1F);         
    bsp_lcd_wr_data(0x02);         
    bsp_lcd_wr_data(0x10);         
    bsp_lcd_wr_data(0x05);         
    bsp_lcd_wr_data(0x32);         
    bsp_lcd_wr_data(0x34);         
    bsp_lcd_wr_data(0x43);         
    bsp_lcd_wr_data(0x02);         
    bsp_lcd_wr_data(0x0A);         
    bsp_lcd_wr_data(0x09);         
    bsp_lcd_wr_data(0x33);         
    bsp_lcd_wr_data(0x37);         
    bsp_lcd_wr_data(0x0F); 


    bsp_lcd_wr_regno(0xF4);      
    bsp_lcd_wr_data(0x00);
    bsp_lcd_wr_data(0x00);
    bsp_lcd_wr_data(0x0f);        
        
    bsp_lcd_wr_regno(0x36);         
    bsp_lcd_wr_data(0x08);         
        
    bsp_lcd_wr_regno(0x3A);  //Interface Mode Control        
    bsp_lcd_wr_data(0x55);  //0x66 18bit; 0x55 16bit                        
        
    bsp_lcd_wr_regno(0x20);         
    bsp_lcd_wr_regno(0x11);         
    delay_ms(120);         
    bsp_lcd_wr_regno(0x29);         
    delay_ms(50); 

} 

26.4.5 LCD画点函数实现

LCD在任意点显示想要的颜色值,需要设置显示光标到目标位置,然后就可以从该光标进行颜色数据写入,颜色信息将显示到LCD的指定坐标上。

C
/**
 * 说明       画点
 * 输入       x,y: 坐标
 * 输入       color: 点的颜色(32位颜色,方便兼容LTDC)
 * 返回值      无
 */
void bsp_lcd_draw_point(uint16_t x, uint16_t y, uint32_t color)
{
    bsp_lcd_set_cursor(x, y);       /* 设置光标位置 */
    EXMC_LCD_C = bsp_lcd_parameter.wramcmd;    /* 开始写入GRAM */
    EXMC_LCD_D = color;
}

26.4.6 窗口设置和色块填充

LCD可以设置需显示的窗口,设置窗口后可以连续的写数据,像素信息会从窗口起始坐标开始自动递增和换行显示颜色。通过图块设置函数可以显示图片,移植到GUI等;这里我们通过DMA的MEM TO MEM模式可以降低色块填充过程的CPU占用率,同时提升刷屏速率。

C
/**
 * 说明       在指定区域内填充指定颜色块
 * 输入       (sx,sy),(ex,ey):填充矩形对角坐标,区域大小为:(ex - sx + 1) * (ey - sy + 1)
 * 输入       color: 要填充的颜色数组首地址
 * 返回值      无
 */
void bsp_lcd_color_fill(uint16_t sx, uint16_t sy, uint16_t ex, uint16_t ey, uint16_t *color)
{
    uint16_t height, width;
//    uint32_t i;
    width = ex - sx + 1;            /* 得到填充的宽度 */
    height = ey - sy + 1;           /* 高度 */
    
    bsp_lcd_set_window(sx,sy,width,height);
    EXMC_LCD_C = bsp_lcd_parameter.wramcmd;
    driver_dma_mem_to_exmclcd_start((void*)&EXMC_LCD_D,&color[0],DMA_Width_16BIT,height*width);     
//    for(uin32_t i = 0; i < height*width; i++)
//    {
//        EXMC_LCD_D = color[i];     
//    }
}

26.4.7 字符显示和LCD Printf

实现上述功能后,通过字库信息结合打点函数就可以实现字符的显示,我们同时实现了在lcd上打印信息,以printf的形式更轻易便捷的输出信息到LCD上。

C
/**
 * 说明       LCD打印
 * 输入       ...和printf相同用法,自动换行
 * 返回值      无
 */
void bsp_lcd_printf(const char * sFormat, ...) 
{
    char printf_buffer[64];
    char* p=printf_buffer;
    uint16_t len=0,count=0;
    va_list ParamList;
    va_start(ParamList, sFormat);    
    vsprintf(printf_buffer,sFormat, ParamList);     
    va_end(ParamList); 

    len=strlen(printf_buffer);    

    while( ( ((*p <= '~') && (*p >= ' ')) || (*p =='\r') || (*p =='\n') ) && (count<len) )   /* 判断是不是非法字符! */
    {
        if((*p =='\r'))
        {
            bsp_lcd_pritnf_parameter.x_now = bsp_lcd_pritnf_parameter.x_start;
        }        
        else if((*p =='\n'))
        {
            bsp_lcd_pritnf_parameter.y_now += bsp_lcd_pritnf_parameter.size; 
            bsp_lcd_fill(bsp_lcd_pritnf_parameter.x_now,bsp_lcd_pritnf_parameter.y_now,bsp_lcd_pritnf_parameter.x_end,bsp_lcd_pritnf_parameter.y_now+bsp_lcd_pritnf_parameter.size,bsp_lcd_pritnf_parameter.back_color);            
        }
        else if( (bsp_lcd_pritnf_parameter.x_now + bsp_lcd_pritnf_parameter.size/2) > bsp_lcd_pritnf_parameter.x_end)
        {
            bsp_lcd_pritnf_parameter.x_now = bsp_lcd_pritnf_parameter.x_start;
            bsp_lcd_pritnf_parameter.y_now += bsp_lcd_pritnf_parameter.size;
            bsp_lcd_fill(bsp_lcd_pritnf_parameter.x_now,bsp_lcd_pritnf_parameter.y_now,bsp_lcd_pritnf_parameter.x_end,bsp_lcd_pritnf_parameter.y_now+bsp_lcd_pritnf_parameter.size,bsp_lcd_pritnf_parameter.back_color);                        
        }
        else if ( (bsp_lcd_pritnf_parameter.y_now+bsp_lcd_pritnf_parameter.size) > bsp_lcd_pritnf_parameter.y_end)
        {
            bsp_lcd_pritnf_parameter.x_now = bsp_lcd_pritnf_parameter.x_start;
            bsp_lcd_pritnf_parameter.y_now = bsp_lcd_pritnf_parameter.y_start;            
            bsp_lcd_fill(bsp_lcd_pritnf_parameter.x_start,bsp_lcd_pritnf_parameter.y_start,bsp_lcd_pritnf_parameter.x_end,bsp_lcd_pritnf_parameter.y_end,bsp_lcd_pritnf_parameter.back_color);
        }

        if((*p !='\r')&&(*p !='\n'))
        {
            bsp_lcd_show_char(bsp_lcd_pritnf_parameter.x_now, bsp_lcd_pritnf_parameter.y_now, *p, bsp_lcd_pritnf_parameter.size, 0, bsp_lcd_pritnf_parameter.point_color,bsp_lcd_pritnf_parameter.back_color);
            bsp_lcd_pritnf_parameter.x_now += bsp_lcd_pritnf_parameter.size / 2;            
        }
        

        p++;
        count++;
    }

}

26.5 实验结果

复位后显示聚沃和GD LOG图片,大字显示LCD ID,刷屏时间、聚沃相关链接地址等。在下方设置了一个printf区窗口,循环打印亮度信息和系统tic信息。左右波动摇杆可以调节LCD亮度。

由聚沃科技原创,来源于

 【红枫派开发板】第二十六讲 EXMC-液晶驱动实验 - 苏州聚沃电子科技有限公司 (gd32bbs.com)

  

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【代码随想录——动态规划——序列问题】

1.最初上升子序列 func lengthOfLIS(nums []int) int {length : len(nums)dp : make([]int, length)for i:0;i<length;i{dp[i] 1}//对于每一个i&#xff0c;我们都需要回过头去遍历是否可以更新长度for i:0;i<length;i{for j:0;j<i;j{if nums[i]>nums[j]{dp[i] m…

如何在前端项目中制定代码注释规范

本文是前端代码规范系列文章&#xff0c;将涵盖前端领域各方面规范整理&#xff0c;其他完整文章可前往主页查阅~ 开始之前&#xff0c;介绍一下​最近很火的开源技术&#xff0c;低代码。 作为一种软件开发技术逐渐进入了人们的视角里&#xff0c;它利用自身独特的优势占领市…

基于SSM+Jsp的雅博书城在线系统

开发语言&#xff1a;Java框架&#xff1a;ssm技术&#xff1a;JSPJDK版本&#xff1a;JDK1.8服务器&#xff1a;tomcat7数据库&#xff1a;mysql 5.7&#xff08;一定要5.7版本&#xff09;数据库工具&#xff1a;Navicat11开发软件&#xff1a;eclipse/myeclipse/ideaMaven包…

【Pandas驯化-17】一文搞懂Pandas如何优雅的连接mysql函数to_sql技巧

【Pandas驯化-17】一文搞懂Pandas如何优雅的连接mysql函数to_sql技巧 本次修炼方法请往下查看 &#x1f308; 欢迎莅临我的个人主页 &#x1f448;这里是我工作、学习、实践 IT领域、真诚分享 踩坑集合&#xff0c;智慧小天地&#xff01; &#x1f387; 相关内容文档获取 微…

mysql数据库中使用存储过程带来的好处和示例(存储过程的概念、定义、作用等详解)

目录 一、概述 二、存储过程的作用 1、代码重用 2、简化复杂操作 3、提高性能 4、安全性和数据完整性 三、相对于sql操纵语句&#xff08;select、insert等&#xff09;存储过程的好处 1、代码重用和封装 &#xff08;1&#xff09;概述 &#xff08;2&#xff09;举…

28. 深度学习中的损失函数:起源、分类及统一理解

在深度学习和机器学习领域&#xff0c;损失函数&#xff08;Loss Function&#xff09;是优化问题的核心&#xff0c;决定了模型参数的调整方向和幅度。尽管损失函数种类繁多&#xff0c;但理解其起源和背后的理论有助于我们更好地选择和应用它们。 损失函数的起源 所有的优化…

吴恩达揭秘:编程Agent如何革新软件开发行业

作为 AI 领域的杰出人物&#xff0c;吴恩达教授对编程 Agent 的兴起表示了极大的兴趣。他认为&#xff0c;编程 Agent 有潜力通过自动执行繁琐的任务、提高代码质量和加速开发周期来彻底改变软件开发行业。 本文将深入探讨吴恩达对编程 Agent 的见解&#xff0c; 多代理系统质…

【RF Transceiver】ADRV9040 THEORY OF OPERATION

工作原理 概述 GENERAL 该 ADRV9040 是一款高度集成的射频收发器&#xff0c;能够针对各种应用进行配置。该器件集成了在单个器件中提供所有发射器、流量接收机和观测接收机功能所需的所有射频、混合信号和数字模块。可编程性使该器件能够适应 TDD 模式下的许多 3G/4G/5G 蜂窝…

Hadoop3:Yarn工作机制

一、流程图 注意&#xff1a;步骤0中&#xff0c;如果是本地运行&#xff0c;则创建的是LocalRunner 二、流程说明 1、首先&#xff0c;我们把自己编写好的MR程序&#xff0c;上传到集群中客户端所在的节点。 2、使用shell客户端命令&#xff0c;执行jar程序&#xff0c;执行…

【线上绘图网站分享】

好用的线上绘图网站分享 使用场景特点使用例子 Excalidraw 使用场景 流程图绘制、组会分享工具等&#xff1b; 特点 最重要的就是&#xff1a;免费&#xff01;&#xff01; 简单&#xff0c;快捷&#xff1a;有时候临时要画一个流程图之类的用来示意、分享知识点&#xff…

一码搞定三种预测!!多变量回归预测+区间预测+核密度估计,LSSVM-ABKDE的多变量回归预测程序,小白上手,不会程序也能用

适用平台&#xff1a;Matlab2022版及以上 区间预测&#xff0b;概率密度估计传统的回归预测无法有效地捕捉新能源等波动的不确定性&#xff0c;很难取得审稿专家的肯定。区间预测能够提供更丰富的不确定信息&#xff0c;也极大地增加了光伏和风能预测在实际应用中的价值&#x…

2024山东大学软件学院创新项目实训(10)项目总结

项目名称&#xff1a;基于InternLM2的题库系统——考研政治助手 一、项目亮点 二、项目分工 三、成果展示 可以看到微调后最直观的首先是答案更加正确&#xff0c;第二点就是&#xff0c;微调完后&#xff0c;给出的回答格式比较鲜明。上来会告诉你选择某个选项&#xff0c;…

时延降低 50%,小红书图数据库如何实现多跳查询性能大幅提升

多跳查询为企业提供了深入的数据洞察和分析能力&#xff0c;它在小红书众多在线业务中扮演重要的角色。然而&#xff0c;这类查询往往很难满足稳定的 P99 时延要求。小红书基础架构存储团队针对这一挑战&#xff0c;基于大规模并行处理&#xff08;MPP&#xff09;的理念&#…

自然语言处理课程论文:《Attention is all you need》复现与解读

目录 1.背景介绍 1.1 文献介绍 1.2 研究背景 1.3 知识概述 1.3.1 机器翻译 1.3.2 attention机制与self-attention机制 2.数据来源与处理 2.1 数据集描述 2.2 数据处理 3. 模型架构 ​​​​​​​3.1 Positional Embedding ​​​​​​​3.2 Multi-Head Attention ​​​​​…

摄影师危!AI绘画即将降维打击摄影行业

你还以为AI绘画影响的只是插画师行业吗&#xff1f;错了&#xff0c;摄影行业也即将面临技术洗牌 话不多说&#xff0c;先看一下这几张图 你能一眼看出这是AI画的迪丽热巴吗&#xff1f; 你是不是还以为AI绘画只能画点动漫艺术风格&#xff1f;那你就低估了AI的发展速度&…

ICE构建坚不可摧的交易环境

金融技术迅速发展的今天,Intercontinental Exchange, Inc.再次站在了行业前沿,首席技术官Mark Wassersug表示:“我们的目标是通过ICE.AI,为全球交易者提供市场领先的技术解决方案。平台的推出标志着我们在利用人工智能技术优化金融市场交易方面迈出了重要一步。我们保证这将大幅…

开发RpcProvider的发布服务(NotifyService)

1.发布服务过程 目前完成了mprpc框架项目中的以上的功能。 作为rpcprovider的使用者&#xff0c;也就是rpc方法的发布方 main函数如下&#xff1a; 首先我们init调用框架的init&#xff0c;然后启动一个provider&#xff0c;然后向provider上注册服务对象方法&#xff0c;即us…

人工智能在医学图像分割中的最新研究进展|顶刊速递·24-06-26

小罗碎碎念 今日推文主题——人工智能在医学图像分割领域中的最新研究进展。 今天的推文都来自同一个期刊——《Med Image Anal 》&#xff0c;最新的IF是10.7。 小罗观点 今天的六篇文献都是直接面向实际应用场景的&#xff0c;可以针对你自己的研究课题选择合适的文章进行阅读…

uniapp标题水平对齐微信小程序胶囊按钮及适配

uniapp标题水平对齐微信小程序胶囊按钮及适配 状态栏高度胶囊按钮的信息计算顶部边距模板样式 标签加样式加动态计算实现效果 状态栏高度 获取系统信息里的状态栏高度 const statusBarHeight uni.getSystemInfoSync().statusBarHeight;//系统信息里的状态栏高度胶囊按钮的…

已解决java.security.GeneralSecurityException: 安全性相关的通用异常的正确解决方法,亲测有效!!!

已解决java.security.GeneralSecurityException: 安全性相关的通用异常的正确解决方法&#xff0c;亲测有效&#xff01;&#xff01;&#xff01; 目录 问题分析 报错原因 解决思路 解决方法 确定具体异常类型 检查输入参数 验证算法支持性 调整安全策略 确保资源可…