stm32硬件I2C测试例程,亲测可用

        对于stm32的硬件I2C确实有不尽人意的地方。但是还是可以实现的,毕竟使用stm32的硬件I2C确实比使用IO口来模拟简单的多。下面的程序代码是使用stm32F03ZET6的I2C1(PB6,PB7)和AT24C02的EEPROM来测试的。希望对于需要的朋友有帮助。

i2c.c

#include "iic.h"
#include "stm32f10x.h"
#include "usart.h"

u16 timeout=TIMEOUT;
/*i2c的初始化*/
void IIC1_Init()
{

	GPIO_InitTypeDef GPIO_InitStruct;
	I2C_InitTypeDef I2C_InitStruct;
	
	RCC_APB1PeriphClockCmd(RCC_APB1Periph_I2C1,ENABLE);
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB,ENABLE);
	//GPIO的配置
	GPIO_InitStruct.GPIO_Mode=GPIO_Mode_AF_OD;
	GPIO_InitStruct.GPIO_Pin=GPIO_Pin_6|GPIO_Pin_7;
	GPIO_InitStruct.GPIO_Speed=GPIO_Speed_50MHz;
	GPIO_Init(GPIOB,&GPIO_InitStruct);
	
	//IIC的配置
	
	I2C_InitStruct.I2C_Ack=I2C_Ack_Enable;
	I2C_InitStruct.I2C_AcknowledgedAddress=I2C_AcknowledgedAddress_7bit;
	I2C_InitStruct.I2C_ClockSpeed=10000;
	I2C_InitStruct.I2C_DutyCycle=I2C_DutyCycle_2;
	I2C_InitStruct.I2C_Mode=I2C_Mode_I2C;
	I2C_InitStruct.I2C_OwnAddress1=0x55;
	I2C_Init(I2C1,&I2C_InitStruct);
	
	I2C_Cmd(I2C1,ENABLE);
	
	
}

/*写入一个字节*/
void IIC_WriteByte(u8 addr,u8 data)
{
	I2C_EE_WaitEepromStandbyState();
	I2C_GenerateSTART(I2C1,ENABLE);//发送起始信号
	while(!I2C_CheckEvent(I2C1,I2C_EVENT_MASTER_MODE_SELECT))//检测	EV5事件
	{
		  if((timeout--)==0)
			printf("EV5 Fail");
	}	timeout=1000;
	
	I2C_Send7bitAddress(I2C1,0xA0,I2C_Direction_Transmitter);//发送7位EEPROM的硬件地址
	while(!I2C_CheckEvent(I2C1,I2C_EVENT_MASTER_TRANSMITTER_MODE_SELECTED))//检测EV6事件
	{
		  if((timeout--)==0)
			printf("EV6 Fail");
	}	timeout=1000;
	
	I2C_SendData(I2C1,addr);//发送操作的内存地址
	while(!I2C_CheckEvent(I2C1,I2C_EVENT_MASTER_BYTE_TRANSMITTED))//检测EV8事件
	{
		  if((timeout--)==0)
			printf("EV8 Fail");
	}	timeout=1000;	
	
	I2C_SendData(I2C1,data);//要写入的数据(一个字节)
	while(!I2C_CheckEvent(I2C1,I2C_EVENT_MASTER_BYTE_TRANSMITTED))//检测EV8事件
	{
		  if((timeout--)==0)
			printf("EV8 Fail");
	}	timeout=1000;
	
	
	I2C_GenerateSTOP(I2C1,ENABLE);//发送结束信号
	
}

/*读取一个字节*/
u8 IIC_ReadByte(u8 addr)
{
	u8 readtemp;
	I2C_EE_WaitEepromStandbyState();//等待EEPROM释放总线
	
	I2C_GenerateSTART(I2C1,ENABLE);
	while(!I2C_CheckEvent(I2C1,I2C_EVENT_MASTER_MODE_SELECT))//5
	{
		  if((timeout--)==0)
			printf("EV5 Fail");
	}	timeout=1000;
	
	I2C_Send7bitAddress(I2C1,0xA1,I2C_Direction_Transmitter);
	while(!I2C_CheckEvent(I2C1,I2C_EVENT_MASTER_TRANSMITTER_MODE_SELECTED))//6
	{
		  if((timeout--)==0)
			printf("EV6 Fail");
	}	timeout=1000;
	
	
	I2C_SendData(I2C1,addr);
  while(I2C_CheckEvent(I2C1,I2C_EVENT_MASTER_BYTE_TRANSMITTED) != SUCCESS )
  {
 		  if((timeout--)==0)
			printf("EV8 Fail");
	}	timeout=1000;
	
	I2C_GenerateSTART(I2C1,ENABLE);
	while(!I2C_CheckEvent(I2C1,I2C_EVENT_MASTER_MODE_SELECT))
	{
		  if((timeout--)==0)
			printf("EV5 Fail");
	}	timeout=1000;
	
	I2C_Send7bitAddress(I2C1,0xA0,I2C_Direction_Receiver);
	while(!I2C_CheckEvent(I2C1,I2C_EVENT_MASTER_RECEIVER_MODE_SELECTED))
	{
		  if((timeout--)==0)
			printf("EV6 Fail");
	}	timeout=1000;
	
	I2C_AcknowledgeConfig(I2C1,DISABLE);

  /* 检测 EV7 事件 */
  while(I2C_CheckEvent(I2C1,I2C_EVENT_MASTER_BYTE_RECEIVED) != SUCCESS )
  {
 	  if((timeout--)==0)
			printf("EV7 Fail");
	}	timeout=1000;
  
  /* 读取接收数据 */
  readtemp = I2C_ReceiveData(I2C1);
   
  /* 停止信号 */
  I2C_GenerateSTOP(I2C1,ENABLE);
  
  return readtemp;
}


/*等待eeprom释放总线*/
void I2C_EE_WaitEepromStandbyState(void)      
{
  do
  {
    I2C_GenerateSTART(I2C1, ENABLE);
    while(I2C_GetFlagStatus(I2C1,I2C_FLAG_SB)==ERROR);//检测EV5事件
		I2C_Send7bitAddress(I2C1,0xA1,I2C_Direction_Transmitter);
  }while(I2C_GetFlagStatus(I2C1,I2C_FLAG_ADDR)==ERROR);//
    I2C_ClearFlag(I2C1, I2C_FLAG_AF);//清楚标志位
    I2C_GenerateSTOP(I2C1, ENABLE); //结束信号
}

/*	向eeprom中写入n个数据*/
void I2C_WriteNumData(u8 addr,u8 *str, u8 numToWrite)
{
	if((addr+numToWrite)>255)//避免写入的数据超过总的内存
	{
		printf("refuse to write\n");
		printf("please enter less than %d char\n",(255-addr));
	}
	else
	{
	 printf("allow to write\n");
	 while(numToWrite)
	 {
		 IIC_WriteByte(addr,*str);
		 addr++;
		 str++;
		 numToWrite--;
	 }
	 printf("write success\n");
 }
}

void I2C_ReadNumData(u8 addr,u8 *str,u8 numToRead)
{
	while(numToRead)
	{
		*str=IIC_ReadByte(addr);
		addr++;
		str++;
		numToRead--;
	}
	
}

usart.c

#include "usart.h"
#include "stm32f10x.h"


void USART1_Init()
{
	
	GPIO_InitTypeDef  GPIO_InitStruct;
	USART_InitTypeDef USART_InitStruct;
	NVIC_InitTypeDef  NVIC_InitStruct;
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA|RCC_APB2Periph_USART1,ENABLE);
	
	GPIO_InitStruct.GPIO_Mode=GPIO_Mode_AF_PP;
	GPIO_InitStruct.GPIO_Pin=GPIO_Pin_9;
	GPIO_InitStruct.GPIO_Speed=GPIO_Speed_50MHz;
	GPIO_Init(GPIOA,&GPIO_InitStruct);
	
	GPIO_InitStruct.GPIO_Mode=GPIO_Mode_IN_FLOATING;
	GPIO_InitStruct.GPIO_Pin=GPIO_Pin_10;
	GPIO_Init(GPIOA,&GPIO_InitStruct);
	
	USART_InitStruct.USART_BaudRate=115200;
	USART_InitStruct.USART_HardwareFlowControl=USART_HardwareFlowControl_None;
	USART_InitStruct.USART_Mode=USART_Mode_Tx|USART_Mode_Rx;
	USART_InitStruct.USART_Parity=USART_Parity_No;
	USART_InitStruct.USART_StopBits=USART_StopBits_1;
	USART_InitStruct.USART_WordLength=USART_WordLength_8b;
	USART_Init(USART1,&USART_InitStruct);	
	USART_Cmd(USART1,ENABLE);
}



void USART1_SendByte(u8 data)
{
	USART_SendData(USART1,data);
	while(!USART_GetFlagStatus(USART1,USART_FLAG_TXE));	
}


void USART1_SendStr(u8 *str)
{
	unsigned int i=0;
	for(i=0;*str!='\0';i++)
	{
		USART1_SendByte(*str);
		str++;
	}
	while(!USART_GetFlagStatus(USART1,USART_FLAG_TC));
}
	
int fputc(int ch, FILE *f)
{
		/* 发送一个字节数据到串口 */
		USART_SendData(USART1, (uint8_t) ch);
		
		/* 等待发送完毕 */
		while (USART_GetFlagStatus(USART1, USART_FLAG_TXE) == RESET);		
	
		return (ch);
}
	

main.c

#include "stm32f10x.h"
#include "usart.h"
#include "iic.h"
#include 

int main()
{
   u8 a[10]={9,8,7,6,5,4,3,2,1,0};
   u8 b[10]={0};
   u8 i=0;
   USART1_Init();
   IIC1_Init();
  	I2C_WriteNumData(125,a,10);
	
	I2C_ReadNumData(125,b,10);
	for(i=0;i<10;i++)
	{
		printf("%d\t",b[i]);
	}
   
}


你可能感兴趣的:(stm32)