Blame view

drivers/leds/leds-da9052.c 4.42 KB
2874c5fd2   Thomas Gleixner   treewide: Replace...
1
  // SPDX-License-Identifier: GPL-2.0-or-later
5ba736311   David Dajun Chen   leds: driver for ...
2
3
4
5
6
7
  /*
   * LED Driver for Dialog DA9052 PMICs.
   *
   * Copyright(c) 2012 Dialog Semiconductor Ltd.
   *
   * Author: David Dajun Chen <dchen@diasemi.com>
5ba736311   David Dajun Chen   leds: driver for ...
8
9
10
11
   */
  
  #include <linux/module.h>
  #include <linux/kernel.h>
5ba736311   David Dajun Chen   leds: driver for ...
12
13
  #include <linux/platform_device.h>
  #include <linux/leds.h>
5ba736311   David Dajun Chen   leds: driver for ...
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
  #include <linux/slab.h>
  
  #include <linux/mfd/da9052/reg.h>
  #include <linux/mfd/da9052/da9052.h>
  #include <linux/mfd/da9052/pdata.h>
  
  #define DA9052_OPENDRAIN_OUTPUT	2
  #define DA9052_SET_HIGH_LVL_OUTPUT	(1 << 3)
  #define DA9052_MASK_UPPER_NIBBLE	0xF0
  #define DA9052_MASK_LOWER_NIBBLE	0x0F
  #define DA9052_NIBBLE_SHIFT		4
  #define DA9052_MAX_BRIGHTNESS		0x5f
  
  struct da9052_led {
  	struct led_classdev cdev;
5ba736311   David Dajun Chen   leds: driver for ...
29
30
31
  	struct da9052 *da9052;
  	unsigned char led_index;
  	unsigned char id;
5ba736311   David Dajun Chen   leds: driver for ...
32
33
34
35
36
37
  };
  
  static unsigned char led_reg[] = {
  	DA9052_LED_CONT_4_REG,
  	DA9052_LED_CONT_5_REG,
  };
0a1f01d38   Andrew Lunn   leds: da9052: Rem...
38
39
  static int da9052_set_led_brightness(struct da9052_led *led,
  				     enum led_brightness brightness)
5ba736311   David Dajun Chen   leds: driver for ...
40
41
42
  {
  	u8 val;
  	int error;
0a1f01d38   Andrew Lunn   leds: da9052: Rem...
43
  	val = (brightness & 0x7f) | DA9052_LED_CONT_DIM;
5ba736311   David Dajun Chen   leds: driver for ...
44
45
46
47
48
49
50
51
  
  	error = da9052_reg_write(led->da9052, led_reg[led->led_index], val);
  	if (error < 0)
  		dev_err(led->da9052->dev, "Failed to set led brightness, %d
  ",
  			error);
  	return error;
  }
0a1f01d38   Andrew Lunn   leds: da9052: Rem...
52
  static int da9052_led_set(struct led_classdev *led_cdev,
5ba736311   David Dajun Chen   leds: driver for ...
53
54
  			   enum led_brightness value)
  {
0a1f01d38   Andrew Lunn   leds: da9052: Rem...
55
56
  	struct da9052_led *led =
  			container_of(led_cdev, struct da9052_led, cdev);
5ba736311   David Dajun Chen   leds: driver for ...
57

0a1f01d38   Andrew Lunn   leds: da9052: Rem...
58
  	return da9052_set_led_brightness(led, value);
5ba736311   David Dajun Chen   leds: driver for ...
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
  }
  
  static int da9052_configure_leds(struct da9052 *da9052)
  {
  	int error;
  	unsigned char register_value = DA9052_OPENDRAIN_OUTPUT
  				       | DA9052_SET_HIGH_LVL_OUTPUT;
  
  	error = da9052_reg_update(da9052, DA9052_GPIO_14_15_REG,
  				  DA9052_MASK_LOWER_NIBBLE,
  				  register_value);
  
  	if (error < 0) {
  		dev_err(da9052->dev, "Failed to write GPIO 14-15 reg, %d
  ",
  			error);
  		return error;
  	}
  
  	error = da9052_reg_update(da9052, DA9052_GPIO_14_15_REG,
  				  DA9052_MASK_UPPER_NIBBLE,
  				  register_value << DA9052_NIBBLE_SHIFT);
  	if (error < 0)
  		dev_err(da9052->dev, "Failed to write GPIO 14-15 reg, %d
  ",
  			error);
  
  	return error;
  }
98ea1ea20   Bill Pemberton   leds: remove use ...
88
  static int da9052_led_probe(struct platform_device *pdev)
5ba736311   David Dajun Chen   leds: driver for ...
89
90
91
92
93
94
95
96
97
  {
  	struct da9052_pdata *pdata;
  	struct da9052 *da9052;
  	struct led_platform_data *pled;
  	struct da9052_led *led = NULL;
  	int error = -ENODEV;
  	int i;
  
  	da9052 = dev_get_drvdata(pdev->dev.parent);
87aae1ea8   Jingoo Han   leds: use dev_get...
98
  	pdata = dev_get_platdata(da9052->dev);
5ba736311   David Dajun Chen   leds: driver for ...
99
100
101
102
103
104
105
106
107
108
109
110
  	if (pdata == NULL) {
  		dev_err(&pdev->dev, "No platform data
  ");
  		goto err;
  	}
  
  	pled = pdata->pled;
  	if (pled == NULL) {
  		dev_err(&pdev->dev, "No platform data for LED
  ");
  		goto err;
  	}
a86854d0c   Kees Cook   treewide: devm_kz...
111
112
  	led = devm_kcalloc(&pdev->dev,
  			   pled->num_leds, sizeof(struct da9052_led),
5ba736311   David Dajun Chen   leds: driver for ...
113
  			   GFP_KERNEL);
0c9a03b68   Xiubo Li   leds: Remove dupl...
114
  	if (!led) {
5ba736311   David Dajun Chen   leds: driver for ...
115
116
117
118
119
120
  		error = -ENOMEM;
  		goto err;
  	}
  
  	for (i = 0; i < pled->num_leds; i++) {
  		led[i].cdev.name = pled->leds[i].name;
0a1f01d38   Andrew Lunn   leds: da9052: Rem...
121
  		led[i].cdev.brightness_set_blocking = da9052_led_set;
5ba736311   David Dajun Chen   leds: driver for ...
122
123
  		led[i].cdev.brightness = LED_OFF;
  		led[i].cdev.max_brightness = DA9052_MAX_BRIGHTNESS;
5ba736311   David Dajun Chen   leds: driver for ...
124
125
  		led[i].led_index = pled->leds[i].flags;
  		led[i].da9052 = dev_get_drvdata(pdev->dev.parent);
5ba736311   David Dajun Chen   leds: driver for ...
126
127
128
129
130
131
132
133
  
  		error = led_classdev_register(pdev->dev.parent, &led[i].cdev);
  		if (error) {
  			dev_err(&pdev->dev, "Failed to register led %d
  ",
  				led[i].led_index);
  			goto err_register;
  		}
0a1f01d38   Andrew Lunn   leds: da9052: Rem...
134
135
  		error = da9052_set_led_brightness(&led[i],
  						  led[i].cdev.brightness);
5ba736311   David Dajun Chen   leds: driver for ...
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
  		if (error) {
  			dev_err(&pdev->dev, "Unable to init led %d
  ",
  				led[i].led_index);
  			continue;
  		}
  	}
  	error = da9052_configure_leds(led->da9052);
  	if (error) {
  		dev_err(&pdev->dev, "Failed to configure GPIO LED%d
  ", error);
  		goto err_register;
  	}
  
  	platform_set_drvdata(pdev, led);
  
  	return 0;
  
  err_register:
0a1f01d38   Andrew Lunn   leds: da9052: Rem...
155
  	for (i = i - 1; i >= 0; i--)
5ba736311   David Dajun Chen   leds: driver for ...
156
  		led_classdev_unregister(&led[i].cdev);
5ba736311   David Dajun Chen   leds: driver for ...
157
158
159
  err:
  	return error;
  }
678e8a6be   Bill Pemberton   leds: remove use ...
160
  static int da9052_led_remove(struct platform_device *pdev)
5ba736311   David Dajun Chen   leds: driver for ...
161
162
163
164
165
166
167
168
  {
  	struct da9052_led *led = platform_get_drvdata(pdev);
  	struct da9052_pdata *pdata;
  	struct da9052 *da9052;
  	struct led_platform_data *pled;
  	int i;
  
  	da9052 = dev_get_drvdata(pdev->dev.parent);
87aae1ea8   Jingoo Han   leds: use dev_get...
169
  	pdata = dev_get_platdata(da9052->dev);
5ba736311   David Dajun Chen   leds: driver for ...
170
171
172
  	pled = pdata->pled;
  
  	for (i = 0; i < pled->num_leds; i++) {
0a1f01d38   Andrew Lunn   leds: da9052: Rem...
173
  		da9052_set_led_brightness(&led[i], LED_OFF);
5ba736311   David Dajun Chen   leds: driver for ...
174
  		led_classdev_unregister(&led[i].cdev);
5ba736311   David Dajun Chen   leds: driver for ...
175
176
177
178
179
180
181
182
  	}
  
  	return 0;
  }
  
  static struct platform_driver da9052_led_driver = {
  	.driver		= {
  		.name	= "da9052-leds",
5ba736311   David Dajun Chen   leds: driver for ...
183
184
  	},
  	.probe		= da9052_led_probe,
df07cf812   Bill Pemberton   leds: remove use ...
185
  	.remove		= da9052_led_remove,
5ba736311   David Dajun Chen   leds: driver for ...
186
187
188
189
190
191
192
  };
  
  module_platform_driver(da9052_led_driver);
  
  MODULE_AUTHOR("Dialog Semiconductor Ltd <dchen@diasemi.com>");
  MODULE_DESCRIPTION("LED driver for Dialog DA9052 PMIC");
  MODULE_LICENSE("GPL");