diff options
Diffstat (limited to 'sys/arch/armv7/imx/imxanatop.c')
-rw-r--r-- | sys/arch/armv7/imx/imxanatop.c | 106 |
1 files changed, 106 insertions, 0 deletions
diff --git a/sys/arch/armv7/imx/imxanatop.c b/sys/arch/armv7/imx/imxanatop.c index 41ab44f9ce2..99bea63417d 100644 --- a/sys/arch/armv7/imx/imxanatop.c +++ b/sys/arch/armv7/imx/imxanatop.c @@ -24,14 +24,37 @@ #include <dev/ofw/openfirm.h> #include <dev/ofw/ofw_misc.h> +#include <dev/ofw/ofw_regulator.h> #include <dev/ofw/fdt.h> +#define HREAD4(sc, reg) \ + (bus_space_read_4((sc)->sc_iot, (sc)->sc_ioh, (reg))) +#define HWRITE4(sc, reg, val) \ + bus_space_write_4((sc)->sc_iot, (sc)->sc_ioh, (reg), (val)) + struct imxanatop_softc { struct device sc_dev; bus_space_tag_t sc_iot; bus_space_handle_t sc_ioh; }; +struct imxanatop_regulator { + struct imxanatop_softc *ir_sc; + + uint32_t ir_reg_offset; + uint32_t ir_vol_bit_shift; + uint32_t ir_vol_bit_width; + uint32_t ir_min_bit_val; + uint32_t ir_min_voltage; + uint32_t ir_max_voltage; + + uint32_t ir_delay_reg_offset; + uint32_t ir_delay_bit_shift; + uint32_t ir_delay_bit_width; + + struct regulator_device ir_rd; +}; + int imxanatop_match(struct device *, void *, void *); void imxanatop_attach(struct device *, struct device *, void *); @@ -43,6 +66,10 @@ struct cfdriver imxanatop_cd = { NULL, "imxanatop", DV_DULL }; +void imxanatop_attach_regulator(struct imxanatop_softc *, int); +uint32_t imxanatop_get_voltage(void *); +int imxanatop_set_voltage(void *, uint32_t); + int imxanatop_match(struct device *parent, void *match, void *aux) { @@ -59,6 +86,7 @@ imxanatop_attach(struct device *parent, struct device *self, void *aux) { struct imxanatop_softc *sc = (struct imxanatop_softc *)self; struct fdt_attach_args *faa = aux; + int node; if (faa->fa_nreg < 1) { printf(": no registers\n"); @@ -77,4 +105,82 @@ imxanatop_attach(struct device *parent, struct device *self, void *aux) faa->fa_reg[0].size); printf("\n"); + + for (node = OF_child(faa->fa_node); node; node = OF_peer(node)) + if (OF_is_compatible(node, "fsl,anatop-regulator")) + imxanatop_attach_regulator(sc, node); +} + +void +imxanatop_attach_regulator(struct imxanatop_softc *sc, int node) +{ + struct imxanatop_regulator *ir; + + ir = malloc(sizeof(*ir), M_DEVBUF, M_WAITOK | M_ZERO); + ir->ir_sc = sc; + + ir->ir_reg_offset = OF_getpropint(node, "anatop-reg-offset", -1); + ir->ir_vol_bit_shift = OF_getpropint(node, "anatop-vol-bit-shift", -1); + ir->ir_vol_bit_width = OF_getpropint(node, "anatop-vol-bit-width", -1); + ir->ir_min_bit_val = OF_getpropint(node, "anatop-min-bit-val", -1); + ir->ir_min_voltage = OF_getpropint(node, "anatop-min-voltage", -1); + ir->ir_max_voltage = OF_getpropint(node, "anatop-max-voltage", -1); + if (ir->ir_reg_offset == -1 || ir->ir_vol_bit_shift == -1 || + ir->ir_vol_bit_width == -1 || ir->ir_min_bit_val == -1 || + ir->ir_min_voltage == -1 || ir->ir_max_voltage == -1) + return; + + ir->ir_delay_reg_offset = + OF_getpropint(node, "anatop-delay-reg-offset", 0); + ir->ir_delay_bit_shift = + OF_getpropint(node, "anatop-delay-bit-shift", 0); + ir->ir_delay_bit_width = + OF_getpropint(node, "anatop-delay-bit-width", 0); + + ir->ir_rd.rd_node = node; + ir->ir_rd.rd_cookie = ir; + ir->ir_rd.rd_get_voltage = imxanatop_get_voltage; + ir->ir_rd.rd_set_voltage = imxanatop_set_voltage; + regulator_register(&ir->ir_rd); +} + +uint32_t +imxanatop_get_voltage(void *cookie) +{ + struct imxanatop_regulator *ir = cookie; + uint32_t bit_val; + + bit_val = HREAD4(ir->ir_sc, ir->ir_reg_offset) >> ir->ir_vol_bit_shift; + bit_val &= ((1 << ir->ir_vol_bit_width) - 1); + return (ir->ir_min_voltage + (bit_val - ir->ir_min_bit_val) * 25000); +} + +int +imxanatop_set_voltage(void *cookie, uint32_t voltage) +{ + struct imxanatop_regulator *ir = cookie; + uint32_t bit_val, old_bit_val, reg; + int steps, usecs; + + if (voltage < ir->ir_min_voltage || voltage > ir->ir_max_voltage) + return -1; + + bit_val = ir->ir_min_bit_val + (voltage - ir->ir_min_voltage) / 25000; + reg = HREAD4(ir->ir_sc, ir->ir_reg_offset); + old_bit_val = (reg >> ir->ir_vol_bit_shift); + old_bit_val &= ((1 << ir->ir_vol_bit_width) -1); + reg &= ~((1 << ir->ir_vol_bit_width) - 1) << ir->ir_vol_bit_shift; + reg |= (bit_val << ir->ir_vol_bit_shift); + HWRITE4(ir->ir_sc, ir->ir_reg_offset, reg); + + steps = bit_val - old_bit_val; + if (steps > 0 && ir->ir_delay_bit_width > 0) { + reg = HREAD4(ir->ir_sc, ir->ir_delay_reg_offset); + reg >>= ir->ir_delay_bit_shift; + reg &= ((1 << ir->ir_delay_bit_width) - 1); + usecs = ((reg + 1) * steps * 64 * 1000000) / 24000000; + delay(usecs); + } + + return 0; } |