summaryrefslogtreecommitdiff
path: root/sys/dev/fdt/fanpwr.c
diff options
context:
space:
mode:
Diffstat (limited to 'sys/dev/fdt/fanpwr.c')
-rw-r--r--sys/dev/fdt/fanpwr.c251
1 files changed, 251 insertions, 0 deletions
diff --git a/sys/dev/fdt/fanpwr.c b/sys/dev/fdt/fanpwr.c
new file mode 100644
index 00000000000..23d40c2e952
--- /dev/null
+++ b/sys/dev/fdt/fanpwr.c
@@ -0,0 +1,251 @@
+/* $OpenBSD: fanpwr.c,v 1.1 2018/06/02 12:19:26 kettenis Exp $ */
+/*
+ * Copyright (c) 2018 Mark Kettenis <kettenis@openbsd.org>
+ *
+ * Permission to use, copy, modify, and 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 <sys/param.h>
+#include <sys/systm.h>
+#include <sys/device.h>
+#include <sys/malloc.h>
+
+#include <dev/ofw/openfirm.h>
+#include <dev/ofw/ofw_regulator.h>
+#include <dev/ofw/fdt.h>
+
+#include <dev/i2c/i2cvar.h>
+
+/* Registers */
+#define FAN53555_VSEL0 0x00
+#define FAN53555_VSEL1 0x01
+#define FAN53555_VSEL_NSEL_MASK 0x3f
+#define FAN53555_CONTROL 0x02
+#define FAN53555_CONTROL_SLEW_MASK (0x7 << 4)
+#define FAN53555_CONTROL_SLEW_SHIFT 4
+#define FAN53555_ID1 0x03
+#define FAN53555_ID2 0x04
+
+/* Distinguish between Failrchild original and Silergy clones. */
+enum fanpwr_id {
+ FANPWR_FAN53555, /* Fairchild FAN53555 */
+ FANPWR_SYR827, /* Silergy SYR827 */
+ FANPWR_SYR828 /* Silergy SYR828 */
+};
+
+struct fanpwr_softc {
+ struct device sc_dev;
+ i2c_tag_t sc_tag;
+ i2c_addr_t sc_addr;
+
+ enum fanpwr_id sc_id;
+ uint8_t sc_vsel;
+
+ struct regulator_device sc_rd;
+ uint32_t sc_vbase;
+ uint32_t sc_vstep;
+};
+
+int fanpwr_match(struct device *, void *, void *);
+void fanpwr_attach(struct device *, struct device *, void *);
+
+struct cfattach fanpwr_ca = {
+ sizeof(struct fanpwr_softc), fanpwr_match, fanpwr_attach
+};
+
+struct cfdriver fanpwr_cd = {
+ NULL, "fanpwr", DV_DULL
+};
+
+uint8_t fanpwr_read(struct fanpwr_softc *, int);
+void fanpwr_write(struct fanpwr_softc *, int, uint8_t);
+uint32_t fanpwr_get_voltage(void *);
+int fanpwr_set_voltage(void *, uint32_t voltage);
+
+int
+fanpwr_match(struct device *parent, void *match, void *aux)
+{
+ struct i2c_attach_args *ia = aux;
+ int node = *(int *)ia->ia_cookie;
+
+ return (OF_is_compatible(node, "fcs,fan53555") ||
+ OF_is_compatible(node, "silergy,syr827") ||
+ OF_is_compatible(node, "silergy,syr828"));
+}
+
+void
+fanpwr_attach(struct device *parent, struct device *self, void *aux)
+{
+ struct fanpwr_softc *sc = (struct fanpwr_softc *)self;
+ struct i2c_attach_args *ia = aux;
+ int node = *(int *)ia->ia_cookie;
+ uint32_t voltage, ramp_delay;
+ uint8_t id1, id2;
+
+ sc->sc_tag = ia->ia_tag;
+ sc->sc_addr = ia->ia_addr;
+
+ if (OF_getpropint(node, "fcs,suspend-voltage-selector", 0))
+ sc->sc_vsel = FAN53555_VSEL0;
+ else
+ sc->sc_vsel = FAN53555_VSEL1;
+
+ if (OF_is_compatible(node, "silergy,syr827")) {
+ printf(": SYR827");
+ sc->sc_id = FANPWR_SYR827;
+ } else if (OF_is_compatible(node, "silergy,syr828")) {
+ printf(": SYR828");
+ sc->sc_id = FANPWR_SYR828;
+ } else {
+ printf(": FAN53555");
+ sc->sc_id = FANPWR_FAN53555;
+ }
+
+ id1 = fanpwr_read(sc, FAN53555_ID1);
+ id2 = fanpwr_read(sc, FAN53555_ID2);
+
+ switch (sc->sc_id) {
+ case FANPWR_FAN53555:
+ switch (id1 << 8 | id2) {
+ case 0x8003: /* 00 Option */
+ case 0x8103: /* 01 Option */
+ case 0x8303: /* 03 Option */
+ case 0x8503: /* 05 Option */
+ case 0x8801: /* 08, 18 Options */
+ case 0x880f: /* BUC08, BUC18 Options */
+ case 0x8108: /* 79 Option */
+ sc->sc_vbase = 600000;
+ sc->sc_vstep = 10000;
+ break;
+ case 0x840f: /* 04 Option */
+ case 0x8c0f: /* 09 Option */
+ sc->sc_vbase = 603000;
+ sc->sc_vstep = 12826;
+ break;
+ case 0x800f: /* 13 Option */
+ sc->sc_vbase = 800000;
+ sc->sc_vstep = 10000;
+ break;
+ case 0x800c: /* 23 Option */
+ sc->sc_vbase = 600000;
+ sc->sc_vstep = 12500;
+ break;
+ case 0x8004: /* 24 Option */
+ sc->sc_vbase = 603000;
+ sc->sc_vstep = 12967;
+ break;
+ default:
+ printf(", unknown ID1 0x%02x ID2 0x%02x\n", id1, id2);
+ return;
+ }
+ break;
+ case FANPWR_SYR827:
+ case FANPWR_SYR828:
+ sc->sc_vbase = 712500;
+ sc->sc_vstep = 12500;
+ break;
+ }
+
+ voltage = fanpwr_get_voltage(sc);
+ printf(", %d.%02d VDC", voltage / 1000000,
+ (voltage % 1000000) / 10000);
+
+ ramp_delay = OF_getpropint(node, "regulator-ramp-delay", 0);
+ if (ramp_delay > 0) {
+ uint8_t ctrl, slew;
+
+ for (slew = 7; slew > 0; slew--)
+ if ((64000 >> slew) >= ramp_delay)
+ break;
+ ctrl = fanpwr_read(sc, FAN53555_CONTROL);
+ ctrl &= ~FAN53555_CONTROL_SLEW_MASK;
+ ctrl |= slew << FAN53555_CONTROL_SLEW_SHIFT;
+ fanpwr_write(sc, FAN53555_CONTROL, ctrl);
+ }
+
+ sc->sc_rd.rd_node = node;
+ sc->sc_rd.rd_cookie = sc;
+ sc->sc_rd.rd_get_voltage = fanpwr_get_voltage;
+ sc->sc_rd.rd_set_voltage = fanpwr_set_voltage;
+ regulator_register(&sc->sc_rd);
+
+ printf("\n");
+}
+
+uint8_t
+fanpwr_read(struct fanpwr_softc *sc, int reg)
+{
+ uint8_t cmd = reg;
+ uint8_t val;
+ int error;
+
+ iic_acquire_bus(sc->sc_tag, I2C_F_POLL);
+ error = iic_exec(sc->sc_tag, I2C_OP_READ_WITH_STOP, sc->sc_addr,
+ &cmd, sizeof cmd, &val, sizeof val, I2C_F_POLL);
+ iic_release_bus(sc->sc_tag, I2C_F_POLL);
+
+ if (error) {
+ printf("error %d\n", error);
+ printf("%s: can't read register 0x%02x\n",
+ sc->sc_dev.dv_xname, reg);
+ val = 0xff;
+ }
+
+ return val;
+}
+
+void
+fanpwr_write(struct fanpwr_softc *sc, int reg, uint8_t val)
+{
+ uint8_t cmd = reg;
+ int error;
+
+ iic_acquire_bus(sc->sc_tag, I2C_F_POLL);
+ error = iic_exec(sc->sc_tag, I2C_OP_WRITE_WITH_STOP, sc->sc_addr,
+ &cmd, sizeof cmd, &val, sizeof val, I2C_F_POLL);
+ iic_release_bus(sc->sc_tag, I2C_F_POLL);
+
+ if (error) {
+ printf("%s: can't write register 0x%02x\n",
+ sc->sc_dev.dv_xname, reg);
+ }
+}
+
+uint32_t
+fanpwr_get_voltage(void *cookie)
+{
+ struct fanpwr_softc *sc = cookie;
+ uint8_t vsel;
+
+ vsel = fanpwr_read(sc, sc->sc_vsel);
+ return sc->sc_vbase + (vsel & FAN53555_VSEL_NSEL_MASK) * sc->sc_vstep;
+}
+
+int
+fanpwr_set_voltage(void *cookie, uint32_t voltage)
+{
+ struct fanpwr_softc *sc = cookie;
+ uint32_t vmin = sc->sc_vbase;
+ uint32_t vmax = vmin + (FAN53555_VSEL_NSEL_MASK) * sc->sc_vstep;
+ uint8_t vsel;
+
+ if (voltage < vmin || voltage > vmax)
+ return EINVAL;
+
+ vsel = fanpwr_read(sc, sc->sc_vsel);
+ vsel &= ~FAN53555_VSEL_NSEL_MASK;
+ vsel |= (voltage - sc->sc_vbase) / sc->sc_vsel;
+ fanpwr_write(sc, sc->sc_vsel, vsel);
+
+ return 0;
+}