diff options
author | Jacek Anaszewski <j.anaszewski@samsung.com> | 2015-10-07 05:10:41 -0400 |
---|---|---|
committer | Jacek Anaszewski <j.anaszewski@samsung.com> | 2016-01-04 03:57:30 -0500 |
commit | 81fe8e5b73e3f4de578ac7f53c1d956d4f58b8d4 (patch) | |
tree | ff084840f52a554f7a902c7e1ffe54416e4c452f | |
parent | 437a4240f26461610cc0d90950b1c88348d9babe (diff) |
leds: core: Add led_set_brightness_nosleep{nopm} functions
This patch adds led_set_brightness_nosleep() and led_set_brightness_nopm()
functions, that guarantee setting LED brightness in a non-blocking way.
The latter is used from pm_ops context and doesn't modify the brightness
cached in the struct led_classdev. Its execution always ends up with
a call to brightness setting op - either directly or through
a set_brightness_work, regardless of LED_SUSPENDED flag state.
The patch also replaces led_set_brightness_async() with
led_set_brightness_nosleep() in all places where the most vital was setting
brightness in a non sleeping way but not necessarily asynchronously, which
is not needed for non-blocking drivers.
Signed-off-by: Jacek Anaszewski <j.anaszewski@samsung.com>
Acked-by: Sakari Ailus <sakari.ailus@linux.intel.com>
-rw-r--r-- | drivers/leds/led-class.c | 4 | ||||
-rw-r--r-- | drivers/leds/led-core.c | 38 | ||||
-rw-r--r-- | drivers/leds/leds.h | 4 | ||||
-rw-r--r-- | drivers/leds/trigger/ledtrig-backlight.c | 8 | ||||
-rw-r--r-- | drivers/leds/trigger/ledtrig-default-on.c | 2 | ||||
-rw-r--r-- | drivers/leds/trigger/ledtrig-gpio.c | 6 | ||||
-rw-r--r-- | drivers/leds/trigger/ledtrig-heartbeat.c | 4 | ||||
-rw-r--r-- | drivers/leds/trigger/ledtrig-oneshot.c | 4 | ||||
-rw-r--r-- | drivers/leds/trigger/ledtrig-transient.c | 8 |
9 files changed, 55 insertions, 23 deletions
diff --git a/drivers/leds/led-class.c b/drivers/leds/led-class.c index 7385f98dd54b..83a1dc725864 100644 --- a/drivers/leds/led-class.c +++ b/drivers/leds/led-class.c | |||
@@ -109,7 +109,7 @@ static const struct attribute_group *led_groups[] = { | |||
109 | void led_classdev_suspend(struct led_classdev *led_cdev) | 109 | void led_classdev_suspend(struct led_classdev *led_cdev) |
110 | { | 110 | { |
111 | led_cdev->flags |= LED_SUSPENDED; | 111 | led_cdev->flags |= LED_SUSPENDED; |
112 | led_cdev->brightness_set(led_cdev, 0); | 112 | led_set_brightness_nopm(led_cdev, 0); |
113 | } | 113 | } |
114 | EXPORT_SYMBOL_GPL(led_classdev_suspend); | 114 | EXPORT_SYMBOL_GPL(led_classdev_suspend); |
115 | 115 | ||
@@ -119,7 +119,7 @@ EXPORT_SYMBOL_GPL(led_classdev_suspend); | |||
119 | */ | 119 | */ |
120 | void led_classdev_resume(struct led_classdev *led_cdev) | 120 | void led_classdev_resume(struct led_classdev *led_cdev) |
121 | { | 121 | { |
122 | led_cdev->brightness_set(led_cdev, led_cdev->brightness); | 122 | led_set_brightness_nopm(led_cdev, led_cdev->brightness); |
123 | 123 | ||
124 | if (led_cdev->flash_resume) | 124 | if (led_cdev->flash_resume) |
125 | led_cdev->flash_resume(led_cdev); | 125 | led_cdev->flash_resume(led_cdev); |
diff --git a/drivers/leds/led-core.c b/drivers/leds/led-core.c index 7cf59243e0ef..3d168592db65 100644 --- a/drivers/leds/led-core.c +++ b/drivers/leds/led-core.c | |||
@@ -32,7 +32,7 @@ static void led_timer_function(unsigned long data) | |||
32 | unsigned long delay; | 32 | unsigned long delay; |
33 | 33 | ||
34 | if (!led_cdev->blink_delay_on || !led_cdev->blink_delay_off) { | 34 | if (!led_cdev->blink_delay_on || !led_cdev->blink_delay_off) { |
35 | led_set_brightness_async(led_cdev, LED_OFF); | 35 | led_set_brightness_nosleep(led_cdev, LED_OFF); |
36 | return; | 36 | return; |
37 | } | 37 | } |
38 | 38 | ||
@@ -60,7 +60,7 @@ static void led_timer_function(unsigned long data) | |||
60 | delay = led_cdev->blink_delay_off; | 60 | delay = led_cdev->blink_delay_off; |
61 | } | 61 | } |
62 | 62 | ||
63 | led_set_brightness_async(led_cdev, brightness); | 63 | led_set_brightness_nosleep(led_cdev, brightness); |
64 | 64 | ||
65 | /* Return in next iteration if led is in one-shot mode and we are in | 65 | /* Return in next iteration if led is in one-shot mode and we are in |
66 | * the final blink state so that the led is toggled each delay_on + | 66 | * the final blink state so that the led is toggled each delay_on + |
@@ -110,13 +110,14 @@ static void led_set_software_blink(struct led_classdev *led_cdev, | |||
110 | 110 | ||
111 | /* never on - just set to off */ | 111 | /* never on - just set to off */ |
112 | if (!delay_on) { | 112 | if (!delay_on) { |
113 | led_set_brightness_async(led_cdev, LED_OFF); | 113 | led_set_brightness_nosleep(led_cdev, LED_OFF); |
114 | return; | 114 | return; |
115 | } | 115 | } |
116 | 116 | ||
117 | /* never off - just set to brightness */ | 117 | /* never off - just set to brightness */ |
118 | if (!delay_off) { | 118 | if (!delay_off) { |
119 | led_set_brightness_async(led_cdev, led_cdev->blink_brightness); | 119 | led_set_brightness_nosleep(led_cdev, |
120 | led_cdev->blink_brightness); | ||
120 | return; | 121 | return; |
121 | } | 122 | } |
122 | 123 | ||
@@ -217,7 +218,7 @@ void led_set_brightness(struct led_classdev *led_cdev, | |||
217 | } | 218 | } |
218 | 219 | ||
219 | if (led_cdev->flags & SET_BRIGHTNESS_ASYNC) { | 220 | if (led_cdev->flags & SET_BRIGHTNESS_ASYNC) { |
220 | led_set_brightness_async(led_cdev, brightness); | 221 | led_set_brightness_nosleep(led_cdev, brightness); |
221 | return; | 222 | return; |
222 | } else if (led_cdev->flags & SET_BRIGHTNESS_SYNC) | 223 | } else if (led_cdev->flags & SET_BRIGHTNESS_SYNC) |
223 | ret = led_set_brightness_sync(led_cdev, brightness); | 224 | ret = led_set_brightness_sync(led_cdev, brightness); |
@@ -230,6 +231,33 @@ void led_set_brightness(struct led_classdev *led_cdev, | |||
230 | } | 231 | } |
231 | EXPORT_SYMBOL_GPL(led_set_brightness); | 232 | EXPORT_SYMBOL_GPL(led_set_brightness); |
232 | 233 | ||
234 | void led_set_brightness_nopm(struct led_classdev *led_cdev, | ||
235 | enum led_brightness value) | ||
236 | { | ||
237 | /* Use brightness_set op if available, it is guaranteed not to sleep */ | ||
238 | if (led_cdev->brightness_set) { | ||
239 | led_cdev->brightness_set(led_cdev, value); | ||
240 | return; | ||
241 | } | ||
242 | |||
243 | /* If brightness setting can sleep, delegate it to a work queue task */ | ||
244 | led_cdev->delayed_set_value = value; | ||
245 | schedule_work(&led_cdev->set_brightness_work); | ||
246 | } | ||
247 | EXPORT_SYMBOL_GPL(led_set_brightness_nopm); | ||
248 | |||
249 | void led_set_brightness_nosleep(struct led_classdev *led_cdev, | ||
250 | enum led_brightness value) | ||
251 | { | ||
252 | led_cdev->brightness = min(value, led_cdev->max_brightness); | ||
253 | |||
254 | if (led_cdev->flags & LED_SUSPENDED) | ||
255 | return; | ||
256 | |||
257 | led_set_brightness_nopm(led_cdev, led_cdev->brightness); | ||
258 | } | ||
259 | EXPORT_SYMBOL_GPL(led_set_brightness_nosleep); | ||
260 | |||
233 | int led_update_brightness(struct led_classdev *led_cdev) | 261 | int led_update_brightness(struct led_classdev *led_cdev) |
234 | { | 262 | { |
235 | int ret = 0; | 263 | int ret = 0; |
diff --git a/drivers/leds/leds.h b/drivers/leds/leds.h index cf6d44886bcb..8e252a2f8662 100644 --- a/drivers/leds/leds.h +++ b/drivers/leds/leds.h | |||
@@ -46,6 +46,10 @@ static inline int led_get_brightness(struct led_classdev *led_cdev) | |||
46 | 46 | ||
47 | void led_init_core(struct led_classdev *led_cdev); | 47 | void led_init_core(struct led_classdev *led_cdev); |
48 | void led_stop_software_blink(struct led_classdev *led_cdev); | 48 | void led_stop_software_blink(struct led_classdev *led_cdev); |
49 | void led_set_brightness_nopm(struct led_classdev *led_cdev, | ||
50 | enum led_brightness value); | ||
51 | void led_set_brightness_nosleep(struct led_classdev *led_cdev, | ||
52 | enum led_brightness value); | ||
49 | 53 | ||
50 | extern struct rw_semaphore leds_list_lock; | 54 | extern struct rw_semaphore leds_list_lock; |
51 | extern struct list_head leds_list; | 55 | extern struct list_head leds_list; |
diff --git a/drivers/leds/trigger/ledtrig-backlight.c b/drivers/leds/trigger/ledtrig-backlight.c index 59eca17d9661..1ca1f1608f76 100644 --- a/drivers/leds/trigger/ledtrig-backlight.c +++ b/drivers/leds/trigger/ledtrig-backlight.c | |||
@@ -51,9 +51,9 @@ static int fb_notifier_callback(struct notifier_block *p, | |||
51 | 51 | ||
52 | if ((n->old_status == UNBLANK) ^ n->invert) { | 52 | if ((n->old_status == UNBLANK) ^ n->invert) { |
53 | n->brightness = led->brightness; | 53 | n->brightness = led->brightness; |
54 | led_set_brightness_async(led, LED_OFF); | 54 | led_set_brightness_nosleep(led, LED_OFF); |
55 | } else { | 55 | } else { |
56 | led_set_brightness_async(led, n->brightness); | 56 | led_set_brightness_nosleep(led, n->brightness); |
57 | } | 57 | } |
58 | 58 | ||
59 | n->old_status = new_status; | 59 | n->old_status = new_status; |
@@ -89,9 +89,9 @@ static ssize_t bl_trig_invert_store(struct device *dev, | |||
89 | 89 | ||
90 | /* After inverting, we need to update the LED. */ | 90 | /* After inverting, we need to update the LED. */ |
91 | if ((n->old_status == BLANK) ^ n->invert) | 91 | if ((n->old_status == BLANK) ^ n->invert) |
92 | led_set_brightness_async(led, LED_OFF); | 92 | led_set_brightness_nosleep(led, LED_OFF); |
93 | else | 93 | else |
94 | led_set_brightness_async(led, n->brightness); | 94 | led_set_brightness_nosleep(led, n->brightness); |
95 | 95 | ||
96 | return num; | 96 | return num; |
97 | } | 97 | } |
diff --git a/drivers/leds/trigger/ledtrig-default-on.c b/drivers/leds/trigger/ledtrig-default-on.c index 6f38f883aaf1..ff455cb46680 100644 --- a/drivers/leds/trigger/ledtrig-default-on.c +++ b/drivers/leds/trigger/ledtrig-default-on.c | |||
@@ -19,7 +19,7 @@ | |||
19 | 19 | ||
20 | static void defon_trig_activate(struct led_classdev *led_cdev) | 20 | static void defon_trig_activate(struct led_classdev *led_cdev) |
21 | { | 21 | { |
22 | led_set_brightness_async(led_cdev, led_cdev->max_brightness); | 22 | led_set_brightness_nosleep(led_cdev, led_cdev->max_brightness); |
23 | } | 23 | } |
24 | 24 | ||
25 | static struct led_trigger defon_led_trigger = { | 25 | static struct led_trigger defon_led_trigger = { |
diff --git a/drivers/leds/trigger/ledtrig-gpio.c b/drivers/leds/trigger/ledtrig-gpio.c index 4cc7040746c6..51288a45fbcb 100644 --- a/drivers/leds/trigger/ledtrig-gpio.c +++ b/drivers/leds/trigger/ledtrig-gpio.c | |||
@@ -54,12 +54,12 @@ static void gpio_trig_work(struct work_struct *work) | |||
54 | 54 | ||
55 | if (tmp) { | 55 | if (tmp) { |
56 | if (gpio_data->desired_brightness) | 56 | if (gpio_data->desired_brightness) |
57 | led_set_brightness_async(gpio_data->led, | 57 | led_set_brightness_nosleep(gpio_data->led, |
58 | gpio_data->desired_brightness); | 58 | gpio_data->desired_brightness); |
59 | else | 59 | else |
60 | led_set_brightness_async(gpio_data->led, LED_FULL); | 60 | led_set_brightness_nosleep(gpio_data->led, LED_FULL); |
61 | } else { | 61 | } else { |
62 | led_set_brightness_async(gpio_data->led, LED_OFF); | 62 | led_set_brightness_nosleep(gpio_data->led, LED_OFF); |
63 | } | 63 | } |
64 | } | 64 | } |
65 | 65 | ||
diff --git a/drivers/leds/trigger/ledtrig-heartbeat.c b/drivers/leds/trigger/ledtrig-heartbeat.c index 8622ce651ae2..410c39c62dc7 100644 --- a/drivers/leds/trigger/ledtrig-heartbeat.c +++ b/drivers/leds/trigger/ledtrig-heartbeat.c | |||
@@ -38,7 +38,7 @@ static void led_heartbeat_function(unsigned long data) | |||
38 | unsigned long delay = 0; | 38 | unsigned long delay = 0; |
39 | 39 | ||
40 | if (unlikely(panic_heartbeats)) { | 40 | if (unlikely(panic_heartbeats)) { |
41 | led_set_brightness(led_cdev, LED_OFF); | 41 | led_set_brightness_nosleep(led_cdev, LED_OFF); |
42 | return; | 42 | return; |
43 | } | 43 | } |
44 | 44 | ||
@@ -81,7 +81,7 @@ static void led_heartbeat_function(unsigned long data) | |||
81 | break; | 81 | break; |
82 | } | 82 | } |
83 | 83 | ||
84 | led_set_brightness_async(led_cdev, brightness); | 84 | led_set_brightness_nosleep(led_cdev, brightness); |
85 | mod_timer(&heartbeat_data->timer, jiffies + delay); | 85 | mod_timer(&heartbeat_data->timer, jiffies + delay); |
86 | } | 86 | } |
87 | 87 | ||
diff --git a/drivers/leds/trigger/ledtrig-oneshot.c b/drivers/leds/trigger/ledtrig-oneshot.c index fbd02cdc3ad7..67293177a07b 100644 --- a/drivers/leds/trigger/ledtrig-oneshot.c +++ b/drivers/leds/trigger/ledtrig-oneshot.c | |||
@@ -63,9 +63,9 @@ static ssize_t led_invert_store(struct device *dev, | |||
63 | oneshot_data->invert = !!state; | 63 | oneshot_data->invert = !!state; |
64 | 64 | ||
65 | if (oneshot_data->invert) | 65 | if (oneshot_data->invert) |
66 | led_set_brightness_async(led_cdev, LED_FULL); | 66 | led_set_brightness_nosleep(led_cdev, LED_FULL); |
67 | else | 67 | else |
68 | led_set_brightness_async(led_cdev, LED_OFF); | 68 | led_set_brightness_nosleep(led_cdev, LED_OFF); |
69 | 69 | ||
70 | return size; | 70 | return size; |
71 | } | 71 | } |
diff --git a/drivers/leds/trigger/ledtrig-transient.c b/drivers/leds/trigger/ledtrig-transient.c index 3c34de404d18..1dddd8f63e2a 100644 --- a/drivers/leds/trigger/ledtrig-transient.c +++ b/drivers/leds/trigger/ledtrig-transient.c | |||
@@ -41,7 +41,7 @@ static void transient_timer_function(unsigned long data) | |||
41 | struct transient_trig_data *transient_data = led_cdev->trigger_data; | 41 | struct transient_trig_data *transient_data = led_cdev->trigger_data; |
42 | 42 | ||
43 | transient_data->activate = 0; | 43 | transient_data->activate = 0; |
44 | led_set_brightness_async(led_cdev, transient_data->restore_state); | 44 | led_set_brightness_nosleep(led_cdev, transient_data->restore_state); |
45 | } | 45 | } |
46 | 46 | ||
47 | static ssize_t transient_activate_show(struct device *dev, | 47 | static ssize_t transient_activate_show(struct device *dev, |
@@ -72,7 +72,7 @@ static ssize_t transient_activate_store(struct device *dev, | |||
72 | if (state == 0 && transient_data->activate == 1) { | 72 | if (state == 0 && transient_data->activate == 1) { |
73 | del_timer(&transient_data->timer); | 73 | del_timer(&transient_data->timer); |
74 | transient_data->activate = state; | 74 | transient_data->activate = state; |
75 | led_set_brightness_async(led_cdev, | 75 | led_set_brightness_nosleep(led_cdev, |
76 | transient_data->restore_state); | 76 | transient_data->restore_state); |
77 | return size; | 77 | return size; |
78 | } | 78 | } |
@@ -81,7 +81,7 @@ static ssize_t transient_activate_store(struct device *dev, | |||
81 | if (state == 1 && transient_data->activate == 0 && | 81 | if (state == 1 && transient_data->activate == 0 && |
82 | transient_data->duration != 0) { | 82 | transient_data->duration != 0) { |
83 | transient_data->activate = state; | 83 | transient_data->activate = state; |
84 | led_set_brightness_async(led_cdev, transient_data->state); | 84 | led_set_brightness_nosleep(led_cdev, transient_data->state); |
85 | transient_data->restore_state = | 85 | transient_data->restore_state = |
86 | (transient_data->state == LED_FULL) ? LED_OFF : LED_FULL; | 86 | (transient_data->state == LED_FULL) ? LED_OFF : LED_FULL; |
87 | mod_timer(&transient_data->timer, | 87 | mod_timer(&transient_data->timer, |
@@ -204,7 +204,7 @@ static void transient_trig_deactivate(struct led_classdev *led_cdev) | |||
204 | 204 | ||
205 | if (led_cdev->activated) { | 205 | if (led_cdev->activated) { |
206 | del_timer_sync(&transient_data->timer); | 206 | del_timer_sync(&transient_data->timer); |
207 | led_set_brightness_async(led_cdev, | 207 | led_set_brightness_nosleep(led_cdev, |
208 | transient_data->restore_state); | 208 | transient_data->restore_state); |
209 | device_remove_file(led_cdev->dev, &dev_attr_activate); | 209 | device_remove_file(led_cdev->dev, &dev_attr_activate); |
210 | device_remove_file(led_cdev->dev, &dev_attr_duration); | 210 | device_remove_file(led_cdev->dev, &dev_attr_duration); |