3   Broadcom B43 wireless driver
 
   6   Copyright (c) 2005 Martin Langer <martin-langer@gmx.de>,
 
   7   Copyright (c) 2005 Stefano Brivio <st3@riseup.net>
 
   8   Copyright (c) 2005-2007 Michael Buesch <mb@bu3sch.de>
 
   9   Copyright (c) 2005 Danny van Dyk <kugelfang@gentoo.org>
 
  10   Copyright (c) 2005 Andreas Jaggi <andreas.jaggi@waterwave.ch>
 
  12   This program is free software; you can redistribute it and/or modify
 
  13   it under the terms of the GNU General Public License as published by
 
  14   the Free Software Foundation; either version 2 of the License, or
 
  15   (at your option) any later version.
 
  17   This program is distributed in the hope that it will be useful,
 
  18   but WITHOUT ANY WARRANTY; without even the implied warranty of
 
  19   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 
  20   GNU General Public License for more details.
 
  22   You should have received a copy of the GNU General Public License
 
  23   along with this program; see the file COPYING.  If not, write to
 
  24   the Free Software Foundation, Inc., 51 Franklin Steet, Fifth Floor,
 
  25   Boston, MA 02110-1301, USA.
 
  33 static void b43_led_turn_on(struct b43_wldev *dev, u8 led_index,
 
  36         struct b43_wl *wl = dev->wl;
 
  40         spin_lock_irqsave(&wl->leds_lock, flags);
 
  41         ctl = b43_read16(dev, B43_MMIO_GPIO_CONTROL);
 
  43                 ctl &= ~(1 << led_index);
 
  45                 ctl |= (1 << led_index);
 
  46         b43_write16(dev, B43_MMIO_GPIO_CONTROL, ctl);
 
  47         spin_unlock_irqrestore(&wl->leds_lock, flags);
 
  50 static void b43_led_turn_off(struct b43_wldev *dev, u8 led_index,
 
  53         struct b43_wl *wl = dev->wl;
 
  57         spin_lock_irqsave(&wl->leds_lock, flags);
 
  58         ctl = b43_read16(dev, B43_MMIO_GPIO_CONTROL);
 
  60                 ctl |= (1 << led_index);
 
  62                 ctl &= ~(1 << led_index);
 
  63         b43_write16(dev, B43_MMIO_GPIO_CONTROL, ctl);
 
  64         spin_unlock_irqrestore(&wl->leds_lock, flags);
 
  67 /* Callback from the LED subsystem. */
 
  68 static void b43_led_brightness_set(struct led_classdev *led_dev,
 
  69                                    enum led_brightness brightness)
 
  71         struct b43_led *led = container_of(led_dev, struct b43_led, led_dev);
 
  72         struct b43_wldev *dev = led->dev;
 
  75         /* Checking the radio-enabled status here is slightly racy,
 
  76          * but we want to avoid the locking overhead and we don't care
 
  77          * whether the LED has the wrong state for a second. */
 
  78         radio_enabled = (dev->phy.radio_on && dev->radio_hw_enable);
 
  80         if (brightness == LED_OFF || !radio_enabled)
 
  81                 b43_led_turn_off(dev, led->index, led->activelow);
 
  83                 b43_led_turn_on(dev, led->index, led->activelow);
 
  86 static int b43_register_led(struct b43_wldev *dev, struct b43_led *led,
 
  87                             const char *name, char *default_trigger,
 
  88                             u8 led_index, bool activelow)
 
  92         b43_led_turn_off(dev, led_index, activelow);
 
  98         led->index = led_index;
 
  99         led->activelow = activelow;
 
 100         strncpy(led->name, name, sizeof(led->name));
 
 102         led->led_dev.name = led->name;
 
 103         led->led_dev.default_trigger = default_trigger;
 
 104         led->led_dev.brightness_set = b43_led_brightness_set;
 
 106         err = led_classdev_register(dev->dev->dev, &led->led_dev);
 
 108                 b43warn(dev->wl, "LEDs: Failed to register %s\n", name);
 
 115 static void b43_unregister_led(struct b43_led *led)
 
 119         led_classdev_unregister(&led->led_dev);
 
 120         b43_led_turn_off(led->dev, led->index, led->activelow);
 
 124 static void b43_map_led(struct b43_wldev *dev,
 
 126                         enum b43_led_behaviour behaviour,
 
 129         struct ieee80211_hw *hw = dev->wl->hw;
 
 130         char name[B43_LED_MAX_NAME_LEN + 1];
 
 132         /* Map the b43 specific LED behaviour value to the
 
 133          * generic LED triggers. */
 
 135         case B43_LED_INACTIVE:
 
 138                 b43_led_turn_off(dev, led_index, activelow);
 
 141                 b43_led_turn_on(dev, led_index, activelow);
 
 143         case B43_LED_ACTIVITY:
 
 144         case B43_LED_TRANSFER:
 
 145         case B43_LED_APTRANSFER:
 
 146                 snprintf(name, sizeof(name),
 
 147                          "b43-%s:tx", wiphy_name(hw->wiphy));
 
 148                 b43_register_led(dev, &dev->led_tx, name,
 
 149                                  ieee80211_get_tx_led_name(hw),
 
 150                                  led_index, activelow);
 
 151                 snprintf(name, sizeof(name),
 
 152                          "b43-%s:rx", wiphy_name(hw->wiphy));
 
 153                 b43_register_led(dev, &dev->led_rx, name,
 
 154                                  ieee80211_get_rx_led_name(hw),
 
 155                                  led_index, activelow);
 
 157         case B43_LED_RADIO_ALL:
 
 158         case B43_LED_RADIO_A:
 
 159         case B43_LED_RADIO_B:
 
 160         case B43_LED_MODE_BG:
 
 161                 snprintf(name, sizeof(name),
 
 162                          "b43-%s:radio", wiphy_name(hw->wiphy));
 
 163                 b43_register_led(dev, &dev->led_radio, name,
 
 164                                  b43_rfkill_led_name(dev),
 
 165                                  led_index, activelow);
 
 169                 snprintf(name, sizeof(name),
 
 170                          "b43-%s:assoc", wiphy_name(hw->wiphy));
 
 171                 b43_register_led(dev, &dev->led_assoc, name,
 
 172                                  ieee80211_get_assoc_led_name(hw),
 
 173                                  led_index, activelow);
 
 176                 b43warn(dev->wl, "LEDs: Unknown behaviour 0x%02X\n",
 
 182 void b43_leds_init(struct b43_wldev *dev)
 
 184         struct ssb_bus *bus = dev->dev->bus;
 
 187         enum b43_led_behaviour behaviour;
 
 190         sprom[0] = bus->sprom.r1.gpio0;
 
 191         sprom[1] = bus->sprom.r1.gpio1;
 
 192         sprom[2] = bus->sprom.r1.gpio2;
 
 193         sprom[3] = bus->sprom.r1.gpio3;
 
 195         for (i = 0; i < 4; i++) {
 
 196                 if (sprom[i] == 0xFF) {
 
 197                         /* There is no LED information in the SPROM
 
 198                          * for this LED. Hardcode it here. */
 
 202                                 behaviour = B43_LED_ACTIVITY;
 
 204                                 if (bus->boardinfo.vendor == PCI_VENDOR_ID_COMPAQ)
 
 205                                         behaviour = B43_LED_RADIO_ALL;
 
 208                                 behaviour = B43_LED_RADIO_B;
 
 209                                 if (bus->boardinfo.vendor == PCI_VENDOR_ID_ASUSTEK)
 
 210                                         behaviour = B43_LED_ASSOC;
 
 213                                 behaviour = B43_LED_RADIO_A;
 
 216                                 behaviour = B43_LED_OFF;
 
 223                         behaviour = sprom[i] & B43_LED_BEHAVIOUR;
 
 224                         activelow = !!(sprom[i] & B43_LED_ACTIVELOW);
 
 226                 b43_map_led(dev, i, behaviour, activelow);
 
 230 void b43_leds_exit(struct b43_wldev *dev)
 
 232         b43_unregister_led(&dev->led_tx);
 
 233         b43_unregister_led(&dev->led_rx);
 
 234         b43_unregister_led(&dev->led_assoc);