From: Sakari Ailus <sakari.ailus@iki.fi>
To: Daniel Scally <djrscally@gmail.com>
Cc: linux-kernel@vger.kernel.org, linux-acpi@vger.kernel.org,
linux-gpio@vger.kernel.org, linux-i2c@vger.kernel.org,
linux-media@vger.kernel.org, devel@acpica.org, rjw@rjwysocki.net,
lenb@kernel.org, gregkh@linuxfoundation.org,
mika.westerberg@linux.intel.com,
andriy.shevchenko@linux.intel.com, linus.walleij@linaro.org,
bgolaszewski@baylibre.com, wsa@kernel.org, yong.zhi@intel.com,
sakari.ailus@linux.intel.com, bingbu.cao@intel.com,
tian.shu.qiu@intel.com, mchehab@kernel.org,
robert.moore@intel.com, erik.kaneda@intel.com, pmladek@suse.com,
rostedt@goodmis.org, sergey.senozhatsky@gmail.com,
linux@rasmusvillemoes.dk,
kieran.bingham+renesas@ideasonboard.com,
jacopo+renesas@jmondi.org,
laurent.pinchart+renesas@ideasonboard.com,
jorhand@linux.microsoft.com, kitakar@gmail.com,
heikki.krogerus@linux.intel.com,
Laurent Pinchart <laurent.pinchart@ideasonboard.com>
Subject: Re: [PATCH 18/18] ipu3: Add driver for dummy INT3472 ACPI device
Date: Mon, 30 Nov 2020 22:52:03 +0200 [thread overview]
Message-ID: <20201130205203.GQ4351@valkosipuli.retiisi.org.uk> (raw)
In-Reply-To: <20201130133129.1024662-19-djrscally@gmail.com>
Hi Daniel,
Thanks for the patch.
On Mon, Nov 30, 2020 at 01:31:29PM +0000, Daniel Scally wrote:
> On platforms where ACPI is designed for use with Windows, resources
> that are intended to be consumed by sensor devices are sometimes in
> the _CRS of a dummy INT3472 device upon which the sensor depends. This
> driver binds to the dummy acpi device (which does not represent a
> physical PMIC) and maps them into GPIO lines and regulators for use by
> the sensor device instead.
>
> Suggested-by: Laurent Pinchart <laurent.pinchart@ideasonboard.com>
> Signed-off-by: Daniel Scally <djrscally@gmail.com>
> ---
> Changes since RFC v3:
>
> - Patch introduced
>
> This patch contains the bits of this process that we're least sure about.
> The sensors in scope for this work are called out as dependent (in their
> DSDT entry's _DEP) on a device with _HID INT3472. These come in at least
> 2 kinds; those with an I2cSerialBusV2 entry (which we presume therefore
> are legitimate tps68470 PMICs that need handling by those drivers - work
> on that in the future). And those without an I2C device. For those without
> an I2C device they instead have an array of GPIO pins defined in _CRS. So
> for example, my Lenovo Miix 510's OVTI2680 sensor is dependent on one of
> the _latter_ kind of INT3472 devices, with this _CRS:
>
> Method (_CRS, 0, NotSerialized) // _CRS: Current Resource Settings
> {
> Name (SBUF, ResourceTemplate ()
> {
> GpioIo (Exclusive, PullDefault, 0x0000, 0x0000,
> IoRestrictionOutputOnly, "\\_SB.PCI0.GPI0",
> 0x00, ResourceConsumer, ,
> )
> { // Pin list
> 0x0079
> }
> GpioIo (Exclusive, PullDefault, 0x0000, 0x0000,
> IoRestrictionOutputOnly, "\\_SB.PCI0.GPI0",
> 0x00, ResourceConsumer, ,
> )
> { // Pin list
> 0x007A
> }
> GpioIo (Exclusive, PullDefault, 0x0000, 0x0000,
> IoRestrictionOutputOnly, "\\_SB.PCI0.GPI0",
> 0x00, ResourceConsumer, ,
> )
> { // Pin list
> 0x008F
> }
> })
> Return (SBUF) /* \_SB_.PCI0.PMI1._CRS.SBUF */
> }
>
> and the same device has a _DSM Method, which returns 32-bit ints where
> the second lowest byte we noticed to match the pin numbers of the GPIO
> lines:
>
> Method (_DSM, 4, NotSerialized) // _DSM: Device-Specific Method
> {
> If ((Arg0 == ToUUID ("79234640-9e10-4fea-a5c1-b5aa8b19756f")))
> {
> If ((Arg2 == One))
> {
> Return (0x03)
> }
>
> If ((Arg2 == 0x02))
> {
> Return (0x01007900)
> }
>
> If ((Arg2 == 0x03))
> {
> Return (0x01007A0C)
> }
>
> If ((Arg2 == 0x04))
> {
> Return (0x01008F01)
> }
> }
>
> Return (Zero)
> }
>
> We know that at least some of those pins have to be toggled active for the
> sensor devices to be available in i2c, so the conclusion we came to was
> that those GPIO entries assigned to the INT3472 device actually represent
> GPIOs and regulators to be consumed by the sensors themselves. Tsuchiya
> noticed that the lowest byte in the return values of the _DSM method
> seemed to represent the type or function of the GPIO line, and we
> confirmed that by testing on each surface device that GPIO lines where the
> low byte in the _DSM entry for that pin was 0x0d controlled the privacy
> LED of the cameras.
>
> We're guessing as to the exact meaning of the function byte, but I
> conclude they're something like this:
>
> 0x00 - probably a reset GPIO
> 0x01 - regulator for the sensor
> 0x0c - regulator for the sensor
> 0x0b - regulator again, but for a VCM or EEPROM
> 0x0d - privacy led (only one we're totally confident of since we can see
> it happen!)
>
> After much internal debate I decided to write this as a standalone
> acpi_driver. Alternative options we considered:
>
> 1. Squash all this into the cio2-bridge code, which I did originally write
> but decided I didn't like.
> 2. Extend the existing tps68470 mfd driver...they share an ACPI ID so this
> kinda makes sense, but ultimately given there is no actual physical
> tps68470 in the scenario this patch handles I decided I didn't like this
> either.
>
> MAINTAINERS | 7 +
> drivers/media/pci/intel/ipu3/Kconfig | 14 +
> drivers/media/pci/intel/ipu3/Makefile | 1 +
> drivers/media/pci/intel/ipu3/int3472.c | 381 +++++++++++++++++++++++++
> drivers/media/pci/intel/ipu3/int3472.h | 96 +++++++
> 5 files changed, 499 insertions(+)
> create mode 100644 drivers/media/pci/intel/ipu3/int3472.c
> create mode 100644 drivers/media/pci/intel/ipu3/int3472.h
>
> diff --git a/MAINTAINERS b/MAINTAINERS
> index 188559a0a610..d73471f9c2a3 100644
> --- a/MAINTAINERS
> +++ b/MAINTAINERS
> @@ -8753,6 +8753,13 @@ L: linux-crypto@vger.kernel.org
> S: Maintained
> F: drivers/crypto/inside-secure/
>
> +INT3472 ACPI DEVICE DRIVER
> +M: Daniel Scally <djrscally@gmail.com>
> +L: linux-media@vger.kernel.org
> +S: Maintained
> +F: drivers/media/pci/intel/ipu3/int3472.c
> +F: drivers/media/pci/intel/ipu3/int3472.h
> +
> INTEGRITY MEASUREMENT ARCHITECTURE (IMA)
> M: Mimi Zohar <zohar@linux.ibm.com>
> M: Dmitry Kasatkin <dmitry.kasatkin@gmail.com>
> diff --git a/drivers/media/pci/intel/ipu3/Kconfig b/drivers/media/pci/intel/ipu3/Kconfig
> index 2b3350d042be..9dd3b280f821 100644
> --- a/drivers/media/pci/intel/ipu3/Kconfig
> +++ b/drivers/media/pci/intel/ipu3/Kconfig
> @@ -34,3 +34,17 @@ config CIO2_BRIDGE
> - Dell 7285
>
> If in doubt, say N here.
> +
> +config INT3472
> + tristate "INT3472 Dummy ACPI Device Driver"
> + depends on VIDEO_IPU3_CIO2
> + depends on ACPI && REGULATOR && GPIOLIB
> + help
> + This module provides an ACPI driver for INT3472 devices that do not
> + represent an actual physical tps68470 device.
> +
> + Say Y here if your device is a detachable / hybrid laptop that comes
> + with Windows installed by the OEM.
> + The module will be called int3472.
> +
> + If in doubt, say N here.
> diff --git a/drivers/media/pci/intel/ipu3/Makefile b/drivers/media/pci/intel/ipu3/Makefile
> index 933777e6ea8a..2285947b2bd2 100644
> --- a/drivers/media/pci/intel/ipu3/Makefile
> +++ b/drivers/media/pci/intel/ipu3/Makefile
> @@ -1,5 +1,6 @@
> # SPDX-License-Identifier: GPL-2.0-only
> obj-$(CONFIG_VIDEO_IPU3_CIO2) += ipu3-cio2.o
> +obj-$(CONFIG_INT3472) += int3472.o
>
> ipu3-cio2-y += ipu3-cio2-main.o
> ipu3-cio2-$(CONFIG_CIO2_BRIDGE) += cio2-bridge.o
> diff --git a/drivers/media/pci/intel/ipu3/int3472.c b/drivers/media/pci/intel/ipu3/int3472.c
> new file mode 100644
> index 000000000000..6b0be75f7f35
> --- /dev/null
> +++ b/drivers/media/pci/intel/ipu3/int3472.c
> @@ -0,0 +1,381 @@
> +// SPDX-License-Identifier: GPL-2.0
> +/* Author: Dan Scally <djrscally@gmail.com> */
> +#include <linux/acpi.h>
> +#include <linux/gpio/consumer.h>
> +#include <linux/gpio/machine.h>
> +#include <linux/i2c.h>
> +#include <linux/kernel.h>
> +#include <linux/list.h>
> +#include <linux/module.h>
> +#include <linux/regulator/driver.h>
> +
> +#include "int3472.h"
> +
> +/*
> + * The regulators have to have .ops to be valid, but the only ops we actually
> + * support are .enable and .disable which are handled via .ena_gpiod. Pass an
> + * empty struct to clear the check without lying about capabilities.
> + */
> +static const struct regulator_ops int3472_gpio_regulator_ops = { 0 };
> +
> +static int int3472_map_gpio_to_sensor(struct int3472_device *int3472,
> + struct acpi_resource *ares, char *func)
> +{
> + char *path = ares->data.gpio.resource_source.string_ptr;
> + struct gpiod_lookup table_entry;
> + struct acpi_device *adev;
> + acpi_handle handle;
> + acpi_status status;
> + int ret;
> +
> + /* Make sure we don't overflow, and leave room for a terminator */
> + if (int3472->n_sensor_gpios >= INT3472_MAX_SENSOR_GPIOS) {
> + dev_warn(&int3472->sensor->dev, "Too many GPIOs mapped\n");
> + return -EINVAL;
> + }
> +
> + /* Fetch ACPI handle for the GPIO chip */
> + status = acpi_get_handle(NULL, path, &handle);
> + if (ACPI_FAILURE(status))
> + return -EINVAL;
> +
> + ret = acpi_bus_get_device(handle, &adev);
> + if (ret)
> + return -ENODEV;
> +
> + table_entry = (struct gpiod_lookup)GPIO_LOOKUP_IDX(acpi_dev_name(adev),
> + ares->data.gpio.pin_table[0],
> + func, 0, GPIO_ACTIVE_HIGH);
> +
> + memcpy(&int3472->gpios.table[int3472->n_sensor_gpios], &table_entry,
> + sizeof(table_entry));
> + int3472->n_sensor_gpios++;
> +
> + return 0;
> +}
> +
> +static struct int3472_sensor_regulator_map *
> +int3472_get_sensor_supply_map(struct int3472_device *int3472)
> +{
> + struct int3472_sensor_regulator_map *ret;
> + union acpi_object *obj;
> + unsigned int i;
> +
> + /*
> + * Sensor modules seem to be identified by a unique string. We use that
> + * to make sure we pass the right device and supply names to the new
> + * regulator's consumer_supplies
> + */
> + obj = acpi_evaluate_dsm_typed(int3472->sensor->handle,
> + &cio2_sensor_module_guid, 0x00,
> + 0x01, NULL, ACPI_TYPE_STRING);
> +
> + if (!obj) {
> + dev_err(&int3472->sensor->dev,
> + "Failed to get sensor module string from _DSM\n");
> + return ERR_PTR(-ENODEV);
> + }
> +
> + if (obj->string.type != ACPI_TYPE_STRING) {
> + dev_err(&int3472->sensor->dev,
> + "Sensor _DSM returned a non-string value\n");
> + ret = ERR_PTR(-EINVAL);
> + goto out_free_obj;
> + }
> +
> + ret = ERR_PTR(-ENODEV);
> + for (i = 0; i < ARRAY_SIZE(int3472_sensor_regulator_maps); i++) {
> + if (!strcmp(int3472_sensor_regulator_maps[i].sensor_module_name,
> + obj->string.pointer)) {
> + ret = &int3472_sensor_regulator_maps[i];
> + goto out_free_obj;
> + }
> + }
> +
> +out_free_obj:
> + ACPI_FREE(obj);
> + return ret;
> +}
> +
> +static int int3472_register_regulator(struct int3472_device *int3472,
> + struct acpi_resource *ares)
> +{
> + char *path = ares->data.gpio.resource_source.string_ptr;
> + struct int3472_sensor_regulator_map *regulator_map;
> + struct regulator_init_data init_data = { };
> + struct int3472_gpio_regulator *regulator;
> + struct regulator_config cfg = { };
> + int ret;
> +
> + /*
> + * We lookup supply names from machine specific tables, based on a
> + * unique identifier in the sensor's _DSM
> + */
> + regulator_map = int3472_get_sensor_supply_map(int3472);
> + if (IS_ERR_OR_NULL(regulator_map)) {
> + dev_err(&int3472->sensor->dev,
> + "Found no supplies defined for this sensor\n");
> + return PTR_ERR(regulator_map);
> + }
> +
> + if (int3472->n_regulators >= regulator_map->n_supplies) {
> + dev_err(&int3472->sensor->dev,
> + "All known supplies are already mapped\n");
> + return -EINVAL;
> + }
> +
> + init_data.supply_regulator = NULL;
> + init_data.constraints.valid_ops_mask = REGULATOR_CHANGE_STATUS;
> + init_data.num_consumer_supplies = 1;
> + init_data.consumer_supplies = ®ulator_map->supplies[int3472->n_regulators];
> +
> + regulator = kmalloc(sizeof(*regulator), GFP_KERNEL);
> + if (!regulator)
> + return -ENOMEM;
> +
> + snprintf(regulator->regulator_name, GPIO_REGULATOR_NAME_LENGTH,
> + "gpio-regulator-%d", int3472->n_regulators);
> + snprintf(regulator->supply_name, GPIO_REGULATOR_SUPPLY_NAME_LENGTH,
> + "supply-%d", int3472->n_regulators);
> +
> + regulator->rdesc = INT3472_REGULATOR(regulator->regulator_name,
> + regulator->supply_name,
> + int3472->n_regulators,
> + &int3472_gpio_regulator_ops);
> +
> + regulator->gpio = acpi_get_gpiod(path, ares->data.gpio.pin_table[0]);
> + if (IS_ERR(regulator->gpio)) {
> + ret = PTR_ERR(regulator->gpio);
> + goto err_free_regulator;
> + }
> +
> + cfg.dev = &int3472->adev->dev;
> + cfg.init_data = &init_data;
> + cfg.ena_gpiod = regulator->gpio;
> +
> + regulator->rdev = regulator_register(®ulator->rdesc, &cfg);
> + if (IS_ERR(regulator->rdev)) {
> + ret = PTR_ERR(regulator->rdev);
> + goto err_free_gpio;
> + }
> +
> + list_add(®ulator->list, &int3472->regulators);
> + int3472->n_regulators++;
> +
> + return 0;
> +
> +err_free_gpio:
> + gpiod_put(regulator->gpio);
> +err_free_regulator:
> + kfree(regulator);
> +
> + return ret;
> +}
> +
> +static int int3472_handle_gpio_resources(struct acpi_resource *ares,
> + void *data)
> +{
> + struct int3472_device *int3472 = data;
> + union acpi_object *obj;
> + int ret = 0;
> +
> + if (ares->type != ACPI_RESOURCE_TYPE_GPIO ||
> + ares->data.gpio.connection_type != ACPI_RESOURCE_GPIO_TYPE_IO)
> + return EINVAL; /* Deliberately positive */
> +
> + /*
> + * n_gpios + 2 because the index of this _DSM function is 1-based and
> + * the first function is just a count.
> + */
> + obj = acpi_evaluate_dsm_typed(int3472->adev->handle,
> + &int3472_gpio_guid, 0x00,
> + int3472->n_gpios + 2,
> + NULL, ACPI_TYPE_INTEGER);
> +
> + if (!obj) {
> + dev_warn(&int3472->adev->dev,
> + "No _DSM entry for this GPIO pin\n");
> + return ENODEV;
> + }
> +
> + switch (obj->integer.value & 0xff) { /* low byte holds type data */
> + case 0x00: /* Purpose unclear, possibly a reset GPIO pin */
> + ret = int3472_map_gpio_to_sensor(int3472, ares, "reset");
> + if (ret)
> + dev_warn(&int3472->adev->dev,
> + "Failed to map reset pin to sensor\n");
> +
> + break;
> + case 0x01: /* Power regulators (we think) */
> + case 0x0c:
> + ret = int3472_register_regulator(int3472, ares);
> + if (ret)
> + dev_warn(&int3472->adev->dev,
> + "Failed to map regulator to sensor\n");
> +
> + break;
> + case 0x0b: /* Power regulators, but to a device separate to sensor */
> + ret = int3472_register_regulator(int3472, ares);
> + if (ret)
> + dev_warn(&int3472->adev->dev,
> + "Failed to map regulator to sensor\n");
> +
> + break;
> + case 0x0d: /* Indicator LEDs */
> + ret = int3472_map_gpio_to_sensor(int3472, ares, "indicator-led");
> + if (ret)
> + dev_warn(&int3472->adev->dev,
> + "Failed to map indicator led to sensor\n");
> +
> + break;
> + default:
> + /* if we've gotten here, we're not sure what they are yet */
> + dev_warn(&int3472->adev->dev,
> + "GPIO type 0x%llx unknown; the sensor may not work\n",
> + (obj->integer.value & 0xff));
> + ret = EINVAL;
> + }
> +
> + int3472->n_gpios++;
> + ACPI_FREE(obj);
> + return abs(ret);
> +}
> +
> +static void int3472_parse_crs(struct int3472_device *int3472)
> +{
> + struct list_head resource_list;
> +
> + INIT_LIST_HEAD(&resource_list);
> +
> + acpi_dev_get_resources(int3472->adev, &resource_list,
> + int3472_handle_gpio_resources, int3472);
> +
> + acpi_dev_free_resource_list(&resource_list);
> + gpiod_add_lookup_table(&int3472->gpios);
> +}
> +
> +static int int3472_add(struct acpi_device *adev)
> +{
> + struct acpi_buffer buffer = { ACPI_ALLOCATE_BUFFER, NULL };
> + struct int3472_device *int3472;
> + struct int3472_cldb cldb;
> + union acpi_object *obj;
> + acpi_status status;
> + int ret = 0;
> +
> + /*
> + * This driver is only intended to support "dummy" INT3472 devices
> + * which appear in ACPI designed for Windows. These are distinguishable
> + * from INT3472 entries representing an actual tps68470 PMIC through
> + * the presence of a CLDB buffer with a particular value set.
> + */
> + status = acpi_evaluate_object(adev->handle, "CLDB", NULL, &buffer);
> + if (ACPI_FAILURE(status))
> + return -ENODEV;
> +
> + obj = buffer.pointer;
> + if (!obj) {
> + dev_err(&adev->dev, "ACPI device has no CLDB object\n");
> + return -ENODEV;
> + }
> +
> + if (obj->type != ACPI_TYPE_BUFFER) {
> + dev_err(&adev->dev, "CLDB object is not an ACPI buffer\n");
> + ret = -EINVAL;
> + goto out_free_buff;
> + }
> +
> + if (obj->buffer.length > sizeof(cldb)) {
> + dev_err(&adev->dev, "The CLDB buffer is too large\n");
> + ret = -EINVAL;
> + goto out_free_buff;
> + }
> +
> + memcpy(&cldb, obj->buffer.pointer, obj->buffer.length);
> +
> + /*
> + * control_logic_type = 1 indicates this is a dummy INT3472 device of
> + * the kind we're looking for. If any other value then we shouldn't try
> + * to handle it
> + */
> + if (cldb.control_logic_type != 1) {
> + ret = -EINVAL;
> + goto out_free_buff;
> + }
> +
> + /* Space for 4 GPIOs - one more than we've seen so far plus a null */
> + int3472 = kzalloc(sizeof(*int3472) +
> + ((INT3472_MAX_SENSOR_GPIOS + 1) * sizeof(struct gpiod_lookup)),
> + GFP_KERNEL);
> + if (!int3472) {
> + ret = -ENOMEM;
> + goto out_free_buff;
> + }
> +
> + int3472->adev = adev;
> + adev->driver_data = int3472;
> +
> + int3472->sensor = acpi_dev_get_next_dep_dev(adev, NULL);
> + if (!int3472->sensor) {
> + dev_err(&adev->dev,
> + "This INT3472 entry seems to have no dependents.\n");
> + ret = -ENODEV;
> + goto out_free_int3472;
> + }
> +
> + int3472->gpios.dev_id = i2c_acpi_dev_name(int3472->sensor);
> +
> + INIT_LIST_HEAD(&int3472->regulators);
> +
> + int3472_parse_crs(int3472);
> +
> + goto out_free_buff;
> +
> +out_free_int3472:
> + kfree(int3472);
> +out_free_buff:
> + kfree(buffer.pointer);
> + return ret;
> +}
> +
> +static int int3472_remove(struct acpi_device *adev)
> +{
> + struct int3472_gpio_regulator *reg;
> + struct int3472_device *int3472;
> +
> + int3472 = acpi_driver_data(adev);
> +
> + acpi_dev_put(int3472->sensor);
> + gpiod_remove_lookup_table(&int3472->gpios);
> +
> + list_for_each_entry(reg, &int3472->regulators, list) {
> + gpiod_put(reg->gpio);
> + regulator_unregister(reg->rdev);
> + }
> +
> + kfree(int3472);
> +
> + return 0;
> +}
> +
> +static const struct acpi_device_id int3472_device_id[] = {
> + { "INT3472", 0 },
The INT3472 _HID is really allocated for the tps68470 PMIC chip. It may not
be used by other drivers; people will want to build kernels where both of
these ACPI table layouts are functional.
Instead, I propose, that you add this as an option to the tps68470 driver
that figures out whether the ACPI device for the tps68470 device actually
describes something else, in a similar fashion you do with the cio2-bridge
driver. I think it may need a separate Kconfig option albeit this and
cio2-bridge cannot be used separately.
> + { },
> +};
> +MODULE_DEVICE_TABLE(acpi, int3472_device_id);
> +
> +static struct acpi_driver int3472_driver = {
> + .name = "int3472",
> + .ids = int3472_device_id,
> + .ops = {
> + .add = int3472_add,
> + .remove = int3472_remove,
> + },
> + .owner = THIS_MODULE,
> +};
> +
> +module_acpi_driver(int3472_driver);
> +
> +MODULE_LICENSE("GPL v2");
> +MODULE_AUTHOR("Dan Scally <djrscally@gmail.com>");
> +MODULE_DESCRIPTION("ACPI Driver for Discrete type INT3472 ACPI Devices");
> diff --git a/drivers/media/pci/intel/ipu3/int3472.h b/drivers/media/pci/intel/ipu3/int3472.h
> new file mode 100644
> index 000000000000..6964726e8e1f
> --- /dev/null
> +++ b/drivers/media/pci/intel/ipu3/int3472.h
> @@ -0,0 +1,96 @@
> +/* SPDX-License-Identifier: GPL-2.0 */
> +/* Author: Dan Scally <djrscally@gmail.com> */
> +#include <linux/regulator/machine.h>
> +
> +#define INT3472_MAX_SENSOR_GPIOS 3
> +#define GPIO_REGULATOR_NAME_LENGTH 17
> +#define GPIO_REGULATOR_SUPPLY_NAME_LENGTH 9
> +
> +#define INT3472_REGULATOR(_NAME, _SUPPLY, _ID, _OPS) \
> + ((const struct regulator_desc) { \
> + .name = _NAME, \
> + .supply_name = _SUPPLY, \
> + .id = _ID, \
> + .type = REGULATOR_VOLTAGE, \
> + .ops = _OPS, \
> + .owner = THIS_MODULE, \
> + })
> +
> +const guid_t int3472_gpio_guid = GUID_INIT(0x79234640, 0x9e10, 0x4fea,
> + 0xa5, 0xc1, 0xb5, 0xaa, 0x8b,
> + 0x19, 0x75, 0x6f);
> +
> +const guid_t cio2_sensor_module_guid = GUID_INIT(0x822ace8f, 0x2814, 0x4174,
> + 0xa5, 0x6b, 0x5f, 0x02, 0x9f,
> + 0xe0, 0x79, 0xee);
> +
> +struct int3472_cldb {
> + u8 version;
> + /*
> + * control logic type
> + * 0: UNKNOWN
> + * 1: DISCRETE(CRD-D)
> + * 2: PMIC TPS68470
> + * 3: PMIC uP6641
> + */
> + u8 control_logic_type;
> + u8 control_logic_id;
> + u8 sensor_card_sku;
> + u8 reserved[28];
> +};
> +
> +struct int3472_device {
> + struct acpi_device *adev;
> + struct acpi_device *sensor;
> +
> + unsigned int n_gpios; /* how many GPIOs have we seen */
> +
> + unsigned int n_regulators;
> + struct list_head regulators;
> +
> + unsigned int n_sensor_gpios; /* how many have we mapped to sensor */
> + struct gpiod_lookup_table gpios;
> +};
> +
> +struct int3472_gpio_regulator {
> + char regulator_name[GPIO_REGULATOR_NAME_LENGTH];
> + char supply_name[GPIO_REGULATOR_SUPPLY_NAME_LENGTH];
> + struct gpio_desc *gpio;
> + struct regulator_dev *rdev;
> + struct regulator_desc rdesc;
> + struct list_head list;
> +};
> +
> +struct int3472_sensor_regulator_map {
> + char *sensor_module_name;
> + unsigned int n_supplies;
> + struct regulator_consumer_supply *supplies;
> +};
> +
> +/*
> + * Here follows platform specific mapping information that we can pass to
> + * regulator_init_data when we register our regulators. They're just mapped
> + * via index, I.E. the first regulator pin that the code finds for the
> + * i2c-OVTI2680:00 device is avdd, the second is dovdd and so on.
> + */
> +
> +static struct regulator_consumer_supply miix_510_ov2680[] = {
> + { "i2c-OVTI2680:00", "avdd" },
> + { "i2c-OVTI2680:00", "dovdd" },
> +};
> +
> +static struct regulator_consumer_supply surface_go2_ov5693[] = {
> + { "i2c-INT33BE:00", "avdd" },
> + { "i2c-INT33BE:00", "dovdd" },
> +};
> +
> +static struct regulator_consumer_supply surface_book_ov5693[] = {
> + { "i2c-INT33BE:00", "avdd" },
> + { "i2c-INT33BE:00", "dovdd" },
> +};
> +
> +static struct int3472_sensor_regulator_map int3472_sensor_regulator_maps[] = {
> + { "GNDF140809R", 2, miix_510_ov2680 },
> + { "YHCU", 2, surface_go2_ov5693 },
> + { "MSHW0070", 2, surface_book_ov5693 },
> +};
--
Kind regards,
Sakari Ailus
next prev parent reply other threads:[~2020-11-30 20:54 UTC|newest]
Thread overview: 144+ messages / expand[flat|nested] mbox.gz Atom feed top
2020-11-30 13:31 [PATCH 00/18] Add functionality to ipu3-cio2 driver allowing software_node connections to sensors on platforms designed for Windows Daniel Scally
2020-11-30 13:31 ` [PATCH 01/18] property: Return true in fwnode_device_is_available for node types that do not implement this operation Daniel Scally
2020-11-30 16:05 ` Laurent Pinchart
2020-11-30 17:21 ` Andy Shevchenko
2020-12-01 3:12 ` Bingbu Cao
2020-12-01 8:46 ` Dan Scally
2020-11-30 13:31 ` [PATCH 02/18] property: Add support for calling fwnode_graph_get_endpoint_by_id() for fwnode->secondary Daniel Scally
2020-11-30 16:08 ` Laurent Pinchart
2020-11-30 17:29 ` Andy Shevchenko
2020-11-30 17:28 ` Laurent Pinchart
2020-11-30 17:53 ` Andy Shevchenko
2020-11-30 18:41 ` Laurent Pinchart
2020-11-30 19:21 ` Andy Shevchenko
2020-12-02 10:13 ` Dan Scally
2020-11-30 13:31 ` [PATCH 03/18] software_node: Fix failure to put() and get() references to children in software_node_get_next_child() Daniel Scally
2020-11-30 16:10 ` Laurent Pinchart
2020-11-30 17:30 ` Andy Shevchenko
2020-11-30 13:31 ` [PATCH 04/18] software_node: Enforce parent before child ordering of nodes array for software_node_register_nodes() Daniel Scally
2020-11-30 16:11 ` Laurent Pinchart
2020-11-30 16:12 ` Laurent Pinchart
2020-11-30 23:10 ` Dan Scally
2020-11-30 17:35 ` Andy Shevchenko
2020-11-30 17:37 ` Andy Shevchenko
2020-11-30 13:31 ` [PATCH 05/18] software_node: Alter software_node_unregister_nodes() to unregister the array in reverse order Daniel Scally
2020-11-30 16:14 ` Laurent Pinchart
2020-11-30 17:45 ` Andy Shevchenko
2020-12-01 23:36 ` Dan Scally
2020-11-30 13:31 ` [PATCH 06/18] software_node: amend software_node_unregister_node_group() to perform unregistration of array in reverse order to be consistent with software_node_unregister_nodes() Daniel Scally
2020-11-30 16:17 ` Laurent Pinchart
2020-11-30 17:47 ` Andy Shevchenko
2020-12-02 10:04 ` Dan Scally
2020-11-30 17:19 ` kernel test robot
2020-12-01 18:18 ` Dan Carpenter
2020-11-30 13:31 ` [PATCH 07/18] software_node: Add support for fwnode_graph*() family of functions Daniel Scally
2020-11-30 16:25 ` Laurent Pinchart
2020-11-30 23:34 ` Dan Scally
2020-11-30 13:31 ` [PATCH 08/18] lib/test_printf.c: Use helper function to unwind array of software_nodes Daniel Scally
2020-11-30 15:54 ` Sergey Senozhatsky
2020-11-30 16:26 ` Laurent Pinchart
2020-11-30 13:31 ` [PATCH 09/18] ipu3-cio2: Add T: entry to MAINTAINERS Daniel Scally
2020-11-30 13:31 ` [PATCH 10/18] ipu3-cio2: Rename ipu3-cio2.c to allow module to be built from multiple source files retaining ipu3-cio2 name Daniel Scally
2020-12-01 6:56 ` Bingbu Cao
2020-12-01 7:07 ` Bingbu Cao
2020-11-30 13:31 ` [PATCH 11/18] media: v4l2-core: v4l2-async: Check possible match in match_fwnode based on sd->fwnode->secondary Daniel Scally
2020-11-30 16:27 ` Laurent Pinchart
2020-11-30 17:55 ` Andy Shevchenko
2020-11-30 13:31 ` [PATCH 12/18] acpi: Add acpi_dev_get_next_match_dev() and macro to iterate through acpi_devices matching a given _HID Daniel Scally
2020-11-30 17:58 ` Andy Shevchenko
2020-11-30 13:31 ` [PATCH 13/18] ipu3-cio2: Add functionality allowing software_node connections to sensors on platforms designed for Windows Daniel Scally
2020-11-30 16:45 ` kernel test robot
2020-11-30 17:09 ` Laurent Pinchart
2020-11-30 18:14 ` Andy Shevchenko
2020-12-01 22:08 ` Dan Scally
2020-12-01 22:11 ` Dan Scally
2020-12-01 22:30 ` Laurent Pinchart
2020-12-01 23:15 ` Dan Scally
2020-12-02 10:38 ` Sakari Ailus
2020-12-02 10:53 ` Dan Scally
2020-12-02 12:02 ` Andy Shevchenko
2020-12-02 22:44 ` Dan Scally
2020-12-02 12:01 ` Andy Shevchenko
2020-11-30 20:27 ` kernel test robot
2020-11-30 20:35 ` Sakari Ailus
2020-12-01 8:13 ` Dan Scally
2020-12-01 15:06 ` Andy Shevchenko
2020-12-15 10:28 ` Daniel Scally
2020-12-15 10:32 ` Daniel Scally
2020-12-15 22:02 ` Sakari Ailus
2020-12-17 14:17 ` Daniel Scally
2020-11-30 13:31 ` [PATCH 14/18] acpi: utils: Add function to fetch dependent acpi_devices Daniel Scally
2020-11-30 18:23 ` Andy Shevchenko
2020-11-30 18:40 ` Laurent Pinchart
2020-11-30 23:54 ` Dan Scally
2020-12-01 15:10 ` Andy Shevchenko
2020-12-01 15:23 ` Dan Scally
2020-11-30 13:31 ` [PATCH 15/18] i2c: i2c-core-acpi: Add i2c_acpi_dev_name() Daniel Scally
2020-11-30 17:11 ` Laurent Pinchart
2020-12-02 22:44 ` Dan Scally
2020-11-30 13:31 ` [PATCH 16/18] i2c: i2c-core-base: Use the new i2c_acpi_dev_name() in i2c_set_dev_name() Daniel Scally
2020-11-30 17:12 ` Laurent Pinchart
2020-11-30 19:18 ` Andy Shevchenko
2020-12-02 9:35 ` Andy Shevchenko
2020-12-02 9:49 ` Dan Scally
2020-12-01 23:50 ` Dan Scally
2020-11-30 13:31 ` [PATCH 17/18] gpio: gpiolib-acpi: Export acpi_get_gpiod() Daniel Scally
2020-11-30 17:02 ` kernel test robot
2020-11-30 18:04 ` kernel test robot
2020-11-30 19:20 ` Andy Shevchenko
2020-11-30 13:31 ` [PATCH 18/18] ipu3: Add driver for dummy INT3472 ACPI device Daniel Scally
2020-11-30 16:17 ` Jean-Michel Hautbois
2020-11-30 23:20 ` Dan Scally
2020-12-01 18:31 ` Andy Shevchenko
2020-12-01 18:36 ` Laurent Pinchart
2020-12-01 18:51 ` Andy Shevchenko
2020-11-30 16:29 ` Kieran Bingham
2020-11-30 17:21 ` Laurent Pinchart
2020-11-30 20:07 ` Andy Shevchenko
2020-11-30 23:32 ` Laurent Pinchart
2020-12-01 15:55 ` Sakari Ailus
2020-12-01 18:37 ` Laurent Pinchart
2020-12-02 11:09 ` Sakari Ailus
2020-12-02 12:42 ` Laurent Pinchart
2020-12-02 15:08 ` Andy Shevchenko
2020-12-03 12:37 ` Dan Scally
2020-12-03 12:57 ` Andy Shevchenko
2020-12-01 18:49 ` Andy Shevchenko
2020-12-01 20:59 ` Dan Scally
2020-12-02 9:39 ` Andy Shevchenko
2020-12-02 12:35 ` Laurent Pinchart
2020-12-02 15:11 ` Andy Shevchenko
2020-12-03 12:25 ` Dan Scally
2020-12-13 22:48 ` Daniel Scally
2020-12-14 15:33 ` Andy Shevchenko
2020-12-01 8:30 ` Dan Scally
2020-12-01 18:54 ` Andy Shevchenko
2020-12-01 18:55 ` Laurent Pinchart
2020-12-01 19:05 ` Andy Shevchenko
2020-12-01 19:06 ` Laurent Pinchart
2020-12-01 19:21 ` Andy Shevchenko
2020-12-01 20:34 ` Hans de Goede
2020-12-01 20:46 ` Andy Shevchenko
2020-12-02 12:48 ` Laurent Pinchart
2020-12-02 15:15 ` Andy Shevchenko
2020-12-01 21:05 ` Dan Scally
2020-12-02 9:42 ` Andy Shevchenko
2021-01-07 23:55 ` Daniel Scally
2021-01-08 12:17 ` Andy Shevchenko
2021-01-08 23:24 ` Daniel Scally
[not found] ` <CAHp75Vcy878xKUUUDH5ory9uS-Vhhx_W1PZc=S6hsSLYJ0i60w@mail.gmail.com>
2021-01-09 9:58 ` Daniel Scally
2021-01-09 0:18 ` Laurent Pinchart
2021-01-09 0:39 ` Daniel Scally
2020-11-30 20:52 ` Sakari Ailus [this message]
2020-11-30 23:06 ` Dan Scally
2020-11-30 23:21 ` Laurent Pinchart
2020-12-01 0:05 ` Dan Scally
2020-12-01 6:44 ` Sakari Ailus
2020-12-01 8:08 ` Dan Scally
2020-12-01 8:09 ` Dan Scally
2020-12-01 12:32 ` Sakari Ailus
2020-12-01 12:40 ` Laurent Pinchart
2020-12-01 12:44 ` Sakari Ailus
2020-12-01 12:48 ` Dan Scally
2020-12-01 19:01 ` Andy Shevchenko
2020-12-01 18:55 ` Andy Shevchenko
Reply instructions:
You may reply publicly to this message via plain-text email
using any one of the following methods:
* Save the following mbox file, import it into your mail client,
and reply-to-all from there: mbox
Avoid top-posting and favor interleaved quoting:
https://en.wikipedia.org/wiki/Posting_style#Interleaved_style
* Reply using the --to, --cc, and --in-reply-to
switches of git-send-email(1):
git send-email \
--in-reply-to=20201130205203.GQ4351@valkosipuli.retiisi.org.uk \
--to=sakari.ailus@iki.fi \
--cc=andriy.shevchenko@linux.intel.com \
--cc=bgolaszewski@baylibre.com \
--cc=bingbu.cao@intel.com \
--cc=devel@acpica.org \
--cc=djrscally@gmail.com \
--cc=erik.kaneda@intel.com \
--cc=gregkh@linuxfoundation.org \
--cc=heikki.krogerus@linux.intel.com \
--cc=jacopo+renesas@jmondi.org \
--cc=jorhand@linux.microsoft.com \
--cc=kieran.bingham+renesas@ideasonboard.com \
--cc=kitakar@gmail.com \
--cc=laurent.pinchart+renesas@ideasonboard.com \
--cc=laurent.pinchart@ideasonboard.com \
--cc=lenb@kernel.org \
--cc=linus.walleij@linaro.org \
--cc=linux-acpi@vger.kernel.org \
--cc=linux-gpio@vger.kernel.org \
--cc=linux-i2c@vger.kernel.org \
--cc=linux-kernel@vger.kernel.org \
--cc=linux-media@vger.kernel.org \
--cc=linux@rasmusvillemoes.dk \
--cc=mchehab@kernel.org \
--cc=mika.westerberg@linux.intel.com \
--cc=pmladek@suse.com \
--cc=rjw@rjwysocki.net \
--cc=robert.moore@intel.com \
--cc=rostedt@goodmis.org \
--cc=sakari.ailus@linux.intel.com \
--cc=sergey.senozhatsky@gmail.com \
--cc=tian.shu.qiu@intel.com \
--cc=wsa@kernel.org \
--cc=yong.zhi@intel.com \
/path/to/YOUR_REPLY
https://kernel.org/pub/software/scm/git/docs/git-send-email.html
* If your mail client supports setting the In-Reply-To header
via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line
before the message body.
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for NNTP newsgroup(s).