ICode9

精准搜索请尝试: 精确搜索
首页 > 编程语言> 文章详细

4位数码管显示模块TM1637芯片C语言驱动程序

2021-11-15 23:59:49  阅读:582  来源: 互联网

标签:驱动程序 void uint8 C语言 数码管 IIC sDisplayData TM1637 GPIO


一、概述

TM1637 是一种带键盘扫描接口的LED(发光二极管显示器)驱动控制专用电路,内部集成有MCU 数
字接口、数据锁存器、LED 高压驱动、键盘扫描等电路。芯片手册已上传到资源,需要的可以下载,链接https://download.csdn.net/download/wanglong3713/40836173
使用的显示模块在某宝、某多价格很便宜,4位数码管,带时间点,适合做电子时钟,另外还有不带时间点的,使用的芯片是GN1637,还有AIP1637,实际与TM1637通用,驱动程序也可以通用。
在这里插入图片描述
本例采用STM32F103C8T6芯片,在IAR环境下编写,配置采用STM32CubeMX,使用了HAL库。

二、 源代码

2.1 IIC接口驱动

在这里插入图片描述
先看数据手册的接口说明,如上图,看这描述,这是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);
}

2.2 TM1637驱动函数

根据手册,写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);
}

2.3 应用层显示控制

应用层函数,用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);
}

显示效果如图:
在这里插入图片描述

三、总结

  1. 采用模拟IIC的方式,STM322CubeMX配置IIC的引脚时,需要配置为开漏输出模式;
  2. 带时间点的模块,中间的时间点,是它前面的数码管的小数点位。

标签:驱动程序,void,uint8,C语言,数码管,IIC,sDisplayData,TM1637,GPIO
来源: https://blog.csdn.net/wanglong3713/article/details/121110104

本站声明: 1. iCode9 技术分享网(下文简称本站)提供的所有内容,仅供技术学习、探讨和分享;
2. 关于本站的所有留言、评论、转载及引用,纯属内容发起人的个人观点,与本站观点和立场无关;
3. 关于本站的所有言论和文字,纯属内容发起人的个人观点,与本站观点和立场无关;
4. 本站文章均是网友提供,不完全保证技术分享内容的完整性、准确性、时效性、风险性和版权归属;如您发现该文章侵犯了您的权益,可联系我们第一时间进行删除;
5. 本站为非盈利性的个人网站,所有内容不会用来进行牟利,也不会利用任何形式的广告来间接获益,纯粹是为了广大技术爱好者提供技术内容和技术思想的分享性交流网站。

专注分享技术,共同学习,共同进步。侵权联系[81616952@qq.com]

Copyright (C)ICode9.com, All Rights Reserved.

ICode9版权所有