⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 twi.c

📁 AVR单片机TWI程序
💻 C
字号:

/*函数的使用说明:
1.该twi模块包含四种工作模式,主发,主收,从收,从发,其中主发由
  twiWriteByte(从机地址,要发送的数据)实现,主收由twiReadByte(从机地址)实现,从机
  的发送与接收均通过中断实现
2.在从机模式下,需调用twi_init()来使能并初始化twi总线
3.在主机模式下,调用twiWriteByte(从机地址,要发送的数据)向指定的从机写入数据,
  调用twiReadByte(从机地址)从指定的从机读取数据,然后可通过查询错误状态error_state
  的值决定下一步如何操作。
*/

//************************* 头文件 ******************************************
#include "twi.h"

//TWI同程序的接口参数
//receive_data,error_state不用设置
extern char receive_data;  		 // 接收到的数据
extern char error_state;   		 // twi错误状态
//以下三个参数根据需求进行设置
extern char slave_address;      // 设置从机地址,Bits 7..1:存放从机地址,Bit 0:最低位为广播识别使能位
extern char bit_race;           // 设置主机模式的比特率,SCL=CPU频率/(16+2*(TWBR)*4TWPS),TWPS在4的指数位置
extern char send_data;     	    // 从机发送模式下从机向主机发送的数据

//*************************** twi错误状态说明*********************************
// 0:twi传送无错误
// 1:在主发模式下,SLA+W已发送,返回NOT ACK
// 2:在主发模式下,DATA已发送,返回NOT ACK
// 3:在主发模式下,SLA+W或者数据的仲裁失败
// 4:在主收模式下,SLA+R或者数据的仲裁失败
// 5:在主收模式下,SLA+R已发送,返回NOT ACK
// 6:在主发模式下,START信号发送不成功
// 7:在主收模式下,START信号发送不成功
// 8:在主收模式下,数据接受完成
// 9:在主收模式下,数据接受不成功
//*****************************************************************************

//*************************** twi stop 信号延时时间 ****************************
//******************************************************************************
void StopDelay(void)
{
	unsigned int i;
	for(i = 0;i < 20;i ++);
}
//******************************* twi 启动总线*********************************
//******************************************************************************
void start(void)
{
	TWCR = (1 << TWINT) | (1 << TWSTA) | (1 << TWEN);	//发出start信号
	_NOP();
	_NOP();
}
//************************** twi 释放总线************************************
//***************************************************************************
void stop(void)
{
	TWCR = (1 << TWINT) |(1 << TWEN)|(1 << TWSTO); //发出stop信号	
	StopDelay();           //等待总线恢复
	TWCR= 0x45;
}
//****************************** twi 初始化*************************************
// TWCR: 控制寄存器,用来控制TWI操作,说明如下:
// Bit 7-TWINT:中断标志位,Bit 6-TWEA:使能应答位,Bit 5-TWSTA:START状态位
// Bit 4-TWSTO:STOP状态位,Bit 3-TWWC: 写冲突标志,Bit 2-TWEN:TWI使能位
// Bit 1-RES:保留,Bit 0-TWIE:中断使能                   
// TWSR: 状态寄存器,Bits 7..3:表示了TWI总线的当前状态,读取时需屏蔽低三位的值,Bits 1..0-TWPS:TWI预分频位
// TWBR: 比特率寄存器,用来设置TWI的工作频率,计算公式为:SCL=CPU频率/(16+2*(TWBR)*4TWPS),TWPS在4的指数位置
// TWAR: 从机地址寄存器,Bits 7..1:存放从机地址,Bit 0:最低位为广播识别使能位
// TWDR: 数据寄存器,用来存放接收或要发送的地址和数据
//******************************************************************************
void twi_init(void)
{
	TWCR= 0x00;    			//disable twi
	TWBR= bit_race;   		//set bit rate
	TWSR= 0x00;             //set prescale为1
	TWAR= slave_address;    //set slave address
	TWCR= 0x45;          //enable twi
}
//************************** 检测发送时的状态码 ****************************
//**************************************************************************
void checkstate(void)
{
	unsigned char TWSR_state;
	TWSR_state = TWSR & 0xf8; //屏蔽第三位读取状态
	switch(TWSR_state)
	{
	case 0x08:error_state=0;break;  // START已发送
	case 0x18:error_state=0;break;  // SLA+W已发送,接收到ACK
	case 0x20:error_state=1;break;  // SLA+W已发送,接收到NOT ACK
	case 0x28:error_state=0;break;  // 数据已发送,接收到ACK
	case 0x30:error_state=2;break;  // 数据已发送,接收到NOT ACK
	case 0x38:error_state=3;break;  // SLA+W或数据的仲裁失败
	default: break;
	}
	TWCR=0x45;
}
//********** twi主机发送,向地址为WriteAddress的从机发送1个字节数据************
// 入口参数1:WriteAddress:从机地址
// TWDR中存放的高七位为地址,最低位表示读写控制,0为写,1为读
// 入口参数2:data,向从机写的数据
//******************************************************************************
void twiWriteByte(unsigned char WriteAddress,char data)
{
	unsigned char TWCR_state;
	char i=0;
	error_state=0;
	
	//初始化 twi 
	twi_init();
	
	//发送START信号 
	start();
	TWCR_state = TWCR & 0x80;
	while((TWCR_state == 0x00))
	{ 
		i++; 
		if(i>=10){error_state=6;break;}
		TWCR_state = TWCR & 0x80;    //轮循等待START信号发送完成
	}	
	
	//若START发送成功,发送从机地址
	if(error_state==0)
	{
		TWDR = WriteAddress;        //发地址
		_NOP();
		_NOP();
		
		TWCR = (1 << TWINT) | (1 << TWEN);
		TWCR_state = TWCR & 0x80;
		while(TWCR_state == 0x00)
		{TWCR_state = TWCR & 0x80;}	  //轮循查询地址是否发送
		checkstate();
	}
	
	//若从机地址发送成功,向从机发送数据
	if(error_state==0)
	{
		TWDR = data;               //发数据
		_NOP();
		_NOP();
		TWCR = (1 << TWINT) | (1 << TWEN);
		TWCR_state = TWCR & 0x80;
		while(TWCR_state == 0x00) TWCR_state = TWCR & 0x80;	//轮循查询数据是否发送
		checkstate();
	}
	
	//发送STOP,退出总线控制
	stop();
}
//************** twi主机接收,从地址为ReadAaddress的从机接收1个字节*************
// 入口参数1:ReadAaddress:从机地址
// TWDR中存放的高七位为地址,最低位表示读写控制,0为写,1为读
//******************************************************************************
void twiReadByte(unsigned char ReadAaddress)
{
	unsigned char TWCR_state;
	char i=0;
	error_state=0;
	
	//初始化 twi 
	twi_init();
	
	//发送START信号
	start();
	TWCR_state = TWCR & 0x80;
	while((TWCR_state == 0x00))
	{ 
		i++; 
		if(i>=10){error_state=7;break;}
		TWCR_state = TWCR & 0x80;    //轮循等待START信号发送完成
	}	
	
	//若START信号发送完成,发送读地址,并读取数据
	if(error_state==0)
	{
		TWDR = ReadAaddress;         //发地址
		_NOP();
		_NOP();
		TWCR = (1 << TWINT) | (1 << TWEN)|( 1<<TWIE )|(1 << TWEA);
		while(error_state==0) 
		{i++;if(i>150){error_state=9;break;}};
	}
	
	//发送STOP,退出总线控制
	stop();
}
//************** twi 中断例程,接收数据 ****************************************
//******************************************************************************
//twi中断服务程序
char twi_isr(void)
{
	unsigned char TWSR_state;
	TWSR_state = TWSR & 0xf8;
	switch(TWSR_state)
	{
		//从机接收模式下的中断程序
    case 0x60: TWCR=0xc5;break;      	       // SLA+W已经接收,ACK已返回(从机被寻址,等待主机发送数据)
    case 0x80: receive_data = TWDR;TWCR=0xc5;break;  // 数据已接收,ACK已返回(接收数据)
    case 0x88: TWCR=0xc5;break;			       // 数据已接收,NOT ACK已返回(下一步:退出总线)
	case 0xA0: TWCR=0xc5;break;			       // 接收到STOP或重复的START信号(停止接收数据)
		
		//主机接收模式下的中断程序
    case 0x38: error_state=4;TWCR=0xc5;break;  // SLA+R或NOT ACK仲裁失败(下一步:退出总线)
	case 0x40: TWCR=0xc5; break;               // SLA+R已发送,接收到ACK(从机被寻址,主机等待接收数据)
	case 0x48: error_state=5;TWCR=0xc5;break;  // SLA+R已发送,接收到NOT ACK(下一步:退出总线)
    case 0x50: receive_data = TWDR;_NOP();_NOP();error_state=8;TWCR = 0xc4;break;	// 接收到数据,ACK已返回(接收数据)
	case 0x58: receive_data = TWDR;_NOP();_NOP();error_state=8;TWCR = 0xc4;break;   // 接收到数据,NOT ACK已返回(下一步:退出总线)
		
		//从机发送模式
	case 0xA8: TWDR= send_data;_NOP();_NOP();TWCR= 0x85; break; // SLA+R已接收,ACK已返回(从机被寻址,开送发送数据)
	case 0xB8: TWCR= 0xc5;break;              // TWDR里的数据已发送,接受到ACK(主机接收到数据,继续发送数据)
	case 0xC0: TWCR= 0xc5;break;			  // TWDR里的数据已发送,接受到NOT ACK(下一步:退出总线)
	case 0xC8: TWCR= 0xc5;break;              // TWEA=0,接收到ACK(停止发送数据,退出发送)
		
		//其他状态码
	case 0xf8: break;						  // 等待或进行当前传输
	case 0x00: TWCR= 0x95;break;              // 非法的START或STOP引起的总线错误(释放总线)
    default:   break;
	}
	return receive_data;					  // 返回接收到的数据
}

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -