Custom I2C peripheral in QEMU
- May 26, 2024
QEMU supports creating various peripherals, not only the memory-mapped shown in previous posts.
In this post we will go through the process of creating a new I2C peripheral, and userspace support for it. The following things will be covered
Custom I2C QEMU peripheral
Emulating I2C peripheral under QEMU is not much harder then emulating a memory-mapped peripheral.
The peripheral needs to be connected to an existing I2C controller, assigned and I2C address and should provide responses when data is written to or read from I2C bus.
Peripheral description
In this example we will implement a simple I2C temperature sensor. Once enabled, it will return a random value in the range of 15.0 to 25.0 with 0.5 degrees (Celsius) step on every read.
The peripheral has following registers
offset | name | description |
---|---|---|
0x0 | ID |
Unique ID register, always returning value 0x5A |
0x1 | CONFIG |
Configuration register, used to enable temperature measurement |
0x2 | TEMPERATURE |
Data register holding current temperature. The value is coded as Q5.1 to be able to hold 0.5 degree step |
Bit values of CONFIG
register are shown in the following table
name | pos | dflt | description |
---|---|---|---|
Reserved | 7:1 | 0 | Reserved |
EN |
0 | 0 | Enable device |
Peripheral implementation
The implementation of the peripheral is available in github repository.
Functions of interest are the ones related to data handling. The global structure is defined in the following way
typedef struct I2CSensor {
/*< private >*/
I2CSlave i2c;
/*< public >*/
uint8_t regs[NR_REGS]; // peripheral registers
uint8_t count; // counter used for tx/rx
uint8_t ptr; // current register index
} I2CSensor;
Writing to the device
The only register that supports writing is the CTRL
register.
In order to set the register that will be written there should be two I2C writes. The first I2C write is the register
address (that will update the ptr
field), and the second write is the value that should be stored in the register.
static int i2c_sens_tx(I2CSlave *i2c, uint8_t data)
{
I2CSensor *s = I2C_SENS(i2c);
if (s->count == 0) {
/* store register address */
s->ptr = data;
s->count++;
} else {
if (s->ptr == REG_CTRL_OFFSET) {
s->regs[s->ptr++] = data;
}
}
return 0;
}
Reading from device
Reading process is similar to writing: first one byte must be written to set the address of register to be read, and then the reading can proceed.
All registers support reading so it is just a matter of returning current register value.
/* Called when master requests read */
static uint8_t i2c_sens_rx(I2CSlave *i2c)
{
I2CSensor *s = I2C_SENS(i2c);
uint8_t ret = 0xff;
if (s->ptr < NR_REGS) {
ret = s->regs[s->ptr++];
}
return ret;
}
We also want each read to trigger loading of random value to the TEMPERATURE
register. That is achieved by defining an
event
callback which modifies the s->regs[2]
value when I2C_START_RECV
event is received.
The register is modified only if device is currently enabled, otherwise a value of 0xff
will be stored.
static int i2c_sens_event(I2CSlave *i2c, enum i2c_event event)
{
I2CSensor *s = I2C_SENS(i2c);
if (event == I2C_START_RECV) {
if (s->ptr == REG_TEMPERATURE_OFFSET) {
if (s->regs[REG_CTRL_OFFSET] & REG_CTRL_EN_MASK) {
s->regs[REG_TEMPERATURE_OFFSET] = i2c_sens_get_temperature();
} else {
s->regs[REG_TEMPERATURE_OFFSET] = 0xff;
}
}
}
s->count = 0;
return 0;
}
Integrating into QEMU
Similarly to the Memory-mapped peripheral, the new I2C peripheral must be integrated into the Cubieboard description. This time, instead of attaching to a place in memory map, it needs to be attached to an I2C controller at a selected I2C address.
We can choose either to use an already existing I2C controller, or to also add a new I2C controller in the memory map, and then attach this component to the I2C controller.
In this example we will use the existing I2C controller and attach our custom I2C component to that I2C controller. The
I2C slave address is chosen as 0x36
.
The excerpt from initialization of emulated Cubieboard is
i2c_slave_create_simple(i2c, "mistra.i2csens", 0x36);
I2C Userspace handling
With the device created in QEMU we can turn to making a userspace application to access the I2C device.
The program flow will be similar to the one used for memory-mapped device
- Device should be initialized
- After device is initialized, we should periodically read the temperature data
The full userspace application can be found in github.
Initialization
The device initialization now involves setting up the I2C userspace communication. Everything is done using the dev
file for the selected I2C controller. In our case it is /dev/i2c-0
.
The process can be summarized as
-
Opening the I2C controller device file
fd = open("/dev/i2c-0", O_RDWR);
-
Configuring I2C slave address
ioctl(fd, I2C_SLAVE, 0x36);
-
Enabling device by writing
1
toCTRL
register. Since this write should be performed in two steps, an array consisting ofCTRL
register offset (0x01
) and value to be written (1
) is passed to thewrite
functionuint8_t buffer[2] = { CTRL_REG, ENABLE }; write(fd, buffer, 2);
Periodic execution
After the device is initialized, we can use a separate thread to periodically initiate reads from I2C device. Unlike the memory-mapped device, this peripheral has no interrupt generation possibilities, so we need to make periodic execution in some other way.
We will use std::this_thread::sleep_until
to enable periodic readouts from the I2C peripheral.
The read process also consists of two steps: writing TEMPERATURE
register offset and then reading the TEMPERATURE
register value.
while (m_running)
{
// Read current time so we know when to timeout
current_time = std::chrono::system_clock::now();
// read I2C device and print
// first write address
uint8_t buffer[1] = { TEMPERATURE_REG };
write(fd, buffer, 1);
// then read value
read(fd, buffer, 1);
std::cout << "Measured " << (buffer[0] / 2.) << std::endl;
// sleep_until
std::this_thread::sleep_until(current_time + std::chrono::seconds(1));
}
Testing in QEMU
i2c-tools
Before testing the userspace application, we need to test that device is integrated properly inside QEMU. For that we
can use i2c-tools
package. In the Ubuntu root filesystem the package is installed using
sudo apt install i2c-tools
Once i2c-tools
are installed, we can use i2cdetect
to verify that a peripheral at 0x36
address exists using
$ i2cdetect -y 0
0 1 2 3 4 5 6 7 8 9 a b c d e f
00: -- -- -- -- -- -- -- -- -- -- -- -- --
10: -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- --
20: -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- --
30: -- -- -- -- UU -- 36 -- -- -- -- -- -- -- -- --
40: -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- --
50: -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- --
60: -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- --
70: -- -- -- -- -- -- -- --
The
UU
at 0x34 is from the AXP209 PMIC driver, indicating that there is a driver allocated to it.
The i2cget
can be used to read a value and i2cset
to set certain value.
We can check that ID
register returns 0x5A
$ i2cget -y 0 0x36 0
0x5a
The last test would be to turn on the device and check that TEMPERATURE
register returns different values every time
$ i2cset -y 0 0x36 1 1
$ i2cget -y 0 0x36 2
0x22
$ i2cget -y 0 0x36 2
0x25
$ i2cget -y 0 0x36 2
0x2c
Userspace application
Userspace application can be compiled in the same manner as the memory-mapped application, via external toolchain or using a Yocto SDK generated toolchain:
mkdir build && cd build
cmake -DCMAKE_TOOLCHAIN_FILE=../cmake/toolchain.cmake ..
make -j
or
source /opt/mistra-framebuffer/4.0/environment-setup-cortexa8t2hf-neon-mistra-linux-gnueabi
mkdir build && cd build
cmake ..
make -j
After the application has been built and copied to the rootfs, it can be started with
./i2csens-app /dev/i2c-0
The application will initialize the I2C peripheral and start printing read temperature value every second.
Summary
In this blog post userspace application development for custom QEMU I2C sensor is presented.
The application encapsulates the I2C userspace access and implements periodic reading of data.
It can be improved with unit tests, as well as more interactive behavior, but it is left for some other time.