15fd752b6SBaolin Wang // SPDX-License-Identifier: GPL-2.0
25fd752b6SBaolin Wang
35fd752b6SBaolin Wang /*
45fd752b6SBaolin Wang * LED pattern trigger
55fd752b6SBaolin Wang *
65fd752b6SBaolin Wang * Idea discussed with Pavel Machek. Raphael Teysseyre implemented
75fd752b6SBaolin Wang * the first version, Baolin Wang simplified and improved the approach.
85fd752b6SBaolin Wang */
95fd752b6SBaolin Wang
105fd752b6SBaolin Wang #include <linux/kernel.h>
115fd752b6SBaolin Wang #include <linux/leds.h>
125fd752b6SBaolin Wang #include <linux/module.h>
135fd752b6SBaolin Wang #include <linux/mutex.h>
145fd752b6SBaolin Wang #include <linux/slab.h>
155fd752b6SBaolin Wang #include <linux/timer.h>
165fd752b6SBaolin Wang
175fd752b6SBaolin Wang #define MAX_PATTERNS 1024
185fd752b6SBaolin Wang /*
195fd752b6SBaolin Wang * When doing gradual dimming, the led brightness will be updated
205fd752b6SBaolin Wang * every 50 milliseconds.
215fd752b6SBaolin Wang */
225fd752b6SBaolin Wang #define UPDATE_INTERVAL 50
235fd752b6SBaolin Wang
245fd752b6SBaolin Wang struct pattern_trig_data {
255fd752b6SBaolin Wang struct led_classdev *led_cdev;
265fd752b6SBaolin Wang struct led_pattern patterns[MAX_PATTERNS];
275fd752b6SBaolin Wang struct led_pattern *curr;
285fd752b6SBaolin Wang struct led_pattern *next;
295fd752b6SBaolin Wang struct mutex lock;
305fd752b6SBaolin Wang u32 npatterns;
315fd752b6SBaolin Wang int repeat;
325fd752b6SBaolin Wang int last_repeat;
335fd752b6SBaolin Wang int delta_t;
345fd752b6SBaolin Wang bool is_indefinite;
355fd752b6SBaolin Wang bool is_hw_pattern;
365fd752b6SBaolin Wang struct timer_list timer;
375fd752b6SBaolin Wang };
385fd752b6SBaolin Wang
pattern_trig_update_patterns(struct pattern_trig_data * data)395fd752b6SBaolin Wang static void pattern_trig_update_patterns(struct pattern_trig_data *data)
405fd752b6SBaolin Wang {
415fd752b6SBaolin Wang data->curr = data->next;
425fd752b6SBaolin Wang if (!data->is_indefinite && data->curr == data->patterns)
435fd752b6SBaolin Wang data->repeat--;
445fd752b6SBaolin Wang
455fd752b6SBaolin Wang if (data->next == data->patterns + data->npatterns - 1)
465fd752b6SBaolin Wang data->next = data->patterns;
475fd752b6SBaolin Wang else
485fd752b6SBaolin Wang data->next++;
495fd752b6SBaolin Wang
505fd752b6SBaolin Wang data->delta_t = 0;
515fd752b6SBaolin Wang }
525fd752b6SBaolin Wang
pattern_trig_compute_brightness(struct pattern_trig_data * data)535fd752b6SBaolin Wang static int pattern_trig_compute_brightness(struct pattern_trig_data *data)
545fd752b6SBaolin Wang {
555fd752b6SBaolin Wang int step_brightness;
565fd752b6SBaolin Wang
575fd752b6SBaolin Wang /*
585fd752b6SBaolin Wang * If current tuple's duration is less than the dimming interval,
595fd752b6SBaolin Wang * we should treat it as a step change of brightness instead of
605fd752b6SBaolin Wang * doing gradual dimming.
615fd752b6SBaolin Wang */
625fd752b6SBaolin Wang if (data->delta_t == 0 || data->curr->delta_t < UPDATE_INTERVAL)
635fd752b6SBaolin Wang return data->curr->brightness;
645fd752b6SBaolin Wang
655fd752b6SBaolin Wang step_brightness = abs(data->next->brightness - data->curr->brightness);
665fd752b6SBaolin Wang step_brightness = data->delta_t * step_brightness / data->curr->delta_t;
675fd752b6SBaolin Wang
685fd752b6SBaolin Wang if (data->next->brightness > data->curr->brightness)
695fd752b6SBaolin Wang return data->curr->brightness + step_brightness;
705fd752b6SBaolin Wang else
715fd752b6SBaolin Wang return data->curr->brightness - step_brightness;
725fd752b6SBaolin Wang }
735fd752b6SBaolin Wang
pattern_trig_timer_function(struct timer_list * t)745fd752b6SBaolin Wang static void pattern_trig_timer_function(struct timer_list *t)
755fd752b6SBaolin Wang {
765fd752b6SBaolin Wang struct pattern_trig_data *data = from_timer(data, t, timer);
775fd752b6SBaolin Wang
785fd752b6SBaolin Wang for (;;) {
795fd752b6SBaolin Wang if (!data->is_indefinite && !data->repeat)
805fd752b6SBaolin Wang break;
815fd752b6SBaolin Wang
825fd752b6SBaolin Wang if (data->curr->brightness == data->next->brightness) {
835fd752b6SBaolin Wang /* Step change of brightness */
845fd752b6SBaolin Wang led_set_brightness(data->led_cdev,
855fd752b6SBaolin Wang data->curr->brightness);
865fd752b6SBaolin Wang mod_timer(&data->timer,
875fd752b6SBaolin Wang jiffies + msecs_to_jiffies(data->curr->delta_t));
881b50bb4dSPavel Machek if (!data->next->delta_t) {
895fd752b6SBaolin Wang /* Skip the tuple with zero duration */
905fd752b6SBaolin Wang pattern_trig_update_patterns(data);
911b50bb4dSPavel Machek }
925fd752b6SBaolin Wang /* Select next tuple */
935fd752b6SBaolin Wang pattern_trig_update_patterns(data);
945fd752b6SBaolin Wang } else {
955fd752b6SBaolin Wang /* Gradual dimming */
965fd752b6SBaolin Wang
975fd752b6SBaolin Wang /*
985fd752b6SBaolin Wang * If the accumulation time is larger than current
995fd752b6SBaolin Wang * tuple's duration, we should go next one and re-check
1005fd752b6SBaolin Wang * if we repeated done.
1015fd752b6SBaolin Wang */
1025fd752b6SBaolin Wang if (data->delta_t > data->curr->delta_t) {
1035fd752b6SBaolin Wang pattern_trig_update_patterns(data);
1045fd752b6SBaolin Wang continue;
1055fd752b6SBaolin Wang }
1065fd752b6SBaolin Wang
1075fd752b6SBaolin Wang led_set_brightness(data->led_cdev,
1085fd752b6SBaolin Wang pattern_trig_compute_brightness(data));
1095fd752b6SBaolin Wang mod_timer(&data->timer,
1105fd752b6SBaolin Wang jiffies + msecs_to_jiffies(UPDATE_INTERVAL));
1115fd752b6SBaolin Wang
1125fd752b6SBaolin Wang /* Accumulate the gradual dimming time */
1135fd752b6SBaolin Wang data->delta_t += UPDATE_INTERVAL;
1145fd752b6SBaolin Wang }
1155fd752b6SBaolin Wang
1165fd752b6SBaolin Wang break;
1175fd752b6SBaolin Wang }
1185fd752b6SBaolin Wang }
1195fd752b6SBaolin Wang
pattern_trig_start_pattern(struct led_classdev * led_cdev)1205fd752b6SBaolin Wang static int pattern_trig_start_pattern(struct led_classdev *led_cdev)
1215fd752b6SBaolin Wang {
1225fd752b6SBaolin Wang struct pattern_trig_data *data = led_cdev->trigger_data;
1235fd752b6SBaolin Wang
1245fd752b6SBaolin Wang if (!data->npatterns)
1255fd752b6SBaolin Wang return 0;
1265fd752b6SBaolin Wang
1275fd752b6SBaolin Wang if (data->is_hw_pattern) {
1285fd752b6SBaolin Wang return led_cdev->pattern_set(led_cdev, data->patterns,
1295fd752b6SBaolin Wang data->npatterns, data->repeat);
1305fd752b6SBaolin Wang }
1315fd752b6SBaolin Wang
1325fd752b6SBaolin Wang /* At least 2 tuples for software pattern. */
1335fd752b6SBaolin Wang if (data->npatterns < 2)
1345fd752b6SBaolin Wang return -EINVAL;
1355fd752b6SBaolin Wang
1365fd752b6SBaolin Wang data->delta_t = 0;
1375fd752b6SBaolin Wang data->curr = data->patterns;
1385fd752b6SBaolin Wang data->next = data->patterns + 1;
1395fd752b6SBaolin Wang data->timer.expires = jiffies;
1405fd752b6SBaolin Wang add_timer(&data->timer);
1415fd752b6SBaolin Wang
1425fd752b6SBaolin Wang return 0;
1435fd752b6SBaolin Wang }
1445fd752b6SBaolin Wang
repeat_show(struct device * dev,struct device_attribute * attr,char * buf)1455fd752b6SBaolin Wang static ssize_t repeat_show(struct device *dev, struct device_attribute *attr,
1465fd752b6SBaolin Wang char *buf)
1475fd752b6SBaolin Wang {
1485fd752b6SBaolin Wang struct led_classdev *led_cdev = dev_get_drvdata(dev);
1495fd752b6SBaolin Wang struct pattern_trig_data *data = led_cdev->trigger_data;
1505fd752b6SBaolin Wang int repeat;
1515fd752b6SBaolin Wang
1525fd752b6SBaolin Wang mutex_lock(&data->lock);
1535fd752b6SBaolin Wang
1545fd752b6SBaolin Wang repeat = data->last_repeat;
1555fd752b6SBaolin Wang
1565fd752b6SBaolin Wang mutex_unlock(&data->lock);
1575fd752b6SBaolin Wang
1583f6fb1cfSye xingchen return sysfs_emit(buf, "%d\n", repeat);
1595fd752b6SBaolin Wang }
1605fd752b6SBaolin Wang
repeat_store(struct device * dev,struct device_attribute * attr,const char * buf,size_t count)1615fd752b6SBaolin Wang static ssize_t repeat_store(struct device *dev, struct device_attribute *attr,
1625fd752b6SBaolin Wang const char *buf, size_t count)
1635fd752b6SBaolin Wang {
1645fd752b6SBaolin Wang struct led_classdev *led_cdev = dev_get_drvdata(dev);
1655fd752b6SBaolin Wang struct pattern_trig_data *data = led_cdev->trigger_data;
1665fd752b6SBaolin Wang int err, res;
1675fd752b6SBaolin Wang
1685fd752b6SBaolin Wang err = kstrtos32(buf, 10, &res);
1695fd752b6SBaolin Wang if (err)
1705fd752b6SBaolin Wang return err;
1715fd752b6SBaolin Wang
1725fd752b6SBaolin Wang /* Number 0 and negative numbers except -1 are invalid. */
1735fd752b6SBaolin Wang if (res < -1 || res == 0)
1745fd752b6SBaolin Wang return -EINVAL;
1755fd752b6SBaolin Wang
1765fd752b6SBaolin Wang mutex_lock(&data->lock);
1775fd752b6SBaolin Wang
1783a40cfe8SBaolin Wang del_timer_sync(&data->timer);
1793a40cfe8SBaolin Wang
1805fd752b6SBaolin Wang if (data->is_hw_pattern)
1815fd752b6SBaolin Wang led_cdev->pattern_clear(led_cdev);
1825fd752b6SBaolin Wang
1835fd752b6SBaolin Wang data->last_repeat = data->repeat = res;
1845fd752b6SBaolin Wang /* -1 means repeat indefinitely */
1855fd752b6SBaolin Wang if (data->repeat == -1)
1865fd752b6SBaolin Wang data->is_indefinite = true;
1875fd752b6SBaolin Wang else
1885fd752b6SBaolin Wang data->is_indefinite = false;
1895fd752b6SBaolin Wang
1905fd752b6SBaolin Wang err = pattern_trig_start_pattern(led_cdev);
1915fd752b6SBaolin Wang
1925fd752b6SBaolin Wang mutex_unlock(&data->lock);
1935fd752b6SBaolin Wang return err < 0 ? err : count;
1945fd752b6SBaolin Wang }
1955fd752b6SBaolin Wang
1965fd752b6SBaolin Wang static DEVICE_ATTR_RW(repeat);
1975fd752b6SBaolin Wang
pattern_trig_show_patterns(struct pattern_trig_data * data,char * buf,bool hw_pattern)1985fd752b6SBaolin Wang static ssize_t pattern_trig_show_patterns(struct pattern_trig_data *data,
1995fd752b6SBaolin Wang char *buf, bool hw_pattern)
2005fd752b6SBaolin Wang {
2015fd752b6SBaolin Wang ssize_t count = 0;
2025fd752b6SBaolin Wang int i;
2035fd752b6SBaolin Wang
2045fd752b6SBaolin Wang mutex_lock(&data->lock);
2055fd752b6SBaolin Wang
2065fd752b6SBaolin Wang if (!data->npatterns || (data->is_hw_pattern ^ hw_pattern))
2075fd752b6SBaolin Wang goto out;
2085fd752b6SBaolin Wang
2095fd752b6SBaolin Wang for (i = 0; i < data->npatterns; i++) {
2105fd752b6SBaolin Wang count += scnprintf(buf + count, PAGE_SIZE - count,
2115fd752b6SBaolin Wang "%d %u ",
2125fd752b6SBaolin Wang data->patterns[i].brightness,
2135fd752b6SBaolin Wang data->patterns[i].delta_t);
2145fd752b6SBaolin Wang }
2155fd752b6SBaolin Wang
2165fd752b6SBaolin Wang buf[count - 1] = '\n';
2175fd752b6SBaolin Wang
2185fd752b6SBaolin Wang out:
2195fd752b6SBaolin Wang mutex_unlock(&data->lock);
2205fd752b6SBaolin Wang return count;
2215fd752b6SBaolin Wang }
2225fd752b6SBaolin Wang
pattern_trig_store_patterns_string(struct pattern_trig_data * data,const char * buf,size_t count)223aa6fd104SKrzysztof Kozlowski static int pattern_trig_store_patterns_string(struct pattern_trig_data *data,
224aa6fd104SKrzysztof Kozlowski const char *buf, size_t count)
225aa6fd104SKrzysztof Kozlowski {
226aa6fd104SKrzysztof Kozlowski int ccount, cr, offset = 0;
227aa6fd104SKrzysztof Kozlowski
228aa6fd104SKrzysztof Kozlowski while (offset < count - 1 && data->npatterns < MAX_PATTERNS) {
229aa6fd104SKrzysztof Kozlowski cr = 0;
230feff7273SPavel Machek ccount = sscanf(buf + offset, "%u %u %n",
231aa6fd104SKrzysztof Kozlowski &data->patterns[data->npatterns].brightness,
232aa6fd104SKrzysztof Kozlowski &data->patterns[data->npatterns].delta_t, &cr);
233feff7273SPavel Machek
234feff7273SPavel Machek if (ccount != 2 ||
235feff7273SPavel Machek data->patterns[data->npatterns].brightness > data->led_cdev->max_brightness) {
236aa6fd104SKrzysztof Kozlowski data->npatterns = 0;
237aa6fd104SKrzysztof Kozlowski return -EINVAL;
238aa6fd104SKrzysztof Kozlowski }
239aa6fd104SKrzysztof Kozlowski
240aa6fd104SKrzysztof Kozlowski offset += cr;
241aa6fd104SKrzysztof Kozlowski data->npatterns++;
242aa6fd104SKrzysztof Kozlowski }
243aa6fd104SKrzysztof Kozlowski
244aa6fd104SKrzysztof Kozlowski return 0;
245aa6fd104SKrzysztof Kozlowski }
246aa6fd104SKrzysztof Kozlowski
pattern_trig_store_patterns_int(struct pattern_trig_data * data,const u32 * buf,size_t count)247aa6fd104SKrzysztof Kozlowski static int pattern_trig_store_patterns_int(struct pattern_trig_data *data,
248aa6fd104SKrzysztof Kozlowski const u32 *buf, size_t count)
249aa6fd104SKrzysztof Kozlowski {
250aa6fd104SKrzysztof Kozlowski unsigned int i;
251aa6fd104SKrzysztof Kozlowski
252aa6fd104SKrzysztof Kozlowski for (i = 0; i < count; i += 2) {
253aa6fd104SKrzysztof Kozlowski data->patterns[data->npatterns].brightness = buf[i];
254aa6fd104SKrzysztof Kozlowski data->patterns[data->npatterns].delta_t = buf[i + 1];
255aa6fd104SKrzysztof Kozlowski data->npatterns++;
256aa6fd104SKrzysztof Kozlowski }
257aa6fd104SKrzysztof Kozlowski
258aa6fd104SKrzysztof Kozlowski return 0;
259aa6fd104SKrzysztof Kozlowski }
260aa6fd104SKrzysztof Kozlowski
pattern_trig_store_patterns(struct led_classdev * led_cdev,const char * buf,const u32 * buf_int,size_t count,bool hw_pattern)2615fd752b6SBaolin Wang static ssize_t pattern_trig_store_patterns(struct led_classdev *led_cdev,
262aa6fd104SKrzysztof Kozlowski const char *buf, const u32 *buf_int,
263aa6fd104SKrzysztof Kozlowski size_t count, bool hw_pattern)
2645fd752b6SBaolin Wang {
2655fd752b6SBaolin Wang struct pattern_trig_data *data = led_cdev->trigger_data;
266aa6fd104SKrzysztof Kozlowski int err = 0;
2675fd752b6SBaolin Wang
2685fd752b6SBaolin Wang mutex_lock(&data->lock);
2695fd752b6SBaolin Wang
2703a40cfe8SBaolin Wang del_timer_sync(&data->timer);
2713a40cfe8SBaolin Wang
2725fd752b6SBaolin Wang if (data->is_hw_pattern)
2735fd752b6SBaolin Wang led_cdev->pattern_clear(led_cdev);
2745fd752b6SBaolin Wang
2755fd752b6SBaolin Wang data->is_hw_pattern = hw_pattern;
2765fd752b6SBaolin Wang data->npatterns = 0;
2775fd752b6SBaolin Wang
278aa6fd104SKrzysztof Kozlowski if (buf)
279aa6fd104SKrzysztof Kozlowski err = pattern_trig_store_patterns_string(data, buf, count);
280aa6fd104SKrzysztof Kozlowski else
281aa6fd104SKrzysztof Kozlowski err = pattern_trig_store_patterns_int(data, buf_int, count);
282aa6fd104SKrzysztof Kozlowski if (err)
2835fd752b6SBaolin Wang goto out;
2845fd752b6SBaolin Wang
2855fd752b6SBaolin Wang err = pattern_trig_start_pattern(led_cdev);
2865fd752b6SBaolin Wang if (err)
2875fd752b6SBaolin Wang data->npatterns = 0;
2885fd752b6SBaolin Wang
2895fd752b6SBaolin Wang out:
2905fd752b6SBaolin Wang mutex_unlock(&data->lock);
2915fd752b6SBaolin Wang return err < 0 ? err : count;
2925fd752b6SBaolin Wang }
2935fd752b6SBaolin Wang
pattern_show(struct device * dev,struct device_attribute * attr,char * buf)2945fd752b6SBaolin Wang static ssize_t pattern_show(struct device *dev, struct device_attribute *attr,
2955fd752b6SBaolin Wang char *buf)
2965fd752b6SBaolin Wang {
2975fd752b6SBaolin Wang struct led_classdev *led_cdev = dev_get_drvdata(dev);
2985fd752b6SBaolin Wang struct pattern_trig_data *data = led_cdev->trigger_data;
2995fd752b6SBaolin Wang
3005fd752b6SBaolin Wang return pattern_trig_show_patterns(data, buf, false);
3015fd752b6SBaolin Wang }
3025fd752b6SBaolin Wang
pattern_store(struct device * dev,struct device_attribute * attr,const char * buf,size_t count)3035fd752b6SBaolin Wang static ssize_t pattern_store(struct device *dev, struct device_attribute *attr,
3045fd752b6SBaolin Wang const char *buf, size_t count)
3055fd752b6SBaolin Wang {
3065fd752b6SBaolin Wang struct led_classdev *led_cdev = dev_get_drvdata(dev);
3075fd752b6SBaolin Wang
308aa6fd104SKrzysztof Kozlowski return pattern_trig_store_patterns(led_cdev, buf, NULL, count, false);
3095fd752b6SBaolin Wang }
3105fd752b6SBaolin Wang
3115fd752b6SBaolin Wang static DEVICE_ATTR_RW(pattern);
3125fd752b6SBaolin Wang
hw_pattern_show(struct device * dev,struct device_attribute * attr,char * buf)3135fd752b6SBaolin Wang static ssize_t hw_pattern_show(struct device *dev,
3145fd752b6SBaolin Wang struct device_attribute *attr, char *buf)
3155fd752b6SBaolin Wang {
3165fd752b6SBaolin Wang struct led_classdev *led_cdev = dev_get_drvdata(dev);
3175fd752b6SBaolin Wang struct pattern_trig_data *data = led_cdev->trigger_data;
3185fd752b6SBaolin Wang
3195fd752b6SBaolin Wang return pattern_trig_show_patterns(data, buf, true);
3205fd752b6SBaolin Wang }
3215fd752b6SBaolin Wang
hw_pattern_store(struct device * dev,struct device_attribute * attr,const char * buf,size_t count)3225fd752b6SBaolin Wang static ssize_t hw_pattern_store(struct device *dev,
3235fd752b6SBaolin Wang struct device_attribute *attr,
3245fd752b6SBaolin Wang const char *buf, size_t count)
3255fd752b6SBaolin Wang {
3265fd752b6SBaolin Wang struct led_classdev *led_cdev = dev_get_drvdata(dev);
3275fd752b6SBaolin Wang
328aa6fd104SKrzysztof Kozlowski return pattern_trig_store_patterns(led_cdev, buf, NULL, count, true);
3295fd752b6SBaolin Wang }
3305fd752b6SBaolin Wang
3315fd752b6SBaolin Wang static DEVICE_ATTR_RW(hw_pattern);
3325fd752b6SBaolin Wang
pattern_trig_attrs_mode(struct kobject * kobj,struct attribute * attr,int index)3335fd752b6SBaolin Wang static umode_t pattern_trig_attrs_mode(struct kobject *kobj,
3345fd752b6SBaolin Wang struct attribute *attr, int index)
3355fd752b6SBaolin Wang {
3365fe09e16STian Tao struct device *dev = kobj_to_dev(kobj);
3375fd752b6SBaolin Wang struct led_classdev *led_cdev = dev_get_drvdata(dev);
3385fd752b6SBaolin Wang
3395fd752b6SBaolin Wang if (attr == &dev_attr_repeat.attr || attr == &dev_attr_pattern.attr)
3405fd752b6SBaolin Wang return attr->mode;
3415fd752b6SBaolin Wang else if (attr == &dev_attr_hw_pattern.attr && led_cdev->pattern_set)
3425fd752b6SBaolin Wang return attr->mode;
3435fd752b6SBaolin Wang
3445fd752b6SBaolin Wang return 0;
3455fd752b6SBaolin Wang }
3465fd752b6SBaolin Wang
3475fd752b6SBaolin Wang static struct attribute *pattern_trig_attrs[] = {
3485fd752b6SBaolin Wang &dev_attr_pattern.attr,
3495fd752b6SBaolin Wang &dev_attr_hw_pattern.attr,
3505fd752b6SBaolin Wang &dev_attr_repeat.attr,
3515fd752b6SBaolin Wang NULL
3525fd752b6SBaolin Wang };
3535fd752b6SBaolin Wang
3545fd752b6SBaolin Wang static const struct attribute_group pattern_trig_group = {
3555fd752b6SBaolin Wang .attrs = pattern_trig_attrs,
3565fd752b6SBaolin Wang .is_visible = pattern_trig_attrs_mode,
3575fd752b6SBaolin Wang };
3585fd752b6SBaolin Wang
3595fd752b6SBaolin Wang static const struct attribute_group *pattern_trig_groups[] = {
3605fd752b6SBaolin Wang &pattern_trig_group,
3615fd752b6SBaolin Wang NULL,
3625fd752b6SBaolin Wang };
3635fd752b6SBaolin Wang
pattern_init(struct led_classdev * led_cdev)364aa6fd104SKrzysztof Kozlowski static void pattern_init(struct led_classdev *led_cdev)
365aa6fd104SKrzysztof Kozlowski {
366aa6fd104SKrzysztof Kozlowski unsigned int size = 0;
367aa6fd104SKrzysztof Kozlowski u32 *pattern;
368aa6fd104SKrzysztof Kozlowski int err;
369aa6fd104SKrzysztof Kozlowski
370aa6fd104SKrzysztof Kozlowski pattern = led_get_default_pattern(led_cdev, &size);
371aa6fd104SKrzysztof Kozlowski if (!pattern)
372aa6fd104SKrzysztof Kozlowski return;
373aa6fd104SKrzysztof Kozlowski
374aa6fd104SKrzysztof Kozlowski if (size % 2) {
375aa6fd104SKrzysztof Kozlowski dev_warn(led_cdev->dev, "Expected pattern of tuples\n");
376aa6fd104SKrzysztof Kozlowski goto out;
377aa6fd104SKrzysztof Kozlowski }
378aa6fd104SKrzysztof Kozlowski
379aa6fd104SKrzysztof Kozlowski err = pattern_trig_store_patterns(led_cdev, NULL, pattern, size, false);
380aa6fd104SKrzysztof Kozlowski if (err < 0)
381aa6fd104SKrzysztof Kozlowski dev_warn(led_cdev->dev,
382aa6fd104SKrzysztof Kozlowski "Pattern initialization failed with error %d\n", err);
383aa6fd104SKrzysztof Kozlowski
384aa6fd104SKrzysztof Kozlowski out:
385aa6fd104SKrzysztof Kozlowski kfree(pattern);
386aa6fd104SKrzysztof Kozlowski }
387aa6fd104SKrzysztof Kozlowski
pattern_trig_activate(struct led_classdev * led_cdev)3885fd752b6SBaolin Wang static int pattern_trig_activate(struct led_classdev *led_cdev)
3895fd752b6SBaolin Wang {
3905fd752b6SBaolin Wang struct pattern_trig_data *data;
3915fd752b6SBaolin Wang
3925fd752b6SBaolin Wang data = kzalloc(sizeof(*data), GFP_KERNEL);
3935fd752b6SBaolin Wang if (!data)
3945fd752b6SBaolin Wang return -ENOMEM;
3955fd752b6SBaolin Wang
3965fd752b6SBaolin Wang if (!!led_cdev->pattern_set ^ !!led_cdev->pattern_clear) {
3975fd752b6SBaolin Wang dev_warn(led_cdev->dev,
3985fd752b6SBaolin Wang "Hardware pattern ops validation failed\n");
3995fd752b6SBaolin Wang led_cdev->pattern_set = NULL;
4005fd752b6SBaolin Wang led_cdev->pattern_clear = NULL;
4015fd752b6SBaolin Wang }
4025fd752b6SBaolin Wang
4035fd752b6SBaolin Wang data->is_indefinite = true;
4045fd752b6SBaolin Wang data->last_repeat = -1;
4055fd752b6SBaolin Wang mutex_init(&data->lock);
4065fd752b6SBaolin Wang data->led_cdev = led_cdev;
4075fd752b6SBaolin Wang led_set_trigger_data(led_cdev, data);
4085fd752b6SBaolin Wang timer_setup(&data->timer, pattern_trig_timer_function, 0);
4095fd752b6SBaolin Wang led_cdev->activated = true;
4105fd752b6SBaolin Wang
411aa6fd104SKrzysztof Kozlowski if (led_cdev->flags & LED_INIT_DEFAULT_TRIGGER) {
412aa6fd104SKrzysztof Kozlowski pattern_init(led_cdev);
413aa6fd104SKrzysztof Kozlowski /*
414aa6fd104SKrzysztof Kozlowski * Mark as initialized even on pattern_init() error because
415aa6fd104SKrzysztof Kozlowski * any consecutive call to it would produce the same error.
416aa6fd104SKrzysztof Kozlowski */
417aa6fd104SKrzysztof Kozlowski led_cdev->flags &= ~LED_INIT_DEFAULT_TRIGGER;
418aa6fd104SKrzysztof Kozlowski }
419aa6fd104SKrzysztof Kozlowski
4205fd752b6SBaolin Wang return 0;
4215fd752b6SBaolin Wang }
4225fd752b6SBaolin Wang
pattern_trig_deactivate(struct led_classdev * led_cdev)4235fd752b6SBaolin Wang static void pattern_trig_deactivate(struct led_classdev *led_cdev)
4245fd752b6SBaolin Wang {
4255fd752b6SBaolin Wang struct pattern_trig_data *data = led_cdev->trigger_data;
4265fd752b6SBaolin Wang
4275fd752b6SBaolin Wang if (!led_cdev->activated)
4285fd752b6SBaolin Wang return;
4295fd752b6SBaolin Wang
4305fd752b6SBaolin Wang if (led_cdev->pattern_clear)
4315fd752b6SBaolin Wang led_cdev->pattern_clear(led_cdev);
4325fd752b6SBaolin Wang
433*292a089dSSteven Rostedt (Google) timer_shutdown_sync(&data->timer);
4345fd752b6SBaolin Wang
4355fd752b6SBaolin Wang led_set_brightness(led_cdev, LED_OFF);
4365fd752b6SBaolin Wang kfree(data);
4375fd752b6SBaolin Wang led_cdev->activated = false;
4385fd752b6SBaolin Wang }
4395fd752b6SBaolin Wang
4405fd752b6SBaolin Wang static struct led_trigger pattern_led_trigger = {
4415fd752b6SBaolin Wang .name = "pattern",
4425fd752b6SBaolin Wang .activate = pattern_trig_activate,
4435fd752b6SBaolin Wang .deactivate = pattern_trig_deactivate,
4445fd752b6SBaolin Wang .groups = pattern_trig_groups,
4455fd752b6SBaolin Wang };
4465fd752b6SBaolin Wang
pattern_trig_init(void)4475fd752b6SBaolin Wang static int __init pattern_trig_init(void)
4485fd752b6SBaolin Wang {
4495fd752b6SBaolin Wang return led_trigger_register(&pattern_led_trigger);
4505fd752b6SBaolin Wang }
4515fd752b6SBaolin Wang
pattern_trig_exit(void)4525fd752b6SBaolin Wang static void __exit pattern_trig_exit(void)
4535fd752b6SBaolin Wang {
4545fd752b6SBaolin Wang led_trigger_unregister(&pattern_led_trigger);
4555fd752b6SBaolin Wang }
4565fd752b6SBaolin Wang
4575fd752b6SBaolin Wang module_init(pattern_trig_init);
4585fd752b6SBaolin Wang module_exit(pattern_trig_exit);
4595fd752b6SBaolin Wang
46030d57d55SPavel Machek MODULE_AUTHOR("Raphael Teysseyre <rteysseyre@gmail.com>");
46130d57d55SPavel Machek MODULE_AUTHOR("Baolin Wang <baolin.wang@linaro.org>");
4625fd752b6SBaolin Wang MODULE_DESCRIPTION("LED Pattern trigger");
4635fd752b6SBaolin Wang MODULE_LICENSE("GPL v2");
464