参考资料:除《【STM32学习笔记】目录》文章中提到的,还有《AT24C02手册》
/* I2C 初始化结构体 */
typedef struct {
uint32_t I2C_ClockSpeed; // 设置SCL 时钟频率,此值要低于400000
uint16_t I2C_Mode; // 指定工作模式,可选 I2C 模式及 SMBUS 模式
uint16_t I2C_DutyCycle; // 指定时钟占空比,可选 low/high = 2:1 及 16:9 模式
uint16_t I2C_OwnAddress1; // 指定自身的 I2C 设备地址
uint16_t I2C_Ack; // 使能或关闭响应(一般都要使能)
uint16_t I2C_AcknowledgedAddress; // 指定地址的长度,可为 7 位及 10 位
} I2C_InitTypeDef;
野火F103[霸道_V2]开发板
工作的过程和随机读取一样,但是在第 5 个步骤结束后,并不发送非应答信号,而是继续(按上一个数据地址增 1 的地址)读取数据。原则上可以无限的读取,但是AT24C02 只有256个字节,当读到最后一个字节时,将执行**“roll over”**,即从 0 号地址开始继续读取数据。
/* I2C GPIO 引脚定义 */
#define EEPROM_I2C_SCL_GPIO_CLK RCC_APB2Periph_GPIOB
#define EEPROM_I2C_SDA_GPIO_CLK RCC_APB2Periph_GPIOB
#define EEPROM_I2C_GPIO_APBxClkCmd RCC_APB2PeriphClockCmd
#define EEPROM_I2C_SCL_GPIO_MODE GPIOB
#define EEPROM_I2C_SCL_GPIO_PIN GPIO_Pin_6
#define EEPROM_I2C_SDA_GPIO_MODE GPIOB
#define EEPROM_I2C_SDA_GPIO_PIN GPIO_Pin_7
void EEPROM_I2C_Config(void){
GPIO_InitTypeDef GPIO_InitStructure;
/*** 初始化 I2C 需要用到的GPIO ***/
/* 打开 I2C GPIO 的时钟 */
EEPROM_I2C_GPIO_APBxClkCmd(EEPROM_I2C_SCL_GPIO_CLK | EEPROM_I2C_SDA_GPIO_CLK, ENABLE);
/* 将 I2C SCL 的 GPIO 配置为开漏复用输出模式 */
GPIO_InitStructure.GPIO_Pin = EEPROM_I2C_SCL_GPIO_PIN;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_OD;
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_Init(EEPROM_I2C_SCL_GPIO_MODE, &GPIO_InitStructure);
/* 将 I2C SDA 的 GPIO 配置为开漏复用输出模式 */
GPIO_InitStructure.GPIO_Pin = EEPROM_I2C_SDA_GPIO_PIN;
GPIO_Init(EEPROM_I2C_SDA_GPIO_MODE, &GPIO_InitStructure);
}
/* I2C 引脚定义 */
#define EEPROM_I2C I2C1
#define EEPROM_I2C_CLK RCC_APB1Periph_I2C1
#define EEPROM_I2C_APBxClkCmd RCC_APB1PeriphClockCmd
#define EEPROM_I2C_BAUDRATE 400000
#define STM32_I2C_OWN_ADDR 0x5f
void EEPROM_I2C_Config(void){
I2C_InitTypeDef I2C_InitStructure;
/*** 配置 I2C 的工作参数 ***/
/* 打开 I2C 外设的时钟 */
EEPROM_I2C_APBxClkCmd(EEPROM_I2C_CLK, ENABLE);
I2C_InitStructure.I2C_ClockSpeed = EEPROM_I2C_BAUDRATE; // 配置SCL时钟频率
I2C_InitStructure.I2C_Mode = I2C_Mode_I2C; // I2C 模式
I2C_InitStructure.I2C_DutyCycle = I2C_DutyCycle_2; // 时钟占空比,low/high = 2:1
I2C_InitStructure.I2C_OwnAddress1 = STM32_I2C_OWN_ADDR; // STM32 IIC 自身设备地址,只要是总线上唯一的地址即可
I2C_InitStructure.I2C_Ack = I2C_Ack_Enable; // 使能应答
I2C_InitStructure.I2C_AcknowledgedAddress = I2C_AcknowledgedAddress_7bit; // 使用7位地址模式
I2C_Init(EEPROM_I2C, &I2C_InitStructure); // 完成 I2C 的初始化配置
/*** 使能 I2C ***/
I2C_Cmd(EEPROM_I2C, ENABLE);
}
/* 发送一个字节数据 */
void EEPROM_Byte_Write(uint8_t addr, uint8_t data){
/* 产生起始信号 */
I2C_GenerateSTART(EEPROM_I2C, ENABLE);
/* 检验事件 EV5 */
while(I2C_CheckEvent(EEPROM_I2C,I2C_EVENT_MASTER_MODE_SELECT) == ERROR);
/* EV5事件被检测到,发送设备地址 */
I2C_Send7bitAddress(EEPROM_I2C, EEPROM_I2C_ADDR, I2C_Direction_Transmitter);
/* 检测事件 EV6 */
while(I2C_CheckEvent(EEPROM_I2C,I2C_EVENT_MASTER_TRANSMITTER_MODE_SELECTED) == ERROR);
/* EV6事件被检测到,发送要操作的存储单元地址 */
I2C_SendData(EEPROM_I2C, addr);
/* 检测事件 EV8 */
while(I2C_CheckEvent(EEPROM_I2C,I2C_EVENT_MASTER_BYTE_TRANSMITTING) == ERROR);
/* EV8事件被检测到,发送要存储的数据 */
I2C_SendData(EEPROM_I2C, data);
/* 检测事件 EV8_2 */
while(I2C_CheckEvent(EEPROM_I2C,I2C_EVENT_MASTER_BYTE_TRANSMITTED) == ERROR);
/* EV8_2 事件被检测到, 数据传输完成 */
I2C_GenerateSTOP(EEPROM_I2C, ENABLE);
}
/* 发送多个字节数据,按页发送,不能超多 8 个字节 */
void EEPROM_Page_Write(uint8_t addr, uint8_t *data, uint8_t numByteToWrite){
/* 产生起始信号 */
I2C_GenerateSTART(EEPROM_I2C, ENABLE);
/* 检验事件 EV5 */
while(I2C_CheckEvent(EEPROM_I2C,I2C_EVENT_MASTER_MODE_SELECT) == ERROR);
/* EV5事件被检测到,发送设备地址 */
I2C_Send7bitAddress(EEPROM_I2C, EEPROM_I2C_ADDR, I2C_Direction_Transmitter);
/* 检测事件 EV6 */
while(I2C_CheckEvent(EEPROM_I2C,I2C_EVENT_MASTER_TRANSMITTER_MODE_SELECTED) == ERROR);
/* EV6事件被检测到,发送要操作的存储单元地址 */
I2C_SendData(EEPROM_I2C, addr);
/* 检测事件 EV8 */
while(I2C_CheckEvent(EEPROM_I2C,I2C_EVENT_MASTER_BYTE_TRANSMITTING) == ERROR);
/* 发送 numByteToWrite 个字节的数据 */
while(numByteToWrite){
/* EV8事件被检测到,发送要存储的数据 */
I2C_SendData(EEPROM_I2C, *data);
/* 检测事件 EV8_2 */
while(I2C_CheckEvent(EEPROM_I2C,I2C_EVENT_MASTER_BYTE_TRANSMITTED) == ERROR);
numByteToWrite--;
data++;
}
/* EV8_2 事件被检测到, 数据传输完成 */
I2C_GenerateSTOP(EEPROM_I2C, ENABLE);
}
/* 从 EEPROM 中读取数据 */
void EEPROM_Read(uint8_t addr, uint8_t *data, uint8_t numByteToRead){
/*--- 第一次发送起始信号 ---*/
I2C_GenerateSTART(EEPROM_I2C, ENABLE);
/* 检验事件 EV5 */
while(I2C_CheckEvent(EEPROM_I2C, I2C_EVENT_MASTER_MODE_SELECT) == ERROR);
/* EV5事件被检测到,发送设备地址 */
/* 注意:此过程为发送地址 */
I2C_Send7bitAddress(EEPROM_I2C, EEPROM_I2C_ADDR, I2C_Direction_Transmitter);
/* 检测事件 EV6 */
/* 注意:此过程为检测发送数据是否完成 */
while(I2C_CheckEvent(EEPROM_I2C, I2C_EVENT_MASTER_TRANSMITTER_MODE_SELECTED) == ERROR);
/* EV6事件被检测到,发送要操作的存储单元地址 */
I2C_SendData(EEPROM_I2C, addr);
/* 检测事件 EV8 */
while(I2C_CheckEvent(EEPROM_I2C,I2C_EVENT_MASTER_BYTE_TRANSMITTING) == ERROR);
/*--- 第二次发送起始信号 ---*/
I2C_GenerateSTART(EEPROM_I2C, ENABLE);
/* 检验事件 EV5 */
while(I2C_CheckEvent(EEPROM_I2C, I2C_EVENT_MASTER_MODE_SELECT) == ERROR);
/* EV5事件被检测到,发送设备地址 */
/* 注意:此过程为接收地址 */
I2C_Send7bitAddress(EEPROM_I2C, EEPROM_I2C_ADDR, I2C_Direction_Receiver);
/* 检测事件 EV6 */
/* 注意:此过程为检测接收数据是否完成 */
while(I2C_CheckEvent(EEPROM_I2C, I2C_EVENT_MASTER_RECEIVER_MODE_SELECTED) == ERROR);
/* 读取 numByteToRead 个字节数据 */
while(numByteToRead){
/* 如果为最后一个字节,产生非应答信号
* 注意:非应答信号需在读取最后一个数据前产生
*/
if(numByteToRead == 1){
I2C_AcknowledgeConfig(EEPROM_I2C, DISABLE);
}
/* 注意:虽然在通讯图中显示的是先有数据DATA,再产生 EV7 事件。
* 但是读过程需要读取数据寄存器,即当数据寄存器非空时,才能读取。
* 因此需先检测 EV7 事件,判断数据寄存器是否非空,若非空,才可读取数据。
*/
/* 检测事件 EV7 */
while(I2C_CheckEvent(EEPROM_I2C, I2C_EVENT_MASTER_BYTE_RECEIVED ) == ERROR);
/* EV7 事件被检测到,读取数据*/
*data = I2C_ReceiveData(EEPROM_I2C);
data++;
numByteToRead--;
}
/* 数据传输完成,产生停止信号 */
I2C_GenerateSTOP(EEPROM_I2C,ENABLE);
/* 重新配置ACK使能,以便下次通讯 */
I2C_AcknowledgeConfig(EEPROM_I2C, ENABLE);
}
/* 等待EEPROM内部时序完成 */
void EEPROM_WaitForWriteEnd(void){
do{
/* 产生起始信号 */
I2C_GenerateSTART(EEPROM_I2C,ENABLE);
/* 检测 SB 位 */
while(I2C_GetFlagStatus (EEPROM_I2C,I2C_FLAG_SB) == RESET);
/* SB 位为 1,发送设备地址 */
I2C_Send7bitAddress(EEPROM_I2C,EEPROM_I2C_ADDR,I2C_Direction_Transmitter);
} while(I2C_GetFlagStatus (EEPROM_I2C,I2C_FLAG_ADDR) == RESET ); // 检测到 ADDR 位为 1 时停止
/* EEPROM内部时序完成传输完成 */
I2C_GenerateSTOP(EEPROM_I2C,ENABLE);
}
#define ArrNum 10
uint8_t writeData[8] = {
2,3,4,5,6,7,8,9};
uint8_t readData[ArrNum] = {
0};
int main(){
uint8_t i;
USART_Config();
EEPROM_I2C_Config();
printf("I2C--EEPROM 实验 \n");
/* 写 读 1 个字节的数据 */
EEPROM_Byte_Write(11, 1); // 在 11 地址写入数据 1
EEPROM_WaitForWriteEnd(); // 等待写操作完成
EEPROM_Read(11, readData, 1); // 读出 11 地址的数据,读取 1 个字节
printf("%d\n",readData[0]); // 通过串口发送给上位机显示
/* 写 读 8 个字节的数据 */
EEPROM_Page_Write(0, writeData, 8); // 在 8 地址写入数据
EEPROM_WaitForWriteEnd(); // 等待写操作完成
EEPROM_Read(0, readData, 8); // 读出 8 地址的数据,读取 8 个字节
for(i=0;i<8;i++){
printf("%d ", readData[i]);
}
putchar('\n');
while(1){
}
}
文章浏览阅读2w次,点赞7次,收藏51次。四个步骤1.创建C++ Win32项目动态库dll 2.在Win32项目动态库中添加 外部依赖项 lib头文件和lib库3.导出C接口4.c#调用c++动态库开始你的表演...①创建一个空白的解决方案,在解决方案中添加 Visual C++ , Win32 项目空白解决方案的创建:添加Visual C++ , Win32 项目这......_c#调用lib
文章浏览阅读4.6k次。苹方字体是苹果系统上的黑体,挺好看的。注重颜值的网站都会使用,例如知乎:font-family: -apple-system, BlinkMacSystemFont, Helvetica Neue, PingFang SC, Microsoft YaHei, Source Han Sans SC, Noto Sans CJK SC, W..._ubuntu pingfang
文章浏览阅读159次。表单表单概述表单标签表单域按钮控件demo表单标签表单标签基本语法结构<form action="处理数据程序的url地址“ method=”get|post“ name="表单名称”></form><!--action,当提交表单时,向何处发送表单中的数据,地址可以是相对地址也可以是绝对地址--><!--method将表单中的数据传送给服务器处理,get方式直接显示在url地址中,数据可以被缓存,且长度有限制;而post方式数据隐藏传输,_html表单的处理程序有那些
文章浏览阅读1.2k次。使用说明:开启Google的登陆二步验证(即Google Authenticator服务)后用户登陆时需要输入额外由手机客户端生成的一次性密码。实现Google Authenticator功能需要服务器端和客户端的支持。服务器端负责密钥的生成、验证一次性密码是否正确。客户端记录密钥后生成一次性密码。下载谷歌验证类库文件放到项目合适位置(我这边放在项目Vender下面)https://github.com/PHPGangsta/GoogleAuthenticatorPHP代码示例://引入谷_php otp 验证器
文章浏览阅读4.3k次,点赞5次,收藏11次。matplotlib.plot画图横坐标混乱及间隔处理_matplotlib更改横轴间距
文章浏览阅读2.2k次。①Storage driver 处理各镜像层及容器层的处理细节,实现了多层数据的堆叠,为用户 提供了多层数据合并后的统一视图②所有 Storage driver 都使用可堆叠图像层和写时复制(CoW)策略③docker info 命令可查看当系统上的 storage driver主要用于测试目的,不建议用于生成环境。_docker 保存容器
文章浏览阅读834次,点赞27次,收藏13次。网络拓扑结构是指计算机网络中各组件(如计算机、服务器、打印机、路由器、交换机等设备)及其连接线路在物理布局或逻辑构型上的排列形式。这种布局不仅描述了设备间的实际物理连接方式,也决定了数据在网络中流动的路径和方式。不同的网络拓扑结构影响着网络的性能、可靠性、可扩展性及管理维护的难易程度。_网络拓扑csdn
文章浏览阅读1.8k次,点赞5次,收藏8次。IOS系统Date的坑要创建一个指定时间的new Date对象时,通常的做法是:new Date("2020-09-21 11:11:00")这行代码在 PC 端和安卓端都是正常的,而在 iOS 端则会提示 Invalid Date 无效日期。在IOS年月日中间的横岗许换成斜杠,也就是new Date("2020/09/21 11:11:00")通常为了兼容IOS的这个坑,需要做一些额外的特殊处理,笔者在开发的时候经常会忘了兼容IOS系统。所以就想试着重写Date函数,一劳永逸,避免每次ne_date.prototype 将所有 ios
文章浏览阅读5.3k次。方法一:用PLSQL Developer工具。 1 在PLSQL Developer的sql window里输入select * from test for update; 2 按F8执行 3 打开锁, 再按一下加号. 鼠标点到第一列的列头,使全列成选中状态,然后粘贴,最后commit提交即可。(前提..._excel导入pl/sql
文章浏览阅读83次。Git常用命令速查手册1、初始化仓库git init2、将文件添加到仓库git add 文件名 # 将工作区的某个文件添加到暂存区 git add -u # 添加所有被tracked文件中被修改或删除的文件信息到暂存区,不处理untracked的文件git add -A # 添加所有被tracked文件中被修改或删除的文件信息到暂存区,包括untracked的文件...
文章浏览阅读202次。分享119个ASP.NET源码总有一个是你想要的_千博二手车源码v2023 build 1120
文章浏览阅读1.8k次。版权声明:转载请注明出处 http://blog.csdn.net/irean_lau。目录(?)[+]1、缺省构造函数。2、缺省拷贝构造函数。3、 缺省析构函数。4、缺省赋值运算符。5、缺省取址运算符。6、 缺省取址运算符 const。[cpp] view plain copy_空类默认产生哪些类成员函数