[PATCH v5 07/14] mfd: Add driver for Maxim 77802 Power Management IC

Javier Martinez Canillas javier.martinez at collabora.co.uk
Tue Jul 1 08:55:47 PDT 2014


Hello Lee,

Thanks a lot for your feedback.

On 07/01/2014 05:15 PM, Lee Jones wrote:
> On Thu, 26 Jun 2014, Javier Martinez Canillas wrote:
> 
>> Maxim MAX77802 is a power management chip that contains 10 high
>> efficiency Buck regulators, 32 Low-dropout (LDO) regulators used
>> to power up application processors and peripherals, a 2-channel
>> 32kHz clock outputs, a Real-Time-Clock (RTC) and a I2C interface
>> to program the individual regulators, clocks outputs and the RTC.
>> 
>> This patch adds the core support for MAX77802 PMIC and is based
>> on a driver added by Simon Glass to the Chrome OS kernel 3.8 tree.
>> 
>> Signed-off-by: Javier Martinez Canillas <javier.martinez at collabora.co.uk>
>> Reviewed-by: Krzysztof Kozlowski <k.kozlowski at samsung.com>
>> Tested-by: Naveen Krishna Chatradhi <ch.naveen at samsung.com>
>> ---
>> 
>> Changes since v4:
>>  - Use consistent expressions when checking for NULL values.
>>    Suggested by Krzysztof Kozlowski.
>>  - Remove unused defines. Suggested by Krzysztof Kozlowski.
>>  - Explain why IRQ is disabled on suspend. Suggested by Krzysztof Kozlowski.
>> 
>> Changes since v3:
>>  - Remove unnecessary OOM error message since the mm subsystem already logs it. 
>> 
>> Changes since v2:
>>  - Split the DT binding docs in a separate patch and improve the documentation.
>>    Suggested by Mark Brown.
>>  - Add all the devices in the MFD driver instead of doing in separate patches.
>>    Suggested by Mark Brown.
>> 
>> Changes since v1:
>>  - Convert max77{686,802} to regmap irq API and get rid of max77{686,802}-irq.c
>>    Suggested by Krzysztof Kozlowski.
>>  - Don't protect max77802 mfd_cells using Kconfig options since mfd core omits
>>    devices that don't match. Suggested by Lee Jones.
>>  - Change mfd driver to be tristate instead of boolean. Suggested by Mark Brown.
>>  - Change binding "voltage-regulators" property to "regulators" to be consistent
>>    with other PMIC drivers. Suggested by Mark Brown.
>>  - Use regulators node names instead of the deprecated "regulator-compatible"
>>    property. Suggested by Mark Brown.
>>  - Use the new descriptor-based GPIO interface instead of the deprecated
>>    integer based GPIO one. Suggested by Mark Brown.
>>  - Remove the type parameter from i2c_device_id table since was not used.
>>  - Fix device not found error message and remove unneeded device found message.
>> 
>>  drivers/mfd/Kconfig                  |  14 ++
>>  drivers/mfd/Makefile                 |   1 +
>>  drivers/mfd/max77802.c               | 366 ++++++++++++++++++++++++++++++++++
> 

> I don't think this needs it's own, brand new file.  You can just as
> easy slot the enablement into max77686, which I suggest you do.
>

Since the series convert max77686 to use regmap IRQ and gets rid of
max77686-irq.c, the mfd max77{802,686} drivers are quite small now and are
composed mostly of data structures definitions (regmap_config and accessors
function handlers, regmap_irq, i2c_driver, etc) and the probe function.

So I thought that it was more maintainable and readable to keep them in separate
files than having lots of conditionals in the max77686 driver.

But since you think that is better to extend the max77686, I'll do it on the
next version of the patch-set. I'll try to keep it as clean as possible.

>>  include/linux/mfd/max77802-private.h | 307 +++++++++++++++++++++++++++++
>>  include/linux/mfd/max77802.h         | 121 ++++++++++++
>>  5 files changed, 809 insertions(+)
>>  create mode 100644 drivers/mfd/max77802.c
>>  create mode 100644 include/linux/mfd/max77802-private.h
>>  create mode 100644 include/linux/mfd/max77802.h
> 
> [...]
> 
>> +#ifdef CONFIG_OF
>> +static struct of_device_id max77802_pmic_dt_match[] = {
>> +	{.compatible = "maxim,max77802", .data = NULL},
> 
> Space before .compatible.
>

Ok.

> No need to set .data to NULL, just omit it.
> 

Ok, I'll have to use .data though now that I'll extend max77686 to support
max77802 too.

>> +	{},
>> +};
>> +
>> +static void max77802_dt_parse_dvs_gpio(struct device *dev,
>> +				       struct max77802_platform_data *pd,
>> +				       struct device_node *np)
> 
> No need to overload these parameters by passing np, you can extract it
> from dev.  Same goes for pd if you populate platform_data _before_
> calling this function.
> 

Right, I'll change that and just get it from dev.

>> +{
>> +	int i;
>> +
>> +	/*
>> +	 * NOTE: we don't consider GPIO errors fatal; board may have some lines
>> +	 * directly pulled high or low and thus doesn't specify them.
>> +	 */
>> +	for (i = 0; i < ARRAY_SIZE(pd->buck_gpio_dvs); i++)
>> +		pd->buck_gpio_dvs[i] =
>> +			devm_gpiod_get_index(dev, "max77802,pmic-buck-dvs", i);
>> +
>> +	for (i = 0; i < ARRAY_SIZE(pd->buck_gpio_selb); i++)
>> +		pd->buck_gpio_selb[i] =
>> +			devm_gpiod_get_index(dev, "max77802,pmic-buck-selb", i);
>> +}
>> +
>> +static struct max77802_platform_data *max77802_i2c_parse_dt_pdata(struct device
>> +								  *dev)
>> +{
>> +	struct device_node *np = dev->of_node;
>> +	struct max77802_platform_data *pd;
>> +
>> +	pd = devm_kzalloc(dev, sizeof(*pd), GFP_KERNEL);
>> +	if (!pd)
>> +		return NULL;
>> +
>> +	/* Read default index and ignore errors, since default is 0 */
>> +	of_property_read_u32(np, "max77802,pmic-buck-default-dvs-idx",
>> +			     &pd->buck_default_idx);
>> +
>> +	max77802_dt_parse_dvs_gpio(dev, pd, np);
>> +
>> +	dev->platform_data = pd;
>> +	return pd;
>> +}
>> +#else
>> +static struct max77802_platform_data *max77802_i2c_parse_dt_pdata(struct device
>> +								  *dev)
>> +{
>> +	return 0;
>> +}
>> +#endif
> 
> No need for dummy functions.
> 
> Use if (IS_ENABLED(CONFIG_OF) && i2c->dev.of_node) instead.
>

Ok.

>> +static int max77802_i2c_probe(struct i2c_client *i2c,
>> +			      const struct i2c_device_id *id)
>> +{
>> +	struct max77802_dev *max77802 = NULL;
>> +	struct max77802_platform_data *pdata = dev_get_platdata(&i2c->dev);
>> +	unsigned int data;
>> +	int ret = 0;
>> +
>> +	if (i2c->dev.of_node)
>> +		pdata = max77802_i2c_parse_dt_pdata(&i2c->dev);
> 
> DT should not over-rule platform data.  If the driver supports pdata
> and it's present, it should over-rule DT.
> 

Ok will change that.

>> +	if (!pdata) {
>> +		dev_err(&i2c->dev, "No platform data found.\n");
>> +		return -EIO;
> 
> -EIO is not the correct error code here.  Either -EINVAL or -ENODEV
> would be better.
> 

Right, I was looking at other drivers and most return -EINVAL indeed if platform
data is not available so I'll use that error code as well.

>> +	}
>> +
>> +	max77802 = devm_kzalloc(&i2c->dev, sizeof(struct max77802_dev),
>> +				GFP_KERNEL);
>> +	if (max77802 == NULL)
> 
> if (!max77802)
> 

Yes, Krzysztof suggested me the same for consistency and I've already changed.

>> +		return -ENOMEM;
>> +
>> +	i2c_set_clientdata(i2c, max77802);
>> +
>> +	max77802->dev = &i2c->dev;
>> +	max77802->i2c = i2c;
>> +	max77802->type = id->driver_data;
>> +
>> +	max77802->wakeup = pdata->wakeup;
>> +	max77802->irq = i2c->irq;
>> +
>> +	max77802->regmap = devm_regmap_init_i2c(i2c, &max77802_regmap_config);
>> +	if (IS_ERR(max77802->regmap)) {
>> +		ret = PTR_ERR(max77802->regmap);
>> +		dev_err(max77802->dev, "Failed to allocate register map: %d\n",
>> +			ret);
>> +		return ret;
>> +	}
>> +
>> +	if (regmap_read(max77802->regmap,
>> +			 MAX77802_REG_DEVICE_ID, &data) < 0) {
>> +		dev_err(max77802->dev,
>> +			"device not found on this channel\n");
>> +		return -ENODEV;
>> +	}
> 
> The return values for all other calls are first placed into a variable
> and _then_ evaluated.  For consistency can you do the same here
> please?
> 

Sure.

>> +	ret = regmap_add_irq_chip(max77802->regmap, max77802->irq,
>> +				  IRQF_TRIGGER_FALLING | IRQF_ONESHOT |
>> +				  IRQF_SHARED, 0, &max77802_irq_chip,
>> +				  &max77802->irq_data);
>> +	if (ret != 0) {
> 
> if (ret)
> 

Ok.

>> +		dev_err(&i2c->dev, "failed to add PMIC irq chip: %d\n", ret);
>> +		return ret;
>> +	}
> 
> '\n'
> 
> [...]
> 
>> +static struct i2c_driver max77802_i2c_driver = {
>> +	.driver = {
>> +		   .name = "max77802",
>> +		   .owner = THIS_MODULE,
>> +		   .pm = &max77802_pm,
>> +		   .of_match_table = of_match_ptr(max77802_pmic_dt_match),
> 
> This tabbing is off.
> 

Right, I'll fix it.

>> +	},
>> +	.probe = max77802_i2c_probe,
>> +	.remove = max77802_i2c_remove,
>> +	.id_table = max77802_i2c_id,
>> +};
>> +
>> +static int __init max77802_i2c_init(void)
>> +{
>> +	return i2c_add_driver(&max77802_i2c_driver);
>> +}
>> +/* init early so consumer devices can complete system boot */
>> +subsys_initcall(max77802_i2c_init);
>> +
>> +static void __exit max77802_i2c_exit(void)
>> +{
>> +	i2c_del_driver(&max77802_i2c_driver);
>> +}
>> +module_exit(max77802_i2c_exit);
> 

Best regards,
Javier



More information about the linux-arm-kernel mailing list