参考资料:除《【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){
}
}
文章浏览阅读1.6k次。安装配置gi、安装数据库软件、dbca建库见下:http://blog.csdn.net/kadwf123/article/details/784299611、检查集群节点及状态:[root@rac2 ~]# olsnodes -srac1 Activerac2 Activerac3 Activerac4 Active[root@rac2 ~]_12c查看crs状态
文章浏览阅读1.3w次,点赞45次,收藏99次。我个人用的是anaconda3的一个python集成环境,自带jupyter notebook,但在我打开jupyter notebook界面后,却找不到对应的虚拟环境,原来是jupyter notebook只是通用于下载anaconda时自带的环境,其他环境要想使用必须手动下载一些库:1.首先进入到自己创建的虚拟环境(pytorch是虚拟环境的名字)activate pytorch2.在该环境下下载这个库conda install ipykernelconda install nb__jupyter没有pytorch环境
文章浏览阅读5.2k次,点赞19次,收藏28次。选择scoop纯属意外,也是无奈,因为电脑用户被锁了管理员权限,所有exe安装程序都无法安装,只可以用绿色软件,最后被我发现scoop,省去了到处下载XXX绿色版的烦恼,当然scoop里需要管理员权限的软件也跟我无缘了(譬如everything)。推荐添加dorado这个bucket镜像,里面很多中文软件,但是部分国外的软件下载地址在github,可能无法下载。以上两个是官方bucket的国内镜像,所有软件建议优先从这里下载。上面可以看到很多bucket以及软件数。如果官网登陆不了可以试一下以下方式。_scoop-cn
文章浏览阅读4.5k次,点赞2次,收藏3次。首先要有一个color-picker组件 <el-color-picker v-model="headcolor"></el-color-picker>在data里面data() { return {headcolor: ’ #278add ’ //这里可以选择一个默认的颜色} }然后在你想要改变颜色的地方用v-bind绑定就好了,例如:这里的:sty..._vue el-color-picker
文章浏览阅读640次。基于芯片日益增长的问题,所以内核开发者们引入了新的方法,就是在内核中只保留函数,而数据则不包含,由用户(应用程序员)自己把数据按照规定的格式编写,并放在约定的地方,为了不占用过多的内存,还要求数据以根精简的方式编写。boot启动时,传参给内核,告诉内核设备树文件和kernel的位置,内核启动时根据地址去找到设备树文件,再利用专用的编译器去反编译dtb文件,将dtb还原成数据结构,以供驱动的函数去调用。firmware是三星的一个固件的设备信息,因为找不到固件,所以内核启动不成功。_exynos 4412 刷机
文章浏览阅读2w次,点赞24次,收藏42次。Linux系统配置jdkLinux学习教程,Linux入门教程(超详细)_linux配置jdk
文章浏览阅读3.3k次,点赞5次,收藏19次。xlabel('\delta');ylabel('AUC');具体符号的对照表参照下图:_matlab微米怎么输入
文章浏览阅读119次。顺序读写指的是按照文件中数据的顺序进行读取或写入。对于文本文件,可以使用fgets、fputs、fscanf、fprintf等函数进行顺序读写。在C语言中,对文件的操作通常涉及文件的打开、读写以及关闭。文件的打开使用fopen函数,而关闭则使用fclose函数。在C语言中,可以使用fread和fwrite函数进行二进制读写。 Biaoge 于2024-03-09 23:51发布 阅读量:7 ️文章类型:【 C语言程序设计 】在C语言中,用于打开文件的函数是____,用于关闭文件的函数是____。
文章浏览阅读3.4k次,点赞2次,收藏13次。跟随鼠标移动的粒子以grid(SOP)为partical(SOP)的资源模板,调整后连接【Geo组合+point spirit(MAT)】,在连接【feedback组合】适当调整。影响粒子动态的节点【metaball(SOP)+force(SOP)】添加mouse in(CHOP)鼠标位置到metaball的坐标,实现鼠标影响。..._touchdesigner怎么让一个模型跟着鼠标移动
文章浏览阅读178次。项目运行环境配置:Jdk1.8 + Tomcat7.0 + Mysql + HBuilderX(Webstorm也行)+ Eclispe(IntelliJ IDEA,Eclispe,MyEclispe,Sts都支持)。项目技术:Springboot + mybatis + Maven +mysql5.7或8.0+html+css+js等等组成,B/S模式 + Maven管理等等。环境需要1.运行环境:最好是java jdk 1.8,我们在这个平台上运行的。其他版本理论上也可以。_基于java技术的停车场管理系统实现与设计
文章浏览阅读3.5k次。前言对于MediaPlayer播放器的源码分析内容相对来说比较多,会从Java-&amp;gt;Jni-&amp;gt;C/C++慢慢分析,后面会慢慢更新。另外,博客只作为自己学习记录的一种方式,对于其他的不过多的评论。MediaPlayerDemopublic class MainActivity extends AppCompatActivity implements SurfaceHolder.Cal..._android多媒体播放源码分析 时序图
文章浏览阅读2.4k次,点赞41次,收藏13次。java 数据结构与算法 ——快速排序法_快速排序法