STM32系列(HAL库)——F103C8T6通过IIC/I2C方式读写AT24C02—(EEPROM 存储模块)

1.软件准备

(1)编程平台:Keil5

(2)CubeMX

(3)XCOM(串口调试助手)

2.硬件准备

(1)AT24C02模块

(2)F1的板子,本例使用经典F103C8T6

(3)ST-link 下载器

(4)USB-TTL模块

(5)杜邦线若干

3.模块资料

(1)模块简介:

        AT24C02是一个2K位串行CMOS E2PROM, 内部含有256个8位字节,CATALYST公司的先进CMOS技术实质上减少了器件的功耗。AT24C02有一个8字节页写缓冲器。该器件通过IIC总线接口进行操作,有一个专门的写保护功能。

(2)模块参数:

●宽范围的工作电压1.8V~5.5V
●2线串行接口,完全兼容I2C总线
●内部写周期(最大5 ms)
●可按字节写
●页写:8字节页(24C02)
●擦写寿命:100万次、数据保持时间:100年

更多详细信息参考:AT24CXX手册

4.CubeMX配置

(1)芯片选择

 (2)配置RCC、SYS、时钟树

配置RCC

配置SYS

配置时钟树

(3)配置串口1

 (4)配置IIC

 (5)设置路径、生成代码工程 

5、Keil5代码

(1)勾选Use MicroLIB

(2) usart.c函数添加

/* USER CODE BEGIN 0 */
#include "stdio.h"
/* USER CODE END 0 */
 
 
 
/* USER CODE BEGIN 1 */
/*********************************************************
*
*重定义 fputc 函数
*
*********************************************************/
int fputc(int ch,FILE *f)
{
	HAL_UART_Transmit (&huart1 ,(uint8_t *)&ch,1,HAL_MAX_DELAY );
	return ch;
}
/* USER CODE END 1 */

(3)main函数

#include "stdio.h"
#define ADDR_24LCxx_Write 0xA0    //写24C02的时候,从器件地址为10100000(0xA0)
#define ADDR_24LCxx_Read 0xA1     //读24C02的时候,从器件地址为10100001(0xA1)
#define BufferSize 256      //缓存数组大小
#define Data_Size 8         //每次写入8个字节(64bit)
#define WriteRead_Num BufferSize/Data_Size //需要重复写入的次数   
uint8_t WriteBuffer[BufferSize],ReadBuffer[BufferSize];     //写缓存、读缓存
uint16_t i,num;

//--------------------------------------------------------------------

int main(void)
{
 
  HAL_Init();

  SystemClock_Config();

  MX_GPIO_Init();
  MX_I2C1_Init();
  MX_USART1_UART_Init();
  
        for(uint16_t i=0; i<256; i++)
        WriteBuffer[i]=i;    //初始化写数组,写入0-255,共256个数据
	
		for (num=0; num<WriteRead_Num; num++)
        {
                          //参数分别为:IIC号、设备地址、从机寄存器地址、从机寄存器地址字节长度(8/16)、发送的数据的起始地址、传输数据的大小、操作超时时间 
                if(HAL_I2C_Mem_Write(&hi2c1, ADDR_24LCxx_Write, Data_Size*num, I2C_MEMADD_SIZE_8BIT,WriteBuffer+Data_Size*num,8, 1000) == HAL_OK)                                    
                    HAL_Delay(5);   //每次写入需要间隔至少5ms     
                else
                    break;          //写入失败跳出for循环                           
		}
        
		if(num==WriteRead_Num)  //数据如果全部写入,num会等于WriteRead_Num 
        {
            printf("---------------写入成功---------------\r\n");
                    //参数分别为:IIC号、读的地址、从机寄存器地址、从机寄存器地址字节长度(8/16)、数据缓存区、读出的字节数、操作超时时间 
            HAL_I2C_Mem_Read(&hi2c1, ADDR_24LCxx_Read, 0, I2C_MEMADD_SIZE_8BIT,ReadBuffer,BufferSize, 1000);
            for(i=0; i<256; i++)
            printf("0x%02x  ",ReadBuffer[i]);   //十六进制串口打印
        }
        else 
            printf("---------------写入失败---------------\r\n");
		


  while (1)
  {
    
  }

}

6.接线、效果

STM32 -> PB6 ——  模块SCL

STM32 -> PB7 ——  模块SDA

本例程源码下载:点击跳转 

  • 11
    点赞
  • 59
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 2
    评论
好的,下面是使用STM32 HAL库,通过SPI方式驱动OLED的基本步骤: 1. 初始化SPI接口 在使用SPI接口前需要先对其进行初始化,包括SPI引脚配置、时钟配置和SPI参数配置等,具体代码如下: ```c /*定义SPI句柄*/ SPI_HandleTypeDef hspi1; /*初始化SPI*/ void MX_SPI1_Init(void) { /* SPI1 parameter configuration*/ hspi1.Instance = SPI1; hspi1.Init.Mode = SPI_MODE_MASTER; hspi1.Init.Direction = SPI_DIRECTION_2LINES; hspi1.Init.DataSize = SPI_DATASIZE_8BIT; hspi1.Init.CLKPolarity = SPI_POLARITY_LOW; hspi1.Init.CLKPhase = SPI_PHASE_1EDGE; hspi1.Init.NSS = SPI_NSS_SOFT; hspi1.Init.BaudRatePrescaler = SPI_BAUDRATEPRESCALER_256; hspi1.Init.FirstBit = SPI_FIRSTBIT_MSB; hspi1.Init.TIMode = SPI_TIMODE_DISABLE; hspi1.Init.CRCCalculation = SPI_CRCCALCULATION_DISABLE; hspi1.Init.CRCPolynomial = 10; if (HAL_SPI_Init(&hspi1) != HAL_OK) { Error_Handler(); } } ``` 2. 初始化OLED 在使用OLED前需要对其进行初始化,包括GPIO配置、时钟配置和OLED参数配置等,具体代码如下: ```c /*OLED硬件SPI引脚定义*/ #define OLED_SPI_CS_GPIO_Port GPIOA #define OLED_SPI_CS_Pin GPIO_PIN_4 #define OLED_SPI_DC_GPIO_Port GPIOA #define OLED_SPI_DC_Pin GPIO_PIN_3 /*初始化OLED*/ void OLED_Init(void) { /*配置OLED SPI接口的片选引脚*/ HAL_GPIO_WritePin(OLED_SPI_CS_GPIO_Port, OLED_SPI_CS_Pin, GPIO_PIN_SET); /*配置OLED SPI接口的DC引脚*/ HAL_GPIO_WritePin(OLED_SPI_DC_GPIO_Port, OLED_SPI_DC_Pin, GPIO_PIN_SET); /*配置OLED SPI接口的RESET引脚*/ OLED_RST_Set(); HAL_Delay(100); OLED_RST_Clr(); HAL_Delay(100); OLED_RST_Set(); HAL_Delay(100); /*初始化OLED*/ OLED_WR_Byte(0xAE,OLED_CMD);//--turn off oled panel OLED_WR_Byte(0x00,OLED_CMD);//---set low column address OLED_WR_Byte(0x10,OLED_CMD);//---set high column address OLED_WR_Byte(0x40,OLED_CMD);//--set start line address Set Mapping RAM Display Start Line (0x00~0x3F) OLED_WR_Byte(0x81,OLED_CMD);//--set contrast control register OLED_WR_Byte(0xCF,OLED_CMD); // Set SEG Output Current Brightness OLED_WR_Byte(0xA1,OLED_CMD);//--Set SEG/Column Mapping 0xa0左右反置 0xa1正常 OLED_WR_Byte(0xC8,OLED_CMD);//Set COM/Row Scan Direction 0xc0上下反置 0xc8正常 OLED_WR_Byte(0xA6,OLED_CMD);//--set normal display OLED_WR_Byte(0xA8,OLED_CMD);//--set multiplex ratio(1 to 64) OLED_WR_Byte(0x3f,OLED_CMD);//--1/64 duty OLED_WR_Byte(0xD3,OLED_CMD);//-set display offset Shift Mapping RAM Counter (0x00~0x3F) OLED_WR_Byte(0x00,OLED_CMD);//-not offset OLED_WR_Byte(0xd5,OLED_CMD);//--set display clock divide ratio/oscillator frequency OLED_WR_Byte(0x80,OLED_CMD);//--set divide ratio, Set Clock as 100 Frames/Sec OLED_WR_Byte(0xD9,OLED_CMD);//--set pre-charge period OLED_WR_Byte(0xF1,OLED_CMD);//Set Pre-Charge as 15 Clocks & Discharge as 1 Clock OLED_WR_Byte(0xDA,OLED_CMD);//--set com pins hardware configuration OLED_WR_Byte(0x12,OLED_CMD); OLED_WR_Byte(0xDB,OLED_CMD);//--set vcomh OLED_WR_Byte(0x40,OLED_CMD);//Set VCOM Deselect Level OLED_WR_Byte(0x20,OLED_CMD);//-Set Page Addressing Mode (0x00/0x01/0x02) OLED_WR_Byte(0x02,OLED_CMD);// OLED_WR_Byte(0x8D,OLED_CMD);//--set Charge Pump enable/disable OLED_WR_Byte(0x14,OLED_CMD);//--set(0x10) disable OLED_WR_Byte(0xA4,OLED_CMD);// Disable Entire Display On (0xa4/0xa5) OLED_WR_Byte(0xA6,OLED_CMD);// Disable Inverse Display On (0xa6/a7) OLED_WR_Byte(0xAF,OLED_CMD);//--turn on oled panel OLED_Clear(); } ``` 3. 实现SPI传输函数 SPI传输函数是使用SPI接口与OLED进行数据传输的函数,具体代码如下: ```c /*OLED硬件SPI引脚定义*/ #define OLED_SPI_CS_GPIO_Port GPIOA #define OLED_SPI_CS_Pin GPIO_PIN_4 #define OLED_SPI_DC_GPIO_Port GPIOA #define OLED_SPI_DC_Pin GPIO_PIN_3 /*OLED写数据*/ void OLED_WR_Byte(uint8_t dat,uint8_t cmd) { if(cmd) { HAL_GPIO_WritePin(OLED_SPI_DC_GPIO_Port, OLED_SPI_DC_Pin, GPIO_PIN_SET); } else { HAL_GPIO_WritePin(OLED_SPI_DC_GPIO_Port, OLED_SPI_DC_Pin, GPIO_PIN_RESET); } /*使能OLED*/ HAL_GPIO_WritePin(OLED_SPI_CS_GPIO_Port, OLED_SPI_CS_Pin, GPIO_PIN_RESET); /*通过SPI发送数据*/ HAL_SPI_Transmit(&hspi1,&dat,1,1000); /*关闭OLED*/ HAL_GPIO_WritePin(OLED_SPI_CS_GPIO_Port, OLED_SPI_CS_Pin, GPIO_PIN_SET); } /*OLED写数据*/ void OLED_WR_LByte(uint8_t dat,uint8_t cmd) { uint8_t data[2]; data[0] = dat; data[1] = 0x00; if(cmd) { HAL_GPIO_WritePin(OLED_SPI_DC_GPIO_Port, OLED_SPI_DC_Pin, GPIO_PIN_SET); } else { HAL_GPIO_WritePin(OLED_SPI_DC_GPIO_Port, OLED_SPI_DC_Pin, GPIO_PIN_RESET); } /*使能OLED*/ HAL_GPIO_WritePin(OLED_SPI_CS_GPIO_Port, OLED_SPI_CS_Pin, GPIO_PIN_RESET); /*通过SPI发送数据*/ HAL_SPI_Transmit(&hspi1,data,2,1000); /*关闭OLED*/ HAL_GPIO_WritePin(OLED_SPI_CS_GPIO_Port, OLED_SPI_CS_Pin, GPIO_PIN_SET); } ``` 4. 实现OLED显示函数 OLED显示函数是向OLED发送数据实现显示的函数,具体代码如下: ```c /*OLED写数据*/ void OLED_WR_BP(uint8_t dat) { uint8_t i; for(i=0;i<8;i++) { if(dat&0x80) OLED_WR_Byte(0xff,OLED_DATA); else OLED_WR_Byte(0x00,OLED_DATA); dat<<=1; } } /*清屏*/ void OLED_Clear(void) { uint8_t i,n; for(i=0;i<8;i++) { OLED_WR_Byte(0xb0+i,OLED_CMD); OLED_WR_Byte(0x00,OLED_CMD); OLED_WR_Byte(0x10,OLED_CMD); for(n=0;n<128;n++)OLED_WR_Byte(0,OLED_DATA); } } ``` 5. 调用OLED显示函数 初始化完成后,就可以调用OLED显示函数来实现向OLED发送数据实现显示了,具体代码如下: ```c int main(void) { /*初始化*/ HAL_Init(); MX_SPI1_Init(); OLED_Init(); /*显示*/ OLED_Clear(); OLED_ShowString(0,0,"Hello World!",16); while (1) { } } ``` 以上就是通过SPI接口驱动OLED的基本步骤,需要注意的是具体的硬件引脚配置和参数配置需要根据实际情况进行修改。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

嵌入式创客工坊

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

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

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

打赏作者

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

抵扣说明:

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

余额充值