xref: /OK3568_Linux_fs/kernel/drivers/leds/leds-an30259a.c (revision 4882a59341e53eb6f0b4789bf948001014eff981)
1*4882a593Smuzhiyun // SPDX-License-Identifier: GPL-2.0+
2*4882a593Smuzhiyun //
3*4882a593Smuzhiyun // Driver for Panasonic AN30259A 3-channel LED driver
4*4882a593Smuzhiyun //
5*4882a593Smuzhiyun // Copyright (c) 2018 Simon Shields <simon@lineageos.org>
6*4882a593Smuzhiyun //
7*4882a593Smuzhiyun // Datasheet:
8*4882a593Smuzhiyun // https://www.alliedelec.com/m/d/a9d2b3ee87c2d1a535a41dd747b1c247.pdf
9*4882a593Smuzhiyun 
10*4882a593Smuzhiyun #include <linux/i2c.h>
11*4882a593Smuzhiyun #include <linux/leds.h>
12*4882a593Smuzhiyun #include <linux/module.h>
13*4882a593Smuzhiyun #include <linux/mutex.h>
14*4882a593Smuzhiyun #include <linux/of.h>
15*4882a593Smuzhiyun #include <linux/regmap.h>
16*4882a593Smuzhiyun 
17*4882a593Smuzhiyun #define AN30259A_MAX_LEDS 3
18*4882a593Smuzhiyun 
19*4882a593Smuzhiyun #define AN30259A_REG_SRESET 0x00
20*4882a593Smuzhiyun #define AN30259A_LED_SRESET BIT(0)
21*4882a593Smuzhiyun 
22*4882a593Smuzhiyun /* LED power registers */
23*4882a593Smuzhiyun #define AN30259A_REG_LED_ON 0x01
24*4882a593Smuzhiyun #define AN30259A_LED_EN(x) BIT((x) - 1)
25*4882a593Smuzhiyun #define AN30259A_LED_SLOPE(x) BIT(((x) - 1) + 4)
26*4882a593Smuzhiyun 
27*4882a593Smuzhiyun #define AN30259A_REG_LEDCC(x) (0x03 + ((x) - 1))
28*4882a593Smuzhiyun 
29*4882a593Smuzhiyun /* slope control registers */
30*4882a593Smuzhiyun #define AN30259A_REG_SLOPE(x) (0x06 + ((x) - 1))
31*4882a593Smuzhiyun #define AN30259A_LED_SLOPETIME1(x) (x)
32*4882a593Smuzhiyun #define AN30259A_LED_SLOPETIME2(x) ((x) << 4)
33*4882a593Smuzhiyun 
34*4882a593Smuzhiyun #define AN30259A_REG_LEDCNT1(x) (0x09 + (4 * ((x) - 1)))
35*4882a593Smuzhiyun #define AN30259A_LED_DUTYMAX(x) ((x) << 4)
36*4882a593Smuzhiyun #define AN30259A_LED_DUTYMID(x) (x)
37*4882a593Smuzhiyun 
38*4882a593Smuzhiyun #define AN30259A_REG_LEDCNT2(x) (0x0A + (4 * ((x) - 1)))
39*4882a593Smuzhiyun #define AN30259A_LED_DELAY(x) ((x) << 4)
40*4882a593Smuzhiyun #define AN30259A_LED_DUTYMIN(x) (x)
41*4882a593Smuzhiyun 
42*4882a593Smuzhiyun /* detention time control (length of each slope step) */
43*4882a593Smuzhiyun #define AN30259A_REG_LEDCNT3(x) (0x0B + (4 * ((x) - 1)))
44*4882a593Smuzhiyun #define AN30259A_LED_DT1(x) (x)
45*4882a593Smuzhiyun #define AN30259A_LED_DT2(x) ((x) << 4)
46*4882a593Smuzhiyun 
47*4882a593Smuzhiyun #define AN30259A_REG_LEDCNT4(x) (0x0C + (4 * ((x) - 1)))
48*4882a593Smuzhiyun #define AN30259A_LED_DT3(x) (x)
49*4882a593Smuzhiyun #define AN30259A_LED_DT4(x) ((x) << 4)
50*4882a593Smuzhiyun 
51*4882a593Smuzhiyun #define AN30259A_REG_MAX 0x14
52*4882a593Smuzhiyun 
53*4882a593Smuzhiyun #define AN30259A_BLINK_MAX_TIME 7500 /* ms */
54*4882a593Smuzhiyun #define AN30259A_SLOPE_RESOLUTION 500 /* ms */
55*4882a593Smuzhiyun 
56*4882a593Smuzhiyun #define AN30259A_NAME "an30259a"
57*4882a593Smuzhiyun 
58*4882a593Smuzhiyun #define STATE_OFF 0
59*4882a593Smuzhiyun #define STATE_KEEP 1
60*4882a593Smuzhiyun #define STATE_ON 2
61*4882a593Smuzhiyun 
62*4882a593Smuzhiyun struct an30259a;
63*4882a593Smuzhiyun 
64*4882a593Smuzhiyun struct an30259a_led {
65*4882a593Smuzhiyun 	struct an30259a *chip;
66*4882a593Smuzhiyun 	struct fwnode_handle *fwnode;
67*4882a593Smuzhiyun 	struct led_classdev cdev;
68*4882a593Smuzhiyun 	u32 num;
69*4882a593Smuzhiyun 	u32 default_state;
70*4882a593Smuzhiyun 	bool sloping;
71*4882a593Smuzhiyun };
72*4882a593Smuzhiyun 
73*4882a593Smuzhiyun struct an30259a {
74*4882a593Smuzhiyun 	struct mutex mutex; /* held when writing to registers */
75*4882a593Smuzhiyun 	struct i2c_client *client;
76*4882a593Smuzhiyun 	struct an30259a_led leds[AN30259A_MAX_LEDS];
77*4882a593Smuzhiyun 	struct regmap *regmap;
78*4882a593Smuzhiyun 	int num_leds;
79*4882a593Smuzhiyun };
80*4882a593Smuzhiyun 
an30259a_brightness_set(struct led_classdev * cdev,enum led_brightness brightness)81*4882a593Smuzhiyun static int an30259a_brightness_set(struct led_classdev *cdev,
82*4882a593Smuzhiyun 				   enum led_brightness brightness)
83*4882a593Smuzhiyun {
84*4882a593Smuzhiyun 	struct an30259a_led *led;
85*4882a593Smuzhiyun 	int ret;
86*4882a593Smuzhiyun 	unsigned int led_on;
87*4882a593Smuzhiyun 
88*4882a593Smuzhiyun 	led = container_of(cdev, struct an30259a_led, cdev);
89*4882a593Smuzhiyun 	mutex_lock(&led->chip->mutex);
90*4882a593Smuzhiyun 
91*4882a593Smuzhiyun 	ret = regmap_read(led->chip->regmap, AN30259A_REG_LED_ON, &led_on);
92*4882a593Smuzhiyun 	if (ret)
93*4882a593Smuzhiyun 		goto error;
94*4882a593Smuzhiyun 
95*4882a593Smuzhiyun 	switch (brightness) {
96*4882a593Smuzhiyun 	case LED_OFF:
97*4882a593Smuzhiyun 		led_on &= ~AN30259A_LED_EN(led->num);
98*4882a593Smuzhiyun 		led_on &= ~AN30259A_LED_SLOPE(led->num);
99*4882a593Smuzhiyun 		led->sloping = false;
100*4882a593Smuzhiyun 		break;
101*4882a593Smuzhiyun 	default:
102*4882a593Smuzhiyun 		led_on |= AN30259A_LED_EN(led->num);
103*4882a593Smuzhiyun 		if (led->sloping)
104*4882a593Smuzhiyun 			led_on |= AN30259A_LED_SLOPE(led->num);
105*4882a593Smuzhiyun 		ret = regmap_write(led->chip->regmap,
106*4882a593Smuzhiyun 				   AN30259A_REG_LEDCNT1(led->num),
107*4882a593Smuzhiyun 				   AN30259A_LED_DUTYMAX(0xf) |
108*4882a593Smuzhiyun 				   AN30259A_LED_DUTYMID(0xf));
109*4882a593Smuzhiyun 		if (ret)
110*4882a593Smuzhiyun 			goto error;
111*4882a593Smuzhiyun 		break;
112*4882a593Smuzhiyun 	}
113*4882a593Smuzhiyun 
114*4882a593Smuzhiyun 	ret = regmap_write(led->chip->regmap, AN30259A_REG_LED_ON, led_on);
115*4882a593Smuzhiyun 	if (ret)
116*4882a593Smuzhiyun 		goto error;
117*4882a593Smuzhiyun 
118*4882a593Smuzhiyun 	ret = regmap_write(led->chip->regmap, AN30259A_REG_LEDCC(led->num),
119*4882a593Smuzhiyun 			   brightness);
120*4882a593Smuzhiyun 
121*4882a593Smuzhiyun error:
122*4882a593Smuzhiyun 	mutex_unlock(&led->chip->mutex);
123*4882a593Smuzhiyun 
124*4882a593Smuzhiyun 	return ret;
125*4882a593Smuzhiyun }
126*4882a593Smuzhiyun 
an30259a_blink_set(struct led_classdev * cdev,unsigned long * delay_off,unsigned long * delay_on)127*4882a593Smuzhiyun static int an30259a_blink_set(struct led_classdev *cdev,
128*4882a593Smuzhiyun 			      unsigned long *delay_off, unsigned long *delay_on)
129*4882a593Smuzhiyun {
130*4882a593Smuzhiyun 	struct an30259a_led *led;
131*4882a593Smuzhiyun 	int ret, num;
132*4882a593Smuzhiyun 	unsigned int led_on;
133*4882a593Smuzhiyun 	unsigned long off = *delay_off, on = *delay_on;
134*4882a593Smuzhiyun 
135*4882a593Smuzhiyun 	led = container_of(cdev, struct an30259a_led, cdev);
136*4882a593Smuzhiyun 
137*4882a593Smuzhiyun 	mutex_lock(&led->chip->mutex);
138*4882a593Smuzhiyun 	num = led->num;
139*4882a593Smuzhiyun 
140*4882a593Smuzhiyun 	/* slope time can only be a multiple of 500ms. */
141*4882a593Smuzhiyun 	if (off % AN30259A_SLOPE_RESOLUTION || on % AN30259A_SLOPE_RESOLUTION) {
142*4882a593Smuzhiyun 		ret = -EINVAL;
143*4882a593Smuzhiyun 		goto error;
144*4882a593Smuzhiyun 	}
145*4882a593Smuzhiyun 
146*4882a593Smuzhiyun 	/* up to a maximum of 7500ms. */
147*4882a593Smuzhiyun 	if (off > AN30259A_BLINK_MAX_TIME || on > AN30259A_BLINK_MAX_TIME) {
148*4882a593Smuzhiyun 		ret = -EINVAL;
149*4882a593Smuzhiyun 		goto error;
150*4882a593Smuzhiyun 	}
151*4882a593Smuzhiyun 
152*4882a593Smuzhiyun 	/* if no blink specified, default to 1 Hz. */
153*4882a593Smuzhiyun 	if (!off && !on) {
154*4882a593Smuzhiyun 		*delay_off = off = 500;
155*4882a593Smuzhiyun 		*delay_on = on = 500;
156*4882a593Smuzhiyun 	}
157*4882a593Smuzhiyun 
158*4882a593Smuzhiyun 	/* convert into values the HW will understand. */
159*4882a593Smuzhiyun 	off /= AN30259A_SLOPE_RESOLUTION;
160*4882a593Smuzhiyun 	on /= AN30259A_SLOPE_RESOLUTION;
161*4882a593Smuzhiyun 
162*4882a593Smuzhiyun 	/* duty min should be zero (=off), delay should be zero. */
163*4882a593Smuzhiyun 	ret = regmap_write(led->chip->regmap, AN30259A_REG_LEDCNT2(num),
164*4882a593Smuzhiyun 			   AN30259A_LED_DELAY(0) | AN30259A_LED_DUTYMIN(0));
165*4882a593Smuzhiyun 	if (ret)
166*4882a593Smuzhiyun 		goto error;
167*4882a593Smuzhiyun 
168*4882a593Smuzhiyun 	/* reset detention time (no "breathing" effect). */
169*4882a593Smuzhiyun 	ret = regmap_write(led->chip->regmap, AN30259A_REG_LEDCNT3(num),
170*4882a593Smuzhiyun 			   AN30259A_LED_DT1(0) | AN30259A_LED_DT2(0));
171*4882a593Smuzhiyun 	if (ret)
172*4882a593Smuzhiyun 		goto error;
173*4882a593Smuzhiyun 	ret = regmap_write(led->chip->regmap, AN30259A_REG_LEDCNT4(num),
174*4882a593Smuzhiyun 			   AN30259A_LED_DT3(0) | AN30259A_LED_DT4(0));
175*4882a593Smuzhiyun 	if (ret)
176*4882a593Smuzhiyun 		goto error;
177*4882a593Smuzhiyun 
178*4882a593Smuzhiyun 	/* slope time controls on/off cycle length. */
179*4882a593Smuzhiyun 	ret = regmap_write(led->chip->regmap, AN30259A_REG_SLOPE(num),
180*4882a593Smuzhiyun 			   AN30259A_LED_SLOPETIME1(off) |
181*4882a593Smuzhiyun 			   AN30259A_LED_SLOPETIME2(on));
182*4882a593Smuzhiyun 	if (ret)
183*4882a593Smuzhiyun 		goto error;
184*4882a593Smuzhiyun 
185*4882a593Smuzhiyun 	/* Finally, enable slope mode. */
186*4882a593Smuzhiyun 	ret = regmap_read(led->chip->regmap, AN30259A_REG_LED_ON, &led_on);
187*4882a593Smuzhiyun 	if (ret)
188*4882a593Smuzhiyun 		goto error;
189*4882a593Smuzhiyun 
190*4882a593Smuzhiyun 	led_on |= AN30259A_LED_SLOPE(num) | AN30259A_LED_EN(led->num);
191*4882a593Smuzhiyun 
192*4882a593Smuzhiyun 	ret = regmap_write(led->chip->regmap, AN30259A_REG_LED_ON, led_on);
193*4882a593Smuzhiyun 
194*4882a593Smuzhiyun 	if (!ret)
195*4882a593Smuzhiyun 		led->sloping = true;
196*4882a593Smuzhiyun error:
197*4882a593Smuzhiyun 	mutex_unlock(&led->chip->mutex);
198*4882a593Smuzhiyun 
199*4882a593Smuzhiyun 	return ret;
200*4882a593Smuzhiyun }
201*4882a593Smuzhiyun 
an30259a_dt_init(struct i2c_client * client,struct an30259a * chip)202*4882a593Smuzhiyun static int an30259a_dt_init(struct i2c_client *client,
203*4882a593Smuzhiyun 			    struct an30259a *chip)
204*4882a593Smuzhiyun {
205*4882a593Smuzhiyun 	struct device_node *np = dev_of_node(&client->dev), *child;
206*4882a593Smuzhiyun 	int count, ret;
207*4882a593Smuzhiyun 	int i = 0;
208*4882a593Smuzhiyun 	const char *str;
209*4882a593Smuzhiyun 	struct an30259a_led *led;
210*4882a593Smuzhiyun 
211*4882a593Smuzhiyun 	count = of_get_available_child_count(np);
212*4882a593Smuzhiyun 	if (!count || count > AN30259A_MAX_LEDS)
213*4882a593Smuzhiyun 		return -EINVAL;
214*4882a593Smuzhiyun 
215*4882a593Smuzhiyun 	for_each_available_child_of_node(np, child) {
216*4882a593Smuzhiyun 		u32 source;
217*4882a593Smuzhiyun 
218*4882a593Smuzhiyun 		ret = of_property_read_u32(child, "reg", &source);
219*4882a593Smuzhiyun 		if (ret != 0 || !source || source > AN30259A_MAX_LEDS) {
220*4882a593Smuzhiyun 			dev_err(&client->dev, "Couldn't read LED address: %d\n",
221*4882a593Smuzhiyun 				ret);
222*4882a593Smuzhiyun 			count--;
223*4882a593Smuzhiyun 			continue;
224*4882a593Smuzhiyun 		}
225*4882a593Smuzhiyun 
226*4882a593Smuzhiyun 		led = &chip->leds[i];
227*4882a593Smuzhiyun 
228*4882a593Smuzhiyun 		led->num = source;
229*4882a593Smuzhiyun 		led->chip = chip;
230*4882a593Smuzhiyun 		led->fwnode = of_fwnode_handle(child);
231*4882a593Smuzhiyun 
232*4882a593Smuzhiyun 		if (!of_property_read_string(child, "default-state", &str)) {
233*4882a593Smuzhiyun 			if (!strcmp(str, "on"))
234*4882a593Smuzhiyun 				led->default_state = STATE_ON;
235*4882a593Smuzhiyun 			else if (!strcmp(str, "keep"))
236*4882a593Smuzhiyun 				led->default_state = STATE_KEEP;
237*4882a593Smuzhiyun 			else
238*4882a593Smuzhiyun 				led->default_state = STATE_OFF;
239*4882a593Smuzhiyun 		}
240*4882a593Smuzhiyun 
241*4882a593Smuzhiyun 		i++;
242*4882a593Smuzhiyun 	}
243*4882a593Smuzhiyun 
244*4882a593Smuzhiyun 	if (!count)
245*4882a593Smuzhiyun 		return -EINVAL;
246*4882a593Smuzhiyun 
247*4882a593Smuzhiyun 	chip->num_leds = i;
248*4882a593Smuzhiyun 
249*4882a593Smuzhiyun 	return 0;
250*4882a593Smuzhiyun }
251*4882a593Smuzhiyun 
252*4882a593Smuzhiyun static const struct regmap_config an30259a_regmap_config = {
253*4882a593Smuzhiyun 	.reg_bits = 8,
254*4882a593Smuzhiyun 	.val_bits = 8,
255*4882a593Smuzhiyun 	.max_register = AN30259A_REG_MAX,
256*4882a593Smuzhiyun };
257*4882a593Smuzhiyun 
an30259a_init_default_state(struct an30259a_led * led)258*4882a593Smuzhiyun static void an30259a_init_default_state(struct an30259a_led *led)
259*4882a593Smuzhiyun {
260*4882a593Smuzhiyun 	struct an30259a *chip = led->chip;
261*4882a593Smuzhiyun 	int led_on, err;
262*4882a593Smuzhiyun 
263*4882a593Smuzhiyun 	switch (led->default_state) {
264*4882a593Smuzhiyun 	case STATE_ON:
265*4882a593Smuzhiyun 		led->cdev.brightness = LED_FULL;
266*4882a593Smuzhiyun 		break;
267*4882a593Smuzhiyun 	case STATE_KEEP:
268*4882a593Smuzhiyun 		err = regmap_read(chip->regmap, AN30259A_REG_LED_ON, &led_on);
269*4882a593Smuzhiyun 		if (err)
270*4882a593Smuzhiyun 			break;
271*4882a593Smuzhiyun 
272*4882a593Smuzhiyun 		if (!(led_on & AN30259A_LED_EN(led->num))) {
273*4882a593Smuzhiyun 			led->cdev.brightness = LED_OFF;
274*4882a593Smuzhiyun 			break;
275*4882a593Smuzhiyun 		}
276*4882a593Smuzhiyun 		regmap_read(chip->regmap, AN30259A_REG_LEDCC(led->num),
277*4882a593Smuzhiyun 			    &led->cdev.brightness);
278*4882a593Smuzhiyun 		break;
279*4882a593Smuzhiyun 	default:
280*4882a593Smuzhiyun 		led->cdev.brightness = LED_OFF;
281*4882a593Smuzhiyun 	}
282*4882a593Smuzhiyun 
283*4882a593Smuzhiyun 	an30259a_brightness_set(&led->cdev, led->cdev.brightness);
284*4882a593Smuzhiyun }
285*4882a593Smuzhiyun 
an30259a_probe(struct i2c_client * client)286*4882a593Smuzhiyun static int an30259a_probe(struct i2c_client *client)
287*4882a593Smuzhiyun {
288*4882a593Smuzhiyun 	struct an30259a *chip;
289*4882a593Smuzhiyun 	int i, err;
290*4882a593Smuzhiyun 
291*4882a593Smuzhiyun 	chip = devm_kzalloc(&client->dev, sizeof(*chip), GFP_KERNEL);
292*4882a593Smuzhiyun 	if (!chip)
293*4882a593Smuzhiyun 		return -ENOMEM;
294*4882a593Smuzhiyun 
295*4882a593Smuzhiyun 	err = an30259a_dt_init(client, chip);
296*4882a593Smuzhiyun 	if (err < 0)
297*4882a593Smuzhiyun 		return err;
298*4882a593Smuzhiyun 
299*4882a593Smuzhiyun 	mutex_init(&chip->mutex);
300*4882a593Smuzhiyun 	chip->client = client;
301*4882a593Smuzhiyun 	i2c_set_clientdata(client, chip);
302*4882a593Smuzhiyun 
303*4882a593Smuzhiyun 	chip->regmap = devm_regmap_init_i2c(client, &an30259a_regmap_config);
304*4882a593Smuzhiyun 
305*4882a593Smuzhiyun 	if (IS_ERR(chip->regmap)) {
306*4882a593Smuzhiyun 		err = PTR_ERR(chip->regmap);
307*4882a593Smuzhiyun 		dev_err(&client->dev, "Failed to allocate register map: %d\n",
308*4882a593Smuzhiyun 			err);
309*4882a593Smuzhiyun 		goto exit;
310*4882a593Smuzhiyun 	}
311*4882a593Smuzhiyun 
312*4882a593Smuzhiyun 	for (i = 0; i < chip->num_leds; i++) {
313*4882a593Smuzhiyun 		struct led_init_data init_data = {};
314*4882a593Smuzhiyun 
315*4882a593Smuzhiyun 		an30259a_init_default_state(&chip->leds[i]);
316*4882a593Smuzhiyun 		chip->leds[i].cdev.brightness_set_blocking =
317*4882a593Smuzhiyun 			an30259a_brightness_set;
318*4882a593Smuzhiyun 		chip->leds[i].cdev.blink_set = an30259a_blink_set;
319*4882a593Smuzhiyun 
320*4882a593Smuzhiyun 		init_data.fwnode = chip->leds[i].fwnode;
321*4882a593Smuzhiyun 		init_data.devicename = AN30259A_NAME;
322*4882a593Smuzhiyun 		init_data.default_label = ":";
323*4882a593Smuzhiyun 
324*4882a593Smuzhiyun 		err = devm_led_classdev_register_ext(&client->dev,
325*4882a593Smuzhiyun 						 &chip->leds[i].cdev,
326*4882a593Smuzhiyun 						 &init_data);
327*4882a593Smuzhiyun 		if (err < 0)
328*4882a593Smuzhiyun 			goto exit;
329*4882a593Smuzhiyun 	}
330*4882a593Smuzhiyun 	return 0;
331*4882a593Smuzhiyun 
332*4882a593Smuzhiyun exit:
333*4882a593Smuzhiyun 	mutex_destroy(&chip->mutex);
334*4882a593Smuzhiyun 	return err;
335*4882a593Smuzhiyun }
336*4882a593Smuzhiyun 
an30259a_remove(struct i2c_client * client)337*4882a593Smuzhiyun static int an30259a_remove(struct i2c_client *client)
338*4882a593Smuzhiyun {
339*4882a593Smuzhiyun 	struct an30259a *chip = i2c_get_clientdata(client);
340*4882a593Smuzhiyun 
341*4882a593Smuzhiyun 	mutex_destroy(&chip->mutex);
342*4882a593Smuzhiyun 
343*4882a593Smuzhiyun 	return 0;
344*4882a593Smuzhiyun }
345*4882a593Smuzhiyun 
346*4882a593Smuzhiyun static const struct of_device_id an30259a_match_table[] = {
347*4882a593Smuzhiyun 	{ .compatible = "panasonic,an30259a", },
348*4882a593Smuzhiyun 	{ /* sentinel */ },
349*4882a593Smuzhiyun };
350*4882a593Smuzhiyun 
351*4882a593Smuzhiyun MODULE_DEVICE_TABLE(of, an30259a_match_table);
352*4882a593Smuzhiyun 
353*4882a593Smuzhiyun static const struct i2c_device_id an30259a_id[] = {
354*4882a593Smuzhiyun 	{ "an30259a", 0 },
355*4882a593Smuzhiyun 	{ /* sentinel */ },
356*4882a593Smuzhiyun };
357*4882a593Smuzhiyun MODULE_DEVICE_TABLE(i2c, an30259a_id);
358*4882a593Smuzhiyun 
359*4882a593Smuzhiyun static struct i2c_driver an30259a_driver = {
360*4882a593Smuzhiyun 	.driver = {
361*4882a593Smuzhiyun 		.name = "leds-an30259a",
362*4882a593Smuzhiyun 		.of_match_table = of_match_ptr(an30259a_match_table),
363*4882a593Smuzhiyun 	},
364*4882a593Smuzhiyun 	.probe_new = an30259a_probe,
365*4882a593Smuzhiyun 	.remove = an30259a_remove,
366*4882a593Smuzhiyun 	.id_table = an30259a_id,
367*4882a593Smuzhiyun };
368*4882a593Smuzhiyun 
369*4882a593Smuzhiyun module_i2c_driver(an30259a_driver);
370*4882a593Smuzhiyun 
371*4882a593Smuzhiyun MODULE_AUTHOR("Simon Shields <simon@lineageos.org>");
372*4882a593Smuzhiyun MODULE_DESCRIPTION("AN30259A LED driver");
373*4882a593Smuzhiyun MODULE_LICENSE("GPL v2");
374