/* $OpenBSD: dwiic_acpi.c,v 1.18 2021/12/21 20:53:46 kettenis Exp $ */ /* * Synopsys DesignWare I2C controller * * Copyright (c) 2015, 2016 joshua stein * * Permission to use, copy, modify, and/or distribute this software for any * purpose with or without fee is hereby granted, provided that the above * copyright notice and this permission notice appear in all copies. * * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. */ #include #include #include #include #include #include #include #include #include struct dwiic_crs { int irq_int; uint8_t irq_flags; uint16_t i2c_addr; struct aml_node *devnode; struct aml_node *gpio_int_node; uint16_t gpio_int_pin; uint16_t gpio_int_flags; }; int dwiic_acpi_match(struct device *, void *, void *); void dwiic_acpi_attach(struct device *, struct device *, void *); int dwiic_acpi_parse_crs(int, union acpi_resource *, void *); int dwiic_acpi_found_ihidev(struct dwiic_softc *, struct aml_node *, char *, struct dwiic_crs); int dwiic_acpi_found_iatp(struct dwiic_softc *, struct aml_node *, char *, struct dwiic_crs); void dwiic_acpi_get_params(struct dwiic_softc *, char *, uint16_t *, uint16_t *, uint32_t *); void dwiic_acpi_power(struct dwiic_softc *, int); void dwiic_acpi_bus_scan(struct device *, struct i2cbus_attach_args *, void *); struct cfattach dwiic_acpi_ca = { sizeof(struct dwiic_softc), dwiic_acpi_match, dwiic_acpi_attach, NULL, dwiic_activate }; const char *dwiic_hids[] = { "AMDI0010", "APMC0D0F", "INT33C2", "INT33C3", "INT3432", "INT3433", "80860F41", "808622C1", NULL }; const char *ihidev_hids[] = { "PNP0C50", "ACPI0C50", NULL }; const char *iatp_hids[] = { "ATML0000", "ATML0001", NULL }; int dwiic_acpi_match(struct device *parent, void *match, void *aux) { struct acpi_attach_args *aaa = aux; struct cfdata *cf = match; if (aaa->aaa_naddr < 1) return 0; return acpi_matchhids(aaa, dwiic_hids, cf->cf_driver->cd_name); } void dwiic_acpi_attach(struct device *parent, struct device *self, void *aux) { struct dwiic_softc *sc = (struct dwiic_softc *)self; struct acpi_attach_args *aaa = aux; struct aml_value res; struct dwiic_crs crs; sc->sc_acpi = (struct acpi_softc *)parent; sc->sc_devnode = aaa->aaa_node; memcpy(&sc->sc_hid, aaa->aaa_dev, sizeof(sc->sc_hid)); printf(" %s", sc->sc_devnode->name); if (aml_evalname(sc->sc_acpi, sc->sc_devnode, "_CRS", 0, NULL, &res)) { printf(", no _CRS method\n"); return; } if (res.type != AML_OBJTYPE_BUFFER || res.length < 5) { printf(", invalid _CRS object (type %d len %d)\n", res.type, res.length); aml_freevalue(&res); return; } memset(&crs, 0, sizeof(crs)); crs.devnode = sc->sc_devnode; aml_parse_resource(&res, dwiic_acpi_parse_crs, &crs); aml_freevalue(&res); printf(" addr 0x%llx/0x%llx", aaa->aaa_addr[0], aaa->aaa_size[0]); sc->sc_iot = aaa->aaa_bst[0]; if (bus_space_map(sc->sc_iot, aaa->aaa_addr[0], aaa->aaa_size[0], 0, &sc->sc_ioh)) { printf(": can't map registers\n"); return; } /* power up the controller */ dwiic_acpi_power(sc, 1); /* fetch timing parameters */ dwiic_acpi_get_params(sc, "SSCN", &sc->ss_hcnt, &sc->ss_lcnt, NULL); dwiic_acpi_get_params(sc, "FMCN", &sc->fs_hcnt, &sc->fs_lcnt, &sc->sda_hold_time); if (dwiic_init(sc)) { printf(", failed initializing\n"); bus_space_unmap(sc->sc_iot, sc->sc_ioh, aaa->aaa_size[0]); return; } /* leave the controller disabled */ dwiic_write(sc, DW_IC_INTR_MASK, 0); dwiic_enable(sc, 0); dwiic_read(sc, DW_IC_CLR_INTR); /* try to register interrupt with apic, but not fatal without it */ if (aaa->aaa_nirq > 0) { printf(" irq %d", aaa->aaa_irq[0]); sc->sc_ih = acpi_intr_establish(aaa->aaa_irq[0], aaa->aaa_irq_flags[0], IPL_BIO, dwiic_intr, sc, sc->sc_dev.dv_xname); if (sc->sc_ih == NULL) printf(": can't establish interrupt"); } printf("\n"); rw_init(&sc->sc_i2c_lock, "iiclk"); /* setup and attach iic bus */ sc->sc_i2c_tag.ic_cookie = sc; sc->sc_i2c_tag.ic_acquire_bus = dwiic_i2c_acquire_bus; sc->sc_i2c_tag.ic_release_bus = dwiic_i2c_release_bus; sc->sc_i2c_tag.ic_exec = dwiic_i2c_exec; sc->sc_i2c_tag.ic_intr_establish = dwiic_i2c_intr_establish; sc->sc_i2c_tag.ic_intr_string = dwiic_i2c_intr_string; bzero(&sc->sc_iba, sizeof(sc->sc_iba)); sc->sc_iba.iba_name = "iic"; sc->sc_iba.iba_tag = &sc->sc_i2c_tag; sc->sc_iba.iba_bus_scan = dwiic_acpi_bus_scan; sc->sc_iba.iba_bus_scan_arg = sc; config_found((struct device *)sc, &sc->sc_iba, iicbus_print); #ifndef SMALL_KERNEL sc->sc_devnode->i2c = &sc->sc_i2c_tag; acpi_register_gsb(sc->sc_acpi, sc->sc_devnode); #endif } int dwiic_acpi_parse_crs(int crsidx, union acpi_resource *crs, void *arg) { struct dwiic_crs *sc_crs = arg; struct aml_node *node; uint16_t pin; uint8_t flags; switch (AML_CRSTYPE(crs)) { case SR_IRQ: sc_crs->irq_int = ffs(letoh16(crs->sr_irq.irq_mask)) - 1; /* Default is exclusive, active-high, edge triggered. */ if (AML_CRSLEN(crs) < 4) flags = SR_IRQ_MODE; else flags = crs->sr_irq.irq_flags; /* Map flags to those of the extended interrupt descriptor. */ if (flags & SR_IRQ_SHR) sc_crs->irq_flags |= LR_EXTIRQ_SHR; if (flags & SR_IRQ_POLARITY) sc_crs->irq_flags |= LR_EXTIRQ_POLARITY; if (flags & SR_IRQ_MODE) sc_crs->irq_flags |= LR_EXTIRQ_MODE; break; case LR_EXTIRQ: sc_crs->irq_int = letoh32(crs->lr_extirq.irq[0]); sc_crs->irq_flags = crs->lr_extirq.flags; break; case LR_GPIO: node = aml_searchname(sc_crs->devnode, (char *)&crs->pad[crs->lr_gpio.res_off]); pin = *(uint16_t *)&crs->pad[crs->lr_gpio.pin_off]; if (crs->lr_gpio.type == LR_GPIO_INT) { sc_crs->gpio_int_node = node; sc_crs->gpio_int_pin = pin; sc_crs->gpio_int_flags = crs->lr_gpio.tflags; } break; case LR_SERBUS: if (crs->lr_serbus.type == LR_SERBUS_I2C) sc_crs->i2c_addr = letoh16(crs->lr_i2cbus._adr); break; case LR_MEM32: case LR_MEM32FIXED: break; default: DPRINTF(("%s: unknown resource type %d\n", __func__, AML_CRSTYPE(crs))); } return 0; } void dwiic_acpi_get_params(struct dwiic_softc *sc, char *method, uint16_t *hcnt, uint16_t *lcnt, uint32_t *sda_hold_time) { struct aml_value res; if (!aml_searchname(sc->sc_devnode, method)) return; if (aml_evalname(sc->sc_acpi, sc->sc_devnode, method, 0, NULL, &res)) { printf(": eval of %s at %s failed", method, aml_nodename(sc->sc_devnode)); return; } if (res.type != AML_OBJTYPE_PACKAGE) { printf(": %s is not a package (%d)", method, res.type); aml_freevalue(&res); return; } if (res.length <= 2) { printf(": %s returned package of len %d", method, res.length); aml_freevalue(&res); return; } *hcnt = aml_val2int(res.v_package[0]); *lcnt = aml_val2int(res.v_package[1]); if (sda_hold_time) *sda_hold_time = aml_val2int(res.v_package[2]); aml_freevalue(&res); } void dwiic_acpi_bus_scan(struct device *iic, struct i2cbus_attach_args *iba, void *aux) { struct dwiic_softc *sc = (struct dwiic_softc *)aux; sc->sc_iic = iic; aml_find_node(sc->sc_devnode, "_HID", dwiic_acpi_found_hid, sc); } void * dwiic_i2c_intr_establish(void *cookie, void *ih, int level, int (*func)(void *), void *arg, const char *name) { struct dwiic_crs *crs = ih; if (crs->gpio_int_node) { if (!crs->gpio_int_node->gpio) /* found ACPI device but no driver for it */ return NULL; struct acpi_gpio *gpio = crs->gpio_int_node->gpio; gpio->intr_establish(gpio->cookie, crs->gpio_int_pin, crs->gpio_int_flags, func, arg); return ih; } return acpi_intr_establish(crs->irq_int, crs->irq_flags, level, func, arg, name); } const char * dwiic_i2c_intr_string(void *cookie, void *ih) { struct dwiic_crs *crs = ih; static char irqstr[64]; if (crs->gpio_int_node) { if (crs->gpio_int_node->gpio) snprintf(irqstr, sizeof(irqstr), "gpio %d", crs->gpio_int_pin); } else snprintf(irqstr, sizeof(irqstr), "irq %d", crs->irq_int); return irqstr; } int dwiic_matchhids(const char *hid, const char *hids[]) { int i; for (i = 0; hids[i]; i++) if (!strcmp(hid, hids[i])) return (1); return (0); } int dwiic_acpi_found_hid(struct aml_node *node, void *arg) { struct dwiic_softc *sc = (struct dwiic_softc *)arg; struct dwiic_crs crs; struct aml_value res; int64_t sta; char cdev[16], dev[16]; struct i2c_attach_args ia; /* Skip our own _HID. */ if (node->parent == sc->sc_devnode) return 0; if (acpi_parsehid(node, arg, cdev, dev, 16) != 0) return 0; if (aml_evalinteger(acpi_softc, node->parent, "_STA", 0, NULL, &sta)) sta = STA_PRESENT | STA_ENABLED | STA_DEV_OK | 0x1000; if ((sta & STA_PRESENT) == 0) return 0; DPRINTF(("%s: found HID %s at %s\n", sc->sc_dev.dv_xname, dev, aml_nodename(node))); if (aml_evalname(acpi_softc, node->parent, "_CRS", 0, NULL, &res)) return 0; if (res.type != AML_OBJTYPE_BUFFER || res.length < 5) { printf("%s: invalid _CRS object (type %d len %d)\n", sc->sc_dev.dv_xname, res.type, res.length); aml_freevalue(&res); return (0); } memset(&crs, 0, sizeof(crs)); crs.devnode = sc->sc_devnode; aml_parse_resource(&res, dwiic_acpi_parse_crs, &crs); aml_freevalue(&res); acpi_attach_deps(acpi_softc, node->parent); if (dwiic_matchhids(cdev, ihidev_hids)) return dwiic_acpi_found_ihidev(sc, node, dev, crs); else if (dwiic_matchhids(dev, iatp_hids)) return dwiic_acpi_found_iatp(sc, node, dev, crs); memset(&ia, 0, sizeof(ia)); ia.ia_tag = sc->sc_iba.iba_tag; ia.ia_name = dev; ia.ia_addr = crs.i2c_addr; ia.ia_cookie = node->parent; if (crs.irq_int != 0 || crs.gpio_int_node != NULL) ia.ia_intr = &crs; config_found(sc->sc_iic, &ia, dwiic_i2c_print); node->parent->attached = 1; return 0; } int dwiic_acpi_found_ihidev(struct dwiic_softc *sc, struct aml_node *node, char *dev, struct dwiic_crs crs) { struct i2c_attach_args ia; struct aml_value cmd[4], res; /* 3cdff6f7-4267-4555-ad05-b30a3d8938de */ static uint8_t i2c_hid_guid[] = { 0xF7, 0xF6, 0xDF, 0x3C, 0x67, 0x42, 0x55, 0x45, 0xAD, 0x05, 0xB3, 0x0A, 0x3D, 0x89, 0x38, 0xDE, }; if (!aml_searchname(node->parent, "_DSM")) { printf("%s: couldn't find _DSM at %s\n", sc->sc_dev.dv_xname, aml_nodename(node->parent)); return 0; } bzero(&cmd, sizeof(cmd)); cmd[0].type = AML_OBJTYPE_BUFFER; cmd[0].v_buffer = (uint8_t *)&i2c_hid_guid; cmd[0].length = sizeof(i2c_hid_guid); /* rev */ cmd[1].type = AML_OBJTYPE_INTEGER; cmd[1].v_integer = 1; cmd[1].length = 1; /* func */ cmd[2].type = AML_OBJTYPE_INTEGER; cmd[2].v_integer = 1; /* HID */ cmd[2].length = 1; /* not used */ cmd[3].type = AML_OBJTYPE_PACKAGE; cmd[3].length = 0; if (aml_evalname(acpi_softc, node->parent, "_DSM", 4, cmd, &res)) { printf("%s: eval of _DSM at %s failed\n", sc->sc_dev.dv_xname, aml_nodename(node->parent)); return 0; } if (res.type != AML_OBJTYPE_INTEGER) { printf("%s: bad _DSM result at %s: %d\n", sc->sc_dev.dv_xname, aml_nodename(node->parent), res.type); aml_freevalue(&res); return 0; } memset(&ia, 0, sizeof(ia)); ia.ia_tag = sc->sc_iba.iba_tag; ia.ia_size = 1; ia.ia_name = "ihidev"; ia.ia_size = aml_val2int(&res); /* hid descriptor address */ ia.ia_addr = crs.i2c_addr; ia.ia_cookie = dev; aml_freevalue(&res); if (sc->sc_poll_ihidev) ia.ia_poll = 1; if (!(crs.irq_int == 0 && crs.gpio_int_node == NULL)) ia.ia_intr = &crs; if (config_found(sc->sc_iic, &ia, dwiic_i2c_print)) { node->parent->attached = 1; return 0; } return 1; } int dwiic_acpi_found_iatp(struct dwiic_softc *sc, struct aml_node *node, char *dev, struct dwiic_crs crs) { struct i2c_attach_args ia; struct aml_value res; if (aml_evalname(acpi_softc, node->parent, "GPIO", 0, NULL, &res)) /* no gpio, assume this is the bootloader interface */ return (0); memset(&ia, 0, sizeof(ia)); ia.ia_tag = sc->sc_iba.iba_tag; ia.ia_size = 1; ia.ia_name = "iatp"; ia.ia_addr = crs.i2c_addr; ia.ia_cookie = dev; if (crs.irq_int <= 0 && crs.gpio_int_node == NULL) { printf("%s: couldn't find irq for %s\n", sc->sc_dev.dv_xname, aml_nodename(node->parent)); return 0; } ia.ia_intr = &crs; if (config_found(sc->sc_iic, &ia, dwiic_i2c_print)) { node->parent->attached = 1; return 0; } return 1; } void dwiic_acpi_power(struct dwiic_softc *sc, int power) { char ps[] = "_PS0"; if (!power) ps[3] = '3'; if (aml_searchname(sc->sc_devnode, ps)) { if (aml_evalname(sc->sc_acpi, sc->sc_devnode, ps, 0, NULL, NULL)) { printf("%s: failed powering %s with %s\n", sc->sc_dev.dv_xname, power ? "on" : "off", ps); return; } DELAY(10000); /* 10 milliseconds */ } else DPRINTF(("%s: no %s method\n", sc->sc_dev.dv_xname, ps)); if (strcmp(sc->sc_hid, "INT3432") == 0 || strcmp(sc->sc_hid, "INT3433") == 0) { /* * XXX: broadwell i2c devices may need this for initial power * up and/or after s3 resume. * * linux does this write via LPSS -> clk_register_gate -> * clk_gate_enable -> clk_gate_endisable -> clk_writel */ dwiic_write(sc, 0x800, 1); } }