162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-only
262306a36Sopenharmony_ci/*
362306a36Sopenharmony_ci * linux/drivers/leds-pwm.c
462306a36Sopenharmony_ci *
562306a36Sopenharmony_ci * simple PWM based LED control
662306a36Sopenharmony_ci *
762306a36Sopenharmony_ci * Copyright 2009 Luotao Fu @ Pengutronix (l.fu@pengutronix.de)
862306a36Sopenharmony_ci *
962306a36Sopenharmony_ci * based on leds-gpio.c by Raphael Assenat <raph@8d.com>
1062306a36Sopenharmony_ci */
1162306a36Sopenharmony_ci
1262306a36Sopenharmony_ci#include <linux/module.h>
1362306a36Sopenharmony_ci#include <linux/kernel.h>
1462306a36Sopenharmony_ci#include <linux/platform_device.h>
1562306a36Sopenharmony_ci#include <linux/of.h>
1662306a36Sopenharmony_ci#include <linux/leds.h>
1762306a36Sopenharmony_ci#include <linux/err.h>
1862306a36Sopenharmony_ci#include <linux/pwm.h>
1962306a36Sopenharmony_ci#include <linux/slab.h>
2062306a36Sopenharmony_ci#include "leds.h"
2162306a36Sopenharmony_ci
2262306a36Sopenharmony_cistruct led_pwm {
2362306a36Sopenharmony_ci	const char	*name;
2462306a36Sopenharmony_ci	u8		active_low;
2562306a36Sopenharmony_ci	u8		default_state;
2662306a36Sopenharmony_ci	unsigned int	max_brightness;
2762306a36Sopenharmony_ci};
2862306a36Sopenharmony_ci
2962306a36Sopenharmony_cistruct led_pwm_data {
3062306a36Sopenharmony_ci	struct led_classdev	cdev;
3162306a36Sopenharmony_ci	struct pwm_device	*pwm;
3262306a36Sopenharmony_ci	struct pwm_state	pwmstate;
3362306a36Sopenharmony_ci	unsigned int		active_low;
3462306a36Sopenharmony_ci};
3562306a36Sopenharmony_ci
3662306a36Sopenharmony_cistruct led_pwm_priv {
3762306a36Sopenharmony_ci	int num_leds;
3862306a36Sopenharmony_ci	struct led_pwm_data leds[];
3962306a36Sopenharmony_ci};
4062306a36Sopenharmony_ci
4162306a36Sopenharmony_cistatic int led_pwm_set(struct led_classdev *led_cdev,
4262306a36Sopenharmony_ci		       enum led_brightness brightness)
4362306a36Sopenharmony_ci{
4462306a36Sopenharmony_ci	struct led_pwm_data *led_dat =
4562306a36Sopenharmony_ci		container_of(led_cdev, struct led_pwm_data, cdev);
4662306a36Sopenharmony_ci	unsigned int max = led_dat->cdev.max_brightness;
4762306a36Sopenharmony_ci	unsigned long long duty = led_dat->pwmstate.period;
4862306a36Sopenharmony_ci
4962306a36Sopenharmony_ci	duty *= brightness;
5062306a36Sopenharmony_ci	do_div(duty, max);
5162306a36Sopenharmony_ci
5262306a36Sopenharmony_ci	if (led_dat->active_low)
5362306a36Sopenharmony_ci		duty = led_dat->pwmstate.period - duty;
5462306a36Sopenharmony_ci
5562306a36Sopenharmony_ci	led_dat->pwmstate.duty_cycle = duty;
5662306a36Sopenharmony_ci	led_dat->pwmstate.enabled = true;
5762306a36Sopenharmony_ci	return pwm_apply_state(led_dat->pwm, &led_dat->pwmstate);
5862306a36Sopenharmony_ci}
5962306a36Sopenharmony_ci
6062306a36Sopenharmony_ci__attribute__((nonnull))
6162306a36Sopenharmony_cistatic int led_pwm_add(struct device *dev, struct led_pwm_priv *priv,
6262306a36Sopenharmony_ci		       struct led_pwm *led, struct fwnode_handle *fwnode)
6362306a36Sopenharmony_ci{
6462306a36Sopenharmony_ci	struct led_pwm_data *led_data = &priv->leds[priv->num_leds];
6562306a36Sopenharmony_ci	struct led_init_data init_data = { .fwnode = fwnode };
6662306a36Sopenharmony_ci	int ret;
6762306a36Sopenharmony_ci
6862306a36Sopenharmony_ci	led_data->active_low = led->active_low;
6962306a36Sopenharmony_ci	led_data->cdev.name = led->name;
7062306a36Sopenharmony_ci	led_data->cdev.brightness = LED_OFF;
7162306a36Sopenharmony_ci	led_data->cdev.max_brightness = led->max_brightness;
7262306a36Sopenharmony_ci	led_data->cdev.flags = LED_CORE_SUSPENDRESUME;
7362306a36Sopenharmony_ci
7462306a36Sopenharmony_ci	led_data->pwm = devm_fwnode_pwm_get(dev, fwnode, NULL);
7562306a36Sopenharmony_ci	if (IS_ERR(led_data->pwm))
7662306a36Sopenharmony_ci		return dev_err_probe(dev, PTR_ERR(led_data->pwm),
7762306a36Sopenharmony_ci				     "unable to request PWM for %s\n",
7862306a36Sopenharmony_ci				     led->name);
7962306a36Sopenharmony_ci
8062306a36Sopenharmony_ci	led_data->cdev.brightness_set_blocking = led_pwm_set;
8162306a36Sopenharmony_ci
8262306a36Sopenharmony_ci	/* init PWM state */
8362306a36Sopenharmony_ci	switch (led->default_state) {
8462306a36Sopenharmony_ci	case LEDS_DEFSTATE_KEEP:
8562306a36Sopenharmony_ci		pwm_get_state(led_data->pwm, &led_data->pwmstate);
8662306a36Sopenharmony_ci		if (led_data->pwmstate.period)
8762306a36Sopenharmony_ci			break;
8862306a36Sopenharmony_ci		led->default_state = LEDS_DEFSTATE_OFF;
8962306a36Sopenharmony_ci		dev_warn(dev,
9062306a36Sopenharmony_ci			"failed to read period for %s, default to off",
9162306a36Sopenharmony_ci			led->name);
9262306a36Sopenharmony_ci		fallthrough;
9362306a36Sopenharmony_ci	default:
9462306a36Sopenharmony_ci		pwm_init_state(led_data->pwm, &led_data->pwmstate);
9562306a36Sopenharmony_ci		break;
9662306a36Sopenharmony_ci	}
9762306a36Sopenharmony_ci
9862306a36Sopenharmony_ci	/* set brightness */
9962306a36Sopenharmony_ci	switch (led->default_state) {
10062306a36Sopenharmony_ci	case LEDS_DEFSTATE_ON:
10162306a36Sopenharmony_ci		led_data->cdev.brightness = led->max_brightness;
10262306a36Sopenharmony_ci		break;
10362306a36Sopenharmony_ci	case LEDS_DEFSTATE_KEEP:
10462306a36Sopenharmony_ci		{
10562306a36Sopenharmony_ci		uint64_t brightness;
10662306a36Sopenharmony_ci
10762306a36Sopenharmony_ci		brightness = led->max_brightness;
10862306a36Sopenharmony_ci		brightness *= led_data->pwmstate.duty_cycle;
10962306a36Sopenharmony_ci		do_div(brightness, led_data->pwmstate.period);
11062306a36Sopenharmony_ci		led_data->cdev.brightness = brightness;
11162306a36Sopenharmony_ci		}
11262306a36Sopenharmony_ci		break;
11362306a36Sopenharmony_ci	}
11462306a36Sopenharmony_ci
11562306a36Sopenharmony_ci	ret = devm_led_classdev_register_ext(dev, &led_data->cdev, &init_data);
11662306a36Sopenharmony_ci	if (ret) {
11762306a36Sopenharmony_ci		dev_err(dev, "failed to register PWM led for %s: %d\n",
11862306a36Sopenharmony_ci			led->name, ret);
11962306a36Sopenharmony_ci		return ret;
12062306a36Sopenharmony_ci	}
12162306a36Sopenharmony_ci
12262306a36Sopenharmony_ci	if (led->default_state != LEDS_DEFSTATE_KEEP) {
12362306a36Sopenharmony_ci		ret = led_pwm_set(&led_data->cdev, led_data->cdev.brightness);
12462306a36Sopenharmony_ci		if (ret) {
12562306a36Sopenharmony_ci			dev_err(dev, "failed to set led PWM value for %s: %d",
12662306a36Sopenharmony_ci				led->name, ret);
12762306a36Sopenharmony_ci			return ret;
12862306a36Sopenharmony_ci		}
12962306a36Sopenharmony_ci	}
13062306a36Sopenharmony_ci
13162306a36Sopenharmony_ci	priv->num_leds++;
13262306a36Sopenharmony_ci	return 0;
13362306a36Sopenharmony_ci}
13462306a36Sopenharmony_ci
13562306a36Sopenharmony_cistatic int led_pwm_create_fwnode(struct device *dev, struct led_pwm_priv *priv)
13662306a36Sopenharmony_ci{
13762306a36Sopenharmony_ci	struct fwnode_handle *fwnode;
13862306a36Sopenharmony_ci	struct led_pwm led;
13962306a36Sopenharmony_ci	int ret;
14062306a36Sopenharmony_ci
14162306a36Sopenharmony_ci	device_for_each_child_node(dev, fwnode) {
14262306a36Sopenharmony_ci		memset(&led, 0, sizeof(led));
14362306a36Sopenharmony_ci
14462306a36Sopenharmony_ci		ret = fwnode_property_read_string(fwnode, "label", &led.name);
14562306a36Sopenharmony_ci		if (ret && is_of_node(fwnode))
14662306a36Sopenharmony_ci			led.name = to_of_node(fwnode)->name;
14762306a36Sopenharmony_ci
14862306a36Sopenharmony_ci		if (!led.name) {
14962306a36Sopenharmony_ci			ret = -EINVAL;
15062306a36Sopenharmony_ci			goto err_child_out;
15162306a36Sopenharmony_ci		}
15262306a36Sopenharmony_ci
15362306a36Sopenharmony_ci		led.active_low = fwnode_property_read_bool(fwnode,
15462306a36Sopenharmony_ci							   "active-low");
15562306a36Sopenharmony_ci		fwnode_property_read_u32(fwnode, "max-brightness",
15662306a36Sopenharmony_ci					 &led.max_brightness);
15762306a36Sopenharmony_ci
15862306a36Sopenharmony_ci		led.default_state = led_init_default_state_get(fwnode);
15962306a36Sopenharmony_ci
16062306a36Sopenharmony_ci		ret = led_pwm_add(dev, priv, &led, fwnode);
16162306a36Sopenharmony_ci		if (ret)
16262306a36Sopenharmony_ci			goto err_child_out;
16362306a36Sopenharmony_ci	}
16462306a36Sopenharmony_ci
16562306a36Sopenharmony_ci	return 0;
16662306a36Sopenharmony_ci
16762306a36Sopenharmony_cierr_child_out:
16862306a36Sopenharmony_ci	fwnode_handle_put(fwnode);
16962306a36Sopenharmony_ci	return ret;
17062306a36Sopenharmony_ci}
17162306a36Sopenharmony_ci
17262306a36Sopenharmony_cistatic int led_pwm_probe(struct platform_device *pdev)
17362306a36Sopenharmony_ci{
17462306a36Sopenharmony_ci	struct led_pwm_priv *priv;
17562306a36Sopenharmony_ci	int ret = 0;
17662306a36Sopenharmony_ci	int count;
17762306a36Sopenharmony_ci
17862306a36Sopenharmony_ci	count = device_get_child_node_count(&pdev->dev);
17962306a36Sopenharmony_ci
18062306a36Sopenharmony_ci	if (!count)
18162306a36Sopenharmony_ci		return -EINVAL;
18262306a36Sopenharmony_ci
18362306a36Sopenharmony_ci	priv = devm_kzalloc(&pdev->dev, struct_size(priv, leds, count),
18462306a36Sopenharmony_ci			    GFP_KERNEL);
18562306a36Sopenharmony_ci	if (!priv)
18662306a36Sopenharmony_ci		return -ENOMEM;
18762306a36Sopenharmony_ci
18862306a36Sopenharmony_ci	ret = led_pwm_create_fwnode(&pdev->dev, priv);
18962306a36Sopenharmony_ci
19062306a36Sopenharmony_ci	if (ret)
19162306a36Sopenharmony_ci		return ret;
19262306a36Sopenharmony_ci
19362306a36Sopenharmony_ci	platform_set_drvdata(pdev, priv);
19462306a36Sopenharmony_ci
19562306a36Sopenharmony_ci	return 0;
19662306a36Sopenharmony_ci}
19762306a36Sopenharmony_ci
19862306a36Sopenharmony_cistatic const struct of_device_id of_pwm_leds_match[] = {
19962306a36Sopenharmony_ci	{ .compatible = "pwm-leds", },
20062306a36Sopenharmony_ci	{},
20162306a36Sopenharmony_ci};
20262306a36Sopenharmony_ciMODULE_DEVICE_TABLE(of, of_pwm_leds_match);
20362306a36Sopenharmony_ci
20462306a36Sopenharmony_cistatic struct platform_driver led_pwm_driver = {
20562306a36Sopenharmony_ci	.probe		= led_pwm_probe,
20662306a36Sopenharmony_ci	.driver		= {
20762306a36Sopenharmony_ci		.name	= "leds_pwm",
20862306a36Sopenharmony_ci		.of_match_table = of_pwm_leds_match,
20962306a36Sopenharmony_ci	},
21062306a36Sopenharmony_ci};
21162306a36Sopenharmony_ci
21262306a36Sopenharmony_cimodule_platform_driver(led_pwm_driver);
21362306a36Sopenharmony_ci
21462306a36Sopenharmony_ciMODULE_AUTHOR("Luotao Fu <l.fu@pengutronix.de>");
21562306a36Sopenharmony_ciMODULE_DESCRIPTION("generic PWM LED driver");
21662306a36Sopenharmony_ciMODULE_LICENSE("GPL v2");
21762306a36Sopenharmony_ciMODULE_ALIAS("platform:leds-pwm");
218