diff options
Diffstat (limited to 'sys/dev/gpio')
| -rw-r--r-- | sys/dev/gpio/acpi_gpiobus.c | 168 | ||||
| -rw-r--r-- | sys/dev/gpio/acpi_gpiobusvar.h | 6 | ||||
| -rw-r--r-- | sys/dev/gpio/bytgpio.c | 3 | ||||
| -rw-r--r-- | sys/dev/gpio/chvgpio.c | 3 | ||||
| -rw-r--r-- | sys/dev/gpio/dwgpio/dwgpio.c | 3 | ||||
| -rw-r--r-- | sys/dev/gpio/gpio_if.m | 26 | ||||
| -rw-r--r-- | sys/dev/gpio/gpioaei.c | 204 | ||||
| -rw-r--r-- | sys/dev/gpio/gpiobus.c | 162 | ||||
| -rw-r--r-- | sys/dev/gpio/gpiobus_if.m | 30 | ||||
| -rw-r--r-- | sys/dev/gpio/gpiobus_internal.h | 3 | ||||
| -rw-r--r-- | sys/dev/gpio/gpiobusvar.h | 1 | ||||
| -rw-r--r-- | sys/dev/gpio/gpioc.c | 191 | ||||
| -rw-r--r-- | sys/dev/gpio/gpioled.c | 106 | ||||
| -rw-r--r-- | sys/dev/gpio/ofw_gpiobus.c | 17 | ||||
| -rw-r--r-- | sys/dev/gpio/pl061.c | 6 | ||||
| -rw-r--r-- | sys/dev/gpio/qoriq_gpio.c | 3 | 
16 files changed, 652 insertions, 280 deletions
| diff --git a/sys/dev/gpio/acpi_gpiobus.c b/sys/dev/gpio/acpi_gpiobus.c index f9468e0deda0..0d2455cab399 100644 --- a/sys/dev/gpio/acpi_gpiobus.c +++ b/sys/dev/gpio/acpi_gpiobus.c @@ -37,6 +37,7 @@  #include <dev/gpio/gpiobusvar.h>  #include <dev/gpio/acpi_gpiobusvar.h>  #include <dev/gpio/gpiobus_internal.h> +#include <sys/sbuf.h>  #include "gpiobus_if.h" @@ -52,12 +53,11 @@ struct acpi_gpiobus_ctx {  struct acpi_gpiobus_ivar  { -	struct gpiobus_ivar	gpiobus;	/* Must come first */ -	ACPI_HANDLE		dev_handle;	/* ACPI handle for bus */ -	uint32_t		flags; +	struct gpiobus_ivar	gpiobus; +	ACPI_HANDLE		handle;  }; -static uint32_t +uint32_t  acpi_gpiobus_convflags(ACPI_RESOURCE_GPIO *gpio_res)  {  	uint32_t flags = 0; @@ -150,70 +150,24 @@ acpi_gpiobus_enumerate_res(ACPI_RESOURCE *res, void *context)  	return (AE_OK);  } -static struct acpi_gpiobus_ivar * -acpi_gpiobus_setup_devinfo(device_t bus, device_t child, -    ACPI_RESOURCE_GPIO *gpio_res) -{ -	struct acpi_gpiobus_ivar *devi; - -	devi = malloc(sizeof(*devi), M_DEVBUF, M_NOWAIT | M_ZERO); -	if (devi == NULL) -		return (NULL); -	resource_list_init(&devi->gpiobus.rl); - -	devi->flags = acpi_gpiobus_convflags(gpio_res); -	if (acpi_quirks & ACPI_Q_AEI_NOPULL) -		devi->flags &= ~GPIO_PIN_PULLUP; - -	devi->gpiobus.npins = 1; -	if (gpiobus_alloc_ivars(&devi->gpiobus) != 0) { -		free(devi, M_DEVBUF); -		return (NULL); -	} - -	for (int i = 0; i < devi->gpiobus.npins; i++) -		devi->gpiobus.pins[i] = gpio_res->PinTable[i]; - -	return (devi); -} -  static ACPI_STATUS  acpi_gpiobus_enumerate_aei(ACPI_RESOURCE *res, void *context)  {  	ACPI_RESOURCE_GPIO *gpio_res = &res->Data.Gpio; -	struct acpi_gpiobus_ctx *ctx = context; -	device_t bus = ctx->sc->sc_busdev; -	device_t child; -	struct acpi_gpiobus_ivar *devi; +	uint32_t *npins = context, *pins = npins + 1; -	/* Check that we have a GpioInt object. */ +	/* +	 * Check that we have a GpioInt object. +	 * Note that according to the spec this +	 * should always be the case. +	 */  	if (res->Type != ACPI_RESOURCE_TYPE_GPIO)  		return (AE_OK);  	if (gpio_res->ConnectionType != ACPI_RESOURCE_GPIO_TYPE_INT)  		return (AE_OK); -	/* Add a child. */ -	child = device_add_child_ordered(bus, 0, "gpio_aei", DEVICE_UNIT_ANY); -	if (child == NULL) -		return (AE_OK); -	devi = acpi_gpiobus_setup_devinfo(bus, child, gpio_res); -	if (devi == NULL) { -		device_delete_child(bus, child); -		return (AE_OK); -	} -	device_set_ivars(child, devi); - -	for (int i = 0; i < devi->gpiobus.npins; i++) { -		if (GPIOBUS_PIN_SETFLAGS(bus, child, 0, devi->flags & -		    ~GPIO_INTR_MASK)) { -			device_delete_child(bus, child); -			return (AE_OK); -		} -	} - -	/* Pass ACPI information to children. */ -	devi->dev_handle = ctx->dev_handle; - +	for (int i = 0; i < gpio_res->PinTableLength; i++) +		pins[(*npins)++] = gpio_res->PinTable[i];  	return (AE_OK);  } @@ -296,6 +250,63 @@ err:  	return (AE_BAD_PARAMETER);  } +static void +acpi_gpiobus_attach_aei(struct acpi_gpiobus_softc *sc, ACPI_HANDLE handle) +{ +	struct acpi_gpiobus_ivar *devi; +	ACPI_HANDLE aei_handle; +	device_t child; +	uint32_t *pins; +	ACPI_STATUS status; +	int err; + +	status = AcpiGetHandle(handle, "_AEI", &aei_handle); +	if (ACPI_FAILURE(status)) +		return; + +	/* pins[0] specifies the length of the array. */ +	pins = mallocarray(sc->super_sc.sc_npins + 1, +	    sizeof(uint32_t), M_DEVBUF, M_WAITOK); +	pins[0] = 0; + +	status = AcpiWalkResources(handle, "_AEI", +	    acpi_gpiobus_enumerate_aei, pins); +	if (ACPI_FAILURE(status)) { +		device_printf(sc->super_sc.sc_busdev, +		    "Failed to enumerate AEI resources\n"); +		free(pins, M_DEVBUF); +		return; +	} + +	child = BUS_ADD_CHILD(sc->super_sc.sc_busdev, 0, "gpio_aei", +	    DEVICE_UNIT_ANY); +	if (child == NULL) { +		device_printf(sc->super_sc.sc_busdev, +		    "Failed to add gpio_aei child\n"); +		free(pins, M_DEVBUF); +		return; +	} + +	devi = device_get_ivars(child); +	devi->gpiobus.npins = pins[0]; +	devi->handle = aei_handle; + +	err = gpiobus_alloc_ivars(&devi->gpiobus); +	if (err != 0) { +		device_printf(sc->super_sc.sc_busdev, +		    "Failed to allocate gpio_aei ivars\n"); +		device_delete_child(sc->super_sc.sc_busdev, child); +		free(pins, M_DEVBUF); +		return; +	} + +	for (int i = 0; i < pins[0]; i++) +		devi->gpiobus.pins[i] = pins[i + 1]; +	free(pins, M_DEVBUF); + +	bus_attach_children(sc->super_sc.sc_busdev); +} +  static int  acpi_gpiobus_probe(device_t dev)  { @@ -353,13 +364,8 @@ acpi_gpiobus_attach(device_t dev)  	if (ACPI_FAILURE(status))  		device_printf(dev, "Failed to enumerate GPIO resources\n"); -	/* Look for AEI children */ -	status = AcpiWalkResources(handle, "_AEI", acpi_gpiobus_enumerate_aei, -	    &ctx); - -	if (ACPI_FAILURE(status)) -		device_printf(dev, "Failed to enumerate AEI resources\n"); - +	/* Look for AEI child */ +	acpi_gpiobus_attach_aei(sc, handle);  	return (0);  } @@ -383,16 +389,14 @@ acpi_gpiobus_detach(device_t dev)  }  static int -acpi_gpiobus_read_ivar(device_t dev, device_t child, int which, uintptr_t *result) +acpi_gpiobus_read_ivar(device_t dev, device_t child, int which, +    uintptr_t *result)  {  	struct acpi_gpiobus_ivar *devi = device_get_ivars(child);  	switch (which) {  	case ACPI_GPIOBUS_IVAR_HANDLE: -		*result = (uintptr_t)devi->dev_handle; -		break; -	case ACPI_GPIOBUS_IVAR_FLAGS: -		*result = (uintptr_t)devi->flags; +		*result = (uintptr_t)devi->handle;  		break;  	default:  		return (gpiobus_read_ivar(dev, child, which, result)); @@ -401,6 +405,28 @@ acpi_gpiobus_read_ivar(device_t dev, device_t child, int which, uintptr_t *resul  	return (0);  } +static device_t +acpi_gpiobus_add_child(device_t dev, u_int order, const char *name, int unit) +{ +	return (gpiobus_add_child_common(dev, order, name, unit, +	    sizeof(struct acpi_gpiobus_ivar))); +} + +static int +acpi_gpiobus_child_location(device_t bus, device_t child, struct sbuf *sb) +{ +	struct acpi_gpiobus_ivar *devi; +	int err; + +	err = gpiobus_child_location(bus, child, sb); +	if (err != 0) +		return (err); + +	devi = device_get_ivars(child); +	sbuf_printf(sb, " handle=%s", acpi_name(devi->handle)); +	return (0); +} +  static device_method_t acpi_gpiobus_methods[] = {  	/* Device interface */  	DEVMETHOD(device_probe,		acpi_gpiobus_probe), @@ -409,6 +435,8 @@ static device_method_t acpi_gpiobus_methods[] = {  	/* Bus interface */  	DEVMETHOD(bus_read_ivar,	acpi_gpiobus_read_ivar), +	DEVMETHOD(bus_add_child,	acpi_gpiobus_add_child), +	DEVMETHOD(bus_child_location,	acpi_gpiobus_child_location),  	DEVMETHOD_END  }; diff --git a/sys/dev/gpio/acpi_gpiobusvar.h b/sys/dev/gpio/acpi_gpiobusvar.h index f8d502eab9d1..288e8bd0f2af 100644 --- a/sys/dev/gpio/acpi_gpiobusvar.h +++ b/sys/dev/gpio/acpi_gpiobusvar.h @@ -33,16 +33,16 @@  #include <contrib/dev/acpica/include/acpi.h>  enum acpi_gpiobus_ivars { -	ACPI_GPIOBUS_IVAR_HANDLE	= 10600, -	ACPI_GPIOBUS_IVAR_FLAGS, +	ACPI_GPIOBUS_IVAR_HANDLE	= 10600  };  #define ACPI_GPIOBUS_ACCESSOR(var, ivar, type)			\  	__BUS_ACCESSOR(acpi_gpiobus, var, ACPI_GPIOBUS, ivar, type)  ACPI_GPIOBUS_ACCESSOR(handle,	HANDLE,		ACPI_HANDLE) -ACPI_GPIOBUS_ACCESSOR(flags,	FLAGS,		uint32_t)  #undef ACPI_GPIOBUS_ACCESSOR +uint32_t acpi_gpiobus_convflags(ACPI_RESOURCE_GPIO *); +  #endif	/* __ACPI_GPIOBUS_H__ */ diff --git a/sys/dev/gpio/bytgpio.c b/sys/dev/gpio/bytgpio.c index f7b2a73ec6cb..5d685c155a03 100644 --- a/sys/dev/gpio/bytgpio.c +++ b/sys/dev/gpio/bytgpio.c @@ -608,7 +608,7 @@ bytgpio_attach(device_t dev)  	    sc->sc_pad_funcs[pin] = val & BYTGPIO_PCONF0_FUNC_MASK;  	} -	sc->sc_busdev = gpiobus_attach_bus(dev); +	sc->sc_busdev = gpiobus_add_bus(dev);  	if (sc->sc_busdev == NULL) {  		BYTGPIO_LOCK_DESTROY(sc);  		bus_release_resource(dev, SYS_RES_MEMORY, @@ -616,6 +616,7 @@ bytgpio_attach(device_t dev)  		return (ENXIO);  	} +	bus_attach_children(dev);  	return (0);  error: diff --git a/sys/dev/gpio/chvgpio.c b/sys/dev/gpio/chvgpio.c index 199ad4d6f373..3273aad9242b 100644 --- a/sys/dev/gpio/chvgpio.c +++ b/sys/dev/gpio/chvgpio.c @@ -441,7 +441,7 @@ chvgpio_attach(device_t dev)  	bus_write_4(sc->sc_mem_res, CHVGPIO_INTERRUPT_MASK, 0);  	bus_write_4(sc->sc_mem_res, CHVGPIO_INTERRUPT_STATUS, 0xffff); -	sc->sc_busdev = gpiobus_attach_bus(dev); +	sc->sc_busdev = gpiobus_add_bus(dev);  	if (sc->sc_busdev == NULL) {  		CHVGPIO_LOCK_DESTROY(sc);  		bus_release_resource(dev, SYS_RES_MEMORY, @@ -451,6 +451,7 @@ chvgpio_attach(device_t dev)  		return (ENXIO);  	} +	bus_attach_children(dev);  	return (0);  } diff --git a/sys/dev/gpio/dwgpio/dwgpio.c b/sys/dev/gpio/dwgpio/dwgpio.c index 5acb99ca591e..3908113d5fd4 100644 --- a/sys/dev/gpio/dwgpio/dwgpio.c +++ b/sys/dev/gpio/dwgpio/dwgpio.c @@ -167,12 +167,13 @@ dwgpio_attach(device_t dev)  		snprintf(sc->gpio_pins[i].gp_name, GPIOMAXNAME,  		    "dwgpio%d.%d", device_get_unit(dev), i);  	} -	sc->busdev = gpiobus_attach_bus(dev); +	sc->busdev = gpiobus_add_bus(dev);  	if (sc->busdev == NULL) {  		mtx_destroy(&sc->sc_mtx);  		return (ENXIO);  	} +	bus_attach_children(dev);  	return (0);  } diff --git a/sys/dev/gpio/gpio_if.m b/sys/dev/gpio/gpio_if.m index 5501b2b5c0e7..0b6988ceba79 100644 --- a/sys/dev/gpio/gpio_if.m +++ b/sys/dev/gpio/gpio_if.m @@ -62,6 +62,22 @@ CODE {  		return (0);  	} + +	static int +	gpio_default_get_pin_list(device_t dev, uint32_t *pin_list) +	{ +		uint32_t maxpin; +		int err; + +		err = GPIO_PIN_MAX(dev, &maxpin); +		if (err != 0) +			return (ENXIO); + +		for (int i = 0; i <= maxpin; i++) +			pin_list[i] = i; + +		return (0); +	}  };  HEADER { @@ -185,3 +201,13 @@ METHOD int pin_config_32 {  	uint32_t num_pins;  	uint32_t *pin_flags;  } DEFAULT gpio_default_nosupport; + +# +# Get the controller's pin numbers. pin_list is expected to be an array with at +# least GPIO_PIN_MAX() elements. Populates pin_list from 0 to GPIO_PIN_MAX() by +# default. +# +METHOD int get_pin_list { +	device_t dev; +	uint32_t *pin_list; +} DEFAULT gpio_default_get_pin_list; diff --git a/sys/dev/gpio/gpioaei.c b/sys/dev/gpio/gpioaei.c index ecae8ccaf2fa..7b97277aaf61 100644 --- a/sys/dev/gpio/gpioaei.c +++ b/sys/dev/gpio/gpioaei.c @@ -45,13 +45,21 @@ enum gpio_aei_type {  	ACPI_AEI_TYPE_EVT  }; -struct gpio_aei_softc { -	ACPI_HANDLE handle; -	enum gpio_aei_type type; -	int pin; +struct gpio_aei_ctx { +	SLIST_ENTRY(gpio_aei_ctx) next;  	struct resource * intr_res; -	int intr_rid;  	void * intr_cookie; +	ACPI_HANDLE handle; +	gpio_pin_t gpio; +	uint32_t pin; +	int intr_rid; +	enum gpio_aei_type type; +}; + +struct gpio_aei_softc { +	SLIST_HEAD(, gpio_aei_ctx) aei_ctx; +	ACPI_HANDLE dev_handle; +	device_t dev;  };  static int @@ -65,69 +73,157 @@ gpio_aei_probe(device_t dev)  static void  gpio_aei_intr(void * arg)  { -	struct gpio_aei_softc * sc = arg; +	struct gpio_aei_ctx * ctx = arg;  	/* Ask ACPI to run the appropriate _EVT, _Exx or _Lxx method. */ -	if (sc->type == ACPI_AEI_TYPE_EVT) -		acpi_SetInteger(sc->handle, NULL, sc->pin); +	if (ctx->type == ACPI_AEI_TYPE_EVT) +		acpi_SetInteger(ctx->handle, NULL, ctx->pin);  	else -		AcpiEvaluateObject(sc->handle, NULL, NULL, NULL); +		AcpiEvaluateObject(ctx->handle, NULL, NULL, NULL); +} + +static ACPI_STATUS +gpio_aei_enumerate(ACPI_RESOURCE * res, void * context) +{ +	ACPI_RESOURCE_GPIO * gpio_res = &res->Data.Gpio; +	struct gpio_aei_softc * sc = context; +	uint32_t flags, maxpin; +	device_t busdev; +	int err; + +	/* +	 * Check that we have a GpioInt object. +	 * Note that according to the spec this +	 * should always be the case. +	 */ +	if (res->Type != ACPI_RESOURCE_TYPE_GPIO) +		return (AE_OK); +	if (gpio_res->ConnectionType != ACPI_RESOURCE_GPIO_TYPE_INT) +		return (AE_OK); + +	flags = acpi_gpiobus_convflags(gpio_res); +	if (acpi_quirks & ACPI_Q_AEI_NOPULL) +		flags &= ~GPIO_PIN_PULLUP; + +	err = GPIO_PIN_MAX(acpi_get_device(sc->dev_handle), &maxpin); +	if (err != 0) +		return (AE_ERROR); + +	busdev = GPIO_GET_BUS(acpi_get_device(sc->dev_handle)); +	for (int i = 0; i < gpio_res->PinTableLength; i++) { +		struct gpio_aei_ctx * ctx; +		uint32_t pin = gpio_res->PinTable[i]; + +		if (__predict_false(pin > maxpin)) { +			device_printf(sc->dev, +			    "Invalid pin 0x%x, max: 0x%x (bad ACPI tables?)\n", +			    pin, maxpin); +			continue; +		} + +		ctx = malloc(sizeof(struct gpio_aei_ctx), M_DEVBUF, M_WAITOK); +		ctx->type = ACPI_AEI_TYPE_UNKNOWN; +		if (pin <= 255) { +			char objname[5];	/* "_EXX" or "_LXX" */ +			sprintf(objname, "_%c%02X", +			    (flags & GPIO_INTR_EDGE_MASK) ? 'E' : 'L', pin); +			if (ACPI_SUCCESS(AcpiGetHandle(sc->dev_handle, objname, +			    &ctx->handle))) +				ctx->type = ACPI_AEI_TYPE_ELX; +		} + +		if (ctx->type == ACPI_AEI_TYPE_UNKNOWN) { +			if (ACPI_SUCCESS(AcpiGetHandle(sc->dev_handle, "_EVT", +			    &ctx->handle))) +				ctx->type = ACPI_AEI_TYPE_EVT; +			else { +				device_printf(sc->dev, +				    "AEI Device type is unknown for pin 0x%x\n", +				    pin); + +				free(ctx, M_DEVBUF); +				continue; +			} +		} + +		err = gpio_pin_get_by_bus_pinnum(busdev, pin, &ctx->gpio); +		if (err != 0) { +			device_printf(sc->dev, "Cannot acquire pin 0x%x\n", +			    pin); + +			free(ctx, M_DEVBUF); +			continue; +		} + +		err = gpio_pin_setflags(ctx->gpio, flags & ~GPIO_INTR_MASK); +		if (err != 0) { +			device_printf(sc->dev, +			    "Cannot set pin flags for pin 0x%x\n", pin); + +			gpio_pin_release(ctx->gpio); +			free(ctx, M_DEVBUF); +			continue; +		} + +		ctx->intr_rid = 0; +		ctx->intr_res = gpio_alloc_intr_resource(sc->dev, +		    &ctx->intr_rid, RF_ACTIVE, ctx->gpio, +		    flags & GPIO_INTR_MASK); +		if (ctx->intr_res == NULL) { +			device_printf(sc->dev, +			    "Cannot allocate an IRQ for pin 0x%x\n", pin); + +			gpio_pin_release(ctx->gpio); +			free(ctx, M_DEVBUF); +			continue; +		} + +		err = bus_setup_intr(sc->dev, ctx->intr_res, INTR_TYPE_MISC | +		    INTR_MPSAFE | INTR_EXCL | INTR_SLEEPABLE, NULL, +		    gpio_aei_intr, ctx, &ctx->intr_cookie); +		if (err != 0) { +			device_printf(sc->dev, +			    "Cannot set up an IRQ for pin 0x%x\n", pin); + +			bus_release_resource(sc->dev, ctx->intr_res); +			gpio_pin_release(ctx->gpio); +			free(ctx, M_DEVBUF); +			continue; +		} + +		ctx->pin = pin; +		SLIST_INSERT_HEAD(&sc->aei_ctx, ctx, next); +	} + +	return (AE_OK);  }  static int  gpio_aei_attach(device_t dev)  {  	struct gpio_aei_softc * sc = device_get_softc(dev); -	gpio_pin_t pin; -	uint32_t flags;  	ACPI_HANDLE handle; -	int err; +	ACPI_STATUS status;  	/* This is us. */  	device_set_desc(dev, "ACPI Event Information Device"); -	/* Store parameters needed by gpio_aei_intr. */  	handle = acpi_gpiobus_get_handle(dev); -	if (gpio_pin_get_by_child_index(dev, 0, &pin) != 0) { -		device_printf(dev, "Unable to get the input pin\n"); +	status = AcpiGetParent(handle, &sc->dev_handle); +	if (ACPI_FAILURE(status)) { +		device_printf(dev, "Cannot get parent of %s\n", +		    acpi_name(handle));  		return (ENXIO);  	} -	sc->type = ACPI_AEI_TYPE_UNKNOWN; -	sc->pin = pin->pin; - -	flags = acpi_gpiobus_get_flags(dev); -	if (pin->pin <= 255) { -		char objname[5];	/* "_EXX" or "_LXX" */ -		sprintf(objname, "_%c%02X", -		    (flags & GPIO_INTR_EDGE_MASK) ? 'E' : 'L', pin->pin); -		if (ACPI_SUCCESS(AcpiGetHandle(handle, objname, &sc->handle))) -			sc->type = ACPI_AEI_TYPE_ELX; -	} -	if (sc->type == ACPI_AEI_TYPE_UNKNOWN) { -		if (ACPI_SUCCESS(AcpiGetHandle(handle, "_EVT", &sc->handle))) -			sc->type = ACPI_AEI_TYPE_EVT; -	} - -	if (sc->type == ACPI_AEI_TYPE_UNKNOWN) { -		device_printf(dev, "ACPI Event Information Device type is unknown"); -		return (ENOTSUP); -	} +	SLIST_INIT(&sc->aei_ctx); +	sc->dev = dev; -	/* Set up the interrupt. */ -	if ((sc->intr_res = gpio_alloc_intr_resource(dev, &sc->intr_rid, -	    RF_ACTIVE, pin, flags & GPIO_INTR_MASK)) == NULL) { -		device_printf(dev, "Cannot allocate an IRQ\n"); -		return (ENOTSUP); -	} -	err = bus_setup_intr(dev, sc->intr_res, INTR_TYPE_MISC | INTR_MPSAFE | -	    INTR_EXCL | INTR_SLEEPABLE, NULL, gpio_aei_intr, sc, -	    &sc->intr_cookie); -	if (err != 0) { -		device_printf(dev, "Cannot set up IRQ\n"); -		bus_release_resource(dev, SYS_RES_IRQ, sc->intr_rid, -		    sc->intr_res); -		return (err); +	status = AcpiWalkResources(sc->dev_handle, "_AEI", +	    gpio_aei_enumerate, sc); +	if (ACPI_FAILURE(status)) { +		device_printf(dev, "Failed to enumerate AEI resources\n"); +		return (ENXIO);  	}  	return (0); @@ -137,9 +233,15 @@ static int  gpio_aei_detach(device_t dev)  {  	struct gpio_aei_softc * sc = device_get_softc(dev); +	struct gpio_aei_ctx * ctx, * tctx; + +	SLIST_FOREACH_SAFE(ctx, &sc->aei_ctx, next, tctx) { +		bus_teardown_intr(dev, ctx->intr_res, ctx->intr_cookie); +		bus_release_resource(dev, ctx->intr_res); +		gpio_pin_release(ctx->gpio); +		free(ctx, M_DEVBUF); +	} -	bus_teardown_intr(dev, sc->intr_res, sc->intr_cookie); -	bus_release_resource(dev, SYS_RES_IRQ, sc->intr_rid, sc->intr_res);  	return (0);  } diff --git a/sys/dev/gpio/gpiobus.c b/sys/dev/gpio/gpiobus.c index ab7f13177969..698b5e5fdd01 100644 --- a/sys/dev/gpio/gpiobus.c +++ b/sys/dev/gpio/gpiobus.c @@ -57,7 +57,6 @@ static int gpiobus_suspend(device_t);  static int gpiobus_resume(device_t);  static void gpiobus_probe_nomatch(device_t, device_t);  static int gpiobus_print_child(device_t, device_t); -static int gpiobus_child_location(device_t, device_t, struct sbuf *);  static device_t gpiobus_add_child(device_t, u_int, const char *, int);  static void gpiobus_hinted_child(device_t, const char *, int); @@ -110,10 +109,9 @@ gpio_alloc_intr_resource(device_t consumer_dev, int *rid, u_int alloc_flags,  	res = bus_alloc_resource(consumer_dev, SYS_RES_IRQ, rid, irq, irq, 1,  	    alloc_flags);  	if (res == NULL) { -		intr_free_intr_map_data((struct intr_map_data *)gpio_data); +		intr_unmap_irq(irq);  		return (NULL);  	} -	rman_set_virtual(res, gpio_data);  	return (res);  }  #else @@ -321,34 +319,12 @@ gpiobus_add_bus(device_t dev)  	busdev = device_add_child(dev, "gpiobus", DEVICE_UNIT_ANY);  	if (busdev == NULL)  		return (NULL); -	if (device_add_child(dev, "gpioc", DEVICE_UNIT_ANY) == NULL) { -		device_delete_child(dev, busdev); -		return (NULL); -	}  #ifdef FDT  	ofw_gpiobus_register_provider(dev);  #endif  	return (busdev);  } -/* - * Attach a gpiobus child. - * Note that the controller is expected - * to be fully initialized at this point. - */ -device_t -gpiobus_attach_bus(device_t dev) -{ -	device_t busdev; - -	busdev = gpiobus_add_bus(dev); -	if (busdev == NULL) -		return (NULL); - -	bus_attach_children(dev); -	return (busdev); -} -  int  gpiobus_detach_bus(device_t dev)  { @@ -392,6 +368,37 @@ gpiobus_init_softc(device_t dev)  }  int +gpiobus_add_gpioc(device_t dev) +{ +	struct gpiobus_ivar *devi; +	struct gpiobus_softc *sc; +	device_t gpioc; +	int err; + +	gpioc = BUS_ADD_CHILD(dev, 0, "gpioc", DEVICE_UNIT_ANY); +	if (gpioc == NULL) +		return (ENXIO); + +	sc = device_get_softc(dev); +	devi = device_get_ivars(gpioc); + +	devi->npins = sc->sc_npins; +	err = gpiobus_alloc_ivars(devi); +	if (err != 0) { +		device_delete_child(dev, gpioc); +		return (err); +	} + +	err = GPIO_GET_PIN_LIST(sc->sc_dev, devi->pins); +	if (err != 0) { +		device_delete_child(dev, gpioc); +		gpiobus_free_ivars(devi); +	} + +	return (err); +} + +int  gpiobus_alloc_ivars(struct gpiobus_ivar *devi)  { @@ -582,6 +589,10 @@ gpiobus_attach(device_t dev)  	if (err != 0)  		return (err); +	err = gpiobus_add_gpioc(dev); +	if (err != 0) +		return (err); +  	/*  	 * Get parent's pins and mark them as unmapped  	 */ @@ -681,7 +692,7 @@ gpiobus_print_child(device_t dev, device_t child)  	return (retval);  } -static int +int  gpiobus_child_location(device_t bus, device_t child, struct sbuf *sb)  {  	struct gpiobus_ivar *devi; @@ -693,16 +704,19 @@ gpiobus_child_location(device_t bus, device_t child, struct sbuf *sb)  	return (0);  } -static device_t -gpiobus_add_child(device_t dev, u_int order, const char *name, int unit) +device_t +gpiobus_add_child_common(device_t dev, u_int order, const char *name, int unit, +    size_t ivars_size)  {  	device_t child;  	struct gpiobus_ivar *devi; +	KASSERT(ivars_size >= sizeof(struct gpiobus_ivar), +	    ("child ivars must include gpiobus_ivar as their first member"));  	child = device_add_child_ordered(dev, order, name, unit);  	if (child == NULL)   		return (child); -	devi = malloc(sizeof(struct gpiobus_ivar), M_DEVBUF, M_NOWAIT | M_ZERO); +	devi = malloc(ivars_size, M_DEVBUF, M_NOWAIT | M_ZERO);  	if (devi == NULL) {  		device_delete_child(dev, child);  		return (NULL); @@ -713,6 +727,13 @@ gpiobus_add_child(device_t dev, u_int order, const char *name, int unit)  	return (child);  } +static device_t +gpiobus_add_child(device_t dev, u_int order, const char *name, int unit) +{ +	return (gpiobus_add_child_common(dev, order, name, unit, +	    sizeof(struct gpiobus_ivar))); +} +  static void  gpiobus_child_deleted(device_t dev, device_t child)  { @@ -866,6 +887,25 @@ gpiobus_alloc_resource(device_t bus, device_t child, int type, int *rid,  	    end, count, flags));  } +static int +gpiobus_release_resource(device_t dev, device_t child, struct resource *r) +{ +	int err; +#ifdef INTRNG +	u_int irq; + +	irq = rman_get_start(r); +	MPASS(irq == rman_get_end(r)); +#endif +	err = bus_generic_rman_release_resource(dev, child, r); +	if (err != 0) +		return (err); +#ifdef INTRNG +	intr_unmap_irq(irq); +#endif +	return (0); +} +  static struct resource_list *  gpiobus_get_resource_list(device_t bus __unused, device_t child)  { @@ -952,7 +992,7 @@ gpiobus_pin_getflags(device_t dev, device_t child, uint32_t pin,  	if (pin >= devi->npins)  		return (EINVAL); -	return GPIO_PIN_GETFLAGS(sc->sc_dev, devi->pins[pin], flags); +	return (GPIO_PIN_GETFLAGS(sc->sc_dev, devi->pins[pin], flags));  }  static int @@ -965,7 +1005,7 @@ gpiobus_pin_getcaps(device_t dev, device_t child, uint32_t pin,  	if (pin >= devi->npins)  		return (EINVAL); -	return GPIO_PIN_GETCAPS(sc->sc_dev, devi->pins[pin], caps); +	return (GPIO_PIN_GETCAPS(sc->sc_dev, devi->pins[pin], caps));  }  static int @@ -978,7 +1018,7 @@ gpiobus_pin_set(device_t dev, device_t child, uint32_t pin,  	if (pin >= devi->npins)  		return (EINVAL); -	return GPIO_PIN_SET(sc->sc_dev, devi->pins[pin], value); +	return (GPIO_PIN_SET(sc->sc_dev, devi->pins[pin], value));  }  static int @@ -991,7 +1031,7 @@ gpiobus_pin_get(device_t dev, device_t child, uint32_t pin,  	if (pin >= devi->npins)  		return (EINVAL); -	return GPIO_PIN_GET(sc->sc_dev, devi->pins[pin], value); +	return (GPIO_PIN_GET(sc->sc_dev, devi->pins[pin], value));  }  static int @@ -1003,7 +1043,57 @@ gpiobus_pin_toggle(device_t dev, device_t child, uint32_t pin)  	if (pin >= devi->npins)  		return (EINVAL); -	return GPIO_PIN_TOGGLE(sc->sc_dev, devi->pins[pin]); +	return (GPIO_PIN_TOGGLE(sc->sc_dev, devi->pins[pin])); +} + +/* + * Verify that a child has all the pins they are requesting + * to access in their ivars. + */ +static bool +gpiobus_pin_verify_32(struct gpiobus_ivar *devi, uint32_t first_pin, +    uint32_t num_pins) +{ +	if (first_pin + num_pins > devi->npins) +		return (false); + +	/* Make sure the pins are consecutive. */ +	for (uint32_t pin = first_pin; pin < first_pin + num_pins - 1; pin++) { +		if (devi->pins[pin] + 1 != devi->pins[pin + 1]) +			return (false); +	} + +	return (true); +} + +static int +gpiobus_pin_access_32(device_t dev, device_t child, uint32_t first_pin, +    uint32_t clear_pins, uint32_t change_pins, uint32_t *orig_pins) +{ +	struct gpiobus_softc *sc = GPIOBUS_SOFTC(dev); +	struct gpiobus_ivar *devi = GPIOBUS_IVAR(child); + +	if (!gpiobus_pin_verify_32(devi, first_pin, 32)) +		return (EINVAL); + +	return (GPIO_PIN_ACCESS_32(sc->sc_dev, devi->pins[first_pin], +	    clear_pins, change_pins, orig_pins)); +} + +static int +gpiobus_pin_config_32(device_t dev, device_t child, uint32_t first_pin, +    uint32_t num_pins, uint32_t *pin_flags) +{ +	struct gpiobus_softc *sc = GPIOBUS_SOFTC(dev); +	struct gpiobus_ivar *devi = GPIOBUS_IVAR(child); + +	if (num_pins > 32) +		return (EINVAL); +	if (!gpiobus_pin_verify_32(devi, first_pin, num_pins)) +		return (EINVAL); + +	return (GPIO_PIN_CONFIG_32(sc->sc_dev, +	    devi->pins[first_pin], num_pins, pin_flags));  }  static int @@ -1060,7 +1150,7 @@ static device_method_t gpiobus_methods[] = {  	DEVMETHOD(bus_get_resource,	bus_generic_rl_get_resource),  	DEVMETHOD(bus_set_resource,	bus_generic_rl_set_resource),  	DEVMETHOD(bus_alloc_resource,	gpiobus_alloc_resource), -	DEVMETHOD(bus_release_resource,	bus_generic_rman_release_resource), +	DEVMETHOD(bus_release_resource,	gpiobus_release_resource),  	DEVMETHOD(bus_activate_resource,	bus_generic_rman_activate_resource),  	DEVMETHOD(bus_deactivate_resource,	bus_generic_rman_deactivate_resource),  	DEVMETHOD(bus_get_resource_list,	gpiobus_get_resource_list), @@ -1084,6 +1174,8 @@ static device_method_t gpiobus_methods[] = {  	DEVMETHOD(gpiobus_pin_get,	gpiobus_pin_get),  	DEVMETHOD(gpiobus_pin_set,	gpiobus_pin_set),  	DEVMETHOD(gpiobus_pin_toggle,	gpiobus_pin_toggle), +	DEVMETHOD(gpiobus_pin_access_32,gpiobus_pin_access_32), +	DEVMETHOD(gpiobus_pin_config_32,gpiobus_pin_config_32),  	DEVMETHOD(gpiobus_pin_getname,	gpiobus_pin_getname),  	DEVMETHOD(gpiobus_pin_setname,	gpiobus_pin_setname), diff --git a/sys/dev/gpio/gpiobus_if.m b/sys/dev/gpio/gpiobus_if.m index 8bf29839ef4e..890738c4e809 100644 --- a/sys/dev/gpio/gpiobus_if.m +++ b/sys/dev/gpio/gpiobus_if.m @@ -107,6 +107,36 @@ METHOD int pin_setflags {  };  # +# Simultaneously read and/or change up to 32 adjacent pins. +# If the device cannot change the pins simultaneously, returns EOPNOTSUPP. +# +# More details about using this interface can be found in sys/gpio.h +# +METHOD int pin_access_32 { +	device_t dev; +	device_t child; +	uint32_t first_pin; +	uint32_t clear_pins; +	uint32_t change_pins; +	uint32_t *orig_pins; +}; + +# +# Simultaneously configure up to 32 adjacent pins. +# This is intended to change the configuration of all the pins simultaneously, +# but unlike pin_access_32, this will not fail if the hardware can't do so. +# +# More details about using this interface can be found in sys/gpio.h +# +METHOD int pin_config_32 { +	device_t dev; +	device_t child; +	uint32_t first_pin; +	uint32_t num_pins; +	uint32_t *pin_flags; +}; + +#  # Get the pin name  #  METHOD int pin_getname { diff --git a/sys/dev/gpio/gpiobus_internal.h b/sys/dev/gpio/gpiobus_internal.h index de3f57663132..58f862343403 100644 --- a/sys/dev/gpio/gpiobus_internal.h +++ b/sys/dev/gpio/gpiobus_internal.h @@ -42,6 +42,9 @@ void gpiobus_free_ivars(struct gpiobus_ivar *);  int gpiobus_read_ivar(device_t, device_t, int, uintptr_t *);  int gpiobus_acquire_pin(device_t, uint32_t);  void gpiobus_release_pin(device_t, uint32_t); +int gpiobus_child_location(device_t, device_t, struct sbuf *); +device_t gpiobus_add_child_common(device_t, u_int, const char *, int, size_t); +int gpiobus_add_gpioc(device_t);  extern driver_t gpiobus_driver;  #endif diff --git a/sys/dev/gpio/gpiobusvar.h b/sys/dev/gpio/gpiobusvar.h index 7f504236a774..0528efe45525 100644 --- a/sys/dev/gpio/gpiobusvar.h +++ b/sys/dev/gpio/gpiobusvar.h @@ -171,7 +171,6 @@ struct resource *gpio_alloc_intr_resource(device_t consumer_dev, int *rid,  int gpio_check_flags(uint32_t, uint32_t);  device_t gpiobus_add_bus(device_t); -device_t gpiobus_attach_bus(device_t);  int gpiobus_detach_bus(device_t);  #endif	/* __GPIOBUS_H__ */ diff --git a/sys/dev/gpio/gpioc.c b/sys/dev/gpio/gpioc.c index 87fed38ebe3e..517f7752daad 100644 --- a/sys/dev/gpio/gpioc.c +++ b/sys/dev/gpio/gpioc.c @@ -45,7 +45,6 @@  #include <dev/gpio/gpiobusvar.h> -#include "gpio_if.h"  #include "gpiobus_if.h"  #undef GPIOC_DEBUG @@ -59,7 +58,7 @@  struct gpioc_softc {  	device_t		sc_dev;		/* gpiocX dev */ -	device_t		sc_pdev;	/* gpioX dev */ +	device_t		sc_pdev;	/* gpiobusX dev */  	struct cdev		*sc_ctl_dev;	/* controller device */  	int			sc_unit;  	int			sc_npins; @@ -69,6 +68,7 @@ struct gpioc_softc {  struct gpioc_pin_intr {  	struct gpioc_softc				*sc;  	gpio_pin_t					pin; +	uint32_t					intr_mode;  	bool						config_locked;  	int						intr_rid;  	struct resource					*intr_res; @@ -112,8 +112,10 @@ struct gpioc_pin_event {  static MALLOC_DEFINE(M_GPIOC, "gpioc", "gpioc device data"); -static int	gpioc_allocate_pin_intr(struct gpioc_pin_intr*, uint32_t); -static int	gpioc_release_pin_intr(struct gpioc_pin_intr*); +static int	gpioc_allocate_pin_intr(struct gpioc_softc*, +		    struct gpioc_pin_intr*, uint32_t, uint32_t); +static int	gpioc_release_pin_intr(struct gpioc_softc*, +		    struct gpioc_pin_intr*);  static int	gpioc_attach_priv_pin(struct gpioc_cdevpriv*,  		    struct gpioc_pin_intr*);  static int	gpioc_detach_priv_pin(struct gpioc_cdevpriv*, @@ -156,7 +158,8 @@ static const struct filterops gpioc_read_filterops = {  	.f_attach =	NULL,  	.f_detach =	gpioc_kqdetach,  	.f_event =	gpioc_kqread, -	.f_touch =	NULL +	.f_touch =	NULL, +	.f_copy =	knote_triv_copy,  };  static struct gpioc_pin_event * @@ -191,27 +194,36 @@ number_of_events(struct gpioc_cdevpriv *priv)  }  static int -gpioc_allocate_pin_intr(struct gpioc_pin_intr *intr_conf, uint32_t flags) +gpioc_allocate_pin_intr(struct gpioc_softc *sc, +    struct gpioc_pin_intr *intr_conf, uint32_t pin, uint32_t flags)  {  	int err;  	intr_conf->config_locked = true;  	mtx_unlock(&intr_conf->mtx); -	intr_conf->intr_res = gpio_alloc_intr_resource(intr_conf->pin->dev, +	MPASS(intr_conf->pin == NULL); +	err = gpio_pin_get_by_bus_pinnum(sc->sc_pdev, pin, &intr_conf->pin); +	if (err != 0) +		goto error_exit; + +	intr_conf->intr_res = gpio_alloc_intr_resource(sc->sc_dev,  	    &intr_conf->intr_rid, RF_ACTIVE, intr_conf->pin, flags);  	if (intr_conf->intr_res == NULL) {  		err = ENXIO; -		goto error_exit; +		goto error_pin;  	} -	err = bus_setup_intr(intr_conf->pin->dev, intr_conf->intr_res, +	err = bus_setup_intr(sc->sc_dev, intr_conf->intr_res,  	    INTR_TYPE_MISC | INTR_MPSAFE, NULL, gpioc_interrupt_handler,  	    intr_conf, &intr_conf->intr_cookie); -	if (err != 0) -		goto error_exit; +	if (err != 0) { +		bus_release_resource(sc->sc_dev, intr_conf->intr_res); +		intr_conf->intr_res = NULL; +		goto error_pin; +	} -	intr_conf->pin->flags = flags; +	intr_conf->intr_mode = flags;  error_exit:  	mtx_lock(&intr_conf->mtx); @@ -219,10 +231,15 @@ error_exit:  	wakeup(&intr_conf->config_locked);  	return (err); + +error_pin: +	gpio_pin_release(intr_conf->pin); +	intr_conf->pin = NULL; +	goto error_exit;  }  static int -gpioc_release_pin_intr(struct gpioc_pin_intr *intr_conf) +gpioc_release_pin_intr(struct gpioc_softc *sc, struct gpioc_pin_intr *intr_conf)  {  	int err; @@ -230,8 +247,8 @@ gpioc_release_pin_intr(struct gpioc_pin_intr *intr_conf)  	mtx_unlock(&intr_conf->mtx);  	if (intr_conf->intr_cookie != NULL) { -		err = bus_teardown_intr(intr_conf->pin->dev, -		    intr_conf->intr_res, intr_conf->intr_cookie); +		err = bus_teardown_intr(sc->sc_dev, intr_conf->intr_res, +		    intr_conf->intr_cookie);  		if (err != 0)  			goto error_exit;  		else @@ -239,7 +256,7 @@ gpioc_release_pin_intr(struct gpioc_pin_intr *intr_conf)  	}  	if (intr_conf->intr_res != NULL) { -		err = bus_release_resource(intr_conf->pin->dev, SYS_RES_IRQ, +		err = bus_release_resource(sc->sc_dev, SYS_RES_IRQ,  		    intr_conf->intr_rid, intr_conf->intr_res);  		if (err != 0)  			goto error_exit; @@ -249,7 +266,10 @@ gpioc_release_pin_intr(struct gpioc_pin_intr *intr_conf)  		}  	} -	intr_conf->pin->flags = 0; +	gpio_pin_release(intr_conf->pin); +	intr_conf->pin = NULL; + +	intr_conf->intr_mode = 0;  	err = 0;  error_exit: @@ -386,7 +406,7 @@ gpioc_get_intr_config(struct gpioc_softc *sc, struct gpioc_cdevpriv *priv,  	struct gpioc_privs	*priv_link;  	uint32_t		flags; -	flags = intr_conf->pin->flags; +	flags = intr_conf->intr_mode;  	if (flags == 0)  		return (0); @@ -411,7 +431,7 @@ gpioc_set_intr_config(struct gpioc_softc *sc, struct gpioc_cdevpriv *priv,  	int res;  	res = 0; -	if (intr_conf->pin->flags == 0 && flags == 0) { +	if (intr_conf->intr_mode == 0 && flags == 0) {  		/* No interrupt configured and none requested: Do nothing. */  		return (0);  	} @@ -419,17 +439,17 @@ gpioc_set_intr_config(struct gpioc_softc *sc, struct gpioc_cdevpriv *priv,  	while (intr_conf->config_locked == true)  		mtx_sleep(&intr_conf->config_locked, &intr_conf->mtx, 0,  		    "gpicfg", 0); -	if (intr_conf->pin->flags == 0 && flags != 0) { +	if (intr_conf->intr_mode == 0 && flags != 0) {  		/*  		 * No interrupt is configured, but one is requested: Allocate  		 * and setup interrupt on the according pin.  		 */ -		res = gpioc_allocate_pin_intr(intr_conf, flags); +		res = gpioc_allocate_pin_intr(sc, intr_conf, pin, flags);  		if (res == 0)  			res = gpioc_attach_priv_pin(priv, intr_conf);  		if (res == EEXIST)  			res = 0; -	} else if (intr_conf->pin->flags == flags) { +	} else if (intr_conf->intr_mode == flags) {  		/*  		 * Same interrupt requested as already configured: Attach the  		 * cdevpriv to the corresponding pin. @@ -437,14 +457,14 @@ gpioc_set_intr_config(struct gpioc_softc *sc, struct gpioc_cdevpriv *priv,  		res = gpioc_attach_priv_pin(priv, intr_conf);  		if (res == EEXIST)  			res = 0; -	} else if (intr_conf->pin->flags != 0 && flags == 0) { +	} else if (intr_conf->intr_mode != 0 && flags == 0) {  		/*  		 * Interrupt configured, but none requested: Teardown and  		 * release the pin when no other cdevpriv is attached. Otherwise  		 * just detach pin and cdevpriv from each other.  		 */  		if (gpioc_intr_reconfig_allowed(priv, intr_conf)) { -			res = gpioc_release_pin_intr(intr_conf); +			res = gpioc_release_pin_intr(sc, intr_conf);  		}  		if (res == 0)  			res = gpioc_detach_priv_pin(priv, intr_conf); @@ -456,9 +476,10 @@ gpioc_set_intr_config(struct gpioc_softc *sc, struct gpioc_cdevpriv *priv,  		if (!gpioc_intr_reconfig_allowed(priv, intr_conf))  			res = EBUSY;  		else { -			res = gpioc_release_pin_intr(intr_conf); +			res = gpioc_release_pin_intr(sc, intr_conf);  			if (res == 0) -				res = gpioc_allocate_pin_intr(intr_conf, flags); +				res = gpioc_allocate_pin_intr(sc, intr_conf, +				    pin, flags);  			if (res == 0)  				res = gpioc_attach_priv_pin(priv, intr_conf);  			if (res == EEXIST) @@ -475,18 +496,16 @@ gpioc_interrupt_handler(void *arg)  {  	struct gpioc_pin_intr *intr_conf;  	struct gpioc_privs *privs; -	struct gpioc_softc *sc;  	sbintime_t evtime; -	uint32_t pin_state; +	bool pin_state;  	intr_conf = arg; -	sc = intr_conf->sc;  	/* Capture time and pin state first. */  	evtime = sbinuptime(); -	if (intr_conf->pin->flags & GPIO_INTR_EDGE_BOTH) -		GPIO_PIN_GET(sc->sc_pdev, intr_conf->pin->pin, &pin_state); -	else if (intr_conf->pin->flags & GPIO_INTR_EDGE_RISING) +	if (intr_conf->intr_mode & GPIO_INTR_EDGE_BOTH) +		gpio_pin_is_active(intr_conf->pin, &pin_state); +	else if (intr_conf->intr_mode & GPIO_INTR_EDGE_RISING)  		pin_state = true;  	else  		pin_state = false; @@ -575,18 +594,11 @@ gpioc_attach(device_t dev)  	sc->sc_pdev = device_get_parent(dev);  	sc->sc_unit = device_get_unit(dev); -	err = GPIO_PIN_MAX(sc->sc_pdev, &sc->sc_npins); -	sc->sc_npins++; /* Number of pins is one more than max pin number. */ -	if (err != 0) -		return (err); +	sc->sc_npins = gpiobus_get_npins(dev);  	sc->sc_pin_intr = malloc(sizeof(struct gpioc_pin_intr) * sc->sc_npins,  	    M_GPIOC, M_WAITOK | M_ZERO);  	for (int i = 0; i < sc->sc_npins; i++) { -		sc->sc_pin_intr[i].pin = malloc(sizeof(struct gpiobus_pin), -		    M_GPIOC, M_WAITOK | M_ZERO);  		sc->sc_pin_intr[i].sc = sc; -		sc->sc_pin_intr[i].pin->pin = i; -		sc->sc_pin_intr[i].pin->dev = sc->sc_pdev;  		mtx_init(&sc->sc_pin_intr[i].mtx, "gpioc pin", NULL, MTX_DEF);  		SLIST_INIT(&sc->sc_pin_intr[i].privs);  	} @@ -610,20 +622,16 @@ static int  gpioc_detach(device_t dev)  {  	struct gpioc_softc *sc = device_get_softc(dev); -	int err;  	if (sc->sc_ctl_dev)  		destroy_dev(sc->sc_ctl_dev);  	for (int i = 0; i < sc->sc_npins; i++) {  		mtx_destroy(&sc->sc_pin_intr[i].mtx); -		free(sc->sc_pin_intr[i].pin, M_GPIOC); +		MPASS(sc->sc_pin_intr[i].pin == NULL);  	}  	free(sc->sc_pin_intr, M_GPIOC); -	if ((err = bus_generic_detach(dev)) != 0) -		return (err); -  	return (0);  } @@ -655,7 +663,7 @@ gpioc_cdevpriv_dtor(void *data)  		KASSERT(consistency == 1,  		    ("inconsistent links between pin config and cdevpriv"));  		if (gpioc_intr_reconfig_allowed(priv, pin_link->pin)) { -			gpioc_release_pin_intr(pin_link->pin); +			gpioc_release_pin_intr(priv->sc, pin_link->pin);  		}  		mtx_unlock(&pin_link->pin->mtx);  		SLIST_REMOVE(&priv->pins, pin_link, gpioc_pins, next); @@ -697,7 +705,7 @@ gpioc_open(struct cdev *dev, int oflags, int devtype, struct thread *td)  	 * npins isn't a horrible fifo size for that either.  	 */  	priv->numevents = priv->sc->sc_npins * 2; -	priv->events = malloc(priv->numevents * sizeof(struct gpio_event_detail), +	priv->events = malloc(priv->numevents * sizeof(struct gpioc_pin_event),  	    M_GPIOC, M_WAITOK | M_ZERO);  	priv->evidx_head = priv->evidx_tail = 0; @@ -778,7 +786,6 @@ static int  gpioc_ioctl(struct cdev *cdev, u_long cmd, caddr_t arg, int fflag,       struct thread *td)  { -	device_t bus;  	int max_pin, res;  	struct gpioc_softc *sc = cdev->si_drv1;  	struct gpioc_cdevpriv *priv; @@ -787,32 +794,35 @@ gpioc_ioctl(struct cdev *cdev, u_long cmd, caddr_t arg, int fflag,  	struct gpio_access_32 *a32;  	struct gpio_config_32 *c32;  	struct gpio_event_config *evcfg; +	struct gpioc_pin_event *tmp;  	uint32_t caps, intrflags; -	bus = GPIO_GET_BUS(sc->sc_pdev); -	if (bus == NULL) -		return (EINVAL);  	switch (cmd) {  	case GPIOMAXPIN: -		max_pin = -1; -		res = GPIO_PIN_MAX(sc->sc_pdev, &max_pin); +		res = 0; +		max_pin = sc->sc_npins - 1;  		bcopy(&max_pin, arg, sizeof(max_pin));  		break;  	case GPIOGETCONFIG:  		bcopy(arg, &pin, sizeof(pin));  		dprintf("get config pin %d\n", pin.gp_pin); -		res = GPIO_PIN_GETFLAGS(sc->sc_pdev, pin.gp_pin, +		res = GPIOBUS_PIN_GETFLAGS(sc->sc_pdev, sc->sc_dev, pin.gp_pin,  		    &pin.gp_flags);  		/* Fail early */ -		if (res) +		if (res != 0)  			break;  		res = devfs_get_cdevpriv((void **)&priv); -		if (res) +		if (res != 0)  			break;  		pin.gp_flags |= gpioc_get_intr_config(sc, priv,  		    pin.gp_pin); -		GPIO_PIN_GETCAPS(sc->sc_pdev, pin.gp_pin, &pin.gp_caps); -		GPIOBUS_PIN_GETNAME(bus, pin.gp_pin, pin.gp_name); +		res = GPIOBUS_PIN_GETCAPS(sc->sc_pdev, sc->sc_dev, pin.gp_pin, +		    &pin.gp_caps); +		if (res != 0) +			break; +		res = GPIOBUS_PIN_GETNAME(sc->sc_pdev, pin.gp_pin, pin.gp_name); +		if (res != 0) +			break;  		bcopy(&pin, arg, sizeof(pin));  		break;  	case GPIOSETCONFIG: @@ -821,7 +831,8 @@ gpioc_ioctl(struct cdev *cdev, u_long cmd, caddr_t arg, int fflag,  		res = devfs_get_cdevpriv((void **)&priv);  		if (res != 0)  			break; -		res = GPIO_PIN_GETCAPS(sc->sc_pdev, pin.gp_pin, &caps); +		res = GPIOBUS_PIN_GETCAPS(sc->sc_pdev, sc->sc_dev, +		    pin.gp_pin, &caps);  		if (res != 0)  			break;  		res = gpio_check_flags(caps, pin.gp_flags); @@ -847,8 +858,8 @@ gpioc_ioctl(struct cdev *cdev, u_long cmd, caddr_t arg, int fflag,  		}  		if (res != 0)  			break; -		res = GPIO_PIN_SETFLAGS(sc->sc_pdev, pin.gp_pin, -		    (pin.gp_flags & ~GPIO_INTR_MASK)); +		res = GPIOBUS_PIN_SETFLAGS(sc->sc_pdev, sc->sc_dev, pin.gp_pin, +		    pin.gp_flags & ~GPIO_INTR_MASK);  		if (res != 0)  			break;  		res = gpioc_set_intr_config(sc, priv, pin.gp_pin, @@ -856,67 +867,78 @@ gpioc_ioctl(struct cdev *cdev, u_long cmd, caddr_t arg, int fflag,  		break;  	case GPIOGET:  		bcopy(arg, &req, sizeof(req)); -		res = GPIO_PIN_GET(sc->sc_pdev, req.gp_pin, +		res = GPIOBUS_PIN_GET(sc->sc_pdev, sc->sc_dev, req.gp_pin,  		    &req.gp_value); -		dprintf("read pin %d -> %d\n",  +		if (res != 0) +			break; +		dprintf("read pin %d -> %d\n",  		    req.gp_pin, req.gp_value);  		bcopy(&req, arg, sizeof(req));  		break;  	case GPIOSET:  		bcopy(arg, &req, sizeof(req)); -		res = GPIO_PIN_SET(sc->sc_pdev, req.gp_pin,  +		res = GPIOBUS_PIN_SET(sc->sc_pdev, sc->sc_dev, req.gp_pin,  		    req.gp_value); -		dprintf("write pin %d -> %d\n",  +		dprintf("write pin %d -> %d\n",  		    req.gp_pin, req.gp_value);  		break;  	case GPIOTOGGLE:  		bcopy(arg, &req, sizeof(req)); -		dprintf("toggle pin %d\n",  +		dprintf("toggle pin %d\n",  		    req.gp_pin); -		res = GPIO_PIN_TOGGLE(sc->sc_pdev, req.gp_pin); +		res = GPIOBUS_PIN_TOGGLE(sc->sc_pdev, sc->sc_dev, req.gp_pin);  		break;  	case GPIOSETNAME:  		bcopy(arg, &pin, sizeof(pin));  		dprintf("set name on pin %d\n", pin.gp_pin); -		res = GPIOBUS_PIN_SETNAME(bus, pin.gp_pin, +		res = GPIOBUS_PIN_SETNAME(sc->sc_pdev, pin.gp_pin,  		    pin.gp_name);  		break;  	case GPIOACCESS32:  		a32 = (struct gpio_access_32 *)arg; -		res = GPIO_PIN_ACCESS_32(sc->sc_pdev, a32->first_pin, -		    a32->clear_pins, a32->change_pins, &a32->orig_pins); +		res = GPIOBUS_PIN_ACCESS_32(sc->sc_pdev, sc->sc_dev, +		    a32->first_pin, a32->clear_pins, a32->change_pins, +		    &a32->orig_pins);  		break;  	case GPIOCONFIG32:  		c32 = (struct gpio_config_32 *)arg; -		res = GPIO_PIN_CONFIG_32(sc->sc_pdev, c32->first_pin, -		    c32->num_pins, c32->pin_flags); +		res = GPIOBUS_PIN_CONFIG_32(sc->sc_pdev, sc->sc_dev, +		    c32->first_pin, c32->num_pins, c32->pin_flags);  		break;  	case GPIOCONFIGEVENTS:  		evcfg = (struct gpio_event_config *)arg;  		res = devfs_get_cdevpriv((void **)&priv);  		if (res != 0)  			break; -		/* If any pins have been configured, changes aren't allowed. */ -		if (!SLIST_EMPTY(&priv->pins)) { -			res = EINVAL; -			break; -		}  		if (evcfg->gp_report_type != GPIO_EVENT_REPORT_DETAIL &&  		    evcfg->gp_report_type != GPIO_EVENT_REPORT_SUMMARY) {  			res = EINVAL;  			break;  		} -		priv->report_option = evcfg->gp_report_type;  		/* Reallocate the events buffer if the user wants it bigger. */ -		if (priv->report_option == GPIO_EVENT_REPORT_DETAIL && +		tmp = NULL; +		if (evcfg->gp_report_type == GPIO_EVENT_REPORT_DETAIL &&  		    priv->numevents < evcfg->gp_fifo_size) { +			tmp = malloc(evcfg->gp_fifo_size * +			    sizeof(struct gpioc_pin_event), M_GPIOC, +			    M_WAITOK | M_ZERO); +		} +		mtx_lock(&priv->mtx); +		/* If any pins have been configured, changes aren't allowed. */ +		if (!SLIST_EMPTY(&priv->pins)) { +			mtx_unlock(&priv->mtx); +			free(tmp, M_GPIOC); +			res = EINVAL; +			break; +		} +		if (tmp != NULL) {  			free(priv->events, M_GPIOC); +			priv->events = tmp;  			priv->numevents = evcfg->gp_fifo_size; -			priv->events = malloc(priv->numevents * -			    sizeof(struct gpio_event_detail), M_GPIOC, -			    M_WAITOK | M_ZERO);  			priv->evidx_head = priv->evidx_tail = 0;  		} +		priv->report_option = evcfg->gp_report_type; +		mtx_unlock(&priv->mtx);  		break;  	case FIONBIO:  		/* @@ -1050,9 +1072,6 @@ static device_method_t gpioc_methods[] = {  	DEVMETHOD(device_probe,		gpioc_probe),  	DEVMETHOD(device_attach,	gpioc_attach),  	DEVMETHOD(device_detach,	gpioc_detach), -	DEVMETHOD(device_shutdown,	bus_generic_shutdown), -	DEVMETHOD(device_suspend,	bus_generic_suspend), -	DEVMETHOD(device_resume,	bus_generic_resume),  	DEVMETHOD_END  }; @@ -1063,5 +1082,5 @@ driver_t gpioc_driver = {  	sizeof(struct gpioc_softc)  }; -DRIVER_MODULE(gpioc, gpio, gpioc_driver, 0, 0); +DRIVER_MODULE(gpioc, gpiobus, gpioc_driver, 0, 0);  MODULE_VERSION(gpioc, 1); diff --git a/sys/dev/gpio/gpioled.c b/sys/dev/gpio/gpioled.c index ba53cb733971..a36c2faef379 100644 --- a/sys/dev/gpio/gpioled.c +++ b/sys/dev/gpio/gpioled.c @@ -55,13 +55,13 @@      device_get_nameunit((_sc)->sc_dev), "gpioled", MTX_DEF)  #define	GPIOLED_LOCK_DESTROY(_sc)	mtx_destroy(&(_sc)->sc_mtx) -struct gpioled_softc  +struct gpioled_softc  {  	device_t	sc_dev;  	device_t	sc_busdev;  	struct mtx	sc_mtx;  	struct cdev	*sc_leddev; -	int		sc_invert; +	int	sc_softinvert;  };  static void gpioled_control(void *, int); @@ -69,20 +69,17 @@ static int gpioled_probe(device_t);  static int gpioled_attach(device_t);  static int gpioled_detach(device_t); -static void  +static void  gpioled_control(void *priv, int onoff)  {  	struct gpioled_softc *sc;  	sc = (struct gpioled_softc *)priv; +	if (sc->sc_softinvert) +		onoff = !onoff;  	GPIOLED_LOCK(sc); -	if (GPIOBUS_PIN_SETFLAGS(sc->sc_busdev, sc->sc_dev, GPIOLED_PIN, -	    GPIO_PIN_OUTPUT) == 0) { -		if (sc->sc_invert) -			onoff = !onoff; -		GPIOBUS_PIN_SET(sc->sc_busdev, sc->sc_dev, GPIOLED_PIN, -		    onoff ? GPIO_PIN_HIGH : GPIO_PIN_LOW); -	} +	GPIOBUS_PIN_SET(sc->sc_busdev, sc->sc_dev, GPIOLED_PIN, +	    onoff ? GPIO_PIN_HIGH : GPIO_PIN_LOW);  	GPIOLED_UNLOCK(sc);  } @@ -95,26 +92,101 @@ gpioled_probe(device_t dev)  }  static int +gpioled_inv(device_t dev, uint32_t *pin_flags) +{ +	struct gpioled_softc *sc; +	int invert; +	uint32_t pin_caps; + +	sc = device_get_softc(dev); + +	if (resource_int_value(device_get_name(dev), +	    device_get_unit(dev), "invert", &invert)) +		invert = 0; + +	if (GPIOBUS_PIN_GETCAPS(sc->sc_busdev, sc->sc_dev, GPIOLED_PIN, +	    &pin_caps) != 0) { +		if (bootverbose) +			device_printf(sc->sc_dev, "unable to get pin caps\n"); +		return (-1); +	} +	if (pin_caps & GPIO_PIN_INVOUT) +		*pin_flags &= ~GPIO_PIN_INVOUT; +	sc->sc_softinvert = 0; +	if (invert) { +		const char *invmode; + +		if (resource_string_value(device_get_name(dev), +		    device_get_unit(dev), "invmode", &invmode)) +			invmode = NULL; + +		if (invmode) { +			if (!strcmp(invmode, "sw")) +				sc->sc_softinvert = 1; +			else if (!strcmp(invmode, "hw")) { +				if (pin_caps & GPIO_PIN_INVOUT) +					*pin_flags |= GPIO_PIN_INVOUT; +				else { +					device_printf(sc->sc_dev, "hardware pin inversion not supported\n"); +					return (-1); +				} +			} else { +				if (strcmp(invmode, "auto") != 0) +					device_printf(sc->sc_dev, "invalid pin inversion mode\n"); +				invmode = NULL; +			} +		} +		/* +		 * auto inversion mode: use hardware support if available, else fallback to +		 * software emulation. +		 */ +		if (invmode == NULL) { +			if (pin_caps & GPIO_PIN_INVOUT) +				*pin_flags |= GPIO_PIN_INVOUT; +			else +				sc->sc_softinvert = 1; +		} +	} +	MPASS(!invert || +	    (((*pin_flags & GPIO_PIN_INVOUT) != 0) && !sc->sc_softinvert) || +	    (((*pin_flags & GPIO_PIN_INVOUT) == 0) && sc->sc_softinvert)); +	return (invert); +} + +static int  gpioled_attach(device_t dev)  {  	struct gpioled_softc *sc;  	int state;  	const char *name; +	uint32_t pin_flags; +	int invert;  	sc = device_get_softc(dev);  	sc->sc_dev = dev;  	sc->sc_busdev = device_get_parent(dev);  	GPIOLED_LOCK_INIT(sc); -	state = 0; - -	if (resource_string_value(device_get_name(dev),  +	if (resource_string_value(device_get_name(dev),  	    device_get_unit(dev), "name", &name))  		name = NULL; -	resource_int_value(device_get_name(dev), -	    device_get_unit(dev), "invert", &sc->sc_invert); -	resource_int_value(device_get_name(dev), -	    device_get_unit(dev), "state", &state); + +	if (resource_int_value(device_get_name(dev), +	    device_get_unit(dev), "state", &state)) +		state = 0; + +	pin_flags = GPIO_PIN_OUTPUT; +	invert = gpioled_inv(dev, &pin_flags); +	if (invert < 0) +		return (ENXIO); +	device_printf(sc->sc_dev, "state %d invert %s\n", +	    state, (invert ? (sc->sc_softinvert ? "sw" : "hw") : "no")); +	if (GPIOBUS_PIN_SETFLAGS(sc->sc_busdev, sc->sc_dev, GPIOLED_PIN, +	    pin_flags) != 0) { +		if (bootverbose) +			device_printf(sc->sc_dev, "unable to set pin flags, %#x\n", pin_flags); +		return (ENXIO); +	}  	sc->sc_leddev = led_create_state(gpioled_control, sc, name ? name :  	    device_get_nameunit(dev), state); diff --git a/sys/dev/gpio/ofw_gpiobus.c b/sys/dev/gpio/ofw_gpiobus.c index fc5fb03d6824..da1bfbc268b8 100644 --- a/sys/dev/gpio/ofw_gpiobus.c +++ b/sys/dev/gpio/ofw_gpiobus.c @@ -426,6 +426,9 @@ ofw_gpiobus_attach(device_t dev)  	err = gpiobus_init_softc(dev);  	if (err != 0)  		return (err); +	err = gpiobus_add_gpioc(dev); +	if (err != 0) +		return (err);  	bus_identify_children(dev);  	bus_enumerate_hinted_children(dev);  	/* @@ -451,28 +454,22 @@ ofw_gpiobus_add_child(device_t dev, u_int order, const char *name, int unit)  	device_t child;  	struct ofw_gpiobus_devinfo *devi; -	child = device_add_child_ordered(dev, order, name, unit); +	child = gpiobus_add_child_common(dev, order, name, unit, +	    sizeof(struct ofw_gpiobus_devinfo));  	if (child == NULL) -		return (child); -	devi = malloc(sizeof(struct ofw_gpiobus_devinfo), M_DEVBUF, -	    M_NOWAIT | M_ZERO); -	if (devi == NULL) { -		device_delete_child(dev, child); -		return (0); -	} +		return (NULL);  	/*  	 * NULL all the OFW-related parts of the ivars for non-OFW  	 * children.  	 */ +	devi = device_get_ivars(child);  	devi->opd_obdinfo.obd_node = -1;  	devi->opd_obdinfo.obd_name = NULL;  	devi->opd_obdinfo.obd_compat = NULL;  	devi->opd_obdinfo.obd_type = NULL;  	devi->opd_obdinfo.obd_model = NULL; -	device_set_ivars(child, devi); -  	return (child);  } diff --git a/sys/dev/gpio/pl061.c b/sys/dev/gpio/pl061.c index 87d4310a6396..9996b0253c7d 100644 --- a/sys/dev/gpio/pl061.c +++ b/sys/dev/gpio/pl061.c @@ -495,13 +495,14 @@ pl061_attach(device_t dev)  		goto free_isrc;  	} -	sc->sc_busdev = gpiobus_attach_bus(dev); +	sc->sc_busdev = gpiobus_add_bus(dev);  	if (sc->sc_busdev == NULL) {  		device_printf(dev, "couldn't attach gpio bus\n");  		PL061_LOCK_DESTROY(sc);  		goto free_isrc;  	} +	bus_attach_children(dev);  	return (0);  free_isrc: @@ -557,8 +558,7 @@ static device_method_t pl061_methods[] = {  	/* Bus interface */  	DEVMETHOD(bus_setup_intr,	bus_generic_setup_intr), -	DEVMETHOD(bus_activate_resource,	bus_generic_activate_resource), -	DEVMETHOD(bus_deactivate_resource,	bus_generic_deactivate_resource), +	DEVMETHOD(bus_teardown_intr,	bus_generic_teardown_intr),  	/* GPIO protocol */  	DEVMETHOD(gpio_get_bus,		pl061_get_bus), diff --git a/sys/dev/gpio/qoriq_gpio.c b/sys/dev/gpio/qoriq_gpio.c index 8b44cd256c79..d11868a23751 100644 --- a/sys/dev/gpio/qoriq_gpio.c +++ b/sys/dev/gpio/qoriq_gpio.c @@ -379,12 +379,13 @@ qoriq_gpio_attach(device_t dev)  	OF_device_register_xref(OF_xref_from_node(ofw_bus_get_node(dev)), dev); -	sc->busdev = gpiobus_attach_bus(dev); +	sc->busdev = gpiobus_add_bus(dev);  	if (sc->busdev == NULL) {  		qoriq_gpio_detach(dev);  		return (ENOMEM);  	} +	bus_attach_children(dev);  	return (0);  } | 
