STM32驱动SPI接口EEPROM,AT25010、AT25020、AT25040

一、源代码

void eeprom_init()
{
	GPIO_InitTypeDef GPIO_InitStructure;
	 SPI_InitTypeDef  SPI_InitStructure;
	RCC_APB2PeriphClockCmd(	RCC_APB2Periph_GPIOB, ENABLE );//PORTB时钟使能 

	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_11|GPIO_Pin_12;  // PB12 推挽 
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;  //推挽输出
	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
	GPIO_Init(GPIOB, &GPIO_InitStructure);
	GPIO_SetBits(GPIOB,GPIO_Pin_11|GPIO_Pin_12);



	RCC_APB2PeriphClockCmd(	RCC_APB2Periph_GPIOB, ENABLE );//PORTB时钟使能 
	RCC_APB1PeriphClockCmd(	RCC_APB1Periph_SPI2,  ENABLE );//SPI2时钟使能 	
 
	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_13 | GPIO_Pin_14 | GPIO_Pin_15;
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;  //PB13/14/15复用推挽输出 
	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
	GPIO_Init(GPIOB, &GPIO_InitStructure);//初始化GPIOB

 	GPIO_SetBits(GPIOB,GPIO_Pin_13|GPIO_Pin_14|GPIO_Pin_15);  //PB13/14/15上拉

	SPI_InitStructure.SPI_Direction = SPI_Direction_2Lines_FullDuplex;  //设置SPI单向或者双向的数据模式:SPI设置为双线双向全双工
	SPI_InitStructure.SPI_Mode = SPI_Mode_Master;		//设置SPI工作模式:设置为主SPI
	SPI_InitStructure.SPI_DataSize = SPI_DataSize_8b;		//设置SPI的数据大小:SPI发送接收8位帧结构
	SPI_InitStructure.SPI_CPOL = SPI_CPOL_High;		//串行同步时钟的空闲状态为高电平
	SPI_InitStructure.SPI_CPHA = SPI_CPHA_2Edge;	//串行同步时钟的第二个跳变沿(上升或下降)数据被采样
	SPI_InitStructure.SPI_NSS = SPI_NSS_Soft;		//NSS信号由硬件(NSS管脚)还是软件(使用SSI位)管理:内部NSS信号有SSI位控制
	SPI_InitStructure.SPI_BaudRatePrescaler = SPI_BaudRatePrescaler_256;		//定义波特率预分频的值:波特率预分频值为256
	SPI_InitStructure.SPI_FirstBit = SPI_FirstBit_MSB;	//指定数据传输从MSB位还是LSB位开始:数据传输从MSB位开始
	SPI_InitStructure.SPI_CRCPolynomial = 7;	//CRC值计算的多项式
	SPI_Init(SPI2, &SPI_InitStructure);  //根据SPI_InitStruct中指定的参数初始化外设SPIx寄存器
 
	SPI_Cmd(SPI2, ENABLE); //使能SPI外设
	
	eeprom_read_write_byte(0xff);//启动传输		
}
//SPI 速度设置函数
//SpeedSet:
//SPI_BaudRatePrescaler_2   2分频   
//SPI_BaudRatePrescaler_8   8分频   
//SPI_BaudRatePrescaler_16  16分频  
//SPI_BaudRatePrescaler_256 256分频 
  
void eeprom_set_speed(u8 SPI_BaudRatePrescaler)
{
  assert_param(IS_SPI_BAUDRATE_PRESCALER(SPI_BaudRatePrescaler));
	SPI2->CR1&=0XFFC7;
	SPI2->CR1|=SPI_BaudRatePrescaler;	//设置SPI2速度 
	SPI_Cmd(SPI2,ENABLE); 

} 
//SPIx 读写一个字节
//TxData:要写入的字节
//返回值:读取到的字节
u8 eeprom_read_write_byte(u8 TxData)
{		
	u8 retry=0;				 	
	while (SPI_I2S_GetFlagStatus(SPI2, SPI_I2S_FLAG_TXE) == RESET) //检查指定的SPI标志位设置与否:发送缓存空标志位
		{
		retry++;
		if(retry>200)return 0;
		}			  
	SPI_I2S_SendData(SPI2, TxData); //通过外设SPIx发送一个数据
	retry=0;

	while (SPI_I2S_GetFlagStatus(SPI2, SPI_I2S_FLAG_RXNE) == RESET) //检查指定的SPI标志位设置与否:接受缓存非空标志位
		{
		retry++;
		if(retry>200)return 0;
		}	  						    
	return SPI_I2S_ReceiveData(SPI2); //返回通过SPIx最近接收的数据					    
}
//==============================================================================
// 描述: 在AT25读状态寄存器
// 输入: 无
// 返回: unsigned char dat 状态寄存器数据
//==============================================================================


u8 eeprom_read_sr(void)
{ 
	u8 dat; 
	eeprom_cs=0; 
	eeprom_read_write_byte(RDSR);        //  定入指令0x05
	dat = eeprom_read_write_byte(0xff);       //  读回数据
	eeprom_cs=1;
	return  dat;
}

//==============================================================================
// 描述: 向AT25写入一个数据
// 输入: unsigned char Dat 字符数据
//       unsigned int addr 写入的地址
// 返回: 无
//==============================================================================
void eeprom_write_byte(u16 addr,u8 Dat)
{ 
	unsigned char Add; 
	while(eeprom_read_sr()&RDY);                   // 读状态寄存器
	
	eeprom_cs=0;
	eeprom_read_write_byte(WREN);                           // 写使能锁存
	eeprom_cs=1;
	
	eeprom_cs=0;                                  // 芯片使能 
	// 写入写操作指令 
	if(addr>0x00ff)
		eeprom_read_write_byte((WRITE|0x08));           // 地址大于255
	else
		eeprom_read_write_byte(WRITE);                  // 地址小于255

	// 写入地址
	Add = (unsigned char)(addr & 0xff);             // 将地址换成8位
	eeprom_read_write_byte(Add); 

	// 写入数据
	eeprom_read_write_byte(Dat);
	eeprom_cs=1;                                // 芯片关闭    
}
//==============================================================================
// 描述: 在AT25读一个字节操作
// 输入: 无
// 返回: unsigned char dat 读出一个字符数据
//==============================================================================
u8 eeprom_read_byte(u16 addr)
{ 
	u8 dat,add;
	while(eeprom_read_sr()&RDY);                   // 读状态寄存器  
	eeprom_cs=0;
	// 写入写操作指令 
	if(addr>0x00ff)
		eeprom_read_write_byte((READ | 0x08));         // 地址大于255
	else
		eeprom_read_write_byte(READ);                  // 地址小于255

	// 写入地址
	add = (unsigned char)(addr & 0xff);             // 将地址换成8位
	eeprom_read_write_byte(add); 

	// 读出一个数据
	dat=eeprom_read_write_byte(0xff);
	eeprom_cs=1;
	return dat;
}
//==============================================================================
// 描述: 向AT25连续写入数据
// 输入: unsigned char *PC 写入数据指针
//       unsigned int count 写入数量计数  
//       unsigned int SatAddr 写入起始地址
// 返回: 无
//==============================================================================
void eeprom_write_buf(u16 startAddr,u8 *buf,u16 count)
{ 
	u8 Add; 
	u16 i;
	while(eeprom_read_sr()&RDY);                   // 读状态寄存器

	eeprom_cs=0;
	eeprom_read_write_byte(WREN);                           // 写使能锁存
	eeprom_cs=1;
	eeprom_cs=0;
	// 写入写操作指令 
	if(startAddr>0x00ff)
		eeprom_read_write_byte((WRITE|0x08));           // 地址大于255
	else
		eeprom_read_write_byte(WRITE);                  // 地址小于255

	// 写入地址
	Add = (unsigned char)(startAddr & 0xff);             // 将地址换成8位
	eeprom_read_write_byte(Add); 

	for(i=0;i<count;i++)// 写入数据
	{ 
		eeprom_read_write_byte(*buf++);
	}
	eeprom_cs=1;                                 // 芯片关闭    
}
//==============================================================================
// 描述: 在AT25连续读数据
// 输入: unsigned char *PC 读出数据指针
//       unsigned int count 读出数量计数  
//       unsigned int SatAddr 读出起始地址
// 返回: 无
//==============================================================================
void eeprom_read_buf(u16 startAddr,u8 *buf,u16 count)
{ 
	unsigned char Add; 
	unsigned int i;
	while(eeprom_read_sr()&RDY);                   // 读状态寄存器  
	eeprom_cs=0;
	// 写入写操作指令 
	if(startAddr>0x00ff)
		eeprom_read_write_byte((READ | 0x08));         // 地址大于255
	else
		eeprom_read_write_byte(READ);                  // 地址小于255

	// 写入地址
	Add = (unsigned char)(startAddr & 0xff);             // 将地址换成8位
	eeprom_read_write_byte(Add); 

	// 读出n个数据
	for(i=0;i<count;i++)
	{
		*buf++ = eeprom_read_write_byte(0xff);
	}
	eeprom_cs=1;
}

二、测试

int main(void)
{	
	u8 buf[5]={0x01,0x02,0x03,0x04,0x05},i;
	uart_init(115200);
	delay_init();	
	program_led_init();	
	
	eeprom_wp=1;
	eeprom_init();
	eeprom_write_buf(0x0000,buf,sizeof(buf));
//	eeprom_write_byte(0x0000,0x08);
	memset(buf,0x00,sizeof(buf));
	eeprom_wp=0;
	while(1)
	{	
		
		eeprom_read_buf(0x0000,buf,sizeof(buf));
	//	buf[0]=eeprom_read_byte(0x0001);
		for(i=0;i<sizeof(buf);i++)
			printf("0x%02x ",buf[i]);
		printf("\r\n");
		memset(buf,0x00,sizeof(buf));
		
		program_led_data(0x10,0x10,0x10);
		delay_ms(500);

		program_led_data(0x00,0x00,0x00);
		delay_ms(500);	
		
		
	}
}

三、原理图
暂留

源代码链接

猜你喜欢

转载自blog.csdn.net/qq_15181569/article/details/106310411