基于stm32软件IIC的oled显示温湿度

IIC简介

特点:
由于他引脚少,硬件实现简单,可拓展性强,不需要UASRT,CAN通讯协议的外部收发设备,现在被广泛使用在系统内多个集成电路IC(芯片)间的通讯。
通讯方式:
半双工的通讯方式

IIC物理层

IIC总线系统架构(IIC物理层)

在这里插入图片描述
关于上述图内容的解释

1.一个IIC总线只使用两条总线线路,
一条双向串行数据线(SDA),数据线用于表示数据
一条串行时钟线(SCL),时钟线用于数据收发同步。
2.他是一个支持多设备的总线。”总线”指多个设备共用的信号线,在一个IIC通讯总线中,可连接多个IIC通讯设备,支持多个通讯主机及多个通讯从机。每个设备都有独立的地址,主机可以通过访问不同的地址来访问从机
3.总线由上拉电阻接到电源VCC,总线设备空闲状态时候出现高阻态,此时由上拉电阻把总线拉成高电平
4.当多个主机设备同时占用总线时候,为了防止冲突会利用仲裁方式来决定谁占用总线

IIC总线分类

软件IIC:一般是配置GPIO管脚,用软件来控制管脚状态,模拟IIC通讯过程
硬件IIC:对应芯片上的IIC外设,有相对应的IIC驱动电路,其所使用的IIC管脚也是专用的

两者的区别
1.硬件IIC的速度远高于软件IIC,但是硬件IIC受引脚的限制,不灵活。
2.软件IIC是通过配置GPIO,软件模拟寄存器的工作方式,而硬件IIC是直接调用内部寄存器进行配置。
综上可以总结如下
1.硬件IIC用法复杂,模拟IIC流程更加清楚
2.硬件IIC速度比模拟快
3.模拟IIC可以在任何管脚上,硬件IIC在固定管脚上

IIC协议层

IIC数据传输过程的一些状态(IIC协议层)

①空闲状态
当IIC总线SDA以及SCL均处于高电平时,规定此状态为空闲状态,对应输出状态为高阻态(各场器件输出效应管截止,导致场效应管电阻很大),由上拉电阻将电平拉高。
②开始信号和截止信号
在这里插入图片描述
如图所示 ,起始条件为 SDA(数据总线)由高电平变为低电平,下降沿的跳变。SCL(时钟总线)保持高电平状态。
终止条件为 SDA由低电平变为高电平,上升沿的跳变。SCL保持高电平状态。
③应答信号
在这里插入图片描述
当发送完一个字节(8位)后,在第9位释放数据线,由接收器件的数据线返回一个应答信号(ACK),并且规定当为低电平时候为有效应答。结合图片可以如下总结
对于反馈有效应答位ACK的要求是,接收器在第九个时钟脉冲之前的低电平期间将SDA线拉低,并且确保在该时钟的高电平期间为稳定的低电平
⑤数据有效性以及数据传输
在这里插入图片描述
具体要求,当SCL为高电平的时候要求SDA数据线数据稳定,SCL为低电平的时候SDA数据线数据可变化。
SDA数据在SCL的每一个时钟周期传递一位数据。数据位的传输是边沿触发
可以总结如下
数据在SCL的上升沿到来前准备好。并在下降沿到来之前必须稳定

IIC通讯——读写数据

IIC的协议定义了通讯的起始和停止信号、数据有效性、响应、仲裁、时钟同步和地址广播等环节。

①主机写数据到从机
在这里插入图片描述
s:start,起始信号
p:stop,停止信号
slave address:从机地址。开始信号结束后,主机就开始广播从机的地址信号,在没有广播出来之前,每一个从机都处于“待命”状态,由于每一个从机地址都是唯一的,只需要主机广播的地址对应到某个从机,某个从机就会做出响应,其他的就不会做出响应。(可以理解为老师上课点名😄)从机地址可以是7位或者10位
R/w:读写数据
A: ACK,应答信号。当数据写完毕后,仅有当从机应答信号产生后主机才能继续写数据。
②主机到从机中读数据
在这里插入图片描述
③主机与从机的通讯,通讯复合格式
在这里插入图片描述
复合格式,与前面的主要区别就是:该传输过程有两次起始信号。
第一次传输过程中,主机通过SLAVE_ADDRESS寻找到从设备后,发送一段”数据”,这段数据通常用于表示从机设备内部的寄存器或存储器地址;
第二次传输中,对该地址的内容进行读或写。(与之前的一样了)
综合所述,第一次通讯是告诉从机读写地址,第二次则是读写的实际内容。

基于stm32软件IIC

驱动代码

oled.c代码

#include "stm32f10x.h"
#include "oled.h"
#include "oledfont.h"
#include "delay.h"			//系统定时器

static void OLED_GPIO_Init(void)		
{
   
	
	 GPIO_InitTypeDef   	oled_GPIOstruct;				//oled管脚初始化
	 RCC_APB2PeriphClockCmd( RCC_APB2Periph_GPIOB, ENABLE );
//PB0 ->SCL    PB1 ->SDA		
	 oled_GPIOstruct.GPIO_Mode = GPIO_Mode_Out_OD;		//开漏输出
	 oled_GPIOstruct.GPIO_Pin =GPIO_Pin_0 | GPIO_Pin_1 ; 
	 oled_GPIOstruct.GPIO_Speed = GPIO_Speed_50MHz;
	 GPIO_Init( GPIOB, &oled_GPIOstruct);

	OLED_SCLK_Set();
	OLED_SDIN_Set();		//将时钟线和数据线全部拉高,让其处于空闲状态
}
//模拟IIC起始信号
static void OLED_IIC_Start(void)
{
   
	OLED_SCLK_Set();
	OLED_SDIN_Set();		
	delay_us(1);
	OLED_SDIN_ReSet();			//数据线拉低,产生下降沿
	delay_us(1);
	OLED_SCLK_ReSet();			//时钟线再数据线产生下降沿之后保持一段时间高电平后将其拉低,起始信号模拟完毕
	delay_us(1);
}
//IIC模拟停止信号
static void OLED_IIC_Stop(void)
{
   
	OLED_SDIN_ReSet();
	OLED_SCLK_ReSet();
	delay_us(1);
	OLED_SCLK_Set();		//时钟线先拉高
	delay_us(1);
	OLED_SDIN_Set();		//数据线拉高,产生上升沿,至此停止信号模拟完毕
	delay_us(1);
}
//IIC模拟应答信号
static unsigned char IIC_Wait_Ack(void)
{
   
	unsigned char Ask;
	OLED_SCLK_ReSet();		//时钟线拉低
	delay_us(1);
	OLED_SDIN_Set();		//数据线拉高
	delay_us(1);
	OLED_SCLK_Set();		//时钟线拉高
//应答信号,在时钟线拉高之前产生下降沿并在时钟线拉高的情况下确保为低电平
	if(OLED_READ_SDIN())	//读取数据线电平为高电平
	{
   
		ack = IIC_NO_Ask;	//对应的应答信号为非应答信号
	}
	else
	{
   
		ack = IIC_Ask;		//对应信号为应答信号
	}
	OLED_SCLK_ReSet();		//接收完应答信号,时钟线拉低
	delay_us(1);
	return ack;
}
//写入一位数据
static void Write_IIC_Byte(unsigned char IIC_Byte)
{
   
	unsigned char i=0;
	for(i=0;i<8;i++)	//每位循环写入
	{
   
		OLED_SCLK_ReSet();		//拉低时钟线,让后面数据可以改变
		delay_us(1);
		if(IIC_Byte & 0x80)		
			OLED_SDIN_Set();	//最高位为1,数据线置高
		else
			OLED_SDIN_ReSet(); 	//最高位为0,数据线拉低
		IIC_Byte <<= 1; 		//数据左移1位,次高位变成最高位
		delay_us(1);
		OLED_SCLK_Set();		//传输并确认一位数据之后,时钟线拉高,读取这一位数据
		delay_us(1);
	}
	  OLED_SCLK_ReSet();		//8位数据全部读完之后时钟线拉低		
		delay_us(1);
		while(IIC_Wait_Ack());	//等待应答信号通过!	
}
//IIC写命令
static void Write_IIC_Command(unsigned char IIC_Command)
{
   
	OLED_IIC_Start();				
	Write_IIC_Byte(0x78);		//第一次发送从机地址
	Write_IIC_Byte(0x00);		//第二次发送命令地址
	Write_IIC_Byte(IIC_Command);//写入指定命令	
	OLED_IIC_Stop();			
}
//IIC写数据
static void Write_IIC_Data(unsigned char IIC_Data)
{
   
	OLED_IIC_Start();				
	Write_IIC_Byte
以下是使用STM32F103C8T6和OLED屏幕显示温湿度的代码示例: ```c #include "stm32f10x.h" #include "dht11.h" #include "OLED_IIC.h" #include "delay.h" #include "stdlib.h" #define DHT11_GPIO GPIOB #define DHT11_PIN GPIO_Pin_10 void GPIO_Configuration(void); void USART_Configuration(void); void NVIC_Configuration(void); void SysTick_Configuration(void); void TIM_Configuration(void); void OLED_Display_Temp_Humi(float temp, float humi); volatile uint32_t TimingDelay; int main(void) { SysTick_Configuration(); GPIO_Configuration(); USART_Configuration(); NVIC_Configuration(); TIM_Configuration(); OLED_Init(); OLED_Clear(); OLED_ShowString(0,0,"Temperature/Humidity"); while(1) { float temp, humi; if(DHT11_Read_Data(DHT11_GPIO, DHT11_PIN, &temp, &humi) == SUCCESS) { OLED_Display_Temp_Humi(temp, humi); } Delay_ms(2000); } } void GPIO_Configuration(void) { GPIO_InitTypeDef GPIO_InitStructure; RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB, ENABLE); GPIO_InitStructure.GPIO_Pin = DHT11_PIN; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; GPIO_Init(DHT11_GPIO, &GPIO_InitStructure); GPIO_SetBits(DHT11_GPIO, DHT11_PIN); } void USART_Configuration(void) { USART_InitTypeDef USART_InitStructure; RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1, ENABLE); USART_InitStructure.USART_BaudRate = 115200; USART_InitStructure.USART_WordLength = USART_WordLength_8b; USART_InitStructure.USART_StopBits = USART_StopBits_1; USART_InitStructure.USART_Parity = USART_Parity_No; USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None; USART_InitStructure.USART_Mode = USART_Mode_Rx | USART_Mode_Tx; USART_Init(USART1, &USART_InitStructure); USART_Cmd(USART1, ENABLE); } void NVIC_Configuration(void) { NVIC_InitTypeDef NVIC_InitStructure; NVIC_InitStructure.NVIC_IRQChannel = USART1_IRQn; NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0; NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0; NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; NVIC_Init(&NVIC_InitStructure); USART_ITConfig(USART1, USART_IT_RXNE, ENABLE); } void SysTick_Configuration(void) { if(SysTick_Config(SystemCoreClock / 1000)) { while(1); } } void TIM_Configuration(void) { TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure; RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3, ENABLE); TIM_TimeBaseStructure.TIM_Period = 999; TIM_TimeBaseStructure.TIM_Prescaler = 7199; TIM_TimeBaseStructure.TIM_ClockDivision = TIM_CKD_DIV1; TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up; TIM_TimeBaseInit(TIM3, &TIM_TimeBaseStructure); TIM_Cmd(TIM3, ENABLE); } void OLED_Display_Temp_Humi(float temp, float humi) { char temp_str[10], humi_str[10]; gcvt(temp, 2, temp_str); gcvt(humi, 2, humi_str); OLED_ShowString(0,20,"Temp:"); OLED_ShowString(32,20,temp_str); OLED_ShowChar(96,20,'C',8); OLED_ShowString(0,40,"Humi:"); OLED_ShowString(32,40,humi_str); OLED_ShowChar(96,40,'%',8); } void Delay_ms(__IO uint32_t nTime) { TimingDelay = nTime; while(TimingDelay != 0); } void TimingDelay_Decrement(void) { if (TimingDelay != 0x00) { TimingDelay--; } } void USART1_IRQHandler(void) { if(USART_GetITStatus(USART1, USART_IT_RXNE) != RESET) { USART_SendData(USART1, USART_ReceiveData(USART1)); while(USART_GetFlagStatus(USART1, USART_FLAG_TXE) == RESET); } } ``` 其中,dht11.h和OLED_IIC.h是DHT11温湿度传感器和OLED屏幕的库文件。注意修改DHT11_GPIO和DHT11_PIN为实际使用的GPIO和引脚号。
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值