Commit 7fafb54c authored by Linus Torvalds's avatar Linus Torvalds

Merge tag 'leds-5.10-rc1' of git://git.kernel.org/pub/scm/linux/kernel/git/pavel/linux-leds

Pull LED updates from Pavel Machek:
 "Quite a lot of stuff is going on here. Great cleanups/fixes from Marek
  and others are biggest part.

  I limited CPU LED trigger to 8 LEDs, because it was willing to
  register 1024 'triggers' on machine with 1024 CPUs. I don't believe it
  will cause any problems, but we can raise the limit if it does"

* tag 'leds-5.10-rc1' of git://git.kernel.org/pub/scm/linux/kernel/git/pavel/linux-leds: (84 commits)
  leds: pwm: Remove platform_data support
  leds: lm3697: Fix out-of-bound access
  leds: ns2: do not guard OF match pointer with of_match_ptr
  leds: ns2: convert to fwnode API
  leds: tlc591xx: fix leak of device node iterator
  leds: pca963x: use struct led_init_data when registering
  leds: pca963x: register LEDs immediately after parsing, get rid of platdata
  leds: tca6507: remove binding comment
  leds: tca6507: cosmetic change: use helper variable
  leds: tca6507: do not set GPIO names
  dt-bindings: leds: tca6507: convert to YAML
  ledtrig-cpu: Limit to 8 CPUs
  leds: TODO: Add documentation about possible subsystem improvements
  leds: pca9532: read pwm settings from device tree
  leds: pca9532: correct shift computation in pca9532_getled
  leds: lm36274: Fix warning for undefined parameters
  leds: lm3532: Fix warnings for undefined parameters
  leds: pca963x: use flexible array
  leds: pca963x: cosmetic: rename variables
  leds: pca963x: cosmetic: rename variables
  ...
parents 55e0500e 19d2e0ce
...@@ -16,6 +16,7 @@ Optional properties: ...@@ -16,6 +16,7 @@ Optional properties:
- audio-gain-db : audio gain selection for external analog modulation input. - audio-gain-db : audio gain selection for external analog modulation input.
Valid values: 0 - 21, step by 3 (rounded down) Valid values: 0 - 21, step by 3 (rounded down)
Default: 0 Default: 0
- shutdown-gpios : Specifier of the GPIO connected to SDB pin of the chip.
Each led is represented as a sub-node of the issi,is31fl319x device. Each led is represented as a sub-node of the issi,is31fl319x device.
There can be less leds subnodes than the chip can support but not more. There can be less leds subnodes than the chip can support but not more.
...@@ -44,6 +45,7 @@ fancy_leds: leds@65 { ...@@ -44,6 +45,7 @@ fancy_leds: leds@65 {
#address-cells = <1>; #address-cells = <1>;
#size-cells = <0>; #size-cells = <0>;
reg = <0x65>; reg = <0x65>;
shutdown-gpios = <&gpio0 11 GPIO_ACTIVE_HIGH>;
red_aux: led@1 { red_aux: led@1 {
label = "red:aux"; label = "red:aux";
......
# SPDX-License-Identifier: (GPL-2.0-only OR BSD-2-Clause)
%YAML 1.2
---
$id: http://devicetree.org/schemas/leds/leds-lp50xx.yaml#
$schema: http://devicetree.org/meta-schemas/core.yaml#
title: LED driver for LP50XX RGB LED from Texas Instruments.
maintainers:
- Dan Murphy <dmurphy@ti.com>
description: |
The LP50XX is multi-channel, I2C RGB LED Drivers that can group RGB LEDs into
a LED group or control them individually.
The difference in these RGB LED drivers is the number of supported RGB
modules.
For more product information please see the link below:
https://www.ti.com/lit/ds/symlink/lp5012.pdf
https://www.ti.com/lit/ds/symlink/lp5024.pdf
https://www.ti.com/lit/ds/symlink/lp5036.pdf
properties:
compatible:
enum:
- ti,lp5009
- ti,lp5012
- ti,lp5018
- ti,lp5024
- ti,lp5030
- ti,lp5036
reg:
maxItems: 1
description:
I2C slave address
lp5009/12 - 0x14, 0x15, 0x16, 0x17
lp5018/24 - 0x28, 0x29, 0x2a, 0x2b
lp5030/36 - 0x30, 0x31, 0x32, 0x33
enable-gpios:
maxItems: 1
description: GPIO pin to enable/disable the device.
vled-supply:
description: LED supply.
patternProperties:
'^multi-led@[0-9a-f]$':
type: object
allOf:
- $ref: leds-class-multicolor.yaml#
properties:
reg:
minItems: 1
maxItems: 12
description:
This property denotes the LED module number(s) that is used on the
for the child node. The LED modules can either be used stand alone
or grouped into a module bank.
patternProperties:
"(^led-[0-9a-f]$|led)":
type: object
$ref: common.yaml#
required:
- compatible
- reg
examples:
- |
#include <dt-bindings/gpio/gpio.h>
#include <dt-bindings/leds/common.h>
i2c {
#address-cells = <1>;
#size-cells = <0>;
led-controller@14 {
compatible = "ti,lp5009";
reg = <0x14>;
#address-cells = <1>;
#size-cells = <0>;
enable-gpios = <&gpio1 16>;
multi-led@1 {
#address-cells = <1>;
#size-cells = <0>;
reg = <0x1>;
color = <LED_COLOR_ID_RGB>;
function = LED_FUNCTION_CHARGING;
led-0 {
color = <LED_COLOR_ID_RED>;
};
led-1 {
color = <LED_COLOR_ID_GREEN>;
};
led-2 {
color = <LED_COLOR_ID_BLUE>;
};
};
multi-led@2 {
#address-cells = <1>;
#size-cells = <2>;
reg = <0x2 0x3 0x5>;
color = <LED_COLOR_ID_RGB>;
function = LED_FUNCTION_STANDBY;
led-6 {
color = <LED_COLOR_ID_RED>;
};
led-7 {
color = <LED_COLOR_ID_GREEN>;
};
led-8 {
color = <LED_COLOR_ID_BLUE>;
};
};
};
};
...
...@@ -189,7 +189,7 @@ examples: ...@@ -189,7 +189,7 @@ examples:
#address-cells = <1>; #address-cells = <1>;
#size-cells = <0>; #size-cells = <0>;
reg = <0x2>; reg = <0x2>;
color = <LED_COLOR_ID_MULTI>; color = <LED_COLOR_ID_RGB>;
function = LED_FUNCTION_STANDBY; function = LED_FUNCTION_STANDBY;
linux,default-trigger = "heartbeat"; linux,default-trigger = "heartbeat";
......
...@@ -9,6 +9,7 @@ Required properties: ...@@ -9,6 +9,7 @@ Required properties:
"nxp,pca9550" "nxp,pca9550"
"nxp,pca9551" "nxp,pca9551"
"nxp,pca9552" "nxp,pca9552"
"ibm,pca9552"
"nxp,pca9553" "nxp,pca9553"
- #address-cells: must be 1 - #address-cells: must be 1
- #size-cells: must be 0 - #size-cells: must be 0
......
LEDs connected to tca6507
Required properties:
- compatible : should be : "ti,tca6507".
- #address-cells: must be 1
- #size-cells: must be 0
- reg: typically 0x45.
Optional properties:
- gpio-controller: allows lines to be used as output-only GPIOs.
- #gpio-cells: if present, must not be 0.
Each led is represented as a sub-node of the ti,tca6507 device.
LED sub-node properties:
- label : (optional) see Documentation/devicetree/bindings/leds/common.txt
- reg : number of LED line (could be from 0 to 6)
- linux,default-trigger : (optional)
see Documentation/devicetree/bindings/leds/common.txt
- compatible: either "led" (the default) or "gpio".
Examples:
tca6507@45 {
compatible = "ti,tca6507";
#address-cells = <1>;
#size-cells = <0>;
reg = <0x45>;
gpio-controller;
#gpio-cells = <2>;
led0: red-aux@0 {
label = "red:aux";
reg = <0x0>;
};
led1: green-aux@1 {
label = "green:aux";
reg = <0x5>;
linux,default-trigger = "default-on";
};
wifi-reset@6 {
reg = <0x6>;
compatible = "gpio";
};
};
# SPDX-License-Identifier: GPL-2.0-only
%YAML 1.2
---
$id: http://devicetree.org/schemas/leds/ti,tca6507.yaml#
$schema: http://devicetree.org/meta-schemas/core.yaml#
title: TCA6507 LED and GPIO controller
maintainers:
- NeilBrown <neilb@suse.de>
description:
The TCA6507 is a programmable LED controller connected via I2C that can drive
7 separate lines either by holding them low, or by pulsing them with modulated
width.
properties:
compatible:
const: ti,tca6507
reg:
description: I2C slave address of the controller.
maxItems: 1
"#address-cells":
const: 1
"#size-cells":
const: 0
gpio-controller: true
"#gpio-cells":
const: 2
gpio-line-names: true
patternProperties:
"^led@[0-6]$":
type: object
$ref: common.yaml#
properties:
reg:
minimum: 0
maximum: 6
required:
- reg
"^gpio@[0-6]$":
type: object
properties:
compatible:
const: gpio
reg:
minimum: 0
maximum: 6
additionalProperties: false
required:
- reg
- compatible
if:
patternProperties:
"^gpio@[0-6]$":
properties:
compatible:
contains:
const: gpio
then:
required:
- gpio-controller
- "#gpio-cells"
additionalProperties: false
examples:
- |
#include <dt-bindings/gpio/gpio.h>
#include <dt-bindings/leds/common.h>
i2c0 {
#address-cells = <1>;
#size-cells = <0>;
led-controller@45 {
compatible = "ti,tca6507";
#address-cells = <1>;
#size-cells = <0>;
reg = <0x45>;
gpio-controller;
#gpio-cells = <2>;
gpio-line-names = "wifi_reset@6";
led@0 {
label = "gta04:red:aux";
reg = <0x0>;
};
led@1 {
label = "gta04:green:aux";
reg = <0x1>;
};
led@3 {
reg = <0x3>;
color = <LED_COLOR_ID_RED>;
function = LED_FUNCTION_POWER;
linux,default-trigger = "default-on";
};
led@4 {
color = <LED_COLOR_ID_GREEN>;
function = LED_FUNCTION_POWER;
reg = <0x4>;
};
gpio@6 {
compatible = "gpio";
reg = <0x6>;
};
};
};
...
...@@ -17,12 +17,6 @@ set a timer to hold a state, however when user space application crashes or ...@@ -17,12 +17,6 @@ set a timer to hold a state, however when user space application crashes or
goes away without deactivating the timer, the hardware will be left in that goes away without deactivating the timer, the hardware will be left in that
state permanently. state permanently.
As a specific example of this use-case, let's look at vibrate feature on
phones. Vibrate function on phones is implemented using PWM pins on SoC or
PMIC. There is a need to activate one shot timer to control the vibrate
feature, to prevent user space crashes leaving the phone in vibrate mode
permanently causing the battery to drain.
Transient trigger addresses the need for one shot timer activation. The Transient trigger addresses the need for one shot timer activation. The
transient trigger can be enabled and disabled just like the other leds transient trigger can be enabled and disabled just like the other leds
triggers. triggers.
...@@ -159,7 +153,6 @@ repeat the following step as needed:: ...@@ -159,7 +153,6 @@ repeat the following step as needed::
This trigger is intended to be used for the following example use cases: This trigger is intended to be used for the following example use cases:
- Control of vibrate (phones, tablets etc.) hardware by user space app.
- Use of LED by user space app as activity indicator. - Use of LED by user space app as activity indicator.
- Use of LED by user space app as a kind of watchdog indicator -- as - Use of LED by user space app as a kind of watchdog indicator -- as
long as the app is alive, it can keep the LED illuminated, if it dies long as the app is alive, it can keep the LED illuminated, if it dies
......
...@@ -274,7 +274,7 @@ config LEDS_MT6323 ...@@ -274,7 +274,7 @@ config LEDS_MT6323
config LEDS_S3C24XX config LEDS_S3C24XX
tristate "LED Support for Samsung S3C24XX GPIO LEDs" tristate "LED Support for Samsung S3C24XX GPIO LEDs"
depends on LEDS_CLASS depends on LEDS_CLASS
depends on ARCH_S3C24XX depends on ARCH_S3C24XX || COMPILE_TEST
help help
This option enables support for LEDs connected to GPIO lines This option enables support for LEDs connected to GPIO lines
on Samsung S3C24XX series CPUs, such as the S3C2410 and S3C2440. on Samsung S3C24XX series CPUs, such as the S3C2410 and S3C2440.
...@@ -304,13 +304,13 @@ config LEDS_WRAP ...@@ -304,13 +304,13 @@ config LEDS_WRAP
config LEDS_COBALT_QUBE config LEDS_COBALT_QUBE
tristate "LED Support for the Cobalt Qube series front LED" tristate "LED Support for the Cobalt Qube series front LED"
depends on LEDS_CLASS depends on LEDS_CLASS
depends on MIPS_COBALT depends on MIPS_COBALT || COMPILE_TEST
help help
This option enables support for the front LED on Cobalt Qube series This option enables support for the front LED on Cobalt Qube series
config LEDS_COBALT_RAQ config LEDS_COBALT_RAQ
bool "LED Support for the Cobalt Raq series" bool "LED Support for the Cobalt Raq series"
depends on LEDS_CLASS=y && MIPS_COBALT depends on LEDS_CLASS=y && (MIPS_COBALT || COMPILE_TEST)
select LEDS_TRIGGERS select LEDS_TRIGGERS
help help
This option enables support for the Cobalt Raq series LEDs. This option enables support for the Cobalt Raq series LEDs.
...@@ -395,8 +395,20 @@ config LEDS_LP3952 ...@@ -395,8 +395,20 @@ config LEDS_LP3952
To compile this driver as a module, choose M here: the To compile this driver as a module, choose M here: the
module will be called leds-lp3952. module will be called leds-lp3952.
config LEDS_LP50XX
tristate "LED Support for TI LP5036/30/24/18/12/9 LED driver chip"
depends on LEDS_CLASS && REGMAP_I2C
depends on LEDS_CLASS_MULTICOLOR || !LEDS_CLASS_MULTICOLOR
help
If you say yes here you get support for the Texas Instruments
LP5036, LP5030, LP5024, LP5018, LP5012 and LP5009 LED driver.
To compile this driver as a module, choose M here: the
module will be called leds-lp50xx.
config LEDS_LP55XX_COMMON config LEDS_LP55XX_COMMON
tristate "Common Driver for TI/National LP5521/5523/55231/5562/8501" tristate "Common Driver for TI/National LP5521/5523/55231/5562/8501"
depends on LEDS_CLASS
depends on LEDS_CLASS_MULTICOLOR || !LEDS_CLASS_MULTICOLOR depends on LEDS_CLASS_MULTICOLOR || !LEDS_CLASS_MULTICOLOR
depends on OF depends on OF
depends on I2C depends on I2C
...@@ -632,7 +644,7 @@ config LEDS_MC13783 ...@@ -632,7 +644,7 @@ config LEDS_MC13783
config LEDS_NS2 config LEDS_NS2
tristate "LED support for Network Space v2 GPIO LEDs" tristate "LED support for Network Space v2 GPIO LEDs"
depends on LEDS_CLASS depends on LEDS_CLASS
depends on MACH_KIRKWOOD || MACH_ARMADA_370 depends on MACH_KIRKWOOD || MACH_ARMADA_370 || COMPILE_TEST
default y default y
help help
This option enables support for the dual-GPIO LEDs found on the This option enables support for the dual-GPIO LEDs found on the
...@@ -646,7 +658,7 @@ config LEDS_NS2 ...@@ -646,7 +658,7 @@ config LEDS_NS2
config LEDS_NETXBIG config LEDS_NETXBIG
tristate "LED support for Big Network series LEDs" tristate "LED support for Big Network series LEDs"
depends on LEDS_CLASS depends on LEDS_CLASS
depends on MACH_KIRKWOOD depends on MACH_KIRKWOOD || COMPILE_TEST
depends on OF_GPIO depends on OF_GPIO
default y default y
help help
...@@ -893,7 +905,7 @@ config LEDS_TPS6105X ...@@ -893,7 +905,7 @@ config LEDS_TPS6105X
config LEDS_IP30 config LEDS_IP30
tristate "LED support for SGI Octane machines" tristate "LED support for SGI Octane machines"
depends on LEDS_CLASS depends on LEDS_CLASS
depends on SGI_MFD_IOC3 depends on SGI_MFD_IOC3 || COMPILE_TEST
help help
This option enables support for the Red and White LEDs of This option enables support for the Red and White LEDs of
SGI Octane machines. SGI Octane machines.
...@@ -909,6 +921,13 @@ config LEDS_SGM3140 ...@@ -909,6 +921,13 @@ config LEDS_SGM3140
This option enables support for the SGM3140 500mA Buck/Boost Charge This option enables support for the SGM3140 500mA Buck/Boost Charge
Pump LED Driver. Pump LED Driver.
config LEDS_ACER_A500
tristate "Power button LED support for Acer Iconia Tab A500"
depends on LEDS_CLASS && MFD_ACER_A500_EC
help
This option enables support for the Power Button LED of
Acer Iconia Tab A500.
comment "LED Triggers" comment "LED Triggers"
source "drivers/leds/trigger/Kconfig" source "drivers/leds/trigger/Kconfig"
......
...@@ -10,6 +10,7 @@ obj-$(CONFIG_LEDS_TRIGGERS) += led-triggers.o ...@@ -10,6 +10,7 @@ obj-$(CONFIG_LEDS_TRIGGERS) += led-triggers.o
# LED Platform Drivers (keep this sorted, M-| sort) # LED Platform Drivers (keep this sorted, M-| sort)
obj-$(CONFIG_LEDS_88PM860X) += leds-88pm860x.o obj-$(CONFIG_LEDS_88PM860X) += leds-88pm860x.o
obj-$(CONFIG_LEDS_AAT1290) += leds-aat1290.o obj-$(CONFIG_LEDS_AAT1290) += leds-aat1290.o
obj-$(CONFIG_LEDS_ACER_A500) += leds-acer-a500.o
obj-$(CONFIG_LEDS_ADP5520) += leds-adp5520.o obj-$(CONFIG_LEDS_ADP5520) += leds-adp5520.o
obj-$(CONFIG_LEDS_AN30259A) += leds-an30259a.o obj-$(CONFIG_LEDS_AN30259A) += leds-an30259a.o
obj-$(CONFIG_LEDS_APU) += leds-apu.o obj-$(CONFIG_LEDS_APU) += leds-apu.o
...@@ -49,6 +50,7 @@ obj-$(CONFIG_LEDS_LM3697) += leds-lm3697.o ...@@ -49,6 +50,7 @@ obj-$(CONFIG_LEDS_LM3697) += leds-lm3697.o
obj-$(CONFIG_LEDS_LOCOMO) += leds-locomo.o obj-$(CONFIG_LEDS_LOCOMO) += leds-locomo.o
obj-$(CONFIG_LEDS_LP3944) += leds-lp3944.o obj-$(CONFIG_LEDS_LP3944) += leds-lp3944.o
obj-$(CONFIG_LEDS_LP3952) += leds-lp3952.o obj-$(CONFIG_LEDS_LP3952) += leds-lp3952.o
obj-$(CONFIG_LEDS_LP50XX) += leds-lp50xx.o
obj-$(CONFIG_LEDS_LP5521) += leds-lp5521.o obj-$(CONFIG_LEDS_LP5521) += leds-lp5521.o
obj-$(CONFIG_LEDS_LP5523) += leds-lp5523.o obj-$(CONFIG_LEDS_LP5523) += leds-lp5523.o
obj-$(CONFIG_LEDS_LP5562) += leds-lp5562.o obj-$(CONFIG_LEDS_LP5562) += leds-lp5562.o
......
-*- org -*-
* On/off LEDs should have max_brightness of 1
* Get rid of enum led_brightness
It is really an integer, as maximum is configurable. Get rid of it, or
make it into typedef or something.
* Review atomicity requirements in LED subsystem
Calls that may and that may not block are mixed in same structure, and
semantics is sometimes non-intuitive. (For example blink callback may
not sleep.) Review the requirements for any bugs and document them
clearly.
* LED names are still a mess
No two LEDs have same name, so the names are probably unusable for the
userland. Nudge authors into creating common LED names for common
functionality.
? Perhaps check for known LED names during boot, and warn if there are
LEDs not on the list?
* Split drivers into subdirectories
The number of drivers is getting big, and driver for on/off LED on a
i/o port is really quite different from camera flash LED, which is
really different from driver for RGB color LED that can run its own
microcode. Split the drivers somehow.
* Figure out what to do with RGB leds
Multicolor is a bit too abstract. Yes, we can have
Green-Magenta-Ultraviolet LED, but so far all the LEDs we support are
RGB, and not even RGB-White or RGB-Yellow variants emerged.
Multicolor is not a good fit for RGB LED. It does not really know
about LED color. In particular, there's no way to make LED "white".
Userspace is interested in knowing "this LED can produce arbitrary
color", which not all multicolor LEDs can.
Proposal: let's add "rgb" to led_colors in drivers/leds/led-core.c,
add corresponding device tree defines, and use that, instead of
multicolor for RGB LEDs.
We really need to do that now; "white" stuff can wait.
RGB LEDs are quite common, and it would be good to be able to turn LED
white and to turn it into any arbitrary color. It is essential that
userspace is able to set arbitrary colors, and it might be good to
have that ability from kernel, too... to allow full-color triggers.
* Command line utility to manipulate the LEDs?
/sys interface is not really suitable to use by hand, should we have
an utility to perform LED control?
In particular, LED names are still a mess (see above) and utility
could help there by presenting both old and new names while we clean
them up.
In future, I'd like utility to accept both old and new names while we
clean them up.
It would be also nice to have useful listing mode -- name, type,
current brightness/trigger...
In future, it would be good to be able to set rgb led to particular
color.
And probably user-friendly interface to access LEDs for particular
ethernet interface would be nice.
...@@ -354,6 +354,11 @@ int led_classdev_register_ext(struct device *parent, ...@@ -354,6 +354,11 @@ int led_classdev_register_ext(struct device *parent,
ret = led_compose_name(parent, init_data, composed_name); ret = led_compose_name(parent, init_data, composed_name);
if (ret < 0) if (ret < 0)
return ret; return ret;
if (init_data->fwnode)
fwnode_property_read_string(init_data->fwnode,
"linux,default-trigger",
&led_cdev->default_trigger);
} else { } else {
proposed_name = led_cdev->name; proposed_name = led_cdev->name;
} }
......
...@@ -118,14 +118,14 @@ static int pm860x_led_dt_init(struct platform_device *pdev, ...@@ -118,14 +118,14 @@ static int pm860x_led_dt_init(struct platform_device *pdev,
struct device_node *nproot, *np; struct device_node *nproot, *np;
int iset = 0; int iset = 0;
if (!pdev->dev.parent->of_node) if (!dev_of_node(pdev->dev.parent))
return -ENODEV; return -ENODEV;
nproot = of_get_child_by_name(pdev->dev.parent->of_node, "leds"); nproot = of_get_child_by_name(dev_of_node(pdev->dev.parent), "leds");
if (!nproot) { if (!nproot) {
dev_err(&pdev->dev, "failed to find leds node\n"); dev_err(&pdev->dev, "failed to find leds node\n");
return -ENODEV; return -ENODEV;
} }
for_each_child_of_node(nproot, np) { for_each_available_child_of_node(nproot, np) {
if (of_node_name_eq(np, data->name)) { if (of_node_name_eq(np, data->name)) {
of_property_read_u32(np, "marvell,88pm860x-iset", of_property_read_u32(np, "marvell,88pm860x-iset",
&iset); &iset);
......
...@@ -248,7 +248,7 @@ static int aat1290_led_parse_dt(struct aat1290_led *led, ...@@ -248,7 +248,7 @@ static int aat1290_led_parse_dt(struct aat1290_led *led,
} }
#endif #endif
child_node = of_get_next_available_child(dev->of_node, NULL); child_node = of_get_next_available_child(dev_of_node(dev), NULL);
if (!child_node) { if (!child_node) {
dev_err(dev, "No DT child node found for connected LED.\n"); dev_err(dev, "No DT child node found for connected LED.\n");
return -EINVAL; return -EINVAL;
......
// SPDX-License-Identifier: GPL-2.0+
#include <linux/leds.h>
#include <linux/module.h>
#include <linux/platform_device.h>
#include <linux/regmap.h>
#define A500_EC_LED_DELAY_USEC (100 * 1000)
enum {
REG_RESET_LEDS = 0x40,
REG_POWER_LED_ON = 0x42,
REG_CHARGE_LED_ON = 0x43,
REG_ANDROID_LEDS_OFF = 0x5a,
};
struct a500_led {
struct led_classdev cdev;
const struct reg_sequence *enable_seq;
struct a500_led *other;
struct regmap *rmap;
};
static const struct reg_sequence a500_ec_leds_reset_seq[] = {
REG_SEQ(REG_RESET_LEDS, 0x0, A500_EC_LED_DELAY_USEC),
REG_SEQ(REG_ANDROID_LEDS_OFF, 0x0, A500_EC_LED_DELAY_USEC),
};
static const struct reg_sequence a500_ec_white_led_enable_seq[] = {
REG_SEQ(REG_POWER_LED_ON, 0x0, A500_EC_LED_DELAY_USEC),
};
static const struct reg_sequence a500_ec_orange_led_enable_seq[] = {
REG_SEQ(REG_CHARGE_LED_ON, 0x0, A500_EC_LED_DELAY_USEC),
};
static int a500_ec_led_brightness_set(struct led_classdev *led_cdev,
enum led_brightness value)
{
struct a500_led *led = container_of(led_cdev, struct a500_led, cdev);
struct reg_sequence control_seq[2];
unsigned int num_regs = 1;
if (value) {
control_seq[0] = led->enable_seq[0];
} else {
/*
* There is no separate controls which can disable LEDs
* individually, there is only RESET_LEDS command that turns
* off both LEDs.
*
* RESET_LEDS turns off both LEDs, thus restore other LED if
* it's turned ON.
*/
if (led->other->cdev.brightness)
num_regs = 2;
control_seq[0] = a500_ec_leds_reset_seq[0];
control_seq[1] = led->other->enable_seq[0];
}
return regmap_multi_reg_write(led->rmap, control_seq, num_regs);
}
static int a500_ec_leds_probe(struct platform_device *pdev)
{
struct a500_led *white_led, *orange_led;
struct regmap *rmap;
int err;
rmap = dev_get_regmap(pdev->dev.parent, "KB930");
if (!rmap)
return -EINVAL;
/* reset and turn off LEDs */
regmap_multi_reg_write(rmap, a500_ec_leds_reset_seq, 2);
white_led = devm_kzalloc(&pdev->dev, sizeof(*white_led), GFP_KERNEL);
if (!white_led)
return -ENOMEM;
white_led->cdev.name = "power:white";
white_led->cdev.brightness_set_blocking = a500_ec_led_brightness_set;
white_led->cdev.flags = LED_CORE_SUSPENDRESUME;
white_led->cdev.max_brightness = 1;
white_led->enable_seq = a500_ec_white_led_enable_seq;
white_led->rmap = rmap;
orange_led = devm_kzalloc(&pdev->dev, sizeof(*orange_led), GFP_KERNEL);
if (!orange_led)
return -ENOMEM;
orange_led->cdev.name = "power:orange";
orange_led->cdev.brightness_set_blocking = a500_ec_led_brightness_set;
orange_led->cdev.flags = LED_CORE_SUSPENDRESUME;
orange_led->cdev.max_brightness = 1;
orange_led->enable_seq = a500_ec_orange_led_enable_seq;
orange_led->rmap = rmap;
white_led->other = orange_led;
orange_led->other = white_led;
err = devm_led_classdev_register(&pdev->dev, &white_led->cdev);
if (err) {
dev_err(&pdev->dev, "failed to register white LED\n");
return err;
}
err = devm_led_classdev_register(&pdev->dev, &orange_led->cdev);
if (err) {
dev_err(&pdev->dev, "failed to register orange LED\n");
return err;
}
return 0;
}
static struct platform_driver a500_ec_leds_driver = {
.driver = {
.name = "acer-a500-iconia-leds",
},
.probe = a500_ec_leds_probe,
};
module_platform_driver(a500_ec_leds_driver);
MODULE_DESCRIPTION("LED driver for Acer Iconia Tab A500 Power Button");
MODULE_AUTHOR("Dmitry Osipenko <digetx@gmail.com>");
MODULE_ALIAS("platform:acer-a500-iconia-leds");
MODULE_LICENSE("GPL");
...@@ -202,13 +202,13 @@ static int an30259a_blink_set(struct led_classdev *cdev, ...@@ -202,13 +202,13 @@ static int an30259a_blink_set(struct led_classdev *cdev,
static int an30259a_dt_init(struct i2c_client *client, static int an30259a_dt_init(struct i2c_client *client,
struct an30259a *chip) struct an30259a *chip)
{ {
struct device_node *np = client->dev.of_node, *child; struct device_node *np = dev_of_node(&client->dev), *child;
int count, ret; int count, ret;
int i = 0; int i = 0;
const char *str; const char *str;
struct an30259a_led *led; struct an30259a_led *led;
count = of_get_child_count(np); count = of_get_available_child_count(np);
if (!count || count > AN30259A_MAX_LEDS) if (!count || count > AN30259A_MAX_LEDS)
return -EINVAL; return -EINVAL;
...@@ -238,9 +238,6 @@ static int an30259a_dt_init(struct i2c_client *client, ...@@ -238,9 +238,6 @@ static int an30259a_dt_init(struct i2c_client *client,
led->default_state = STATE_OFF; led->default_state = STATE_OFF;
} }
of_property_read_string(child, "linux,default-trigger",
&led->cdev.default_trigger);
i++; i++;
} }
......
...@@ -261,11 +261,11 @@ static int aw2013_blink_set(struct led_classdev *cdev, ...@@ -261,11 +261,11 @@ static int aw2013_blink_set(struct led_classdev *cdev,
static int aw2013_probe_dt(struct aw2013 *chip) static int aw2013_probe_dt(struct aw2013 *chip)
{ {
struct device_node *np = chip->client->dev.of_node, *child; struct device_node *np = dev_of_node(&chip->client->dev), *child;
int count, ret = 0, i = 0; int count, ret = 0, i = 0;
struct aw2013_led *led; struct aw2013_led *led;
count = of_get_child_count(np); count = of_get_available_child_count(np);
if (!count || count > AW2013_MAX_LEDS) if (!count || count > AW2013_MAX_LEDS)
return -EINVAL; return -EINVAL;
...@@ -297,16 +297,15 @@ static int aw2013_probe_dt(struct aw2013 *chip) ...@@ -297,16 +297,15 @@ static int aw2013_probe_dt(struct aw2013 *chip)
"DT property led-max-microamp is missing\n"); "DT property led-max-microamp is missing\n");
} }
of_property_read_string(child, "linux,default-trigger",
&led->cdev.default_trigger);
led->cdev.brightness_set_blocking = aw2013_brightness_set; led->cdev.brightness_set_blocking = aw2013_brightness_set;
led->cdev.blink_set = aw2013_blink_set; led->cdev.blink_set = aw2013_blink_set;
ret = devm_led_classdev_register_ext(&chip->client->dev, ret = devm_led_classdev_register_ext(&chip->client->dev,
&led->cdev, &init_data); &led->cdev, &init_data);
if (ret < 0) if (ret < 0) {
of_node_put(child);
return ret; return ret;
}
i++; i++;
} }
......
...@@ -328,6 +328,7 @@ static int bcm6328_led(struct device *dev, struct device_node *nc, u32 reg, ...@@ -328,6 +328,7 @@ static int bcm6328_led(struct device *dev, struct device_node *nc, u32 reg,
void __iomem *mem, spinlock_t *lock, void __iomem *mem, spinlock_t *lock,
unsigned long *blink_leds, unsigned long *blink_delay) unsigned long *blink_leds, unsigned long *blink_delay)
{ {
struct led_init_data init_data = {};
struct bcm6328_led *led; struct bcm6328_led *led;
const char *state; const char *state;
int rc; int rc;
...@@ -345,11 +346,6 @@ static int bcm6328_led(struct device *dev, struct device_node *nc, u32 reg, ...@@ -345,11 +346,6 @@ static int bcm6328_led(struct device *dev, struct device_node *nc, u32 reg,
if (of_property_read_bool(nc, "active-low")) if (of_property_read_bool(nc, "active-low"))
led->active_low = true; led->active_low = true;
led->cdev.name = of_get_property(nc, "label", NULL) ? : nc->name;
led->cdev.default_trigger = of_get_property(nc,
"linux,default-trigger",
NULL);
if (!of_property_read_string(nc, "default-state", &state)) { if (!of_property_read_string(nc, "default-state", &state)) {
if (!strcmp(state, "on")) { if (!strcmp(state, "on")) {
led->cdev.brightness = LED_FULL; led->cdev.brightness = LED_FULL;
...@@ -382,8 +378,9 @@ static int bcm6328_led(struct device *dev, struct device_node *nc, u32 reg, ...@@ -382,8 +378,9 @@ static int bcm6328_led(struct device *dev, struct device_node *nc, u32 reg,
led->cdev.brightness_set = bcm6328_led_set; led->cdev.brightness_set = bcm6328_led_set;
led->cdev.blink_set = bcm6328_blink_set; led->cdev.blink_set = bcm6328_blink_set;
init_data.fwnode = of_fwnode_handle(nc);
rc = led_classdev_register(dev, &led->cdev); rc = devm_led_classdev_register_ext(dev, &led->cdev, &init_data);
if (rc < 0) if (rc < 0)
return rc; return rc;
...@@ -395,7 +392,7 @@ static int bcm6328_led(struct device *dev, struct device_node *nc, u32 reg, ...@@ -395,7 +392,7 @@ static int bcm6328_led(struct device *dev, struct device_node *nc, u32 reg,
static int bcm6328_leds_probe(struct platform_device *pdev) static int bcm6328_leds_probe(struct platform_device *pdev)
{ {
struct device *dev = &pdev->dev; struct device *dev = &pdev->dev;
struct device_node *np = pdev->dev.of_node; struct device_node *np = dev_of_node(&pdev->dev);
struct device_node *child; struct device_node *child;
void __iomem *mem; void __iomem *mem;
spinlock_t *lock; /* memory lock */ spinlock_t *lock; /* memory lock */
......
...@@ -94,6 +94,7 @@ static void bcm6358_led_set(struct led_classdev *led_cdev, ...@@ -94,6 +94,7 @@ static void bcm6358_led_set(struct led_classdev *led_cdev,
static int bcm6358_led(struct device *dev, struct device_node *nc, u32 reg, static int bcm6358_led(struct device *dev, struct device_node *nc, u32 reg,
void __iomem *mem, spinlock_t *lock) void __iomem *mem, spinlock_t *lock)
{ {
struct led_init_data init_data = {};
struct bcm6358_led *led; struct bcm6358_led *led;
const char *state; const char *state;
int rc; int rc;
...@@ -109,11 +110,6 @@ static int bcm6358_led(struct device *dev, struct device_node *nc, u32 reg, ...@@ -109,11 +110,6 @@ static int bcm6358_led(struct device *dev, struct device_node *nc, u32 reg,
if (of_property_read_bool(nc, "active-low")) if (of_property_read_bool(nc, "active-low"))
led->active_low = true; led->active_low = true;
led->cdev.name = of_get_property(nc, "label", NULL) ? : nc->name;
led->cdev.default_trigger = of_get_property(nc,
"linux,default-trigger",
NULL);
if (!of_property_read_string(nc, "default-state", &state)) { if (!of_property_read_string(nc, "default-state", &state)) {
if (!strcmp(state, "on")) { if (!strcmp(state, "on")) {
led->cdev.brightness = LED_FULL; led->cdev.brightness = LED_FULL;
...@@ -136,8 +132,9 @@ static int bcm6358_led(struct device *dev, struct device_node *nc, u32 reg, ...@@ -136,8 +132,9 @@ static int bcm6358_led(struct device *dev, struct device_node *nc, u32 reg,
bcm6358_led_set(&led->cdev, led->cdev.brightness); bcm6358_led_set(&led->cdev, led->cdev.brightness);
led->cdev.brightness_set = bcm6358_led_set; led->cdev.brightness_set = bcm6358_led_set;
init_data.fwnode = of_fwnode_handle(nc);
rc = led_classdev_register(dev, &led->cdev); rc = devm_led_classdev_register_ext(dev, &led->cdev, &init_data);
if (rc < 0) if (rc < 0)
return rc; return rc;
...@@ -149,7 +146,7 @@ static int bcm6358_led(struct device *dev, struct device_node *nc, u32 reg, ...@@ -149,7 +146,7 @@ static int bcm6358_led(struct device *dev, struct device_node *nc, u32 reg,
static int bcm6358_leds_probe(struct platform_device *pdev) static int bcm6358_leds_probe(struct platform_device *pdev)
{ {
struct device *dev = &pdev->dev; struct device *dev = &pdev->dev;
struct device_node *np = pdev->dev.of_node; struct device_node *np = dev_of_node(&pdev->dev);
struct device_node *child; struct device_node *child;
void __iomem *mem; void __iomem *mem;
spinlock_t *lock; /* memory lock */ spinlock_t *lock; /* memory lock */
......
...@@ -158,19 +158,14 @@ MODULE_DEVICE_TABLE(of, cpcap_led_of_match); ...@@ -158,19 +158,14 @@ MODULE_DEVICE_TABLE(of, cpcap_led_of_match);
static int cpcap_led_probe(struct platform_device *pdev) static int cpcap_led_probe(struct platform_device *pdev)
{ {
const struct of_device_id *match;
struct cpcap_led *led; struct cpcap_led *led;
int err; int err;
match = of_match_device(of_match_ptr(cpcap_led_of_match), &pdev->dev);
if (!match || !match->data)
return -EINVAL;
led = devm_kzalloc(&pdev->dev, sizeof(*led), GFP_KERNEL); led = devm_kzalloc(&pdev->dev, sizeof(*led), GFP_KERNEL);
if (!led) if (!led)
return -ENOMEM; return -ENOMEM;
platform_set_drvdata(pdev, led); platform_set_drvdata(pdev, led);
led->info = match->data; led->info = device_get_match_data(&pdev->dev);
led->dev = &pdev->dev; led->dev = &pdev->dev;
if (led->info->reg == 0x0000) { if (led->info->reg == 0x0000) {
......
...@@ -188,9 +188,6 @@ static int cr0014114_probe_dt(struct cr0014114 *priv) ...@@ -188,9 +188,6 @@ static int cr0014114_probe_dt(struct cr0014114 *priv)
device_for_each_child_node(priv->dev, child) { device_for_each_child_node(priv->dev, child) {
led = &priv->leds[i]; led = &priv->leds[i];
fwnode_property_read_string(child, "linux,default-trigger",
&led->ldev.default_trigger);
led->priv = priv; led->priv = priv;
led->ldev.max_brightness = CR_MAX_BRIGHTNESS; led->ldev.max_brightness = CR_MAX_BRIGHTNESS;
led->ldev.brightness_set_blocking = cr0014114_set_sync; led->ldev.brightness_set_blocking = cr0014114_set_sync;
......
...@@ -263,9 +263,6 @@ static int el15203000_probe_dt(struct el15203000 *priv) ...@@ -263,9 +263,6 @@ static int el15203000_probe_dt(struct el15203000 *priv)
return -EINVAL; return -EINVAL;
} }
fwnode_property_read_string(child, "linux,default-trigger",
&led->ldev.default_trigger);
led->priv = priv; led->priv = priv;
led->ldev.max_brightness = LED_ON; led->ldev.max_brightness = LED_ON;
led->ldev.brightness_set_blocking = el15203000_set_blocking; led->ldev.brightness_set_blocking = el15203000_set_blocking;
......
...@@ -160,9 +160,6 @@ static struct gpio_leds_priv *gpio_leds_create(struct platform_device *pdev) ...@@ -160,9 +160,6 @@ static struct gpio_leds_priv *gpio_leds_create(struct platform_device *pdev)
led_dat->gpiod = led.gpiod; led_dat->gpiod = led.gpiod;
fwnode_property_read_string(child, "linux,default-trigger",
&led.default_trigger);
if (!fwnode_property_read_string(child, "default-state", if (!fwnode_property_read_string(child, "default-state",
&state)) { &state)) {
if (!strcmp(state, "keep")) if (!strcmp(state, "keep"))
......
...@@ -3,6 +3,7 @@ ...@@ -3,6 +3,7 @@
* LED Driver for SGI Octane machines * LED Driver for SGI Octane machines
*/ */
#include <asm/io.h>
#include <linux/module.h> #include <linux/module.h>
#include <linux/kernel.h> #include <linux/kernel.h>
#include <linux/platform_device.h> #include <linux/platform_device.h>
......
...@@ -16,6 +16,8 @@ ...@@ -16,6 +16,8 @@
#include <linux/of_device.h> #include <linux/of_device.h>
#include <linux/regmap.h> #include <linux/regmap.h>
#include <linux/slab.h> #include <linux/slab.h>
#include <linux/delay.h>
#include <linux/gpio/consumer.h>
/* register numbers */ /* register numbers */
#define IS31FL319X_SHUTDOWN 0x00 #define IS31FL319X_SHUTDOWN 0x00
...@@ -61,6 +63,7 @@ ...@@ -61,6 +63,7 @@
struct is31fl319x_chip { struct is31fl319x_chip {
const struct is31fl319x_chipdef *cdef; const struct is31fl319x_chipdef *cdef;
struct i2c_client *client; struct i2c_client *client;
struct gpio_desc *shutdown_gpio;
struct regmap *regmap; struct regmap *regmap;
struct mutex lock; struct mutex lock;
u32 audio_gain_db; u32 audio_gain_db;
...@@ -199,26 +202,27 @@ static int is31fl319x_parse_child_dt(const struct device *dev, ...@@ -199,26 +202,27 @@ static int is31fl319x_parse_child_dt(const struct device *dev,
static int is31fl319x_parse_dt(struct device *dev, static int is31fl319x_parse_dt(struct device *dev,
struct is31fl319x_chip *is31) struct is31fl319x_chip *is31)
{ {
struct device_node *np = dev->of_node, *child; struct device_node *np = dev_of_node(dev), *child;
const struct of_device_id *of_dev_id;
int count; int count;
int ret; int ret;
if (!np) if (!np)
return -ENODEV; return -ENODEV;
of_dev_id = of_match_device(of_is31fl319x_match, dev); is31->shutdown_gpio = devm_gpiod_get_optional(dev,
if (!of_dev_id) { "shutdown",
dev_err(dev, "Failed to match device with supported chips\n"); GPIOD_OUT_HIGH);
return -EINVAL; if (IS_ERR(is31->shutdown_gpio)) {
ret = PTR_ERR(is31->shutdown_gpio);
dev_err(dev, "Failed to get shutdown gpio: %d\n", ret);
return ret;
} }
is31->cdef = of_dev_id->data; is31->cdef = device_get_match_data(dev);
count = of_get_child_count(np); count = of_get_available_child_count(np);
dev_dbg(dev, "probe %s with %d leds defined in DT\n", dev_dbg(dev, "probing with %d leds defined in DT\n", count);
of_dev_id->compatible, count);
if (!count || count > is31->cdef->num_leds) { if (!count || count > is31->cdef->num_leds) {
dev_err(dev, "Number of leds defined must be between 1 and %u\n", dev_err(dev, "Number of leds defined must be between 1 and %u\n",
...@@ -226,7 +230,7 @@ static int is31fl319x_parse_dt(struct device *dev, ...@@ -226,7 +230,7 @@ static int is31fl319x_parse_dt(struct device *dev,
return -ENODEV; return -ENODEV;
} }
for_each_child_of_node(np, child) { for_each_available_child_of_node(np, child) {
struct is31fl319x_led *led; struct is31fl319x_led *led;
u32 reg; u32 reg;
...@@ -350,6 +354,12 @@ static int is31fl319x_probe(struct i2c_client *client, ...@@ -350,6 +354,12 @@ static int is31fl319x_probe(struct i2c_client *client,
if (err) if (err)
goto free_mutex; goto free_mutex;
if (is31->shutdown_gpio) {
gpiod_direction_output(is31->shutdown_gpio, 0);
mdelay(5);
gpiod_direction_output(is31->shutdown_gpio, 1);
}
is31->client = client; is31->client = client;
is31->regmap = devm_regmap_init_i2c(client, &regmap_config); is31->regmap = devm_regmap_init_i2c(client, &regmap_config);
if (IS_ERR(is31->regmap)) { if (IS_ERR(is31->regmap)) {
......
...@@ -332,9 +332,6 @@ static int is31fl32xx_parse_child_dt(const struct device *dev, ...@@ -332,9 +332,6 @@ static int is31fl32xx_parse_child_dt(const struct device *dev,
int ret = 0; int ret = 0;
u32 reg; u32 reg;
if (of_property_read_string(child, "label", &cdev->name))
cdev->name = child->name;
ret = of_property_read_u32(child, "reg", &reg); ret = of_property_read_u32(child, "reg", &reg);
if (ret || reg < 1 || reg > led_data->priv->cdef->channels) { if (ret || reg < 1 || reg > led_data->priv->cdef->channels) {
dev_err(dev, dev_err(dev,
...@@ -344,9 +341,6 @@ static int is31fl32xx_parse_child_dt(const struct device *dev, ...@@ -344,9 +341,6 @@ static int is31fl32xx_parse_child_dt(const struct device *dev,
} }
led_data->channel = reg; led_data->channel = reg;
of_property_read_string(child, "linux,default-trigger",
&cdev->default_trigger);
cdev->brightness_set_blocking = is31fl32xx_brightness_set; cdev->brightness_set_blocking = is31fl32xx_brightness_set;
return 0; return 0;
...@@ -372,7 +366,8 @@ static int is31fl32xx_parse_dt(struct device *dev, ...@@ -372,7 +366,8 @@ static int is31fl32xx_parse_dt(struct device *dev,
struct device_node *child; struct device_node *child;
int ret = 0; int ret = 0;
for_each_child_of_node(dev->of_node, child) { for_each_available_child_of_node(dev_of_node(dev), child) {
struct led_init_data init_data = {};
struct is31fl32xx_led_data *led_data = struct is31fl32xx_led_data *led_data =
&priv->leds[priv->num_leds]; &priv->leds[priv->num_leds];
const struct is31fl32xx_led_data *other_led_data; const struct is31fl32xx_led_data *other_led_data;
...@@ -388,17 +383,18 @@ static int is31fl32xx_parse_dt(struct device *dev, ...@@ -388,17 +383,18 @@ static int is31fl32xx_parse_dt(struct device *dev,
led_data->channel); led_data->channel);
if (other_led_data) { if (other_led_data) {
dev_err(dev, dev_err(dev,
"%s and %s both attempting to use channel %d\n", "Node %pOF 'reg' conflicts with another LED\n",
led_data->cdev.name, child);
other_led_data->cdev.name,
led_data->channel);
goto err; goto err;
} }
ret = devm_led_classdev_register(dev, &led_data->cdev); init_data.fwnode = of_fwnode_handle(child);
ret = devm_led_classdev_register_ext(dev, &led_data->cdev,
&init_data);
if (ret) { if (ret) {
dev_err(dev, "failed to register PWM led for %s: %d\n", dev_err(dev, "Failed to register LED for %pOF: %d\n",
led_data->cdev.name, ret); child, ret);
goto err; goto err;
} }
...@@ -428,19 +424,14 @@ static int is31fl32xx_probe(struct i2c_client *client, ...@@ -428,19 +424,14 @@ static int is31fl32xx_probe(struct i2c_client *client,
const struct i2c_device_id *id) const struct i2c_device_id *id)
{ {
const struct is31fl32xx_chipdef *cdef; const struct is31fl32xx_chipdef *cdef;
const struct of_device_id *of_dev_id;
struct device *dev = &client->dev; struct device *dev = &client->dev;
struct is31fl32xx_priv *priv; struct is31fl32xx_priv *priv;
int count; int count;
int ret = 0; int ret = 0;
of_dev_id = of_match_device(of_is31fl32xx_match, dev); cdef = device_get_match_data(dev);
if (!of_dev_id)
return -EINVAL;
cdef = of_dev_id->data;
count = of_get_child_count(dev->of_node); count = of_get_available_child_count(dev_of_node(dev));
if (!count) if (!count)
return -EINVAL; return -EINVAL;
......
...@@ -259,11 +259,11 @@ static void ktd2692_setup(struct ktd2692_context *led) ...@@ -259,11 +259,11 @@ static void ktd2692_setup(struct ktd2692_context *led)
static int ktd2692_parse_dt(struct ktd2692_context *led, struct device *dev, static int ktd2692_parse_dt(struct ktd2692_context *led, struct device *dev,
struct ktd2692_led_config_data *cfg) struct ktd2692_led_config_data *cfg)
{ {
struct device_node *np = dev->of_node; struct device_node *np = dev_of_node(dev);
struct device_node *child_node; struct device_node *child_node;
int ret; int ret;
if (!dev->of_node) if (!dev_of_node(dev))
return -ENXIO; return -ENXIO;
led->ctrl_gpio = devm_gpiod_get(dev, "ctrl", GPIOD_ASIS); led->ctrl_gpio = devm_gpiod_get(dev, "ctrl", GPIOD_ASIS);
......
...@@ -96,15 +96,15 @@ ...@@ -96,15 +96,15 @@
/* /*
* struct lm3532_als_data * struct lm3532_als_data
* @config - value of ALS configuration register * @config: value of ALS configuration register
* @als1_imp_sel - value of ALS1 resistor select register * @als1_imp_sel: value of ALS1 resistor select register
* @als2_imp_sel - value of ALS2 resistor select register * @als2_imp_sel: value of ALS2 resistor select register
* @als_avrg_time - ALS averaging time * @als_avrg_time: ALS averaging time
* @als_input_mode - ALS input mode for brightness control * @als_input_mode: ALS input mode for brightness control
* @als_vmin - Minimum ALS voltage * @als_vmin: Minimum ALS voltage
* @als_vmax - Maximum ALS voltage * @als_vmax: Maximum ALS voltage
* @zone_lo - values of ALS lo ZB(Zone Boundary) registers * @zone_lo: values of ALS lo ZB(Zone Boundary) registers
* @zone_hi - values of ALS hi ZB(Zone Boundary) registers * @zone_hi: values of ALS hi ZB(Zone Boundary) registers
*/ */
struct lm3532_als_data { struct lm3532_als_data {
u8 config; u8 config;
...@@ -121,15 +121,14 @@ struct lm3532_als_data { ...@@ -121,15 +121,14 @@ struct lm3532_als_data {
/** /**
* struct lm3532_led * struct lm3532_led
* @led_dev: led class device * @led_dev: led class device
* @priv - Pointer the device data structure * @priv: Pointer the device data structure
* @control_bank - Control bank the LED is associated to * @control_bank: Control bank the LED is associated to
* @mode - Mode of the LED string * @mode: Mode of the LED string
* @ctrl_brt_pointer - Zone target register that controls the sink * @ctrl_brt_pointer: Zone target register that controls the sink
* @num_leds - Number of LED strings are supported in this array * @num_leds: Number of LED strings are supported in this array
* @full_scale_current - The full-scale current setting for the current sink. * @full_scale_current: The full-scale current setting for the current sink.
* @led_strings - The LED strings supported in this array * @led_strings: The LED strings supported in this array
* @enabled - Enabled status * @enabled: Enabled status
* @label - LED label
*/ */
struct lm3532_led { struct lm3532_led {
struct led_classdev led_dev; struct led_classdev led_dev;
...@@ -142,21 +141,20 @@ struct lm3532_led { ...@@ -142,21 +141,20 @@ struct lm3532_led {
int full_scale_current; int full_scale_current;
unsigned int enabled:1; unsigned int enabled:1;
u32 led_strings[LM3532_MAX_CONTROL_BANKS]; u32 led_strings[LM3532_MAX_CONTROL_BANKS];
char label[LED_MAX_NAME_SIZE];
}; };
/** /**
* struct lm3532_data * struct lm3532_data
* @enable_gpio - Hardware enable gpio * @enable_gpio: Hardware enable gpio
* @regulator: regulator * @regulator: regulator
* @client: i2c client * @client: i2c client
* @regmap - Devices register map * @regmap: Devices register map
* @dev - Pointer to the devices device struct * @dev: Pointer to the devices device struct
* @lock - Lock for reading/writing the device * @lock: Lock for reading/writing the device
* @als_data - Pointer to the als data struct * @als_data: Pointer to the als data struct
* @runtime_ramp_up - Runtime ramp up setting * @runtime_ramp_up: Runtime ramp up setting
* @runtime_ramp_down - Runtime ramp down setting * @runtime_ramp_down: Runtime ramp down setting
* @leds - Array of LED strings * @leds: Array of LED strings
*/ */
struct lm3532_data { struct lm3532_data {
struct gpio_desc *enable_gpio; struct gpio_desc *enable_gpio;
...@@ -548,7 +546,6 @@ static int lm3532_parse_node(struct lm3532_data *priv) ...@@ -548,7 +546,6 @@ static int lm3532_parse_node(struct lm3532_data *priv)
{ {
struct fwnode_handle *child = NULL; struct fwnode_handle *child = NULL;
struct lm3532_led *led; struct lm3532_led *led;
const char *name;
int control_bank; int control_bank;
u32 ramp_time; u32 ramp_time;
size_t i = 0; size_t i = 0;
...@@ -643,19 +640,7 @@ static int lm3532_parse_node(struct lm3532_data *priv) ...@@ -643,19 +640,7 @@ static int lm3532_parse_node(struct lm3532_data *priv)
goto child_out; goto child_out;
} }
fwnode_property_read_string(child, "linux,default-trigger",
&led->led_dev.default_trigger);
ret = fwnode_property_read_string(child, "label", &name);
if (ret)
snprintf(led->label, sizeof(led->label),
"%s::", priv->client->name);
else
snprintf(led->label, sizeof(led->label),
"%s:%s", priv->client->name, name);
led->priv = priv; led->priv = priv;
led->led_dev.name = led->label;
led->led_dev.brightness_set_blocking = lm3532_brightness_set; led->led_dev.brightness_set_blocking = lm3532_brightness_set;
ret = devm_led_classdev_register_ext(priv->dev, &led->led_dev, &idata); ret = devm_led_classdev_register_ext(priv->dev, &led->led_dev, &idata);
......
...@@ -26,8 +26,8 @@ ...@@ -26,8 +26,8 @@
* @lmu_data: Register and setting values for common code * @lmu_data: Register and setting values for common code
* @regmap: Devices register map * @regmap: Devices register map
* @dev: Pointer to the devices device struct * @dev: Pointer to the devices device struct
* @led_sources - The LED strings supported in this array * @led_sources: The LED strings supported in this array
* @num_leds - Number of LED strings are supported in this array * @num_leds: Number of LED strings are supported in this array
*/ */
struct lm36274 { struct lm36274 {
struct platform_device *pdev; struct platform_device *pdev;
...@@ -43,120 +43,111 @@ struct lm36274 { ...@@ -43,120 +43,111 @@ struct lm36274 {
static int lm36274_brightness_set(struct led_classdev *led_cdev, static int lm36274_brightness_set(struct led_classdev *led_cdev,
enum led_brightness brt_val) enum led_brightness brt_val)
{ {
struct lm36274 *led = container_of(led_cdev, struct lm36274, led_dev); struct lm36274 *chip = container_of(led_cdev, struct lm36274, led_dev);
return ti_lmu_common_set_brightness(&led->lmu_data, brt_val); return ti_lmu_common_set_brightness(&chip->lmu_data, brt_val);
} }
static int lm36274_init(struct lm36274 *lm36274_data) static int lm36274_init(struct lm36274 *chip)
{ {
int enable_val = 0; int enable_val = 0;
int i; int i;
for (i = 0; i < lm36274_data->num_leds; i++) for (i = 0; i < chip->num_leds; i++)
enable_val |= (1 << lm36274_data->led_sources[i]); enable_val |= (1 << chip->led_sources[i]);
if (!enable_val) { if (!enable_val) {
dev_err(lm36274_data->dev, "No LEDs were enabled\n"); dev_err(chip->dev, "No LEDs were enabled\n");
return -EINVAL; return -EINVAL;
} }
enable_val |= LM36274_BL_EN; enable_val |= LM36274_BL_EN;
return regmap_write(lm36274_data->regmap, LM36274_REG_BL_EN, return regmap_write(chip->regmap, LM36274_REG_BL_EN, enable_val);
enable_val);
} }
static int lm36274_parse_dt(struct lm36274 *lm36274_data) static int lm36274_parse_dt(struct lm36274 *chip,
struct led_init_data *init_data)
{ {
struct fwnode_handle *child = NULL; struct device *dev = chip->dev;
char label[LED_MAX_NAME_SIZE]; struct fwnode_handle *child;
struct device *dev = &lm36274_data->pdev->dev; int ret;
const char *name;
int child_cnt;
int ret = -EINVAL;
/* There should only be 1 node */ /* There should only be 1 node */
child_cnt = device_get_child_node_count(dev); if (device_get_child_node_count(dev) != 1)
if (child_cnt != 1)
return -EINVAL; return -EINVAL;
device_for_each_child_node(dev, child) { child = device_get_next_child_node(dev, NULL);
ret = fwnode_property_read_string(child, "label", &name);
if (ret)
snprintf(label, sizeof(label),
"%s::", lm36274_data->pdev->name);
else
snprintf(label, sizeof(label),
"%s:%s", lm36274_data->pdev->name, name);
lm36274_data->num_leds = fwnode_property_count_u32(child, "led-sources"); init_data->fwnode = child;
if (lm36274_data->num_leds <= 0) init_data->devicename = chip->pdev->name;
return -ENODEV; /* for backwards compatibility when `label` property is not present */
init_data->default_label = ":";
chip->num_leds = fwnode_property_count_u32(child, "led-sources");
if (chip->num_leds <= 0) {
ret = -ENODEV;
goto err;
}
ret = fwnode_property_read_u32_array(child, "led-sources", ret = fwnode_property_read_u32_array(child, "led-sources",
lm36274_data->led_sources, chip->led_sources, chip->num_leds);
lm36274_data->num_leds);
if (ret) { if (ret) {
dev_err(dev, "led-sources property missing\n"); dev_err(dev, "led-sources property missing\n");
return ret; goto err;
} }
fwnode_property_read_string(child, "linux,default-trigger",
&lm36274_data->led_dev.default_trigger);
}
lm36274_data->lmu_data.regmap = lm36274_data->regmap;
lm36274_data->lmu_data.max_brightness = MAX_BRIGHTNESS_11BIT;
lm36274_data->lmu_data.msb_brightness_reg = LM36274_REG_BRT_MSB;
lm36274_data->lmu_data.lsb_brightness_reg = LM36274_REG_BRT_LSB;
lm36274_data->led_dev.name = label;
lm36274_data->led_dev.max_brightness = MAX_BRIGHTNESS_11BIT;
lm36274_data->led_dev.brightness_set_blocking = lm36274_brightness_set;
return 0; return 0;
err:
fwnode_handle_put(child);
return ret;
} }
static int lm36274_probe(struct platform_device *pdev) static int lm36274_probe(struct platform_device *pdev)
{ {
struct ti_lmu *lmu = dev_get_drvdata(pdev->dev.parent); struct ti_lmu *lmu = dev_get_drvdata(pdev->dev.parent);
struct lm36274 *lm36274_data; struct led_init_data init_data = {};
struct lm36274 *chip;
int ret; int ret;
lm36274_data = devm_kzalloc(&pdev->dev, sizeof(*lm36274_data), chip = devm_kzalloc(&pdev->dev, sizeof(*chip), GFP_KERNEL);
GFP_KERNEL); if (!chip)
if (!lm36274_data)
return -ENOMEM; return -ENOMEM;
lm36274_data->pdev = pdev; chip->pdev = pdev;
lm36274_data->dev = lmu->dev; chip->dev = &pdev->dev;
lm36274_data->regmap = lmu->regmap; chip->regmap = lmu->regmap;
platform_set_drvdata(pdev, lm36274_data); platform_set_drvdata(pdev, chip);
ret = lm36274_parse_dt(lm36274_data); ret = lm36274_parse_dt(chip, &init_data);
if (ret) { if (ret) {
dev_err(lm36274_data->dev, "Failed to parse DT node\n"); dev_err(chip->dev, "Failed to parse DT node\n");
return ret; return ret;
} }
ret = lm36274_init(lm36274_data); ret = lm36274_init(chip);
if (ret) { if (ret) {
dev_err(lm36274_data->dev, "Failed to init the device\n"); dev_err(chip->dev, "Failed to init the device\n");
return ret; return ret;
} }
return led_classdev_register(lm36274_data->dev, &lm36274_data->led_dev); chip->lmu_data.regmap = chip->regmap;
} chip->lmu_data.max_brightness = MAX_BRIGHTNESS_11BIT;
chip->lmu_data.msb_brightness_reg = LM36274_REG_BRT_MSB;
chip->lmu_data.lsb_brightness_reg = LM36274_REG_BRT_LSB;
static int lm36274_remove(struct platform_device *pdev) chip->led_dev.max_brightness = MAX_BRIGHTNESS_11BIT;
{ chip->led_dev.brightness_set_blocking = lm36274_brightness_set;
struct lm36274 *lm36274_data = platform_get_drvdata(pdev);
led_classdev_unregister(&lm36274_data->led_dev); ret = devm_led_classdev_register_ext(chip->dev, &chip->led_dev,
&init_data);
if (ret)
dev_err(chip->dev, "Failed to register LED for node %pfw\n",
init_data.fwnode);
return 0; fwnode_handle_put(init_data.fwnode);
return ret;
} }
static const struct of_device_id of_lm36274_leds_match[] = { static const struct of_device_id of_lm36274_leds_match[] = {
...@@ -167,9 +158,9 @@ MODULE_DEVICE_TABLE(of, of_lm36274_leds_match); ...@@ -167,9 +158,9 @@ MODULE_DEVICE_TABLE(of, of_lm36274_leds_match);
static struct platform_driver lm36274_driver = { static struct platform_driver lm36274_driver = {
.probe = lm36274_probe, .probe = lm36274_probe,
.remove = lm36274_remove,
.driver = { .driver = {
.name = "lm36274-leds", .name = "lm36274-leds",
.of_match_table = of_lm36274_leds_match,
}, },
}; };
module_platform_driver(lm36274_driver) module_platform_driver(lm36274_driver)
......
...@@ -394,13 +394,10 @@ static int lm3692x_probe_dt(struct lm3692x_led *led) ...@@ -394,13 +394,10 @@ static int lm3692x_probe_dt(struct lm3692x_led *led)
led->regulator = devm_regulator_get_optional(&led->client->dev, "vled"); led->regulator = devm_regulator_get_optional(&led->client->dev, "vled");
if (IS_ERR(led->regulator)) { if (IS_ERR(led->regulator)) {
ret = PTR_ERR(led->regulator); ret = PTR_ERR(led->regulator);
if (ret != -ENODEV) { if (ret != -ENODEV)
if (ret != -EPROBE_DEFER) return dev_err_probe(&led->client->dev, ret,
dev_err(&led->client->dev, "Failed to get vled regulator\n");
"Failed to get vled regulator: %d\n",
ret);
return ret;
}
led->regulator = NULL; led->regulator = NULL;
} }
...@@ -436,9 +433,6 @@ static int lm3692x_probe_dt(struct lm3692x_led *led) ...@@ -436,9 +433,6 @@ static int lm3692x_probe_dt(struct lm3692x_led *led)
return -ENODEV; return -ENODEV;
} }
fwnode_property_read_string(child, "linux,default-trigger",
&led->led_dev.default_trigger);
ret = fwnode_property_read_u32(child, "reg", &led->led_enable); ret = fwnode_property_read_u32(child, "reg", &led->led_enable);
if (ret) { if (ret) {
dev_err(&led->client->dev, "reg DT property missing\n"); dev_err(&led->client->dev, "reg DT property missing\n");
......
...@@ -78,6 +78,7 @@ struct lm3697 { ...@@ -78,6 +78,7 @@ struct lm3697 {
struct mutex lock; struct mutex lock;
int bank_cfg; int bank_cfg;
int num_banks;
struct lm3697_led leds[]; struct lm3697_led leds[];
}; };
...@@ -115,6 +116,7 @@ static int lm3697_brightness_set(struct led_classdev *led_cdev, ...@@ -115,6 +116,7 @@ static int lm3697_brightness_set(struct led_classdev *led_cdev,
struct lm3697_led *led = container_of(led_cdev, struct lm3697_led, struct lm3697_led *led = container_of(led_cdev, struct lm3697_led,
led_dev); led_dev);
int ctrl_en_val = (1 << led->control_bank); int ctrl_en_val = (1 << led->control_bank);
struct device *dev = led->priv->dev;
int ret; int ret;
mutex_lock(&led->priv->lock); mutex_lock(&led->priv->lock);
...@@ -123,7 +125,7 @@ static int lm3697_brightness_set(struct led_classdev *led_cdev, ...@@ -123,7 +125,7 @@ static int lm3697_brightness_set(struct led_classdev *led_cdev,
ret = regmap_update_bits(led->priv->regmap, LM3697_CTRL_ENABLE, ret = regmap_update_bits(led->priv->regmap, LM3697_CTRL_ENABLE,
ctrl_en_val, ~ctrl_en_val); ctrl_en_val, ~ctrl_en_val);
if (ret) { if (ret) {
dev_err(&led->priv->client->dev, "Cannot write ctrl register\n"); dev_err(dev, "Cannot write ctrl register\n");
goto brightness_out; goto brightness_out;
} }
...@@ -131,8 +133,7 @@ static int lm3697_brightness_set(struct led_classdev *led_cdev, ...@@ -131,8 +133,7 @@ static int lm3697_brightness_set(struct led_classdev *led_cdev,
} else { } else {
ret = ti_lmu_common_set_brightness(&led->lmu_data, brt_val); ret = ti_lmu_common_set_brightness(&led->lmu_data, brt_val);
if (ret) { if (ret) {
dev_err(&led->priv->client->dev, dev_err(dev, "Cannot write brightness\n");
"Cannot write brightness\n");
goto brightness_out; goto brightness_out;
} }
...@@ -141,8 +142,7 @@ static int lm3697_brightness_set(struct led_classdev *led_cdev, ...@@ -141,8 +142,7 @@ static int lm3697_brightness_set(struct led_classdev *led_cdev,
LM3697_CTRL_ENABLE, LM3697_CTRL_ENABLE,
ctrl_en_val, ctrl_en_val); ctrl_en_val, ctrl_en_val);
if (ret) { if (ret) {
dev_err(&led->priv->client->dev, dev_err(dev, "Cannot enable the device\n");
"Cannot enable the device\n");
goto brightness_out; goto brightness_out;
} }
...@@ -157,6 +157,7 @@ static int lm3697_brightness_set(struct led_classdev *led_cdev, ...@@ -157,6 +157,7 @@ static int lm3697_brightness_set(struct led_classdev *led_cdev,
static int lm3697_init(struct lm3697 *priv) static int lm3697_init(struct lm3697 *priv)
{ {
struct device *dev = priv->dev;
struct lm3697_led *led; struct lm3697_led *led;
int i, ret; int i, ret;
...@@ -165,26 +166,26 @@ static int lm3697_init(struct lm3697 *priv) ...@@ -165,26 +166,26 @@ static int lm3697_init(struct lm3697 *priv)
} else { } else {
ret = regmap_write(priv->regmap, LM3697_RESET, LM3697_SW_RESET); ret = regmap_write(priv->regmap, LM3697_RESET, LM3697_SW_RESET);
if (ret) { if (ret) {
dev_err(&priv->client->dev, "Cannot reset the device\n"); dev_err(dev, "Cannot reset the device\n");
goto out; goto out;
} }
} }
ret = regmap_write(priv->regmap, LM3697_CTRL_ENABLE, 0x0); ret = regmap_write(priv->regmap, LM3697_CTRL_ENABLE, 0x0);
if (ret) { if (ret) {
dev_err(&priv->client->dev, "Cannot write ctrl enable\n"); dev_err(dev, "Cannot write ctrl enable\n");
goto out; goto out;
} }
ret = regmap_write(priv->regmap, LM3697_OUTPUT_CONFIG, priv->bank_cfg); ret = regmap_write(priv->regmap, LM3697_OUTPUT_CONFIG, priv->bank_cfg);
if (ret) if (ret)
dev_err(&priv->client->dev, "Cannot write OUTPUT config\n"); dev_err(dev, "Cannot write OUTPUT config\n");
for (i = 0; i < LM3697_MAX_CONTROL_BANKS; i++) { for (i = 0; i < priv->num_banks; i++) {
led = &priv->leds[i]; led = &priv->leds[i];
ret = ti_lmu_common_set_ramp(&led->lmu_data); ret = ti_lmu_common_set_ramp(&led->lmu_data);
if (ret) if (ret)
dev_err(&priv->client->dev, "Setting the ramp rate failed\n"); dev_err(dev, "Setting the ramp rate failed\n");
} }
out: out:
return ret; return ret;
...@@ -193,36 +194,37 @@ static int lm3697_init(struct lm3697 *priv) ...@@ -193,36 +194,37 @@ static int lm3697_init(struct lm3697 *priv)
static int lm3697_probe_dt(struct lm3697 *priv) static int lm3697_probe_dt(struct lm3697 *priv)
{ {
struct fwnode_handle *child = NULL; struct fwnode_handle *child = NULL;
struct device *dev = priv->dev;
struct lm3697_led *led; struct lm3697_led *led;
const char *name; int ret = -EINVAL;
int control_bank; int control_bank;
size_t i = 0; size_t i = 0;
int ret = -EINVAL;
int j; int j;
priv->enable_gpio = devm_gpiod_get_optional(&priv->client->dev, priv->enable_gpio = devm_gpiod_get_optional(dev, "enable",
"enable", GPIOD_OUT_LOW); GPIOD_OUT_LOW);
if (IS_ERR(priv->enable_gpio)) { if (IS_ERR(priv->enable_gpio)) {
ret = PTR_ERR(priv->enable_gpio); ret = PTR_ERR(priv->enable_gpio);
dev_err(&priv->client->dev, "Failed to get enable gpio: %d\n", dev_err(dev, "Failed to get enable gpio: %d\n", ret);
ret);
return ret; return ret;
} }
priv->regulator = devm_regulator_get(&priv->client->dev, "vled"); priv->regulator = devm_regulator_get(dev, "vled");
if (IS_ERR(priv->regulator)) if (IS_ERR(priv->regulator))
priv->regulator = NULL; priv->regulator = NULL;
device_for_each_child_node(priv->dev, child) { device_for_each_child_node(dev, child) {
struct led_init_data init_data = {};
ret = fwnode_property_read_u32(child, "reg", &control_bank); ret = fwnode_property_read_u32(child, "reg", &control_bank);
if (ret) { if (ret) {
dev_err(&priv->client->dev, "reg property missing\n"); dev_err(dev, "reg property missing\n");
fwnode_handle_put(child); fwnode_handle_put(child);
goto child_out; goto child_out;
} }
if (control_bank > LM3697_CONTROL_B) { if (control_bank > LM3697_CONTROL_B) {
dev_err(&priv->client->dev, "reg property is invalid\n"); dev_err(dev, "reg property is invalid\n");
ret = -EINVAL; ret = -EINVAL;
fwnode_handle_put(child); fwnode_handle_put(child);
goto child_out; goto child_out;
...@@ -230,10 +232,10 @@ static int lm3697_probe_dt(struct lm3697 *priv) ...@@ -230,10 +232,10 @@ static int lm3697_probe_dt(struct lm3697 *priv)
led = &priv->leds[i]; led = &priv->leds[i];
ret = ti_lmu_common_get_brt_res(&priv->client->dev, ret = ti_lmu_common_get_brt_res(dev, child, &led->lmu_data);
child, &led->lmu_data);
if (ret) if (ret)
dev_warn(&priv->client->dev, "brightness resolution property missing\n"); dev_warn(dev,
"brightness resolution property missing\n");
led->control_bank = control_bank; led->control_bank = control_bank;
led->lmu_data.regmap = priv->regmap; led->lmu_data.regmap = priv->regmap;
...@@ -246,7 +248,7 @@ static int lm3697_probe_dt(struct lm3697 *priv) ...@@ -246,7 +248,7 @@ static int lm3697_probe_dt(struct lm3697 *priv)
led->num_leds = fwnode_property_count_u32(child, "led-sources"); led->num_leds = fwnode_property_count_u32(child, "led-sources");
if (led->num_leds > LM3697_MAX_LED_STRINGS) { if (led->num_leds > LM3697_MAX_LED_STRINGS) {
dev_err(&priv->client->dev, "Too many LED strings defined\n"); dev_err(dev, "Too many LED strings defined\n");
continue; continue;
} }
...@@ -254,7 +256,7 @@ static int lm3697_probe_dt(struct lm3697 *priv) ...@@ -254,7 +256,7 @@ static int lm3697_probe_dt(struct lm3697 *priv)
led->hvled_strings, led->hvled_strings,
led->num_leds); led->num_leds);
if (ret) { if (ret) {
dev_err(&priv->client->dev, "led-sources property missing\n"); dev_err(dev, "led-sources property missing\n");
fwnode_handle_put(child); fwnode_handle_put(child);
goto child_out; goto child_out;
} }
...@@ -263,31 +265,23 @@ static int lm3697_probe_dt(struct lm3697 *priv) ...@@ -263,31 +265,23 @@ static int lm3697_probe_dt(struct lm3697 *priv)
priv->bank_cfg |= priv->bank_cfg |=
(led->control_bank << led->hvled_strings[j]); (led->control_bank << led->hvled_strings[j]);
ret = ti_lmu_common_get_ramp_params(&priv->client->dev, ret = ti_lmu_common_get_ramp_params(dev, child, &led->lmu_data);
child, &led->lmu_data);
if (ret) if (ret)
dev_warn(&priv->client->dev, "runtime-ramp properties missing\n"); dev_warn(dev, "runtime-ramp properties missing\n");
fwnode_property_read_string(child, "linux,default-trigger", init_data.fwnode = child;
&led->led_dev.default_trigger); init_data.devicename = priv->client->name;
/* for backwards compatibility if `label` is not present */
ret = fwnode_property_read_string(child, "label", &name); init_data.default_label = ":";
if (ret)
snprintf(led->label, sizeof(led->label),
"%s::", priv->client->name);
else
snprintf(led->label, sizeof(led->label),
"%s:%s", priv->client->name, name);
led->priv = priv; led->priv = priv;
led->led_dev.name = led->label;
led->led_dev.max_brightness = led->lmu_data.max_brightness; led->led_dev.max_brightness = led->lmu_data.max_brightness;
led->led_dev.brightness_set_blocking = lm3697_brightness_set; led->led_dev.brightness_set_blocking = lm3697_brightness_set;
ret = devm_led_classdev_register(priv->dev, &led->led_dev); ret = devm_led_classdev_register_ext(dev, &led->led_dev,
&init_data);
if (ret) { if (ret) {
dev_err(&priv->client->dev, "led register err: %d\n", dev_err(dev, "led register err: %d\n", ret);
ret);
fwnode_handle_put(child); fwnode_handle_put(child);
goto child_out; goto child_out;
} }
...@@ -302,18 +296,18 @@ static int lm3697_probe_dt(struct lm3697 *priv) ...@@ -302,18 +296,18 @@ static int lm3697_probe_dt(struct lm3697 *priv)
static int lm3697_probe(struct i2c_client *client, static int lm3697_probe(struct i2c_client *client,
const struct i2c_device_id *id) const struct i2c_device_id *id)
{ {
struct device *dev = &client->dev;
struct lm3697 *led; struct lm3697 *led;
int count; int count;
int ret; int ret;
count = device_get_child_node_count(&client->dev); count = device_get_child_node_count(dev);
if (!count) { if (!count || count > LM3697_MAX_CONTROL_BANKS) {
dev_err(&client->dev, "LEDs are not defined in device tree!"); dev_err(dev, "Strange device tree!");
return -ENODEV; return -ENODEV;
} }
led = devm_kzalloc(&client->dev, struct_size(led, leds, count), led = devm_kzalloc(dev, struct_size(led, leds, count), GFP_KERNEL);
GFP_KERNEL);
if (!led) if (!led)
return -ENOMEM; return -ENOMEM;
...@@ -321,12 +315,12 @@ static int lm3697_probe(struct i2c_client *client, ...@@ -321,12 +315,12 @@ static int lm3697_probe(struct i2c_client *client,
i2c_set_clientdata(client, led); i2c_set_clientdata(client, led);
led->client = client; led->client = client;
led->dev = &client->dev; led->dev = dev;
led->num_banks = count;
led->regmap = devm_regmap_init_i2c(client, &lm3697_regmap_config); led->regmap = devm_regmap_init_i2c(client, &lm3697_regmap_config);
if (IS_ERR(led->regmap)) { if (IS_ERR(led->regmap)) {
ret = PTR_ERR(led->regmap); ret = PTR_ERR(led->regmap);
dev_err(&client->dev, "Failed to allocate register map: %d\n", dev_err(dev, "Failed to allocate register map: %d\n", ret);
ret);
return ret; return ret;
} }
...@@ -340,12 +334,13 @@ static int lm3697_probe(struct i2c_client *client, ...@@ -340,12 +334,13 @@ static int lm3697_probe(struct i2c_client *client,
static int lm3697_remove(struct i2c_client *client) static int lm3697_remove(struct i2c_client *client)
{ {
struct lm3697 *led = i2c_get_clientdata(client); struct lm3697 *led = i2c_get_clientdata(client);
struct device *dev = &led->client->dev;
int ret; int ret;
ret = regmap_update_bits(led->regmap, LM3697_CTRL_ENABLE, ret = regmap_update_bits(led->regmap, LM3697_CTRL_ENABLE,
LM3697_CTRL_A_B_EN, 0); LM3697_CTRL_A_B_EN, 0);
if (ret) { if (ret) {
dev_err(&led->client->dev, "Failed to disable the device\n"); dev_err(dev, "Failed to disable the device\n");
return ret; return ret;
} }
...@@ -355,8 +350,7 @@ static int lm3697_remove(struct i2c_client *client) ...@@ -355,8 +350,7 @@ static int lm3697_remove(struct i2c_client *client)
if (led->regulator) { if (led->regulator) {
ret = regulator_disable(led->regulator); ret = regulator_disable(led->regulator);
if (ret) if (ret)
dev_err(&led->client->dev, dev_err(dev, "Failed to disable regulator\n");
"Failed to disable regulator\n");
} }
mutex_destroy(&led->lock); mutex_destroy(&led->lock);
......
// SPDX-License-Identifier: GPL-2.0
// TI LP50XX LED chip family driver
// Copyright (C) 2018-20 Texas Instruments Incorporated - https://www.ti.com/
#include <linux/gpio/consumer.h>
#include <linux/i2c.h>
#include <linux/init.h>
#include <linux/leds.h>
#include <linux/module.h>
#include <linux/mutex.h>
#include <linux/of.h>
#include <linux/of_gpio.h>
#include <linux/regmap.h>
#include <linux/regulator/consumer.h>
#include <linux/slab.h>
#include <uapi/linux/uleds.h>
#include <linux/led-class-multicolor.h>
#include "leds.h"
#define LP50XX_DEV_CFG0 0x00
#define LP50XX_DEV_CFG1 0x01
#define LP50XX_LED_CFG0 0x02
/* LP5009 and LP5012 registers */
#define LP5012_BNK_BRT 0x03
#define LP5012_BNKA_CLR 0x04
#define LP5012_BNKB_CLR 0x05
#define LP5012_BNKC_CLR 0x06
#define LP5012_LED0_BRT 0x07
#define LP5012_OUT0_CLR 0x0b
#define LP5012_RESET 0x17
/* LP5018 and LP5024 registers */
#define LP5024_BNK_BRT 0x03
#define LP5024_BNKA_CLR 0x04
#define LP5024_BNKB_CLR 0x05
#define LP5024_BNKC_CLR 0x06
#define LP5024_LED0_BRT 0x07
#define LP5024_OUT0_CLR 0x0f
#define LP5024_RESET 0x27
/* LP5030 and LP5036 registers */
#define LP5036_LED_CFG1 0x03
#define LP5036_BNK_BRT 0x04
#define LP5036_BNKA_CLR 0x05
#define LP5036_BNKB_CLR 0x06
#define LP5036_BNKC_CLR 0x07
#define LP5036_LED0_BRT 0x08
#define LP5036_OUT0_CLR 0x14
#define LP5036_RESET 0x38
#define LP50XX_SW_RESET 0xff
#define LP50XX_CHIP_EN BIT(6)
/* There are 3 LED outputs per bank */
#define LP50XX_LEDS_PER_MODULE 3
#define LP5009_MAX_LED_MODULES 2
#define LP5012_MAX_LED_MODULES 4
#define LP5018_MAX_LED_MODULES 6
#define LP5024_MAX_LED_MODULES 8
#define LP5030_MAX_LED_MODULES 10
#define LP5036_MAX_LED_MODULES 12
static const struct reg_default lp5012_reg_defs[] = {
{LP50XX_DEV_CFG0, 0x0},
{LP50XX_DEV_CFG1, 0x3c},
{LP50XX_LED_CFG0, 0x0},
{LP5012_BNK_BRT, 0xff},
{LP5012_BNKA_CLR, 0x0f},
{LP5012_BNKB_CLR, 0x0f},
{LP5012_BNKC_CLR, 0x0f},
{LP5012_LED0_BRT, 0x0f},
/* LEDX_BRT registers are all 0xff for defaults */
{0x08, 0xff}, {0x09, 0xff}, {0x0a, 0xff},
{LP5012_OUT0_CLR, 0x0f},
/* OUTX_CLR registers are all 0x0 for defaults */
{0x0c, 0x00}, {0x0d, 0x00}, {0x0e, 0x00}, {0x0f, 0x00}, {0x10, 0x00},
{0x11, 0x00}, {0x12, 0x00}, {0x13, 0x00}, {0x14, 0x00}, {0x15, 0x00},
{0x16, 0x00},
{LP5012_RESET, 0x00}
};
static const struct reg_default lp5024_reg_defs[] = {
{LP50XX_DEV_CFG0, 0x0},
{LP50XX_DEV_CFG1, 0x3c},
{LP50XX_LED_CFG0, 0x0},
{LP5024_BNK_BRT, 0xff},
{LP5024_BNKA_CLR, 0x0f},
{LP5024_BNKB_CLR, 0x0f},
{LP5024_BNKC_CLR, 0x0f},
{LP5024_LED0_BRT, 0x0f},
/* LEDX_BRT registers are all 0xff for defaults */
{0x08, 0xff}, {0x09, 0xff}, {0x0a, 0xff}, {0x0b, 0xff}, {0x0c, 0xff},
{0x0d, 0xff}, {0x0e, 0xff},
{LP5024_OUT0_CLR, 0x0f},
/* OUTX_CLR registers are all 0x0 for defaults */
{0x10, 0x00}, {0x11, 0x00}, {0x12, 0x00}, {0x13, 0x00}, {0x14, 0x00},
{0x15, 0x00}, {0x16, 0x00}, {0x17, 0x00}, {0x18, 0x00}, {0x19, 0x00},
{0x1a, 0x00}, {0x1b, 0x00}, {0x1c, 0x00}, {0x1d, 0x00}, {0x1e, 0x00},
{0x1f, 0x00}, {0x20, 0x00}, {0x21, 0x00}, {0x22, 0x00}, {0x23, 0x00},
{0x24, 0x00}, {0x25, 0x00}, {0x26, 0x00},
{LP5024_RESET, 0x00}
};
static const struct reg_default lp5036_reg_defs[] = {
{LP50XX_DEV_CFG0, 0x0},
{LP50XX_DEV_CFG1, 0x3c},
{LP50XX_LED_CFG0, 0x0},
{LP5036_LED_CFG1, 0x0},
{LP5036_BNK_BRT, 0xff},
{LP5036_BNKA_CLR, 0x0f},
{LP5036_BNKB_CLR, 0x0f},
{LP5036_BNKC_CLR, 0x0f},
{LP5036_LED0_BRT, 0x0f},
/* LEDX_BRT registers are all 0xff for defaults */
{0x08, 0xff}, {0x09, 0xff}, {0x0a, 0xff}, {0x0b, 0xff}, {0x0c, 0xff},
{0x0d, 0xff}, {0x0e, 0xff}, {0x0f, 0xff}, {0x10, 0xff}, {0x11, 0xff},
{0x12, 0xff}, {0x13, 0xff},
{LP5036_OUT0_CLR, 0x0f},
/* OUTX_CLR registers are all 0x0 for defaults */
{0x15, 0x00}, {0x16, 0x00}, {0x17, 0x00}, {0x18, 0x00}, {0x19, 0x00},
{0x1a, 0x00}, {0x1b, 0x00}, {0x1c, 0x00}, {0x1d, 0x00}, {0x1e, 0x00},
{0x1f, 0x00}, {0x20, 0x00}, {0x21, 0x00}, {0x22, 0x00}, {0x23, 0x00},
{0x24, 0x00}, {0x25, 0x00}, {0x26, 0x00}, {0x27, 0x00}, {0x28, 0x00},
{0x29, 0x00}, {0x2a, 0x00}, {0x2b, 0x00}, {0x2c, 0x00}, {0x2d, 0x00},
{0x2e, 0x00}, {0x2f, 0x00}, {0x30, 0x00}, {0x31, 0x00}, {0x32, 0x00},
{0x33, 0x00}, {0x34, 0x00}, {0x35, 0x00}, {0x36, 0x00}, {0x37, 0x00},
{LP5036_RESET, 0x00}
};
static const struct regmap_config lp5012_regmap_config = {
.reg_bits = 8,
.val_bits = 8,
.max_register = LP5012_RESET,
.reg_defaults = lp5012_reg_defs,
.num_reg_defaults = ARRAY_SIZE(lp5012_reg_defs),
.cache_type = REGCACHE_FLAT,
};
static const struct regmap_config lp5024_regmap_config = {
.reg_bits = 8,
.val_bits = 8,
.max_register = LP5024_RESET,
.reg_defaults = lp5024_reg_defs,
.num_reg_defaults = ARRAY_SIZE(lp5024_reg_defs),
.cache_type = REGCACHE_FLAT,
};
static const struct regmap_config lp5036_regmap_config = {
.reg_bits = 8,
.val_bits = 8,
.max_register = LP5036_RESET,
.reg_defaults = lp5036_reg_defs,
.num_reg_defaults = ARRAY_SIZE(lp5036_reg_defs),
.cache_type = REGCACHE_FLAT,
};
enum lp50xx_model {
LP5009,
LP5012,
LP5018,
LP5024,
LP5030,
LP5036,
};
/**
* struct lp50xx_chip_info -
* @lp50xx_regmap_config: regmap register configuration
* @model_id: LED device model
* @max_modules: total number of supported LED modules
* @num_leds: number of LED outputs available on the device
* @led_brightness0_reg: first brightness register of the device
* @mix_out0_reg: first color mix register of the device
* @bank_brt_reg: bank brightness register
* @bank_mix_reg: color mix register
* @reset_reg: device reset register
*/
struct lp50xx_chip_info {
const struct regmap_config *lp50xx_regmap_config;
int model_id;
u8 max_modules;
u8 num_leds;
u8 led_brightness0_reg;
u8 mix_out0_reg;
u8 bank_brt_reg;
u8 bank_mix_reg;
u8 reset_reg;
};
static const struct lp50xx_chip_info lp50xx_chip_info_tbl[] = {
[LP5009] = {
.model_id = LP5009,
.max_modules = LP5009_MAX_LED_MODULES,
.num_leds = LP5009_MAX_LED_MODULES * LP50XX_LEDS_PER_MODULE,
.led_brightness0_reg = LP5012_LED0_BRT,
.mix_out0_reg = LP5012_OUT0_CLR,
.bank_brt_reg = LP5012_BNK_BRT,
.bank_mix_reg = LP5012_BNKA_CLR,
.reset_reg = LP5012_RESET,
.lp50xx_regmap_config = &lp5012_regmap_config,
},
[LP5012] = {
.model_id = LP5012,
.max_modules = LP5012_MAX_LED_MODULES,
.num_leds = LP5012_MAX_LED_MODULES * LP50XX_LEDS_PER_MODULE,
.led_brightness0_reg = LP5012_LED0_BRT,
.mix_out0_reg = LP5012_OUT0_CLR,
.bank_brt_reg = LP5012_BNK_BRT,
.bank_mix_reg = LP5012_BNKA_CLR,
.reset_reg = LP5012_RESET,
.lp50xx_regmap_config = &lp5012_regmap_config,
},
[LP5018] = {
.model_id = LP5018,
.max_modules = LP5018_MAX_LED_MODULES,
.num_leds = LP5018_MAX_LED_MODULES * LP50XX_LEDS_PER_MODULE,
.led_brightness0_reg = LP5024_LED0_BRT,
.mix_out0_reg = LP5024_OUT0_CLR,
.bank_brt_reg = LP5024_BNK_BRT,
.bank_mix_reg = LP5024_BNKA_CLR,
.reset_reg = LP5024_RESET,
.lp50xx_regmap_config = &lp5024_regmap_config,
},
[LP5024] = {
.model_id = LP5024,
.max_modules = LP5024_MAX_LED_MODULES,
.num_leds = LP5024_MAX_LED_MODULES * LP50XX_LEDS_PER_MODULE,
.led_brightness0_reg = LP5024_LED0_BRT,
.mix_out0_reg = LP5024_OUT0_CLR,
.bank_brt_reg = LP5024_BNK_BRT,
.bank_mix_reg = LP5024_BNKA_CLR,
.reset_reg = LP5024_RESET,
.lp50xx_regmap_config = &lp5024_regmap_config,
},
[LP5030] = {
.model_id = LP5030,
.max_modules = LP5030_MAX_LED_MODULES,
.num_leds = LP5030_MAX_LED_MODULES * LP50XX_LEDS_PER_MODULE,
.led_brightness0_reg = LP5036_LED0_BRT,
.mix_out0_reg = LP5036_OUT0_CLR,
.bank_brt_reg = LP5036_BNK_BRT,
.bank_mix_reg = LP5036_BNKA_CLR,
.reset_reg = LP5036_RESET,
.lp50xx_regmap_config = &lp5036_regmap_config,
},
[LP5036] = {
.model_id = LP5036,
.max_modules = LP5036_MAX_LED_MODULES,
.num_leds = LP5036_MAX_LED_MODULES * LP50XX_LEDS_PER_MODULE,
.led_brightness0_reg = LP5036_LED0_BRT,
.mix_out0_reg = LP5036_OUT0_CLR,
.bank_brt_reg = LP5036_BNK_BRT,
.bank_mix_reg = LP5036_BNKA_CLR,
.reset_reg = LP5036_RESET,
.lp50xx_regmap_config = &lp5036_regmap_config,
},
};
struct lp50xx_led {
struct led_classdev_mc mc_cdev;
struct lp50xx *priv;
unsigned long bank_modules;
int led_intensity[LP50XX_LEDS_PER_MODULE];
u8 ctrl_bank_enabled;
int led_number;
};
/**
* struct lp50xx -
* @enable_gpio: hardware enable gpio
* @regulator: LED supply regulator pointer
* @client: pointer to the I2C client
* @regmap: device register map
* @dev: pointer to the devices device struct
* @lock: lock for reading/writing the device
* @chip_info: chip specific information (ie num_leds)
* @num_of_banked_leds: holds the number of banked LEDs
* @leds: array of LED strings
*/
struct lp50xx {
struct gpio_desc *enable_gpio;
struct regulator *regulator;
struct i2c_client *client;
struct regmap *regmap;
struct device *dev;
struct mutex lock;
const struct lp50xx_chip_info *chip_info;
int num_of_banked_leds;
/* This needs to be at the end of the struct */
struct lp50xx_led leds[];
};
static struct lp50xx_led *mcled_cdev_to_led(struct led_classdev_mc *mc_cdev)
{
return container_of(mc_cdev, struct lp50xx_led, mc_cdev);
}
static int lp50xx_brightness_set(struct led_classdev *cdev,
enum led_brightness brightness)
{
struct led_classdev_mc *mc_dev = lcdev_to_mccdev(cdev);
struct lp50xx_led *led = mcled_cdev_to_led(mc_dev);
const struct lp50xx_chip_info *led_chip = led->priv->chip_info;
u8 led_offset, reg_val;
int ret = 0;
int i;
mutex_lock(&led->priv->lock);
if (led->ctrl_bank_enabled)
reg_val = led_chip->bank_brt_reg;
else
reg_val = led_chip->led_brightness0_reg +
led->led_number;
ret = regmap_write(led->priv->regmap, reg_val, brightness);
if (ret) {
dev_err(&led->priv->client->dev,
"Cannot write brightness value %d\n", ret);
goto out;
}
for (i = 0; i < led->mc_cdev.num_colors; i++) {
if (led->ctrl_bank_enabled) {
reg_val = led_chip->bank_mix_reg + i;
} else {
led_offset = (led->led_number * 3) + i;
reg_val = led_chip->mix_out0_reg + led_offset;
}
ret = regmap_write(led->priv->regmap, reg_val,
mc_dev->subled_info[i].intensity);
if (ret) {
dev_err(&led->priv->client->dev,
"Cannot write intensity value %d\n", ret);
goto out;
}
}
out:
mutex_unlock(&led->priv->lock);
return ret;
}
static int lp50xx_set_banks(struct lp50xx *priv, u32 led_banks[])
{
u8 led_config_lo, led_config_hi;
u32 bank_enable_mask = 0;
int ret;
int i;
for (i = 0; i < priv->chip_info->max_modules; i++) {
if (led_banks[i])
bank_enable_mask |= (1 << led_banks[i]);
}
led_config_lo = (u8)(bank_enable_mask & 0xff);
led_config_hi = (u8)(bank_enable_mask >> 8) & 0xff;
ret = regmap_write(priv->regmap, LP50XX_LED_CFG0, led_config_lo);
if (ret)
return ret;
if (priv->chip_info->model_id >= LP5030)
ret = regmap_write(priv->regmap, LP5036_LED_CFG1, led_config_hi);
return ret;
}
static int lp50xx_reset(struct lp50xx *priv)
{
return regmap_write(priv->regmap, priv->chip_info->reset_reg, LP50XX_SW_RESET);
}
static int lp50xx_enable_disable(struct lp50xx *priv, int enable_disable)
{
int ret;
if (priv->enable_gpio) {
ret = gpiod_direction_output(priv->enable_gpio, enable_disable);
if (ret)
return ret;
}
if (enable_disable)
return regmap_write(priv->regmap, LP50XX_DEV_CFG0, LP50XX_CHIP_EN);
else
return regmap_write(priv->regmap, LP50XX_DEV_CFG0, 0);
}
static int lp50xx_probe_leds(struct fwnode_handle *child, struct lp50xx *priv,
struct lp50xx_led *led, int num_leds)
{
u32 led_banks[LP5036_MAX_LED_MODULES] = {0};
int led_number;
int ret;
if (num_leds > 1) {
if (num_leds > priv->chip_info->max_modules) {
dev_err(&priv->client->dev, "reg property is invalid\n");
return -EINVAL;
}
priv->num_of_banked_leds = num_leds;
ret = fwnode_property_read_u32_array(child, "reg", led_banks, num_leds);
if (ret) {
dev_err(&priv->client->dev, "reg property is missing\n");
return ret;
}
ret = lp50xx_set_banks(priv, led_banks);
if (ret) {
dev_err(&priv->client->dev, "Cannot setup banked LEDs\n");
return ret;
}
led->ctrl_bank_enabled = 1;
} else {
ret = fwnode_property_read_u32(child, "reg", &led_number);
if (ret) {
dev_err(&priv->client->dev, "led reg property missing\n");
return ret;
}
if (led_number > priv->chip_info->num_leds) {
dev_err(&priv->client->dev, "led-sources property is invalid\n");
return -EINVAL;
}
led->led_number = led_number;
}
return 0;
}
static int lp50xx_probe_dt(struct lp50xx *priv)
{
struct fwnode_handle *child = NULL;
struct fwnode_handle *led_node = NULL;
struct led_init_data init_data = {};
struct led_classdev *led_cdev;
struct mc_subled *mc_led_info;
struct lp50xx_led *led;
int ret = -EINVAL;
int num_colors;
u32 color_id;
int i = 0;
priv->enable_gpio = devm_gpiod_get_optional(priv->dev, "enable", GPIOD_OUT_LOW);
if (IS_ERR(priv->enable_gpio)) {
ret = PTR_ERR(priv->enable_gpio);
dev_err(&priv->client->dev, "Failed to get enable gpio: %d\n",
ret);
return ret;
}
priv->regulator = devm_regulator_get(priv->dev, "vled");
if (IS_ERR(priv->regulator))
priv->regulator = NULL;
device_for_each_child_node(priv->dev, child) {
led = &priv->leds[i];
ret = fwnode_property_count_u32(child, "reg");
if (ret < 0) {
dev_err(&priv->client->dev, "reg property is invalid\n");
goto child_out;
}
ret = lp50xx_probe_leds(child, priv, led, ret);
if (ret)
goto child_out;
init_data.fwnode = child;
num_colors = 0;
/*
* There are only 3 LEDs per module otherwise they should be
* banked which also is presented as 3 LEDs.
*/
mc_led_info = devm_kcalloc(priv->dev, LP50XX_LEDS_PER_MODULE,
sizeof(*mc_led_info), GFP_KERNEL);
if (!mc_led_info)
return -ENOMEM;
fwnode_for_each_child_node(child, led_node) {
ret = fwnode_property_read_u32(led_node, "color",
&color_id);
if (ret) {
dev_err(priv->dev, "Cannot read color\n");
goto child_out;
}
mc_led_info[num_colors].color_index = color_id;
num_colors++;
}
led->priv = priv;
led->mc_cdev.num_colors = num_colors;
led->mc_cdev.subled_info = mc_led_info;
led_cdev = &led->mc_cdev.led_cdev;
led_cdev->brightness_set_blocking = lp50xx_brightness_set;
ret = devm_led_classdev_multicolor_register_ext(&priv->client->dev,
&led->mc_cdev,
&init_data);
if (ret) {
dev_err(&priv->client->dev, "led register err: %d\n",
ret);
goto child_out;
}
i++;
fwnode_handle_put(child);
}
return 0;
child_out:
fwnode_handle_put(child);
return ret;
}
static int lp50xx_probe(struct i2c_client *client,
const struct i2c_device_id *id)
{
struct lp50xx *led;
int count;
int ret;
count = device_get_child_node_count(&client->dev);
if (!count) {
dev_err(&client->dev, "LEDs are not defined in device tree!");
return -ENODEV;
}
led = devm_kzalloc(&client->dev, struct_size(led, leds, count),
GFP_KERNEL);
if (!led)
return -ENOMEM;
mutex_init(&led->lock);
led->client = client;
led->dev = &client->dev;
led->chip_info = &lp50xx_chip_info_tbl[id->driver_data];
i2c_set_clientdata(client, led);
led->regmap = devm_regmap_init_i2c(client,
led->chip_info->lp50xx_regmap_config);
if (IS_ERR(led->regmap)) {
ret = PTR_ERR(led->regmap);
dev_err(&client->dev, "Failed to allocate register map: %d\n",
ret);
return ret;
}
ret = lp50xx_reset(led);
if (ret)
return ret;
ret = lp50xx_enable_disable(led, 1);
if (ret)
return ret;
return lp50xx_probe_dt(led);
}
static int lp50xx_remove(struct i2c_client *client)
{
struct lp50xx *led = i2c_get_clientdata(client);
int ret;
ret = lp50xx_enable_disable(led, 0);
if (ret) {
dev_err(&led->client->dev, "Failed to disable chip\n");
return ret;
}
if (led->regulator) {
ret = regulator_disable(led->regulator);
if (ret)
dev_err(&led->client->dev,
"Failed to disable regulator\n");
}
mutex_destroy(&led->lock);
return 0;
}
static const struct i2c_device_id lp50xx_id[] = {
{ "lp5009", LP5009 },
{ "lp5012", LP5012 },
{ "lp5018", LP5018 },
{ "lp5024", LP5024 },
{ "lp5030", LP5030 },
{ "lp5036", LP5036 },
{ }
};
MODULE_DEVICE_TABLE(i2c, lp50xx_id);
static const struct of_device_id of_lp50xx_leds_match[] = {
{ .compatible = "ti,lp5009", .data = (void *)LP5009 },
{ .compatible = "ti,lp5012", .data = (void *)LP5012 },
{ .compatible = "ti,lp5018", .data = (void *)LP5018 },
{ .compatible = "ti,lp5024", .data = (void *)LP5024 },
{ .compatible = "ti,lp5030", .data = (void *)LP5030 },
{ .compatible = "ti,lp5036", .data = (void *)LP5036 },
{},
};
MODULE_DEVICE_TABLE(of, of_lp50xx_leds_match);
static struct i2c_driver lp50xx_driver = {
.driver = {
.name = "lp50xx",
.of_match_table = of_lp50xx_leds_match,
},
.probe = lp50xx_probe,
.remove = lp50xx_remove,
.id_table = lp50xx_id,
};
module_i2c_driver(lp50xx_driver);
MODULE_DESCRIPTION("Texas Instruments LP50XX LED driver");
MODULE_AUTHOR("Dan Murphy <dmurphy@ti.com>");
MODULE_LICENSE("GPL v2");
...@@ -523,7 +523,7 @@ static int lp5521_probe(struct i2c_client *client, ...@@ -523,7 +523,7 @@ static int lp5521_probe(struct i2c_client *client,
struct lp55xx_chip *chip; struct lp55xx_chip *chip;
struct lp55xx_led *led; struct lp55xx_led *led;
struct lp55xx_platform_data *pdata = dev_get_platdata(&client->dev); struct lp55xx_platform_data *pdata = dev_get_platdata(&client->dev);
struct device_node *np = client->dev.of_node; struct device_node *np = dev_of_node(&client->dev);
chip = devm_kzalloc(&client->dev, sizeof(*chip), GFP_KERNEL); chip = devm_kzalloc(&client->dev, sizeof(*chip), GFP_KERNEL);
if (!chip) if (!chip)
......
...@@ -891,7 +891,7 @@ static int lp5523_probe(struct i2c_client *client, ...@@ -891,7 +891,7 @@ static int lp5523_probe(struct i2c_client *client,
struct lp55xx_chip *chip; struct lp55xx_chip *chip;
struct lp55xx_led *led; struct lp55xx_led *led;
struct lp55xx_platform_data *pdata = dev_get_platdata(&client->dev); struct lp55xx_platform_data *pdata = dev_get_platdata(&client->dev);
struct device_node *np = client->dev.of_node; struct device_node *np = dev_of_node(&client->dev);
chip = devm_kzalloc(&client->dev, sizeof(*chip), GFP_KERNEL); chip = devm_kzalloc(&client->dev, sizeof(*chip), GFP_KERNEL);
if (!chip) if (!chip)
......
...@@ -518,7 +518,7 @@ static int lp5562_probe(struct i2c_client *client, ...@@ -518,7 +518,7 @@ static int lp5562_probe(struct i2c_client *client,
struct lp55xx_chip *chip; struct lp55xx_chip *chip;
struct lp55xx_led *led; struct lp55xx_led *led;
struct lp55xx_platform_data *pdata = dev_get_platdata(&client->dev); struct lp55xx_platform_data *pdata = dev_get_platdata(&client->dev);
struct device_node *np = client->dev.of_node; struct device_node *np = dev_of_node(&client->dev);
chip = devm_kzalloc(&client->dev, sizeof(*chip), GFP_KERNEL); chip = devm_kzalloc(&client->dev, sizeof(*chip), GFP_KERNEL);
if (!chip) if (!chip)
......
...@@ -611,11 +611,13 @@ static int lp55xx_parse_multi_led(struct device_node *np, ...@@ -611,11 +611,13 @@ static int lp55xx_parse_multi_led(struct device_node *np,
struct device_node *child; struct device_node *child;
int num_colors = 0, ret; int num_colors = 0, ret;
for_each_child_of_node(np, child) { for_each_available_child_of_node(np, child) {
ret = lp55xx_parse_multi_led_child(child, cfg, child_number, ret = lp55xx_parse_multi_led_child(child, cfg, child_number,
num_colors); num_colors);
if (ret) if (ret) {
of_node_put(child);
return ret; return ret;
}
num_colors++; num_colors++;
} }
...@@ -665,7 +667,7 @@ struct lp55xx_platform_data *lp55xx_of_populate_pdata(struct device *dev, ...@@ -665,7 +667,7 @@ struct lp55xx_platform_data *lp55xx_of_populate_pdata(struct device *dev,
if (!pdata) if (!pdata)
return ERR_PTR(-ENOMEM); return ERR_PTR(-ENOMEM);
num_channels = of_get_child_count(np); num_channels = of_get_available_child_count(np);
if (num_channels == 0) { if (num_channels == 0) {
dev_err(dev, "no LED channels\n"); dev_err(dev, "no LED channels\n");
return ERR_PTR(-EINVAL); return ERR_PTR(-EINVAL);
...@@ -679,10 +681,12 @@ struct lp55xx_platform_data *lp55xx_of_populate_pdata(struct device *dev, ...@@ -679,10 +681,12 @@ struct lp55xx_platform_data *lp55xx_of_populate_pdata(struct device *dev,
pdata->num_channels = num_channels; pdata->num_channels = num_channels;
cfg->max_channel = chip->cfg->max_channel; cfg->max_channel = chip->cfg->max_channel;
for_each_child_of_node(np, child) { for_each_available_child_of_node(np, child) {
ret = lp55xx_parse_logical_led(child, cfg, i); ret = lp55xx_parse_logical_led(child, cfg, i);
if (ret) if (ret) {
of_node_put(child);
return ERR_PTR(-EINVAL); return ERR_PTR(-EINVAL);
}
i++; i++;
} }
......
...@@ -306,7 +306,7 @@ static int lp8501_probe(struct i2c_client *client, ...@@ -306,7 +306,7 @@ static int lp8501_probe(struct i2c_client *client,
struct lp55xx_chip *chip; struct lp55xx_chip *chip;
struct lp55xx_led *led; struct lp55xx_led *led;
struct lp55xx_platform_data *pdata = dev_get_platdata(&client->dev); struct lp55xx_platform_data *pdata = dev_get_platdata(&client->dev);
struct device_node *np = client->dev.of_node; struct device_node *np = dev_of_node(&client->dev);
chip = devm_kzalloc(&client->dev, sizeof(*chip), GFP_KERNEL); chip = devm_kzalloc(&client->dev, sizeof(*chip), GFP_KERNEL);
if (!chip) if (!chip)
......
...@@ -380,7 +380,7 @@ static int lp8860_probe(struct i2c_client *client, ...@@ -380,7 +380,7 @@ static int lp8860_probe(struct i2c_client *client,
{ {
int ret; int ret;
struct lp8860_led *led; struct lp8860_led *led;
struct device_node *np = client->dev.of_node; struct device_node *np = dev_of_node(&client->dev);
struct device_node *child_node; struct device_node *child_node;
struct led_init_data init_data = {}; struct led_init_data init_data = {};
...@@ -392,10 +392,6 @@ static int lp8860_probe(struct i2c_client *client, ...@@ -392,10 +392,6 @@ static int lp8860_probe(struct i2c_client *client,
if (!child_node) if (!child_node)
return -EINVAL; return -EINVAL;
led->led_dev.default_trigger = of_get_property(child_node,
"linux,default-trigger",
NULL);
led->enable_gpio = devm_gpiod_get_optional(&client->dev, led->enable_gpio = devm_gpiod_get_optional(&client->dev,
"enable", GPIOD_OUT_LOW); "enable", GPIOD_OUT_LOW);
if (IS_ERR(led->enable_gpio)) { if (IS_ERR(led->enable_gpio)) {
......
...@@ -68,7 +68,7 @@ static int lt3593_led_probe(struct platform_device *pdev) ...@@ -68,7 +68,7 @@ static int lt3593_led_probe(struct platform_device *pdev)
struct led_init_data init_data = {}; struct led_init_data init_data = {};
const char *tmp; const char *tmp;
if (!dev->of_node) if (!dev_of_node(dev))
return -ENODEV; return -ENODEV;
led_data = devm_kzalloc(dev, sizeof(*led_data), GFP_KERNEL); led_data = devm_kzalloc(dev, sizeof(*led_data), GFP_KERNEL);
...@@ -86,9 +86,6 @@ static int lt3593_led_probe(struct platform_device *pdev) ...@@ -86,9 +86,6 @@ static int lt3593_led_probe(struct platform_device *pdev)
child = device_get_next_child_node(dev, NULL); child = device_get_next_child_node(dev, NULL);
fwnode_property_read_string(child, "linux,default-trigger",
&led_data->cdev.default_trigger);
if (!fwnode_property_read_string(child, "default-state", &tmp)) { if (!fwnode_property_read_string(child, "default-state", &tmp)) {
if (!strcmp(tmp, "on")) if (!strcmp(tmp, "on"))
state = LEDS_GPIO_DEFSTATE_ON; state = LEDS_GPIO_DEFSTATE_ON;
...@@ -107,7 +104,6 @@ static int lt3593_led_probe(struct platform_device *pdev) ...@@ -107,7 +104,6 @@ static int lt3593_led_probe(struct platform_device *pdev)
return ret; return ret;
} }
led_data->cdev.dev->of_node = dev->of_node;
platform_set_drvdata(pdev, led_data); platform_set_drvdata(pdev, led_data);
return 0; return 0;
......
...@@ -66,7 +66,6 @@ static int max77650_led_probe(struct platform_device *pdev) ...@@ -66,7 +66,6 @@ static int max77650_led_probe(struct platform_device *pdev)
struct max77650_led *leds, *led; struct max77650_led *leds, *led;
struct device *dev; struct device *dev;
struct regmap *map; struct regmap *map;
const char *label;
int rv, num_leds; int rv, num_leds;
u32 reg; u32 reg;
...@@ -86,6 +85,8 @@ static int max77650_led_probe(struct platform_device *pdev) ...@@ -86,6 +85,8 @@ static int max77650_led_probe(struct platform_device *pdev)
return -ENODEV; return -ENODEV;
device_for_each_child_node(dev, child) { device_for_each_child_node(dev, child) {
struct led_init_data init_data = {};
rv = fwnode_property_read_u32(child, "reg", &reg); rv = fwnode_property_read_u32(child, "reg", &reg);
if (rv || reg >= MAX77650_LED_NUM_LEDS) { if (rv || reg >= MAX77650_LED_NUM_LEDS) {
rv = -EINVAL; rv = -EINVAL;
...@@ -99,22 +100,13 @@ static int max77650_led_probe(struct platform_device *pdev) ...@@ -99,22 +100,13 @@ static int max77650_led_probe(struct platform_device *pdev)
led->cdev.brightness_set_blocking = max77650_led_brightness_set; led->cdev.brightness_set_blocking = max77650_led_brightness_set;
led->cdev.max_brightness = MAX77650_LED_MAX_BRIGHTNESS; led->cdev.max_brightness = MAX77650_LED_MAX_BRIGHTNESS;
rv = fwnode_property_read_string(child, "label", &label); init_data.fwnode = child;
if (rv) { init_data.devicename = "max77650";
led->cdev.name = "max77650::"; /* for backwards compatibility if `label` is not present */
} else { init_data.default_label = ":";
led->cdev.name = devm_kasprintf(dev, GFP_KERNEL,
"max77650:%s", label);
if (!led->cdev.name) {
rv = -ENOMEM;
goto err_node_put;
}
}
fwnode_property_read_string(child, "linux,default-trigger",
&led->cdev.default_trigger);
rv = devm_led_classdev_register(dev, &led->cdev); rv = devm_led_classdev_register_ext(dev, &led->cdev,
&init_data);
if (rv) if (rv)
goto err_node_put; goto err_node_put;
......
...@@ -599,7 +599,7 @@ static int max77693_led_parse_dt(struct max77693_led_device *led, ...@@ -599,7 +599,7 @@ static int max77693_led_parse_dt(struct max77693_led_device *led,
{ {
struct device *dev = &led->pdev->dev; struct device *dev = &led->pdev->dev;
struct max77693_sub_led *sub_leds = led->sub_leds; struct max77693_sub_led *sub_leds = led->sub_leds;
struct device_node *node = dev->of_node, *child_node; struct device_node *node = dev_of_node(dev), *child_node;
struct property *prop; struct property *prop;
u32 led_sources[2]; u32 led_sources[2];
int i, ret, fled_id; int i, ret, fled_id;
......
...@@ -121,7 +121,7 @@ static struct mc13xxx_leds_platform_data __init *mc13xxx_led_probe_dt( ...@@ -121,7 +121,7 @@ static struct mc13xxx_leds_platform_data __init *mc13xxx_led_probe_dt(
if (!pdata) if (!pdata)
return ERR_PTR(-ENOMEM); return ERR_PTR(-ENOMEM);
parent = of_get_child_by_name(dev->parent->of_node, "leds"); parent = of_get_child_by_name(dev_of_node(dev->parent), "leds");
if (!parent) if (!parent)
goto out_node_put; goto out_node_put;
...@@ -131,7 +131,7 @@ static struct mc13xxx_leds_platform_data __init *mc13xxx_led_probe_dt( ...@@ -131,7 +131,7 @@ static struct mc13xxx_leds_platform_data __init *mc13xxx_led_probe_dt(
if (ret) if (ret)
goto out_node_put; goto out_node_put;
pdata->num_leds = of_get_child_count(parent); pdata->num_leds = of_get_available_child_count(parent);
pdata->led = devm_kcalloc(dev, pdata->num_leds, sizeof(*pdata->led), pdata->led = devm_kcalloc(dev, pdata->num_leds, sizeof(*pdata->led),
GFP_KERNEL); GFP_KERNEL);
...@@ -140,7 +140,7 @@ static struct mc13xxx_leds_platform_data __init *mc13xxx_led_probe_dt( ...@@ -140,7 +140,7 @@ static struct mc13xxx_leds_platform_data __init *mc13xxx_led_probe_dt(
goto out_node_put; goto out_node_put;
} }
for_each_child_of_node(parent, child) { for_each_available_child_of_node(parent, child) {
const char *str; const char *str;
u32 tmp; u32 tmp;
...@@ -192,7 +192,7 @@ static int __init mc13xxx_led_probe(struct platform_device *pdev) ...@@ -192,7 +192,7 @@ static int __init mc13xxx_led_probe(struct platform_device *pdev)
leds->master = mcdev; leds->master = mcdev;
platform_set_drvdata(pdev, leds); platform_set_drvdata(pdev, leds);
if (dev->parent->of_node) { if (dev_of_node(dev->parent)) {
pdata = mc13xxx_led_probe_dt(pdev); pdata = mc13xxx_led_probe_dt(pdev);
if (IS_ERR(pdata)) if (IS_ERR(pdata))
return PTR_ERR(pdata); return PTR_ERR(pdata);
......
...@@ -248,15 +248,6 @@ static int mt6323_led_set_blink(struct led_classdev *cdev, ...@@ -248,15 +248,6 @@ static int mt6323_led_set_blink(struct led_classdev *cdev,
u8 duty_hw; u8 duty_hw;
int ret; int ret;
/*
* Units are in ms, if over the hardware able
* to support, fallback into software blink
*/
period = *delay_on + *delay_off;
if (period > MT6323_MAX_PERIOD)
return -EINVAL;
/* /*
* LED subsystem requires a default user * LED subsystem requires a default user
* friendly blink pattern for the LED so using * friendly blink pattern for the LED so using
...@@ -268,6 +259,15 @@ static int mt6323_led_set_blink(struct led_classdev *cdev, ...@@ -268,6 +259,15 @@ static int mt6323_led_set_blink(struct led_classdev *cdev,
*delay_off = 500; *delay_off = 500;
} }
/*
* Units are in ms, if over the hardware able
* to support, fallback into software blink
*/
period = *delay_on + *delay_off;
if (period > MT6323_MAX_PERIOD)
return -EINVAL;
/* /*
* Calculate duty_hw based on the percentage of period during * Calculate duty_hw based on the percentage of period during
* which the led is ON. * which the led is ON.
...@@ -342,11 +342,6 @@ static int mt6323_led_set_dt_default(struct led_classdev *cdev, ...@@ -342,11 +342,6 @@ static int mt6323_led_set_dt_default(struct led_classdev *cdev,
const char *state; const char *state;
int ret = 0; int ret = 0;
led->cdev.name = of_get_property(np, "label", NULL) ? : np->name;
led->cdev.default_trigger = of_get_property(np,
"linux,default-trigger",
NULL);
state = of_get_property(np, "default-state", NULL); state = of_get_property(np, "default-state", NULL);
if (state) { if (state) {
if (!strcmp(state, "keep")) { if (!strcmp(state, "keep")) {
...@@ -369,9 +364,9 @@ static int mt6323_led_set_dt_default(struct led_classdev *cdev, ...@@ -369,9 +364,9 @@ static int mt6323_led_set_dt_default(struct led_classdev *cdev,
static int mt6323_led_probe(struct platform_device *pdev) static int mt6323_led_probe(struct platform_device *pdev)
{ {
struct device *dev = &pdev->dev; struct device *dev = &pdev->dev;
struct device_node *np = pdev->dev.of_node; struct device_node *np = dev_of_node(dev);
struct device_node *child; struct device_node *child;
struct mt6397_chip *hw = dev_get_drvdata(pdev->dev.parent); struct mt6397_chip *hw = dev_get_drvdata(dev->parent);
struct mt6323_leds *leds; struct mt6323_leds *leds;
struct mt6323_led *led; struct mt6323_led *led;
int ret; int ret;
...@@ -402,6 +397,8 @@ static int mt6323_led_probe(struct platform_device *pdev) ...@@ -402,6 +397,8 @@ static int mt6323_led_probe(struct platform_device *pdev)
} }
for_each_available_child_of_node(np, child) { for_each_available_child_of_node(np, child) {
struct led_init_data init_data = {};
ret = of_property_read_u32(child, "reg", &reg); ret = of_property_read_u32(child, "reg", &reg);
if (ret) { if (ret) {
dev_err(dev, "Failed to read led 'reg' property\n"); dev_err(dev, "Failed to read led 'reg' property\n");
...@@ -437,13 +434,14 @@ static int mt6323_led_probe(struct platform_device *pdev) ...@@ -437,13 +434,14 @@ static int mt6323_led_probe(struct platform_device *pdev)
goto put_child_node; goto put_child_node;
} }
ret = devm_led_classdev_register(dev, &leds->led[reg]->cdev); init_data.fwnode = of_fwnode_handle(child);
ret = devm_led_classdev_register_ext(dev, &leds->led[reg]->cdev,
&init_data);
if (ret) { if (ret) {
dev_err(&pdev->dev, "Failed to register LED: %d\n", dev_err(dev, "Failed to register LED: %d\n", ret);
ret);
goto put_child_node; goto put_child_node;
} }
leds->led[reg]->cdev.dev->of_node = child;
} }
return 0; return 0;
......
...@@ -419,7 +419,7 @@ static int netxbig_gpio_ext_get(struct device *dev, ...@@ -419,7 +419,7 @@ static int netxbig_gpio_ext_get(struct device *dev,
static int netxbig_leds_get_of_pdata(struct device *dev, static int netxbig_leds_get_of_pdata(struct device *dev,
struct netxbig_led_platform_data *pdata) struct netxbig_led_platform_data *pdata)
{ {
struct device_node *np = dev->of_node; struct device_node *np = dev_of_node(dev);
struct device_node *gpio_ext_np; struct device_node *gpio_ext_np;
struct platform_device *gpio_ext_pdev; struct platform_device *gpio_ext_pdev;
struct device *gpio_ext_dev; struct device *gpio_ext_dev;
...@@ -485,7 +485,7 @@ static int netxbig_leds_get_of_pdata(struct device *dev, ...@@ -485,7 +485,7 @@ static int netxbig_leds_get_of_pdata(struct device *dev,
} }
/* LEDs */ /* LEDs */
num_leds = of_get_child_count(np); num_leds = of_get_available_child_count(np);
if (!num_leds) { if (!num_leds) {
dev_err(dev, "No LED subnodes found in DT\n"); dev_err(dev, "No LED subnodes found in DT\n");
return -ENODEV; return -ENODEV;
...@@ -496,7 +496,7 @@ static int netxbig_leds_get_of_pdata(struct device *dev, ...@@ -496,7 +496,7 @@ static int netxbig_leds_get_of_pdata(struct device *dev,
return -ENOMEM; return -ENOMEM;
led = leds; led = leds;
for_each_child_of_node(np, child) { for_each_available_child_of_node(np, child) {
const char *string; const char *string;
int *mode_val; int *mode_val;
int num_modes; int num_modes;
......
...@@ -24,25 +24,16 @@ enum ns2_led_modes { ...@@ -24,25 +24,16 @@ enum ns2_led_modes {
NS_V2_LED_SATA, NS_V2_LED_SATA,
}; };
/*
* If the size of this structure or types of its members is changed,
* the filling of array modval in function ns2_led_register must be changed
* accordingly.
*/
struct ns2_led_modval { struct ns2_led_modval {
enum ns2_led_modes mode; u32 mode;
int cmd_level; u32 cmd_level;
int slow_level; u32 slow_level;
}; } __packed;
struct ns2_led {
const char *name;
const char *default_trigger;
struct gpio_desc *cmd;
struct gpio_desc *slow;
int num_modes;
struct ns2_led_modval *modval;
};
struct ns2_led_platform_data {
int num_leds;
struct ns2_led *leds;
};
/* /*
* The Network Space v2 dual-GPIO LED is wired to a CPLD. Three different LED * The Network Space v2 dual-GPIO LED is wired to a CPLD. Three different LED
...@@ -51,7 +42,7 @@ struct ns2_led_platform_data { ...@@ -51,7 +42,7 @@ struct ns2_led_platform_data {
* for the command/slow GPIOs corresponds to a LED mode. * for the command/slow GPIOs corresponds to a LED mode.
*/ */
struct ns2_led_data { struct ns2_led {
struct led_classdev cdev; struct led_classdev cdev;
struct gpio_desc *cmd; struct gpio_desc *cmd;
struct gpio_desc *slow; struct gpio_desc *slow;
...@@ -62,77 +53,67 @@ struct ns2_led_data { ...@@ -62,77 +53,67 @@ struct ns2_led_data {
struct ns2_led_modval *modval; struct ns2_led_modval *modval;
}; };
static int ns2_led_get_mode(struct ns2_led_data *led_dat, static int ns2_led_get_mode(struct ns2_led *led, enum ns2_led_modes *mode)
enum ns2_led_modes *mode)
{ {
int i; int i;
int ret = -EINVAL;
int cmd_level; int cmd_level;
int slow_level; int slow_level;
cmd_level = gpiod_get_value_cansleep(led_dat->cmd); cmd_level = gpiod_get_value_cansleep(led->cmd);
slow_level = gpiod_get_value_cansleep(led_dat->slow); slow_level = gpiod_get_value_cansleep(led->slow);
for (i = 0; i < led_dat->num_modes; i++) { for (i = 0; i < led->num_modes; i++) {
if (cmd_level == led_dat->modval[i].cmd_level && if (cmd_level == led->modval[i].cmd_level &&
slow_level == led_dat->modval[i].slow_level) { slow_level == led->modval[i].slow_level) {
*mode = led_dat->modval[i].mode; *mode = led->modval[i].mode;
ret = 0; return 0;
break;
} }
} }
return ret; return -EINVAL;
} }
static void ns2_led_set_mode(struct ns2_led_data *led_dat, static void ns2_led_set_mode(struct ns2_led *led, enum ns2_led_modes mode)
enum ns2_led_modes mode)
{ {
int i; int i;
bool found = false;
unsigned long flags; unsigned long flags;
for (i = 0; i < led_dat->num_modes; i++) for (i = 0; i < led->num_modes; i++)
if (mode == led_dat->modval[i].mode) { if (mode == led->modval[i].mode)
found = true;
break; break;
}
if (!found) if (i == led->num_modes)
return; return;
write_lock_irqsave(&led_dat->rw_lock, flags); write_lock_irqsave(&led->rw_lock, flags);
if (!led_dat->can_sleep) { if (!led->can_sleep) {
gpiod_set_value(led_dat->cmd, gpiod_set_value(led->cmd, led->modval[i].cmd_level);
led_dat->modval[i].cmd_level); gpiod_set_value(led->slow, led->modval[i].slow_level);
gpiod_set_value(led_dat->slow,
led_dat->modval[i].slow_level);
goto exit_unlock; goto exit_unlock;
} }
gpiod_set_value_cansleep(led_dat->cmd, led_dat->modval[i].cmd_level); gpiod_set_value_cansleep(led->cmd, led->modval[i].cmd_level);
gpiod_set_value_cansleep(led_dat->slow, led_dat->modval[i].slow_level); gpiod_set_value_cansleep(led->slow, led->modval[i].slow_level);
exit_unlock: exit_unlock:
write_unlock_irqrestore(&led_dat->rw_lock, flags); write_unlock_irqrestore(&led->rw_lock, flags);
} }
static void ns2_led_set(struct led_classdev *led_cdev, static void ns2_led_set(struct led_classdev *led_cdev,
enum led_brightness value) enum led_brightness value)
{ {
struct ns2_led_data *led_dat = struct ns2_led *led = container_of(led_cdev, struct ns2_led, cdev);
container_of(led_cdev, struct ns2_led_data, cdev);
enum ns2_led_modes mode; enum ns2_led_modes mode;
if (value == LED_OFF) if (value == LED_OFF)
mode = NS_V2_LED_OFF; mode = NS_V2_LED_OFF;
else if (led_dat->sata) else if (led->sata)
mode = NS_V2_LED_SATA; mode = NS_V2_LED_SATA;
else else
mode = NS_V2_LED_ON; mode = NS_V2_LED_ON;
ns2_led_set_mode(led_dat, mode); ns2_led_set_mode(led, mode);
} }
static int ns2_led_set_blocking(struct led_classdev *led_cdev, static int ns2_led_set_blocking(struct led_classdev *led_cdev,
...@@ -147,8 +128,7 @@ static ssize_t ns2_led_sata_store(struct device *dev, ...@@ -147,8 +128,7 @@ static ssize_t ns2_led_sata_store(struct device *dev,
const char *buff, size_t count) const char *buff, size_t count)
{ {
struct led_classdev *led_cdev = dev_get_drvdata(dev); struct led_classdev *led_cdev = dev_get_drvdata(dev);
struct ns2_led_data *led_dat = struct ns2_led *led = container_of(led_cdev, struct ns2_led, cdev);
container_of(led_cdev, struct ns2_led_data, cdev);
int ret; int ret;
unsigned long enable; unsigned long enable;
...@@ -158,18 +138,18 @@ static ssize_t ns2_led_sata_store(struct device *dev, ...@@ -158,18 +138,18 @@ static ssize_t ns2_led_sata_store(struct device *dev,
enable = !!enable; enable = !!enable;
if (led_dat->sata == enable) if (led->sata == enable)
goto exit; goto exit;
led_dat->sata = enable; led->sata = enable;
if (!led_get_brightness(led_cdev)) if (!led_get_brightness(led_cdev))
goto exit; goto exit;
if (enable) if (enable)
ns2_led_set_mode(led_dat, NS_V2_LED_SATA); ns2_led_set_mode(led, NS_V2_LED_SATA);
else else
ns2_led_set_mode(led_dat, NS_V2_LED_ON); ns2_led_set_mode(led, NS_V2_LED_ON);
exit: exit:
return count; return count;
...@@ -179,10 +159,9 @@ static ssize_t ns2_led_sata_show(struct device *dev, ...@@ -179,10 +159,9 @@ static ssize_t ns2_led_sata_show(struct device *dev,
struct device_attribute *attr, char *buf) struct device_attribute *attr, char *buf)
{ {
struct led_classdev *led_cdev = dev_get_drvdata(dev); struct led_classdev *led_cdev = dev_get_drvdata(dev);
struct ns2_led_data *led_dat = struct ns2_led *led = container_of(led_cdev, struct ns2_led, cdev);
container_of(led_cdev, struct ns2_led_data, cdev);
return sprintf(buf, "%d\n", led_dat->sata); return sprintf(buf, "%d\n", led->sata);
} }
static DEVICE_ATTR(sata, 0644, ns2_led_sata_show, ns2_led_sata_store); static DEVICE_ATTR(sata, 0644, ns2_led_sata_show, ns2_led_sata_store);
...@@ -193,147 +172,94 @@ static struct attribute *ns2_led_attrs[] = { ...@@ -193,147 +172,94 @@ static struct attribute *ns2_led_attrs[] = {
}; };
ATTRIBUTE_GROUPS(ns2_led); ATTRIBUTE_GROUPS(ns2_led);
static int static int ns2_led_register(struct device *dev, struct fwnode_handle *node,
create_ns2_led(struct platform_device *pdev, struct ns2_led_data *led_dat, struct ns2_led *led)
const struct ns2_led *template)
{ {
int ret; struct led_init_data init_data = {};
struct ns2_led_modval *modval;
enum ns2_led_modes mode; enum ns2_led_modes mode;
int nmodes, ret;
led->cmd = devm_fwnode_gpiod_get_index(dev, node, "cmd", 0, GPIOD_ASIS,
fwnode_get_name(node));
if (IS_ERR(led->cmd))
return PTR_ERR(led->cmd);
rwlock_init(&led_dat->rw_lock); led->slow = devm_fwnode_gpiod_get_index(dev, node, "slow", 0,
GPIOD_ASIS,
led_dat->cdev.name = template->name; fwnode_get_name(node));
led_dat->cdev.default_trigger = template->default_trigger; if (IS_ERR(led->slow))
led_dat->cdev.blink_set = NULL; return PTR_ERR(led->slow);
led_dat->cdev.flags |= LED_CORE_SUSPENDRESUME;
led_dat->cdev.groups = ns2_led_groups; ret = fwnode_property_count_u32(node, "modes-map");
led_dat->cmd = template->cmd; if (ret < 0 || ret % 3) {
led_dat->slow = template->slow; dev_err(dev, "Missing or malformed modes-map for %pfw\n", node);
led_dat->can_sleep = gpiod_cansleep(led_dat->cmd) | return -EINVAL;
gpiod_cansleep(led_dat->slow); }
if (led_dat->can_sleep)
led_dat->cdev.brightness_set_blocking = ns2_led_set_blocking; nmodes = ret / 3;
modval = devm_kcalloc(dev, nmodes, sizeof(*modval), GFP_KERNEL);
if (!modval)
return -ENOMEM;
fwnode_property_read_u32_array(node, "modes-map", (void *)modval,
nmodes * 3);
rwlock_init(&led->rw_lock);
led->cdev.blink_set = NULL;
led->cdev.flags |= LED_CORE_SUSPENDRESUME;
led->cdev.groups = ns2_led_groups;
led->can_sleep = gpiod_cansleep(led->cmd) || gpiod_cansleep(led->slow);
if (led->can_sleep)
led->cdev.brightness_set_blocking = ns2_led_set_blocking;
else else
led_dat->cdev.brightness_set = ns2_led_set; led->cdev.brightness_set = ns2_led_set;
led_dat->modval = template->modval; led->num_modes = nmodes;
led_dat->num_modes = template->num_modes; led->modval = modval;
ret = ns2_led_get_mode(led_dat, &mode); ret = ns2_led_get_mode(led, &mode);
if (ret < 0) if (ret < 0)
return ret; return ret;
/* Set LED initial state. */ /* Set LED initial state. */
led_dat->sata = (mode == NS_V2_LED_SATA) ? 1 : 0; led->sata = (mode == NS_V2_LED_SATA) ? 1 : 0;
led_dat->cdev.brightness = led->cdev.brightness = (mode == NS_V2_LED_OFF) ? LED_OFF : LED_FULL;
(mode == NS_V2_LED_OFF) ? LED_OFF : LED_FULL;
ret = led_classdev_register(&pdev->dev, &led_dat->cdev); init_data.fwnode = node;
if (ret < 0)
return ret;
return 0; ret = devm_led_classdev_register_ext(dev, &led->cdev, &init_data);
} if (ret)
dev_err(dev, "Failed to register LED for node %pfw\n", node);
static void delete_ns2_led(struct ns2_led_data *led_dat) return ret;
{
led_classdev_unregister(&led_dat->cdev);
} }
#ifdef CONFIG_OF_GPIO static int ns2_led_probe(struct platform_device *pdev)
/*
* Translate OpenFirmware node properties into platform_data.
*/
static int
ns2_leds_get_of_pdata(struct device *dev, struct ns2_led_platform_data *pdata)
{ {
struct device_node *np = dev->of_node; struct device *dev = &pdev->dev;
struct device_node *child; struct fwnode_handle *child;
struct ns2_led *led, *leds; struct ns2_led *leds;
int ret, num_leds = 0; int count;
int ret;
num_leds = of_get_child_count(np); count = device_get_child_node_count(dev);
if (!num_leds) if (!count)
return -ENODEV; return -ENODEV;
leds = devm_kcalloc(dev, num_leds, sizeof(struct ns2_led), leds = devm_kzalloc(dev, array_size(sizeof(*leds), count), GFP_KERNEL);
GFP_KERNEL);
if (!leds) if (!leds)
return -ENOMEM; return -ENOMEM;
led = leds; device_for_each_child_node(dev, child) {
for_each_child_of_node(np, child) { ret = ns2_led_register(dev, child, leds++);
const char *string; if (ret) {
int i, num_modes; fwnode_handle_put(child);
struct ns2_led_modval *modval; return ret;
struct gpio_desc *gd;
ret = of_property_read_string(child, "label", &string);
led->name = (ret == 0) ? string : child->name;
gd = gpiod_get_from_of_node(child, "cmd-gpio", 0,
GPIOD_ASIS, led->name);
if (IS_ERR(gd)) {
ret = PTR_ERR(gd);
goto err_node_put;
}
led->cmd = gd;
gd = gpiod_get_from_of_node(child, "slow-gpio", 0,
GPIOD_ASIS, led->name);
if (IS_ERR(gd)) {
ret = PTR_ERR(gd);
goto err_node_put;
}
led->slow = gd;
ret = of_property_read_string(child, "linux,default-trigger",
&string);
if (ret == 0)
led->default_trigger = string;
ret = of_property_count_u32_elems(child, "modes-map");
if (ret < 0 || ret % 3) {
dev_err(dev,
"Missing or malformed modes-map property\n");
ret = -EINVAL;
goto err_node_put;
}
num_modes = ret / 3;
modval = devm_kcalloc(dev,
num_modes,
sizeof(struct ns2_led_modval),
GFP_KERNEL);
if (!modval) {
ret = -ENOMEM;
goto err_node_put;
}
for (i = 0; i < num_modes; i++) {
of_property_read_u32_index(child,
"modes-map", 3 * i,
(u32 *) &modval[i].mode);
of_property_read_u32_index(child,
"modes-map", 3 * i + 1,
(u32 *) &modval[i].cmd_level);
of_property_read_u32_index(child,
"modes-map", 3 * i + 2,
(u32 *) &modval[i].slow_level);
} }
led->num_modes = num_modes;
led->modval = modval;
led++;
} }
pdata->leds = leds;
pdata->num_leds = num_leds;
return 0; return 0;
err_node_put:
of_node_put(child);
return ret;
} }
static const struct of_device_id of_ns2_leds_match[] = { static const struct of_device_id of_ns2_leds_match[] = {
...@@ -341,76 +267,12 @@ static const struct of_device_id of_ns2_leds_match[] = { ...@@ -341,76 +267,12 @@ static const struct of_device_id of_ns2_leds_match[] = {
{}, {},
}; };
MODULE_DEVICE_TABLE(of, of_ns2_leds_match); MODULE_DEVICE_TABLE(of, of_ns2_leds_match);
#endif /* CONFIG_OF_GPIO */
struct ns2_led_priv {
int num_leds;
struct ns2_led_data leds_data[];
};
static int ns2_led_probe(struct platform_device *pdev)
{
struct ns2_led_platform_data *pdata = dev_get_platdata(&pdev->dev);
struct ns2_led_priv *priv;
int i;
int ret;
#ifdef CONFIG_OF_GPIO
if (!pdata) {
pdata = devm_kzalloc(&pdev->dev,
sizeof(struct ns2_led_platform_data),
GFP_KERNEL);
if (!pdata)
return -ENOMEM;
ret = ns2_leds_get_of_pdata(&pdev->dev, pdata);
if (ret)
return ret;
}
#else
if (!pdata)
return -EINVAL;
#endif /* CONFIG_OF_GPIO */
priv = devm_kzalloc(&pdev->dev, struct_size(priv, leds_data, pdata->num_leds), GFP_KERNEL);
if (!priv)
return -ENOMEM;
priv->num_leds = pdata->num_leds;
for (i = 0; i < priv->num_leds; i++) {
ret = create_ns2_led(pdev, &priv->leds_data[i],
&pdata->leds[i]);
if (ret < 0) {
for (i = i - 1; i >= 0; i--)
delete_ns2_led(&priv->leds_data[i]);
return ret;
}
}
platform_set_drvdata(pdev, priv);
return 0;
}
static int ns2_led_remove(struct platform_device *pdev)
{
int i;
struct ns2_led_priv *priv;
priv = platform_get_drvdata(pdev);
for (i = 0; i < priv->num_leds; i++)
delete_ns2_led(&priv->leds_data[i]);
return 0;
}
static struct platform_driver ns2_led_driver = { static struct platform_driver ns2_led_driver = {
.probe = ns2_led_probe, .probe = ns2_led_probe,
.remove = ns2_led_remove,
.driver = { .driver = {
.name = "leds-ns2", .name = "leds-ns2",
.of_match_table = of_match_ptr(of_ns2_leds_match), .of_match_table = of_ns2_leds_match,
}, },
}; };
......
...@@ -27,6 +27,8 @@ ...@@ -27,6 +27,8 @@
#define PCA9532_REG_PWM(m, i) (PCA9532_REG_OFFSET(m) + 0x2 + (i) * 2) #define PCA9532_REG_PWM(m, i) (PCA9532_REG_OFFSET(m) + 0x2 + (i) * 2)
#define LED_REG(m, led) (PCA9532_REG_OFFSET(m) + 0x5 + (led >> 2)) #define LED_REG(m, led) (PCA9532_REG_OFFSET(m) + 0x5 + (led >> 2))
#define LED_NUM(led) (led & 0x3) #define LED_NUM(led) (led & 0x3)
#define LED_SHIFT(led) (LED_NUM(led) * 2)
#define LED_MASK(led) (0x3 << LED_SHIFT(led))
#define ldev_to_led(c) container_of(c, struct pca9532_led, ldev) #define ldev_to_led(c) container_of(c, struct pca9532_led, ldev)
...@@ -162,9 +164,9 @@ static void pca9532_setled(struct pca9532_led *led) ...@@ -162,9 +164,9 @@ static void pca9532_setled(struct pca9532_led *led)
mutex_lock(&data->update_lock); mutex_lock(&data->update_lock);
reg = i2c_smbus_read_byte_data(client, LED_REG(maxleds, led->id)); reg = i2c_smbus_read_byte_data(client, LED_REG(maxleds, led->id));
/* zero led bits */ /* zero led bits */
reg = reg & ~(0x3<<LED_NUM(led->id)*2); reg = reg & ~LED_MASK(led->id);
/* set the new value */ /* set the new value */
reg = reg | (led->state << LED_NUM(led->id)*2); reg = reg | (led->state << LED_SHIFT(led->id));
i2c_smbus_write_byte_data(client, LED_REG(maxleds, led->id), reg); i2c_smbus_write_byte_data(client, LED_REG(maxleds, led->id), reg);
mutex_unlock(&data->update_lock); mutex_unlock(&data->update_lock);
} }
...@@ -260,7 +262,7 @@ static enum pca9532_state pca9532_getled(struct pca9532_led *led) ...@@ -260,7 +262,7 @@ static enum pca9532_state pca9532_getled(struct pca9532_led *led)
mutex_lock(&data->update_lock); mutex_lock(&data->update_lock);
reg = i2c_smbus_read_byte_data(client, LED_REG(maxleds, led->id)); reg = i2c_smbus_read_byte_data(client, LED_REG(maxleds, led->id));
ret = reg >> LED_NUM(led->id)/2; ret = (reg & LED_MASK(led->id)) >> LED_SHIFT(led->id);
mutex_unlock(&data->update_lock); mutex_unlock(&data->update_lock);
return ret; return ret;
} }
...@@ -478,7 +480,12 @@ pca9532_of_populate_pdata(struct device *dev, struct device_node *np) ...@@ -478,7 +480,12 @@ pca9532_of_populate_pdata(struct device *dev, struct device_node *np)
if (!pdata) if (!pdata)
return ERR_PTR(-ENOMEM); return ERR_PTR(-ENOMEM);
for_each_child_of_node(np, child) { of_property_read_u8_array(np, "nxp,pwm", &pdata->pwm[0],
ARRAY_SIZE(pdata->pwm));
of_property_read_u8_array(np, "nxp,psc", &pdata->psc[0],
ARRAY_SIZE(pdata->psc));
for_each_available_child_of_node(np, child) {
if (of_property_read_string(child, "label", if (of_property_read_string(child, "label",
&pdata->leds[i].name)) &pdata->leds[i].name))
pdata->leds[i].name = child->name; pdata->leds[i].name = child->name;
...@@ -507,7 +514,7 @@ static int pca9532_probe(struct i2c_client *client, ...@@ -507,7 +514,7 @@ static int pca9532_probe(struct i2c_client *client,
struct pca9532_data *data = i2c_get_clientdata(client); struct pca9532_data *data = i2c_get_clientdata(client);
struct pca9532_platform_data *pca9532_pdata = struct pca9532_platform_data *pca9532_pdata =
dev_get_platdata(&client->dev); dev_get_platdata(&client->dev);
struct device_node *np = client->dev.of_node; struct device_node *np = dev_of_node(&client->dev);
if (!pca9532_pdata) { if (!pca9532_pdata) {
if (np) { if (np) {
...@@ -545,13 +552,8 @@ static int pca9532_probe(struct i2c_client *client, ...@@ -545,13 +552,8 @@ static int pca9532_probe(struct i2c_client *client,
static int pca9532_remove(struct i2c_client *client) static int pca9532_remove(struct i2c_client *client)
{ {
struct pca9532_data *data = i2c_get_clientdata(client); struct pca9532_data *data = i2c_get_clientdata(client);
int err;
err = pca9532_destroy_devices(data, data->chip_info->num_leds); return pca9532_destroy_devices(data, data->chip_info->num_leds);
if (err)
return err;
return 0;
} }
module_i2c_driver(pca9532_driver); module_i2c_driver(pca9532_driver);
......
...@@ -65,6 +65,7 @@ enum pca955x_type { ...@@ -65,6 +65,7 @@ enum pca955x_type {
pca9550, pca9550,
pca9551, pca9551,
pca9552, pca9552,
ibm_pca9552,
pca9553, pca9553,
}; };
...@@ -90,6 +91,11 @@ static struct pca955x_chipdef pca955x_chipdefs[] = { ...@@ -90,6 +91,11 @@ static struct pca955x_chipdef pca955x_chipdefs[] = {
.slv_addr = /* 1100xxx */ 0x60, .slv_addr = /* 1100xxx */ 0x60,
.slv_addr_shift = 3, .slv_addr_shift = 3,
}, },
[ibm_pca9552] = {
.bits = 16,
.slv_addr = /* 0110xxx */ 0x30,
.slv_addr_shift = 3,
},
[pca9553] = { [pca9553] = {
.bits = 4, .bits = 4,
.slv_addr = /* 110001x */ 0x62, .slv_addr = /* 110001x */ 0x62,
...@@ -101,6 +107,7 @@ static const struct i2c_device_id pca955x_id[] = { ...@@ -101,6 +107,7 @@ static const struct i2c_device_id pca955x_id[] = {
{ "pca9550", pca9550 }, { "pca9550", pca9550 },
{ "pca9551", pca9551 }, { "pca9551", pca9551 },
{ "pca9552", pca9552 }, { "pca9552", pca9552 },
{ "ibm-pca9552", ibm_pca9552 },
{ "pca9553", pca9553 }, { "pca9553", pca9553 },
{ } { }
}; };
...@@ -412,6 +419,7 @@ static const struct of_device_id of_pca955x_match[] = { ...@@ -412,6 +419,7 @@ static const struct of_device_id of_pca955x_match[] = {
{ .compatible = "nxp,pca9550", .data = (void *)pca9550 }, { .compatible = "nxp,pca9550", .data = (void *)pca9550 },
{ .compatible = "nxp,pca9551", .data = (void *)pca9551 }, { .compatible = "nxp,pca9551", .data = (void *)pca9551 },
{ .compatible = "nxp,pca9552", .data = (void *)pca9552 }, { .compatible = "nxp,pca9552", .data = (void *)pca9552 },
{ .compatible = "ibm,pca9552", .data = (void *)ibm_pca9552 },
{ .compatible = "nxp,pca9553", .data = (void *)pca9553 }, { .compatible = "nxp,pca9553", .data = (void *)pca9553 },
{}, {},
}; };
......
...@@ -32,7 +32,6 @@ ...@@ -32,7 +32,6 @@
#include <linux/property.h> #include <linux/property.h>
#include <linux/slab.h> #include <linux/slab.h>
#include <linux/of.h> #include <linux/of.h>
#include <linux/platform_data/leds-pca963x.h>
/* LED select registers determine the source that drives LED outputs */ /* LED select registers determine the source that drives LED outputs */
#define PCA963X_LED_OFF 0x0 /* LED driver off */ #define PCA963X_LED_OFF 0x0 /* LED driver off */
...@@ -96,102 +95,108 @@ static const struct i2c_device_id pca963x_id[] = { ...@@ -96,102 +95,108 @@ static const struct i2c_device_id pca963x_id[] = {
}; };
MODULE_DEVICE_TABLE(i2c, pca963x_id); MODULE_DEVICE_TABLE(i2c, pca963x_id);
struct pca963x_led; struct pca963x;
struct pca963x {
struct pca963x_chipdef *chipdef;
struct mutex mutex;
struct i2c_client *client;
struct pca963x_led *leds;
unsigned long leds_on;
};
struct pca963x_led { struct pca963x_led {
struct pca963x *chip; struct pca963x *chip;
struct led_classdev led_cdev; struct led_classdev led_cdev;
int led_num; /* 0 .. 15 potentially */ int led_num; /* 0 .. 15 potentially */
char name[32];
u8 gdc; u8 gdc;
u8 gfrq; u8 gfrq;
}; };
static int pca963x_brightness(struct pca963x_led *pca963x, struct pca963x {
struct pca963x_chipdef *chipdef;
struct mutex mutex;
struct i2c_client *client;
unsigned long leds_on;
struct pca963x_led leds[];
};
static int pca963x_brightness(struct pca963x_led *led,
enum led_brightness brightness) enum led_brightness brightness)
{ {
u8 ledout_addr = pca963x->chip->chipdef->ledout_base struct i2c_client *client = led->chip->client;
+ (pca963x->led_num / 4); struct pca963x_chipdef *chipdef = led->chip->chipdef;
u8 ledout; u8 ledout_addr, ledout, mask, val;
int shift = 2 * (pca963x->led_num % 4); int shift;
u8 mask = 0x3 << shift;
int ret; int ret;
ledout = i2c_smbus_read_byte_data(pca963x->chip->client, ledout_addr); ledout_addr = chipdef->ledout_base + (led->led_num / 4);
shift = 2 * (led->led_num % 4);
mask = 0x3 << shift;
ledout = i2c_smbus_read_byte_data(client, ledout_addr);
switch (brightness) { switch (brightness) {
case LED_FULL: case LED_FULL:
ret = i2c_smbus_write_byte_data(pca963x->chip->client, val = (ledout & ~mask) | (PCA963X_LED_ON << shift);
ledout_addr, ret = i2c_smbus_write_byte_data(client, ledout_addr, val);
(ledout & ~mask) | (PCA963X_LED_ON << shift));
break; break;
case LED_OFF: case LED_OFF:
ret = i2c_smbus_write_byte_data(pca963x->chip->client, val = ledout & ~mask;
ledout_addr, ledout & ~mask); ret = i2c_smbus_write_byte_data(client, ledout_addr, val);
break; break;
default: default:
ret = i2c_smbus_write_byte_data(pca963x->chip->client, ret = i2c_smbus_write_byte_data(client,
PCA963X_PWM_BASE + pca963x->led_num, PCA963X_PWM_BASE +
led->led_num,
brightness); brightness);
if (ret < 0) if (ret < 0)
return ret; return ret;
ret = i2c_smbus_write_byte_data(pca963x->chip->client,
ledout_addr, val = (ledout & ~mask) | (PCA963X_LED_PWM << shift);
(ledout & ~mask) | (PCA963X_LED_PWM << shift)); ret = i2c_smbus_write_byte_data(client, ledout_addr, val);
break; break;
} }
return ret; return ret;
} }
static void pca963x_blink(struct pca963x_led *pca963x) static void pca963x_blink(struct pca963x_led *led)
{ {
u8 ledout_addr = pca963x->chip->chipdef->ledout_base + struct i2c_client *client = led->chip->client;
(pca963x->led_num / 4); struct pca963x_chipdef *chipdef = led->chip->chipdef;
u8 ledout; u8 ledout_addr, ledout, mask, val, mode2;
u8 mode2 = i2c_smbus_read_byte_data(pca963x->chip->client, int shift;
PCA963X_MODE2);
int shift = 2 * (pca963x->led_num % 4);
u8 mask = 0x3 << shift;
i2c_smbus_write_byte_data(pca963x->chip->client, ledout_addr = chipdef->ledout_base + (led->led_num / 4);
pca963x->chip->chipdef->grppwm, pca963x->gdc); shift = 2 * (led->led_num % 4);
mask = 0x3 << shift;
mode2 = i2c_smbus_read_byte_data(client, PCA963X_MODE2);
i2c_smbus_write_byte_data(pca963x->chip->client, i2c_smbus_write_byte_data(client, chipdef->grppwm, led->gdc);
pca963x->chip->chipdef->grpfreq, pca963x->gfrq);
i2c_smbus_write_byte_data(client, chipdef->grpfreq, led->gfrq);
if (!(mode2 & PCA963X_MODE2_DMBLNK)) if (!(mode2 & PCA963X_MODE2_DMBLNK))
i2c_smbus_write_byte_data(pca963x->chip->client, PCA963X_MODE2, i2c_smbus_write_byte_data(client, PCA963X_MODE2,
mode2 | PCA963X_MODE2_DMBLNK); mode2 | PCA963X_MODE2_DMBLNK);
mutex_lock(&pca963x->chip->mutex); mutex_lock(&led->chip->mutex);
ledout = i2c_smbus_read_byte_data(pca963x->chip->client, ledout_addr);
if ((ledout & mask) != (PCA963X_LED_GRP_PWM << shift)) ledout = i2c_smbus_read_byte_data(client, ledout_addr);
i2c_smbus_write_byte_data(pca963x->chip->client, ledout_addr, if ((ledout & mask) != (PCA963X_LED_GRP_PWM << shift)) {
(ledout & ~mask) | (PCA963X_LED_GRP_PWM << shift)); val = (ledout & ~mask) | (PCA963X_LED_GRP_PWM << shift);
mutex_unlock(&pca963x->chip->mutex); i2c_smbus_write_byte_data(client, ledout_addr, val);
}
mutex_unlock(&led->chip->mutex);
} }
static int pca963x_power_state(struct pca963x_led *pca963x) static int pca963x_power_state(struct pca963x_led *led)
{ {
unsigned long *leds_on = &pca963x->chip->leds_on; struct i2c_client *client = led->chip->client;
unsigned long cached_leds = pca963x->chip->leds_on; unsigned long *leds_on = &led->chip->leds_on;
unsigned long cached_leds = *leds_on;
if (pca963x->led_cdev.brightness) if (led->led_cdev.brightness)
set_bit(pca963x->led_num, leds_on); set_bit(led->led_num, leds_on);
else else
clear_bit(pca963x->led_num, leds_on); clear_bit(led->led_num, leds_on);
if (!(*leds_on) != !cached_leds) if (!(*leds_on) != !cached_leds)
return i2c_smbus_write_byte_data(pca963x->chip->client, return i2c_smbus_write_byte_data(client, PCA963X_MODE1,
PCA963X_MODE1, *leds_on ? 0 : BIT(4)); *leds_on ? 0 : BIT(4));
return 0; return 0;
} }
...@@ -199,27 +204,27 @@ static int pca963x_power_state(struct pca963x_led *pca963x) ...@@ -199,27 +204,27 @@ static int pca963x_power_state(struct pca963x_led *pca963x)
static int pca963x_led_set(struct led_classdev *led_cdev, static int pca963x_led_set(struct led_classdev *led_cdev,
enum led_brightness value) enum led_brightness value)
{ {
struct pca963x_led *pca963x; struct pca963x_led *led;
int ret; int ret;
pca963x = container_of(led_cdev, struct pca963x_led, led_cdev); led = container_of(led_cdev, struct pca963x_led, led_cdev);
mutex_lock(&pca963x->chip->mutex); mutex_lock(&led->chip->mutex);
ret = pca963x_brightness(pca963x, value); ret = pca963x_brightness(led, value);
if (ret < 0) if (ret < 0)
goto unlock; goto unlock;
ret = pca963x_power_state(pca963x); ret = pca963x_power_state(led);
unlock: unlock:
mutex_unlock(&pca963x->chip->mutex); mutex_unlock(&led->chip->mutex);
return ret; return ret;
} }
static unsigned int pca963x_period_scale(struct pca963x_led *pca963x, static unsigned int pca963x_period_scale(struct pca963x_led *led,
unsigned int val) unsigned int val)
{ {
unsigned int scaling = pca963x->chip->chipdef->scaling; unsigned int scaling = led->chip->chipdef->scaling;
return scaling ? DIV_ROUND_CLOSEST(val * scaling, 1000) : val; return scaling ? DIV_ROUND_CLOSEST(val * scaling, 1000) : val;
} }
...@@ -227,11 +232,11 @@ static unsigned int pca963x_period_scale(struct pca963x_led *pca963x, ...@@ -227,11 +232,11 @@ static unsigned int pca963x_period_scale(struct pca963x_led *pca963x,
static int pca963x_blink_set(struct led_classdev *led_cdev, static int pca963x_blink_set(struct led_classdev *led_cdev,
unsigned long *delay_on, unsigned long *delay_off) unsigned long *delay_on, unsigned long *delay_off)
{ {
struct pca963x_led *pca963x;
unsigned long time_on, time_off, period; unsigned long time_on, time_off, period;
struct pca963x_led *led;
u8 gdc, gfrq; u8 gdc, gfrq;
pca963x = container_of(led_cdev, struct pca963x_led, led_cdev); led = container_of(led_cdev, struct pca963x_led, led_cdev);
time_on = *delay_on; time_on = *delay_on;
time_off = *delay_off; time_off = *delay_off;
...@@ -242,14 +247,14 @@ static int pca963x_blink_set(struct led_classdev *led_cdev, ...@@ -242,14 +247,14 @@ static int pca963x_blink_set(struct led_classdev *led_cdev,
time_off = 500; time_off = 500;
} }
period = pca963x_period_scale(pca963x, time_on + time_off); period = pca963x_period_scale(led, time_on + time_off);
/* If period not supported by hardware, default to someting sane. */ /* If period not supported by hardware, default to someting sane. */
if ((period < PCA963X_BLINK_PERIOD_MIN) || if ((period < PCA963X_BLINK_PERIOD_MIN) ||
(period > PCA963X_BLINK_PERIOD_MAX)) { (period > PCA963X_BLINK_PERIOD_MAX)) {
time_on = 500; time_on = 500;
time_off = 500; time_off = 500;
period = pca963x_period_scale(pca963x, 1000); period = pca963x_period_scale(led, 1000);
} }
/* /*
...@@ -257,7 +262,7 @@ static int pca963x_blink_set(struct led_classdev *led_cdev, ...@@ -257,7 +262,7 @@ static int pca963x_blink_set(struct led_classdev *led_cdev,
* (time_on / period) = (GDC / 256) -> * (time_on / period) = (GDC / 256) ->
* GDC = ((time_on * 256) / period) * GDC = ((time_on * 256) / period)
*/ */
gdc = (pca963x_period_scale(pca963x, time_on) * 256) / period; gdc = (pca963x_period_scale(led, time_on) * 256) / period;
/* /*
* From manual: period = ((GFRQ + 1) / 24) in seconds. * From manual: period = ((GFRQ + 1) / 24) in seconds.
...@@ -266,10 +271,10 @@ static int pca963x_blink_set(struct led_classdev *led_cdev, ...@@ -266,10 +271,10 @@ static int pca963x_blink_set(struct led_classdev *led_cdev,
*/ */
gfrq = (period * 24 / 1000) - 1; gfrq = (period * 24 / 1000) - 1;
pca963x->gdc = gdc; led->gdc = gdc;
pca963x->gfrq = gfrq; led->gfrq = gfrq;
pca963x_blink(pca963x); pca963x_blink(led);
*delay_on = time_on; *delay_on = time_on;
*delay_off = time_off; *delay_off = time_off;
...@@ -277,72 +282,84 @@ static int pca963x_blink_set(struct led_classdev *led_cdev, ...@@ -277,72 +282,84 @@ static int pca963x_blink_set(struct led_classdev *led_cdev,
return 0; return 0;
} }
static struct pca963x_platform_data * static int pca963x_register_leds(struct i2c_client *client,
pca963x_get_pdata(struct i2c_client *client, struct pca963x_chipdef *chip) struct pca963x *chip)
{ {
struct pca963x_platform_data *pdata; struct pca963x_chipdef *chipdef = chip->chipdef;
struct led_info *pca963x_leds; struct pca963x_led *led = chip->leds;
struct device *dev = &client->dev;
struct fwnode_handle *child; struct fwnode_handle *child;
int count; bool hw_blink;
s32 mode2;
count = device_get_child_node_count(&client->dev);
if (!count || count > chip->n_leds)
return ERR_PTR(-ENODEV);
pca963x_leds = devm_kcalloc(&client->dev,
chip->n_leds, sizeof(struct led_info), GFP_KERNEL);
if (!pca963x_leds)
return ERR_PTR(-ENOMEM);
device_for_each_child_node(&client->dev, child) {
struct led_info led = {};
u32 reg; u32 reg;
int res; int ret;
res = fwnode_property_read_u32(child, "reg", &reg);
if ((res != 0) || (reg >= chip->n_leds))
continue;
res = fwnode_property_read_string(child, "label", &led.name);
if ((res != 0) && is_of_node(child))
led.name = to_of_node(child)->name;
fwnode_property_read_string(child, "linux,default-trigger", if (device_property_read_u32(dev, "nxp,period-scale",
&led.default_trigger); &chipdef->scaling))
chipdef->scaling = 1000;
pca963x_leds[reg] = led; hw_blink = device_property_read_bool(dev, "nxp,hw-blink");
}
pdata = devm_kzalloc(&client->dev,
sizeof(struct pca963x_platform_data), GFP_KERNEL);
if (!pdata)
return ERR_PTR(-ENOMEM);
pdata->leds.leds = pca963x_leds; mode2 = i2c_smbus_read_byte_data(client, PCA963X_MODE2);
pdata->leds.num_leds = chip->n_leds; if (mode2 < 0)
return mode2;
/* default to open-drain unless totem pole (push-pull) is specified */ /* default to open-drain unless totem pole (push-pull) is specified */
if (device_property_read_bool(&client->dev, "nxp,totem-pole")) if (device_property_read_bool(dev, "nxp,totem-pole"))
pdata->outdrv = PCA963X_TOTEM_POLE; mode2 |= PCA963X_MODE2_OUTDRV;
else else
pdata->outdrv = PCA963X_OPEN_DRAIN; mode2 &= ~PCA963X_MODE2_OUTDRV;
/* default to software blinking unless hardware blinking is specified */ /* default to non-inverted output, unless inverted is specified */
if (device_property_read_bool(&client->dev, "nxp,hw-blink")) if (device_property_read_bool(dev, "nxp,inverted-out"))
pdata->blink_type = PCA963X_HW_BLINK; mode2 |= PCA963X_MODE2_INVRT;
else else
pdata->blink_type = PCA963X_SW_BLINK; mode2 &= ~PCA963X_MODE2_INVRT;
if (device_property_read_u32(&client->dev, "nxp,period-scale", ret = i2c_smbus_write_byte_data(client, PCA963X_MODE2, mode2);
&chip->scaling)) if (ret < 0)
chip->scaling = 1000; return ret;
/* default to non-inverted output, unless inverted is specified */ device_for_each_child_node(dev, child) {
if (device_property_read_bool(&client->dev, "nxp,inverted-out")) struct led_init_data init_data = {};
pdata->dir = PCA963X_INVERTED; char default_label[32];
else
pdata->dir = PCA963X_NORMAL;
return pdata; ret = fwnode_property_read_u32(child, "reg", &reg);
if (ret || reg >= chipdef->n_leds) {
dev_err(dev, "Invalid 'reg' property for node %pfw\n",
child);
ret = -EINVAL;
goto err;
}
led->led_num = reg;
led->chip = chip;
led->led_cdev.brightness_set_blocking = pca963x_led_set;
if (hw_blink)
led->led_cdev.blink_set = pca963x_blink_set;
init_data.fwnode = child;
/* for backwards compatibility */
init_data.devicename = "pca963x";
snprintf(default_label, sizeof(default_label), "%d:%.2x:%u",
client->adapter->nr, client->addr, reg);
init_data.default_label = default_label;
ret = devm_led_classdev_register_ext(dev, &led->led_cdev,
&init_data);
if (ret) {
dev_err(dev, "Failed to register LED for node %pfw\n",
child);
goto err;
}
++led;
}
return 0;
err:
fwnode_handle_put(child);
return ret;
} }
static const struct of_device_id of_pca963x_match[] = { static const struct of_device_id of_pca963x_match[] = {
...@@ -357,117 +374,38 @@ MODULE_DEVICE_TABLE(of, of_pca963x_match); ...@@ -357,117 +374,38 @@ MODULE_DEVICE_TABLE(of, of_pca963x_match);
static int pca963x_probe(struct i2c_client *client, static int pca963x_probe(struct i2c_client *client,
const struct i2c_device_id *id) const struct i2c_device_id *id)
{ {
struct pca963x *pca963x_chip; struct device *dev = &client->dev;
struct pca963x_led *pca963x; struct pca963x_chipdef *chipdef;
struct pca963x_platform_data *pdata; struct pca963x *chip;
struct pca963x_chipdef *chip; int i, count;
int i, err;
chipdef = &pca963x_chipdefs[id->driver_data];
chip = &pca963x_chipdefs[id->driver_data];
pdata = dev_get_platdata(&client->dev);
if (!pdata) {
pdata = pca963x_get_pdata(client, chip);
if (IS_ERR(pdata)) {
dev_warn(&client->dev, "could not parse configuration\n");
pdata = NULL;
}
}
if (pdata && (pdata->leds.num_leds < 1 || count = device_get_child_node_count(dev);
pdata->leds.num_leds > chip->n_leds)) { if (!count || count > chipdef->n_leds) {
dev_err(&client->dev, "board info must claim 1-%d LEDs", dev_err(dev, "Node %pfw must define between 1 and %d LEDs\n",
chip->n_leds); dev_fwnode(dev), chipdef->n_leds);
return -EINVAL; return -EINVAL;
} }
pca963x_chip = devm_kzalloc(&client->dev, sizeof(*pca963x_chip), chip = devm_kzalloc(dev, struct_size(chip, leds, count), GFP_KERNEL);
GFP_KERNEL); if (!chip)
if (!pca963x_chip)
return -ENOMEM;
pca963x = devm_kcalloc(&client->dev, chip->n_leds, sizeof(*pca963x),
GFP_KERNEL);
if (!pca963x)
return -ENOMEM; return -ENOMEM;
i2c_set_clientdata(client, pca963x_chip); i2c_set_clientdata(client, chip);
mutex_init(&pca963x_chip->mutex); mutex_init(&chip->mutex);
pca963x_chip->chipdef = chip; chip->chipdef = chipdef;
pca963x_chip->client = client; chip->client = client;
pca963x_chip->leds = pca963x;
/* Turn off LEDs by default*/ /* Turn off LEDs by default*/
for (i = 0; i < chip->n_leds / 4; i++) for (i = 0; i < chipdef->n_leds / 4; i++)
i2c_smbus_write_byte_data(client, chip->ledout_base + i, 0x00); i2c_smbus_write_byte_data(client, chipdef->ledout_base + i, 0x00);
for (i = 0; i < chip->n_leds; i++) {
pca963x[i].led_num = i;
pca963x[i].chip = pca963x_chip;
/* Platform data can specify LED names and default triggers */
if (pdata && i < pdata->leds.num_leds) {
if (pdata->leds.leds[i].name)
snprintf(pca963x[i].name,
sizeof(pca963x[i].name), "pca963x:%s",
pdata->leds.leds[i].name);
if (pdata->leds.leds[i].default_trigger)
pca963x[i].led_cdev.default_trigger =
pdata->leds.leds[i].default_trigger;
}
if (!pdata || i >= pdata->leds.num_leds ||
!pdata->leds.leds[i].name)
snprintf(pca963x[i].name, sizeof(pca963x[i].name),
"pca963x:%d:%.2x:%d", client->adapter->nr,
client->addr, i);
pca963x[i].led_cdev.name = pca963x[i].name;
pca963x[i].led_cdev.brightness_set_blocking = pca963x_led_set;
if (pdata && pdata->blink_type == PCA963X_HW_BLINK)
pca963x[i].led_cdev.blink_set = pca963x_blink_set;
err = led_classdev_register(&client->dev, &pca963x[i].led_cdev);
if (err < 0)
goto exit;
}
/* Disable LED all-call address, and power down initially */ /* Disable LED all-call address, and power down initially */
i2c_smbus_write_byte_data(client, PCA963X_MODE1, BIT(4)); i2c_smbus_write_byte_data(client, PCA963X_MODE1, BIT(4));
if (pdata) { return pca963x_register_leds(client, chip);
u8 mode2 = i2c_smbus_read_byte_data(pca963x->chip->client,
PCA963X_MODE2);
/* Configure output: open-drain or totem pole (push-pull) */
if (pdata->outdrv == PCA963X_OPEN_DRAIN)
mode2 &= ~PCA963X_MODE2_OUTDRV;
else
mode2 |= PCA963X_MODE2_OUTDRV;
/* Configure direction: normal or inverted */
if (pdata->dir == PCA963X_INVERTED)
mode2 |= PCA963X_MODE2_INVRT;
i2c_smbus_write_byte_data(pca963x->chip->client, PCA963X_MODE2,
mode2);
}
return 0;
exit:
while (i--)
led_classdev_unregister(&pca963x[i].led_cdev);
return err;
}
static int pca963x_remove(struct i2c_client *client)
{
struct pca963x *pca963x = i2c_get_clientdata(client);
int i;
for (i = 0; i < pca963x->chipdef->n_leds; i++)
led_classdev_unregister(&pca963x->leds[i].led_cdev);
return 0;
} }
static struct i2c_driver pca963x_driver = { static struct i2c_driver pca963x_driver = {
...@@ -476,7 +414,6 @@ static struct i2c_driver pca963x_driver = { ...@@ -476,7 +414,6 @@ static struct i2c_driver pca963x_driver = {
.of_match_table = of_pca963x_match, .of_match_table = of_pca963x_match,
}, },
.probe = pca963x_probe, .probe = pca963x_probe,
.remove = pca963x_remove,
.id_table = pca963x_id, .id_table = pca963x_id,
}; };
......
...@@ -87,36 +87,36 @@ static enum led_brightness pm8058_led_get(struct led_classdev *cled) ...@@ -87,36 +87,36 @@ static enum led_brightness pm8058_led_get(struct led_classdev *cled)
static int pm8058_led_probe(struct platform_device *pdev) static int pm8058_led_probe(struct platform_device *pdev)
{ {
struct led_init_data init_data = {};
struct device *dev = &pdev->dev;
struct pm8058_led *led; struct pm8058_led *led;
struct device_node *np = pdev->dev.of_node; struct device_node *np;
int ret; int ret;
struct regmap *map; struct regmap *map;
const char *state; const char *state;
enum led_brightness maxbright; enum led_brightness maxbright;
led = devm_kzalloc(&pdev->dev, sizeof(*led), GFP_KERNEL); led = devm_kzalloc(dev, sizeof(*led), GFP_KERNEL);
if (!led) if (!led)
return -ENOMEM; return -ENOMEM;
led->ledtype = (u32)(unsigned long)of_device_get_match_data(&pdev->dev); led->ledtype = (u32)(unsigned long)of_device_get_match_data(dev);
map = dev_get_regmap(pdev->dev.parent, NULL); map = dev_get_regmap(dev->parent, NULL);
if (!map) { if (!map) {
dev_err(&pdev->dev, "Parent regmap unavailable.\n"); dev_err(dev, "Parent regmap unavailable.\n");
return -ENXIO; return -ENXIO;
} }
led->map = map; led->map = map;
np = dev_of_node(dev);
ret = of_property_read_u32(np, "reg", &led->reg); ret = of_property_read_u32(np, "reg", &led->reg);
if (ret) { if (ret) {
dev_err(&pdev->dev, "no register offset specified\n"); dev_err(dev, "no register offset specified\n");
return -EINVAL; return -EINVAL;
} }
/* Use label else node name */
led->cdev.name = of_get_property(np, "label", NULL) ? : np->name;
led->cdev.default_trigger =
of_get_property(np, "linux,default-trigger", NULL);
led->cdev.brightness_set = pm8058_led_set; led->cdev.brightness_set = pm8058_led_set;
led->cdev.brightness_get = pm8058_led_get; led->cdev.brightness_get = pm8058_led_get;
if (led->ledtype == PM8058_LED_TYPE_COMMON) if (led->ledtype == PM8058_LED_TYPE_COMMON)
...@@ -142,14 +142,13 @@ static int pm8058_led_probe(struct platform_device *pdev) ...@@ -142,14 +142,13 @@ static int pm8058_led_probe(struct platform_device *pdev)
led->ledtype == PM8058_LED_TYPE_FLASH) led->ledtype == PM8058_LED_TYPE_FLASH)
led->cdev.flags = LED_CORE_SUSPENDRESUME; led->cdev.flags = LED_CORE_SUSPENDRESUME;
ret = devm_led_classdev_register(&pdev->dev, &led->cdev); init_data.fwnode = of_fwnode_handle(np);
if (ret) {
dev_err(&pdev->dev, "unable to register led \"%s\"\n",
led->cdev.name);
return ret;
}
return 0; ret = devm_led_classdev_register_ext(dev, &led->cdev, &init_data);
if (ret)
dev_err(dev, "Failed to register LED for %pOF\n", np);
return ret;
} }
static const struct of_device_id pm8058_leds_id_table[] = { static const struct of_device_id pm8058_leds_id_table[] = {
......
...@@ -250,7 +250,7 @@ static int powernv_led_classdev(struct platform_device *pdev, ...@@ -250,7 +250,7 @@ static int powernv_led_classdev(struct platform_device *pdev,
struct powernv_led_data *powernv_led; struct powernv_led_data *powernv_led;
struct device *dev = &pdev->dev; struct device *dev = &pdev->dev;
for_each_child_of_node(led_node, np) { for_each_available_child_of_node(led_node, np) {
p = of_find_property(np, "led-types", NULL); p = of_find_property(np, "led-types", NULL);
while ((cur = of_prop_next_string(p, cur)) != NULL) { while ((cur = of_prop_next_string(p, cur)) != NULL) {
......
...@@ -20,16 +20,10 @@ ...@@ -20,16 +20,10 @@
struct led_pwm { struct led_pwm {
const char *name; const char *name;
const char *default_trigger;
u8 active_low; u8 active_low;
unsigned int max_brightness; unsigned int max_brightness;
}; };
struct led_pwm_platform_data {
int num_leds;
struct led_pwm *leds;
};
struct led_pwm_data { struct led_pwm_data {
struct led_classdev cdev; struct led_classdev cdev;
struct pwm_device *pwm; struct pwm_device *pwm;
...@@ -61,36 +55,31 @@ static int led_pwm_set(struct led_classdev *led_cdev, ...@@ -61,36 +55,31 @@ static int led_pwm_set(struct led_classdev *led_cdev,
return pwm_apply_state(led_dat->pwm, &led_dat->pwmstate); return pwm_apply_state(led_dat->pwm, &led_dat->pwmstate);
} }
__attribute__((nonnull))
static int led_pwm_add(struct device *dev, struct led_pwm_priv *priv, static int led_pwm_add(struct device *dev, struct led_pwm_priv *priv,
struct led_pwm *led, struct fwnode_handle *fwnode) struct led_pwm *led, struct fwnode_handle *fwnode)
{ {
struct led_pwm_data *led_data = &priv->leds[priv->num_leds]; struct led_pwm_data *led_data = &priv->leds[priv->num_leds];
struct led_init_data init_data = { .fwnode = fwnode };
int ret; int ret;
led_data->active_low = led->active_low; led_data->active_low = led->active_low;
led_data->cdev.name = led->name; led_data->cdev.name = led->name;
led_data->cdev.default_trigger = led->default_trigger;
led_data->cdev.brightness = LED_OFF; led_data->cdev.brightness = LED_OFF;
led_data->cdev.max_brightness = led->max_brightness; led_data->cdev.max_brightness = led->max_brightness;
led_data->cdev.flags = LED_CORE_SUSPENDRESUME; led_data->cdev.flags = LED_CORE_SUSPENDRESUME;
if (fwnode)
led_data->pwm = devm_fwnode_pwm_get(dev, fwnode, NULL); led_data->pwm = devm_fwnode_pwm_get(dev, fwnode, NULL);
else if (IS_ERR(led_data->pwm))
led_data->pwm = devm_pwm_get(dev, led->name); return dev_err_probe(dev, PTR_ERR(led_data->pwm),
if (IS_ERR(led_data->pwm)) { "unable to request PWM for %s\n",
ret = PTR_ERR(led_data->pwm); led->name);
if (ret != -EPROBE_DEFER)
dev_err(dev, "unable to request PWM for %s: %d\n",
led->name, ret);
return ret;
}
led_data->cdev.brightness_set_blocking = led_pwm_set; led_data->cdev.brightness_set_blocking = led_pwm_set;
pwm_init_state(led_data->pwm, &led_data->pwmstate); pwm_init_state(led_data->pwm, &led_data->pwmstate);
ret = devm_led_classdev_register(dev, &led_data->cdev); ret = devm_led_classdev_register_ext(dev, &led_data->cdev, &init_data);
if (ret) { if (ret) {
dev_err(dev, "failed to register PWM led for %s: %d\n", dev_err(dev, "failed to register PWM led for %s: %d\n",
led->name, ret); led->name, ret);
...@@ -126,9 +115,6 @@ static int led_pwm_create_fwnode(struct device *dev, struct led_pwm_priv *priv) ...@@ -126,9 +115,6 @@ static int led_pwm_create_fwnode(struct device *dev, struct led_pwm_priv *priv)
return -EINVAL; return -EINVAL;
} }
fwnode_property_read_string(fwnode, "linux,default-trigger",
&led.default_trigger);
led.active_low = fwnode_property_read_bool(fwnode, led.active_low = fwnode_property_read_bool(fwnode,
"active-low"); "active-low");
fwnode_property_read_u32(fwnode, "max-brightness", fwnode_property_read_u32(fwnode, "max-brightness",
...@@ -146,14 +132,10 @@ static int led_pwm_create_fwnode(struct device *dev, struct led_pwm_priv *priv) ...@@ -146,14 +132,10 @@ static int led_pwm_create_fwnode(struct device *dev, struct led_pwm_priv *priv)
static int led_pwm_probe(struct platform_device *pdev) static int led_pwm_probe(struct platform_device *pdev)
{ {
struct led_pwm_platform_data *pdata = dev_get_platdata(&pdev->dev);
struct led_pwm_priv *priv; struct led_pwm_priv *priv;
int count, i;
int ret = 0; int ret = 0;
int count;
if (pdata)
count = pdata->num_leds;
else
count = device_get_child_node_count(&pdev->dev); count = device_get_child_node_count(&pdev->dev);
if (!count) if (!count)
...@@ -164,16 +146,7 @@ static int led_pwm_probe(struct platform_device *pdev) ...@@ -164,16 +146,7 @@ static int led_pwm_probe(struct platform_device *pdev)
if (!priv) if (!priv)
return -ENOMEM; return -ENOMEM;
if (pdata) {
for (i = 0; i < count; i++) {
ret = led_pwm_add(&pdev->dev, priv, &pdata->leds[i],
NULL);
if (ret)
break;
}
} else {
ret = led_pwm_create_fwnode(&pdev->dev, priv); ret = led_pwm_create_fwnode(&pdev->dev, priv);
}
if (ret) if (ret)
return ret; return ret;
......
...@@ -16,8 +16,6 @@ ...@@ -16,8 +16,6 @@
#include <linux/module.h> #include <linux/module.h>
#include <linux/platform_data/leds-s3c24xx.h> #include <linux/platform_data/leds-s3c24xx.h>
#include <mach/regs-gpio.h>
/* our context */ /* our context */
struct s3c24xx_gpio_led { struct s3c24xx_gpio_led {
......
...@@ -276,12 +276,12 @@ static int sc27xx_led_register(struct device *dev, struct sc27xx_led_priv *priv) ...@@ -276,12 +276,12 @@ static int sc27xx_led_register(struct device *dev, struct sc27xx_led_priv *priv)
static int sc27xx_led_probe(struct platform_device *pdev) static int sc27xx_led_probe(struct platform_device *pdev)
{ {
struct device *dev = &pdev->dev; struct device *dev = &pdev->dev;
struct device_node *np = dev->of_node, *child; struct device_node *np = dev_of_node(dev), *child;
struct sc27xx_led_priv *priv; struct sc27xx_led_priv *priv;
u32 base, count, reg; u32 base, count, reg;
int err; int err;
count = of_get_child_count(np); count = of_get_available_child_count(np);
if (!count || count > SC27XX_LEDS_MAX) if (!count || count > SC27XX_LEDS_MAX)
return -EINVAL; return -EINVAL;
...@@ -305,7 +305,7 @@ static int sc27xx_led_probe(struct platform_device *pdev) ...@@ -305,7 +305,7 @@ static int sc27xx_led_probe(struct platform_device *pdev)
return err; return err;
} }
for_each_child_of_node(np, child) { for_each_available_child_of_node(np, child) {
err = of_property_read_u32(child, "reg", &reg); err = of_property_read_u32(child, "reg", &reg);
if (err) { if (err) {
of_node_put(child); of_node_put(child);
......
...@@ -195,30 +195,21 @@ static int sgm3140_probe(struct platform_device *pdev) ...@@ -195,30 +195,21 @@ static int sgm3140_probe(struct platform_device *pdev)
priv->flash_gpio = devm_gpiod_get(&pdev->dev, "flash", GPIOD_OUT_LOW); priv->flash_gpio = devm_gpiod_get(&pdev->dev, "flash", GPIOD_OUT_LOW);
ret = PTR_ERR_OR_ZERO(priv->flash_gpio); ret = PTR_ERR_OR_ZERO(priv->flash_gpio);
if (ret) { if (ret)
if (ret != -EPROBE_DEFER) return dev_err_probe(&pdev->dev, ret,
dev_err(&pdev->dev, "Failed to request flash gpio\n");
"Failed to request flash gpio: %d\n", ret);
return ret;
}
priv->enable_gpio = devm_gpiod_get(&pdev->dev, "enable", GPIOD_OUT_LOW); priv->enable_gpio = devm_gpiod_get(&pdev->dev, "enable", GPIOD_OUT_LOW);
ret = PTR_ERR_OR_ZERO(priv->enable_gpio); ret = PTR_ERR_OR_ZERO(priv->enable_gpio);
if (ret) { if (ret)
if (ret != -EPROBE_DEFER) return dev_err_probe(&pdev->dev, ret,
dev_err(&pdev->dev, "Failed to request enable gpio\n");
"Failed to request enable gpio: %d\n", ret);
return ret;
}
priv->vin_regulator = devm_regulator_get(&pdev->dev, "vin"); priv->vin_regulator = devm_regulator_get(&pdev->dev, "vin");
ret = PTR_ERR_OR_ZERO(priv->vin_regulator); ret = PTR_ERR_OR_ZERO(priv->vin_regulator);
if (ret) { if (ret)
if (ret != -EPROBE_DEFER) return dev_err_probe(&pdev->dev, ret,
dev_err(&pdev->dev, "Failed to request regulator\n");
"Failed to request regulator: %d\n", ret);
return ret;
}
child_node = fwnode_get_next_available_child_node(pdev->dev.fwnode, child_node = fwnode_get_next_available_child_node(pdev->dev.fwnode,
NULL); NULL);
...@@ -316,5 +307,5 @@ static struct platform_driver sgm3140_driver = { ...@@ -316,5 +307,5 @@ static struct platform_driver sgm3140_driver = {
module_platform_driver(sgm3140_driver); module_platform_driver(sgm3140_driver);
MODULE_AUTHOR("Luca Weiss <luca@z3ntu.xyz>"); MODULE_AUTHOR("Luca Weiss <luca@z3ntu.xyz>");
MODULE_DESCRIPTION("SG Micro SGM3140 charge pump led driver"); MODULE_DESCRIPTION("SG Micro SGM3140 charge pump LED driver");
MODULE_LICENSE("GPL v2"); MODULE_LICENSE("GPL v2");
...@@ -80,7 +80,6 @@ static int spi_byte_brightness_set_blocking(struct led_classdev *dev, ...@@ -80,7 +80,6 @@ static int spi_byte_brightness_set_blocking(struct led_classdev *dev,
static int spi_byte_probe(struct spi_device *spi) static int spi_byte_probe(struct spi_device *spi)
{ {
const struct of_device_id *of_dev_id;
struct device_node *child; struct device_node *child;
struct device *dev = &spi->dev; struct device *dev = &spi->dev;
struct spi_byte_led *led; struct spi_byte_led *led;
...@@ -88,15 +87,11 @@ static int spi_byte_probe(struct spi_device *spi) ...@@ -88,15 +87,11 @@ static int spi_byte_probe(struct spi_device *spi)
const char *state; const char *state;
int ret; int ret;
of_dev_id = of_match_device(spi_byte_dt_ids, dev); if (of_get_available_child_count(dev_of_node(dev)) != 1) {
if (!of_dev_id)
return -EINVAL;
if (of_get_child_count(dev->of_node) != 1) {
dev_err(dev, "Device must have exactly one LED sub-node."); dev_err(dev, "Device must have exactly one LED sub-node.");
return -EINVAL; return -EINVAL;
} }
child = of_get_next_child(dev->of_node, NULL); child = of_get_next_available_child(dev_of_node(dev), NULL);
led = devm_kzalloc(dev, sizeof(*led), GFP_KERNEL); led = devm_kzalloc(dev, sizeof(*led), GFP_KERNEL);
if (!led) if (!led)
...@@ -106,7 +101,7 @@ static int spi_byte_probe(struct spi_device *spi) ...@@ -106,7 +101,7 @@ static int spi_byte_probe(struct spi_device *spi)
strlcpy(led->name, name, sizeof(led->name)); strlcpy(led->name, name, sizeof(led->name));
led->spi = spi; led->spi = spi;
mutex_init(&led->mutex); mutex_init(&led->mutex);
led->cdef = of_dev_id->data; led->cdef = device_get_match_data(dev);
led->ldev.name = led->name; led->ldev.name = led->name;
led->ldev.brightness = LED_OFF; led->ldev.brightness = LED_OFF;
led->ldev.max_brightness = led->cdef->max_value - led->cdef->off_value; led->ldev.max_brightness = led->cdef->max_value - led->cdef->off_value;
......
...@@ -55,8 +55,9 @@ static void syscon_led_set(struct led_classdev *led_cdev, ...@@ -55,8 +55,9 @@ static void syscon_led_set(struct led_classdev *led_cdev,
static int syscon_led_probe(struct platform_device *pdev) static int syscon_led_probe(struct platform_device *pdev)
{ {
struct led_init_data init_data = {};
struct device *dev = &pdev->dev; struct device *dev = &pdev->dev;
struct device_node *np = dev->of_node; struct device_node *np = dev_of_node(dev);
struct device *parent; struct device *parent;
struct regmap *map; struct regmap *map;
struct syscon_led *sled; struct syscon_led *sled;
...@@ -68,7 +69,7 @@ static int syscon_led_probe(struct platform_device *pdev) ...@@ -68,7 +69,7 @@ static int syscon_led_probe(struct platform_device *pdev)
dev_err(dev, "no parent for syscon LED\n"); dev_err(dev, "no parent for syscon LED\n");
return -ENODEV; return -ENODEV;
} }
map = syscon_node_to_regmap(parent->of_node); map = syscon_node_to_regmap(dev_of_node(parent));
if (IS_ERR(map)) { if (IS_ERR(map)) {
dev_err(dev, "no regmap for syscon LED parent\n"); dev_err(dev, "no regmap for syscon LED parent\n");
return PTR_ERR(map); return PTR_ERR(map);
...@@ -84,10 +85,6 @@ static int syscon_led_probe(struct platform_device *pdev) ...@@ -84,10 +85,6 @@ static int syscon_led_probe(struct platform_device *pdev)
return -EINVAL; return -EINVAL;
if (of_property_read_u32(np, "mask", &sled->mask)) if (of_property_read_u32(np, "mask", &sled->mask))
return -EINVAL; return -EINVAL;
sled->cdev.name =
of_get_property(np, "label", NULL) ? : np->name;
sled->cdev.default_trigger =
of_get_property(np, "linux,default-trigger", NULL);
state = of_get_property(np, "default-state", NULL); state = of_get_property(np, "default-state", NULL);
if (state) { if (state) {
...@@ -115,7 +112,9 @@ static int syscon_led_probe(struct platform_device *pdev) ...@@ -115,7 +112,9 @@ static int syscon_led_probe(struct platform_device *pdev)
} }
sled->cdev.brightness_set = syscon_led_set; sled->cdev.brightness_set = syscon_led_set;
ret = devm_led_classdev_register(dev, &sled->cdev); init_data.fwnode = of_fwnode_handle(np);
ret = devm_led_classdev_register_ext(dev, &sled->cdev, &init_data);
if (ret < 0) if (ret < 0)
return ret; return ret;
......
...@@ -69,23 +69,6 @@ ...@@ -69,23 +69,6 @@
* defaulted. Similarly the banks know if each time was explicit or a * defaulted. Similarly the banks know if each time was explicit or a
* default. Defaults are permitted to be changed freely - they are * default. Defaults are permitted to be changed freely - they are
* not recognised when matching. * not recognised when matching.
*
*
* An led-tca6507 device must be provided with platform data or
* configured via devicetree.
*
* The platform-data lists for each output: the name, default trigger,
* and whether the signal is being used as a GPIO rather than an LED.
* 'struct led_plaform_data' is used for this. If 'name' is NULL, the
* output isn't used. If 'flags' is TCA6507_MAKE_GPIO, the output is
* a GPO. The "struct led_platform_data" can be embedded in a "struct
* tca6507_platform_data" which adds a 'gpio_base' for the GPIOs, and
* a 'setup' callback which is called once the GPIOs are available.
*
* When configured via devicetree there is one child for each output.
* The "reg" determines the output number and "compatible" determines
* whether it is an LED or a GPIO. "linux,default-trigger" can set a
* default trigger.
*/ */
#include <linux/module.h> #include <linux/module.h>
...@@ -94,9 +77,8 @@ ...@@ -94,9 +77,8 @@
#include <linux/err.h> #include <linux/err.h>
#include <linux/i2c.h> #include <linux/i2c.h>
#include <linux/gpio/driver.h> #include <linux/gpio/driver.h>
#include <linux/property.h>
#include <linux/workqueue.h> #include <linux/workqueue.h>
#include <linux/leds-tca6507.h>
#include <linux/of.h>
/* LED select registers determine the source that drives LED outputs */ /* LED select registers determine the source that drives LED outputs */
#define TCA6507_LS_LED_OFF 0x0 /* Output HI-Z (off) */ #define TCA6507_LS_LED_OFF 0x0 /* Output HI-Z (off) */
...@@ -108,6 +90,15 @@ ...@@ -108,6 +90,15 @@
#define TCA6507_LS_BLINK0 0x6 /* Blink at Bank0 rate */ #define TCA6507_LS_BLINK0 0x6 /* Blink at Bank0 rate */
#define TCA6507_LS_BLINK1 0x7 /* Blink at Bank1 rate */ #define TCA6507_LS_BLINK1 0x7 /* Blink at Bank1 rate */
struct tca6507_platform_data {
struct led_platform_data leds;
#ifdef CONFIG_GPIOLIB
int gpio_base;
#endif
};
#define TCA6507_MAKE_GPIO 1
enum { enum {
BANK0, BANK0,
BANK1, BANK1,
...@@ -189,7 +180,6 @@ struct tca6507_chip { ...@@ -189,7 +180,6 @@ struct tca6507_chip {
} leds[NUM_LEDS]; } leds[NUM_LEDS];
#ifdef CONFIG_GPIOLIB #ifdef CONFIG_GPIOLIB
struct gpio_chip gpio; struct gpio_chip gpio;
const char *gpio_name[NUM_LEDS];
int gpio_map[NUM_LEDS]; int gpio_map[NUM_LEDS];
#endif #endif
}; };
...@@ -628,7 +618,7 @@ static int tca6507_gpio_direction_output(struct gpio_chip *gc, ...@@ -628,7 +618,7 @@ static int tca6507_gpio_direction_output(struct gpio_chip *gc,
return 0; return 0;
} }
static int tca6507_probe_gpios(struct i2c_client *client, static int tca6507_probe_gpios(struct device *dev,
struct tca6507_chip *tca, struct tca6507_chip *tca,
struct tca6507_platform_data *pdata) struct tca6507_platform_data *pdata)
{ {
...@@ -639,7 +629,6 @@ static int tca6507_probe_gpios(struct i2c_client *client, ...@@ -639,7 +629,6 @@ static int tca6507_probe_gpios(struct i2c_client *client,
for (i = 0; i < NUM_LEDS; i++) for (i = 0; i < NUM_LEDS; i++)
if (pdata->leds.leds[i].name && pdata->leds.leds[i].flags) { if (pdata->leds.leds[i].name && pdata->leds.leds[i].flags) {
/* Configure as a gpio */ /* Configure as a gpio */
tca->gpio_name[gpios] = pdata->leds.leds[i].name;
tca->gpio_map[gpios] = i; tca->gpio_map[gpios] = i;
gpios++; gpios++;
} }
...@@ -648,23 +637,20 @@ static int tca6507_probe_gpios(struct i2c_client *client, ...@@ -648,23 +637,20 @@ static int tca6507_probe_gpios(struct i2c_client *client,
return 0; return 0;
tca->gpio.label = "gpio-tca6507"; tca->gpio.label = "gpio-tca6507";
tca->gpio.names = tca->gpio_name;
tca->gpio.ngpio = gpios; tca->gpio.ngpio = gpios;
tca->gpio.base = pdata->gpio_base; tca->gpio.base = pdata->gpio_base;
tca->gpio.owner = THIS_MODULE; tca->gpio.owner = THIS_MODULE;
tca->gpio.direction_output = tca6507_gpio_direction_output; tca->gpio.direction_output = tca6507_gpio_direction_output;
tca->gpio.set = tca6507_gpio_set_value; tca->gpio.set = tca6507_gpio_set_value;
tca->gpio.parent = &client->dev; tca->gpio.parent = dev;
#ifdef CONFIG_OF_GPIO #ifdef CONFIG_OF_GPIO
tca->gpio.of_node = of_node_get(client->dev.of_node); tca->gpio.of_node = of_node_get(dev_of_node(dev));
#endif #endif
err = gpiochip_add_data(&tca->gpio, tca); err = gpiochip_add_data(&tca->gpio, tca);
if (err) { if (err) {
tca->gpio.ngpio = 0; tca->gpio.ngpio = 0;
return err; return err;
} }
if (pdata->setup)
pdata->setup(tca->gpio.base, tca->gpio.ngpio);
return 0; return 0;
} }
...@@ -674,7 +660,7 @@ static void tca6507_remove_gpio(struct tca6507_chip *tca) ...@@ -674,7 +660,7 @@ static void tca6507_remove_gpio(struct tca6507_chip *tca)
gpiochip_remove(&tca->gpio); gpiochip_remove(&tca->gpio);
} }
#else /* CONFIG_GPIOLIB */ #else /* CONFIG_GPIOLIB */
static int tca6507_probe_gpios(struct i2c_client *client, static int tca6507_probe_gpios(struct device *dev,
struct tca6507_chip *tca, struct tca6507_chip *tca,
struct tca6507_platform_data *pdata) struct tca6507_platform_data *pdata)
{ {
...@@ -685,44 +671,50 @@ static void tca6507_remove_gpio(struct tca6507_chip *tca) ...@@ -685,44 +671,50 @@ static void tca6507_remove_gpio(struct tca6507_chip *tca)
} }
#endif /* CONFIG_GPIOLIB */ #endif /* CONFIG_GPIOLIB */
#ifdef CONFIG_OF
static struct tca6507_platform_data * static struct tca6507_platform_data *
tca6507_led_dt_init(struct i2c_client *client) tca6507_led_dt_init(struct device *dev)
{ {
struct device_node *np = client->dev.of_node, *child;
struct tca6507_platform_data *pdata; struct tca6507_platform_data *pdata;
struct fwnode_handle *child;
struct led_info *tca_leds; struct led_info *tca_leds;
int count; int count;
count = of_get_child_count(np); count = device_get_child_node_count(dev);
if (!count || count > NUM_LEDS) if (!count || count > NUM_LEDS)
return ERR_PTR(-ENODEV); return ERR_PTR(-ENODEV);
tca_leds = devm_kcalloc(&client->dev, tca_leds = devm_kcalloc(dev, NUM_LEDS, sizeof(struct led_info),
NUM_LEDS, sizeof(struct led_info), GFP_KERNEL); GFP_KERNEL);
if (!tca_leds) if (!tca_leds)
return ERR_PTR(-ENOMEM); return ERR_PTR(-ENOMEM);
for_each_child_of_node(np, child) { device_for_each_child_node(dev, child) {
struct led_info led; struct led_info led;
u32 reg; u32 reg;
int ret; int ret;
led.name = if (fwnode_property_read_string(child, "label", &led.name))
of_get_property(child, "label", NULL) ? : child->name; led.name = fwnode_get_name(child);
led.default_trigger =
of_get_property(child, "linux,default-trigger", NULL); fwnode_property_read_string(child, "linux,default-trigger",
&led.default_trigger);
led.flags = 0; led.flags = 0;
if (of_property_match_string(child, "compatible", "gpio") >= 0) if (fwnode_property_match_string(child, "compatible",
"gpio") >= 0)
led.flags |= TCA6507_MAKE_GPIO; led.flags |= TCA6507_MAKE_GPIO;
ret = of_property_read_u32(child, "reg", &reg);
if (ret != 0 || reg >= NUM_LEDS) ret = fwnode_property_read_u32(child, "reg", &reg);
continue; if (ret || reg >= NUM_LEDS) {
fwnode_handle_put(child);
return ERR_PTR(ret ? : -EINVAL);
}
tca_leds[reg] = led; tca_leds[reg] = led;
} }
pdata = devm_kzalloc(&client->dev,
sizeof(struct tca6507_platform_data), GFP_KERNEL); pdata = devm_kzalloc(dev, sizeof(struct tca6507_platform_data),
GFP_KERNEL);
if (!pdata) if (!pdata)
return ERR_PTR(-ENOMEM); return ERR_PTR(-ENOMEM);
...@@ -731,48 +723,37 @@ tca6507_led_dt_init(struct i2c_client *client) ...@@ -731,48 +723,37 @@ tca6507_led_dt_init(struct i2c_client *client)
#ifdef CONFIG_GPIOLIB #ifdef CONFIG_GPIOLIB
pdata->gpio_base = -1; pdata->gpio_base = -1;
#endif #endif
return pdata; return pdata;
} }
static const struct of_device_id of_tca6507_leds_match[] = { static const struct of_device_id __maybe_unused of_tca6507_leds_match[] = {
{ .compatible = "ti,tca6507", }, { .compatible = "ti,tca6507", },
{}, {},
}; };
MODULE_DEVICE_TABLE(of, of_tca6507_leds_match); MODULE_DEVICE_TABLE(of, of_tca6507_leds_match);
#else
static struct tca6507_platform_data *
tca6507_led_dt_init(struct i2c_client *client)
{
return ERR_PTR(-ENODEV);
}
#endif
static int tca6507_probe(struct i2c_client *client, static int tca6507_probe(struct i2c_client *client,
const struct i2c_device_id *id) const struct i2c_device_id *id)
{ {
struct tca6507_chip *tca; struct device *dev = &client->dev;
struct i2c_adapter *adapter; struct i2c_adapter *adapter;
struct tca6507_chip *tca;
struct tca6507_platform_data *pdata; struct tca6507_platform_data *pdata;
int err; int err;
int i = 0; int i = 0;
adapter = client->adapter; adapter = client->adapter;
pdata = dev_get_platdata(&client->dev);
if (!i2c_check_functionality(adapter, I2C_FUNC_I2C)) if (!i2c_check_functionality(adapter, I2C_FUNC_I2C))
return -EIO; return -EIO;
if (!pdata || pdata->leds.num_leds != NUM_LEDS) { pdata = tca6507_led_dt_init(dev);
pdata = tca6507_led_dt_init(client);
if (IS_ERR(pdata)) { if (IS_ERR(pdata)) {
dev_err(&client->dev, "Need %d entries in platform-data list\n", dev_err(dev, "Need %d entries in platform-data list\n", NUM_LEDS);
NUM_LEDS);
return PTR_ERR(pdata); return PTR_ERR(pdata);
} }
} tca = devm_kzalloc(dev, sizeof(*tca), GFP_KERNEL);
tca = devm_kzalloc(&client->dev, sizeof(*tca), GFP_KERNEL);
if (!tca) if (!tca)
return -ENOMEM; return -ENOMEM;
...@@ -793,13 +774,12 @@ static int tca6507_probe(struct i2c_client *client, ...@@ -793,13 +774,12 @@ static int tca6507_probe(struct i2c_client *client,
l->led_cdev.brightness_set = tca6507_brightness_set; l->led_cdev.brightness_set = tca6507_brightness_set;
l->led_cdev.blink_set = tca6507_blink_set; l->led_cdev.blink_set = tca6507_blink_set;
l->bank = -1; l->bank = -1;
err = led_classdev_register(&client->dev, err = led_classdev_register(dev, &l->led_cdev);
&l->led_cdev);
if (err < 0) if (err < 0)
goto exit; goto exit;
} }
} }
err = tca6507_probe_gpios(client, tca, pdata); err = tca6507_probe_gpios(dev, tca, pdata);
if (err) if (err)
goto exit; goto exit;
/* set all registers to known state - zero */ /* set all registers to known state - zero */
......
...@@ -148,22 +148,17 @@ static int ...@@ -148,22 +148,17 @@ static int
tlc591xx_probe(struct i2c_client *client, tlc591xx_probe(struct i2c_client *client,
const struct i2c_device_id *id) const struct i2c_device_id *id)
{ {
struct device_node *np = client->dev.of_node, *child; struct device_node *np = dev_of_node(&client->dev), *child;
struct device *dev = &client->dev; struct device *dev = &client->dev;
const struct of_device_id *match;
const struct tlc591xx *tlc591xx; const struct tlc591xx *tlc591xx;
struct tlc591xx_priv *priv; struct tlc591xx_priv *priv;
int err, count, reg; int err, count, reg;
match = of_match_device(of_tlc591xx_leds_match, dev); tlc591xx = device_get_match_data(dev);
if (!match)
return -ENODEV;
tlc591xx = match->data;
if (!np) if (!np)
return -ENODEV; return -ENODEV;
count = of_get_child_count(np); count = of_get_available_child_count(np);
if (!count || count > tlc591xx->max_leds) if (!count || count > tlc591xx->max_leds)
return -EINVAL; return -EINVAL;
...@@ -185,7 +180,7 @@ tlc591xx_probe(struct i2c_client *client, ...@@ -185,7 +180,7 @@ tlc591xx_probe(struct i2c_client *client,
if (err < 0) if (err < 0)
return err; return err;
for_each_child_of_node(np, child) { for_each_available_child_of_node(np, child) {
struct tlc591xx_led *led; struct tlc591xx_led *led;
struct led_init_data init_data = {}; struct led_init_data init_data = {};
...@@ -204,9 +199,6 @@ tlc591xx_probe(struct i2c_client *client, ...@@ -204,9 +199,6 @@ tlc591xx_probe(struct i2c_client *client,
led = &priv->leds[reg]; led = &priv->leds[reg];
led->active = true; led->active = true;
led->ldev.default_trigger =
of_get_property(child, "linux,default-trigger", NULL);
led->priv = priv; led->priv = priv;
led->led_no = reg; led->led_no = reg;
led->ldev.brightness_set_blocking = tlc591xx_brightness_set; led->ldev.brightness_set_blocking = tlc591xx_brightness_set;
...@@ -214,10 +206,10 @@ tlc591xx_probe(struct i2c_client *client, ...@@ -214,10 +206,10 @@ tlc591xx_probe(struct i2c_client *client,
err = devm_led_classdev_register_ext(dev, &led->ldev, err = devm_led_classdev_register_ext(dev, &led->ldev,
&init_data); &init_data);
if (err < 0) { if (err < 0) {
if (err != -EPROBE_DEFER) of_node_put(child);
dev_err(dev, "couldn't register LED %s\n", return dev_err_probe(dev, err,
"couldn't register LED %s\n",
led->ldev.name); led->ldev.name);
return err;
} }
} }
return 0; return 0;
......
...@@ -121,8 +121,6 @@ static int omnia_led_register(struct i2c_client *client, struct omnia_led *led, ...@@ -121,8 +121,6 @@ static int omnia_led_register(struct i2c_client *client, struct omnia_led *led,
cdev->max_brightness = 255; cdev->max_brightness = 255;
cdev->brightness_set_blocking = omnia_led_brightness_set_blocking; cdev->brightness_set_blocking = omnia_led_brightness_set_blocking;
of_property_read_string(np, "linux,default-trigger", &cdev->default_trigger);
/* put the LED into software mode */ /* put the LED into software mode */
ret = i2c_smbus_write_byte_data(client, CMD_LED_MODE, ret = i2c_smbus_write_byte_data(client, CMD_LED_MODE,
CMD_LED_MODE_LED(led->reg) | CMD_LED_MODE_LED(led->reg) |
...@@ -210,7 +208,7 @@ static int omnia_leds_probe(struct i2c_client *client, ...@@ -210,7 +208,7 @@ static int omnia_leds_probe(struct i2c_client *client,
const struct i2c_device_id *id) const struct i2c_device_id *id)
{ {
struct device *dev = &client->dev; struct device *dev = &client->dev;
struct device_node *np = dev->of_node, *child; struct device_node *np = dev_of_node(dev), *child;
struct omnia_leds *leds; struct omnia_leds *leds;
struct omnia_led *led; struct omnia_led *led;
int ret, count; int ret, count;
...@@ -236,8 +234,10 @@ static int omnia_leds_probe(struct i2c_client *client, ...@@ -236,8 +234,10 @@ static int omnia_leds_probe(struct i2c_client *client,
led = &leds->leds[0]; led = &leds->leds[0];
for_each_available_child_of_node(np, child) { for_each_available_child_of_node(np, child) {
ret = omnia_led_register(client, led, child); ret = omnia_led_register(client, led, child);
if (ret < 0) if (ret < 0) {
of_node_put(child);
return ret; return ret;
}
led += ret; led += ret;
} }
......
...@@ -2,14 +2,18 @@ ...@@ -2,14 +2,18 @@
/* /*
* ledtrig-cpu.c - LED trigger based on CPU activity * ledtrig-cpu.c - LED trigger based on CPU activity
* *
* This LED trigger will be registered for each possible CPU and named as * This LED trigger will be registered for first 8 CPUs and named
* cpu0, cpu1, cpu2, cpu3, etc. * as cpu0..cpu7. There's additional trigger called cpu that
* is on when any CPU is active.
*
* If you want support for arbitrary number of CPUs, make it one trigger,
* with additional sysfs file selecting which CPU to watch.
* *
* It can be bound to any LED just like other triggers using either a * It can be bound to any LED just like other triggers using either a
* board file or via sysfs interface. * board file or via sysfs interface.
* *
* An API named ledtrig_cpu is exported for any user, who want to add CPU * An API named ledtrig_cpu is exported for any user, who want to add CPU
* activity indication in their code * activity indication in their code.
* *
* Copyright 2011 Linus Walleij <linus.walleij@linaro.org> * Copyright 2011 Linus Walleij <linus.walleij@linaro.org>
* Copyright 2011 - 2012 Bryan Wu <bryan.wu@canonical.com> * Copyright 2011 - 2012 Bryan Wu <bryan.wu@canonical.com>
...@@ -145,6 +149,9 @@ static int __init ledtrig_cpu_init(void) ...@@ -145,6 +149,9 @@ static int __init ledtrig_cpu_init(void)
for_each_possible_cpu(cpu) { for_each_possible_cpu(cpu) {
struct led_trigger_cpu *trig = &per_cpu(cpu_trig, cpu); struct led_trigger_cpu *trig = &per_cpu(cpu_trig, cpu);
if (cpu >= 8)
continue;
snprintf(trig->name, MAX_NAME_LEN, "cpu%d", cpu); snprintf(trig->name, MAX_NAME_LEN, "cpu%d", cpu);
led_trigger_register_simple(trig->name, &trig->_trig); led_trigger_register_simple(trig->name, &trig->_trig);
......
/* SPDX-License-Identifier: GPL-2.0-only */
/*
* TCA6507 LED chip driver.
*
* Copyright (C) 2011 Neil Brown <neil@brown.name>
*/
#ifndef __LINUX_TCA6507_H
#define __LINUX_TCA6507_H
#include <linux/leds.h>
struct tca6507_platform_data {
struct led_platform_data leds;
#ifdef CONFIG_GPIOLIB
int gpio_base;
void (*setup)(unsigned gpio_base, unsigned ngpio);
#endif
};
#define TCA6507_MAKE_GPIO 1
#endif /* __LINUX_TCA6507_H*/
/* SPDX-License-Identifier: GPL-2.0-only */
/*
* PCA963X LED chip driver.
*
* Copyright 2012 bct electronic GmbH
* Copyright 2013 Qtechnology A/S
*/
#ifndef __LINUX_PCA963X_H
#define __LINUX_PCA963X_H
#include <linux/leds.h>
enum pca963x_outdrv {
PCA963X_OPEN_DRAIN,
PCA963X_TOTEM_POLE, /* aka push-pull */
};
enum pca963x_blink_type {
PCA963X_SW_BLINK,
PCA963X_HW_BLINK,
};
enum pca963x_direction {
PCA963X_NORMAL,
PCA963X_INVERTED,
};
struct pca963x_platform_data {
struct led_platform_data leds;
enum pca963x_outdrv outdrv;
enum pca963x_blink_type blink_type;
enum pca963x_direction dir;
};
#endif /* __LINUX_PCA963X_H*/
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment