Files
micropython/tests/extmod_hardware/machine_i2c_target.py
Damien George 6558d519a2 tests/extmod_hardware: Add self unittest for I2CTarget.
This test uses a SoftI2C controller wired to an I2CTarget on the one board,
and tests all functionality of the I2CTarget class.

Signed-off-by: Damien George <damien@micropython.org>
2025-08-01 23:03:17 +10:00

308 lines
10 KiB
Python

# Test machine.I2CTarget.
#
# IMPORTANT: This test requires hardware connections: a SoftI2C instance must be
# wired to a hardware I2C target. See pin definitions below.
import sys
try:
from machine import Pin, SoftI2C, I2CTarget
except ImportError:
print("SKIP")
raise SystemExit
import unittest
ADDR = 67
kwargs_target = {}
# Configure pins based on the target.
if sys.platform == "alif" and sys.implementation._build == "ALIF_ENSEMBLE":
args_controller = {"scl": "P1_1", "sda": "P1_0"}
args_target = (0,) # on pins P0_3/P0_2
elif sys.platform == "esp32":
args_controller = {"scl": 5, "sda": 6}
args_target = (0,) # on pins 9/8 for C3 and S3, 18/19 for others
kwargs_target = {"scl": 9, "sda": 8}
elif sys.platform == "rp2":
args_controller = {"scl": 5, "sda": 4}
args_target = (1,)
elif sys.platform == "pyboard":
if sys.implementation._build == "NUCLEO_WB55":
args_controller = {"scl": "B8", "sda": "B9"}
args_target = (3,)
else:
args_controller = {"scl": "X1", "sda": "X2"}
args_target = ("X",)
elif "zephyr-nucleo_wb55rg" in sys.implementation._machine:
# PB8=I2C1_SCL, PB9=I2C1_SDA (on Arduino header D15/D14)
# PC0=I2C3_SCL, PC1=I2C3_SDA (on Arduino header A0/A1)
args_controller = {"scl": Pin(("gpiob", 8)), "sda": Pin(("gpiob", 9))}
args_target = ("i2c3",)
elif "zephyr-rpi_pico" in sys.implementation._machine:
args_controller = {"scl": Pin(("gpio0", 5)), "sda": Pin(("gpio0", 4))}
args_target = ("i2c1",) # on gpio7/gpio6
elif sys.platform == "mimxrt":
if "Teensy" in sys.implementation._machine:
args_controller = {"scl": "A6", "sda": "A3"} # D20/D17
else:
args_controller = {"scl": "D0", "sda": "D1"}
args_target = (0,) # pins 19/18 On Teensy 4.x
elif sys.platform == "samd":
args_controller = {"scl": "D5", "sda": "D1"}
args_target = ()
else:
print("Please add support for this test on this platform.")
raise SystemExit
def config_pull_up():
Pin(args_controller["scl"], Pin.OPEN_DRAIN, Pin.PULL_UP)
Pin(args_controller["sda"], Pin.OPEN_DRAIN, Pin.PULL_UP)
class TestMemory(unittest.TestCase):
@classmethod
def setUpClass(cls):
cls.mem = bytearray(8)
cls.i2c = SoftI2C(**args_controller)
cls.i2c_target = I2CTarget(*args_target, **kwargs_target, addr=ADDR, mem=cls.mem)
config_pull_up()
@classmethod
def tearDownClass(cls):
cls.i2c_target.deinit()
def test_scan(self):
self.assertIn(ADDR, self.i2c.scan())
def test_write(self):
self.mem[:] = b"01234567"
self.i2c.writeto_mem(ADDR, 0, b"test")
self.assertEqual(self.mem, bytearray(b"test4567"))
self.i2c.writeto_mem(ADDR, 4, b"TEST")
self.assertEqual(self.mem, bytearray(b"testTEST"))
def test_write_wrap(self):
self.mem[:] = b"01234567"
self.i2c.writeto_mem(ADDR, 6, b"test")
self.assertEqual(self.mem, bytearray(b"st2345te"))
@unittest.skipIf(sys.platform == "esp32", "write lengths larger than buffer unsupported")
def test_write_wrap_large(self):
self.mem[:] = b"01234567"
self.i2c.writeto_mem(ADDR, 0, b"testTESTmore")
self.assertEqual(self.mem, bytearray(b"moreTEST"))
def test_read(self):
self.mem[:] = b"01234567"
self.assertEqual(self.i2c.readfrom_mem(ADDR, 0, 4), b"0123")
self.assertEqual(self.i2c.readfrom_mem(ADDR, 4, 4), b"4567")
def test_read_wrap(self):
self.mem[:] = b"01234567"
self.assertEqual(self.i2c.readfrom_mem(ADDR, 0, 4), b"0123")
self.assertEqual(self.i2c.readfrom_mem(ADDR, 2, 4), b"2345")
self.assertEqual(self.i2c.readfrom_mem(ADDR, 6, 4), b"6701")
@unittest.skipIf(sys.platform == "esp32", "read lengths larger than buffer unsupported")
def test_read_wrap_large(self):
self.mem[:] = b"01234567"
self.assertEqual(self.i2c.readfrom_mem(ADDR, 0, 12), b"012345670123")
def test_write_read(self):
self.mem[:] = b"01234567"
self.assertEqual(self.i2c.writeto(ADDR, b"\x02"), 1)
self.assertEqual(self.i2c.readfrom(ADDR, 4), b"2345")
@unittest.skipIf(sys.platform == "esp32", "read after read unsupported")
def test_write_read_read(self):
self.mem[:] = b"01234567"
self.assertEqual(self.i2c.writeto(ADDR, b"\x02"), 1)
self.assertEqual(self.i2c.readfrom(ADDR, 4), b"2345")
self.assertEqual(self.i2c.readfrom(ADDR, 4), b"7012")
@unittest.skipUnless(hasattr(I2CTarget, "IRQ_END_READ"), "IRQ unsupported")
class TestMemoryIRQ(unittest.TestCase):
@staticmethod
def irq_handler(i2c_target):
flags = i2c_target.irq().flags()
TestMemoryIRQ.events[TestMemoryIRQ.num_events] = flags
TestMemoryIRQ.events[TestMemoryIRQ.num_events + 1] = i2c_target.memaddr
TestMemoryIRQ.num_events += 2
@classmethod
def setUpClass(cls):
cls.mem = bytearray(8)
cls.events = [0] * 8
cls.num_events = 0
cls.i2c = SoftI2C(**args_controller)
cls.i2c_target = I2CTarget(*args_target, **kwargs_target, addr=ADDR, mem=cls.mem)
cls.i2c_target.irq(TestMemoryIRQ.irq_handler)
config_pull_up()
@classmethod
def tearDownClass(cls):
cls.i2c_target.deinit()
@unittest.skipIf(sys.platform == "esp32", "scan doesn't trigger IRQ_END_WRITE")
def test_scan(self):
TestMemoryIRQ.num_events = 0
self.i2c.scan()
self.assertEqual(self.events[: self.num_events], [I2CTarget.IRQ_END_WRITE, 0])
def test_write(self):
TestMemoryIRQ.num_events = 0
self.mem[:] = b"01234567"
self.i2c.writeto_mem(ADDR, 2, b"test")
self.assertEqual(self.mem, bytearray(b"01test67"))
self.assertEqual(self.events[: self.num_events], [I2CTarget.IRQ_END_WRITE, 2])
def test_read(self):
TestMemoryIRQ.num_events = 0
self.mem[:] = b"01234567"
self.assertEqual(self.i2c.readfrom_mem(ADDR, 2, 4), b"2345")
self.assertEqual(self.events[: self.num_events], [I2CTarget.IRQ_END_READ, 2])
@unittest.skipUnless(hasattr(I2CTarget, "IRQ_WRITE_REQ"), "IRQ unsupported")
@unittest.skipIf(sys.platform == "mimxrt", "not working")
@unittest.skipIf(sys.platform == "pyboard", "can't queue more than one byte")
@unittest.skipIf(sys.platform == "samd", "not working")
@unittest.skipIf(sys.platform == "zephyr", "must call readinto/write in IRQ handler")
class TestPolling(unittest.TestCase):
@staticmethod
def irq_handler(i2c_target, buf=bytearray(1)):
flags = i2c_target.irq().flags()
if flags & I2CTarget.IRQ_READ_REQ:
i2c_target.write(b"0123")
@classmethod
def setUpClass(cls):
cls.i2c = SoftI2C(**args_controller)
cls.i2c_target = I2CTarget(*args_target, addr=ADDR)
cls.i2c_target.irq(
TestPolling.irq_handler,
I2CTarget.IRQ_WRITE_REQ | I2CTarget.IRQ_READ_REQ,
hard=True,
)
config_pull_up()
@classmethod
def tearDownClass(cls):
cls.i2c_target.deinit()
def test_read(self):
# Can't write data up front, must wait until IRQ_READ_REQ.
# self.assertEqual(self.i2c_target.write(b"abcd"), 4)
self.assertEqual(self.i2c.readfrom(ADDR, 4), b"0123")
def test_write(self):
# Can do the read outside the IRQ, but requires IRQ_WRITE_REQ trigger to be set.
self.assertEqual(self.i2c.writeto(ADDR, b"0123"), 4)
buf = bytearray(8)
self.assertEqual(self.i2c_target.readinto(buf), 4)
self.assertEqual(buf, b"0123\x00\x00\x00\x00")
@unittest.skipUnless(hasattr(I2CTarget, "IRQ_ADDR_MATCH_READ"), "IRQ unsupported")
class TestIRQ(unittest.TestCase):
@staticmethod
def irq_handler(i2c_target, buf=bytearray(1)):
flags = i2c_target.irq().flags()
TestIRQ.events[TestIRQ.num_events] = flags
TestIRQ.num_events += 1
if flags & I2CTarget.IRQ_READ_REQ:
i2c_target.write(b"Y")
if flags & I2CTarget.IRQ_WRITE_REQ:
i2c_target.readinto(buf)
TestIRQ.events[TestIRQ.num_events] = buf[0]
TestIRQ.num_events += 1
@classmethod
def setUpClass(cls):
cls.events = [0] * 8
cls.num_events = 0
cls.i2c = SoftI2C(**args_controller)
cls.i2c_target = I2CTarget(*args_target, addr=ADDR)
cls.i2c_target.irq(
TestIRQ.irq_handler,
I2CTarget.IRQ_ADDR_MATCH_READ
| I2CTarget.IRQ_ADDR_MATCH_WRITE
| I2CTarget.IRQ_WRITE_REQ
| I2CTarget.IRQ_READ_REQ
| I2CTarget.IRQ_END_READ
| I2CTarget.IRQ_END_WRITE,
hard=True,
)
config_pull_up()
@classmethod
def tearDownClass(cls):
cls.i2c_target.deinit()
def test_scan(self):
TestIRQ.num_events = 0
self.i2c.scan()
self.assertEqual(
self.events[: self.num_events],
[
I2CTarget.IRQ_ADDR_MATCH_WRITE,
I2CTarget.IRQ_END_WRITE,
],
)
def test_write(self):
TestIRQ.num_events = 0
self.i2c.writeto(ADDR, b"XYZ")
self.assertEqual(
self.events[: self.num_events],
[
I2CTarget.IRQ_ADDR_MATCH_WRITE,
I2CTarget.IRQ_WRITE_REQ,
ord(b"X"),
I2CTarget.IRQ_WRITE_REQ,
ord(b"Y"),
I2CTarget.IRQ_WRITE_REQ,
ord(b"Z"),
I2CTarget.IRQ_END_WRITE,
],
)
def test_read(self):
TestIRQ.num_events = 0
self.assertEqual(self.i2c.readfrom(ADDR, 1), b"Y")
self.assertEqual(
self.events[: self.num_events],
[
I2CTarget.IRQ_ADDR_MATCH_READ,
I2CTarget.IRQ_READ_REQ,
I2CTarget.IRQ_READ_REQ,
I2CTarget.IRQ_END_READ,
],
)
def test_write_read(self):
TestIRQ.num_events = 0
self.i2c.writeto(ADDR, b"X", False)
self.assertEqual(self.i2c.readfrom(ADDR, 1), b"Y")
self.assertEqual(
self.events[: self.num_events],
[
I2CTarget.IRQ_ADDR_MATCH_WRITE,
I2CTarget.IRQ_WRITE_REQ,
ord(b"X"),
I2CTarget.IRQ_END_WRITE,
I2CTarget.IRQ_ADDR_MATCH_READ,
I2CTarget.IRQ_READ_REQ,
I2CTarget.IRQ_READ_REQ,
I2CTarget.IRQ_END_READ,
],
)
if __name__ == "__main__":
unittest.main()