blob: 7c500dfdcfa3cac6e15b78660c0bc3c85691a66e [file] [log] [blame]
Thomas Gleixner1a59d1b82019-05-27 08:55:05 +02001// SPDX-License-Identifier: GPL-2.0-or-later
Simon Guinot11efe712010-07-06 16:08:46 +02002/*
3 * leds-ns2.c - Driver for the Network Space v2 (and parents) dual-GPIO LED
4 *
5 * Copyright (C) 2010 LaCie
6 *
7 * Author: Simon Guinot <sguinot@lacie.com>
8 *
9 * Based on leds-gpio.c by Raphael Assenat <raph@8d.com>
Simon Guinot11efe712010-07-06 16:08:46 +020010 */
11
12#include <linux/kernel.h>
Simon Guinot11efe712010-07-06 16:08:46 +020013#include <linux/platform_device.h>
14#include <linux/slab.h>
15#include <linux/gpio.h>
16#include <linux/leds.h>
Paul Gortmaker54f4ded2011-07-03 13:56:03 -040017#include <linux/module.h>
Arnd Bergmannc02cecb2012-08-24 15:21:54 +020018#include <linux/platform_data/leds-kirkwood-ns2.h>
Sachin Kamatc68f46d2013-09-28 04:38:30 -070019#include <linux/of.h>
Simon Guinot72052fc2012-10-17 12:09:03 +020020#include <linux/of_gpio.h>
Simon Guinot4b904322015-07-02 19:56:42 +020021#include "leds.h"
Simon Guinot11efe712010-07-06 16:08:46 +020022
23/*
Vincent Donnefortf7fafd02015-07-02 19:56:40 +020024 * The Network Space v2 dual-GPIO LED is wired to a CPLD. Three different LED
25 * modes are available: off, on and SATA activity blinking. The LED modes are
26 * controlled through two GPIOs (command and slow): each combination of values
27 * for the command/slow GPIOs corresponds to a LED mode.
Simon Guinot11efe712010-07-06 16:08:46 +020028 */
29
Simon Guinot11efe712010-07-06 16:08:46 +020030struct ns2_led_data {
31 struct led_classdev cdev;
Kitone Elvis Peter2224f2f2018-08-06 20:27:59 +030032 unsigned int cmd;
33 unsigned int slow;
Simon Guinot4b904322015-07-02 19:56:42 +020034 bool can_sleep;
Simon Guinot11efe712010-07-06 16:08:46 +020035 unsigned char sata; /* True when SATA mode active. */
36 rwlock_t rw_lock; /* Lock GPIOs. */
Vincent Donnefortf7fafd02015-07-02 19:56:40 +020037 int num_modes;
38 struct ns2_led_modval *modval;
Simon Guinot11efe712010-07-06 16:08:46 +020039};
40
41static int ns2_led_get_mode(struct ns2_led_data *led_dat,
42 enum ns2_led_modes *mode)
43{
44 int i;
45 int ret = -EINVAL;
46 int cmd_level;
47 int slow_level;
48
Simon Guinot4b904322015-07-02 19:56:42 +020049 cmd_level = gpio_get_value_cansleep(led_dat->cmd);
50 slow_level = gpio_get_value_cansleep(led_dat->slow);
Simon Guinot11efe712010-07-06 16:08:46 +020051
Vincent Donnefortf7fafd02015-07-02 19:56:40 +020052 for (i = 0; i < led_dat->num_modes; i++) {
53 if (cmd_level == led_dat->modval[i].cmd_level &&
54 slow_level == led_dat->modval[i].slow_level) {
55 *mode = led_dat->modval[i].mode;
Simon Guinot11efe712010-07-06 16:08:46 +020056 ret = 0;
57 break;
58 }
59 }
60
Simon Guinot11efe712010-07-06 16:08:46 +020061 return ret;
62}
63
64static void ns2_led_set_mode(struct ns2_led_data *led_dat,
65 enum ns2_led_modes mode)
66{
67 int i;
Simon Guinot4b904322015-07-02 19:56:42 +020068 bool found = false;
Simon Guinotf539dfe2010-09-19 15:30:59 +020069 unsigned long flags;
Simon Guinot11efe712010-07-06 16:08:46 +020070
Simon Guinot4b904322015-07-02 19:56:42 +020071 for (i = 0; i < led_dat->num_modes; i++)
72 if (mode == led_dat->modval[i].mode) {
73 found = true;
74 break;
75 }
76
77 if (!found)
78 return;
79
Simon Guinotf539dfe2010-09-19 15:30:59 +020080 write_lock_irqsave(&led_dat->rw_lock, flags);
Simon Guinot11efe712010-07-06 16:08:46 +020081
Simon Guinot4b904322015-07-02 19:56:42 +020082 if (!led_dat->can_sleep) {
83 gpio_set_value(led_dat->cmd,
84 led_dat->modval[i].cmd_level);
85 gpio_set_value(led_dat->slow,
86 led_dat->modval[i].slow_level);
87 goto exit_unlock;
Simon Guinot11efe712010-07-06 16:08:46 +020088 }
89
Jacek Anaszewskic29e6502015-11-20 11:39:41 +010090 gpio_set_value_cansleep(led_dat->cmd, led_dat->modval[i].cmd_level);
91 gpio_set_value_cansleep(led_dat->slow, led_dat->modval[i].slow_level);
Simon Guinot4b904322015-07-02 19:56:42 +020092
93exit_unlock:
Simon Guinotf539dfe2010-09-19 15:30:59 +020094 write_unlock_irqrestore(&led_dat->rw_lock, flags);
Simon Guinot11efe712010-07-06 16:08:46 +020095}
96
97static void ns2_led_set(struct led_classdev *led_cdev,
98 enum led_brightness value)
99{
100 struct ns2_led_data *led_dat =
101 container_of(led_cdev, struct ns2_led_data, cdev);
102 enum ns2_led_modes mode;
103
104 if (value == LED_OFF)
105 mode = NS_V2_LED_OFF;
106 else if (led_dat->sata)
107 mode = NS_V2_LED_SATA;
108 else
109 mode = NS_V2_LED_ON;
110
111 ns2_led_set_mode(led_dat, mode);
112}
113
Jacek Anaszewskic29e6502015-11-20 11:39:41 +0100114static int ns2_led_set_blocking(struct led_classdev *led_cdev,
115 enum led_brightness value)
116{
117 ns2_led_set(led_cdev, value);
118 return 0;
119}
120
Simon Guinot11efe712010-07-06 16:08:46 +0200121static ssize_t ns2_led_sata_store(struct device *dev,
122 struct device_attribute *attr,
123 const char *buff, size_t count)
124{
Simon Guinote5971bb2010-10-07 16:35:40 +0200125 struct led_classdev *led_cdev = dev_get_drvdata(dev);
126 struct ns2_led_data *led_dat =
127 container_of(led_cdev, struct ns2_led_data, cdev);
Simon Guinot11efe712010-07-06 16:08:46 +0200128 int ret;
129 unsigned long enable;
Simon Guinot11efe712010-07-06 16:08:46 +0200130
Jingoo Han38743502012-10-23 05:25:35 -0700131 ret = kstrtoul(buff, 10, &enable);
Simon Guinot11efe712010-07-06 16:08:46 +0200132 if (ret < 0)
133 return ret;
134
135 enable = !!enable;
136
137 if (led_dat->sata == enable)
Simon Guinot4b904322015-07-02 19:56:42 +0200138 goto exit;
Simon Guinot11efe712010-07-06 16:08:46 +0200139
140 led_dat->sata = enable;
141
Simon Guinot4b904322015-07-02 19:56:42 +0200142 if (!led_get_brightness(led_cdev))
143 goto exit;
144
145 if (enable)
146 ns2_led_set_mode(led_dat, NS_V2_LED_SATA);
147 else
148 ns2_led_set_mode(led_dat, NS_V2_LED_ON);
149
150exit:
Simon Guinot11efe712010-07-06 16:08:46 +0200151 return count;
152}
153
154static ssize_t ns2_led_sata_show(struct device *dev,
155 struct device_attribute *attr, char *buf)
156{
Simon Guinote5971bb2010-10-07 16:35:40 +0200157 struct led_classdev *led_cdev = dev_get_drvdata(dev);
158 struct ns2_led_data *led_dat =
159 container_of(led_cdev, struct ns2_led_data, cdev);
Simon Guinot11efe712010-07-06 16:08:46 +0200160
161 return sprintf(buf, "%d\n", led_dat->sata);
162}
163
164static DEVICE_ATTR(sata, 0644, ns2_led_sata_show, ns2_led_sata_store);
165
Johan Hovold475f8542014-06-25 10:08:51 -0700166static struct attribute *ns2_led_attrs[] = {
167 &dev_attr_sata.attr,
168 NULL
169};
170ATTRIBUTE_GROUPS(ns2_led);
171
Bill Pemberton98ea1ea2012-11-19 13:23:02 -0500172static int
Simon Guinot11efe712010-07-06 16:08:46 +0200173create_ns2_led(struct platform_device *pdev, struct ns2_led_data *led_dat,
174 const struct ns2_led *template)
175{
176 int ret;
177 enum ns2_led_modes mode;
178
Sachin Kamat04195822012-11-25 10:28:10 +0530179 ret = devm_gpio_request_one(&pdev->dev, template->cmd,
Simon Guinot4b904322015-07-02 19:56:42 +0200180 gpio_get_value_cansleep(template->cmd) ?
Jingoo Han9d04cba2013-03-07 18:38:26 -0800181 GPIOF_OUT_INIT_HIGH : GPIOF_OUT_INIT_LOW,
Jingoo Han31c3dc72012-10-23 05:18:21 -0700182 template->name);
Simon Guinot11efe712010-07-06 16:08:46 +0200183 if (ret) {
184 dev_err(&pdev->dev, "%s: failed to setup command GPIO\n",
185 template->name);
Jingoo Han31c3dc72012-10-23 05:18:21 -0700186 return ret;
Simon Guinot11efe712010-07-06 16:08:46 +0200187 }
188
Sachin Kamat04195822012-11-25 10:28:10 +0530189 ret = devm_gpio_request_one(&pdev->dev, template->slow,
Simon Guinot4b904322015-07-02 19:56:42 +0200190 gpio_get_value_cansleep(template->slow) ?
Jingoo Han9d04cba2013-03-07 18:38:26 -0800191 GPIOF_OUT_INIT_HIGH : GPIOF_OUT_INIT_LOW,
Jingoo Han31c3dc72012-10-23 05:18:21 -0700192 template->name);
Simon Guinot11efe712010-07-06 16:08:46 +0200193 if (ret) {
194 dev_err(&pdev->dev, "%s: failed to setup slow GPIO\n",
195 template->name);
Sachin Kamat04195822012-11-25 10:28:10 +0530196 return ret;
Simon Guinot11efe712010-07-06 16:08:46 +0200197 }
198
199 rwlock_init(&led_dat->rw_lock);
200
201 led_dat->cdev.name = template->name;
202 led_dat->cdev.default_trigger = template->default_trigger;
203 led_dat->cdev.blink_set = NULL;
Simon Guinot11efe712010-07-06 16:08:46 +0200204 led_dat->cdev.flags |= LED_CORE_SUSPENDRESUME;
Johan Hovold475f8542014-06-25 10:08:51 -0700205 led_dat->cdev.groups = ns2_led_groups;
Simon Guinot11efe712010-07-06 16:08:46 +0200206 led_dat->cmd = template->cmd;
207 led_dat->slow = template->slow;
Simon Guinot4b904322015-07-02 19:56:42 +0200208 led_dat->can_sleep = gpio_cansleep(led_dat->cmd) |
209 gpio_cansleep(led_dat->slow);
Jacek Anaszewskic29e6502015-11-20 11:39:41 +0100210 if (led_dat->can_sleep)
211 led_dat->cdev.brightness_set_blocking = ns2_led_set_blocking;
212 else
213 led_dat->cdev.brightness_set = ns2_led_set;
Vincent Donnefortf7fafd02015-07-02 19:56:40 +0200214 led_dat->modval = template->modval;
215 led_dat->num_modes = template->num_modes;
Simon Guinot11efe712010-07-06 16:08:46 +0200216
217 ret = ns2_led_get_mode(led_dat, &mode);
218 if (ret < 0)
Sachin Kamat04195822012-11-25 10:28:10 +0530219 return ret;
Simon Guinot11efe712010-07-06 16:08:46 +0200220
221 /* Set LED initial state. */
222 led_dat->sata = (mode == NS_V2_LED_SATA) ? 1 : 0;
223 led_dat->cdev.brightness =
224 (mode == NS_V2_LED_OFF) ? LED_OFF : LED_FULL;
225
226 ret = led_classdev_register(&pdev->dev, &led_dat->cdev);
227 if (ret < 0)
Sachin Kamat04195822012-11-25 10:28:10 +0530228 return ret;
Simon Guinot11efe712010-07-06 16:08:46 +0200229
Simon Guinot11efe712010-07-06 16:08:46 +0200230 return 0;
Simon Guinot11efe712010-07-06 16:08:46 +0200231}
232
Arnd Bergmannb8cd7422012-05-10 13:01:46 -0700233static void delete_ns2_led(struct ns2_led_data *led_dat)
Simon Guinot11efe712010-07-06 16:08:46 +0200234{
Simon Guinot11efe712010-07-06 16:08:46 +0200235 led_classdev_unregister(&led_dat->cdev);
Simon Guinot11efe712010-07-06 16:08:46 +0200236}
237
Simon Guinot72052fc2012-10-17 12:09:03 +0200238#ifdef CONFIG_OF_GPIO
239/*
240 * Translate OpenFirmware node properties into platform_data.
241 */
Linus Torvaldscf4af012012-12-12 12:14:06 -0800242static int
Simon Guinot72052fc2012-10-17 12:09:03 +0200243ns2_leds_get_of_pdata(struct device *dev, struct ns2_led_platform_data *pdata)
244{
245 struct device_node *np = dev->of_node;
246 struct device_node *child;
Vincent Donnefortf7fafd02015-07-02 19:56:40 +0200247 struct ns2_led *led, *leds;
Nishka Dasgupta79937a42019-07-16 12:54:24 +0530248 int ret, num_leds = 0;
Simon Guinot72052fc2012-10-17 12:09:03 +0200249
250 num_leds = of_get_child_count(np);
251 if (!num_leds)
252 return -ENODEV;
253
Kees Cooka86854d2018-06-12 14:07:58 -0700254 leds = devm_kcalloc(dev, num_leds, sizeof(struct ns2_led),
Simon Guinot72052fc2012-10-17 12:09:03 +0200255 GFP_KERNEL);
256 if (!leds)
257 return -ENOMEM;
258
Vincent Donnefortf7fafd02015-07-02 19:56:40 +0200259 led = leds;
Simon Guinot72052fc2012-10-17 12:09:03 +0200260 for_each_child_of_node(np, child) {
261 const char *string;
Nishka Dasgupta79937a42019-07-16 12:54:24 +0530262 int i, num_modes;
Vincent Donnefortf7fafd02015-07-02 19:56:40 +0200263 struct ns2_led_modval *modval;
Simon Guinot72052fc2012-10-17 12:09:03 +0200264
265 ret = of_get_named_gpio(child, "cmd-gpio", 0);
266 if (ret < 0)
Nishka Dasgupta79937a42019-07-16 12:54:24 +0530267 goto err_node_put;
Vincent Donnefortf7fafd02015-07-02 19:56:40 +0200268 led->cmd = ret;
Simon Guinot72052fc2012-10-17 12:09:03 +0200269 ret = of_get_named_gpio(child, "slow-gpio", 0);
270 if (ret < 0)
Nishka Dasgupta79937a42019-07-16 12:54:24 +0530271 goto err_node_put;
Vincent Donnefortf7fafd02015-07-02 19:56:40 +0200272 led->slow = ret;
Simon Guinot72052fc2012-10-17 12:09:03 +0200273 ret = of_property_read_string(child, "label", &string);
Vincent Donnefortf7fafd02015-07-02 19:56:40 +0200274 led->name = (ret == 0) ? string : child->name;
Simon Guinot72052fc2012-10-17 12:09:03 +0200275 ret = of_property_read_string(child, "linux,default-trigger",
276 &string);
277 if (ret == 0)
Vincent Donnefortf7fafd02015-07-02 19:56:40 +0200278 led->default_trigger = string;
Simon Guinot72052fc2012-10-17 12:09:03 +0200279
Vincent Donnefortf7fafd02015-07-02 19:56:40 +0200280 ret = of_property_count_u32_elems(child, "modes-map");
281 if (ret < 0 || ret % 3) {
282 dev_err(dev,
283 "Missing or malformed modes-map property\n");
Nishka Dasgupta79937a42019-07-16 12:54:24 +0530284 ret = -EINVAL;
285 goto err_node_put;
Vincent Donnefortf7fafd02015-07-02 19:56:40 +0200286 }
287
288 num_modes = ret / 3;
Kees Cooka86854d2018-06-12 14:07:58 -0700289 modval = devm_kcalloc(dev,
290 num_modes,
291 sizeof(struct ns2_led_modval),
Vincent Donnefortf7fafd02015-07-02 19:56:40 +0200292 GFP_KERNEL);
Nishka Dasgupta79937a42019-07-16 12:54:24 +0530293 if (!modval) {
294 ret = -ENOMEM;
295 goto err_node_put;
296 }
Vincent Donnefortf7fafd02015-07-02 19:56:40 +0200297
298 for (i = 0; i < num_modes; i++) {
299 of_property_read_u32_index(child,
300 "modes-map", 3 * i,
301 (u32 *) &modval[i].mode);
302 of_property_read_u32_index(child,
303 "modes-map", 3 * i + 1,
304 (u32 *) &modval[i].cmd_level);
305 of_property_read_u32_index(child,
306 "modes-map", 3 * i + 2,
307 (u32 *) &modval[i].slow_level);
308 }
309
310 led->num_modes = num_modes;
311 led->modval = modval;
312
313 led++;
Simon Guinot72052fc2012-10-17 12:09:03 +0200314 }
315
316 pdata->leds = leds;
317 pdata->num_leds = num_leds;
318
319 return 0;
Nishka Dasgupta79937a42019-07-16 12:54:24 +0530320
321err_node_put:
322 of_node_put(child);
323 return ret;
Simon Guinot72052fc2012-10-17 12:09:03 +0200324}
325
326static const struct of_device_id of_ns2_leds_match[] = {
327 { .compatible = "lacie,ns2-leds", },
328 {},
329};
Luis de Bethencourt98f9cc72015-09-01 23:36:59 +0200330MODULE_DEVICE_TABLE(of, of_ns2_leds_match);
Simon Guinot72052fc2012-10-17 12:09:03 +0200331#endif /* CONFIG_OF_GPIO */
332
Simon Guinot3de19292013-03-19 11:07:29 -0700333struct ns2_led_priv {
334 int num_leds;
335 struct ns2_led_data leds_data[];
336};
337
338static inline int sizeof_ns2_led_priv(int num_leds)
339{
340 return sizeof(struct ns2_led_priv) +
341 (sizeof(struct ns2_led_data) * num_leds);
342}
343
Bill Pemberton98ea1ea2012-11-19 13:23:02 -0500344static int ns2_led_probe(struct platform_device *pdev)
Simon Guinot11efe712010-07-06 16:08:46 +0200345{
Jingoo Han87aae1e2013-07-30 01:07:35 -0700346 struct ns2_led_platform_data *pdata = dev_get_platdata(&pdev->dev);
Simon Guinot3de19292013-03-19 11:07:29 -0700347 struct ns2_led_priv *priv;
Simon Guinot11efe712010-07-06 16:08:46 +0200348 int i;
349 int ret;
350
Simon Guinot72052fc2012-10-17 12:09:03 +0200351#ifdef CONFIG_OF_GPIO
352 if (!pdata) {
353 pdata = devm_kzalloc(&pdev->dev,
354 sizeof(struct ns2_led_platform_data),
355 GFP_KERNEL);
356 if (!pdata)
357 return -ENOMEM;
358
359 ret = ns2_leds_get_of_pdata(&pdev->dev, pdata);
360 if (ret)
361 return ret;
362 }
363#else
Simon Guinot11efe712010-07-06 16:08:46 +0200364 if (!pdata)
365 return -EINVAL;
Simon Guinot72052fc2012-10-17 12:09:03 +0200366#endif /* CONFIG_OF_GPIO */
Simon Guinot11efe712010-07-06 16:08:46 +0200367
Simon Guinot3de19292013-03-19 11:07:29 -0700368 priv = devm_kzalloc(&pdev->dev,
369 sizeof_ns2_led_priv(pdata->num_leds), GFP_KERNEL);
370 if (!priv)
Simon Guinot11efe712010-07-06 16:08:46 +0200371 return -ENOMEM;
Simon Guinot3de19292013-03-19 11:07:29 -0700372 priv->num_leds = pdata->num_leds;
Simon Guinot11efe712010-07-06 16:08:46 +0200373
Simon Guinot3de19292013-03-19 11:07:29 -0700374 for (i = 0; i < priv->num_leds; i++) {
375 ret = create_ns2_led(pdev, &priv->leds_data[i],
376 &pdata->leds[i]);
Bryan Wua209f762012-07-04 12:30:50 +0800377 if (ret < 0) {
378 for (i = i - 1; i >= 0; i--)
Simon Guinot3de19292013-03-19 11:07:29 -0700379 delete_ns2_led(&priv->leds_data[i]);
Bryan Wua209f762012-07-04 12:30:50 +0800380 return ret;
381 }
Simon Guinot11efe712010-07-06 16:08:46 +0200382 }
383
Simon Guinot3de19292013-03-19 11:07:29 -0700384 platform_set_drvdata(pdev, priv);
Simon Guinot11efe712010-07-06 16:08:46 +0200385
386 return 0;
Simon Guinot11efe712010-07-06 16:08:46 +0200387}
388
Bill Pemberton678e8a62012-11-19 13:26:00 -0500389static int ns2_led_remove(struct platform_device *pdev)
Simon Guinot11efe712010-07-06 16:08:46 +0200390{
391 int i;
Simon Guinot3de19292013-03-19 11:07:29 -0700392 struct ns2_led_priv *priv;
Simon Guinot11efe712010-07-06 16:08:46 +0200393
Simon Guinot3de19292013-03-19 11:07:29 -0700394 priv = platform_get_drvdata(pdev);
Simon Guinot11efe712010-07-06 16:08:46 +0200395
Simon Guinot3de19292013-03-19 11:07:29 -0700396 for (i = 0; i < priv->num_leds; i++)
397 delete_ns2_led(&priv->leds_data[i]);
Simon Guinot11efe712010-07-06 16:08:46 +0200398
Simon Guinot11efe712010-07-06 16:08:46 +0200399 return 0;
400}
401
402static struct platform_driver ns2_led_driver = {
403 .probe = ns2_led_probe,
Bill Pembertondf07cf82012-11-19 13:20:20 -0500404 .remove = ns2_led_remove,
Simon Guinot11efe712010-07-06 16:08:46 +0200405 .driver = {
Simon Guinot72052fc2012-10-17 12:09:03 +0200406 .name = "leds-ns2",
Simon Guinot72052fc2012-10-17 12:09:03 +0200407 .of_match_table = of_match_ptr(of_ns2_leds_match),
Simon Guinot11efe712010-07-06 16:08:46 +0200408 },
409};
Simon Guinot11efe712010-07-06 16:08:46 +0200410
Axel Lin892a8842012-01-10 15:09:24 -0800411module_platform_driver(ns2_led_driver);
Simon Guinot11efe712010-07-06 16:08:46 +0200412
413MODULE_AUTHOR("Simon Guinot <sguinot@lacie.com>");
414MODULE_DESCRIPTION("Network Space v2 LED driver");
415MODULE_LICENSE("GPL");
Axel Lin892a8842012-01-10 15:09:24 -0800416MODULE_ALIAS("platform:leds-ns2");