1223637Sbz/* $OpenBSD: gpioleds.c,v 1.4 2022/04/06 18:59:28 naddy Exp $ */ 2126353Smlaier/* 3126353Smlaier * Copyright (c) 2021 Klemens Nanni <kn@openbsd.org> 4126353Smlaier * 5126353Smlaier * Permission to use, copy, modify, and distribute this software for any 6126353Smlaier * purpose with or without fee is hereby granted, provided that the above 7126353Smlaier * copyright notice and this permission notice appear in all copies. 8126353Smlaier * 9126353Smlaier * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES 10126353Smlaier * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF 11126353Smlaier * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR 12126353Smlaier * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES 13126353Smlaier * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN 14126353Smlaier * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF 15126353Smlaier * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. 16126353Smlaier */ 17126353Smlaier 18126353Smlaier#include <sys/param.h> 19126353Smlaier#include <sys/systm.h> 20126353Smlaier#include <sys/device.h> 21126353Smlaier#include <sys/gpio.h> 22126353Smlaier#include <sys/malloc.h> 23126353Smlaier 24126353Smlaier#include <machine/bus.h> 25126353Smlaier#include <machine/fdt.h> 26126353Smlaier 27126353Smlaier#include <dev/gpio/gpiovar.h> 28126353Smlaier#include <dev/ofw/ofw_gpio.h> 29126353Smlaier#include <dev/ofw/ofw_pinctrl.h> 30126353Smlaier#include <dev/ofw/openfirm.h> 31126353Smlaier#include <dev/ofw/fdt.h> 32126353Smlaier 33127082Sobrienstruct gpioleds_softc { 34127082Sobrien struct device sc_dev; 35127082Sobrien int sc_node; 36126353Smlaier}; 37130617Smlaier 38126353Smlaierint gpioleds_match(struct device *, void *, void *); 39126353Smlaiervoid gpioleds_attach(struct device *, struct device *, void *); 40223637Sbz 41223637Sbzconst struct cfattach gpioleds_ca = { 42126353Smlaier sizeof (struct gpioleds_softc), gpioleds_match, gpioleds_attach 43126353Smlaier}; 44126353Smlaier 45126353Smlaierstruct cfdriver gpioleds_cd = { 46126353Smlaier NULL, "gpioleds", DV_DULL 47126353Smlaier}; 48126353Smlaier 49126353Smlaierint 50223637Sbzgpioleds_match(struct device *parent, void *match, void *aux) 51126353Smlaier{ 52126353Smlaier const struct fdt_attach_args *faa = aux; 53126353Smlaier 54127024Smlaier return OF_is_compatible(faa->fa_node, "gpio-leds"); 55223637Sbz} 56126355Smlaier 57126355Smlaiervoid 58126353Smlaiergpioleds_attach(struct device *parent, struct device *self, void *aux) 59126355Smlaier{ 60130617Smlaier struct fdt_attach_args *faa = aux; 61126353Smlaier uint32_t *led_pin; 62126353Smlaier char *function, *default_state; 63130617Smlaier char *function_prop = "function"; 64126353Smlaier int function_len, default_state_len, gpios_len; 65126353Smlaier int node, leds = 0; 66130617Smlaier 67130617Smlaier pinctrl_byname(faa->fa_node, "default"); 68126353Smlaier 69223637Sbz for (node = OF_child(faa->fa_node); node; node = OF_peer(node)) { 70126353Smlaier function_len = OF_getproplen(node, function_prop); 71126353Smlaier if (function_len <= 0) { 72126353Smlaier function_prop = "label"; 73126353Smlaier function_len = OF_getproplen(node, function_prop); 74126353Smlaier if (function_len <= 0) 75126353Smlaier continue; 76126353Smlaier } 77126353Smlaier default_state_len = OF_getproplen(node, "default-state"); 78126353Smlaier if (default_state_len <= 0) 79126353Smlaier continue; 80130617Smlaier gpios_len = OF_getproplen(node, "gpios"); 81130617Smlaier if (gpios_len <= 0) 82130617Smlaier continue; 83223637Sbz 84130617Smlaier function = malloc(function_len, M_TEMP, M_WAITOK); 85223637Sbz OF_getprop(node, function_prop, function, function_len); 86126353Smlaier default_state = malloc(default_state_len, M_TEMP, M_WAITOK); 87130617Smlaier OF_getprop(node, "default-state", default_state, default_state_len); 88130617Smlaier led_pin = malloc(gpios_len, M_TEMP, M_WAITOK); 89171172Smlaier OF_getpropintarray(node, "gpios", led_pin, gpios_len); 90130617Smlaier gpio_controller_config_pin(led_pin, GPIO_CONFIG_OUTPUT); 91130617Smlaier if (strcmp(default_state, "on") == 0) 92130617Smlaier gpio_controller_set_pin(led_pin, 1); 93126353Smlaier else if (strcmp(default_state, "off") == 0) 94223637Sbz gpio_controller_set_pin(led_pin, 0); 95126353Smlaier 96126353Smlaier printf("%s \"%s\"", leds++ ? "," : ":", function); 97126353Smlaier free(function, M_TEMP, function_len); 98126353Smlaier free(default_state, M_TEMP, default_state_len); 99171172Smlaier free(led_pin, M_TEMP, gpios_len); 100171172Smlaier } 101130617Smlaier 102130617Smlaier if (leds == 0) 103130617Smlaier printf(": no LEDs"); 104130617Smlaier printf("\n"); 105130617Smlaier} 106130617Smlaier