summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorIan Lepore <ian@FreeBSD.org>2018-05-04 19:28:05 +0000
committerIan Lepore <ian@FreeBSD.org>2018-05-04 19:28:05 +0000
commitd8cf9c4f8b7abad78e3359aa50ea171dbadb403c (patch)
treea67c53fbd745c6eca4a487249053fb757c4a4379
parent8fd222ebb4be73c7478d44d4b415b634a1841f8b (diff)
Notes
-rw-r--r--sys/arm/freescale/imx/imx_gpio.c30
1 files changed, 25 insertions, 5 deletions
diff --git a/sys/arm/freescale/imx/imx_gpio.c b/sys/arm/freescale/imx/imx_gpio.c
index 503bd0e3800e..5e0ad3430ce4 100644
--- a/sys/arm/freescale/imx/imx_gpio.c
+++ b/sys/arm/freescale/imx/imx_gpio.c
@@ -506,21 +506,41 @@ static void
imx51_gpio_pin_configure(struct imx51_gpio_softc *sc, struct gpio_pin *pin,
unsigned int flags)
{
- u_int newflags;
+ u_int newflags, pad;
mtx_lock_spin(&sc->sc_mtx);
/*
- * Manage input/output; other flags not supported yet.
+ * Manage input/output; other flags not supported yet (maybe not ever,
+ * since we have no connection to the pad config registers from here).
+ *
+ * When setting a pin to output, honor the PRESET_[LOW,HIGH] flags if
+ * present. Otherwise, for glitchless transistions on pins with pulls,
+ * read the current state of the pad and preset the DR register to drive
+ * the current value onto the pin before enabling the pin for output.
*
* Note that changes to pin->gp_flags must be acccumulated in newflags
* and stored with a single writeback to gp_flags at the end, to enable
- * unlocked reads of that value elsewhere.
+ * unlocked reads of that value elsewhere. This is only about unlocked
+ * access to gp_flags from elsewhere; we still use locking in this
+ * function to protect r-m-w access to the hardware registers.
*/
if (flags & (GPIO_PIN_INPUT | GPIO_PIN_OUTPUT)) {
newflags = pin->gp_flags & ~(GPIO_PIN_INPUT | GPIO_PIN_OUTPUT);
if (flags & GPIO_PIN_OUTPUT) {
+ if (flags & GPIO_PIN_PRESET_LOW) {
+ pad = 0;
+ } else if (flags & GPIO_PIN_PRESET_HIGH) {
+ pad = 1;
+ } else {
+ if (flags & GPIO_PIN_OPENDRAIN)
+ pad = READ4(sc, IMX_GPIO_PSR_REG);
+ else
+ pad = READ4(sc, IMX_GPIO_DR_REG);
+ pad = (pad >> pin->gp_pin) & 1;
+ }
newflags |= GPIO_PIN_OUTPUT;
+ SET4(sc, IMX_GPIO_DR_REG, (pad << pin->gp_pin));
SET4(sc, IMX_GPIO_OE_REG, (1U << pin->gp_pin));
} else {
newflags |= GPIO_PIN_INPUT;
@@ -692,7 +712,7 @@ imx51_gpio_pin_access_32(device_t dev, uint32_t first_pin, uint32_t clear_pins,
sc = device_get_softc(dev);
if (orig_pins != NULL)
- *orig_pins = READ4(sc, IMX_GPIO_PSR_REG);
+ *orig_pins = READ4(sc, IMX_GPIO_DR_REG);
if ((clear_pins | change_pins) != 0) {
mtx_lock_spin(&sc->sc_mtx);
@@ -718,7 +738,7 @@ imx51_gpio_pin_config_32(device_t dev, uint32_t first_pin, uint32_t num_pins,
return (EINVAL);
drclr = drset = oeclr = oeset = 0;
- pads = READ4(sc, IMX_GPIO_PSR_REG);
+ pads = READ4(sc, IMX_GPIO_DR_REG);
for (i = 0; i < num_pins; ++i) {
bit = 1u << i;