在全志H616核桃派开发板上配置I2C引脚并读取温度数据

配置引脚

找到板子上的i2c引脚

为了方便查找,我们加入了一个显示功能引脚位置的功能,运行以下命令,查看板子的40pin引脚上有几个可用i2c

gpio pin i2c

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

启用i2c

我们使用set-device指令来使能/关闭指定设备的底层驱动,使能后,引脚就会由gpio模式切换为对应的引脚复用功能。(配置后要重启才能生效)

首先查看各设备的状态

set-device status

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

运行指令启用i2c1,注意要重启后才能生效

sudo set-device enable i2c1

重启后查看引脚状态,可以看到3和5都处于alt5了 外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

并且存在/dev/i2c-1这个文件,因为后续我们需要通过操作这个文件来控制i2c通讯

i2c读写程序

linux下一切皆是文件。而i2c1也被抽象为/dev/i2c-1这个文件。通过open打开它,用ioctl触发读写,close关闭文件。

1. 打开文件

linux下一切皆是文件,先使用open函数打开我们要操作的设备对应文件,获取文件描述符。

需要这些头文件

#include<sys/types.h>
#include<sys/stat.h>
#include<fcntl.h>
#include<unistd.h>

open设备节点

int fd =open("/dev/i2c-1", O_RDWR);
if(fd <0)
{
perror("Fail to Open\n");
return-1;
}

2. i2c_msg

linux下操作i2c不是使用write和read函数,而是使用一个i2c_msg结构体来配置从i2c从起始到停止之间要做的内容。

  • addr: 目标地址
  • flags: 读还是写
  • buf : buf的地址,根据flags是读还是写,会在地址帧发送后,将其内容发送出去,或是读取总线内容存放进来。
  • len : buf的大小

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

3. 向i2c总线写

从德州仪器那下载的i2c时序图↓ 外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

加入我现在想往地址是0x3c的设备上,把寄存器0x01赋值为0x55,那msg结构体的设置就该如下。addrflags共同决定第一帧地址帧的内容。因为flags是写,所以在地址帧发送完后,会将buf的内容依次发送出去。

首先,需要这些头文件

#include<sys/ioctl.h>
#include<linux/i2c.h>
#include<linux/i2c-dev.h>
uint8_t addr =0x3c;
uint8_t reg =0x01;
uint8_t value =0x55;

uint8_t buf[2];
    buf[0]= reg;//寄存器
    buf[1]= value;//数据

structi2c_msg msg;
    msg.addr = addr;//目标地址
    msg.flags =0;//读写标志
    msg.len =2;//buf长度
    msg.buf = buf;

编写完msg后,还需要创建i2c_rdwr_ioctl_data结构体,写清楚本次i2c通信要处理几个msg,然后用ioctl函数触发一次i2c通信。

structi2c_rdwr_ioctl_data data;
    data.msgs =&msg;
    data.nmsgs =1;

int ret =ioctl(fd, I2C_RDWR,&data);
if(ret <0)
printf("i2c write failed");
return ret;

4. 从i2c总线读

从德州仪器那下载的i2c时序图↓ 外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

我现在想往地址是0x3c的设备上,读取寄存器0x01的值.

根据时序图,需要两个msg,第一个msg是写,地址帧后只跟寄存器编号。第二个msg是读,地址帧发完就是从机将该寄存器的内容返回。

uint8_t addr =0x3c;
uint8_t reg =0x01;
uint8_t value;

structi2c_msg msgs[2];
    msgs[0].addr = addr;
    msgs[0].flags =0;
    msgs[0].len =1;
    msgs[0].buf =&reg;

    msgs[1].addr = addr;
    msgs[1].flags = I2C_M_RD;
    msgs[1].len =1;
    msgs[1].buf =&value;

编写完msg后,还需要创建i2c_rdwr_ioctl_data结构体,写清楚本次i2c通信要处理几个msg,然后用ioctl函数触发一次i2c通信。

structi2c_rdwr_ioctl_data data;
    data.msgs = msgs;
    data.nmsgs =2;

int res =ioctl(fd, I2C_RDWR,&data);
if(res <0)
printf("Read failed\n");

return res;

5. 关闭文件

每次open后记得调用close来手动关闭,不然文件描述符会保留直到程序关闭。而系统限制单个程序最大只能同时打开1024个文件,如果程序不停的open却不close,没一会就要报错退出了。

close(fd);

示例-从mlx90614中读取温度数据

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

首先阅读mlx90614的手册,查看他的读写时序↓

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

根据手册给的时序,这里要创建两个msg,第一个是写,buf内存放读取温度的命令0x07。第二个是读,连续3个字节,其中前两个是温度数据。再加上mlx90614的地址是0,最后代码如下

#include<stdio.h>
#include<stdint.h>
#include<string.h>

#include<sys/types.h>
#include<sys/stat.h>
#include<fcntl.h>
#include<unistd.h>

#include<sys/ioctl.h>
#include<linux/i2c.h>
#include<linux/i2c-dev.h>


#defineDEV_I2C"/dev/i2c-1"

//计算mlx9614实际温度值
uint16_tmlx_data_transform(uint8_t Data[3])
{
uint16_t temp;
    temp =(Data[1]<<8)+ Data[0];// 高位与低位结合
    temp = temp *2-27315;// 将数据扩大100倍
return temp;
}

intmain()
{

int fd =open(DEV_I2C, O_RDWR);
if(fd <0)
{
perror("Fail to Open\n");
return-1;
}

uint8_t addr =0;
uint8_t reg =0x07;
uint8_t value[3];

structi2c_msg msgs[2];
    msgs[0].addr = addr;
    msgs[0].flags =0;
    msgs[0].len =1;
    msgs[0].buf =&reg;

    msgs[1].addr = addr;
    msgs[1].flags = I2C_M_RD;
    msgs[1].len =3;
    msgs[1].buf = value;

structi2c_rdwr_ioctl_data data;
    data.msgs = msgs;
    data.nmsgs =2;

while(1)
{

int res =ioctl(fd, I2C_RDWR,&data);
if(res <0)
printf("Read failed\n");
printf("temp=[%d]\r\n",mlx_data_transform(value));
sleep(1);
}
}

我将代码写在文件i2c.c内,想将其编译成名为exe的可执行文件,只需要执行下面这一句

gcc i2c.c -o exe

执行结果如下,显示我周围环境的温度是30左右,手靠近后上升到34:

  • 21
    点赞
  • 18
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值