leds: Add snapshot of QTI Tri-LED driver

The QTI Tri-LED driver enables the configuration of the output of LED
channels in the TRI_LED and HR_LED peripherals found in QTI PMICs.

This is a snapshot of the driver taken from msm-4.19 as of commit
cf2cbb63fb60 ("Merge "input: touchscreen: synaptics_tcm: enable touch
driver"") with the following modifications:
  - Remove the setting of struct pwm_output_pattern to NULL in
    __tri_led_config_pwm()
  - Remove LED_KEEP_TRIGGER from qpnp_tri_led_register()
  - Add a MODULE_SOFTDEP link to the QTI PWM LPG driver so that modprobe
    can enforce the functional dependency on it.

Change-Id: I7ceb2ff1a9ed1fec09651744ec2e7b2ba73f1bae
Signed-off-by: Guru Das Srinagesh <gurus@codeaurora.org>
This commit is contained in:
Guru Das Srinagesh 2019-11-25 12:09:32 -08:00
parent f0ae1932b0
commit 67222c1d60
3 changed files with 655 additions and 0 deletions

View File

@ -823,6 +823,18 @@ config LEDS_LM36274
Say Y to enable the LM36274 LED driver for TI LMU devices.
This supports the LED device LM36274.
config LEDS_QTI_TRI_LED
tristate "LED support for Qualcomm Technologies, Inc. TRI_LED"
depends on LEDS_CLASS && MFD_SPMI_PMIC && PWM && OF
help
This driver supports the TRI_LED module found in Qualcomm
Technologies, Inc. PMIC chips. TRI_LED supports 3 LED drivers
at max and each is controlled by a PWM channel used for dimming
or blinking.
To compile this driver as a module, choose M here: the module
will be called leds-qti-tri-led.
comment "LED Triggers"
source "drivers/leds/trigger/Kconfig"

View File

@ -85,6 +85,7 @@ obj-$(CONFIG_LEDS_LM3601X) += leds-lm3601x.o
obj-$(CONFIG_LEDS_TI_LMU_COMMON) += leds-ti-lmu-common.o
obj-$(CONFIG_LEDS_LM3697) += leds-lm3697.o
obj-$(CONFIG_LEDS_LM36274) += leds-lm36274.o
obj-$(CONFIG_LEDS_QTI_TRI_LED) += leds-qti-tri-led.o
# LED SPI Drivers
obj-$(CONFIG_LEDS_CR0014114) += leds-cr0014114.o

View File

@ -0,0 +1,642 @@
// SPDX-License-Identifier: GPL-2.0-only
/*
* Copyright (c) 2018-2019, The Linux Foundation. All rights reserved.
*/
#include <linux/bitops.h>
#include <linux/device.h>
#include <linux/err.h>
#include <linux/init.h>
#include <linux/kernel.h>
#include <linux/leds.h>
#include <linux/module.h>
#include <linux/mutex.h>
#include <linux/nvmem-consumer.h>
#include <linux/of.h>
#include <linux/of_address.h>
#include <linux/platform_device.h>
#include <linux/pwm.h>
#include <linux/regmap.h>
#include <linux/types.h>
#define TRILED_REG_TYPE 0x04
#define TRILED_REG_SUBTYPE 0x05
#define TRILED_REG_EN_CTL 0x46
/* TRILED_REG_EN_CTL */
#define TRILED_EN_CTL_MASK GENMASK(7, 5)
#define TRILED_EN_CTL_MAX_BIT 7
#define TRILED_TYPE 0x19
#define TRILED_SUBTYPE_LED3H0L12 0x02
#define TRILED_SUBTYPE_LED2H0L12 0x03
#define TRILED_SUBTYPE_LED1H2L12 0x04
#define TRILED_NUM_MAX 3
#define PWM_PERIOD_DEFAULT_NS 1000000
struct pwm_setting {
u64 pre_period_ns;
u64 period_ns;
u64 duty_ns;
};
struct led_setting {
u64 on_ms;
u64 off_ms;
enum led_brightness brightness;
bool blink;
bool breath;
};
struct qpnp_led_dev {
struct led_classdev cdev;
struct pwm_device *pwm_dev;
struct pwm_setting pwm_setting;
struct led_setting led_setting;
struct qpnp_tri_led_chip *chip;
struct mutex lock;
const char *label;
const char *default_trigger;
u8 id;
bool blinking;
bool breathing;
};
struct qpnp_tri_led_chip {
struct device *dev;
struct regmap *regmap;
struct qpnp_led_dev *leds;
struct nvmem_device *pbs_nvmem;
struct mutex bus_lock;
int num_leds;
u16 reg_base;
u8 subtype;
u8 bitmap;
};
static int qpnp_tri_led_read(struct qpnp_tri_led_chip *chip, u16 addr, u8 *val)
{
int rc;
unsigned int tmp;
mutex_lock(&chip->bus_lock);
rc = regmap_read(chip->regmap, chip->reg_base + addr, &tmp);
if (rc < 0)
dev_err(chip->dev, "Read addr 0x%x failed, rc=%d\n", addr, rc);
else
*val = (u8)tmp;
mutex_unlock(&chip->bus_lock);
return rc;
}
static int qpnp_tri_led_masked_write(struct qpnp_tri_led_chip *chip,
u16 addr, u8 mask, u8 val)
{
int rc;
mutex_lock(&chip->bus_lock);
rc = regmap_update_bits(chip->regmap, chip->reg_base + addr, mask, val);
if (rc < 0)
dev_err(chip->dev, "Update addr 0x%x to val 0x%x with mask 0x%x failed, rc=%d\n",
addr, val, mask, rc);
mutex_unlock(&chip->bus_lock);
return rc;
}
static int __tri_led_config_pwm(struct qpnp_led_dev *led,
struct pwm_setting *pwm)
{
struct pwm_state pstate;
int rc;
pwm_get_state(led->pwm_dev, &pstate);
pstate.enabled = !!(pwm->duty_ns != 0);
pstate.period = pwm->period_ns;
pstate.duty_cycle = pwm->duty_ns;
pstate.output_type = led->led_setting.breath ?
PWM_OUTPUT_MODULATED : PWM_OUTPUT_FIXED;
rc = pwm_apply_state(led->pwm_dev, &pstate);
if (rc < 0)
dev_err(led->chip->dev, "Apply PWM state for %s led failed, rc=%d\n",
led->cdev.name, rc);
return rc;
}
#define PBS_ENABLE 1
#define PBS_DISABLE 2
#define PBS_ARG 0x42
#define PBS_TRIG_CLR 0xE6
#define PBS_TRIG_SET 0xE5
static int __tri_led_set(struct qpnp_led_dev *led)
{
int rc = 0;
u8 val = 0, mask = 0, pbs_val;
u8 prev_bitmap;
rc = __tri_led_config_pwm(led, &led->pwm_setting);
if (rc < 0) {
dev_err(led->chip->dev, "Configure PWM for %s led failed, rc=%d\n",
led->cdev.name, rc);
return rc;
}
mask |= 1 << (TRILED_EN_CTL_MAX_BIT - led->id);
if (led->pwm_setting.duty_ns == 0)
val = 0;
else
val = mask;
if (led->chip->subtype == TRILED_SUBTYPE_LED2H0L12 &&
led->chip->pbs_nvmem) {
/*
* Control BOB_CONFIG_EXT_CTRL2_FORCE_EN for HR_LED through
* PBS trigger. PBS trigger for enable happens if any one of
* LEDs are turned on. PBS trigger for disable happens only
* if both LEDs are turned off.
*/
prev_bitmap = led->chip->bitmap;
if (val)
led->chip->bitmap |= (1 << led->id);
else
led->chip->bitmap &= ~(1 << led->id);
if (!(led->chip->bitmap & prev_bitmap)) {
pbs_val = led->chip->bitmap ? PBS_ENABLE : PBS_DISABLE;
rc = nvmem_device_write(led->chip->pbs_nvmem, PBS_ARG,
1, &pbs_val);
if (rc < 0) {
dev_err(led->chip->dev, "Couldn't set PBS_ARG, rc=%d\n",
rc);
return rc;
}
pbs_val = 1;
rc = nvmem_device_write(led->chip->pbs_nvmem,
PBS_TRIG_CLR, 1, &pbs_val);
if (rc < 0) {
dev_err(led->chip->dev, "Couldn't set PBS_TRIG_CLR, rc=%d\n",
rc);
return rc;
}
pbs_val = 1;
rc = nvmem_device_write(led->chip->pbs_nvmem,
PBS_TRIG_SET, 1, &pbs_val);
if (rc < 0) {
dev_err(led->chip->dev, "Couldn't set PBS_TRIG_SET, rc=%d\n",
rc);
return rc;
}
}
}
rc = qpnp_tri_led_masked_write(led->chip, TRILED_REG_EN_CTL,
mask, val);
if (rc < 0)
dev_err(led->chip->dev, "Update addr 0x%x failed, rc=%d\n",
TRILED_REG_EN_CTL, rc);
return rc;
}
static int qpnp_tri_led_set(struct qpnp_led_dev *led)
{
u64 on_ms, off_ms, period_ns, duty_ns;
enum led_brightness brightness = led->led_setting.brightness;
int rc = 0;
if (led->led_setting.blink) {
on_ms = led->led_setting.on_ms;
off_ms = led->led_setting.off_ms;
duty_ns = on_ms * NSEC_PER_MSEC;
period_ns = (on_ms + off_ms) * NSEC_PER_MSEC;
if (period_ns < duty_ns && duty_ns != 0)
period_ns = duty_ns + 1;
} else {
/* Use initial period if no blinking is required */
period_ns = led->pwm_setting.pre_period_ns;
if (brightness == LED_OFF)
duty_ns = 0;
duty_ns = period_ns * brightness;
do_div(duty_ns, LED_FULL);
if (period_ns < duty_ns && duty_ns != 0)
period_ns = duty_ns + 1;
}
dev_dbg(led->chip->dev, "PWM settings for %s led: period = %lluns, duty = %lluns\n",
led->cdev.name, period_ns, duty_ns);
led->pwm_setting.duty_ns = duty_ns;
led->pwm_setting.period_ns = period_ns;
rc = __tri_led_set(led);
if (rc < 0) {
dev_err(led->chip->dev, "__tri_led_set %s failed, rc=%d\n",
led->cdev.name, rc);
return rc;
}
if (led->led_setting.blink) {
led->cdev.brightness = LED_FULL;
led->blinking = true;
led->breathing = false;
} else if (led->led_setting.breath) {
led->cdev.brightness = LED_FULL;
led->blinking = false;
led->breathing = true;
} else {
led->cdev.brightness = led->led_setting.brightness;
led->blinking = false;
led->breathing = false;
}
return rc;
}
static int qpnp_tri_led_set_brightness(struct led_classdev *led_cdev,
enum led_brightness brightness)
{
struct qpnp_led_dev *led =
container_of(led_cdev, struct qpnp_led_dev, cdev);
int rc = 0;
mutex_lock(&led->lock);
if (brightness > LED_FULL)
brightness = LED_FULL;
if (brightness == led->led_setting.brightness &&
!led->blinking && !led->breathing) {
mutex_unlock(&led->lock);
return 0;
}
led->led_setting.brightness = brightness;
if (!!brightness)
led->led_setting.off_ms = 0;
else
led->led_setting.on_ms = 0;
led->led_setting.blink = false;
led->led_setting.breath = false;
rc = qpnp_tri_led_set(led);
if (rc)
dev_err(led->chip->dev, "Set led failed for %s, rc=%d\n",
led->label, rc);
mutex_unlock(&led->lock);
return rc;
}
static enum led_brightness qpnp_tri_led_get_brightness(
struct led_classdev *led_cdev)
{
return led_cdev->brightness;
}
static int qpnp_tri_led_set_blink(struct led_classdev *led_cdev,
unsigned long *on_ms, unsigned long *off_ms)
{
struct qpnp_led_dev *led =
container_of(led_cdev, struct qpnp_led_dev, cdev);
int rc = 0;
mutex_lock(&led->lock);
if (led->blinking && *on_ms == led->led_setting.on_ms &&
*off_ms == led->led_setting.off_ms) {
dev_dbg(led_cdev->dev, "Ignore, on/off setting is not changed: on %lums, off %lums\n",
*on_ms, *off_ms);
mutex_unlock(&led->lock);
return 0;
}
if (*on_ms == 0) {
led->led_setting.blink = false;
led->led_setting.breath = false;
led->led_setting.brightness = LED_OFF;
} else if (*off_ms == 0) {
led->led_setting.blink = false;
led->led_setting.breath = false;
led->led_setting.brightness = led->cdev.brightness;
} else {
led->led_setting.on_ms = *on_ms;
led->led_setting.off_ms = *off_ms;
led->led_setting.blink = true;
led->led_setting.breath = false;
}
rc = qpnp_tri_led_set(led);
if (rc)
dev_err(led->chip->dev, "Set led failed for %s, rc=%d\n",
led->label, rc);
mutex_unlock(&led->lock);
return rc;
}
static ssize_t breath_show(struct device *dev, struct device_attribute *attr,
char *buf)
{
struct led_classdev *led_cdev = dev_get_drvdata(dev);
struct qpnp_led_dev *led =
container_of(led_cdev, struct qpnp_led_dev, cdev);
return scnprintf(buf, PAGE_SIZE, "%d\n", led->led_setting.breath);
}
static ssize_t breath_store(struct device *dev, struct device_attribute *attr,
const char *buf, size_t count)
{
int rc;
bool breath;
struct led_classdev *led_cdev = dev_get_drvdata(dev);
struct qpnp_led_dev *led =
container_of(led_cdev, struct qpnp_led_dev, cdev);
rc = kstrtobool(buf, &breath);
if (rc < 0)
return rc;
cancel_work_sync(&led_cdev->set_brightness_work);
mutex_lock(&led->lock);
if (led->breathing == breath)
goto unlock;
led->led_setting.blink = false;
led->led_setting.breath = breath;
led->led_setting.brightness = breath ? LED_FULL : LED_OFF;
rc = qpnp_tri_led_set(led);
if (rc < 0)
dev_err(led->chip->dev, "Set led failed for %s, rc=%d\n",
led->label, rc);
unlock:
mutex_unlock(&led->lock);
return (rc < 0) ? rc : count;
}
static DEVICE_ATTR_RW(breath);
static const struct attribute *breath_attrs[] = {
&dev_attr_breath.attr,
NULL
};
static int qpnp_tri_led_register(struct qpnp_tri_led_chip *chip)
{
struct qpnp_led_dev *led;
int rc, i, j;
for (i = 0; i < chip->num_leds; i++) {
led = &chip->leds[i];
mutex_init(&led->lock);
led->cdev.name = led->label;
led->cdev.max_brightness = LED_FULL;
led->cdev.brightness_set_blocking = qpnp_tri_led_set_brightness;
led->cdev.brightness_get = qpnp_tri_led_get_brightness;
led->cdev.blink_set = qpnp_tri_led_set_blink;
led->cdev.default_trigger = led->default_trigger;
led->cdev.brightness = LED_OFF;
rc = devm_led_classdev_register(chip->dev, &led->cdev);
if (rc < 0) {
dev_err(chip->dev, "%s led class device registering failed, rc=%d\n",
led->label, rc);
goto err_out;
}
if (pwm_get_output_type_supported(led->pwm_dev)
& PWM_OUTPUT_MODULATED) {
rc = sysfs_create_files(&led->cdev.dev->kobj,
breath_attrs);
if (rc < 0) {
dev_err(chip->dev, "Create breath file for %s led failed, rc=%d\n",
led->label, rc);
goto err_out;
}
}
}
return 0;
err_out:
for (j = 0; j <= i; j++) {
if (j < i)
sysfs_remove_files(&chip->leds[j].cdev.dev->kobj,
breath_attrs);
mutex_destroy(&chip->leds[j].lock);
}
return rc;
}
static int qpnp_tri_led_hw_init(struct qpnp_tri_led_chip *chip)
{
int rc = 0;
u8 val;
rc = qpnp_tri_led_read(chip, TRILED_REG_TYPE, &val);
if (rc < 0) {
dev_err(chip->dev, "Read REG_TYPE failed, rc=%d\n", rc);
return rc;
}
if (val != TRILED_TYPE) {
dev_err(chip->dev, "invalid subtype(%d)\n", val);
return -ENODEV;
}
rc = qpnp_tri_led_read(chip, TRILED_REG_SUBTYPE, &val);
if (rc < 0) {
dev_err(chip->dev, "Read REG_SUBTYPE failed, rc=%d\n", rc);
return rc;
}
chip->subtype = val;
return 0;
}
static int qpnp_tri_led_parse_dt(struct qpnp_tri_led_chip *chip)
{
struct device_node *node = chip->dev->of_node, *child_node;
struct qpnp_led_dev *led;
struct pwm_args pargs;
const __be32 *addr;
int rc = 0, id, i = 0;
addr = of_get_address(chip->dev->of_node, 0, NULL, NULL);
if (!addr) {
dev_err(chip->dev, "Getting address failed\n");
return -EINVAL;
}
chip->reg_base = be32_to_cpu(addr[0]);
chip->num_leds = of_get_available_child_count(node);
if (chip->num_leds == 0) {
dev_err(chip->dev, "No led child node defined\n");
return -ENODEV;
}
if (chip->num_leds > TRILED_NUM_MAX) {
dev_err(chip->dev, "can't support %d leds(max %d)\n",
chip->num_leds, TRILED_NUM_MAX);
return -EINVAL;
}
if (of_find_property(chip->dev->of_node, "nvmem", NULL)) {
chip->pbs_nvmem = devm_nvmem_device_get(chip->dev, "pbs_sdam");
if (IS_ERR_OR_NULL(chip->pbs_nvmem)) {
rc = PTR_ERR(chip->pbs_nvmem);
if (rc != -EPROBE_DEFER) {
dev_err(chip->dev, "Couldn't get nvmem device, rc=%d\n",
rc);
return -ENODEV;
}
chip->pbs_nvmem = NULL;
return rc;
}
}
chip->leds = devm_kcalloc(chip->dev, chip->num_leds,
sizeof(struct qpnp_led_dev), GFP_KERNEL);
if (!chip->leds)
return -ENOMEM;
for_each_available_child_of_node(node, child_node) {
rc = of_property_read_u32(child_node, "led-sources", &id);
if (rc) {
dev_err(chip->dev, "Get led-sources failed, rc=%d\n",
rc);
return rc;
}
if (id >= TRILED_NUM_MAX) {
dev_err(chip->dev, "only support 0~%d current source\n",
TRILED_NUM_MAX - 1);
return -EINVAL;
}
led = &chip->leds[i++];
led->chip = chip;
led->id = id;
led->label =
of_get_property(child_node, "label", NULL) ? :
child_node->name;
led->pwm_dev =
devm_of_pwm_get(chip->dev, child_node, NULL);
if (IS_ERR(led->pwm_dev)) {
rc = PTR_ERR(led->pwm_dev);
if (rc != -EPROBE_DEFER)
dev_err(chip->dev, "Get pwm device for %s led failed, rc=%d\n",
led->label, rc);
return rc;
}
pwm_get_args(led->pwm_dev, &pargs);
if (pargs.period == 0)
led->pwm_setting.pre_period_ns = PWM_PERIOD_DEFAULT_NS;
else
led->pwm_setting.pre_period_ns = pargs.period;
led->default_trigger = of_get_property(child_node,
"linux,default-trigger", NULL);
}
return rc;
}
static int qpnp_tri_led_probe(struct platform_device *pdev)
{
struct qpnp_tri_led_chip *chip;
int rc = 0;
chip = devm_kzalloc(&pdev->dev, sizeof(*chip), GFP_KERNEL);
if (!chip)
return -ENOMEM;
chip->dev = &pdev->dev;
chip->regmap = dev_get_regmap(chip->dev->parent, NULL);
if (!chip->regmap) {
dev_err(chip->dev, "Getting regmap failed\n");
return -EINVAL;
}
rc = qpnp_tri_led_parse_dt(chip);
if (rc < 0) {
dev_err(chip->dev, "Devicetree properties parsing failed, rc=%d\n",
rc);
return rc;
}
mutex_init(&chip->bus_lock);
rc = qpnp_tri_led_hw_init(chip);
if (rc) {
dev_err(chip->dev, "HW initialization failed, rc=%d\n", rc);
goto destroy;
}
dev_set_drvdata(chip->dev, chip);
rc = qpnp_tri_led_register(chip);
if (rc < 0) {
dev_err(chip->dev, "Registering LED class devices failed, rc=%d\n",
rc);
goto destroy;
}
dev_dbg(chip->dev, "Tri-led module with subtype 0x%x is detected\n",
chip->subtype);
return 0;
destroy:
mutex_destroy(&chip->bus_lock);
dev_set_drvdata(chip->dev, NULL);
return rc;
}
static int qpnp_tri_led_remove(struct platform_device *pdev)
{
int i;
struct qpnp_tri_led_chip *chip = dev_get_drvdata(&pdev->dev);
mutex_destroy(&chip->bus_lock);
for (i = 0; i < chip->num_leds; i++) {
sysfs_remove_files(&chip->leds[i].cdev.dev->kobj, breath_attrs);
mutex_destroy(&chip->leds[i].lock);
}
dev_set_drvdata(chip->dev, NULL);
return 0;
}
static const struct of_device_id qpnp_tri_led_of_match[] = {
{ .compatible = "qcom,tri-led",},
{ },
};
static struct platform_driver qpnp_tri_led_driver = {
.driver = {
.name = "leds-qti-tri-led",
.of_match_table = qpnp_tri_led_of_match,
},
.probe = qpnp_tri_led_probe,
.remove = qpnp_tri_led_remove,
};
module_platform_driver(qpnp_tri_led_driver);
MODULE_DESCRIPTION("QTI TRI_LED driver");
MODULE_LICENSE("GPL v2");
MODULE_SOFTDEP("pre: pwm-qti-lpg");