123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128 |
- // SPDX-License-Identifier: GPL-2.0
- #include <net/mac80211.h>
- #include <linux/bcma/bcma_driver_chipcommon.h>
- #include <linux/gpio.h>
- #include "mac80211_if.h"
- #include "pub.h"
- #include "main.h"
- #include "led.h"
- /* number of leds */
- #define BRCMS_LED_NO 4
- /* behavior mask */
- #define BRCMS_LED_BEH_MASK 0x7f
- /* activelow (polarity) bit */
- #define BRCMS_LED_AL_MASK 0x80
- /* radio enabled */
- #define BRCMS_LED_RADIO 3
- static void brcms_radio_led_ctrl(struct brcms_info *wl, bool state)
- {
- if (wl->radio_led.gpio == -1)
- return;
- if (wl->radio_led.active_low)
- state = !state;
- if (state)
- gpio_set_value(wl->radio_led.gpio, 1);
- else
- gpio_set_value(wl->radio_led.gpio, 0);
- }
- /* Callback from the LED subsystem. */
- static void brcms_led_brightness_set(struct led_classdev *led_dev,
- enum led_brightness brightness)
- {
- struct brcms_info *wl = container_of(led_dev,
- struct brcms_info, led_dev);
- brcms_radio_led_ctrl(wl, brightness);
- }
- void brcms_led_unregister(struct brcms_info *wl)
- {
- if (wl->led_dev.dev)
- led_classdev_unregister(&wl->led_dev);
- if (wl->radio_led.gpio != -1)
- gpio_free(wl->radio_led.gpio);
- }
- int brcms_led_register(struct brcms_info *wl)
- {
- int i, err;
- struct brcms_led *radio_led = &wl->radio_led;
- /* get CC core */
- struct bcma_drv_cc *cc_drv = &wl->wlc->hw->d11core->bus->drv_cc;
- struct gpio_chip *bcma_gpio = &cc_drv->gpio;
- struct ssb_sprom *sprom = &wl->wlc->hw->d11core->bus->sprom;
- u8 *leds[] = { &sprom->gpio0,
- &sprom->gpio1,
- &sprom->gpio2,
- &sprom->gpio3 };
- unsigned gpio = -1;
- bool active_low = false;
- /* none by default */
- radio_led->gpio = -1;
- radio_led->active_low = false;
- if (!bcma_gpio || !gpio_is_valid(bcma_gpio->base))
- return -ENODEV;
- /* find radio enabled LED */
- for (i = 0; i < BRCMS_LED_NO; i++) {
- u8 led = *leds[i];
- if ((led & BRCMS_LED_BEH_MASK) == BRCMS_LED_RADIO) {
- gpio = bcma_gpio->base + i;
- if (led & BRCMS_LED_AL_MASK)
- active_low = true;
- break;
- }
- }
- if (gpio == -1 || !gpio_is_valid(gpio))
- return -ENODEV;
- /* request and configure LED gpio */
- err = gpio_request_one(gpio,
- active_low ? GPIOF_OUT_INIT_HIGH
- : GPIOF_OUT_INIT_LOW,
- "radio on");
- if (err) {
- wiphy_err(wl->wiphy, "requesting led gpio %d failed (err: %d)\n",
- gpio, err);
- return err;
- }
- err = gpio_direction_output(gpio, 1);
- if (err) {
- wiphy_err(wl->wiphy, "cannot set led gpio %d to output (err: %d)\n",
- gpio, err);
- return err;
- }
- snprintf(wl->radio_led.name, sizeof(wl->radio_led.name),
- "brcmsmac-%s:radio", wiphy_name(wl->wiphy));
- wl->led_dev.name = wl->radio_led.name;
- wl->led_dev.default_trigger =
- ieee80211_get_radio_led_name(wl->pub->ieee_hw);
- wl->led_dev.brightness_set = brcms_led_brightness_set;
- err = led_classdev_register(wiphy_dev(wl->wiphy), &wl->led_dev);
- if (err) {
- wiphy_err(wl->wiphy, "cannot register led device: %s (err: %d)\n",
- wl->radio_led.name, err);
- return err;
- }
- wiphy_info(wl->wiphy, "registered radio enabled led device: %s gpio: %d\n",
- wl->radio_led.name,
- gpio);
- radio_led->gpio = gpio;
- radio_led->active_low = active_low;
- return 0;
- }
|