2010-12-18 13:36:02 +00:00
|
|
|
/*
|
|
|
|
* gpio LED support for barebox
|
|
|
|
*
|
|
|
|
* (C) Copyright 2010 Sascha Hauer, Pengutronix
|
|
|
|
*
|
|
|
|
* See file CREDITS for list of people who contributed to this
|
|
|
|
* project.
|
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU General Public License as
|
|
|
|
* published by the Free Software Foundation; either version 2 of
|
|
|
|
* the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
#include <common.h>
|
2013-07-02 18:30:47 +00:00
|
|
|
#include <init.h>
|
2010-12-18 13:36:02 +00:00
|
|
|
#include <led.h>
|
2012-12-26 16:33:56 +00:00
|
|
|
#include <gpio.h>
|
2013-07-02 18:30:47 +00:00
|
|
|
#include <of_gpio.h>
|
2010-12-18 13:36:02 +00:00
|
|
|
|
|
|
|
static void led_gpio_set(struct led *led, unsigned int value)
|
|
|
|
{
|
|
|
|
struct gpio_led *gpio_led = container_of(led, struct gpio_led, led);
|
|
|
|
|
|
|
|
gpio_direction_output(gpio_led->gpio, !!value ^ gpio_led->active_low);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* led_gpio_register - register a gpio controlled LED
|
|
|
|
* @param led The gpio LED
|
|
|
|
*
|
|
|
|
* This function registers a single gpio as a LED. led->gpio
|
|
|
|
* should be initialized to the gpio to control.
|
|
|
|
*/
|
|
|
|
int led_gpio_register(struct gpio_led *led)
|
|
|
|
{
|
2012-12-26 16:33:56 +00:00
|
|
|
int ret;
|
|
|
|
char *name = led->led.name;
|
|
|
|
|
|
|
|
ret = gpio_request(led->gpio, name ? name : "led");
|
|
|
|
if (ret)
|
|
|
|
return ret;
|
|
|
|
|
2010-12-18 13:36:02 +00:00
|
|
|
led->led.set = led_gpio_set;
|
|
|
|
led->led.max_value = 1;
|
|
|
|
|
2012-12-26 16:33:56 +00:00
|
|
|
ret = led_register(&led->led);
|
|
|
|
if (ret)
|
|
|
|
gpio_free(led->gpio);
|
|
|
|
|
|
|
|
return ret;
|
2010-12-18 13:36:02 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* led_gpio_unregister - remove a gpio controlled LED from the framework
|
|
|
|
* @param led The gpio LED
|
|
|
|
*/
|
|
|
|
void led_gpio_unregister(struct gpio_led *led)
|
|
|
|
{
|
|
|
|
led_unregister(&led->led);
|
|
|
|
}
|
|
|
|
|
2012-11-16 20:17:23 +00:00
|
|
|
#ifdef CONFIG_LED_GPIO_BICOLOR
|
|
|
|
static void led_gpio_bicolor_set(struct led *led, unsigned int value)
|
|
|
|
{
|
|
|
|
struct gpio_bicolor_led *bi = container_of(led, struct gpio_bicolor_led, led);
|
|
|
|
int al = bi->active_low;
|
|
|
|
|
|
|
|
switch (value) {
|
|
|
|
case 0:
|
|
|
|
gpio_direction_output(bi->gpio_c0, al);
|
|
|
|
gpio_direction_output(bi->gpio_c1, al);
|
|
|
|
break;
|
|
|
|
case 1:
|
|
|
|
gpio_direction_output(bi->gpio_c0, !al);
|
|
|
|
gpio_direction_output(bi->gpio_c1, al);
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
gpio_direction_output(bi->gpio_c0, al);
|
|
|
|
gpio_direction_output(bi->gpio_c1, !al);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* led_gpio_bicolor_register - register three gpios as a bicolor LED
|
|
|
|
* @param led The gpio bicolor LED
|
|
|
|
*
|
|
|
|
* This function registers three gpios as a bicolor LED. led->gpio[rg]
|
|
|
|
* should be initialized to the gpios to control.
|
|
|
|
*/
|
|
|
|
int led_gpio_bicolor_register(struct gpio_bicolor_led *led)
|
|
|
|
{
|
2012-12-26 16:33:56 +00:00
|
|
|
int ret;
|
|
|
|
char *name = led->led.name;
|
|
|
|
|
|
|
|
ret = gpio_request(led->gpio_c0, name ? name : "led_c0");
|
|
|
|
if (ret)
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
ret = gpio_request(led->gpio_c1, name ? name : "led_c1");
|
|
|
|
if (ret)
|
|
|
|
goto err_gpio_c0;
|
|
|
|
|
2012-11-16 20:17:23 +00:00
|
|
|
led->led.set = led_gpio_bicolor_set;
|
|
|
|
led->led.max_value = 2;
|
|
|
|
|
2012-12-26 16:33:56 +00:00
|
|
|
ret = led_register(&led->led);
|
|
|
|
if (ret)
|
|
|
|
goto err_gpio_c1;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
err_gpio_c1:
|
|
|
|
gpio_free(led->gpio_c1);
|
|
|
|
err_gpio_c0:
|
|
|
|
gpio_free(led->gpio_c0);
|
|
|
|
return ret;
|
2012-11-16 20:17:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* led_gpio_bicolor_unregister - remove a gpio controlled bicolor LED from the framework
|
|
|
|
* @param led The gpio LED
|
|
|
|
*/
|
|
|
|
void led_gpio_bicolor_unregister(struct gpio_bicolor_led *led)
|
|
|
|
{
|
|
|
|
led_unregister(&led->led);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2010-12-18 13:36:02 +00:00
|
|
|
#ifdef CONFIG_LED_GPIO_RGB
|
|
|
|
|
|
|
|
static void led_gpio_rgb_set(struct led *led, unsigned int value)
|
|
|
|
{
|
|
|
|
struct gpio_rgb_led *rgb = container_of(led, struct gpio_rgb_led, led);
|
|
|
|
int al = rgb->active_low;
|
|
|
|
|
|
|
|
gpio_direction_output(rgb->gpio_r, !!(value & 4) ^ al);
|
|
|
|
gpio_direction_output(rgb->gpio_g, !!(value & 2) ^ al);
|
|
|
|
gpio_direction_output(rgb->gpio_b, !!(value & 1) ^ al);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* led_gpio_rgb_register - register three gpios as a rgb LED
|
|
|
|
* @param led The gpio rg LED
|
|
|
|
*
|
|
|
|
* This function registers three gpios as a rgb LED. led->gpio[rgb]
|
|
|
|
* should be initialized to the gpios to control.
|
|
|
|
*/
|
|
|
|
int led_gpio_rgb_register(struct gpio_rgb_led *led)
|
|
|
|
{
|
2012-12-26 16:33:56 +00:00
|
|
|
int ret;
|
|
|
|
char *name = led->led.name;
|
|
|
|
|
|
|
|
ret = gpio_request(led->gpio_r, name ? name : "led_r");
|
|
|
|
if (ret)
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
ret = gpio_request(led->gpio_g, name ? name : "led_g");
|
|
|
|
if (ret)
|
|
|
|
goto err_gpio_r;
|
|
|
|
|
|
|
|
ret = gpio_request(led->gpio_b, name ? name : "led_b");
|
|
|
|
if (ret)
|
|
|
|
goto err_gpio_g;
|
|
|
|
|
2010-12-18 13:36:02 +00:00
|
|
|
led->led.set = led_gpio_rgb_set;
|
|
|
|
led->led.max_value = 7;
|
|
|
|
|
2012-12-26 16:33:56 +00:00
|
|
|
ret = led_register(&led->led);
|
|
|
|
if (ret)
|
|
|
|
goto err_gpio_b;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
err_gpio_b:
|
|
|
|
gpio_free(led->gpio_b);
|
|
|
|
err_gpio_g:
|
|
|
|
gpio_free(led->gpio_g);
|
|
|
|
err_gpio_r:
|
|
|
|
gpio_free(led->gpio_r);
|
|
|
|
return ret;
|
2010-12-18 13:36:02 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* led_gpio_rgb_unregister - remove a gpio controlled rgb LED from the framework
|
|
|
|
* @param led The gpio LED
|
|
|
|
*/
|
|
|
|
void led_gpio_rgb_unregister(struct gpio_led *led)
|
|
|
|
{
|
|
|
|
led_unregister(&led->led);
|
|
|
|
}
|
|
|
|
#endif /* CONFIG_LED_GPIO_RGB */
|
2013-07-02 18:30:47 +00:00
|
|
|
|
|
|
|
#ifdef CONFIG_LED_GPIO_OF
|
|
|
|
static int led_gpio_of_probe(struct device_d *dev)
|
|
|
|
{
|
|
|
|
struct device_node *child;
|
|
|
|
|
|
|
|
for_each_child_of_node(dev->device_node, child) {
|
|
|
|
struct gpio_led *gled;
|
|
|
|
enum of_gpio_flags flags;
|
|
|
|
int gpio;
|
2014-07-29 05:36:36 +00:00
|
|
|
const char *label;
|
2013-07-02 18:30:47 +00:00
|
|
|
|
|
|
|
gpio = of_get_named_gpio_flags(child, "gpios", 0, &flags);
|
|
|
|
if (gpio < 0)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
gled = xzalloc(sizeof(*gled));
|
2014-07-29 05:36:36 +00:00
|
|
|
if (of_property_read_string(child, "label", &label))
|
|
|
|
label = child->name;
|
|
|
|
gled->led.name = xstrdup(label);
|
|
|
|
|
2013-07-02 18:30:47 +00:00
|
|
|
gled->gpio = gpio;
|
|
|
|
gled->active_low = (flags & OF_GPIO_ACTIVE_LOW) ? 1 : 0;
|
|
|
|
|
|
|
|
dev_dbg(dev, "register led %s on gpio%d, active_low = %d\n",
|
|
|
|
gled->led.name, gled->gpio, gled->active_low);
|
|
|
|
|
|
|
|
led_gpio_register(gled);
|
2013-11-09 13:24:06 +00:00
|
|
|
led_of_parse_trigger(&gled->led, child);
|
2013-07-02 18:30:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct of_device_id led_gpio_of_ids[] = {
|
|
|
|
{ .compatible = "gpio-leds", },
|
|
|
|
{ }
|
|
|
|
};
|
|
|
|
|
|
|
|
static struct driver_d led_gpio_of_driver = {
|
|
|
|
.name = "gpio-leds",
|
|
|
|
.probe = led_gpio_of_probe,
|
|
|
|
.of_compatible = DRV_OF_COMPAT(led_gpio_of_ids),
|
|
|
|
};
|
|
|
|
device_platform_driver(led_gpio_of_driver);
|
|
|
|
|
|
|
|
#endif /* CONFIG LED_GPIO_OF */
|