博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
nRF52832之硬件I2C
阅读量:5885 次
发布时间:2019-06-19

本文共 3076 字,大约阅读时间需要 10 分钟。

这几天一直在折腾nRF52832的硬件I2C,到了今天最终出现了成果,在此也印证了那句话:“耕耘就有收获”

52832的硬件I2C尽管官方提供了demo,可是自己对I2C通信理解的不够深入,再一个52832的代码也封装的太深了。可是对接口函数没有一个明白的解释(也能够说是我英文太渣,别人写了可是我没看懂。

。),这样对于首次接触nRF产品的人就造成了一定的难度

依据我的开发过程,还是先说明一下I2C的一些相关知识,由于我是先调硬件I2C搞了半天不正确头,然后再开发模拟I2C,模拟的成功了再来调试的硬件TWI(也就是52832的硬件I2C,全称预计是two wire interface)

I2C通信须要两条线:SDA,SCL。

I2C通信设备有两种角色:master和slave。一般用户开发程序都是开发master端,然后去读写作为slave的外设,比方:eeprom,flash,sensor,display device。

在通信过程中,有两组特殊控制信号:

start :scl为高电平时,sda由高电平变为低电平。

stop: scl为高电平时,sda由低电平变为高电平。

(注意在通信过程中。SCL始终由master控制。这句话在做模拟I2C的时候就显得意义非凡了)

master做写数据操作时。先是SCL和SDA都处于空暇状态(两者都是高电平),然后SDA由高变低(start信号);变低后SCL拉低,这个时候SDA就能够变成想要的电平。高电平代表bit为1,低电平代表bit为0。电平稳定后拉高时钟(拉高的目的是为了让slave读取数据,SCL为高时,SDA要保持不变。slave读取SDA的电平)。传输数据完了后要结束:先拉高SCL。然后拉高SDA。然后拉低SDA,一个完整的stop信号完毕了。

读数据操作时,start和stop这些时序一样,可是主机要去解析slave传来的数据(电平信号), 拉低SCL,然后释放SDA(即拉高SDA),一段延时之后拉高SCL再去读取SDA电平信号(既然是读取电平。这里必要设置为输入引脚啦),假设是高电平则记下是一个H_bit,否则是L_bit,读取到8位数据后假设还要继续读取则回复ACK,否则回复NCK。

ACK信号是SCL拉低后给SDA一个低电平,然后拉高SCL;
ANK信号是SCL拉低和给SDA一个高电平。然后拉高SCL。

以下以解说下master 和 slave传输时整体操作:

master向slave写数据,一般slave端写数据都要一个确定的寄存器地址。即你要往这个外设的哪个位置传数据
以eeprom为例,先发送器件地址0xAE(8位数据。高7位是地址。LSB是传输数据方向:0;0代表写。1代表读,能够当做out。in来理解这样easy记住)。
然后发送寄存器地址,然后发送数据。
时序上面能够是
start–slave_address_write–register_address–N*Send_data–stop
Send_data每发一个字节,slave会回应一个“CK信号”。假设是ACK则发送数据成功了。否则失败
由于是连续的写数据,因此中间能够没有stop,start

读数据操作。要先写进一个寄存器地址,再传递一个读命令

start–slave_address_write–register_address–start–slave_address_Read–N*Receive_data–stop

发送slave_address_Read前要先re_start,跟start信号一样

Receive_data 是接收数据,这时要去识别SDA电平而且解析数据,作出ACK回应。最后一个字节接收完毕回复NCK信号;然后stop。

以下说明nRF52832的硬件I2C代码问题

nRF留出的API接口是

ret_code_t app_twi_perform(app_twi_t *  \              p_app_twi,app_twi_transfer_t const * p_transfers,\uint8_t         number_of_transfers,\void (* user_function)(void)\)

这个函数调用了app_twi_schedule函数,以此来导入到队列

ret_code_t app_twi_schedule(app_twi_t *                   p_app_twi,                            app_twi_transaction_t const * p_transaction)

想要调用app_twi_perform函数那么得准备好參数

1、p_app_twi。这是在TWI传输队列里申请一个位置
英文原话是creating an instance of the TWI transaction manager.
2、p_transfers,这是包括了要传输的数组块的一个数组
3、number_of_transfers,这个是你传输数据块的个数
4、user_function,一个用户的回调函数的函数指针,数据块传输完了API内部会调用这个user_function

解释:上面说的数据块的意思就是一个完整的I2C操作须要用到的信息:包括了slave地址。数据传递方向(读或者写),传输的数据data_buffer,数据长度length,有无结束标志(意思就是这团传输数据完了后是否结束通信了)

在官方SDK里面文件夹

examples\peripheral\twi_master_using_app_twi里打开project

首先初始化TWI

初始化
传输数据
传输数据

传输的内容

这里写图片描写叙述
注意AT24C02_init_transfers是一个全局变量数组
也就是它的地址是在堆里面的,不会自己主动释放。这么做的原因是这个数组的地址可能会被多次调用,而放在某个函数里面会造成地址不同造成错误
demo里面有解释

// [these structures have to be "static" - they cannot be placed on stack    //  since the transaction is scheduled and these structures most likely    //  will be referred after this function returns]      static app_twi_transfer_t const transfers[] =    {         AT24C02_READ(&AT24C02_first_page_addr,AT_buffer,5)    };

注意这里的AT_WRITE_NUMBER数组能够理解为一个数据缓冲区,能够通过改变这个数组的内容然后调用app_twi_perform来发数据出去(把const去掉)

读数据相似,demo用的自己定义传输函数

仿写一个
这里写图片描写叙述

这里写图片描写叙述

没想到这篇博客会有这么多人看。写的挺乱的。可是也不想再做改动了~~~写博客真的耗费时间啊~

建议大家去看twi_sensor这个project

路径:NORDIC官方SDK\nRF52_SDK_11.0.0\examples\peripheral\twi_sensor

你可能感兴趣的文章
异类的Javascript处理和解析URL的方式
查看>>
Andrew Ng机器学习公开课笔记 -- 线性回归和梯度下降
查看>>
Webkit内核探究【1】——Webkit简介
查看>>
欧拉计划6-10题
查看>>
Nested transactions in stored procedure of SQLServer
查看>>
SQL Server 2014新特性——基数评估(白皮书阅读笔记)
查看>>
[WinAPI] 串口1-创建[包括: 打不开串口]
查看>>
【jquery】$(document).ready() 与window.onload的区别
查看>>
C#往SQLServer中插入大数据
查看>>
使用UNetbootin制作U盘启动
查看>>
flask 知识点总结
查看>>
ORACLE字符集基础知识
查看>>
<摘录>详谈高性能TCP服务器的开发
查看>>
手机按键控制
查看>>
U3D之Editor扩展学习
查看>>
【C语言】练习1-23
查看>>
一级缓存相关
查看>>
GDB调试之暂停
查看>>
c语言全局变量和局部变量问题汇总
查看>>
scott权限
查看>>