Lines Matching +full:led +full:- +full:4

1 // SPDX-License-Identifier: GPL-2.0-only
6 * Author: Peter Meerwald <p.meerwald@bct-electronic.com>
9 * Based on leds-pca955x.c
11 * LED driver for the PCA9633 I2C LED driver (7-bit slave address 0x62)
12 * LED driver for the PCA9634/5 I2C LED driver (7-bit slave address set by hw.)
18 * in identical fashion. The delay_on/delay_off values of the last LED
22 * or by adding the 'nxp,hw-blink' property to the DTS.
36 /* LED select registers determine the source that drives LED outputs */
37 #define PCA963X_LED_OFF 0x0 /* LED driver off */
38 #define PCA963X_LED_ON 0x1 /* LED driver on */
43 #define PCA963X_MODE2_OUTDRV 0x04 /* Open-drain or totem pole */
70 .n_leds = 4,
118 static int pca963x_brightness(struct pca963x_led *led, in pca963x_brightness() argument
121 struct i2c_client *client = led->chip->client; in pca963x_brightness()
122 struct pca963x_chipdef *chipdef = led->chip->chipdef; in pca963x_brightness()
127 ledout_addr = chipdef->ledout_base + (led->led_num / 4); in pca963x_brightness()
128 shift = 2 * (led->led_num % 4); in pca963x_brightness()
134 if (led->blinking) { in pca963x_brightness()
138 led->led_num, in pca963x_brightness()
148 led->blinking = false; in pca963x_brightness()
153 led->led_num, in pca963x_brightness()
158 if (led->blinking) in pca963x_brightness()
170 static void pca963x_blink(struct pca963x_led *led) in pca963x_blink() argument
172 struct i2c_client *client = led->chip->client; in pca963x_blink()
173 struct pca963x_chipdef *chipdef = led->chip->chipdef; in pca963x_blink()
177 ledout_addr = chipdef->ledout_base + (led->led_num / 4); in pca963x_blink()
178 shift = 2 * (led->led_num % 4); in pca963x_blink()
182 i2c_smbus_write_byte_data(client, chipdef->grppwm, led->gdc); in pca963x_blink()
184 i2c_smbus_write_byte_data(client, chipdef->grpfreq, led->gfrq); in pca963x_blink()
190 mutex_lock(&led->chip->mutex); in pca963x_blink()
198 mutex_unlock(&led->chip->mutex); in pca963x_blink()
199 led->blinking = true; in pca963x_blink()
202 static int pca963x_power_state(struct pca963x_led *led) in pca963x_power_state() argument
204 struct i2c_client *client = led->chip->client; in pca963x_power_state()
205 unsigned long *leds_on = &led->chip->leds_on; in pca963x_power_state()
208 if (led->led_cdev.brightness) in pca963x_power_state()
209 set_bit(led->led_num, leds_on); in pca963x_power_state()
211 clear_bit(led->led_num, leds_on); in pca963x_power_state()
215 *leds_on ? 0 : BIT(4)); in pca963x_power_state()
223 struct pca963x_led *led; in pca963x_led_set() local
226 led = container_of(led_cdev, struct pca963x_led, led_cdev); in pca963x_led_set()
228 mutex_lock(&led->chip->mutex); in pca963x_led_set()
230 ret = pca963x_brightness(led, value); in pca963x_led_set()
233 ret = pca963x_power_state(led); in pca963x_led_set()
236 mutex_unlock(&led->chip->mutex); in pca963x_led_set()
240 static unsigned int pca963x_period_scale(struct pca963x_led *led, in pca963x_period_scale() argument
243 unsigned int scaling = led->chip->chipdef->scaling; in pca963x_period_scale()
252 struct pca963x_led *led; in pca963x_blink_set() local
255 led = container_of(led_cdev, struct pca963x_led, led_cdev); in pca963x_blink_set()
266 period = pca963x_period_scale(led, time_on + time_off); in pca963x_blink_set()
273 period = pca963x_period_scale(led, 1000); in pca963x_blink_set()
277 * From manual: duty cycle = (GDC / 256) -> in pca963x_blink_set()
278 * (time_on / period) = (GDC / 256) -> in pca963x_blink_set()
281 gdc = (pca963x_period_scale(led, time_on) * 256) / period; in pca963x_blink_set()
285 * So, period (in ms) = (((GFRQ + 1) / 24) * 1000) -> in pca963x_blink_set()
286 * GFRQ = ((period * 24 / 1000) - 1) in pca963x_blink_set()
288 gfrq = (period * 24 / 1000) - 1; in pca963x_blink_set()
290 led->gdc = gdc; in pca963x_blink_set()
291 led->gfrq = gfrq; in pca963x_blink_set()
293 pca963x_blink(led); in pca963x_blink_set()
294 led->led_cdev.brightness = LED_FULL; in pca963x_blink_set()
306 struct pca963x_chipdef *chipdef = chip->chipdef; in pca963x_register_leds()
307 struct pca963x_led *led = chip->leds; in pca963x_register_leds() local
308 struct device *dev = &client->dev; in pca963x_register_leds()
315 if (device_property_read_u32(dev, "nxp,period-scale", in pca963x_register_leds()
316 &chipdef->scaling)) in pca963x_register_leds()
317 chipdef->scaling = 1000; in pca963x_register_leds()
319 hw_blink = device_property_read_bool(dev, "nxp,hw-blink"); in pca963x_register_leds()
325 /* default to open-drain unless totem pole (push-pull) is specified */ in pca963x_register_leds()
326 if (device_property_read_bool(dev, "nxp,totem-pole")) in pca963x_register_leds()
331 /* default to non-inverted output, unless inverted is specified */ in pca963x_register_leds()
332 if (device_property_read_bool(dev, "nxp,inverted-out")) in pca963x_register_leds()
346 if (ret || reg >= chipdef->n_leds) { in pca963x_register_leds()
349 ret = -EINVAL; in pca963x_register_leds()
353 led->led_num = reg; in pca963x_register_leds()
354 led->chip = chip; in pca963x_register_leds()
355 led->led_cdev.brightness_set_blocking = pca963x_led_set; in pca963x_register_leds()
357 led->led_cdev.blink_set = pca963x_blink_set; in pca963x_register_leds()
358 led->blinking = false; in pca963x_register_leds()
364 client->adapter->nr, client->addr, reg); in pca963x_register_leds()
367 ret = devm_led_classdev_register_ext(dev, &led->led_cdev, in pca963x_register_leds()
370 dev_err(dev, "Failed to register LED for node %pfw\n", in pca963x_register_leds()
375 ++led; in pca963x_register_leds()
389 reg = i2c_smbus_read_byte_data(chip->client, PCA963X_MODE1); in pca963x_suspend()
391 i2c_smbus_write_byte_data(chip->client, PCA963X_MODE1, reg); in pca963x_suspend()
401 reg = i2c_smbus_read_byte_data(chip->client, PCA963X_MODE1); in pca963x_resume()
403 i2c_smbus_write_byte_data(chip->client, PCA963X_MODE1, reg); in pca963x_resume()
422 struct device *dev = &client->dev; in pca963x_probe()
427 chipdef = &pca963x_chipdefs[id->driver_data]; in pca963x_probe()
430 if (!count || count > chipdef->n_leds) { in pca963x_probe()
432 dev_fwnode(dev), chipdef->n_leds); in pca963x_probe()
433 return -EINVAL; in pca963x_probe()
438 return -ENOMEM; in pca963x_probe()
442 mutex_init(&chip->mutex); in pca963x_probe()
443 chip->chipdef = chipdef; in pca963x_probe()
444 chip->client = client; in pca963x_probe()
447 for (i = 0; i < chipdef->n_leds / 4; i++) in pca963x_probe()
448 i2c_smbus_write_byte_data(client, chipdef->ledout_base + i, 0x00); in pca963x_probe()
450 /* Disable LED all-call address, and power down initially */ in pca963x_probe()
451 i2c_smbus_write_byte_data(client, PCA963X_MODE1, BIT(4)); in pca963x_probe()
458 .name = "leds-pca963x",
468 MODULE_AUTHOR("Peter Meerwald <p.meerwald@bct-electronic.com>");
469 MODULE_DESCRIPTION("PCA963X LED driver");