TM1637 是一种带键盘扫描接口的LED(发光二极管显示器)驱动控制专用电路,内部集成有MCU 数
字接口、数据锁存器、LED 高压驱动、键盘扫描等电路。芯片手册已上传到资源,需要的可以下载,链接https://download.csdn.net/download/wanglong3713/40836173。
使用的显示模块在某宝、某多价格很便宜,4位数码管,带时间点,适合做电子时钟,另外还有不带时间点的,使用的芯片是GN1637,还有AIP1637,实际与TM1637通用,驱动程序也可以通用。
本例采用STM32F103C8T6芯片,在IAR环境下编写,配置采用STM32CubeMX,使用了HAL库。
先看数据手册的接口说明,如上图,看这描述,这是IIC啊,因此找来之前写过的IIC驱动函数移植,但是实际在使用的时候发现给TM1637发送命令时并无响应,后来仔细看手册中的下面的内容:
我们知道,标准的IIC协议是从高位到低位传输的,即MSB方式,而TM1637实际是非标准的IIC协议,而且这个器件连地址都没有,不能多个TM1637同时使用同一条总线!
所以,IIC驱动需要修改,为了方便移植,采用条件编译的方式:
/* 说明:标准IIC协议传输数据时为MSB方式,即高位在前低位在后,但有些器件为LSB方式, 即低位在前,高位在后,如TM1637数码管驱动芯片。 */ #define IIC_LSB//定义了则IIC在数据传输时低位在前 /******************************************************************************* * 函数名:IIC_Start * 功 能:起始信号 * 参 数:无 * 返回值:无 * 说 明:无 *******************************************************************************/ void IIC_Start(void) { IIC_SdaModeOut(); IIC_SdaOutput_H(); IIC_SclOutput_H(); delay_us(5);//>4.7us IIC_SdaOutput_L(); delay_us(4);//>4us IIC_SclOutput_L(); } /******************************************************************************* * 函数名:IIC_Stop * 功 能:结束信号 * 参 数:无 * 返回值:无 * 说 明:无 *******************************************************************************/ void IIC_Stop(void) { IIC_SdaModeOut(); IIC_SclOutput_L(); IIC_SdaOutput_L(); IIC_SclOutput_H(); delay_us(5);//>4us IIC_SdaOutput_H(); delay_us(4);//>4.7us IIC_SdaOutput_L(); } /******************************************************************************* * 函数名:IIC_Ack * 功 能:应答信号 * 参 数:无 * 返回值:无 * 说 明:无 *******************************************************************************/ void IIC_Ack(void) { IIC_SdaModeOut(); IIC_SclOutput_L(); IIC_SdaOutput_L(); IIC_SclOutput_H(); delay_us(4);//>4us IIC_SclOutput_L(); } /******************************************************************************* * 函数名:IIC_NoAck * 功 能:非应答信号 * 参 数:无 * 返回值:无 * 说 明:无 *******************************************************************************/ void IIC_NoAck(void) { IIC_SdaModeOut(); IIC_SclOutput_L(); IIC_SdaOutput_H(); IIC_SclOutput_H(); delay_us(4);//>4us IIC_SclOutput_L(); } /******************************************************************************* * 函数名:IIC_WaitAck * 功 能:等待应答信号 * 参 数:无 * 返回值:0应答成功,1应答失败 * 说 明:从机把总线拉低,为应答成功 *******************************************************************************/ uint8_t IIC_WaitAck(void) { uint8_t u8ErrCnt = 0; IIC_SdaModeIn();//输入状态 IIC_SdaOutput_H(); IIC_SclOutput_H(); while (IIC_SdaRead() == 1) { u8ErrCnt++; if (u8ErrCnt > 250) { IIC_Stop();//发送停止信号 return 1; } } IIC_SclOutput_L(); return 0; } /******************************************************************************* * 函数名:IIC_WriteByte * 功 能:SDA线上输出一个字节 * 参 数:u8Data需要写入的数据 * 返回值:无 * 说 明:无 *******************************************************************************/ void IIC_WriteByte(uint8_t u8Data) { uint8_t i; uint8_t u8Temp; IIC_SdaModeOut(); IIC_SclOutput_L(); for (i = 0; i < 8; i++) { delay_us(2); #ifdef IIC_LSB//低位在前 u8Temp = ((u8Data << (7 - i)) & 0x80); (u8Temp == 0x80) ? (IIC_SdaOutput_H()) : (IIC_SdaOutput_L()); #else//高位在前 u8Temp = ((u8Data >> (7 - i)) & 0x01); (u8Temp == 0x01) ? (IIC_SdaOutput_H()) : (IIC_SdaOutput_L()); #endif IIC_SclOutput_H();//时钟保持高电平 delay_us(2); IIC_SclOutput_L();//时钟拉低,才允许SDA变化 } } /******************************************************************************* * 函数名:IIC_ReadByte * 功 能:读一个字节 * 参 数:无 * 返回值:读出的数据 * 说 明:无 *******************************************************************************/ uint8_t IIC_ReadByte(void) { uint8_t i; uint8_t bit = 0; uint8_t data = 0; IIC_SdaModeIn();//输入状态 for (i = 0; i < 8; i++) { IIC_SclOutput_L(); delay_us(2); IIC_SclOutput_H(); bit = IIC_SdaRead();//读出1位 #ifdef IIC_LSB//低位在前 data |= (bit << i); #else//高位在前 data = (data << 1) | bit; #endif delay_us(2); } return data; }
以上代码中的delay_us()函数,如果使用的是STM32F103单片机,可参考通用定时器实现STM32单片机微秒级延时函数
其中的SCL/SDA引脚的操作,采用宏定义:
#define IIC_SdaModeOut() Port_SetMode(GPIOB, GPIO_PIN_7, GPIO_MODE_OUTPUT_OD) #define IIC_SdaModeIn() Port_SetMode(GPIOB, GPIO_PIN_7, GPIO_MODE_INPUT) #define IIC_SdaOutput_H() HAL_GPIO_WritePin(GPIOB, GPIO_PIN_7, GPIO_PIN_SET) #define IIC_SdaOutput_L() HAL_GPIO_WritePin(GPIOB, GPIO_PIN_7, GPIO_PIN_RESET) #define IIC_SdaRead() HAL_GPIO_ReadPin(GPIOB, GPIO_PIN_7) #define IIC_SclModeOut() Port_SetMode(GPIOB, GPIO_PIN_6, GPIO_MODE_OUTPUT_OD) #define IIC_SclModeIn() Port_SetMode(GPIOB, GPIO_PIN_6, GPIO_MODE_INPUT) #define IIC_SclOutput_H() HAL_GPIO_WritePin(GPIOB, GPIO_PIN_6, GPIO_PIN_SET) #define IIC_SclOutput_L() HAL_GPIO_WritePin(GPIOB, GPIO_PIN_6, GPIO_PIN_RESET)
Port_SetMode()函数:
/******************************************************************************* * 函数名:Port_SetMode * 功 能:GPIO设置输入或输出模式 * 参 数:*GPIOx 引脚组号 GPIO_Pin引脚号 u32Mode输入或输出模式 * 返回值:无 * 说 明:无 *******************************************************************************/ void Port_SetMode(GPIO_TypeDef *GPIOx, uint16_t GPIO_Pin, uint32_t u32Mode) { GPIO_InitTypeDef GPIO_InitStruct = {0}; GPIO_InitStruct.Pin = GPIO_Pin; GPIO_InitStruct.Mode = u32Mode; //GPIO_InitStruct.Pull = GPIO_NOPULL; GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_HIGH; HAL_GPIO_Init(GPIOx, &GPIO_InitStruct); }
根据手册,写SRAM数据时,我们采用固定地址的方式,这样可以方便地对任意一个数码管写入数据,
再构造写命令、写数据、设置亮度、开关等功能的函数:
TM1637.h文件:
/******************************************************************************* * 文件:TM1637.h * 作者:https://blog.csdn.net/wanglong3713 * 版本:v1.0 * 日期:2021-11-2 * 说明:TM1637驱动 *******************************************************************************/ #ifndef _TM1637_H_ #define _TM1637_H_ #include "Typedefine.h" #define TUBE_DISPLAY_NULL 26//不显示 #define TUBE_DISPLAY_DECIMAL_PIONT_OFFSET 16//带小数点的偏移量 /******************************************************************************* Typedefine *******************************************************************************/ typedef struct { uint8_t tube0; uint8_t tube1; uint8_t tube2; uint8_t tube3; }TM1637Tube_ts; /******************************************************************************* Global Functions *******************************************************************************/ void TM1637_WriteCmd(uint8_t u8Cmd); void TM1637_WriteData(uint8_t u8Addr, uint8_t u8Data); void TM1637_TubeDisplay(TM1637Tube_ts sData); void TM1637_SetBrightness(uint8_t u8Brt); void TM1637_Switch(bool bState); #endif
TM637.c文件:
/******************************************************************************* * 文件:TM1637.c * 作者:https://blog.csdn.net/wanglong3713 * 版本:v1.0 * 日期:2021-11-2 * 说明:TM1637驱动 *******************************************************************************/ #include "IIC.h" #include "TM1637.h" //段码表 const uint8_t u8NumTab[] = { //0, 1, 2, 3, 4, 5, 6, 7, 8, 9, A, b, C, d, E, F, 0x3F,0x06,0x5B,0x4F,0x66,0x6D,0x7D,0x07,0x7F,0x6F,0x77,0x7C,0x39,0x5E,0x79,0x71, //0., 1., 2., 3., 4., 5., 6., 7., 8., 9. Null 0xBF,0x86,0xDB,0xCF,0xE6,0xED,0xFD,0x87,0xFF,0xEF,0x00 }; //最左至最右数码管 ,依次为0-3号,对应的显示寄存器地址 const uint8_t u8TubeAddrTab[] = { 0xC0,0xC1,0xC2,0xC3 }; /******************************************************************************* * 函数名:TM1637_WriteCmd * 功 能:写命令 * 参 数:无 * 返回值:无 * 说 明:无 *******************************************************************************/ void TM1637_WriteCmd(uint8_t u8Cmd) { IIC_Start(); IIC_WriteByte(u8Cmd); IIC_Ack(); IIC_Stop(); } /******************************************************************************* * 函数名:TM1637_WriteData * 功 能:向地址中写入数据 * 参 数:u8Addr地址,u8Data数据 * 返回值:无 * 说 明:用于数码管固定地址写入显示数据 *******************************************************************************/ void TM1637_WriteData(uint8_t u8Addr, uint8_t u8Data) { IIC_Start(); IIC_WriteByte(u8Addr); IIC_Ack(); IIC_WriteByte(u8Data); IIC_Ack(); IIC_Stop(); } /******************************************************************************* * 函数名:TM1637_TubeDisplay * 功 能:4个数码管显示 * 参 数:sData显示数据结构体 * 返回值:无 * 说 明:无 *******************************************************************************/ void TM1637_TubeDisplay(TM1637Tube_ts sData) { uint8_t temp[4], i; temp[0] = u8NumTab[sData.tube0]; temp[1] = u8NumTab[sData.tube1]; temp[2] = u8NumTab[sData.tube2]; temp[3] = u8NumTab[sData.tube3]; for (i = 0; i < 4; i++) { TM1637_WriteData(u8TubeAddrTab[i], temp[i]); } } /******************************************************************************* * 函数名:TM1637_SetBrightness * 功 能:设置亮度 * 参 数:u8Brt亮度 * 返回值:无 * 说 明:0x88为开显示 *******************************************************************************/ void TM1637_SetBrightness(uint8_t u8Brt) { TM1637_WriteCmd(0x88 | u8Brt); } /******************************************************************************* * 函数名:TM1637_Switch * 功 能:显示开关 * 参 数:0关,1开 * 返回值:无 * 说 明:0x88为开显示,0x80关显示 *******************************************************************************/ void TM1637_Switch(bool bState) { bState ? TM1637_WriteCmd(0x88) : TM1637_WriteCmd(0x80); }
应用层函数,用TM1637.h中的类型TM1637Tube_ts定义一个结构体,用来存放4个数码管的显示数据:
static TM1637Tube_ts sDisplayData; /******************************************************************************* * 函数名:Display_Init * 功 能:初始化 * 参 数:无 * 返回值:无 * 说 明:无 *******************************************************************************/ void Display_Init(void) { TM1637_Switch(0);//关显示 TM1637_SetBrightness(0x87);//设置亮度,开显示 TM1637_WriteCmd(0x44);//写数据到寄存器,固定地址模式 memset(&sDisplayData, 0xFF, sizeof(sDisplayData)); }
例如显示数据1234,则运行以下函数即可:
/******************************************************************************* * 函数名:Display_TubeDataProcess * 功 能:显示数据处理 * 参 数:无 * 返回值:无 * 说 明:4位数码管,根据十进制数据位数,不需要的不显示 *******************************************************************************/ void Display_TubeDataProcess(void) { uint16_t u16Data = 1234;//需要显示的数据,自定义,或者从其他接口函数获得,本例程直接赋值为1234 memset(&sDisplayData, 0xFF, sizeof(sDisplayData)); if (u16Data > 9999) { u16Data = 9999;//最多四位数 } if (u16Data > 999)//四位数 { sDisplayData.tube0 = (uint8_t)(u16Data / 1000);//千位 sDisplayData.tube1 = (uint8_t)(u16Data / 100 % 10);//百位 sDisplayData.tube2 = (uint8_t)(u16Data % 100 / 10);//十位 sDisplayData.tube3 = (uint8_t)(u16Data % 10);//个位 }else if (u16Data > 99)//三位数 { sDisplayData.tube0 = TUBE_DISPLAY_NULL;//不显示 sDisplayData.tube1 = (uint8_t)(u16Data / 100);//百位 sDisplayData.tube2 = (uint8_t)(u16Data / 10 % 10);//十位 sDisplayData.tube3 = (uint8_t)(u16Data % 10);//个位 }else if (u16Data > 9)//两位数 { sDisplayData.tube0 = TUBE_DISPLAY_NULL;//不显示 sDisplayData.tube1 = TUBE_DISPLAY_NULL;//不显示 sDisplayData.tube2 = (uint8_t)(u16Data / 10);//十位 sDisplayData.tube3 = (uint8_t)(u16Data % 10);//个位 }else//一位数 { sDisplayData.tube0 = TUBE_DISPLAY_NULL;//不显示 sDisplayData.tube1 = TUBE_DISPLAY_NULL;//不显示 sDisplayData.tube2 = TUBE_DISPLAY_NULL;//不显示 sDisplayData.tube3 = (uint8_t)u16Data;//个位 } TM1637_TubeDisplay(sDisplayData); }
显示效果如图: