工业物联网LCD数码屏的驱动原理及低功耗设计(华大半导体HC32L136)
目录
在工业物联网传感器可视化设计时,仅仅为显示传感器的数值变化,多选用低成本、低功耗、尺寸合适的LCD数码屏,本次博客为各位分享华大半导体HC32L136驱动LCD数码屏的实现方法以及低功耗设计。

1、驱动原理
LCD数码屏本质上就是数码管,因为其主要是为了显示传感器数据,多为若干个7段数码管(7个亮段和1个小数点组成
 )组成,7个亮段实际上就是7个条形的发光二极管,按顺时针方向,这7个亮段分别为a、b、c、d、e、f、g大多数七段数码管还带有一个小数点位dp。如下图所示:

7段数码管中亮段的发光原理和普通的发光二极管是一样的,所以可以把这7个亮段看成7个发光二极管,根据内部7个发光二极管的共连端不同,可将七段数码管分为共阳(共阳极)和共阴(共阴极)两种。共阳极就是把所有LED的阳极(正极)连接到
 共同接点COM,而每个LED的阴极分别为a、b、c、d、e、f、g及dp (小数点) ;共阴极则是把所有LED的阴极(负极)连接到共同接点COM,而每个LED的阳极分别为a、b、c、d、e、f、g及dp(小数点),通过控制各个LED的亮灭来显示数字。如下图所示:

7段数码管有多种颜色、多种尺寸供设计时使用,它们的显示原理相同。如果要7段数码管显示数字1,只要点亮b、c两段即可;如要显示数字5,则需要点亮a、f、g、c、d段。其他数字和一些字母可以按照下图中的说明点亮对应的亮段来显示,7个亮段可以灵活地表现数字和一些字母信息。

在实际应用中,从节约端口数量、降低成本等角度考虑,LCD数码屏中的多个数码管并联,采用动态扫描的方式,一位一位地轮流点亮各位显示器(扫描),对于显示器的每一位而言,每隔一段时间点亮一次。虽然在同一时刻只有一位显示器在工作(点亮),但利用人眼的视觉暂留效应和发光二极管熄灭时的余辉效应,看到的却是多个字符“同时”显示。显示器亮度既与点亮时的导通电流有关,也与点亮时间和间隔时间的比例有关,调整电流和时间参,可实现亮度较高较稳定的显示。
2、驱动程序
最近在研究国产华大半导体的MCU,本次将基于HC32L136实现LCD数码屏的驱动程序设计,这里我选用的是自定制LCD数码屏,驱动原理和市面上的LCD屏一致,如下图所示:

该LCD数码屏有4个公共端,29个端口,在不借助驱动芯片的前提下,要保证MCU有29个富余的IO口,LCD数码屏引脚对应特性如下图所示:

华大半导体的HC32L136支持LCD 控制器可适用于单色无源液晶显示器(LCD)的数字控制器/驱动器,最多具有 8 个公用端子(COM)和 40 个区段端子(SEG),用以驱动 160 (4x40)或 288 (8x36)个 LCD 图像元素。可以选择电容分压或电阻分压,支持内部电阻分压,内部电阻分压可以调节对比度,支持 DMA 硬件数据传输,明显足够我使用了,特性如下所示:
- 高度灵活的帧速率控制。
 - 支持静态、1/2、1/3、1/4、1/6 和 1/8 占空比。
 - 支持 1/2、1/3 偏置。
 - 多达 16 个寄存器的 LCD 数据 RAM。
 - 可通过软件配置 LCD 的对比度。
 - 3 种驱动波形生成方式:内部电阻分压、外部电阻分压,外部电容分压方式,可通过软件配置内部电阻分压方式的功耗,从而匹配 LCD 面板所需的电容电荷。
 - 支持低功耗模式:LCD 控制器可在 Active、Sleep、DeepSleep 模式下进行显示。
 - 可配置帧中断。
 - 支持 LCD 闪烁功能且可配置多种闪烁频率
 - 未使用的 LCD 区段和公共引脚可配置为数字或模拟功能。
 
LCD 控制器框架图如下所示:

了解了 LCD数码屏的特性后就要开始设计程序了~
基于HC32L136 LCD 控制器需要完成基本的配置,使用相关的API即可快速配置好扫描模式、驱动波形、帧速率等信息。
第1步:使能RCL时钟、配置内部低速时钟频率为32.768kHz、开启LCD时钟和GPIO时钟,配置代码如下所示:
  
   - 
    
     
    
    
     
       Sysctrl_ClkSourceEnable(SysctrlClkRCL,TRUE); ///< 使能RCL时钟
     
    
 
   - 
    
     
    
    
     
       Sysctrl_SetRCLTrim(SysctrlRclFreq32768); ///< 配置内部低速时钟频率为32.768kHz
     
    
 
   - 
    
     
    
    
      
     
    
 
   - 
    
     
    
    
     
       Sysctrl_SetPeripheralGate(SysctrlPeripheralLcd,TRUE);   ///< 开启LCD时钟
     
    
 
   - 
    
     
    
    
     
       Sysctrl_SetPeripheralGate(SysctrlPeripheralGpio,TRUE);  ///< 开启GPIO时钟
     
    
 
  
 
第2步:LCD端口配置,因为是基于LCD 控制器,所以使用特定GPIO无法自定义,可参阅HC32L136管脚功能查询及配置表进行了解。

配置代码如下所示:
  
   - 
    
     
    
    
     
       ******************************************************************************
     
    
 
   - 
    
     
    
    
     
       ** \brief  初始化外部GPIO引脚
     
    
 
   - 
    
     
    
    
     
       **
     
    
 
   - 
    
     
    
    
     
       ** \return 无
     
    
 
   - 
    
     
    
    
     
       ******************************************************************************/
     
    
 
   - 
    
     
    
    
     
      void App_PortCfg(void)
     
    
 
   - 
    
     
    
    
     
      {
     
    
 
   - 
    
     
    
    
     
       Gpio_SetAnalogMode(GpioPortA, GpioPin9);  //COM1
     
    
 
   - 
    
     
    
    
     
       Gpio_SetAnalogMode(GpioPortA, GpioPin10); //COM2
     
    
 
   - 
    
     
    
    
     
       Gpio_SetAnalogMode(GpioPortA, GpioPin11); //COM3
     
    
 
   - 
    
     
    
    
     
       Gpio_SetAnalogMode(GpioPortA, GpioPin12); //COM4 
     
    
 
   - 
    
     
    
    
      
     
    
 
   - 
    
     
    
    
     
       Gpio_SetAnalogMode(GpioPortA, GpioPin8);  //SEG0
     
    
 
   - 
    
     
    
    
     
       Gpio_SetAnalogMode(GpioPortC, GpioPin9);  //SEG1
     
    
 
   - 
    
     
    
    
     
       Gpio_SetAnalogMode(GpioPortC, GpioPin8);  //SEG2
     
    
 
   - 
    
     
    
    
     
       Gpio_SetAnalogMode(GpioPortC, GpioPin7);  //SEG3
     
    
 
   - 
    
     
    
    
     
       Gpio_SetAnalogMode(GpioPortC, GpioPin6);  //SEG4
     
    
 
   - 
    
     
    
    
     
       Gpio_SetAnalogMode(GpioPortB, GpioPin15); //SEG5
     
    
 
   - 
    
     
    
    
     
       Gpio_SetAnalogMode(GpioPortB, GpioPin14); //SEG6
     
    
 
   - 
    
     
    
    
     
       Gpio_SetAnalogMode(GpioPortB, GpioPin13); //SEG7
     
    
 
   - 
    
     
    
    
      
     
    
 
   - 
    
     
    
    
      
     
    
 
   - 
    
     
    
    
     
       Gpio_SetAnalogMode(GpioPortB, GpioPin12);  //SEG8
     
    
 
   - 
    
     
    
    
     
       Gpio_SetAnalogMode(GpioPortB, GpioPin11);  //SEG9
     
    
 
   - 
    
     
    
    
     
       Gpio_SetAnalogMode(GpioPortB, GpioPin10);  //SEG10
     
    
 
   - 
    
     
    
    
      ///< SEG11不用,凑成16位,方便计算
     
    
 
   - 
    
     
    
    
      
     
    
 
   - 
    
     
    
    
      
     
    
 
   - 
    
     
    
    
     
       Gpio_SetAnalogMode(GpioPortB, GpioPin1);   //SEG12
     
    
 
   - 
    
     
    
    
     
       Gpio_SetAnalogMode(GpioPortB, GpioPin0);   //SEG13
     
    
 
   - 
    
     
    
    
     
       Gpio_SetAnalogMode(GpioPortC, GpioPin5);   //SEG14
     
    
 
   - 
    
     
    
    
     
       Gpio_SetAnalogMode(GpioPortC, GpioPin4);   //SEG15
     
    
 
   - 
    
     
    
    
     
       Gpio_SetAnalogMode(GpioPortA, GpioPin7);   //SEG16
     
    
 
   - 
    
     
    
    
     
       Gpio_SetAnalogMode(GpioPortA, GpioPin6);   //SEG17
     
    
 
   - 
    
     
    
    
     
       Gpio_SetAnalogMode(GpioPortA, GpioPin5);   //SEG18
     
    
 
   - 
    
     
    
    
     
       Gpio_SetAnalogMode(GpioPortA, GpioPin4);   //SEG19
     
    
 
   - 
    
     
    
    
      
     
    
 
   - 
    
     
    
    
     
       Gpio_SetAnalogMode(GpioPortA, GpioPin3);  //SEG20
     
    
 
   - 
    
     
    
    
     
       Gpio_SetAnalogMode(GpioPortA, GpioPin2);  //SEG21
     
    
 
   - 
    
     
    
    
     
       Gpio_SetAnalogMode(GpioPortA, GpioPin1);  //SEG22
     
    
 
   - 
    
     
    
    
     
       Gpio_SetAnalogMode(GpioPortA, GpioPin0);  //SEG23
     
    
 
   - 
    
     
    
    
      
     
    
 
   - 
    
     
    
    
     
       Gpio_SetAnalogMode(GpioPortC, GpioPin3);  //SEG24
     
    
 
   - 
    
     
    
    
     
       Gpio_SetAnalogMode(GpioPortC, GpioPin2);  //SEG25
     
    
 
   - 
    
     
    
    
      
     
    
 
   - 
    
     
    
    
     
       Gpio_SetAnalogMode(GpioPortB, GpioPin3);  //VLCDH
     
    
 
   - 
    
     
    
    
     
       Gpio_SetAnalogMode(GpioPortB, GpioPin4);  //VLCD3
     
    
 
   - 
    
     
    
    
     
       Gpio_SetAnalogMode(GpioPortB, GpioPin5);  //VLCD2
     
    
 
   - 
    
     
    
    
     
       Gpio_SetAnalogMode(GpioPortB, GpioPin6);  //VLCD1
     
    
 
   - 
    
     
    
    
     
      }
     
    
 
  
 
第3步:配置LCD,这里我使用的是配置是:外部电容工作模式、1/4duty、1/3 BIAS、电压泵时钟频率选择2kHz、LCD扫描频率选择128Hz、LCD时钟选择RCL、选择模式0,具体配置可查阅用户手册,讲解的比较细致,代码如下所示:
  
   - 
    
     
    
    
     
      /**
     
    
 
   - 
    
     
    
    
     
       ******************************************************************************
     
    
 
   - 
    
     
    
    
     
       ** \brief 配置LCD
     
    
 
   - 
    
     
    
    
     
       **
     
    
 
   - 
    
     
    
    
     
       ** \return 无
     
    
 
   - 
    
     
    
    
     
       ******************************************************************************/
     
    
 
   - 
    
     
    
    
     
      void App_LcdCfg(void)
     
    
 
   - 
    
     
    
    
     
      {
     
    
 
   - 
    
     
    
    
      stc_lcd_cfg_t LcdInitStruct;
     
    
 
   - 
    
     
    
    
      stc_lcd_segcom_t LcdSegCom;
     
    
 
   - 
    
     
    
    
      
     
    
 
   - 
    
     
    
    
     
       LcdSegCom.u32Seg0_31 = 0xFC000800; ///< 配置LCD_POEN0寄存器 开启SEG0~SEG25,SEG12不开 11111100 00000000 00001000 00000000
     
    
 
   - 
    
     
    
    
     
       LcdSegCom.stc_seg32_51_com0_8_t.seg32_51_com0_8 = 0xffffffff;   ///< 初始化LCD_POEN1寄存器 全部关闭输出端口
     
    
 
   - 
    
     
    
    
     
       LcdSegCom.stc_seg32_51_com0_8_t.segcom_bit.Com0_3 = 0; ///< 使能COM0~COM3
     
    
 
   - 
    
     
    
    
     
       LcdSegCom.stc_seg32_51_com0_8_t.segcom_bit.Mux = 0; ///< Mux=0,Seg32_35=0,BSEL=1表示:选择外部电容工作模式,内部电阻断路
     
    
 
   - 
    
     
    
    
     
       LcdSegCom.stc_seg32_51_com0_8_t.segcom_bit.Seg32_35 = 0;
     
    
 
   - 
    
     
    
    
     
       Lcd_SetSegCom(&LcdSegCom); ///< LCD COMSEG端口配置
     
    
 
   - 
    
     
    
    
      
     
    
 
   - 
    
     
    
    
     
       LcdInitStruct.LcdBiasSrc = LcdExtCap; ///< 电容分压模式,需要外部电路配合
     
    
 
   - 
    
     
    
    
     
       LcdInitStruct.LcdDuty = LcdDuty4; ///< 1/4duty 占空比(DUTY):定义为 1/(LCD 显示器上的公用端子数)的数字
     
    
 
   - 
    
     
    
    
     
       LcdInitStruct.LcdBias = LcdBias3; ///< 1/3 BIAS 偏置(BIAS):驱动 LCD 时使用的电压等级,定义为 1/(驱动 LCD 显示的电压等级数–1)
     
    
 
   - 
    
     
    
    
     
       LcdInitStruct.LcdCpClk = LcdClk2k; ///< 电压泵时钟频率选择2kHz
     
    
 
   - 
    
     
    
    
     
       LcdInitStruct.LcdScanClk = LcdClk128hz; ///< LCD扫描频率选择128Hz
     
    
 
   - 
    
     
    
    
     
       LcdInitStruct.LcdMode = LcdMode0; ///< 选择模式0
     
    
 
   - 
    
     
    
    
     
       LcdInitStruct.LcdClkSrc = LcdRCL; ///< LCD时钟选择RCL
     
    
 
   - 
    
     
    
    
     
       LcdInitStruct.LcdEn   = LcdEnable; ///< 使能LCD模块
     
    
 
   - 
    
     
    
    
     
       Lcd_Init(&LcdInitStruct);
     
    
 
   - 
    
     
    
    
     
      }
     
    
 
  
 
第4步:建立LCD驱动GPIO和LCD数码屏中数码管之间的驱动关系,以LCD数码屏中左上角4个数码管为例建立(之后的数码管显示规律有差异)。

十六进制如下表所示:
| 数字 | 通断控制位 | A | B | C | DP | 通断控制位 | F | G | E | D | LCD十六进制 | ||||||
| 0 | 默认为0 | 1 | 1 | 1 | 0 | 默认为0 | 1 | 0 | 1 | 1 | 0x0E0B | ||||||
| 1 | 0 | 1 | 1 | 0 | 0 | 0 | 0 | 0 | 0x0600 | ||||||||
| 2 | 1 | 1 | 0 | 0 | 0 | 1 | 1 | 1 | 0x0C07 | ||||||||
| 3 | 1 | 1 | 1 | 0 | 0 | 1 | 0 | 1 | 0x0E05 | ||||||||
| 4 | 0 | 1 | 1 | 0 | 1 | 1 | 0 | 0 | 0x060C | ||||||||
| 5 | 1 | 0 | 1 | 0 | 1 | 1 | 0 | 1 | 0x0A0D | ||||||||
| 6 | 1 | 0 | 1 | 0 | 1 | 1 | 1 | 1 | 0x0A0F | ||||||||
| 7 | 1 | 1 | 1 | 0 | 0 | 0 | 0 | 0 | 0x0E00 | ||||||||
| 8 | 1 | 1 | 1 | 0 | 1 | 1 | 1 | 1 | 0x0E0F | ||||||||
| 9 | 1 | 1 | 1 | 0 | 1 | 1 | 0 | 0 | 0x0E0C | ||||||||
| DP | 0 | 0 | 0 | 1 | 0 | 0 | 0 | 0 | 0x0100 | ||||||||
| 高8位 | 低8位 | 高8位 | 低8位 | ||||||||||||||
根据以上关系建立关系数组,该部分实现代码如下所示:
  
   - 
    
     
    
    
     
      void Lcd_Drive(int8_t id,int16_t num1,int16_t num2,int8_t point)
     
    
 
   - 
    
     
    
    
     
      {
     
    
 
   - 
    
     
    
    
      ///< LCD数字0~9和DP/COL1
     
    
 
   - 
    
     
    
    
       uint16_t Numerical_Tables1[11]={0x0E0B,0x0600,0x0C07,0x0E05,0x060C,0x0A0D,0x0A0F,0x0E00,0x0E0F,0x0E0C,0x0100}; 
     
    
 
   - 
    
     
    
    
      
     
    
 
   - 
    
     
    
    
       uint32_t Num_Collect=0;
     
    
 
   - 
    
     
    
    
      
     
    
 
   - 
    
     
    
    
       ///< 屏幕第1、2(id==0)、3、4(id==1)小数字
     
    
 
   - 
    
     
    
    
       if(id==0 || id==1)
     
    
 
   - 
    
     
    
    
     
        {
     
    
 
   - 
    
     
    
    
      if(num1>=0 && num1<=9)
     
    
 
   - 
    
     
    
    
     
       {
     
    
 
   - 
    
     
    
    
     
       Num_Collect=Numerical_Tables1[num1]; ///< 第一个数字
     
    
 
   - 
    
     
    
    
     
       }
     
    
 
   - 
    
     
    
    
      else if(num1==-1)
     
    
 
   - 
    
     
    
    
     
       {
     
    
 
   - 
    
     
    
    
     
       Num_Collect=0x0000; ///< 熄灭
     
    
 
   - 
    
     
    
    
     
       }
     
    
 
   - 
    
     
    
    
      
     
    
 
   - 
    
     
    
    
      if(num2>=0 && num2<=9)
     
    
 
   - 
    
     
    
    
     
       {
     
    
 
   - 
    
     
    
    
     
       Num_Collect|=Numerical_Tables1[num2]<<16;  ///< 第二个数字
     
    
 
   - 
    
     
    
    
     
       }
     
    
 
   - 
    
     
    
    
      else if(num1==-1)
     
    
 
   - 
    
     
    
    
     
       {
     
    
 
   - 
    
     
    
    
     
       Num_Collect|=0x0000<<16; ///< 熄灭
     
    
 
   - 
    
     
    
    
     
       }
     
    
 
   - 
    
     
    
    
      
     
    
 
   - 
    
     
    
    
      if(point==1)
     
    
 
   - 
    
     
    
    
     
       {
     
    
 
   - 
    
     
    
    
     
       Num_Collect|=Numerical_Tables1[10]; ///< 第一个DP
     
    
 
   - 
    
     
    
    
     
       }
     
    
 
   - 
    
     
    
    
      else if(point==2)
     
    
 
   - 
    
     
    
    
     
       {
     
    
 
   - 
    
     
    
    
     
       Num_Collect|=Numerical_Tables1[10]<<16; ///< 第二个DP/COL1
     
    
 
   - 
    
     
    
    
     
       }
     
    
 
   - 
    
     
    
    
      else if(point==3)
     
    
 
   - 
    
     
    
    
     
       {
     
    
 
   - 
    
     
    
    
     
       Num_Collect|=Numerical_Tables1[10]; ///< 第一个DP
     
    
 
   - 
    
     
    
    
     
       Num_Collect|=Numerical_Tables1[10]<<16; ///< 第二个DP/COL1
     
    
 
   - 
    
     
    
    
     
       }
     
    
 
   - 
    
     
    
    
      
     
    
 
   - 
    
     
    
    
     
       Lcd_WriteRam(id,Num_Collect);
     
    
 
   - 
    
     
    
    
      
     
    
 
   - 
    
     
    
    
     
       Num_Collect=0;
     
    
 
   - 
    
     
    
    
     
        }
     
    
 
   - 
    
     
    
    
     
      }
     
    
 
  
 
至此就可以实现LCD数码屏幕的驱动。
3、低功耗设计
HC32L136进入深度休眠状态,不会改变端口状态,在进入休眠前根据需要更改 IO 的状态为休眠下的状态,所以在深度休眠状态下LCD数码屏可以继续显示工作。
运行上述程序,LCD数码屏所有数码管工作点亮耗能约890毫安左右,如下图所示:

使用低功耗设计后,耗能约为2.6微安,极大降低了功耗,如下图所示:

低功耗设计实现代码如下所示:
  
   - 
    
     
    
    
     
      void App_LowPowerMode(void)
     
    
 
   - 
    
     
    
    
     
      {
     
    
 
   - 
    
     
    
    
      ///< 打开GPIO外设时钟门控
     
    
 
   - 
    
     
    
    
     
       Sysctrl_SetPeripheralGate(SysctrlPeripheralGpio, TRUE);
     
    
 
   - 
    
     
    
    
      
     
    
 
   - 
    
     
    
    
      //swd as gpio
     
    
 
   - 
    
     
    
    
     
       Sysctrl_SetFunc(SysctrlSWDUseIOEn, TRUE);
     
    
 
   - 
    
     
    
    
      
     
    
 
   - 
    
     
    
    
      ///< 配置为数字端口
     
    
 
   - 
    
     
    
    
     
       M0P_GPIO->PAADS = ~0xE000;
     
    
 
   - 
    
     
    
    
     
       M0P_GPIO->PBADS = ~0x0384;
     
    
 
   - 
    
     
    
    
     
       M0P_GPIO->PCADS = ~0xFC03;
     
    
 
   - 
    
     
    
    
     
       M0P_GPIO->PDADS = ~0xFFEF;
     
    
 
   - 
    
     
    
    
      
     
    
 
   - 
    
     
    
    
      ///< 配置为端口输入
     
    
 
   - 
    
     
    
    
     
       M0P_GPIO->PADIR = 0XFFFF;
     
    
 
   - 
    
     
    
    
     
       M0P_GPIO->PBDIR = 0XFFFF;
     
    
 
   - 
    
     
    
    
     
       M0P_GPIO->PCDIR = 0XFFFF;
     
    
 
   - 
    
     
    
    
     
       M0P_GPIO->PDDIR = 0XFFFF;
     
    
 
   - 
    
     
    
    
      
     
    
 
   - 
    
     
    
    
      ///< 输入下拉(除LCD端口以外)
     
    
 
   - 
    
     
    
    
     
       M0P_GPIO->PAPD = 0xE000;
     
    
 
   - 
    
     
    
    
     
       M0P_GPIO->PBPD = 0x0384;
     
    
 
   - 
    
     
    
    
     
       M0P_GPIO->PCPD = 0xFC03;
     
    
 
   - 
    
     
    
    
     
       M0P_GPIO->PDPD = 0xFFEF;
     
    
 
   - 
    
     
    
    
      
     
    
 
   - 
    
     
    
    
     
       Lpm_GotoDeepSleep(TRUE);
     
    
 
   - 
    
     
    
    
     
      }
     
    
 
  
 
文章来源: handsome-man.blog.csdn.net,作者:不脱发的程序猿,版权归原作者所有,如需转载,请联系作者。
原文链接:handsome-man.blog.csdn.net/article/details/104591679
- 点赞
 - 收藏
 - 关注作者
 
            
           
评论(0)