|  | .. 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! | 
|  |  | 
|  | Instantiating the device is regular. Example for bus 0, address 0x30:: | 
|  |  | 
|  | # echo "slave-testunit 0x1030" > /sys/bus/i2c/devices/i2c-0/new_device | 
|  |  | 
|  | 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)>; | 
|  | }; | 
|  | }; | 
|  |  | 
|  | 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: | 
|  |  | 
|  | .. csv-table:: | 
|  | :header: "Offset", "Name", "Description" | 
|  |  | 
|  | 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 | 
|  |  | 
|  | Using 'i2cset' from the i2c-tools package, the generic command looks like:: | 
|  |  | 
|  | # i2cset -y <bus_num> <testunit_address> <CMD> <DATAL> <DATAH> <DELAY> i | 
|  |  | 
|  | DELAY is a generic parameter which will delay the execution of the test in CMD. | 
|  | While a command is running (including the delay), new commands will not be | 
|  | acknowledged. You need to wait until the old one is completed. | 
|  |  | 
|  | The commands are described in the following section. An invalid command will | 
|  | result in the transfer not being acknowledged. | 
|  |  | 
|  | Commands | 
|  | -------- | 
|  |  | 
|  | 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:: | 
|  |  | 
|  | # i2cset -y 0 0x30 1 0x50 0x80 5 i | 
|  |  | 
|  | 0x02 SMBUS_HOST_NOTIFY | 
|  | ~~~~~~~~~~~~~~~~~~~~~~ | 
|  |  | 
|  | .. list-table:: | 
|  | :header-rows: 1 | 
|  |  | 
|  | * - CMD | 
|  | - DATAL | 
|  | - DATAH | 
|  | - DELAY | 
|  |  | 
|  | * - 0x02 | 
|  | - low byte of the status word to send | 
|  | - high byte of the status word to send | 
|  | - n * 10ms | 
|  |  | 
|  | 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. | 
|  | Example to send a notification with status word 0x6442 after 10ms:: | 
|  |  | 
|  | # i2cset -y 0 0x30 2 0x42 0x64 1 i | 
|  |  | 
|  | If the host controller supports HostNotify, this message with debug level | 
|  | should appear (Linux 6.11 and later):: | 
|  |  | 
|  | Detected HostNotify from address 0x30 | 
|  |  | 
|  | 0x03 SMBUS_BLOCK_PROC_CALL | 
|  | ~~~~~~~~~~~~~~~~~~~~~~~~~~ | 
|  |  | 
|  | .. list-table:: | 
|  | :header-rows: 1 | 
|  |  | 
|  | * - CMD | 
|  | - DATAL | 
|  | - DATAH | 
|  | - DELAY | 
|  |  | 
|  | * - 0x03 | 
|  | - 0x01 (i.e. one further byte will be written) | 
|  | - number of bytes to be sent back | 
|  | - leave out, partial command! | 
|  |  | 
|  | 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):: | 
|  |  | 
|  | # i2ctransfer -y 0 w3@0x30 3 1 0x10 r? | 
|  | 0x10 0x0f 0x0e 0x0d 0x0c 0x0b 0x0a 0x09 0x08 0x07 0x06 0x05 0x04 0x03 0x02 0x01 0x00 | 
|  |  | 
|  | 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 | 
|  | 0x00 | 
|  |  | 
|  | 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. |