linux: Fast Linux MCU i2c_read() with I2C_RDRW (#6101)
Reading an I2C device from the Linux MCU used a separate write(2) to select the target register & read(2) to get the value(s). This implementation uses ioctl(file, I2C_RDWR, ...) to skip a large bus idle period and extra process sleep by combining them like the stm32. I2C_RDRW requires I2C_FUNC_I2C flag in the I2C driver. I2C_FUNC_I2C is defined in: BCM2835: Pi 1 Models A, A+, B, B+, the Raspberry Pi Zero, the Raspberry Pi Zero W, and the Raspberry Pi Compute Module 1 BCM2836: Pi 2 Model B Identical to BCM2835 except Cortex BCM2837: Pi 3 Model B, later models of the Raspberry Pi 2 Model B, and the Raspberry Pi Compute Module 3 BCM2837B0: Pi 3 Models A+, B+, and the Raspberry Pi Compute Module 3+ BCM2711: Pi 4 Model B, the Raspberry Pi 400, and the Raspberry Pi Compute Module 4 RK3xxx: Rockchips SoCs NanoPi, RockPi, Tinker, etc. SUNXI: H2, H3, etc. Orange Pi AMLOGIC: S905x, Banana Pi, Odroid, etc. TEGRA: NVidia Jetson etc. MediaTek: Several SBCs in other ranges Signed-off-by: Matthew Swabey <matthew@swabey.org>
This commit is contained in:
parent
ca6e5fe514
commit
9d77f44995
|
@ -45,6 +45,7 @@ void gpio_pwm_write(struct gpio_pwm g, uint16_t val);
|
||||||
|
|
||||||
struct i2c_config {
|
struct i2c_config {
|
||||||
int fd;
|
int fd;
|
||||||
|
uint8_t addr;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct i2c_config i2c_setup(uint32_t bus, uint32_t rate, uint8_t addr);
|
struct i2c_config i2c_setup(uint32_t bus, uint32_t rate, uint8_t addr);
|
||||||
|
|
|
@ -4,7 +4,8 @@
|
||||||
//
|
//
|
||||||
// This file may be distributed under the terms of the GNU GPLv3 license.
|
// This file may be distributed under the terms of the GNU GPLv3 license.
|
||||||
#include <fcntl.h> // open
|
#include <fcntl.h> // open
|
||||||
#include <linux/i2c-dev.h> // I2C_SLAVE
|
#include <linux/i2c-dev.h> // I2C_SLAVE i2c_msg
|
||||||
|
#include <linux/i2c.h> // i2c_rdwr_ioctl_data I2C_M_RD I2C_FUNC_I2C
|
||||||
#include <stdio.h> // snprintf
|
#include <stdio.h> // snprintf
|
||||||
#include <sys/ioctl.h> // ioctl
|
#include <sys/ioctl.h> // ioctl
|
||||||
#include <unistd.h> // write
|
#include <unistd.h> // write
|
||||||
|
@ -42,6 +43,13 @@ i2c_open(uint32_t bus, uint8_t addr)
|
||||||
report_errno("open i2c", fd);
|
report_errno("open i2c", fd);
|
||||||
goto fail;
|
goto fail;
|
||||||
}
|
}
|
||||||
|
// Test for I2C_RDWR support
|
||||||
|
unsigned long i2c_funcs; // datatype from ioctl spec.
|
||||||
|
ioctl(fd, I2C_FUNCS, &i2c_funcs);
|
||||||
|
if ((i2c_funcs & I2C_FUNC_I2C) == 0) {
|
||||||
|
report_errno("i2c does not support I2C_RDWR", fd);
|
||||||
|
goto fail;
|
||||||
|
}
|
||||||
int ret = ioctl(fd, I2C_SLAVE, addr);
|
int ret = ioctl(fd, I2C_SLAVE, addr);
|
||||||
if (ret < 0) {
|
if (ret < 0) {
|
||||||
report_errno("ioctl i2c", fd);
|
report_errno("ioctl i2c", fd);
|
||||||
|
@ -73,7 +81,7 @@ i2c_setup(uint32_t bus, uint32_t rate, uint8_t addr)
|
||||||
// dtparam=i2c_baudrate=<rate>
|
// dtparam=i2c_baudrate=<rate>
|
||||||
|
|
||||||
int fd = i2c_open(bus, addr);
|
int fd = i2c_open(bus, addr);
|
||||||
return (struct i2c_config){.fd=fd};
|
return (struct i2c_config){.fd=fd, .addr=addr};
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
|
@ -91,12 +99,29 @@ void
|
||||||
i2c_read(struct i2c_config config, uint8_t reg_len, uint8_t *reg
|
i2c_read(struct i2c_config config, uint8_t reg_len, uint8_t *reg
|
||||||
, uint8_t read_len, uint8_t *data)
|
, uint8_t read_len, uint8_t *data)
|
||||||
{
|
{
|
||||||
if(reg_len != 0)
|
struct i2c_rdwr_ioctl_data i2c_data;
|
||||||
i2c_write(config, reg_len, reg);
|
struct i2c_msg msgs[2];
|
||||||
int ret = read(config.fd, data, read_len);
|
|
||||||
if (ret != read_len) {
|
if(reg_len != 0) {
|
||||||
if (ret < 0)
|
msgs[0].addr = config.addr;
|
||||||
report_errno("read value i2c", ret);
|
msgs[0].flags = 0x0;
|
||||||
|
msgs[0].len = reg_len;
|
||||||
|
msgs[0].buf = reg;
|
||||||
|
i2c_data.nmsgs = 2;
|
||||||
|
i2c_data.msgs = &msgs[0];
|
||||||
|
} else {
|
||||||
|
i2c_data.nmsgs = 1;
|
||||||
|
i2c_data.msgs = &msgs[1];
|
||||||
|
}
|
||||||
|
|
||||||
|
msgs[1].addr = config.addr;
|
||||||
|
msgs[1].flags = I2C_M_RD;
|
||||||
|
msgs[1].len = read_len;
|
||||||
|
msgs[1].buf = data;
|
||||||
|
|
||||||
|
int ret = ioctl(config.fd, I2C_RDWR, &i2c_data);
|
||||||
|
|
||||||
|
if(ret < 0) {
|
||||||
try_shutdown("Unable to read i2c device");
|
try_shutdown("Unable to read i2c device");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue