mirror of
				https://github.com/torvalds/linux.git
				synced 2025-11-04 10:40:15 +02:00 
			
		
		
		
	gpio: Rework of_gpiochip_set_names() to use device property accessors
In order to use "gpio-line-names" property in systems not having DT as their boot firmware, rework of_gpiochip_set_names() to use device property accessors. This reworked function is placed in a separate file making it clear it deals with universal device properties. Signed-off-by: Mika Westerberg <mika.westerberg@linux.intel.com> Signed-off-by: Linus Walleij <linus.walleij@linaro.org>
This commit is contained in:
		
							parent
							
								
									c80f1ba75d
								
							
						
					
					
						commit
						9427ecbed4
					
				
					 4 changed files with 66 additions and 46 deletions
				
			
		| 
						 | 
					@ -5,6 +5,7 @@ ccflags-$(CONFIG_DEBUG_GPIO)	+= -DDEBUG
 | 
				
			||||||
obj-$(CONFIG_GPIO_DEVRES)	+= devres.o
 | 
					obj-$(CONFIG_GPIO_DEVRES)	+= devres.o
 | 
				
			||||||
obj-$(CONFIG_GPIOLIB)		+= gpiolib.o
 | 
					obj-$(CONFIG_GPIOLIB)		+= gpiolib.o
 | 
				
			||||||
obj-$(CONFIG_GPIOLIB)		+= gpiolib-legacy.o
 | 
					obj-$(CONFIG_GPIOLIB)		+= gpiolib-legacy.o
 | 
				
			||||||
 | 
					obj-$(CONFIG_GPIOLIB)		+= gpiolib-devprop.o
 | 
				
			||||||
obj-$(CONFIG_OF_GPIO)		+= gpiolib-of.o
 | 
					obj-$(CONFIG_OF_GPIO)		+= gpiolib-of.o
 | 
				
			||||||
obj-$(CONFIG_GPIO_SYSFS)	+= gpiolib-sysfs.o
 | 
					obj-$(CONFIG_GPIO_SYSFS)	+= gpiolib-sysfs.o
 | 
				
			||||||
obj-$(CONFIG_GPIO_ACPI)		+= gpiolib-acpi.o
 | 
					obj-$(CONFIG_GPIO_ACPI)		+= gpiolib-acpi.o
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
							
								
								
									
										62
									
								
								drivers/gpio/gpiolib-devprop.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										62
									
								
								drivers/gpio/gpiolib-devprop.c
									
									
									
									
									
										Normal file
									
								
							| 
						 | 
					@ -0,0 +1,62 @@
 | 
				
			||||||
 | 
					/*
 | 
				
			||||||
 | 
					 * Device property helpers for GPIO chips.
 | 
				
			||||||
 | 
					 *
 | 
				
			||||||
 | 
					 * Copyright (C) 2016, Intel Corporation
 | 
				
			||||||
 | 
					 * Author: Mika Westerberg <mika.westerberg@linux.intel.com>
 | 
				
			||||||
 | 
					 *
 | 
				
			||||||
 | 
					 * This program is free software; you can redistribute it and/or modify
 | 
				
			||||||
 | 
					 * it under the terms of the GNU General Public License version 2 as
 | 
				
			||||||
 | 
					 * published by the Free Software Foundation.
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#include <linux/property.h>
 | 
				
			||||||
 | 
					#include <linux/slab.h>
 | 
				
			||||||
 | 
					#include <linux/gpio/consumer.h>
 | 
				
			||||||
 | 
					#include <linux/gpio/driver.h>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#include "gpiolib.h"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/**
 | 
				
			||||||
 | 
					 * devprop_gpiochip_set_names - Set GPIO line names using device properties
 | 
				
			||||||
 | 
					 * @chip: GPIO chip whose lines should be named, if possible
 | 
				
			||||||
 | 
					 *
 | 
				
			||||||
 | 
					 * Looks for device property "gpio-line-names" and if it exists assigns
 | 
				
			||||||
 | 
					 * GPIO line names for the chip. The memory allocated for the assigned
 | 
				
			||||||
 | 
					 * names belong to the underlying firmware node and should not be released
 | 
				
			||||||
 | 
					 * by the caller.
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					void devprop_gpiochip_set_names(struct gpio_chip *chip)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
						struct gpio_device *gdev = chip->gpiodev;
 | 
				
			||||||
 | 
						const char **names;
 | 
				
			||||||
 | 
						int ret, i;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						ret = device_property_read_string_array(chip->parent, "gpio-line-names",
 | 
				
			||||||
 | 
											NULL, 0);
 | 
				
			||||||
 | 
						if (ret < 0)
 | 
				
			||||||
 | 
							return;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if (ret != gdev->ngpio) {
 | 
				
			||||||
 | 
							dev_warn(chip->parent,
 | 
				
			||||||
 | 
								 "names %d do not match number of GPIOs %d\n", ret,
 | 
				
			||||||
 | 
								 gdev->ngpio);
 | 
				
			||||||
 | 
							return;
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						names = kcalloc(gdev->ngpio, sizeof(*names), GFP_KERNEL);
 | 
				
			||||||
 | 
						if (!names)
 | 
				
			||||||
 | 
							return;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						ret = device_property_read_string_array(chip->parent, "gpio-line-names",
 | 
				
			||||||
 | 
											names, gdev->ngpio);
 | 
				
			||||||
 | 
						if (ret < 0) {
 | 
				
			||||||
 | 
							dev_warn(chip->parent, "failed to read GPIO line names\n");
 | 
				
			||||||
 | 
							kfree(names);
 | 
				
			||||||
 | 
							return;
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						for (i = 0; i < gdev->ngpio; i++)
 | 
				
			||||||
 | 
							gdev->descs[i].name = names[i];
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						kfree(names);
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
| 
						 | 
					@ -221,51 +221,6 @@ static struct gpio_desc *of_parse_own_gpio(struct device_node *np,
 | 
				
			||||||
	return desc;
 | 
						return desc;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
/**
 | 
					 | 
				
			||||||
 * of_gpiochip_set_names() - set up the names of the lines
 | 
					 | 
				
			||||||
 * @chip: GPIO chip whose lines should be named, if possible
 | 
					 | 
				
			||||||
 */
 | 
					 | 
				
			||||||
static void of_gpiochip_set_names(struct gpio_chip *gc)
 | 
					 | 
				
			||||||
{
 | 
					 | 
				
			||||||
	struct gpio_device *gdev = gc->gpiodev;
 | 
					 | 
				
			||||||
	struct device_node *np = gc->of_node;
 | 
					 | 
				
			||||||
	int i;
 | 
					 | 
				
			||||||
	int nstrings;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	nstrings = of_property_count_strings(np, "gpio-line-names");
 | 
					 | 
				
			||||||
	if (nstrings <= 0)
 | 
					 | 
				
			||||||
		/* Lines names not present */
 | 
					 | 
				
			||||||
		return;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	/* This is normally not what you want */
 | 
					 | 
				
			||||||
	if (gdev->ngpio != nstrings)
 | 
					 | 
				
			||||||
		dev_info(&gdev->dev, "gpio-line-names specifies %d line "
 | 
					 | 
				
			||||||
			 "names but there are %d lines on the chip\n",
 | 
					 | 
				
			||||||
			 nstrings, gdev->ngpio);
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	/*
 | 
					 | 
				
			||||||
	 * Make sure to not index beyond the end of the number of descriptors
 | 
					 | 
				
			||||||
	 * of the GPIO device.
 | 
					 | 
				
			||||||
	 */
 | 
					 | 
				
			||||||
	for (i = 0; i < gdev->ngpio; i++) {
 | 
					 | 
				
			||||||
		const char *name;
 | 
					 | 
				
			||||||
		int ret;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
		ret = of_property_read_string_index(np,
 | 
					 | 
				
			||||||
						    "gpio-line-names",
 | 
					 | 
				
			||||||
						    i,
 | 
					 | 
				
			||||||
						    &name);
 | 
					 | 
				
			||||||
		if (ret) {
 | 
					 | 
				
			||||||
			if (ret != -ENODATA)
 | 
					 | 
				
			||||||
                                dev_err(&gdev->dev,
 | 
					 | 
				
			||||||
                                        "unable to name line %d: %d\n",
 | 
					 | 
				
			||||||
                                        i, ret);
 | 
					 | 
				
			||||||
			break;
 | 
					 | 
				
			||||||
		}
 | 
					 | 
				
			||||||
		gdev->descs[i].name = name;
 | 
					 | 
				
			||||||
	}
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
/**
 | 
					/**
 | 
				
			||||||
 * of_gpiochip_scan_gpios - Scan gpio-controller for gpio definitions
 | 
					 * of_gpiochip_scan_gpios - Scan gpio-controller for gpio definitions
 | 
				
			||||||
 * @chip:	gpio chip to act on
 | 
					 * @chip:	gpio chip to act on
 | 
				
			||||||
| 
						 | 
					@ -522,7 +477,7 @@ int of_gpiochip_add(struct gpio_chip *chip)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	/* If the chip defines names itself, these take precedence */
 | 
						/* If the chip defines names itself, these take precedence */
 | 
				
			||||||
	if (!chip->names)
 | 
						if (!chip->names)
 | 
				
			||||||
		of_gpiochip_set_names(chip);
 | 
							devprop_gpiochip_set_names(chip);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	of_node_get(chip->of_node);
 | 
						of_node_get(chip->of_node);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -209,6 +209,8 @@ static int __maybe_unused gpio_chip_hwgpio(const struct gpio_desc *desc)
 | 
				
			||||||
	return desc - &desc->gdev->descs[0];
 | 
						return desc - &desc->gdev->descs[0];
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					void devprop_gpiochip_set_names(struct gpio_chip *chip);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
/* With descriptor prefix */
 | 
					/* With descriptor prefix */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
#define gpiod_emerg(desc, fmt, ...)					       \
 | 
					#define gpiod_emerg(desc, fmt, ...)					       \
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
		Loading…
	
		Reference in a new issue