GD32F407之硬件IIC(从机模式)

2024-02-08 15:40

本文主要是介绍GD32F407之硬件IIC(从机模式),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

承接上一篇GD32F407硬件IIC主机模式,下面这一片介绍GD32F407硬件IIC从机模式,用MCU来做从机模式百度上有用的资源比较少,都是STM32里面的源码,千篇一律,有点水帖的感觉。

网上百度用GPIO模拟方式来做从机好像没有找到资料,也咨询了GD32F407的FE没有做过GPIO模拟从机,所以就用硬件方式来,官方源码这一次终于不是while结构了,而且官方还是IIC0做主机,IIC1从机的方式,真的太给力了。

1、从机接收模式下的软件流程

2、代码,官方源码从机发送模式

/*!\brief      handle I2C1 event interrupt request\param[in]  none\param[out] none\retval     none
*/
void I2C1_EventIRQ_Handler(void)
{if(i2c_flag_get(I2C1, I2C_ADDSEND)){/* clear the ADDSEND bit */i2c_flag_clear(I2C1, I2C_STAT0_ADDSEND);}else if((i2c_flag_get(I2C1, I2C_TBE))&&(!i2c_flag_get(I2C1, I2C_AERR))){/* send a data byte */i2c_transmit_data(I2C1, *i2c_txbuffer++);}
}

是不是感觉太少了,有点怀疑可行性啊,单独测试官方源码可以使用,但是拿到项目就不能用了,修改如下,加了一个清除ADDSEND bit后的标志,让它必须清除ADDSEND后才能下一步

void Si2c_Send_Data(uint32_t i2c_periph,BYTE Channel)
{/* wait until ADDSEND bit is set */if(i2c_flag_get(i2c_periph, I2C_ADDSEND)){ADDSEND_FLAG=1;/* clear ADDSEND bit */i2c_flag_clear(i2c_periph, I2C_ADDSEND);}       if(ADDSEND_FLAG){if(i2c_flag_get(i2c_periph, I2C_RBNE)){/*  reception data register  */SReceAddrBuffer[Channel] = i2c_receive_data(i2c_periph);}   }
}

也是很简单的

3、从机发送模式下的软件流程

4、从机发送模式代码,官方源码

*!\brief      handle I2C1 event interrupt request\param[in]  none\param[out] none\retval     none
*/
void I2C1_EventIRQ_Handler(void)
{if(i2c_flag_get(I2C1, I2C_ADDSEND)){/* clear the ADDSEND bit */i2c_flag_clear(I2C1, I2C_STAT0_ADDSEND);}else if(i2c_flag_get(I2C1, I2C_RBNE)){/* if reception data register is not empty ,I2C1 will read a data from I2C_DATA */*i2c_rxbuffer++ = i2c_receive_data(I2C1);}else if(i2c_flag_get(I2C1, I2C_STPDET)){Status = SUCCESS;/* clear the STPDET bit */i2c_enable(I2C1);/* disable I2C1 interrupt */i2c_interrupt_disable(I2C1, I2C_CTL1_ERRIE | I2C_CTL1_BUFIE | I2C_CTL1_EVIE);}
}

也是很简单,直接测试没有问题,

将发送和接收数据整合在一起,整合一定要注意接收和发送的标志位不同,

oid Si2c_Transfer_Data(uint32_t i2c_periph,BYTE Channel)
{/* wait until ADDSEND bit is set */if(i2c_flag_get(i2c_periph, I2C_ADDSEND)){ADDSEND_FLAG=1;/* clear ADDSEND bit */i2c_flag_clear(i2c_periph, I2C_ADDSEND);}if(ADDSEND_FLAG){if(i2c_flag_get(i2c_periph, I2C_RBNE) && (SRFalg ==0)){/*  Receive register  addr*/SReceAddrBuffer[Channel] = i2c_receive_data(i2c_periph);SRFalg = 1;}if(SReceAddrBuffer[Channel] != 0XFF){if(i2c_flag_get(i2c_periph, I2C_TBE)){/* Send a word data */MatchAdd_SendData(i2c_periph,SReceAddrBuffer[Channel]);ADDSEND_FLAG=0;SRFalg = 0;}if((SRFalg==1) && i2c_flag_get(i2c_periph, I2C_RBNE)){/*Receive a byte data */SReceDataBuffer[Channel] = i2c_receive_data(i2c_periph);MatchAdd_ReceData(SReceAddrBuffer[Channel],SReceDataBuffer[Channel]);}if(i2c_flag_get(i2c_periph, I2C_STPDET)){/* Receive mode clear the STPDET bit */ic20Flag = 2;SRFalg = 0;ADDSEND_FLAG=0;i2c_enable(i2c_periph);i2c_interrupt_disable(i2c_periph, I2C_CTL1_ERRIE | I2C_CTL1_BUFIE | I2C_CTL1_EVIE);}if(ic20Flag == 2){i2c_interrupt_enable(i2c_periph, I2C_CTL1_ERRIE | I2C_CTL1_BUFIE | I2C_CTL1_EVIE);ic20Flag = 0;}}}
}

注意:首先我们知道,,一般的IIC从设备都是有很多寄存器地址的,所以我们主机肯定也会发一个寄存器地址过来,官方源码里面是没有做处理的就当做(0x00),所以在整合的代码里面我有两次接收数据一个是寄存器地址放到地址数组和一个是数据放到了数据数组中。

5、接收发送处理函数

接收一个WORD数据

void MatchAdd_ReceData(BYTE Cmd,BYTE Data)
{ if (Cmd == 0x02){   /* Currently system power state. */PwrState.byte = Data;} else if (Cmd == 0xB3){   /* button event state */ButtState.byte = Data;}else if (Cmd == 0xB4){   /* button event number */SCINumber = Data;}else if (Cmd == 0xB5){  HeartBeatEnable = Data;} else if (Cmd == 0xB6){  HeartBeatTimer = Data;}else if (Cmd == 0xB7){  HeartBeatFlag = Data;}
}

发送一个WORD数据

void MatchAdd_SendData(uint32_t i2c_periph ,BYTE Cmd)
{WORD rval = 0;WORD *pntr;if (Cmd == 0x02){   /* Currently system power state. */rval = (WORD) PwrState.byte;} else if (Cmd == 0xB3){   /* button event state */rval = (WORD) ButtState.byte;}else if (Cmd == 0xB4){   /* button sci event number */rval = (WORD) SCINumber;}i2c_transmit_worddata(i2c_periph,rval);
}

注意:接受和发送的寄存器其实是同一个地址,通俗说就是从机有一个0x00--0xFF的地址的寄存器来让主机读写

官方里面没有发送一个Word数据函数,修改如下

void i2c_transmit_worddata(uint32_t i2c_periph, uint16_t Data)
{/* send a data byte */I2C_DATA(i2c_periph) = Data;I2C_DATA(i2c_periph) = Data>>8;
}

6、中断函数

void I2C0_EV_IRQHandler(void)
{//DEBUG(0XAA);Si2c_Transfer_Data(I2C0,0);
}

到此GD32F407硬件IIC就结束了,其实硬件模式的从机还是比主机简单很多,主要是文档非常详细。

 

这篇关于GD32F407之硬件IIC(从机模式)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/691484

相关文章

在JS中的设计模式的单例模式、策略模式、代理模式、原型模式浅讲

1. 单例模式(Singleton Pattern) 确保一个类只有一个实例,并提供一个全局访问点。 示例代码: class Singleton {constructor() {if (Singleton.instance) {return Singleton.instance;}Singleton.instance = this;this.data = [];}addData(value)

【STM32】SPI通信-软件与硬件读写SPI

SPI通信-软件与硬件读写SPI 软件SPI一、SPI通信协议1、SPI通信2、硬件电路3、移位示意图4、SPI时序基本单元(1)开始通信和结束通信(2)模式0---用的最多(3)模式1(4)模式2(5)模式3 5、SPI时序(1)写使能(2)指定地址写(3)指定地址读 二、W25Q64模块介绍1、W25Q64简介2、硬件电路3、W25Q64框图4、Flash操作注意事项软件SPI读写W2

模版方法模式template method

学习笔记,原文链接 https://refactoringguru.cn/design-patterns/template-method 超类中定义了一个算法的框架, 允许子类在不修改结构的情况下重写算法的特定步骤。 上层接口有默认实现的方法和子类需要自己实现的方法

【iOS】MVC模式

MVC模式 MVC模式MVC模式demo MVC模式 MVC模式全称为model(模型)view(视图)controller(控制器),他分为三个不同的层分别负责不同的职责。 View:该层用于存放视图,该层中我们可以对页面及控件进行布局。Model:模型一般都拥有很好的可复用性,在该层中,我们可以统一管理一些数据。Controlller:该层充当一个CPU的功能,即该应用程序

迭代器模式iterator

学习笔记,原文链接 https://refactoringguru.cn/design-patterns/iterator 不暴露集合底层表现形式 (列表、 栈和树等) 的情况下遍历集合中所有的元素

《x86汇编语言:从实模式到保护模式》视频来了

《x86汇编语言:从实模式到保护模式》视频来了 很多朋友留言,说我的专栏《x86汇编语言:从实模式到保护模式》写得很详细,还有的朋友希望我能写得更细,最好是覆盖全书的所有章节。 毕竟我不是作者,只有作者的解读才是最权威的。 当初我学习这本书的时候,只能靠自己摸索,网上搜不到什么好资源。 如果你正在学这本书或者汇编语言,那你有福气了。 本书作者李忠老师,以此书为蓝本,录制了全套视频。 试

利用命令模式构建高效的手游后端架构

在现代手游开发中,后端架构的设计对于支持高并发、快速迭代和复杂游戏逻辑至关重要。命令模式作为一种行为设计模式,可以有效地解耦请求的发起者与接收者,提升系统的可维护性和扩展性。本文将深入探讨如何利用命令模式构建一个强大且灵活的手游后端架构。 1. 命令模式的概念与优势 命令模式通过将请求封装为对象,使得请求的发起者和接收者之间的耦合度降低。这种模式的主要优势包括: 解耦请求发起者与处理者

springboot实战学习(1)(开发模式与环境)

目录 一、实战学习的引言 (1)前后端的大致学习模块 (2)后端 (3)前端 二、开发模式 一、实战学习的引言 (1)前后端的大致学习模块 (2)后端 Validation:做参数校验Mybatis:做数据库的操作Redis:做缓存Junit:单元测试项目部署:springboot项目部署相关的知识 (3)前端 Vite:Vue项目的脚手架Router:路由Pina:状态管理Eleme

状态模式state

学习笔记,原文链接 https://refactoringguru.cn/design-patterns/state 在一个对象的内部状态变化时改变其行为, 使其看上去就像改变了自身所属的类一样。 在状态模式中,player.getState()获取的是player的当前状态,通常是一个实现了状态接口的对象。 onPlay()是状态模式中定义的一个方法,不同状态下(例如“正在播放”、“暂停

软件架构模式:5 分钟阅读

原文: https://orkhanscience.medium.com/software-architecture-patterns-5-mins-read-e9e3c8eb47d2 软件架构模式:5 分钟阅读 当有人潜入软件工程世界时,有一天他需要学习软件架构模式的基础知识。当我刚接触编码时,我不知道从哪里获得简要介绍现有架构模式的资源,这样它就不会太详细和混乱,而是非常抽象和易