]> git.dujemihanovic.xyz Git - u-boot.git/commitdiff
acpi: Support generation of I2C descriptor
authorSimon Glass <sjg@chromium.org>
Tue, 7 Jul 2020 19:11:48 +0000 (13:11 -0600)
committerBin Meng <bmeng.cn@gmail.com>
Fri, 17 Jul 2020 06:32:24 +0000 (14:32 +0800)
Add a function to write a GPIO descriptor to the generated ACPI code.

Signed-off-by: Simon Glass <sjg@chromium.org>
Reviewed-by: Wolfgang Wallner <wolfgang.wallner@br-automation.com>
Reviewed-by: Bin Meng <bmeng.cn@gmail.com>
drivers/i2c/sandbox_i2c.c
drivers/rtc/sandbox_rtc.c
include/acpi/acpi_device.h
lib/acpi/acpi_device.c
test/dm/acpigen.c

index f4ae2397a03f98822e4515cbb47b295c626712a3..125026da9083a198b0072a6d2f67954b89004d1d 100644 (file)
@@ -11,6 +11,7 @@
 #include <i2c.h>
 #include <log.h>
 #include <asm/test.h>
+#include <dm/acpi.h>
 #include <dm/lists.h>
 #include <dm/device-internal.h>
 
@@ -83,6 +84,15 @@ static int sandbox_i2c_xfer(struct udevice *bus, struct i2c_msg *msg,
        return ops->xfer(emul, msg, nmsgs);
 }
 
+static int sandbox_i2c_get_name(const struct udevice *dev, char *out_name)
+{
+       return acpi_copy_name(out_name, "SI2C");
+}
+
+struct acpi_ops sandbox_i2c_acpi_ops = {
+       .get_name       = sandbox_i2c_get_name,
+};
+
 static const struct dm_i2c_ops sandbox_i2c_ops = {
        .xfer           = sandbox_i2c_xfer,
 };
@@ -98,4 +108,5 @@ U_BOOT_DRIVER(i2c_sandbox) = {
        .of_match = sandbox_i2c_ids,
        .ops    = &sandbox_i2c_ops,
        .priv_auto_alloc_size = sizeof(struct sandbox_i2c_priv),
+       ACPI_OPS_PTR(&sandbox_i2c_acpi_ops)
 };
index 77065e49c76975a356fbbc322b43f1d00bcb7a62..852770a49cf8d9b97696b317c3f53f3b427b7684 100644 (file)
@@ -9,6 +9,7 @@
 #include <i2c.h>
 #include <rtc.h>
 #include <asm/rtc.h>
+#include <dm/acpi.h>
 
 #define REG_COUNT 0x80
 
@@ -67,6 +68,17 @@ static int sandbox_rtc_write8(struct udevice *dev, unsigned int reg, int val)
        return dm_i2c_reg_write(dev, reg, val);
 }
 
+#if CONFIG_IS_ENABLED(ACPIGEN)
+static int sandbox_rtc_get_name(const struct udevice *dev, char *out_name)
+{
+       return acpi_copy_name(out_name, "RTCC");
+}
+
+struct acpi_ops sandbox_rtc_acpi_ops = {
+       .get_name       = sandbox_rtc_get_name,
+};
+#endif
+
 static const struct rtc_ops sandbox_rtc_ops = {
        .get = sandbox_rtc_get,
        .set = sandbox_rtc_set,
@@ -85,4 +97,5 @@ U_BOOT_DRIVER(rtc_sandbox) = {
        .id     = UCLASS_RTC,
        .of_match = sandbox_rtc_ids,
        .ops    = &sandbox_rtc_ops,
+       ACPI_OPS_PTR(&sandbox_rtc_acpi_ops)
 };
index 67a242eb75d67ca0154eb9476e17df4005f02ffd..7a65dadbb2f4d364ebf5d539a4c3e149f9e2c130 100644 (file)
@@ -9,6 +9,7 @@
 #ifndef __ACPI_DEVICE_H
 #define __ACPI_DEVICE_H
 
+#include <i2c.h>
 #include <linux/bitops.h>
 
 struct acpi_ctx;
@@ -183,6 +184,26 @@ struct acpi_gpio {
        enum acpi_gpio_polarity polarity;
 };
 
+/* ACPI Descriptors for Serial Bus interfaces */
+#define ACPI_SERIAL_BUS_TYPE_I2C               1
+#define ACPI_I2C_SERIAL_BUS_REVISION_ID                1 /* TODO: upgrade to 2 */
+#define ACPI_I2C_TYPE_SPECIFIC_REVISION_ID     1
+
+/**
+ * struct acpi_i2c - representation of an ACPI I2C device
+ *
+ * @address: 7-bit or 10-bit I2C address
+ * @mode_10bit: Which address size is used
+ * @speed: Bus speed in Hz
+ * @resource: Resource name for the I2C controller
+ */
+struct acpi_i2c {
+       u16 address;
+       enum i2c_address_mode mode_10bit;
+       enum i2c_speed_rate speed;
+       const char *resource;
+};
+
 /**
  * acpi_device_path() - Get the full path to an ACPI device
  *
@@ -270,4 +291,16 @@ int acpi_device_write_gpio_desc(struct acpi_ctx *ctx,
 int acpi_device_write_interrupt_or_gpio(struct acpi_ctx *ctx,
                                        struct udevice *dev, const char *prop);
 
+/**
+ * acpi_device_write_i2c_dev() - Write an I2C device to ACPI
+ *
+ * This creates a I2cSerialBus descriptor for an I2C device, including
+ * information ACPI needs to use it.
+ *
+ * @ctx: ACPI context pointer
+ * @dev: I2C device to write
+ * @return I2C address of device if OK, -ve on error
+ */
+int acpi_device_write_i2c_dev(struct acpi_ctx *ctx, const struct udevice *dev);
+
 #endif
index c93af2e58370c9016b47aab6d06b6057b661f940..c0d5a9acaed63ccfb112542cf263f5e8b101f404 100644 (file)
@@ -385,3 +385,110 @@ int acpi_device_write_interrupt_or_gpio(struct acpi_ctx *ctx,
 
        return pin;
 }
+
+/* ACPI 6.3 section 6.4.3.8.2.1 - I2cSerialBus() */
+static void acpi_device_write_i2c(struct acpi_ctx *ctx,
+                                 const struct acpi_i2c *i2c)
+{
+       void *desc_length, *type_length;
+
+       /* Byte 0: Descriptor Type */
+       acpigen_emit_byte(ctx, ACPI_DESCRIPTOR_SERIAL_BUS);
+
+       /* Byte 1+2: Length (filled in later) */
+       desc_length = largeres_write_len_f(ctx);
+
+       /* Byte 3: Revision ID */
+       acpigen_emit_byte(ctx, ACPI_I2C_SERIAL_BUS_REVISION_ID);
+
+       /* Byte 4: Resource Source Index is Reserved */
+       acpigen_emit_byte(ctx, 0);
+
+       /* Byte 5: Serial Bus Type is I2C */
+       acpigen_emit_byte(ctx, ACPI_SERIAL_BUS_TYPE_I2C);
+
+       /*
+        * Byte 6: Flags
+        *  [7:2]: 0 => Reserved
+        *    [1]: 1 => ResourceConsumer
+        *    [0]: 0 => ControllerInitiated
+        */
+       acpigen_emit_byte(ctx, 1 << 1);
+
+       /*
+        * Byte 7-8: Type Specific Flags
+        *   [15:1]: 0 => Reserved
+        *      [0]: 0 => 7bit, 1 => 10bit
+        */
+       acpigen_emit_word(ctx, i2c->mode_10bit);
+
+       /* Byte 9: Type Specific Revision ID */
+       acpigen_emit_byte(ctx, ACPI_I2C_TYPE_SPECIFIC_REVISION_ID);
+
+       /* Byte 10-11: I2C Type Data Length */
+       type_length = largeres_write_len_f(ctx);
+
+       /* Byte 12-15: I2C Bus Speed */
+       acpigen_emit_dword(ctx, i2c->speed);
+
+       /* Byte 16-17: I2C Slave Address */
+       acpigen_emit_word(ctx, i2c->address);
+
+       /* Fill in Type Data Length */
+       largeres_fill_len(ctx, type_length);
+
+       /* Byte 18+: ResourceSource */
+       acpigen_emit_string(ctx, i2c->resource);
+
+       /* Fill in I2C Descriptor Length */
+       largeres_fill_len(ctx, desc_length);
+}
+
+/**
+ * acpi_device_set_i2c() - Set up an ACPI I2C struct from a device
+ *
+ * The value of @scope is not copied, but only referenced. This implies the
+ * caller has to ensure it stays valid for the lifetime of @i2c.
+ *
+ * @dev: I2C device to convert
+ * @i2c: Place to put the new structure
+ * @scope: Scope of the I2C device (this is the controller path)
+ * @return chip address of device
+ */
+static int acpi_device_set_i2c(const struct udevice *dev, struct acpi_i2c *i2c,
+                              const char *scope)
+{
+       struct dm_i2c_chip *chip = dev_get_parent_platdata(dev);
+       struct udevice *bus = dev_get_parent(dev);
+
+       memset(i2c, '\0', sizeof(*i2c));
+       i2c->address = chip->chip_addr;
+       i2c->mode_10bit = 0;
+
+       /*
+        * i2c_bus->speed_hz is set if this device is probed, but if not we
+        * must use the device tree
+        */
+       i2c->speed = dev_read_u32_default(bus, "clock-frequency",
+                                         I2C_SPEED_STANDARD_RATE);
+       i2c->resource = scope;
+
+       return i2c->address;
+}
+
+int acpi_device_write_i2c_dev(struct acpi_ctx *ctx, const struct udevice *dev)
+{
+       char scope[ACPI_PATH_MAX];
+       struct acpi_i2c i2c;
+       int ret;
+
+       ret = acpi_device_scope(dev, scope, sizeof(scope));
+       if (ret)
+               return log_msg_ret("scope", ret);
+       ret = acpi_device_set_i2c(dev, &i2c, scope);
+       if (ret < 0)
+               return log_msg_ret("set", ret);
+       acpi_device_write_i2c(ctx, &i2c);
+
+       return ret;
+}
index 7d81652295e291be5e58dcc90edd4dbbdce6b602..c210ceb404cd5f0965d0322a0101c626a66ed6e8 100644 (file)
@@ -276,3 +276,35 @@ static int dm_test_acpi_interrupt_or_gpio(struct unit_test_state *uts)
 }
 DM_TEST(dm_test_acpi_interrupt_or_gpio,
        DM_TESTF_SCAN_PDATA | DM_TESTF_SCAN_FDT);
+
+/* Test emitting an I2C descriptor */
+static int dm_test_acpi_i2c(struct unit_test_state *uts)
+{
+       struct acpi_ctx *ctx;
+       struct udevice *dev;
+       u8 *ptr;
+
+       ut_assertok(alloc_context(&ctx));
+
+       ptr = acpigen_get_current(ctx);
+
+       ut_assertok(uclass_get_device(UCLASS_RTC, 0, &dev));
+       ut_asserteq(0x43, acpi_device_write_i2c_dev(ctx, dev));
+       ut_asserteq(28, acpigen_get_current(ctx) - ptr);
+       ut_asserteq(ACPI_DESCRIPTOR_SERIAL_BUS, ptr[0]);
+       ut_asserteq(25, get_unaligned((u16 *)(ptr + 1)));
+       ut_asserteq(ACPI_I2C_SERIAL_BUS_REVISION_ID, ptr[3]);
+       ut_asserteq(0, ptr[4]);
+       ut_asserteq(ACPI_SERIAL_BUS_TYPE_I2C, ptr[5]);
+       ut_asserteq(0, get_unaligned((u16 *)(ptr + 7)));
+       ut_asserteq(ACPI_I2C_TYPE_SPECIFIC_REVISION_ID, ptr[9]);
+       ut_asserteq(6, get_unaligned((u16 *)(ptr + 10)));
+       ut_asserteq(100000, get_unaligned((u32 *)(ptr + 12)));
+       ut_asserteq(0x43, get_unaligned((u16 *)(ptr + 16)));
+       ut_asserteq_str("\\_SB.I2C0", (char *)ptr + 18);
+
+       free_context(&ctx);
+
+       return 0;
+}
+DM_TEST(dm_test_acpi_i2c, DM_TESTF_SCAN_PDATA | DM_TESTF_SCAN_FDT);