diff options
Diffstat (limited to 'roms/u-boot/drivers/i2c/i2c-emul-uclass.c')
-rw-r--r-- | roms/u-boot/drivers/i2c/i2c-emul-uclass.c | 94 |
1 files changed, 94 insertions, 0 deletions
diff --git a/roms/u-boot/drivers/i2c/i2c-emul-uclass.c b/roms/u-boot/drivers/i2c/i2c-emul-uclass.c new file mode 100644 index 000000000..7917b63c5 --- /dev/null +++ b/roms/u-boot/drivers/i2c/i2c-emul-uclass.c @@ -0,0 +1,94 @@ +// SPDX-License-Identifier: GPL-2.0+ +/* + * Copyright (c) 2014 Google, Inc + */ + +#include <common.h> +#include <dm.h> +#include <i2c.h> +#include <log.h> +#include <asm/i2c.h> +#include <dm/device-internal.h> +#include <dm/uclass-internal.h> + +/* + * i2c emulation works using an 'emul' node at the bus level. Each device in + * that node is in the UCLASS_I2C_EMUL uclass, and emulates one i2c device. A + * pointer to the device it emulates is in the 'dev' property of the emul device + * uclass plat (struct i2c_emul_plat), put there by i2c_emul_find(). + * When sandbox wants an emulator for a device, it calls i2c_emul_find() which + * searches for the emulator with the correct address. To find the device for an + * emulator, call i2c_emul_get_device(). + * + * The 'emul' node is in the UCLASS_I2C_EMUL_PARENT uclass. We use a separate + * uclass so avoid having strange devices on the I2C bus. + */ + +struct udevice *i2c_emul_get_device(struct udevice *emul) +{ + struct i2c_emul_uc_plat *uc_plat = dev_get_uclass_plat(emul); + + return uc_plat->dev; +} + +void i2c_emul_set_idx(struct udevice *dev, int emul_idx) +{ + struct dm_i2c_chip *plat = dev_get_parent_plat(dev); + + plat->emul_idx = emul_idx; +} + +int i2c_emul_find(struct udevice *dev, struct udevice **emulp) +{ + struct i2c_emul_uc_plat *uc_plat; + struct udevice *emul; + int ret; + + if (!CONFIG_IS_ENABLED(OF_PLATDATA)) { + ret = uclass_find_device_by_phandle(UCLASS_I2C_EMUL, dev, + "sandbox,emul", &emul); + } else { + struct dm_i2c_chip *plat = dev_get_parent_plat(dev); + + ret = device_get_by_ofplat_idx(plat->emul_idx, &emul); + } + if (ret) { + log_err("No emulators for device '%s'\n", dev->name); + return ret; + } + uc_plat = dev_get_uclass_plat(emul); + uc_plat->dev = dev; + *emulp = emul; + + return device_probe(emul); +} + +UCLASS_DRIVER(i2c_emul) = { + .id = UCLASS_I2C_EMUL, + .name = "i2c_emul", + .per_device_plat_auto = sizeof(struct i2c_emul_uc_plat), +}; + +/* + * This uclass is a child of the i2c bus. Its plat is not defined here so + * is defined by its parent, UCLASS_I2C, which uses struct dm_i2c_chip. See + * per_child_plat_auto in UCLASS_DRIVER(i2c). + */ +UCLASS_DRIVER(i2c_emul_parent) = { + .id = UCLASS_I2C_EMUL_PARENT, + .name = "i2c_emul_parent", +#if !CONFIG_IS_ENABLED(OF_PLATDATA) + .post_bind = dm_scan_fdt_dev, +#endif +}; + +static const struct udevice_id i2c_emul_parent_ids[] = { + { .compatible = "sandbox,i2c-emul-parent" }, + { } +}; + +U_BOOT_DRIVER(sandbox_i2c_emul_parent) = { + .name = "sandbox_i2c_emul_parent", + .id = UCLASS_I2C_EMUL_PARENT, + .of_match = i2c_emul_parent_ids, +}; |