ARM接口实验—IIC实验

一、IIC总线简介

I2C总线为两线制,只有两根双向信号线。一根是数据线SDA,另一根是时钟线SCL

IIC总线有两个上拉电阻,这两个上拉电阻作用在IIC总线为空闲状态时保持高电平。

IIC总线属于同步、半双工、串行总线

二、IIC时序

1,起始信号:在SCL为高电平期间,SDA从高电平到低电平的变化

2,停止信号:在SCL为高电平期间,SDA从低电平到高电平的变化

3,数据传输:在SCL高电平时,只允许读取数据,SDA必须保持稳定

                        在SCL低电平时,允许往SDA上写数据

4,应答/非应答信号:在8位数据位传输完毕后,必须在第9个时钟周期内返回一个应答/非应答信号

        如果读到的是高电平,代表是非应答信号

        如果读到的是低电平,代表是应答信号

5,寻址,由于需要确定从机地址,所以在起始信号产生后,必须传递一个7位从机地址

                第8位是数据传输的方向位,用0表示主机发送数据(写),用1表示主机接收数据(读)

 

 三、IIC协议

3.1 主机给从机发送一个字节

3.2 主机给从机发送多个连续字节

3.3 从机给主机发送一个字节

3.4 从机给主机发送多个连续字节

四、温湿度采集实验

使用GPIO模拟IIC协议

 iic.h

#ifndef __IIC_H__
#define __IIC_H__
#include "stm32mp1xx_gpio.h"
#include "stm32mp1xx_rcc.h"
#include "gpio.h"
/* 通过程序模拟实现I2C总线的时序和协议
 * GPIOF ---> AHB4
 * I2C1_SCL ---> PF14
 * I2C1_SDA ---> PF15
 *
 * */

#define SET_SDA_OUT     do{GPIOF->MODER &= (~(0x3 << 30)); \
							GPIOF->MODER |= (0x1 << 30);}while(0)
#define SET_SDA_IN      do{GPIOF->MODER &= (~(0x3 << 30));}while(0)

#define I2C_SCL_H       do{GPIOF->BSRR |= (0x1 << 14);}while(0)
#define I2C_SCL_L       do{GPIOF->BRR |= (0x1 << 14);}while(0)

#define I2C_SDA_H       do{GPIOF->BSRR |= (0x1 << 15);}while(0)
#define I2C_SDA_L       do{GPIOF->BRR |= (0x1 << 15);}while(0)

#define I2C_SDA_READ    (GPIOF->IDR & (0x1 << 15))

void delay_us(void);
void i2c_init(void);
void i2c_start(void);
void i2c_stop(void);
void i2c_write_byte(unsigned char  dat);
unsigned char i2c_read_byte(unsigned char ack);
unsigned char i2c_wait_ack(void);       
void i2c_ack(void);
void i2c_nack(void);

#endif 

iic.c

#include "../include/iic.h"

extern void printf(const char *fmt, ...);
/*
 * 函数名 : delay_us
 * 函数功能:延时函数
 * 函数参数:无
 * 函数返回值:无
 * */
void delay_us(void)
{
	unsigned int i = 2000;
	while (i--);
}
/*
 * 函数名 : i2c_init
 * 函数功能: i2C总线引脚的初始化, 通用输出,推挽输出,输出速度,
 * 函数参数:无
 * 函数返回值:无
 * */
void i2c_init(void)
{
	// 使能GPIOF端口的时钟
	RCC->MP_AHB4ENSETR |= (0x1 << 5);
	// 设置PF14,PF15引脚为通用的输出功能
	GPIOF->MODER &= (~(0xF << 28));
	GPIOF->MODER |= (0x5 << 28);
	// 设置PF14, PF15引脚为推挽输出
	GPIOF->OTYPER &= (~(0x3 << 14));
	// 设置PF14, PF15引脚为高速输出
	GPIOF->OSPEEDR |= (0xF << 28);
	// 设置PF14, PF15引脚的禁止上拉和下拉
	GPIOF->PUPDR &= (~(0xF << 28));
	// 空闲状态SDA和SCL拉高
	I2C_SCL_H;
	I2C_SDA_H;
}

/*
 * 函数名:i2c_start
 * 函数功能:模拟i2c开始信号的时序
 * 函数参数:无
 * 函数返回值:无
 * */
void i2c_start(void)
{
	/*
	 * 开始信号:时钟在高电平期间,数据线从高到低的变化
	 *     --------
	 * SCL         \
	 *              --------
	 *     ----
	 * SDA     \
	 *          --------
	 * */
	SET_SDA_OUT;   // 设置SDA为输出模式,确保一定为输出模式
	I2C_SDA_H;     // SDA拉高
	I2C_SCL_H;     // SCL拉高
	delay_us();
	I2C_SDA_L;     // SDA拉低
	delay_us();
	I2C_SCL_L;     // SCL拉低,I2C总线就处于占用态
}

/*
 * 函数名:i2c_stop
 * 函数功能:模拟i2c停止信号的时序
 * 函数参数:无
 * 函数返回值:无
 * */

void i2c_stop(void)
{
	/*
	 * 停止信号 : 时钟在高电平期间,数据线从低到高的变化
	 *             ----------
	 * SCL        /
	 *    --------
	 *    ---         -------
	 * SDA   X       /
	 *    --- -------
	 *    为了确保停止信号是一个上升沿,因此在时钟为低电平期间
	 *    将数据线拉低,确保可以产生上升沿。
	 * */
	SET_SDA_OUT;    // 设置SDA为输出
	I2C_SCL_L;      // SCL拉低
	delay_us();
	I2C_SDA_L;      // SDA拉低
	delay_us();
	I2C_SCL_H;      // SCL拉高
	delay_us();
	I2C_SDA_H;      // SDA拉高

}

/*
 * 函数名: i2c_write_byte
 * 函数功能:主机向i2c总线上的从设备写8bits数据
 * 函数参数:dat : 等待发送的字节数据
 * 函数返回值: 无
 * */

void i2c_write_byte(unsigned char dat)
{
	/*
	 * 数据信号:时钟在低电平期间,发送器向数据线上写入数据
	 * 			时钟在高电平期间,接收器从数据线上读取数据
	 *      ----          --------
	 * 	SCL     \        /        \
	 *           --------          --------
	 *      -------- ------------------ ---
	 * 	SDA         X                  X
	 *      -------- ------------------ ---
	 *      先发送高位在发送低位
	 * */
	unsigned int i;
	SET_SDA_OUT;   // SDA为输出
	for (i = 0 ; i < 8; i++) {
		I2C_SCL_L;   // SCL拉低
		delay_us();
		if (dat & 0x80)  // 判断最高位的值
			I2C_SDA_H;   // 向数据线上写入高电平
		else 
			I2C_SDA_L;   // 向数据线上写入低电平
		delay_us();
		I2C_SCL_H;   // SCL拉高
		delay_us();
		delay_us();  // 等待从机将数据线上的数据读走
		dat <<= 1;   // 将次高位的数据移动到最高位
	}
}

/*
 * 函数名:i2c_read_byte
 * 函数功能: 主机从i2c总线上的从设备读8bits数据,
 *          主机发送一个应答或者非应答信号给从机
 * 函数参数: 0 : 应答信号   1 : 非应答信号
 * 函数返回值:读到的有效数据
 *
 * */
unsigned char i2c_read_byte(unsigned char ack)
{
	/*
	 * 数据信号:时钟在低电平期间,发送器向数据线上写入数据
	 * 			时钟在高电平期间,接收器从数据线上读取数据
	 *      ----          --------
	 * 	SCL     \        /        \
	 *           --------          --------
	 *      -------- ------------------ ---
	 * 	SDA         X                  X
	 *      -------- ------------------ ---
	 *
	 *      先接收高位, 在接收低位
	 * */
	unsigned char dat;
	unsigned int i;
	SET_SDA_IN;   // 设置SDA为输入
	for (i = 0; i < 8; i++) {
		I2C_SCL_L;   // SCL拉低
		delay_us();
		delay_us();  // 等待从机向数据线上写入数据
		I2C_SCL_H;   // SCL拉高
		delay_us();
		dat <<= 1;
		if (I2C_SDA_READ) 
			dat |= 1;
		else 
			dat |= 0;
		
		delay_us();
	}

	if (!ack) 
		i2c_ack();   // 发送应答信号
	else 
		i2c_nack();  // 发送非应答信号
	return dat;

}
/*
	将dat中数据的左移1位写到接收数据的上边。

	为什么将dat中的数据左移1位,写道接收数据的下边不可以?
	假设接收的数据为0xFF,
					先接收数据    在进行左移
	第1次循环        0000 0001    0000 0010
	第2次循环        0000 0011    0000 0110
	第3次循环        0000 0111    0000 1110
	第4次循环        0000 1111    0001 1110
	第5次循环        0001 1111    0011 1110
	第6次循环        0011 1111    0111 1110
	第7次循环        0111 1111    1111 1110
	第8次循环        1111 1111    1111 1110

*/

/*
 * 函数名: i2c_wait_ack
 * 函数功能: 主机作为发送器时,等待接收器返回的应答信号
 * 函数参数:无
 * 函数返回值:
 *					0:接收到的应答信号
 *                  1:接收到的非应答信号
 * */
unsigned char i2c_wait_ack(void)
{
	int ack;
	/*
	 * 主机发送一个字节之后,从机给主机返回一个应答信号,主机接收应答信号
	 *                       -----------
	 * SCL                  /   M:读    \
	 *     -----------------             --------
	 *     --- -------- --------------------
	 * SDA    X        X
	 *     ---          --------------------
	 *     主  释 设  从机    主机
	 *     机  放 置  向数据  读数据线
	 *         总 SDA 线写    上的数据
	 *         线 输  数据
	 *            入
	 * */
	I2C_SCL_L;   // SCL拉低
	delay_us();
	I2C_SCL_L;
	I2C_SDA_H;   // SDA拉高, 释放总线
	SET_SDA_IN;  // 设置SDA为输入
	delay_us();
	delay_us();  // 等待从机向数据线上写入应答信号
	I2C_SCL_H;   // SCL拉高
	delay_us();
	if (I2C_SDA_READ) 
		ack = 1;  // 非应答信号
	else 
		ack = 0;  // 应答信号
	delay_us();
	I2C_SCL_L;
	return ack;
}
/*
 * 函数名: iic_ack
 * 函数功能: 主机作为接收器时,给发送器发送应答信号
 * 函数参数:无
 * 函数返回值:无
 * */
void i2c_ack(void)
{
	/*            --------
	 * SCL       /        \
	 *    -------          ------
	 *    ---
	 * SDA   X
	 *    --- -------------
	 *    在第九个时钟周期的低电平期间,接收器向数据线写入数据,
	 *    在第九个时钟周期的高电平期间,发送器从数据线上读取数据,
	 *    如果读到低电平表示应答信号
	 * */
	SET_SDA_OUT;   // 设置SDA为输出
	I2C_SCL_L;     // SCL拉低
	delay_us();
	I2C_SDA_L;     // SDA拉低,发送的是应答信号
	delay_us();   
	I2C_SCL_H;     // SCL拉高
	delay_us();
	delay_us();    // 等待从机接收应答信号
	I2C_SCL_L;     // SCL拉低
}
/*
 * 函数名: iic_nack
 * 函数功能: 主机作为接收器时,给发送器发送非应答信号
 * 函数参数:无
 * 函数返回值:无
 * */
void i2c_nack(void)
{
	/*            --------
	 * SCL       /        \
	 *    -------          ------
	 *    --- ---------------
	 * SDA   X
	 *    ---
	 *    在第九个时钟周期的低电平期间,接收器向数据线写入数据,
	 *    在第九个时钟周期的高电平期间,发送器从数据线上读取数据,
	 *    如果读到高电平表示非应答信号
	 * */
	SET_SDA_OUT;   // 设置SDA为输出
	I2C_SCL_L;     // SCL拉低
	delay_us();
	I2C_SDA_H;     // SDA拉高,发送的是非应答信号
	delay_us();   
	I2C_SCL_H;     // SCL拉高
	delay_us();
	delay_us();    // 等待从机接收非应答信号
	I2C_SCL_L;     // SCL拉低
}

main.c

#include "led.h"
#include "uart4.h"
#include "command.h"
#include "beep.h"
#include "fan.h"
#include "motor.h"
#include "interrupt.h"
#include "timer2_it.h"
#include "./include/si7006.h"

extern void printf(const char *fmt, ...);
void delay_ms(unsigned int ms)
{
	int i, j;
	for (i = 0; i < ms; i++)
		for (j = 0; j < 1800; j++)
			;
}

extern char buffer[LEN];

int main()
{
	unsigned short hum;
	short temp;
	si7006_init();

	while (1)
	{
		hum = si7006_read_hum_data(SI7006_SLAVE, MEASURE_HUM_CMD);
		temp = si7006_read_temp_data(SI7006_SLAVE, MEASURE_TEMP_CMD);
		hum = (125 * hum / 65536 - 6) * 10;
		temp = (175.72 * temp / 65536 - 46.85) * 10;

		printf("hum = %d.%d\n", hum / 10, hum % 10);
		printf("temp = %d.%d\n", temp / 10, temp % 10);
		delay_ms(1000);
		
	}
	return 0;
}

 实验现象:

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

林某某..

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值