[<prev] [next>] [<thread-prev] [day] [month] [year] [list]
Message-ID: <20250508143146.GP3865826@google.com>
Date: Thu, 8 May 2025 15:31:46 +0100
From: Lee Jones <lee@...nel.org>
To: Matthias Fend <matthias.fend@...end.at>
Cc: Pavel Machek <pavel@....cz>, Rob Herring <robh@...nel.org>,
Krzysztof Kozlowski <krzk+dt@...nel.org>,
Conor Dooley <conor+dt@...nel.org>, Pavel Machek <pavel@...nel.org>,
linux-leds@...r.kernel.org, devicetree@...r.kernel.org,
linux-kernel@...r.kernel.org,
bsp-development.geo@...ca-geosystems.com
Subject: Re: [PATCH v3 2/2] leds: tps6131x: add support for Texas Instruments
TPS6131X flash LED driver
On Fri, 02 May 2025, Matthias Fend wrote:
> Hi Lee,
>
> thank you for taking the time for this review.
>
> Am 01.05.2025 um 13:03 schrieb Lee Jones:
> > On Wed, 23 Apr 2025, Matthias Fend wrote:
> >
> > > The TPS61310/TPS61311 is a flash LED driver with I2C interface. Its power
> > > stage is capable of supplying a maximum total current of roughly 1500mA.
> > > The TPS6131x provides three constant-current sinks, capable of sinking up
> > > to 2 × 400mA (LED1 and LED3) and 800mA (LED2) in flash mode. In torch mode
> > > each sink (LED1, LED2, LED3) supports currents up to 175mA.
> > >
> > > Signed-off-by: Matthias Fend <matthias.fend@...end.at>
> > > ---
> > > MAINTAINERS | 7 +
> > > drivers/leds/flash/Kconfig | 11 +
> > > drivers/leds/flash/Makefile | 1 +
> > > drivers/leds/flash/leds-tps6131x.c | 798 +++++++++++++++++++++++++++++++++++++
> > > 4 files changed, 817 insertions(+)
> >
> > Looks pretty good in general. Just a few nits, really.
> >
> > > diff --git a/MAINTAINERS b/MAINTAINERS
> > > index fa1e04e87d1d63c1b01b59744d7ade00fe5a1885..67e7c165efd332dfe3bc5ec64a4cc10845d7bbd6 100644
> > > --- a/MAINTAINERS
> > > +++ b/MAINTAINERS
> > > @@ -23983,6 +23983,13 @@ F: Documentation/devicetree/bindings/hwmon/ti,tps23861.yaml
> > > F: Documentation/hwmon/tps23861.rst
> > > F: drivers/hwmon/tps23861.c
> > > +TEXAS INSTRUMENTS TPS6131X FLASH LED DRIVER
> > > +M: Matthias Fend <matthias.fend@...end.at>
> > > +L: linux-leds@...r.kernel.org
> > > +S: Maintained
> > > +F: Documentation/devicetree/bindings/leds/ti,tps6131x.yaml
> > > +F: drivers/leds/flash/leds-tps6131x.c
> > > +
> > > TEXAS INSTRUMENTS' DAC7612 DAC DRIVER
> > > M: Ricardo Ribalda <ribalda@...nel.org>
> > > L: linux-iio@...r.kernel.org
> > > diff --git a/drivers/leds/flash/Kconfig b/drivers/leds/flash/Kconfig
> > > index f39f0bfe6eefcd376405d9d35dc36e323a485002..55ca663ca506ad8be627f58f6d6308368ea2b928 100644
> > > --- a/drivers/leds/flash/Kconfig
> > > +++ b/drivers/leds/flash/Kconfig
> > > @@ -132,4 +132,15 @@ config LEDS_SY7802
> > > This driver can be built as a module, it will be called "leds-sy7802".
> > > +config LEDS_TPS6131X
> > > + tristate "LED support for TI TPS6131x flash LED driver"
> > > + depends on I2C && OF
> > > + depends on GPIOLIB
> > > + select REGMAP_I2C
> > > + help
> > > + This option enables support for Texas Instruments TPS61310/TPS61311
> > > + flash LED driver.
> > > +
> > > + This driver can be built as a module, it will be called "leds-tps6131x".
> > > +
> > > endif # LEDS_CLASS_FLASH
> > > diff --git a/drivers/leds/flash/Makefile b/drivers/leds/flash/Makefile
> > > index 48860eeced79513a0ed303e4af3db9bfe9724b7e..712fb737a428e42747e1aa339058dc4306ade9c8 100644
> > > --- a/drivers/leds/flash/Makefile
> > > +++ b/drivers/leds/flash/Makefile
> > > @@ -12,3 +12,4 @@ obj-$(CONFIG_LEDS_RT4505) += leds-rt4505.o
> > > obj-$(CONFIG_LEDS_RT8515) += leds-rt8515.o
> > > obj-$(CONFIG_LEDS_SGM3140) += leds-sgm3140.o
> > > obj-$(CONFIG_LEDS_SY7802) += leds-sy7802.o
> > > +obj-$(CONFIG_LEDS_TPS6131X) += leds-tps6131x.o
> > > diff --git a/drivers/leds/flash/leds-tps6131x.c b/drivers/leds/flash/leds-tps6131x.c
> > > new file mode 100644
> > > index 0000000000000000000000000000000000000000..c123fb3f908c865f8a9e60b0339067cdb5a864f6
> > > --- /dev/null
> > > +++ b/drivers/leds/flash/leds-tps6131x.c
> > > @@ -0,0 +1,798 @@
> > > +// SPDX-License-Identifier: GPL-2.0-or-later
> > > +/*
> > > + * Texas Instruments TPS61310/TPS61311 flash LED driver with I2C interface
> > > + *
> > > + * Copyright 2025 Matthias Fend <matthias.fend@...end.at>
> > > + */
> > > +
> > > +#include <linux/gpio/consumer.h>
> > > +#include <linux/i2c.h>
> > > +#include <linux/led-class-flash.h>
> > > +#include <linux/leds.h>
> > > +#include <linux/module.h>
> > > +#include <linux/regmap.h>
> > > +#include <media/v4l2-flash-led-class.h>
> > > +
> > > +/* Registers */
> >
> > This little header comments are superfluous IMHO.
> >
> > The nomenclature of the defines is good enough.
>
> Okay, I'll remove them.
>
> >
> > > +#define TPS6131X_REG_0 0x00
> > > +#define TPS6131X_REG_0_RESET BIT(7)
> >
> > Suggest adding a space or two before the bit defines, like:
> >
> > #define TPS6131X_REG_0 0x00
> > #define TPS6131X_REG_0_RESET BIT(7)
>
> I actually like that too, but I couldn't convince clang-format to keep the
> spaces. The automatic formatting was more valuable to me in the end. If you
> know how to have both, I'd be very happy :)
I wouldn't let the tooling push you around.
IMHO that's a false positive and should be reported.
> > > +#define TPS6131X_REG_0_DCLC13 GENMASK(5, 3)
> > > +#define TPS6131X_REG_0_DCLC13_SHIFT 3
> > > +#define TPS6131X_REG_0_DCLC2 GENMASK(2, 0)
> > > +#define TPS6131X_REG_0_DCLC2_SHIFT 0
> > > +
> > > +#define TPS6131X_REG_1 0x01
> > > +#define TPS6131X_REG_1_MODE GENMASK(7, 6)
> > > +#define TPS6131X_REG_1_MODE_SHIFT 6
> > > +#define TPS6131X_REG_1_FC2 GENMASK(5, 0)
> > > +#define TPS6131X_REG_1_FC2_SHIFT 0
> > > +
> > > +#define TPS6131X_REG_2 0x02
> > > +#define TPS6131X_REG_2_MODE GENMASK(7, 6)
> > > +#define TPS6131X_REG_2_MODE_SHIFT 6
> > > +#define TPS6131X_REG_2_ENVM BIT(5)
> > > +#define TPS6131X_REG_2_FC13 GENMASK(4, 0)
> > > +#define TPS6131X_REG_2_FC13_SHIFT 0
> > > +
> > > +#define TPS6131X_REG_3 0x03
> > > +#define TPS6131X_REG_3_STIM GENMASK(7, 5)
> > > +#define TPS6131X_REG_3_STIM_SHIFT 5
> > > +#define TPS6131X_REG_3_HPFL BIT(4)
> > > +#define TPS6131X_REG_3_SELSTIM_TO BIT(3)
> > > +#define TPS6131X_REG_3_STT BIT(2)
> > > +#define TPS6131X_REG_3_SFT BIT(1)
> > > +#define TPS6131X_REG_3_TXMASK BIT(0)
> > > +
> > > +#define TPS6131X_REG_4 0x04
> > > +#define TPS6131X_REG_4_PG BIT(7)
> > > +#define TPS6131X_REG_4_HOTDIE_HI BIT(6)
> > > +#define TPS6131X_REG_4_HOTDIE_LO BIT(5)
> > > +#define TPS6131X_REG_4_ILIM BIT(4)
> > > +#define TPS6131X_REG_4_INDC GENMASK(3, 0)
> > > +#define TPS6131X_REG_4_INDC_SHIFT 0
> > > +
> > > +#define TPS6131X_REG_5 0x05
> > > +#define TPS6131X_REG_5_SELFCAL BIT(7)
> > > +#define TPS6131X_REG_5_ENPSM BIT(6)
> > > +#define TPS6131X_REG_5_STSTRB1_DIR BIT(5)
> > > +#define TPS6131X_REG_5_GPIO BIT(4)
> > > +#define TPS6131X_REG_5_GPIOTYPE BIT(3)
> > > +#define TPS6131X_REG_5_ENLED3 BIT(2)
> > > +#define TPS6131X_REG_5_ENLED2 BIT(1)
> > > +#define TPS6131X_REG_5_ENLED1 BIT(0)
> > > +
> > > +#define TPS6131X_REG_6 0x06
> > > +#define TPS6131X_REG_6_ENTS BIT(7)
> > > +#define TPS6131X_REG_6_LEDHOT BIT(6)
> > > +#define TPS6131X_REG_6_LEDWARN BIT(5)
> > > +#define TPS6131X_REG_6_LEDHDR BIT(4)
> > > +#define TPS6131X_REG_6_OV GENMASK(3, 0)
> > > +#define TPS6131X_REG_6_OV_SHIFT 0
> > > +
> > > +#define TPS6131X_REG_7 0x07
> > > +#define TPS6131X_REG_7_ENBATMON BIT(7)
> > > +#define TPS6131X_REG_7_BATDROOP GENMASK(6, 4)
> > > +#define TPS6131X_REG_7_BATDROOP_SHIFT 4
> > > +#define TPS6131X_REG_7_REVID GENMASK(2, 0)
> > > +#define TPS6131X_REG_7_REVID_SHIFT 0
> > > +
> > > +/* Constants */
> > > +
> > > +#define TPS6131X_MAX_CHANNELS 3
> > > +
> > > +#define TPS6131X_FLASH_MAX_I_CHAN13_MA 400
> > > +#define TPS6131X_FLASH_MAX_I_CHAN2_MA 800
> > > +#define TPS6131X_FLASH_STEP_I_MA 25
> > > +
> > > +#define TPS6131X_TORCH_MAX_I_CHAN13_MA 175
> > > +#define TPS6131X_TORCH_MAX_I_CHAN2_MA 175
> > > +#define TPS6131X_TORCH_STEP_I_MA 25
> > > +
> > > +#define TPS6131X_TORCH_REFRESH_INTERVAL_JIFFIES msecs_to_jiffies(10000)
> >
> > Why 10s?
>
> At the point where the timer starts, there's a comment that says the
> register must be updated within a timeout of 13 seconds.
> So I thought 10 seconds would be a nice, appropriate interval.
Maybe put this useful info in a small comment.
> >
> > > +#define UA_TO_MA(UA) ((UA) / 1000)
> >
> > I'm surprised a generic macro like this doesn't exist already.
> >
> > > +enum tps6131x_mode {
> > > + TPS6131X_MODE_SHUTDOWN = 0x0,
> > > + TPS6131X_MODE_TORCH = 0x1,
> > > + TPS6131X_MODE_FLASH = 0x2,
> >
> > Are these set in stone or are they arbitrary?
>
> The values are fixed because they are written directly to a register.
> In V1, I used an enum without values and mapped it to the register value in
> a function. I was asked to omit this mapping and use the enum directly.
>
> TLDR:
> [x] stone
> [ ] arbitrary
>
> >
> > > +};
> > > +
> > > +struct tps6131x {
> > > + struct device *dev;
> > > + struct regmap *regmap;
> > > + struct gpio_desc *reset_gpio;
> > > + /*
> > > + * Registers 0, 1, 2, and 3 control parts of the controller that are not completely
> > > + * independent of each other. Since some operations require the registers to be written in
> > > + * a specific order to avoid unwanted side effects, they are synchronized with a lock.
> > > + */
> > > + struct mutex lock; /* Hardware access lock for register 0, 1, 2 and 3 */
> > > + struct delayed_work torch_refresh_work;
> > > + bool valley_current_limit;
> > > + bool chan1_en;
> > > + bool chan2_en;
> > > + bool chan3_en;
> > > + struct fwnode_handle *led_node;
> > > + u32 max_flash_current_ma;
> > > + u32 step_flash_current_ma;
> > > + u32 max_torch_current_ma;
> > > + u32 step_torch_current_ma;
> > > + u32 max_timeout_us;
> > > + struct led_classdev_flash fled_cdev;
> > > + struct v4l2_flash *v4l2_flash;
> > > +};
> > > +
> > > +static struct tps6131x *fled_cdev_to_tps6131x(struct led_classdev_flash *fled_cdev)
> > > +{
> > > + return container_of(fled_cdev, struct tps6131x, fled_cdev);
> > > +}
> > > +
> > > +/*
> > > + * Register contents after a power on/reset. These values cannot be changed.
> > > + */
> > > +static const struct reg_default tps6131x_regmap_defaults[] = {
> > > + { TPS6131X_REG_0, (1 << TPS6131X_REG_0_DCLC13_SHIFT) | (2 << TPS6131X_REG_0_DCLC2_SHIFT) },
> >
> > Real values should be defined. What does 2 even do, etc?
>
> Initially, I only had the bare values here, so that no one would mistakenly
> think that something could be changed here.
> This information only contains the information about the reset state of the
> registers from the datasheet.
> The contents are only needed for correct cache handling and have no meaning
> for the driver itself.
> If I now create a separate 'INITIAL' define for each value, which is only
> used here, it would probably make it a bit harder to read, wouldn't it? What
> do you think?
I think that I like magic numbers to be defined. :)
> > > + { TPS6131X_REG_1, (0 << TPS6131X_REG_1_MODE_SHIFT) | (16 << TPS6131X_REG_1_FC2_SHIFT) },
> >
> > What's the point in oring 0 values? Can't you just omit them?
>
> Of course, I could simply omit that. But the point was that I had to
> describe which fields have which values.
> If I omitted that, the information would somehow disappear.
It's okay to leave things like this in for documentation purposes,
however since '0' is not documented with a define, it's effectively
meaningless. What is mode 0? If you define all of these magic numbers,
you can use the 'documentation' ~excuse~ reason guilt-free. :)
> > > + { TPS6131X_REG_2, (0 << TPS6131X_REG_2_MODE_SHIFT) | (8 << TPS6131X_REG_2_FC13_SHIFT) },
> > > + { TPS6131X_REG_3, (6 << TPS6131X_REG_3_STIM_SHIFT) | TPS6131X_REG_3_TXMASK },
> > > + { TPS6131X_REG_4, (0 << TPS6131X_REG_4_INDC_SHIFT) },
> > > + { TPS6131X_REG_5, TPS6131X_REG_5_ENPSM | TPS6131X_REG_5_STSTRB1_DIR |
> > > + TPS6131X_REG_5_GPIOTYPE | TPS6131X_REG_5_ENLED2 },
> > > + { TPS6131X_REG_6, (9 << TPS6131X_REG_6_OV_SHIFT) },
> > > + { TPS6131X_REG_7, (4 << TPS6131X_REG_7_BATDROOP_SHIFT) },
> > > +};
> > > +
> > > +/*
> > > + * These registers contain flags that are reset when read. Ensure that these registers are not read
> > > + * outside of a call from the driver.
> >
> > You can simplify to just the first sentence. The rest is implied.
>
> ACK.
>
> >
> > > + */
> > > +static bool tps6131x_regmap_precious(struct device *dev, unsigned int reg)
> > > +{
> > > + switch (reg) {
> > > + case TPS6131X_REG_3:
> > > + case TPS6131X_REG_4:
> > > + case TPS6131X_REG_6:
> > > + return true;
> > > + default:
> > > + return false;
> > > + }
> > > +}
> > > +
> > > +static const struct regmap_config tps6131x_regmap = {
> > > + .reg_bits = 8,
> > > + .val_bits = 8,
> > > + .max_register = TPS6131X_REG_7,
> > > + .reg_defaults = tps6131x_regmap_defaults,
> > > + .num_reg_defaults = ARRAY_SIZE(tps6131x_regmap_defaults),
> > > + .cache_type = REGCACHE_FLAT,
> >
> > This is new to me. Why was this chosen over the usual REGCACHE_MAPLE?
>
> Since the register space for this chip is very small, simple, compact, and
> contiguous, I think this is the right cache type for it. Did I miss
> something here?
No, it's just unusual. It's probably correct (you decide).
> > > + .precious_reg = &tps6131x_regmap_precious,
> > > +};
> > > +
> > > +struct tps6131x_timer_config {
> > > + u8 val;
> > > + u8 range;
> > > + u32 time_us;
> > > +};
> > > +
> > > +static const struct tps6131x_timer_config tps6131x_timer_configs[] = {
> > > + { .val = 0, .range = 1, .time_us = 5300 },
> > > + { .val = 1, .range = 1, .time_us = 10700 },
> > > + { .val = 2, .range = 1, .time_us = 16000 },
> > > + { .val = 3, .range = 1, .time_us = 21300 },
> > > + { .val = 4, .range = 1, .time_us = 26600 },
> > > + { .val = 5, .range = 1, .time_us = 32000 },
> > > + { .val = 6, .range = 1, .time_us = 37300 },
> > > + { .val = 0, .range = 0, .time_us = 68200 },
> > > + { .val = 7, .range = 1, .time_us = 71500 },
> > > + { .val = 1, .range = 0, .time_us = 102200 },
> > > + { .val = 2, .range = 0, .time_us = 136300 },
> > > + { .val = 3, .range = 0, .time_us = 170400 },
> > > + { .val = 4, .range = 0, .time_us = 204500 },
> > > + { .val = 5, .range = 0, .time_us = 340800 },
> > > + { .val = 6, .range = 0, .time_us = 579300 },
> > > + { .val = 7, .range = 0, .time_us = 852000 },
> > > +};
> > > +
> > > +static const struct tps6131x_timer_config *tps6131x_find_closest_timer_config(u32 timeout_us)
> > > +{
> > > + const struct tps6131x_timer_config *timer_config = &tps6131x_timer_configs[0];
> > > + u32 diff, min_diff = U32_MAX;
> > > + int i;
> > > +
> > > + for (i = 0; i < ARRAY_SIZE(tps6131x_timer_configs); i++) {
> > > + diff = abs(tps6131x_timer_configs[i].time_us - timeout_us);
> > > + if (diff < min_diff) {
> > > + timer_config = &tps6131x_timer_configs[i];
> > > + min_diff = diff;
> > > + if (!min_diff)
> > > + break;
> > > + }
> > > + }
> > > +
> > > + return timer_config;
> > > +}
> > > +
> > > +static int tps6131x_reset_chip(struct tps6131x *tps6131x)
> > > +{
> > > + int ret;
> > > +
> > > + if (tps6131x->reset_gpio) {
> > > + gpiod_set_value_cansleep(tps6131x->reset_gpio, 1);
> > > + fsleep(10);
> > > + gpiod_set_value_cansleep(tps6131x->reset_gpio, 0);
> > > + fsleep(100);
> > > + } else {
> > > + ret = regmap_update_bits(tps6131x->regmap, TPS6131X_REG_0, TPS6131X_REG_0_RESET,
> > > + TPS6131X_REG_0_RESET);
> > > + if (ret)
> > > + return ret;
> > > +
> > > + fsleep(100);
> > > +
> > > + ret = regmap_update_bits(tps6131x->regmap, TPS6131X_REG_0, TPS6131X_REG_0_RESET, 0);
> > > + if (ret)
> > > + return ret;
> > > + }
> > > +
> > > + return 0;
> > > +}
> > > +
> > > +static int tps6131x_init_chip(struct tps6131x *tps6131x)
> > > +{
> > > + u32 reg4, reg5, reg6;
> >
> > Why cant we just reuse a single variable called 'value'?
>
> This way, it's easy to spot a mistake. Each line can only contain parts that
> use the same 'reg<x>'.
> But I can easily change it and use only one variable.
Yes please. It's common practice.
> > > + int ret;
> > > +
> > > + reg4 = tps6131x->valley_current_limit ? TPS6131X_REG_4_ILIM : 0;
> >
> > Nicer on the eye if we un-squidge these.
> >
> > No need to group everything like this.
>
> I'm happy to adapt the code here as well, but unfortunately I haven't quite
> figured out how you'd like it to look. Could you please tell me more
> specifically what I should change?
Like this:
> > > + ret = regmap_write(tps6131x->regmap, TPS6131X_REG_4, reg4);
> > > + if (ret)
> > > + return ret;
> > > +
> > > + reg5 = TPS6131X_REG_5_ENPSM | TPS6131X_REG_5_STSTRB1_DIR | TPS6131X_REG_5_GPIOTYPE;
> > > + if (tps6131x->chan1_en)
> > > + reg5 |= TPS6131X_REG_5_ENLED1;
> > > +
> > > + if (tps6131x->chan2_en)
> > > + reg5 |= TPS6131X_REG_5_ENLED2;
> > > +
> > > + if (tps6131x->chan3_en)
> > > + reg5 |= TPS6131X_REG_5_ENLED3;
> > > + ret = regmap_write(tps6131x->regmap, TPS6131X_REG_5, reg5);
> > > + if (ret)
> > > + return ret;
> > > +
> > > + reg6 = TPS6131X_REG_6_ENTS;
> > > + ret = regmap_write(tps6131x->regmap, TPS6131X_REG_6, reg6);
> > > + if (ret)
> > > + return ret;
> > > +
> > > + return 0;
> > > +}
> > > +
> > > +static int tps6131x_set_mode(struct tps6131x *tps6131x, enum tps6131x_mode mode, bool force)
> > > +{
> > > + u8 val;
> > > +
> > > + val = mode << TPS6131X_REG_1_MODE_SHIFT;
> >
> > You could do this during assignment. Or, seeing as there is already a
> > line break required, in the call itself.
>
> ACK.
>
> >
> > > +
> > > + return regmap_update_bits_base(tps6131x->regmap, TPS6131X_REG_1, TPS6131X_REG_1_MODE, val,
> > > + NULL, false, force);
> > > +}
> > > +
> > > +static void tps6131x_torch_refresh_handler(struct work_struct *work)
> > > +{
> > > + struct tps6131x *tps6131x = container_of(work, struct tps6131x, torch_refresh_work.work);
> > > +
> > > + guard(mutex)(&tps6131x->lock);
> > > +
> > > + tps6131x_set_mode(tps6131x, TPS6131X_MODE_TORCH, true);
> >
> > This can fail.
>
> That's true. I wasn't sure how to respond to this, other than just keep
> trying.
> What do you think about not starting the timer in case of an error and
> instead outputting a message with dev_err?
Either check the value and/or provide a comment why checking is not required.
> > > +
> > > + schedule_delayed_work(&tps6131x->torch_refresh_work,
> > > + TPS6131X_TORCH_REFRESH_INTERVAL_JIFFIES);
> > > +}
> > > +
> > > +static int tps6131x_brightness_set(struct led_classdev *cdev, enum led_brightness brightness)
> > > +{
> > > + struct led_classdev_flash *fled_cdev = lcdev_to_flcdev(cdev);
> > > + struct tps6131x *tps6131x = fled_cdev_to_tps6131x(fled_cdev);
> >
> > > + u32 num_chans;
> > > + u32 steps_chan13, steps_chan2;
> > > + u32 steps_remaining;
> >
> > Why not group these?
>
> I will do so.
>
> >
> > > + u8 reg0;
> > > + int ret;
> > > +
> > > + cancel_delayed_work_sync(&tps6131x->torch_refresh_work);
> > > +
> > > + /*
> > > + * The brightness parameter uses the number of current steps as the unit (not the current
> > > + * value itself). Since the reported step size can vary depending on the configuration,
> > > + * this value must be converted into actual register steps.
> > > + */
> > > + steps_remaining = (brightness * tps6131x->step_torch_current_ma) / TPS6131X_TORCH_STEP_I_MA;
> > > +
> > > + num_chans = tps6131x->chan1_en + tps6131x->chan2_en + tps6131x->chan3_en;
> > > +
> > > + /*
> > > + * The currents are distributed as evenly as possible across the activated channels.
> > > + * Since channels 1 and 3 share the same register setting, they always use the same current
> > > + * value. Channel 2 supports higher currents and thus takes over the remaining additional
> > > + * portion that cannot be covered by the other channels.
> > > + */
> > > + steps_chan13 = min_t(u32, steps_remaining / num_chans,
> > > + TPS6131X_TORCH_MAX_I_CHAN13_MA / TPS6131X_TORCH_STEP_I_MA);
> > > + if (tps6131x->chan1_en)
> > > + steps_remaining -= steps_chan13;
> > > + if (tps6131x->chan3_en)
> > > + steps_remaining -= steps_chan13;
> > > +
> > > + steps_chan2 = min_t(u32, steps_remaining,
> > > + TPS6131X_TORCH_MAX_I_CHAN2_MA / TPS6131X_TORCH_STEP_I_MA);
> > > +
> > > + guard(mutex)(&tps6131x->lock);
> > > +
> > > + reg0 = (steps_chan13 << TPS6131X_REG_0_DCLC13_SHIFT) |
> > > + (steps_chan2 << TPS6131X_REG_0_DCLC2_SHIFT);
> >
> > Indent.
>
> The indent was intention ;) The operands of arithmetic assignments are at
> least intentionally and automatically aligned to be on the same column. This
> is also the case elsewhere in the code.
> What is the alternative expectation here?
I'd normally expect line feeds to be indented.
This is an odd one because nothing is in front of the open parenthesis.
I'm struggling to find any other examples like it - perhaps it is correct?!
[..]
> > > +static int tps6131x_v4l2_setup(struct tps6131x *tps6131x)
> > > +{
> > > + struct v4l2_flash_config v4l2_cfg = { 0 };
> > > + struct led_flash_setting *intensity = &v4l2_cfg.intensity;
> > > +
> > > + intensity->min = tps6131x->step_torch_current_ma;
> > > + intensity->max = tps6131x->max_torch_current_ma;
> > > + intensity->step = tps6131x->step_torch_current_ma;
> > > + intensity->val = intensity->min;
> > > +
> > > + strscpy(v4l2_cfg.dev_name, tps6131x->fled_cdev.led_cdev.dev->kobj.name,
> > > + sizeof(v4l2_cfg.dev_name));
> > > +
> > > + v4l2_cfg.has_external_strobe = true;
> > > + v4l2_cfg.flash_faults = LED_FAULT_TIMEOUT | LED_FAULT_OVER_TEMPERATURE |
> > > + LED_FAULT_SHORT_CIRCUIT | LED_FAULT_UNDER_VOLTAGE |
> > > + LED_FAULT_LED_OVER_TEMPERATURE;
> > > +
> > > + tps6131x->v4l2_flash = v4l2_flash_init(tps6131x->dev, tps6131x->led_node,
> > > + &tps6131x->fled_cdev, &tps6131x_v4l2_flash_ops,
> > > + &v4l2_cfg);
> > > + if (IS_ERR(tps6131x->v4l2_flash)) {
> > > + dev_err(tps6131x->dev, "Failed to initialize v4l2 flash LED\n");
> > > + return PTR_ERR(tps6131x->v4l2_flash);
> > > + }
> > > +
> > > + return 0;
> > > +}
> > > +
> > > +static int tps6131x_probe(struct i2c_client *client)
> > > +{
> > > + struct tps6131x *tps6131x;
> > > + int ret;
> > > +
> > > + tps6131x = devm_kzalloc(&client->dev, sizeof(*tps6131x), GFP_KERNEL);
> > > + if (!tps6131x)
> > > + return -ENOMEM;
> > > +
> > > + tps6131x->dev = &client->dev;
> > > + i2c_set_clientdata(client, tps6131x);
> >
> > If you already have client, to fetch this, you'll already have access to dev.
>
> I understand that in principle. However, I'm still not entirely sure what
> exactly I should change. Could you please provide me with some further
> guidance?
Yes, don't store 'dev' in 'tps6131x'.
> > > + mutex_init(&tps6131x->lock);
> > > + INIT_DELAYED_WORK(&tps6131x->torch_refresh_work, tps6131x_torch_refresh_handler);
> > > +
> > > + ret = tps6131x_parse_node(tps6131x);
> > > + if (ret)
> > > + return -ENODEV;
> >
> > Why aren't we propagating the real error?
>
> Good point. I'll change that.
>
> >
> > > + tps6131x->regmap = devm_regmap_init_i2c(client, &tps6131x_regmap);
> > > + if (IS_ERR(tps6131x->regmap)) {
> > > + ret = PTR_ERR(tps6131x->regmap);
> >
> > Use dev_err_ptr_probe() instead.
>
> Hmm, I don't quite understand that. dev_err_ptr_probe() returns an error
> pointer instead of an int, right?
Yes, disregard - I had a senior moment!
--
Lee Jones [李琼斯]
Powered by blists - more mailing lists