2 Copyright (C) 2004 - 2009 Ivo van Doorn <IvDoorn@gmail.com>
3 <http://rt2x00.serialmonkey.com>
5 This program is free software; you can redistribute it and/or modify
6 it under the terms of the GNU General Public License as published by
7 the Free Software Foundation; either version 2 of the License, or
8 (at your option) any later version.
10 This program is distributed in the hope that it will be useful,
11 but WITHOUT ANY WARRANTY; without even the implied warranty of
12 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 GNU General Public License for more details.
15 You should have received a copy of the GNU General Public License
16 along with this program; if not, see <http://www.gnu.org/licenses/>.
21 Abstract: rt2x00 led specific routines.
24 #include <linux/kernel.h>
25 #include <linux/module.h>
28 #include "rt2x00lib.h"
30 void rt2x00leds_led_quality(struct rt2x00_dev
*rt2x00dev
, int rssi
)
32 struct rt2x00_led
*led
= &rt2x00dev
->led_qual
;
33 unsigned int brightness
;
35 if ((led
->type
!= LED_TYPE_QUALITY
) || !(led
->flags
& LED_REGISTERED
))
39 * Led handling requires a positive value for the rssi,
40 * to do that correctly we need to add the correction.
42 rssi
+= rt2x00dev
->rssi_offset
;
45 * Get the rssi level, this is used to convert the rssi
46 * to a LED value inside the range LED_OFF - LED_FULL.
62 * Note that we must _not_ send LED_OFF since the driver
63 * is going to calculate the value and might use it in a
66 brightness
= ((LED_FULL
/ 6) * rssi
) + 1;
67 if (brightness
!= led
->led_dev
.brightness
) {
68 led
->led_dev
.brightness_set(&led
->led_dev
, brightness
);
69 led
->led_dev
.brightness
= brightness
;
73 static void rt2x00led_led_simple(struct rt2x00_led
*led
, bool enabled
)
75 unsigned int brightness
= enabled
? LED_FULL
: LED_OFF
;
77 if (!(led
->flags
& LED_REGISTERED
))
80 led
->led_dev
.brightness_set(&led
->led_dev
, brightness
);
81 led
->led_dev
.brightness
= brightness
;
84 void rt2x00led_led_activity(struct rt2x00_dev
*rt2x00dev
, bool enabled
)
86 if (rt2x00dev
->led_qual
.type
== LED_TYPE_ACTIVITY
)
87 rt2x00led_led_simple(&rt2x00dev
->led_qual
, enabled
);
90 void rt2x00leds_led_assoc(struct rt2x00_dev
*rt2x00dev
, bool enabled
)
92 if (rt2x00dev
->led_assoc
.type
== LED_TYPE_ASSOC
)
93 rt2x00led_led_simple(&rt2x00dev
->led_assoc
, enabled
);
96 void rt2x00leds_led_radio(struct rt2x00_dev
*rt2x00dev
, bool enabled
)
98 if (rt2x00dev
->led_radio
.type
== LED_TYPE_RADIO
)
99 rt2x00led_led_simple(&rt2x00dev
->led_radio
, enabled
);
102 static int rt2x00leds_register_led(struct rt2x00_dev
*rt2x00dev
,
103 struct rt2x00_led
*led
,
106 struct device
*device
= wiphy_dev(rt2x00dev
->hw
->wiphy
);
109 led
->led_dev
.name
= name
;
110 led
->led_dev
.brightness
= LED_OFF
;
112 retval
= led_classdev_register(device
, &led
->led_dev
);
114 rt2x00_err(rt2x00dev
, "Failed to register led handler\n");
118 led
->flags
|= LED_REGISTERED
;
123 void rt2x00leds_register(struct rt2x00_dev
*rt2x00dev
)
127 unsigned long on_period
;
128 unsigned long off_period
;
129 const char *phy_name
= wiphy_name(rt2x00dev
->hw
->wiphy
);
131 if (rt2x00dev
->led_radio
.flags
& LED_INITIALIZED
) {
132 snprintf(name
, sizeof(name
), "%s-%s::radio",
133 rt2x00dev
->ops
->name
, phy_name
);
135 retval
= rt2x00leds_register_led(rt2x00dev
,
136 &rt2x00dev
->led_radio
,
142 if (rt2x00dev
->led_assoc
.flags
& LED_INITIALIZED
) {
143 snprintf(name
, sizeof(name
), "%s-%s::assoc",
144 rt2x00dev
->ops
->name
, phy_name
);
146 retval
= rt2x00leds_register_led(rt2x00dev
,
147 &rt2x00dev
->led_assoc
,
153 if (rt2x00dev
->led_qual
.flags
& LED_INITIALIZED
) {
154 snprintf(name
, sizeof(name
), "%s-%s::quality",
155 rt2x00dev
->ops
->name
, phy_name
);
157 retval
= rt2x00leds_register_led(rt2x00dev
,
158 &rt2x00dev
->led_qual
,
165 * Initialize blink time to default value:
169 if (rt2x00dev
->led_radio
.led_dev
.blink_set
) {
172 rt2x00dev
->led_radio
.led_dev
.blink_set(
173 &rt2x00dev
->led_radio
.led_dev
, &on_period
, &off_period
);
179 rt2x00leds_unregister(rt2x00dev
);
182 static void rt2x00leds_unregister_led(struct rt2x00_led
*led
)
184 led_classdev_unregister(&led
->led_dev
);
187 * This might look weird, but when we are unregistering while
188 * suspended the led is already off, and since we haven't
189 * fully resumed yet, access to the device might not be
192 if (!(led
->led_dev
.flags
& LED_SUSPENDED
))
193 led
->led_dev
.brightness_set(&led
->led_dev
, LED_OFF
);
195 led
->flags
&= ~LED_REGISTERED
;
198 void rt2x00leds_unregister(struct rt2x00_dev
*rt2x00dev
)
200 if (rt2x00dev
->led_qual
.flags
& LED_REGISTERED
)
201 rt2x00leds_unregister_led(&rt2x00dev
->led_qual
);
202 if (rt2x00dev
->led_assoc
.flags
& LED_REGISTERED
)
203 rt2x00leds_unregister_led(&rt2x00dev
->led_assoc
);
204 if (rt2x00dev
->led_radio
.flags
& LED_REGISTERED
)
205 rt2x00leds_unregister_led(&rt2x00dev
->led_radio
);
208 static inline void rt2x00leds_suspend_led(struct rt2x00_led
*led
)
210 led_classdev_suspend(&led
->led_dev
);
212 /* This shouldn't be needed, but just to be safe */
213 led
->led_dev
.brightness_set(&led
->led_dev
, LED_OFF
);
214 led
->led_dev
.brightness
= LED_OFF
;
217 void rt2x00leds_suspend(struct rt2x00_dev
*rt2x00dev
)
219 if (rt2x00dev
->led_qual
.flags
& LED_REGISTERED
)
220 rt2x00leds_suspend_led(&rt2x00dev
->led_qual
);
221 if (rt2x00dev
->led_assoc
.flags
& LED_REGISTERED
)
222 rt2x00leds_suspend_led(&rt2x00dev
->led_assoc
);
223 if (rt2x00dev
->led_radio
.flags
& LED_REGISTERED
)
224 rt2x00leds_suspend_led(&rt2x00dev
->led_radio
);
227 static inline void rt2x00leds_resume_led(struct rt2x00_led
*led
)
229 led_classdev_resume(&led
->led_dev
);
231 /* Device might have enabled the LEDS during resume */
232 led
->led_dev
.brightness_set(&led
->led_dev
, LED_OFF
);
233 led
->led_dev
.brightness
= LED_OFF
;
236 void rt2x00leds_resume(struct rt2x00_dev
*rt2x00dev
)
238 if (rt2x00dev
->led_radio
.flags
& LED_REGISTERED
)
239 rt2x00leds_resume_led(&rt2x00dev
->led_radio
);
240 if (rt2x00dev
->led_assoc
.flags
& LED_REGISTERED
)
241 rt2x00leds_resume_led(&rt2x00dev
->led_assoc
);
242 if (rt2x00dev
->led_qual
.flags
& LED_REGISTERED
)
243 rt2x00leds_resume_led(&rt2x00dev
->led_qual
);