2020-09-11 07:16:58 +00:00
|
|
|
.. SPDX-License-Identifier: GPL-2.0
|
|
|
|
|
|
|
|
================================
|
|
|
|
Linux I2C slave testunit backend
|
|
|
|
================================
|
|
|
|
|
|
|
|
by Wolfram Sang <wsa@sang-engineering.com> in 2020
|
|
|
|
|
|
|
|
This backend can be used to trigger test cases for I2C bus masters which
|
|
|
|
require a remote device with certain capabilities (and which are usually not so
|
|
|
|
easy to obtain). Examples include multi-master testing, and SMBus Host Notify
|
|
|
|
testing. For some tests, the I2C slave controller must be able to switch
|
|
|
|
between master and slave mode because it needs to send data, too.
|
|
|
|
|
|
|
|
Note that this is a device for testing and debugging. It should not be enabled
|
|
|
|
in a production build. And while there is some versioning and we try hard to
|
|
|
|
keep backward compatibility, there is no stable ABI guaranteed!
|
|
|
|
|
2024-06-11 09:50:31 +00:00
|
|
|
Instantiating the device is regular. Example for bus 0, address 0x30::
|
2020-09-11 07:16:58 +00:00
|
|
|
|
2024-06-11 09:50:31 +00:00
|
|
|
# echo "slave-testunit 0x1030" > /sys/bus/i2c/devices/i2c-0/new_device
|
2020-09-11 07:16:58 +00:00
|
|
|
|
2024-08-14 18:22:07 +00:00
|
|
|
Or using firmware nodes. Here is a devicetree example (note this is only a
|
|
|
|
debug device, so there are no official DT bindings)::
|
|
|
|
|
|
|
|
&i2c0 {
|
|
|
|
...
|
|
|
|
|
|
|
|
testunit@30 {
|
|
|
|
compatible = "slave-testunit";
|
|
|
|
reg = <(0x30 | I2C_OWN_SLAVE_ADDRESS)>;
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
2024-08-11 21:23:16 +00:00
|
|
|
After that, you will have the device listening. Reading will return a single
|
|
|
|
byte. Its value is 0 if the testunit is idle, otherwise the command number of
|
|
|
|
the currently running command.
|
|
|
|
|
|
|
|
When writing, the device consists of 4 8-bit registers and, except for some
|
|
|
|
"partial" commands, all registers must be written to start a testcase, i.e. you
|
|
|
|
usually write 4 bytes to the device. The registers are:
|
2020-09-11 07:16:58 +00:00
|
|
|
|
2024-06-11 09:50:31 +00:00
|
|
|
.. csv-table::
|
|
|
|
:header: "Offset", "Name", "Description"
|
2020-09-11 07:16:58 +00:00
|
|
|
|
2024-06-11 09:50:31 +00:00
|
|
|
0x00, CMD, which test to trigger
|
|
|
|
0x01, DATAL, configuration byte 1 for the test
|
|
|
|
0x02, DATAH, configuration byte 2 for the test
|
|
|
|
0x03, DELAY, delay in n * 10ms until test is started
|
2020-09-11 07:16:58 +00:00
|
|
|
|
2024-06-11 09:50:31 +00:00
|
|
|
Using 'i2cset' from the i2c-tools package, the generic command looks like::
|
|
|
|
|
|
|
|
# i2cset -y <bus_num> <testunit_address> <CMD> <DATAL> <DATAH> <DELAY> i
|
2020-09-11 07:16:58 +00:00
|
|
|
|
|
|
|
DELAY is a generic parameter which will delay the execution of the test in CMD.
|
2020-09-28 11:06:47 +00:00
|
|
|
While a command is running (including the delay), new commands will not be
|
|
|
|
acknowledged. You need to wait until the old one is completed.
|
|
|
|
|
2020-09-11 07:16:58 +00:00
|
|
|
The commands are described in the following section. An invalid command will
|
|
|
|
result in the transfer not being acknowledged.
|
|
|
|
|
|
|
|
Commands
|
|
|
|
--------
|
|
|
|
|
2024-06-11 09:50:31 +00:00
|
|
|
0x00 NOOP
|
|
|
|
~~~~~~~~~
|
|
|
|
|
|
|
|
Reserved for future use.
|
|
|
|
|
|
|
|
0x01 READ_BYTES
|
|
|
|
~~~~~~~~~~~~~~~
|
|
|
|
|
|
|
|
.. list-table::
|
|
|
|
:header-rows: 1
|
|
|
|
|
|
|
|
* - CMD
|
|
|
|
- DATAL
|
|
|
|
- DATAH
|
|
|
|
- DELAY
|
|
|
|
|
|
|
|
* - 0x01
|
|
|
|
- address to read data from (lower 7 bits, highest bit currently unused)
|
|
|
|
- number of bytes to read
|
|
|
|
- n * 10ms
|
|
|
|
|
|
|
|
Also needs master mode. This is useful to test if your bus master driver is
|
|
|
|
handling multi-master correctly. You can trigger the testunit to read bytes
|
|
|
|
from another device on the bus. If the bus master under test also wants to
|
|
|
|
access the bus at the same time, the bus will be busy. Example to read 128
|
|
|
|
bytes from device 0x50 after 50ms of delay::
|
|
|
|
|
2024-08-11 21:23:14 +00:00
|
|
|
# i2cset -y 0 0x30 1 0x50 0x80 5 i
|
2024-06-11 09:50:31 +00:00
|
|
|
|
|
|
|
0x02 SMBUS_HOST_NOTIFY
|
|
|
|
~~~~~~~~~~~~~~~~~~~~~~
|
|
|
|
|
|
|
|
.. list-table::
|
|
|
|
:header-rows: 1
|
|
|
|
|
|
|
|
* - CMD
|
|
|
|
- DATAL
|
|
|
|
- DATAH
|
|
|
|
- DELAY
|
2020-09-11 07:16:58 +00:00
|
|
|
|
2024-06-11 09:50:31 +00:00
|
|
|
* - 0x02
|
|
|
|
- low byte of the status word to send
|
|
|
|
- high byte of the status word to send
|
|
|
|
- n * 10ms
|
2020-09-11 07:16:58 +00:00
|
|
|
|
2024-06-11 09:50:31 +00:00
|
|
|
Also needs master mode. This test will send an SMBUS_HOST_NOTIFY message to the
|
|
|
|
host. Note that the status word is currently ignored in the Linux Kernel.
|
2024-08-11 21:23:14 +00:00
|
|
|
Example to send a notification with status word 0x6442 after 10ms::
|
2020-09-11 07:16:58 +00:00
|
|
|
|
2024-08-11 21:23:14 +00:00
|
|
|
# i2cset -y 0 0x30 2 0x42 0x64 1 i
|
2020-09-11 07:16:58 +00:00
|
|
|
|
2024-07-04 03:28:59 +00:00
|
|
|
If the host controller supports HostNotify, this message with debug level
|
|
|
|
should appear (Linux 6.11 and later)::
|
|
|
|
|
|
|
|
Detected HostNotify from address 0x30
|
|
|
|
|
2024-06-11 09:50:31 +00:00
|
|
|
0x03 SMBUS_BLOCK_PROC_CALL
|
|
|
|
~~~~~~~~~~~~~~~~~~~~~~~~~~
|
2020-09-11 07:16:58 +00:00
|
|
|
|
2024-06-11 09:50:31 +00:00
|
|
|
.. list-table::
|
|
|
|
:header-rows: 1
|
2020-09-11 07:16:58 +00:00
|
|
|
|
2024-06-11 09:50:31 +00:00
|
|
|
* - CMD
|
|
|
|
- DATAL
|
|
|
|
- DATAH
|
|
|
|
- DELAY
|
2021-02-09 11:19:27 +00:00
|
|
|
|
2024-06-11 09:50:31 +00:00
|
|
|
* - 0x03
|
2024-08-11 21:23:14 +00:00
|
|
|
- 0x01 (i.e. one further byte will be written)
|
2024-06-11 09:50:31 +00:00
|
|
|
- number of bytes to be sent back
|
|
|
|
- leave out, partial command!
|
2021-02-09 11:19:27 +00:00
|
|
|
|
2024-06-11 09:50:31 +00:00
|
|
|
Partial command. This test will respond to a block process call as defined by
|
|
|
|
the SMBus specification. The one data byte written specifies how many bytes
|
|
|
|
will be sent back in the following read transfer. Note that in this read
|
|
|
|
transfer, the testunit will prefix the length of the bytes to follow. So, if
|
|
|
|
your host bus driver emulates SMBus calls like the majority does, it needs to
|
|
|
|
support the I2C_M_RECV_LEN flag of an i2c_msg. This is a good testcase for it.
|
|
|
|
The returned data consists of the length first, and then of an array of bytes
|
|
|
|
from length-1 to 0. Here is an example which emulates
|
|
|
|
i2c_smbus_block_process_call() using i2ctransfer (you need i2c-tools v4.2 or
|
|
|
|
later)::
|
2021-02-09 11:19:27 +00:00
|
|
|
|
2024-08-11 21:23:14 +00:00
|
|
|
# i2ctransfer -y 0 w3@0x30 3 1 0x10 r?
|
2024-06-11 09:50:31 +00:00
|
|
|
0x10 0x0f 0x0e 0x0d 0x0c 0x0b 0x0a 0x09 0x08 0x07 0x06 0x05 0x04 0x03 0x02 0x01 0x00
|
2024-08-11 21:23:15 +00:00
|
|
|
|
|
|
|
0x04 GET_VERSION_WITH_REP_START
|
|
|
|
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
|
|
|
|
|
|
|
.. list-table::
|
|
|
|
:header-rows: 1
|
|
|
|
|
|
|
|
* - CMD
|
|
|
|
- DATAL
|
|
|
|
- DATAH
|
|
|
|
- DELAY
|
|
|
|
|
|
|
|
* - 0x04
|
|
|
|
- currently unused
|
|
|
|
- currently unused
|
|
|
|
- leave out, partial command!
|
|
|
|
|
|
|
|
Partial command. After sending this command, the testunit will reply to a read
|
|
|
|
message with a NUL terminated version string based on UTS_RELEASE. The first
|
|
|
|
character is always a 'v' and the length of the version string is at maximum
|
|
|
|
128 bytes. However, it will only respond if the read message is connected to
|
|
|
|
the write message via repeated start. If your controller driver handles
|
|
|
|
repeated start correctly, this will work::
|
|
|
|
|
|
|
|
# i2ctransfer -y 0 w3@0x30 4 0 0 r128
|
|
|
|
0x76 0x36 0x2e 0x31 0x31 0x2e 0x30 0x2d 0x72 0x63 0x31 0x2d 0x30 0x30 0x30 0x30 ...
|
|
|
|
|
|
|
|
If you have i2c-tools 4.4 or later, you can print out the data right away::
|
|
|
|
|
|
|
|
# i2ctransfer -y -b 0 w3@0x30 4 0 0 r128
|
|
|
|
v6.11.0-rc1-00009-gd37a1b4d3fd0
|
|
|
|
|
|
|
|
STOP/START combinations between the two messages will *not* work because they
|
|
|
|
are not equivalent to a REPEATED START. As an example, this returns just the
|
|
|
|
default response::
|
|
|
|
|
|
|
|
# i2cset -y 0 0x30 4 0 0 i; i2cget -y 0 0x30
|
2024-08-11 21:23:16 +00:00
|
|
|
0x00
|
2024-08-14 18:22:09 +00:00
|
|
|
|
|
|
|
0x05 SMBUS_ALERT_REQUEST
|
|
|
|
~~~~~~~~~~~~~~~~~~~~~~~~
|
|
|
|
|
|
|
|
.. list-table::
|
|
|
|
:header-rows: 1
|
|
|
|
|
|
|
|
* - CMD
|
|
|
|
- DATAL
|
|
|
|
- DATAH
|
|
|
|
- DELAY
|
|
|
|
|
|
|
|
* - 0x05
|
|
|
|
- response value (7 MSBs interpreted as I2C address)
|
|
|
|
- currently unused
|
|
|
|
- n * 10ms
|
|
|
|
|
|
|
|
This test raises an interrupt via the SMBAlert pin which the host controller
|
|
|
|
must handle. The pin must be connected to the testunit as a GPIO. GPIO access
|
|
|
|
is not allowed to sleep. Currently, this can only be described using firmware
|
|
|
|
nodes. So, for devicetree, you would add something like this to the testunit
|
|
|
|
node::
|
|
|
|
|
|
|
|
gpios = <&gpio1 24 GPIO_ACTIVE_LOW>;
|
|
|
|
|
|
|
|
The following command will trigger the alert with a response of 0xc9 after 1
|
|
|
|
second of delay::
|
|
|
|
|
|
|
|
# i2cset -y 0 0x30 5 0xc9 0x00 100 i
|
|
|
|
|
|
|
|
If the host controller supports SMBusAlert, this message with debug level
|
|
|
|
should appear::
|
|
|
|
|
|
|
|
smbus_alert 0-000c: SMBALERT# from dev 0x64, flag 1
|
|
|
|
|
|
|
|
This message may appear more than once because the testunit is software not
|
|
|
|
hardware and, thus, may not be able to react to the response of the host fast
|
|
|
|
enough. The interrupt count should increase only by one, though::
|
|
|
|
|
|
|
|
# cat /proc/interrupts | grep smbus_alert
|
|
|
|
93: 1 gpio-rcar 26 Edge smbus_alert
|
|
|
|
|
|
|
|
If the host does not respond to the alert within 1 second, the test will be
|
|
|
|
aborted and the testunit will report an error.
|
|
|
|
|
|
|
|
For this test, the testunit will shortly drop its assigned address and listen
|
|
|
|
on the SMBus Alert Response Address (0x0c). It will reassign its original
|
|
|
|
address afterwards.
|