gpt4 book ai didi

c - 在 Linux 的 Raspberry Pi 上使用 C 语言从头开始进行 I²C 通信

转载 作者:行者123 更新时间:2023-12-04 07:22:17 26 4
gpt4 key购买 nike

我想创建一个用于与 I²C 设备通信的小型库,尤其是 C 语言中从头开始的 MPU6050 加速度计/陀螺仪模块。从技术上讲,我可以使用诸如wiringPi 之类的库,但是……它的乐趣在哪里?现在我正在尝试编写一个函数来扫描所有可能的 7 位地址(0 - 127)并读取确认位以查找连接的 I²C 设备的所有“事件”地址。为此,我编写了一个小型库,它可以通过写入/sys/class/gpio 中对应的 GPIO 文件来控制 GPIO 引脚。例如,它可以设置引脚、设置其方向(输入或输出)、设置其数字输出并从引脚读取数字输入。我测试了所有这些功能,它们工作得很好而且很快。
为了与 MPU6050 I²C 从机进行通信,我将 MPU 板的电源和接地引脚连接到 Raspberry 的 GPIO 电源和接地引脚,并将 SDA 和 SCL 线连接到 GPIO 引脚 14 和 15。我最近阅读了很多关于 I²C 的信息,我确实了解这个怎么运作。问题是,我扫描所有地址的代码不起作用。这是我的代码:

#include <stdio.h>
#include <unistd.h>
#include "string.h"
#include "gpio.h"

#define SCL 14
#define SDA 15
#define FREQ 100000
#define high 1
#define low 0

char binret[8];
void scanI2C();
void charToBin(unsigned char);
void delay();

int main () {
setupPin(SCL, OUTPUT);
setupPin(SDA, OUTPUT);
scanI2C();
}

void scanI2C() {
int addr = 0;

for (int i = 90; i < 128; i++) {
charToBin(i);

// Start Condition
setPin(SDA, high);
setPin(SCL, high);
delay();
setPin(SDA, low);

for (int j = 0; j < 9; j++) {
delay();
setPin(SCL, low);
delay();
setPin(SCL, high);

if (j < 7) {
setPin(SDA, binret[j+1]);
}

// WRITE Bit
if (j == 7) {
setPin(SDA, high);
}

// Listen for ACK bit
if (j == 8) {
setPinDirection(SDA, INPUT);
printf("Checking Address %d (%d%d%d%d%d%d%d%d)", i, binret[0], binret[1], binret[2], binret[3], binret[4], binret[5], binret[6], binret[7]);
if (readPin(SDA) <= 0.5) {
printf(" Bingo!");
}
setPinDirection(SDA, OUTPUT);
printf("\n");
}
}
// Stop Condition
setPin(SCL, low);
delay();
setPin(SDA, high);
delay();
setPin(SCL, high);
delay();
delay();
delay();
delay();
delay();

}
}

void charToBin(unsigned char a) {
memset(binret, 0, sizeof(binret));
int val = 128;
for (int i = 0; i < 8; i++) {
int bit = a & val;
if (bit != 0) {
binret[i] = 1;
}
val /= 2;
}
}

void delay() {
usleep(1000000.0/FREQ);
}
因为我确实知道 MPU6050 I²C 从机的地址是十进制的 0x69 或 105,所以我希望输出是这样的:
... 
Checking Address 103 (01100111)
Checking Address 104 (01101000)
Checking Address 105 (01101001) Bingo!
Checking Address 106 (01101010)
Checking Address 107 (01101011)
Checking Address 108 (01101100)
Checking Address 109 (01101101)
Checking Address 110 (01101110)
...
但是不,我没有得到 Bingo!在地址 105,这意味着我的 I²C 通信肯定无法正常工作。非常感谢有关如何解决此问题的建议!谢谢你,祝你有美好的一天 :)

最佳答案

通常它是 8 位“地址”的 LSB,即 R/W 位。如果您有一个地址为 105 的设备,我希望它会在您的 8 位地址为 210 或 211 时做出响应。您是否尝试过整个地址空间?
readPin() 是否返回浮点数?如果不是,为什么要与浮点数进行比较?
看起来您正在更改 SDA 而 SCL 为高并且您正在发送地址位。我认为这是不允许的。
SDA上有上拉电阻吗?
您是否使用示波器验证过信号电平和时序?

关于c - 在 Linux 的 Raspberry Pi 上使用 C 语言从头开始进行 I²C 通信,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/68415261/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com