繁体   English   中英

I2C-dev:更改写入和读取参数

[英]I2C-dev: Change write and read parameters

我目前正在 Raspberry Pi 3 和 PsoC 4 之间处理 I2C。我正在使用 I2C-dev 库来处理 I2C 通信。 这样I2C总线就正常运行了,读写功能已经正确实现了。 然而,我想让函数指针指向读取和写入函数,因此 to 函数使用相同类型的参数(至少看起来像)我有以下主要代码称为 I

2Ctest-tools.cpp

#include <unistd.h>             //Needed for I2C port
#include <fcntl.h>              //Needed for I2C port
#include <sys/ioctl.h>          //Needed for I2C port
#include <linux/i2c-dev.h>      //Needed for I2C port
#include <stdio.h>
#include "i2c.h"
#include "functions.h"


int main() {


        int addr = 0x49;
        int cmd = 2; //Write
        int length = 5;
        int file_i2c = 0;
        I2C myI2C;



        unsigned char buffer[5];

        buffer[0] = 0x01;
        buffer[1] = 0x02;
        buffer[2] = 0x20;
        buffer[3] = 0x00;
        buffer[4] = 0x17;

        i2c_init(&myI2C, cmd, addr, &file_i2c);
        i2c_exe(&myI2C, file_i2c, buffer, length);

    return 0;
}

如代码所示,im 使用了名为 myI2C 的结构对象,该对象在 i2c_init、i2c_exe 两个函数中传递。 functions.cpp 的源代码如下:

#include <unistd.h>             //Needed for I2C port
#include <fcntl.h>              //Needed for I2C port
#include <sys/ioctl.h>          //Needed for I2C port
#include <linux/i2c-dev.h>      //Needed for I2C port
#include <stdio.h>
#include "i2c.h"
#include "functions.h"

int i2c_init(I2C *cthis, int cmd, int addr, int *ptrFile_i2c ) {
    char *filename = (char*)"/dev/i2c-1";
    if ((*ptrFile_i2c = open(filename, O_RDWR)) < 0)
    {
        //ERROR HANDLING: you can check errno to see what went wrong
        printf("Failed to open the i2c bus");
        return 0;
    }

    if (ioctl(*ptrFile_i2c, I2C_SLAVE, addr) < 0)
    {
        printf("Failed to acquire bus access and/or talk to slave.\n");
        //ERROR HANDLING; you can check errno to see what went wrong
        return 0;
    }

    switch(cmd) {
        case 1: 
            //cthis->WR = write;
            break;
        case 2:
            cthis->WR = read;
            break;
    }
    return 0;
}

int i2c_exe(I2C *cthis, int file_i2c, unsigned char *buffer, size_t length) {

    cthis->WR(file_i2c, buffer, length);
    return 0;
}

所以这里要注意的重要一点是,在函数 i2c_init im 中打开了 varialbe cmd,它决定了函数指针是指向写函数还是读函数。 现在失败部分来了。函数指针在它自己的名为 i2c.h 的 .h 文件中声明,如下所示:

struct I2C {
    ssize_t (*WR)(int, const void   *, size_t);
};

正如你所看到的,函数指针必须指向一个带有参数(int、const void*、size_t)的函数,当函数指向写函数时,这就像一个魅力,但当它指向读函数时,我得到了错误,错误说:

functions.cpp:在函数'诠释i2c_init(I2C *,INT,INT,INT *)':functions.cpp:30:14:错误:从“ssize_t供无效的转换()(INT,空隙为size_t){又名INT( )(int, void , unsigned int)}' 到 'ssize_t ( )(int, const void , size_t) {aka int ( )(int, const void , unsigned int)}' [-fpermissive] cthis->WR = read ;

我研究了错误并得出结论,这是因为读写函数以某种方式不采用相同的参数,这很奇怪,因为我在其中传递了相同的参数(int i2s_file,int buffer,int length)所以如果我更改函数指向

ssize_t (*WR)(int, void *, size_t);

该函数适用于读取但不适用于写入..所以我的问题是:我可以通过更改 i2c-dev 库以某种方式更改 write 或 read 函数以采用相同的参数,或者我还能做些什么来解决这个问题?

这是 i2c-dev 库的链接,我已经完全放弃理解http://textuploader.com/5yioi

提前致谢

您可以将函数显式转换为正确的类型:

typedef ssize_t (*I2CCMD)(int, void *, size_t);

cthis->WR = (I2CCMD)write;

cthis->WR = (I2CCMD)read;

这应该消除错误。

显然read函数不能使用指向const的指针,因为与write函数不同,它必须修改该参数。

所以你不能用函数指针来切换功能,因为函数指针是不同的类型。

最好的解决方法似乎是将“cmd”参数移动到i2c_exe函数,或者创建一个包含cmd的私有变量(在struct I2C内是理想的)。 然后在i2c_exe中调用 read 或 write。

更糟糕的解决方案是将结构更改为联合,例如:

union I2C {
    ssize_t (*WR)(int, const void   *, size_t);
    ssize_t (*read)(int, void*, size_t);
};

然后在读取的情况下,使用读取成员分配给结构。 但请注意,这是一种“肮脏的黑客”,理论上它是指定不明确的行为。 在实践中,它很可能适用于任何给定的系统。 (const 的正确性不太可能影响调用约定。)

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM