Source-Changes-HG archive
[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index][Old Index]
[src/trunk]: src/sys/arch/arm/sunxi Enable sunxi pinctrl support
details: https://anonhg.NetBSD.org/src/rev/6699763f1067
branches: trunk
changeset: 354873:6699763f1067
user: jmcneill <jmcneill%NetBSD.org@localhost>
date: Sun Jul 02 15:28:25 2017 +0000
description:
Enable sunxi pinctrl support
diffstat:
sys/arch/arm/sunxi/sun6i_a31_gpio.c | 30 +++++++-
sys/arch/arm/sunxi/sunxi_gpio.c | 130 +++++++++++++++++++++++++++++++++++-
sys/arch/arm/sunxi/sunxi_gpio.h | 3 +-
3 files changed, 158 insertions(+), 5 deletions(-)
diffs (258 lines):
diff -r 6a65d12e24ac -r 6699763f1067 sys/arch/arm/sunxi/sun6i_a31_gpio.c
--- a/sys/arch/arm/sunxi/sun6i_a31_gpio.c Sun Jul 02 15:27:58 2017 +0000
+++ b/sys/arch/arm/sunxi/sun6i_a31_gpio.c Sun Jul 02 15:28:25 2017 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: sun6i_a31_gpio.c,v 1.1 2017/07/02 13:36:46 jmcneill Exp $ */
+/* $NetBSD: sun6i_a31_gpio.c,v 1.2 2017/07/02 15:28:25 jmcneill Exp $ */
/*-
* Copyright (c) 2016 Emmanuel Vadot <manu%freebsd.org@localhost>
@@ -28,7 +28,7 @@
*/
#include <sys/cdefs.h>
-__KERNEL_RCSID(0, "$NetBSD: sun6i_a31_gpio.c,v 1.1 2017/07/02 13:36:46 jmcneill Exp $");
+__KERNEL_RCSID(0, "$NetBSD: sun6i_a31_gpio.c,v 1.2 2017/07/02 15:28:25 jmcneill Exp $");
#include <sys/param.h>
#include <sys/systm.h>
@@ -212,7 +212,33 @@
{"PH30", 7, 30, {"gpio_in", "gpio_out", "nand1", NULL, NULL, NULL, NULL, NULL}},
};
+static const struct sunxi_gpio_pins a31_r_pins[] = {
+ {"PL0", 0, 0, {"gpio_in", "gpio_out", "s_twi", "s_p2wi", NULL, NULL, NULL, NULL}},
+ {"PL1", 0, 1, {"gpio_in", "gpio_out", "s_twi", "s_p2wi", NULL, NULL, NULL, NULL}},
+ {"PL2", 0, 2, {"gpio_in", "gpio_out", "s_uart", NULL, NULL, NULL, NULL, NULL}},
+ {"PL3", 0, 3, {"gpio_in", "gpio_out", "s_uart", NULL, NULL, NULL, NULL, NULL}},
+ {"PL4", 0, 4, {"gpio_in", "gpio_out", "s_ir", NULL, NULL, NULL, NULL, NULL}},
+ {"PL5", 0, 5, {"gpio_in", "gpio_out", "pl_eint0", "s_jtag", NULL, NULL, NULL, NULL}, 2, 0},
+ {"PL6", 0, 6, {"gpio_in", "gpio_out", "pl_eint1", "s_jtag", NULL, NULL, NULL, NULL}, 2, 1},
+ {"PL7", 0, 7, {"gpio_in", "gpio_out", "pl_eint2", "s_jtag", NULL, NULL, NULL, NULL}, 2, 2},
+ {"PL8", 0, 8, {"gpio_in", "gpio_out", "pl_eint3", "s_jtag", NULL, NULL, NULL, NULL}, 2, 3},
+
+ {"PM0", 1, 0, {"gpio_in", "gpio_out", "pm_eint0", NULL, NULL, NULL, NULL, NULL}, 2, 0},
+ {"PM1", 1, 1, {"gpio_in", "gpio_out", "pm_eint1", NULL, NULL, NULL, NULL, NULL}, 2, 1},
+ {"PM2", 1, 2, {"gpio_in", "gpio_out", "pm_eint2", "1wire", NULL, NULL, NULL, NULL}, 2, 2},
+ {"PM3", 1, 3, {"gpio_in", "gpio_out", "pm_eint3", NULL, NULL, NULL, NULL, NULL}, 2, 3},
+ {"PM4", 1, 4, {"gpio_in", "gpio_out", "pm_eint4", NULL, NULL, NULL, NULL, NULL}, 2, 4},
+ {"PM5", 1, 5, {"gpio_in", "gpio_out", "pm_eint5", NULL, NULL, NULL, NULL, NULL}, 2, 5},
+ {"PM6", 1, 6, {"gpio_in", "gpio_out", "pm_eint6", NULL, NULL, NULL, NULL, NULL}, 2, 6},
+ {"PM7", 1, 7, {"gpio_in", "gpio_out", "pm_eint7", "rtc", NULL, NULL, NULL, NULL}, 2, 7},
+};
+
const struct sunxi_gpio_padconf sun6i_a31_padconf = {
.npins = __arraycount(a31_pins),
.pins = a31_pins,
};
+
+const struct sunxi_gpio_padconf sun6i_a31_r_padconf = {
+ .npins = __arraycount(a31_r_pins),
+ .pins = a31_r_pins,
+};
diff -r 6a65d12e24ac -r 6699763f1067 sys/arch/arm/sunxi/sunxi_gpio.c
--- a/sys/arch/arm/sunxi/sunxi_gpio.c Sun Jul 02 15:27:58 2017 +0000
+++ b/sys/arch/arm/sunxi/sunxi_gpio.c Sun Jul 02 15:28:25 2017 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: sunxi_gpio.c,v 1.1 2017/07/02 13:36:46 jmcneill Exp $ */
+/* $NetBSD: sunxi_gpio.c,v 1.2 2017/07/02 15:28:25 jmcneill Exp $ */
/*-
* Copyright (c) 2017 Jared McNeill <jmcneill%invisible.ca@localhost>
@@ -29,7 +29,7 @@
#include "opt_soc.h"
#include <sys/cdefs.h>
-__KERNEL_RCSID(0, "$NetBSD: sunxi_gpio.c,v 1.1 2017/07/02 13:36:46 jmcneill Exp $");
+__KERNEL_RCSID(0, "$NetBSD: sunxi_gpio.c,v 1.2 2017/07/02 15:28:25 jmcneill Exp $");
#include <sys/param.h>
#include <sys/bus.h>
@@ -49,11 +49,17 @@
#define SUNXI_GPIO_CFG_PINMASK(pin) (0x7 << (((pin) % 8) * 4))
#define SUNXI_GPIO_DATA(port) (SUNXI_GPIO_PORT(port) + 0x10)
#define SUNXI_GPIO_DRV(port, pin) (SUNXI_GPIO_PORT(port) + 0x14 + (0x4 * ((pin) / 16)))
+#define SUNXI_GPIO_DRV_PINMASK(pin) (0x3 << (((pin) % 16) * 4))
#define SUNXI_GPIO_PULL(port, pin) (SUNXI_GPIO_PORT(port) + 0x1c + (0x4 * ((pin) / 16)))
+#define SUNXI_GPIO_PULL_DISABLE 0
+#define SUNXI_GPIO_PULL_UP 1
+#define SUNXI_GPIO_PULL_DOWN 2
+#define SUNXI_GPIO_PULL_PINMASK(pin) (0x3 << (((pin) % 16) * 4))
static const struct of_compat_data compat_data[] = {
#ifdef SOC_SUN6I_A31
{ "allwinner,sun6i-a31-pinctrl", (uintptr_t)&sun6i_a31_padconf },
+ { "allwinner,sun6i-a31-r-pinctrl", (uintptr_t)&sun6i_a31_r_padconf },
#endif
#ifdef SOC_SUN8I_H3
{ "allwinner,sun8i-h3-pinctrl", (uintptr_t)&sun8i_h3_padconf },
@@ -102,6 +108,21 @@
return NULL;
}
+static const struct sunxi_gpio_pins *
+sunxi_gpio_lookup_byname(struct sunxi_gpio_softc *sc, const char *name)
+{
+ const struct sunxi_gpio_pins *pin_def;
+ u_int n;
+
+ for (n = 0; n < sc->sc_padconf->npins; n++) {
+ pin_def = &sc->sc_padconf->pins[n];
+ if (strcmp(pin_def->name, name) == 0)
+ return pin_def;
+ }
+
+ return NULL;
+}
+
static int
sunxi_gpio_setfunc(struct sunxi_gpio_softc *sc,
const struct sunxi_gpio_pins *pin_def, const char *func)
@@ -132,6 +153,48 @@
}
static int
+sunxi_gpio_setpull(struct sunxi_gpio_softc *sc,
+ const struct sunxi_gpio_pins *pin_def, int flags)
+{
+ uint32_t pull;
+
+ const bus_size_t pull_reg = SUNXI_GPIO_PULL(pin_def->port, pin_def->pin);
+ const uint32_t pull_mask = SUNXI_GPIO_PULL_PINMASK(pin_def->pin);
+
+ pull = GPIO_READ(sc, pull_reg);
+ pull &= ~pull_mask;
+ if (flags & GPIO_PIN_PULLUP)
+ pull |= __SHIFTIN(SUNXI_GPIO_PULL_UP, pull_mask);
+ else if (flags & GPIO_PIN_PULLDOWN)
+ pull |= __SHIFTIN(SUNXI_GPIO_PULL_DOWN, pull_mask);
+ else
+ pull |= __SHIFTIN(SUNXI_GPIO_PULL_DISABLE, pull_mask);
+ GPIO_WRITE(sc, pull_reg, pull);
+
+ return 0;
+}
+
+static int
+sunxi_gpio_setdrv(struct sunxi_gpio_softc *sc,
+ const struct sunxi_gpio_pins *pin_def, int drive_strength)
+{
+ uint32_t drv;
+
+ if (drive_strength < 10 || drive_strength > 40)
+ return EINVAL;
+
+ const bus_size_t drv_reg = SUNXI_GPIO_DRV(pin_def->port, pin_def->pin);
+ const uint32_t drv_mask = SUNXI_GPIO_DRV_PINMASK(pin_def->pin);
+
+ drv = GPIO_READ(sc, drv_reg);
+ drv &= ~drv_mask;
+ drv |= __SHIFTIN((drive_strength / 10) - 1, drv_mask);
+ GPIO_WRITE(sc, drv_reg, drv);
+
+ return 0;
+}
+
+static int
sunxi_gpio_ctl(struct sunxi_gpio_softc *sc, const struct sunxi_gpio_pins *pin_def,
int flags)
{
@@ -256,6 +319,60 @@
};
static int
+sunxi_pinctrl_set_config(device_t dev, const void *data, size_t len)
+{
+ struct sunxi_gpio_softc * const sc = device_private(dev);
+ const struct sunxi_gpio_pins *pin_def;
+ u_int drive_strength;
+
+ if (len != 4)
+ return -1;
+
+ const int phandle = fdtbus_get_phandle_from_native(be32dec(data));
+
+ /*
+ * Required: pins, function
+ * Optional: bias-disable, bias-pull-up, bias-pull-down, drive-strength
+ */
+
+ const char *function = fdtbus_get_string(phandle, "function");
+ if (function == NULL)
+ return -1;
+ int pins_len = OF_getproplen(phandle, "pins");
+ if (pins_len <= 0)
+ return -1;
+ const char *pins = fdtbus_get_string(phandle, "pins");
+
+ for (pins = fdtbus_get_string(phandle, "pins");
+ pins_len > 0;
+ pins_len -= strlen(pins) + 1, pins += strlen(pins) + 1) {
+ pin_def = sunxi_gpio_lookup_byname(sc, pins);
+ if (pin_def == NULL) {
+ aprint_error_dev(dev, "unknown pin name '%s'\n", pins);
+ continue;
+ }
+ if (sunxi_gpio_setfunc(sc, pin_def, function) != 0)
+ continue;
+
+ if (of_hasprop(phandle, "bias-disable"))
+ sunxi_gpio_setpull(sc, pin_def, 0);
+ else if (of_hasprop(phandle, "bias-pull-up"))
+ sunxi_gpio_setpull(sc, pin_def, GPIO_PIN_PULLUP);
+ else if (of_hasprop(phandle, "bias-pull-down"))
+ sunxi_gpio_setpull(sc, pin_def, GPIO_PIN_PULLDOWN);
+
+ if (of_getprop_uint32(phandle, "drive-strength", &drive_strength) == 0)
+ sunxi_gpio_setdrv(sc, pin_def, drive_strength);
+ }
+
+ return 0;
+}
+
+static struct fdtbus_pinctrl_controller_func sunxi_pinctrl_funcs = {
+ .set_config = sunxi_pinctrl_set_config,
+};
+
+static int
sunxi_gpio_match(device_t parent, cfdata_t cf, void *aux)
{
struct fdt_attach_args * const faa = aux;
@@ -271,6 +388,7 @@
const int phandle = faa->faa_phandle;
bus_addr_t addr;
bus_size_t size;
+ int child;
if (fdtbus_get_reg(phandle, 0, &addr, &size) != 0) {
aprint_error(": couldn't get registers\n");
@@ -289,4 +407,12 @@
aprint_normal(": PIO\n");
fdtbus_register_gpio_controller(self, phandle, &sunxi_gpio_funcs);
+
+ for (child = OF_child(phandle); child; child = OF_peer(child)) {
+ if (!of_hasprop(child, "function") || !of_hasprop(child, "pins"))
+ continue;
+ fdtbus_register_pinctrl_config(self, child, &sunxi_pinctrl_funcs);
+ }
+
+ fdtbus_pinctrl_configure();
}
diff -r 6a65d12e24ac -r 6699763f1067 sys/arch/arm/sunxi/sunxi_gpio.h
--- a/sys/arch/arm/sunxi/sunxi_gpio.h Sun Jul 02 15:27:58 2017 +0000
+++ b/sys/arch/arm/sunxi/sunxi_gpio.h Sun Jul 02 15:28:25 2017 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: sunxi_gpio.h,v 1.1 2017/07/02 13:36:46 jmcneill Exp $ */
+/* $NetBSD: sunxi_gpio.h,v 1.2 2017/07/02 15:28:25 jmcneill Exp $ */
/*-
* Copyright (c) 2017 Jared McNeill <jmcneill%invisible.ca@localhost>
@@ -52,6 +52,7 @@
#ifdef SOC_SUN6I_A31
extern const struct sunxi_gpio_padconf sun6i_a31_padconf;
+extern const struct sunxi_gpio_padconf sun6i_a31_r_padconf;
#endif
#ifdef SOC_SUN8I_H3
Home |
Main Index |
Thread Index |
Old Index