mirror of
				https://github.com/torvalds/linux.git
				synced 2025-11-04 10:40:15 +02:00 
			
		
		
		
	pinctrl: API changes to support multiple states per device
The API model is changed from: p = pinctrl_get(dev, "state1"); pinctrl_enable(p); ... pinctrl_disable(p); pinctrl_put(p); p = pinctrl_get(dev, "state2"); pinctrl_enable(p); ... pinctrl_disable(p); pinctrl_put(p); to this: p = pinctrl_get(dev); s1 = pinctrl_lookup_state(p, "state1"); s2 = pinctrl_lookup_state(p, "state2"); pinctrl_select_state(p, s1); ... pinctrl_select_state(p, s2); ... pinctrl_put(p); This allows devices to directly transition between states without disabling the pin controller programming and put()/get()ing the configuration data each time. This model will also better suit pinconf programming, which doesn't have a concept of "disable". The special-case hogging feature of pin controllers is re-written to use the regular APIs instead of special-case code. Hence, the pinmux-hogs debugfs file is removed; see the top-level pinctrl-handles files for equivalent data. Signed-off-by: Stephen Warren <swarren@nvidia.com> Acked-by: Dong Aisheng <dong.aisheng@linaro.org> Signed-off-by: Linus Walleij <linus.walleij@linaro.org>
This commit is contained in:
		
							parent
							
								
									0e3db173e2
								
							
						
					
					
						commit
						6e5e959dde
					
				
					 7 changed files with 379 additions and 229 deletions
				
			
		| 
						 | 
				
			
			@ -847,8 +847,8 @@ As it is possible to map a function to different groups of pins an optional
 | 
			
		|||
This example mapping is used to switch between two positions for spi0 at
 | 
			
		||||
runtime, as described further below under the heading "Runtime pinmuxing".
 | 
			
		||||
 | 
			
		||||
Further it is possible to match several groups of pins to the same function
 | 
			
		||||
for a single device, say for example in the mmc0 example above, where you can
 | 
			
		||||
Further it is possible for one named state to affect the muxing of several
 | 
			
		||||
groups of pins, say for example in the mmc0 example above, where you can
 | 
			
		||||
additively expand the mmc0 bus from 2 to 4 to 8 pins. If we want to use all
 | 
			
		||||
three groups for a total of 2+2+4 = 8 pins (for an 8-bit MMC bus as is the
 | 
			
		||||
case), we define a mapping like this:
 | 
			
		||||
| 
						 | 
				
			
			@ -879,6 +879,7 @@ case), we define a mapping like this:
 | 
			
		|||
	.dev_name = "foo-mmc.0",
 | 
			
		||||
	.name = "8bit"
 | 
			
		||||
	.ctrl_dev_name = "pinctrl-foo",
 | 
			
		||||
	.function = "mmc0",
 | 
			
		||||
	.group = "mmc0_1_grp",
 | 
			
		||||
},
 | 
			
		||||
{
 | 
			
		||||
| 
						 | 
				
			
			@ -900,10 +901,16 @@ case), we define a mapping like this:
 | 
			
		|||
The result of grabbing this mapping from the device with something like
 | 
			
		||||
this (see next paragraph):
 | 
			
		||||
 | 
			
		||||
	p = pinctrl_get(&device, "8bit");
 | 
			
		||||
	p = pinctrl_get(dev);
 | 
			
		||||
	s = pinctrl_lookup_state(p, "8bit");
 | 
			
		||||
	ret = pinctrl_select_state(p, s);
 | 
			
		||||
 | 
			
		||||
or more simply:
 | 
			
		||||
 | 
			
		||||
	p = pinctrl_get_select(dev, "8bit");
 | 
			
		||||
 | 
			
		||||
Will be that you activate all the three bottom records in the mapping at
 | 
			
		||||
once. Since they share the same name, pin controller device, funcion and
 | 
			
		||||
once. Since they share the same name, pin controller device, function and
 | 
			
		||||
device, and since we allow multiple groups to match to a single device, they
 | 
			
		||||
all get selected, and they all get enabled and disable simultaneously by the
 | 
			
		||||
pinmux core.
 | 
			
		||||
| 
						 | 
				
			
			@ -925,45 +932,63 @@ default state like this:
 | 
			
		|||
 | 
			
		||||
struct foo_state {
 | 
			
		||||
       struct pinctrl *p;
 | 
			
		||||
       struct pinctrl_state *s;
 | 
			
		||||
       ...
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
foo_probe()
 | 
			
		||||
{
 | 
			
		||||
	/* Allocate a state holder named "state" etc */
 | 
			
		||||
	struct pinctrl p;
 | 
			
		||||
	/* Allocate a state holder named "foo" etc */
 | 
			
		||||
	struct foo_state *foo = ...;
 | 
			
		||||
 | 
			
		||||
	p = pinctrl_get(&device, PINCTRL_STATE_DEFAULT);
 | 
			
		||||
	if IS_ERR(p)
 | 
			
		||||
		return PTR_ERR(p);
 | 
			
		||||
	pinctrl_enable(p);
 | 
			
		||||
	foo->p = pinctrl_get(&device);
 | 
			
		||||
	if (IS_ERR(foo->p)) {
 | 
			
		||||
		/* FIXME: clean up "foo" here */
 | 
			
		||||
		return PTR_ERR(foo->p);
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	state->p = p;
 | 
			
		||||
	foo->s = pinctrl_lookup_state(foo->p, PINCTRL_STATE_DEFAULT);
 | 
			
		||||
	if (IS_ERR(foo->s)) {
 | 
			
		||||
		pinctrl_put(foo->p);
 | 
			
		||||
		/* FIXME: clean up "foo" here */
 | 
			
		||||
		return PTR_ERR(s);
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	ret = pinctrl_select_state(foo->s);
 | 
			
		||||
	if (ret < 0) {
 | 
			
		||||
		pinctrl_put(foo->p);
 | 
			
		||||
		/* FIXME: clean up "foo" here */
 | 
			
		||||
		return ret;
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
foo_remove()
 | 
			
		||||
{
 | 
			
		||||
	pinctrl_disable(state->p);
 | 
			
		||||
	pinctrl_put(state->p);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
This get/enable/disable/put sequence can just as well be handled by bus drivers
 | 
			
		||||
This get/lookup/select/put sequence can just as well be handled by bus drivers
 | 
			
		||||
if you don't want each and every driver to handle it and you know the
 | 
			
		||||
arrangement on your bus.
 | 
			
		||||
 | 
			
		||||
The semantics of the get/enable respective disable/put is as follows:
 | 
			
		||||
The semantics of the pinctrl APIs are:
 | 
			
		||||
 | 
			
		||||
- pinctrl_get() is called in process context to reserve the pins affected with
 | 
			
		||||
  a certain mapping and set up the pinmux core and the driver. It will allocate
 | 
			
		||||
  a struct from the kernel memory to hold the pinmux state.
 | 
			
		||||
- pinctrl_get() is called in process context to obtain a handle to all pinctrl
 | 
			
		||||
  information for a given client device. It will allocate a struct from the
 | 
			
		||||
  kernel memory to hold the pinmux state. All mapping table parsing or similar
 | 
			
		||||
  slow operations take place within this API.
 | 
			
		||||
 | 
			
		||||
- pinctrl_enable()/pinctrl_disable() is quick and can be called from fastpath
 | 
			
		||||
  (irq context) when you quickly want to set up/tear down the hardware muxing
 | 
			
		||||
  when running a device driver. Usually it will just poke some values into a
 | 
			
		||||
  register.
 | 
			
		||||
- pinctrl_lookup_state() is called in process context to obtain a handle to a
 | 
			
		||||
  specific state for a the client device. This operation may be slow too.
 | 
			
		||||
 | 
			
		||||
- pinctrl_disable() is called in process context to tear down the pin requests
 | 
			
		||||
  and release the state holder struct for the mux setting etc.
 | 
			
		||||
- pinctrl_select_state() programs pin controller hardware according to the
 | 
			
		||||
  definition of the state as given by the mapping table. In theory this is a
 | 
			
		||||
  fast-path operation, since it only involved blasting some register settings
 | 
			
		||||
  into hardware. However, note that some pin controllers may have their
 | 
			
		||||
  registers on a slow/IRQ-based bus, so client devices should not assume they
 | 
			
		||||
  can call pinctrl_select_state() from non-blocking contexts.
 | 
			
		||||
 | 
			
		||||
- pinctrl_put() frees all information associated with a pinctrl handle.
 | 
			
		||||
 | 
			
		||||
Usually the pin control core handled the get/put pair and call out to the
 | 
			
		||||
device drivers bookkeeping operations, like checking available functions and
 | 
			
		||||
| 
						 | 
				
			
			@ -979,12 +1004,12 @@ System pin control hogging
 | 
			
		|||
==========================
 | 
			
		||||
 | 
			
		||||
Pin control map entries can be hogged by the core when the pin controller
 | 
			
		||||
is registered. This means that the core will attempt to call pinctrl_get() and
 | 
			
		||||
pinctrl_enable() on it immediately after the pin control device has been
 | 
			
		||||
registered.
 | 
			
		||||
is registered. This means that the core will attempt to call pinctrl_get(),
 | 
			
		||||
lookup_state() and select_state() on it immediately after the pin control
 | 
			
		||||
device has been registered.
 | 
			
		||||
 | 
			
		||||
This is enabled by simply setting the .dev_name field in the map to the name
 | 
			
		||||
of the pin controller itself, like this:
 | 
			
		||||
This occurs for mapping table entries where the client device name is equal
 | 
			
		||||
to the pin controller device name, and the state name is PINCTRL_STATE_DEFAULT.
 | 
			
		||||
 | 
			
		||||
{
 | 
			
		||||
	.dev_name = "pinctrl-foo",
 | 
			
		||||
| 
						 | 
				
			
			@ -1009,8 +1034,8 @@ It is possible to mux a certain function in and out at runtime, say to move
 | 
			
		|||
an SPI port from one set of pins to another set of pins. Say for example for
 | 
			
		||||
spi0 in the example above, we expose two different groups of pins for the same
 | 
			
		||||
function, but with different named in the mapping as described under
 | 
			
		||||
"Advanced mapping" above. So we have two mappings named "spi0-pos-A" and
 | 
			
		||||
"spi0-pos-B".
 | 
			
		||||
"Advanced mapping" above. So that for an SPI device, we have two states named
 | 
			
		||||
"pos-A" and "pos-B".
 | 
			
		||||
 | 
			
		||||
This snippet first muxes the function in the pins defined by group A, enables
 | 
			
		||||
it, disables and releases it, and muxes it in on the pins defined by group B:
 | 
			
		||||
| 
						 | 
				
			
			@ -1020,23 +1045,36 @@ it, disables and releases it, and muxes it in on the pins defined by group B:
 | 
			
		|||
foo_switch()
 | 
			
		||||
{
 | 
			
		||||
	struct pinctrl *p;
 | 
			
		||||
	struct pinctrl_state *s1, *s2;
 | 
			
		||||
 | 
			
		||||
	/* Setup */
 | 
			
		||||
	p = pinctrl_get(&device);
 | 
			
		||||
	if (IS_ERR(p))
 | 
			
		||||
		...
 | 
			
		||||
 | 
			
		||||
	s1 = pinctrl_lookup_state(foo->p, "pos-A");
 | 
			
		||||
	if (IS_ERR(s1))
 | 
			
		||||
		...
 | 
			
		||||
 | 
			
		||||
	s2 = pinctrl_lookup_state(foo->p, "pos-B");
 | 
			
		||||
	if (IS_ERR(s2))
 | 
			
		||||
		...
 | 
			
		||||
 | 
			
		||||
	/* Enable on position A */
 | 
			
		||||
	p = pinctrl_get(&device, "spi0-pos-A");
 | 
			
		||||
	if IS_ERR(p)
 | 
			
		||||
		return PTR_ERR(p);
 | 
			
		||||
	pinctrl_enable(p);
 | 
			
		||||
	ret = pinctrl_select_state(s1);
 | 
			
		||||
	if (ret < 0)
 | 
			
		||||
	    ...
 | 
			
		||||
 | 
			
		||||
	/* This releases the pins again */
 | 
			
		||||
	pinctrl_disable(p);
 | 
			
		||||
	pinctrl_put(p);
 | 
			
		||||
	...
 | 
			
		||||
 | 
			
		||||
	/* Enable on position B */
 | 
			
		||||
	p = pinctrl_get(&device, "spi0-pos-B");
 | 
			
		||||
	if IS_ERR(p)
 | 
			
		||||
		return PTR_ERR(p);
 | 
			
		||||
	pinctrl_enable(p);
 | 
			
		||||
	ret = pinctrl_select_state(s2);
 | 
			
		||||
	if (ret < 0)
 | 
			
		||||
	    ...
 | 
			
		||||
 | 
			
		||||
	...
 | 
			
		||||
 | 
			
		||||
	pinctrl_put(p);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
The above has to be done from process context.
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -1618,22 +1618,18 @@ static struct pinctrl_map __initdata u300_pinmux_map[] = {
 | 
			
		|||
};
 | 
			
		||||
 | 
			
		||||
struct u300_mux_hog {
 | 
			
		||||
	const char *name;
 | 
			
		||||
	struct device *dev;
 | 
			
		||||
	struct pinctrl *p;
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
static struct u300_mux_hog u300_mux_hogs[] = {
 | 
			
		||||
	{
 | 
			
		||||
		.name = "uart0",
 | 
			
		||||
		.dev = &uart0_device.dev,
 | 
			
		||||
	},
 | 
			
		||||
	{
 | 
			
		||||
		.name = "spi0",
 | 
			
		||||
		.dev = &pl022_device.dev,
 | 
			
		||||
	},
 | 
			
		||||
	{
 | 
			
		||||
		.name = "mmc0",
 | 
			
		||||
		.dev = &mmcsd_device.dev,
 | 
			
		||||
	},
 | 
			
		||||
};
 | 
			
		||||
| 
						 | 
				
			
			@ -1646,16 +1642,10 @@ static int __init u300_pinctrl_fetch(void)
 | 
			
		|||
		struct pinctrl *p;
 | 
			
		||||
		int ret;
 | 
			
		||||
 | 
			
		||||
		p = pinctrl_get(u300_mux_hogs[i].dev, PINCTRL_STATE_DEFAULT);
 | 
			
		||||
		p = pinctrl_get_select_default(u300_mux_hogs[i].dev);
 | 
			
		||||
		if (IS_ERR(p)) {
 | 
			
		||||
			pr_err("u300: could not get pinmux hog %s\n",
 | 
			
		||||
			       u300_mux_hogs[i].name);
 | 
			
		||||
			continue;
 | 
			
		||||
		}
 | 
			
		||||
		ret = pinctrl_enable(p);
 | 
			
		||||
		if (ret) {
 | 
			
		||||
			pr_err("u300: could enable pinmux hog %s\n",
 | 
			
		||||
			       u300_mux_hogs[i].name);
 | 
			
		||||
			pr_err("u300: could not get pinmux hog for dev %s\n",
 | 
			
		||||
			       dev_name(u300_mux_hogs[i].dev));
 | 
			
		||||
			continue;
 | 
			
		||||
		}
 | 
			
		||||
		u300_mux_hogs[i].p = p;
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -458,25 +458,98 @@ int pinctrl_gpio_direction_output(unsigned gpio)
 | 
			
		|||
}
 | 
			
		||||
EXPORT_SYMBOL_GPL(pinctrl_gpio_direction_output);
 | 
			
		||||
 | 
			
		||||
static struct pinctrl *pinctrl_get_locked(struct device *dev, const char *name)
 | 
			
		||||
static struct pinctrl_state *find_state(struct pinctrl *p,
 | 
			
		||||
					const char *name)
 | 
			
		||||
{
 | 
			
		||||
	struct pinctrl_dev *pctldev;
 | 
			
		||||
	const char *devname;
 | 
			
		||||
	struct pinctrl *p;
 | 
			
		||||
	unsigned num_maps = 0;
 | 
			
		||||
	struct pinctrl_state *state;
 | 
			
		||||
 | 
			
		||||
	list_for_each_entry(state, &p->states, node)
 | 
			
		||||
		if (!strcmp(state->name, name))
 | 
			
		||||
			return state;
 | 
			
		||||
 | 
			
		||||
	return NULL;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static struct pinctrl_state *create_state(struct pinctrl *p,
 | 
			
		||||
					  const char *name)
 | 
			
		||||
{
 | 
			
		||||
	struct pinctrl_state *state;
 | 
			
		||||
 | 
			
		||||
	state = kzalloc(sizeof(*state), GFP_KERNEL);
 | 
			
		||||
	if (state == NULL) {
 | 
			
		||||
		dev_err(p->dev,
 | 
			
		||||
			"failed to alloc struct pinctrl_state\n");
 | 
			
		||||
		return ERR_PTR(-ENOMEM);
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	state->name = name;
 | 
			
		||||
	INIT_LIST_HEAD(&state->settings);
 | 
			
		||||
 | 
			
		||||
	list_add_tail(&state->node, &p->states);
 | 
			
		||||
 | 
			
		||||
	return state;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static int add_setting(struct pinctrl *p, struct pinctrl_map const *map)
 | 
			
		||||
{
 | 
			
		||||
	struct pinctrl_state *state;
 | 
			
		||||
	struct pinctrl_setting *setting;
 | 
			
		||||
	int ret;
 | 
			
		||||
 | 
			
		||||
	state = find_state(p, map->name);
 | 
			
		||||
	if (!state)
 | 
			
		||||
		state = create_state(p, map->name);
 | 
			
		||||
	if (IS_ERR(state))
 | 
			
		||||
		return PTR_ERR(state);
 | 
			
		||||
 | 
			
		||||
	setting = kzalloc(sizeof(*setting), GFP_KERNEL);
 | 
			
		||||
	if (setting == NULL) {
 | 
			
		||||
		dev_err(p->dev,
 | 
			
		||||
			"failed to alloc struct pinctrl_setting\n");
 | 
			
		||||
		return -ENOMEM;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	setting->pctldev = get_pinctrl_dev_from_devname(map->ctrl_dev_name);
 | 
			
		||||
	if (setting->pctldev == NULL) {
 | 
			
		||||
		dev_err(p->dev, "unknown pinctrl device %s in map entry",
 | 
			
		||||
			map->ctrl_dev_name);
 | 
			
		||||
		kfree(setting);
 | 
			
		||||
		/* Eventually, this should trigger deferred probe */
 | 
			
		||||
		return -ENODEV;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	ret = pinmux_map_to_setting(map, setting);
 | 
			
		||||
	if (ret < 0) {
 | 
			
		||||
		kfree(setting);
 | 
			
		||||
		return ret;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	list_add_tail(&setting->node, &state->settings);
 | 
			
		||||
 | 
			
		||||
	return 0;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static struct pinctrl *find_pinctrl(struct device *dev)
 | 
			
		||||
{
 | 
			
		||||
	struct pinctrl *p;
 | 
			
		||||
 | 
			
		||||
	list_for_each_entry(p, &pinctrldev_list, node)
 | 
			
		||||
		if (p->dev == dev)
 | 
			
		||||
			return p;
 | 
			
		||||
 | 
			
		||||
	return NULL;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void pinctrl_put_locked(struct pinctrl *p, bool inlist);
 | 
			
		||||
 | 
			
		||||
static struct pinctrl *create_pinctrl(struct device *dev)
 | 
			
		||||
{
 | 
			
		||||
	struct pinctrl *p;
 | 
			
		||||
	const char *devname;
 | 
			
		||||
	struct pinctrl_maps *maps_node;
 | 
			
		||||
	int i;
 | 
			
		||||
	struct pinctrl_map const *map;
 | 
			
		||||
	struct pinctrl_setting *setting;
 | 
			
		||||
 | 
			
		||||
	/* We must have both a dev and state name */
 | 
			
		||||
	if (WARN_ON(!dev || !name))
 | 
			
		||||
		return ERR_PTR(-EINVAL);
 | 
			
		||||
 | 
			
		||||
	devname = dev_name(dev);
 | 
			
		||||
 | 
			
		||||
	dev_dbg(dev, "pinctrl_get() for device %s state %s\n", devname, name);
 | 
			
		||||
	int ret;
 | 
			
		||||
 | 
			
		||||
	/*
 | 
			
		||||
	 * create the state cookie holder struct pinctrl for each
 | 
			
		||||
| 
						 | 
				
			
			@ -489,8 +562,9 @@ static struct pinctrl *pinctrl_get_locked(struct device *dev, const char *name)
 | 
			
		|||
		return ERR_PTR(-ENOMEM);
 | 
			
		||||
	}
 | 
			
		||||
	p->dev = dev;
 | 
			
		||||
	p->state = name;
 | 
			
		||||
	INIT_LIST_HEAD(&p->settings);
 | 
			
		||||
	INIT_LIST_HEAD(&p->states);
 | 
			
		||||
 | 
			
		||||
	devname = dev_name(dev);
 | 
			
		||||
 | 
			
		||||
	/* Iterate over the pin control maps to locate the right ones */
 | 
			
		||||
	for_each_maps(maps_node, i, map) {
 | 
			
		||||
| 
						 | 
				
			
			@ -498,139 +572,157 @@ static struct pinctrl *pinctrl_get_locked(struct device *dev, const char *name)
 | 
			
		|||
		if (strcmp(map->dev_name, devname))
 | 
			
		||||
			continue;
 | 
			
		||||
 | 
			
		||||
		/* State name must be the one we're looking for */
 | 
			
		||||
		if (strcmp(map->name, name))
 | 
			
		||||
			continue;
 | 
			
		||||
 | 
			
		||||
		/*
 | 
			
		||||
		 * Try to find the pctldev given in the map
 | 
			
		||||
		 */
 | 
			
		||||
		pctldev = get_pinctrl_dev_from_devname(map->ctrl_dev_name);
 | 
			
		||||
		if (!pctldev) {
 | 
			
		||||
			dev_err(dev, "unknown pinctrl device %s in map entry",
 | 
			
		||||
				map->ctrl_dev_name);
 | 
			
		||||
			/* Eventually, this should trigger deferred probe */
 | 
			
		||||
			ret = -ENODEV;
 | 
			
		||||
			goto error;
 | 
			
		||||
		ret = add_setting(p, map);
 | 
			
		||||
		if (ret < 0) {
 | 
			
		||||
			pinctrl_put_locked(p, false);
 | 
			
		||||
			return ERR_PTR(ret);
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		dev_dbg(dev, "in map, found pctldev %s to handle function %s",
 | 
			
		||||
			dev_name(pctldev->dev), map->function);
 | 
			
		||||
 | 
			
		||||
		setting = kzalloc(sizeof(*setting), GFP_KERNEL);
 | 
			
		||||
		if (setting == NULL) {
 | 
			
		||||
			dev_err(dev,
 | 
			
		||||
				"failed to alloc struct pinctrl_setting\n");
 | 
			
		||||
			ret = -ENOMEM;
 | 
			
		||||
			goto error;
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		setting->pctldev = pctldev;
 | 
			
		||||
		ret = pinmux_map_to_setting(map, setting);
 | 
			
		||||
		if (ret < 0)
 | 
			
		||||
			goto error;
 | 
			
		||||
 | 
			
		||||
		list_add_tail(&setting->node, &p->settings);
 | 
			
		||||
 | 
			
		||||
		num_maps++;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	/*
 | 
			
		||||
	 * This may be perfectly legitimate. An IP block may get re-used
 | 
			
		||||
	 * across SoCs. Not all of those SoCs may need pinmux settings for the
 | 
			
		||||
	 * IP block, e.g. if one SoC dedicates pins to that function but
 | 
			
		||||
	 * another doesn't. The driver won't know this, and will always
 | 
			
		||||
	 * attempt to set up the pinmux. The mapping table defines whether any
 | 
			
		||||
	 * HW programming is actually needed.
 | 
			
		||||
	 */
 | 
			
		||||
	if (!num_maps)
 | 
			
		||||
		dev_info(dev, "zero maps found for mapping %s\n", name);
 | 
			
		||||
 | 
			
		||||
	dev_dbg(dev, "found %u maps for device %s state %s\n",
 | 
			
		||||
		num_maps, devname, name ? name : "(undefined)");
 | 
			
		||||
 | 
			
		||||
	/* Add the pinmux to the global list */
 | 
			
		||||
	list_add_tail(&p->node, &pinctrl_list);
 | 
			
		||||
 | 
			
		||||
	return p;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
error:
 | 
			
		||||
	list_for_each_entry(setting, &p->settings, node)
 | 
			
		||||
		pinmux_free_setting(setting);
 | 
			
		||||
static struct pinctrl *pinctrl_get_locked(struct device *dev)
 | 
			
		||||
{
 | 
			
		||||
	struct pinctrl *p;
 | 
			
		||||
 | 
			
		||||
	kfree(p);
 | 
			
		||||
	if (WARN_ON(!dev))
 | 
			
		||||
		return ERR_PTR(-EINVAL);
 | 
			
		||||
 | 
			
		||||
	return ERR_PTR(ret);
 | 
			
		||||
	p = find_pinctrl(dev);
 | 
			
		||||
	if (p != NULL)
 | 
			
		||||
		return ERR_PTR(-EBUSY);
 | 
			
		||||
 | 
			
		||||
	p = create_pinctrl(dev);
 | 
			
		||||
	if (IS_ERR(p))
 | 
			
		||||
		return p;
 | 
			
		||||
 | 
			
		||||
	return p;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * pinctrl_get() - retrieves the pin controller handle for a certain device
 | 
			
		||||
 * @dev: the device to get the pin controller handle for
 | 
			
		||||
 * @name: an optional specific control mapping name or NULL, the name is only
 | 
			
		||||
 *	needed if you want to have more than one mapping per device, or if you
 | 
			
		||||
 *	need an anonymous pin control (not tied to any specific device)
 | 
			
		||||
 * pinctrl_get() - retrieves the pinctrl handle for a device
 | 
			
		||||
 * @dev: the device to obtain the handle for
 | 
			
		||||
 */
 | 
			
		||||
struct pinctrl *pinctrl_get(struct device *dev, const char *name)
 | 
			
		||||
struct pinctrl *pinctrl_get(struct device *dev)
 | 
			
		||||
{
 | 
			
		||||
	struct pinctrl *p;
 | 
			
		||||
 | 
			
		||||
	mutex_lock(&pinctrl_mutex);
 | 
			
		||||
	p = pinctrl_get_locked(dev, name);
 | 
			
		||||
	p = pinctrl_get_locked(dev);
 | 
			
		||||
	mutex_unlock(&pinctrl_mutex);
 | 
			
		||||
 | 
			
		||||
	return p;
 | 
			
		||||
}
 | 
			
		||||
EXPORT_SYMBOL_GPL(pinctrl_get);
 | 
			
		||||
 | 
			
		||||
static void pinctrl_put_locked(struct pinctrl *p)
 | 
			
		||||
static void pinctrl_put_locked(struct pinctrl *p, bool inlist)
 | 
			
		||||
{
 | 
			
		||||
	struct pinctrl_setting *setting, *n;
 | 
			
		||||
	struct pinctrl_state *state, *n1;
 | 
			
		||||
	struct pinctrl_setting *setting, *n2;
 | 
			
		||||
 | 
			
		||||
	if (p == NULL)
 | 
			
		||||
		return;
 | 
			
		||||
 | 
			
		||||
	if (p->usecount)
 | 
			
		||||
		pr_warn("releasing pin control handle with active users!\n");
 | 
			
		||||
	list_for_each_entry_safe(setting, n, &p->settings, node) {
 | 
			
		||||
		pinmux_free_setting(setting);
 | 
			
		||||
		list_del(&setting->node);
 | 
			
		||||
		kfree(setting);
 | 
			
		||||
	list_for_each_entry_safe(state, n1, &p->states, node) {
 | 
			
		||||
		list_for_each_entry_safe(setting, n2, &state->settings, node) {
 | 
			
		||||
			if (state == p->state)
 | 
			
		||||
				pinmux_disable_setting(setting);
 | 
			
		||||
			pinmux_free_setting(setting);
 | 
			
		||||
			list_del(&setting->node);
 | 
			
		||||
			kfree(setting);
 | 
			
		||||
		}
 | 
			
		||||
		list_del(&state->node);
 | 
			
		||||
		kfree(state);
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	/* Remove from list */
 | 
			
		||||
	list_del(&p->node);
 | 
			
		||||
 | 
			
		||||
	if (inlist)
 | 
			
		||||
		list_del(&p->node);
 | 
			
		||||
	kfree(p);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * pinctrl_put() - release a previously claimed pin control handle
 | 
			
		||||
 * @p: a pin control handle previously claimed by pinctrl_get()
 | 
			
		||||
 * pinctrl_put() - release a previously claimed pinctrl handle
 | 
			
		||||
 * @p: the pinctrl handle to release
 | 
			
		||||
 */
 | 
			
		||||
void pinctrl_put(struct pinctrl *p)
 | 
			
		||||
{
 | 
			
		||||
	mutex_lock(&pinctrl_mutex);
 | 
			
		||||
	pinctrl_put(p);
 | 
			
		||||
	pinctrl_put_locked(p, true);
 | 
			
		||||
	mutex_unlock(&pinctrl_mutex);
 | 
			
		||||
}
 | 
			
		||||
EXPORT_SYMBOL_GPL(pinctrl_put);
 | 
			
		||||
 | 
			
		||||
static int pinctrl_enable_locked(struct pinctrl *p)
 | 
			
		||||
static struct pinctrl_state *pinctrl_lookup_state_locked(struct pinctrl *p,
 | 
			
		||||
							 const char *name)
 | 
			
		||||
{
 | 
			
		||||
	struct pinctrl_setting *setting;
 | 
			
		||||
	struct pinctrl_state *state;
 | 
			
		||||
 | 
			
		||||
	state = find_state(p, name);
 | 
			
		||||
	if (!state)
 | 
			
		||||
		return ERR_PTR(-ENODEV);
 | 
			
		||||
 | 
			
		||||
	return state;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * pinctrl_lookup_state() - retrieves a state handle from a pinctrl handle
 | 
			
		||||
 * @p: the pinctrl handle to retrieve the state from
 | 
			
		||||
 * @name: the state name to retrieve
 | 
			
		||||
 */
 | 
			
		||||
struct pinctrl_state *pinctrl_lookup_state(struct pinctrl *p, const char *name)
 | 
			
		||||
{
 | 
			
		||||
	struct pinctrl_state *s;
 | 
			
		||||
 | 
			
		||||
	mutex_lock(&pinctrl_mutex);
 | 
			
		||||
	s = pinctrl_lookup_state_locked(p, name);
 | 
			
		||||
	mutex_unlock(&pinctrl_mutex);
 | 
			
		||||
 | 
			
		||||
	return s;
 | 
			
		||||
}
 | 
			
		||||
EXPORT_SYMBOL_GPL(pinctrl_lookup_state);
 | 
			
		||||
 | 
			
		||||
static int pinctrl_select_state_locked(struct pinctrl *p,
 | 
			
		||||
				       struct pinctrl_state *state)
 | 
			
		||||
{
 | 
			
		||||
	struct pinctrl_setting *setting, *setting2;
 | 
			
		||||
	int ret;
 | 
			
		||||
 | 
			
		||||
	if (p == NULL)
 | 
			
		||||
		return -EINVAL;
 | 
			
		||||
	if (p->state == state)
 | 
			
		||||
		return 0;
 | 
			
		||||
 | 
			
		||||
	if (p->usecount++ == 0) {
 | 
			
		||||
		list_for_each_entry(setting, &p->settings, node) {
 | 
			
		||||
			ret = pinmux_enable_setting(setting);
 | 
			
		||||
			if (ret < 0) {
 | 
			
		||||
				/* FIXME: Difficult to return to prev state */
 | 
			
		||||
				p->usecount--;
 | 
			
		||||
				return ret;
 | 
			
		||||
	if (p->state) {
 | 
			
		||||
		/*
 | 
			
		||||
		 * The set of groups with a mux configuration in the old state
 | 
			
		||||
		 * may not be identical to the set of groups with a mux setting
 | 
			
		||||
		 * in the new state. While this might be unusual, it's entirely
 | 
			
		||||
		 * possible for the "user"-supplied mapping table to be written
 | 
			
		||||
		 * that way. For each group that was configured in the old state
 | 
			
		||||
		 * but not in the new state, this code puts that group into a
 | 
			
		||||
		 * safe/disabled state.
 | 
			
		||||
		 */
 | 
			
		||||
		list_for_each_entry(setting, &p->state->settings, node) {
 | 
			
		||||
			bool found = false;
 | 
			
		||||
			list_for_each_entry(setting2, &state->settings, node) {
 | 
			
		||||
				if (setting2->group_selector ==
 | 
			
		||||
						setting->group_selector) {
 | 
			
		||||
					found = true;
 | 
			
		||||
					break;
 | 
			
		||||
				}
 | 
			
		||||
			}
 | 
			
		||||
			if (!found)
 | 
			
		||||
				pinmux_disable_setting(setting);
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	p->state = state;
 | 
			
		||||
 | 
			
		||||
	/* Apply all the settings for the new state */
 | 
			
		||||
	list_for_each_entry(setting, &state->settings, node) {
 | 
			
		||||
		ret = pinmux_enable_setting(setting);
 | 
			
		||||
		if (ret < 0) {
 | 
			
		||||
			/* FIXME: Difficult to return to prev state */
 | 
			
		||||
			return ret;
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			@ -638,43 +730,21 @@ static int pinctrl_enable_locked(struct pinctrl *p)
 | 
			
		|||
}
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * pinctrl_enable() - enable a certain pin controller setting
 | 
			
		||||
 * @p: the pin control handle to enable, previously claimed by pinctrl_get()
 | 
			
		||||
 * pinctrl_select() - select/activate/program a pinctrl state to HW
 | 
			
		||||
 * @p: the pinctrl handle for the device that requests configuratio
 | 
			
		||||
 * @state: the state handle to select/activate/program
 | 
			
		||||
 */
 | 
			
		||||
int pinctrl_enable(struct pinctrl *p)
 | 
			
		||||
int pinctrl_select_state(struct pinctrl *p, struct pinctrl_state *state)
 | 
			
		||||
{
 | 
			
		||||
	int ret;
 | 
			
		||||
 | 
			
		||||
	mutex_lock(&pinctrl_mutex);
 | 
			
		||||
	ret = pinctrl_enable_locked(p);
 | 
			
		||||
	ret = pinctrl_select_state_locked(p, state);
 | 
			
		||||
	mutex_unlock(&pinctrl_mutex);
 | 
			
		||||
 | 
			
		||||
	return ret;
 | 
			
		||||
}
 | 
			
		||||
EXPORT_SYMBOL_GPL(pinctrl_enable);
 | 
			
		||||
 | 
			
		||||
static void pinctrl_disable_locked(struct pinctrl *p)
 | 
			
		||||
{
 | 
			
		||||
	struct pinctrl_setting *setting;
 | 
			
		||||
 | 
			
		||||
	if (p == NULL)
 | 
			
		||||
		return;
 | 
			
		||||
 | 
			
		||||
	if (--p->usecount == 0) {
 | 
			
		||||
		list_for_each_entry(setting, &p->settings, node)
 | 
			
		||||
			pinmux_disable_setting(setting);
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * pinctrl_disable() - disable a certain pin control setting
 | 
			
		||||
 * @p: the pin control handle to disable, previously claimed by pinctrl_get()
 | 
			
		||||
 */
 | 
			
		||||
void pinctrl_disable(struct pinctrl *p)
 | 
			
		||||
{
 | 
			
		||||
	mutex_lock(&pinctrl_mutex);
 | 
			
		||||
	pinctrl_disable_locked(p);
 | 
			
		||||
	mutex_unlock(&pinctrl_mutex);
 | 
			
		||||
}
 | 
			
		||||
EXPORT_SYMBOL_GPL(pinctrl_disable);
 | 
			
		||||
EXPORT_SYMBOL_GPL(pinctrl_select_state);
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * pinctrl_register_mappings() - register a set of pin controller mappings
 | 
			
		||||
| 
						 | 
				
			
			@ -891,6 +961,7 @@ static int pinctrl_maps_show(struct seq_file *s, void *what)
 | 
			
		|||
static int pinctrl_show(struct seq_file *s, void *what)
 | 
			
		||||
{
 | 
			
		||||
	struct pinctrl *p;
 | 
			
		||||
	struct pinctrl_state *state;
 | 
			
		||||
	struct pinctrl_setting *setting;
 | 
			
		||||
 | 
			
		||||
	seq_puts(s, "Requested pin control handlers their pinmux maps:\n");
 | 
			
		||||
| 
						 | 
				
			
			@ -898,12 +969,17 @@ static int pinctrl_show(struct seq_file *s, void *what)
 | 
			
		|||
	mutex_lock(&pinctrl_mutex);
 | 
			
		||||
 | 
			
		||||
	list_for_each_entry(p, &pinctrl_list, node) {
 | 
			
		||||
		seq_printf(s, "device: %s state: %s users: %u\n",
 | 
			
		||||
			   dev_name(p->dev), p->state, p->usecount);
 | 
			
		||||
		seq_printf(s, "device: %s current state: %s\n",
 | 
			
		||||
			   dev_name(p->dev),
 | 
			
		||||
			   p->state ? p->state->name : "none");
 | 
			
		||||
 | 
			
		||||
		list_for_each_entry(setting, &p->settings, node) {
 | 
			
		||||
			seq_printf(s, "  ");
 | 
			
		||||
			pinmux_dbg_show(s, setting);
 | 
			
		||||
		list_for_each_entry(state, &p->states, node) {
 | 
			
		||||
			seq_printf(s, "  state: %s\n", state->name);
 | 
			
		||||
 | 
			
		||||
			list_for_each_entry(setting, &state->settings, node) {
 | 
			
		||||
				seq_printf(s, "    ");
 | 
			
		||||
				pinmux_dbg_show(s, setting);
 | 
			
		||||
			}
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			@ -1113,9 +1189,14 @@ struct pinctrl_dev *pinctrl_register(struct pinctrl_desc *pctldesc,
 | 
			
		|||
 | 
			
		||||
	list_add_tail(&pctldev->node, &pinctrldev_list);
 | 
			
		||||
 | 
			
		||||
	pctldev->p = pinctrl_get_locked(pctldev->dev, PINCTRL_STATE_DEFAULT);
 | 
			
		||||
	if (!IS_ERR(pctldev->p))
 | 
			
		||||
		pinctrl_enable_locked(pctldev->p);
 | 
			
		||||
	pctldev->p = pinctrl_get_locked(pctldev->dev);
 | 
			
		||||
	if (!IS_ERR(pctldev->p)) {
 | 
			
		||||
		struct pinctrl_state *s =
 | 
			
		||||
			pinctrl_lookup_state_locked(pctldev->p,
 | 
			
		||||
						    PINCTRL_STATE_DEFAULT);
 | 
			
		||||
		if (!IS_ERR(s))
 | 
			
		||||
			pinctrl_select_state_locked(pctldev->p, s);
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	mutex_unlock(&pinctrl_mutex);
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			@ -1144,10 +1225,8 @@ void pinctrl_unregister(struct pinctrl_dev *pctldev)
 | 
			
		|||
 | 
			
		||||
	mutex_lock(&pinctrl_mutex);
 | 
			
		||||
 | 
			
		||||
	if (!IS_ERR(pctldev->p)) {
 | 
			
		||||
		pinctrl_disable_locked(pctldev->p);
 | 
			
		||||
		pinctrl_put_locked(pctldev->p);
 | 
			
		||||
	}
 | 
			
		||||
	if (!IS_ERR(pctldev->p))
 | 
			
		||||
		pinctrl_put_locked(pctldev->p, true);
 | 
			
		||||
 | 
			
		||||
	/* TODO: check that no pinmuxes are still active? */
 | 
			
		||||
	list_del(&pctldev->node);
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -49,22 +49,31 @@ struct pinctrl_dev {
 | 
			
		|||
 * struct pinctrl - per-device pin control state holder
 | 
			
		||||
 * @node: global list node
 | 
			
		||||
 * @dev: the device using this pin control handle
 | 
			
		||||
 * @state: the state name passed to pinctrl_get()
 | 
			
		||||
 * @usecount: the number of active users of this pin controller setting, used
 | 
			
		||||
 *	to keep track of nested use cases
 | 
			
		||||
 * @settings: a list of settings for this device/state
 | 
			
		||||
 * @states: a list of states for this device
 | 
			
		||||
 * @state: the current state
 | 
			
		||||
 */
 | 
			
		||||
struct pinctrl {
 | 
			
		||||
	struct list_head node;
 | 
			
		||||
	struct device *dev;
 | 
			
		||||
	const char *state;
 | 
			
		||||
	unsigned usecount;
 | 
			
		||||
	struct list_head states;
 | 
			
		||||
	struct pinctrl_state *state;
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * struct pinctrl_state - a pinctrl state for a device
 | 
			
		||||
 * @node: list not for struct pinctrl's @states field
 | 
			
		||||
 * @name: the name of this state
 | 
			
		||||
 * @settings: a list of settings for this state
 | 
			
		||||
 */
 | 
			
		||||
struct pinctrl_state {
 | 
			
		||||
	struct list_head node;
 | 
			
		||||
	const char *name;
 | 
			
		||||
	struct list_head settings;
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * struct pinctrl_setting - an individual mux setting
 | 
			
		||||
 * @node: list node for struct pinctrl's @settings field
 | 
			
		||||
 * @node: list node for struct pinctrl_settings's @settings field
 | 
			
		||||
 * @pctldev: pin control device handling to be programmed
 | 
			
		||||
 * @group_selector: the group selector to program
 | 
			
		||||
 * @func_selector: the function selector to program
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -673,12 +673,10 @@ int sirfsoc_uart_probe(struct platform_device *pdev)
 | 
			
		|||
	port->irq = res->start;
 | 
			
		||||
 | 
			
		||||
	if (sirfport->hw_flow_ctrl) {
 | 
			
		||||
		sirfport->p = pinctrl_get(&pdev->dev, PINCTRL_STATE_DEFAULT);
 | 
			
		||||
		sirfport->p = pinctrl_get_select_default(&pdev->dev);
 | 
			
		||||
		ret = IS_ERR(sirfport->p);
 | 
			
		||||
		if (ret)
 | 
			
		||||
			goto pin_err;
 | 
			
		||||
 | 
			
		||||
		pinctrl_enable(sirfport->p);
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	port->ops = &sirfsoc_uart_ops;
 | 
			
		||||
| 
						 | 
				
			
			@ -695,10 +693,8 @@ int sirfsoc_uart_probe(struct platform_device *pdev)
 | 
			
		|||
 | 
			
		||||
port_err:
 | 
			
		||||
	platform_set_drvdata(pdev, NULL);
 | 
			
		||||
	if (sirfport->hw_flow_ctrl) {
 | 
			
		||||
		pinctrl_disable(sirfport->p);
 | 
			
		||||
	if (sirfport->hw_flow_ctrl)
 | 
			
		||||
		pinctrl_put(sirfport->p);
 | 
			
		||||
	}
 | 
			
		||||
pin_err:
 | 
			
		||||
irq_err:
 | 
			
		||||
	devm_iounmap(&pdev->dev, port->membase);
 | 
			
		||||
| 
						 | 
				
			
			@ -711,10 +707,8 @@ static int sirfsoc_uart_remove(struct platform_device *pdev)
 | 
			
		|||
	struct sirfsoc_uart_port *sirfport = platform_get_drvdata(pdev);
 | 
			
		||||
	struct uart_port *port = &sirfport->port;
 | 
			
		||||
	platform_set_drvdata(pdev, NULL);
 | 
			
		||||
	if (sirfport->hw_flow_ctrl) {
 | 
			
		||||
		pinctrl_disable(sirfport->p);
 | 
			
		||||
	if (sirfport->hw_flow_ctrl)
 | 
			
		||||
		pinctrl_put(sirfport->p);
 | 
			
		||||
	}
 | 
			
		||||
	devm_iounmap(&pdev->dev, port->membase);
 | 
			
		||||
	uart_remove_one_port(&sirfsoc_uart_drv, port);
 | 
			
		||||
	return 0;
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -12,12 +12,14 @@
 | 
			
		|||
#ifndef __LINUX_PINCTRL_CONSUMER_H
 | 
			
		||||
#define __LINUX_PINCTRL_CONSUMER_H
 | 
			
		||||
 | 
			
		||||
#include <linux/err.h>
 | 
			
		||||
#include <linux/list.h>
 | 
			
		||||
#include <linux/seq_file.h>
 | 
			
		||||
#include "pinctrl.h"
 | 
			
		||||
 | 
			
		||||
/* This struct is private to the core and should be regarded as a cookie */
 | 
			
		||||
struct pinctrl;
 | 
			
		||||
struct pinctrl_state;
 | 
			
		||||
 | 
			
		||||
#ifdef CONFIG_PINCTRL
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			@ -26,10 +28,13 @@ extern int pinctrl_request_gpio(unsigned gpio);
 | 
			
		|||
extern void pinctrl_free_gpio(unsigned gpio);
 | 
			
		||||
extern int pinctrl_gpio_direction_input(unsigned gpio);
 | 
			
		||||
extern int pinctrl_gpio_direction_output(unsigned gpio);
 | 
			
		||||
extern struct pinctrl * __must_check pinctrl_get(struct device *dev, const char *name);
 | 
			
		||||
 | 
			
		||||
extern struct pinctrl * __must_check pinctrl_get(struct device *dev);
 | 
			
		||||
extern void pinctrl_put(struct pinctrl *p);
 | 
			
		||||
extern int pinctrl_enable(struct pinctrl *p);
 | 
			
		||||
extern void pinctrl_disable(struct pinctrl *p);
 | 
			
		||||
extern struct pinctrl_state * __must_check pinctrl_lookup_state(
 | 
			
		||||
							struct pinctrl *p,
 | 
			
		||||
							const char *name);
 | 
			
		||||
extern int pinctrl_select_state(struct pinctrl *p, struct pinctrl_state *s);
 | 
			
		||||
 | 
			
		||||
#else /* !CONFIG_PINCTRL */
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			@ -52,7 +57,7 @@ static inline int pinctrl_gpio_direction_output(unsigned gpio)
 | 
			
		|||
	return 0;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static inline struct pinctrl * __must_check pinctrl_get(struct device *dev, const char *name)
 | 
			
		||||
static inline struct pinctrl * __must_check pinctrl_get(struct device *dev)
 | 
			
		||||
{
 | 
			
		||||
	return NULL;
 | 
			
		||||
}
 | 
			
		||||
| 
						 | 
				
			
			@ -61,16 +66,52 @@ static inline void pinctrl_put(struct pinctrl *p)
 | 
			
		|||
{
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static inline int pinctrl_enable(struct pinctrl *p)
 | 
			
		||||
static inline struct pinctrl_state * __must_check pinctrl_lookup_state(
 | 
			
		||||
							struct pinctrl *p,
 | 
			
		||||
							const char *name)
 | 
			
		||||
{
 | 
			
		||||
	return NULL;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static inline int pinctrl_select_state(struct pinctrl *p,
 | 
			
		||||
				       struct pinctrl_state *s)
 | 
			
		||||
{
 | 
			
		||||
	return 0;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static inline void pinctrl_disable(struct pinctrl *p)
 | 
			
		||||
#endif /* CONFIG_PINCTRL */
 | 
			
		||||
 | 
			
		||||
static inline struct pinctrl * __must_check pinctrl_get_select(
 | 
			
		||||
					struct device *dev, const char *name)
 | 
			
		||||
{
 | 
			
		||||
	struct pinctrl *p;
 | 
			
		||||
	struct pinctrl_state *s;
 | 
			
		||||
	int ret;
 | 
			
		||||
 | 
			
		||||
	p = pinctrl_get(dev);
 | 
			
		||||
	if (IS_ERR(p))
 | 
			
		||||
		return p;
 | 
			
		||||
 | 
			
		||||
	s = pinctrl_lookup_state(p, name);
 | 
			
		||||
	if (IS_ERR(s)) {
 | 
			
		||||
		pinctrl_put(p);
 | 
			
		||||
		return ERR_PTR(PTR_ERR(s));
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	ret = pinctrl_select_state(p, s);
 | 
			
		||||
	if (ret < 0) {
 | 
			
		||||
		pinctrl_put(p);
 | 
			
		||||
		return ERR_PTR(ret);
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	return p;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
#endif /* CONFIG_PINCTRL */
 | 
			
		||||
static inline struct pinctrl * __must_check pinctrl_get_select_default(
 | 
			
		||||
					struct device *dev)
 | 
			
		||||
{
 | 
			
		||||
	return pinctrl_get_select(dev, PINCTRL_STATE_DEFAULT);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
#ifdef CONFIG_PINCONF
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -21,23 +21,22 @@
 | 
			
		|||
 *	same name as the pin controllers own dev_name(), the map entry will be
 | 
			
		||||
 *	hogged by the driver itself upon registration
 | 
			
		||||
 * @name: the name of this specific map entry for the particular machine.
 | 
			
		||||
 *	This is the second parameter passed to pinmux_get() when you want
 | 
			
		||||
 *	to have several mappings to the same device
 | 
			
		||||
 *	This is the parameter passed to pinmux_lookup_state()
 | 
			
		||||
 * @ctrl_dev_name: the name of the device controlling this specific mapping,
 | 
			
		||||
 *	the name must be the same as in your struct device*
 | 
			
		||||
 * @function: a function in the driver to use for this mapping, the driver
 | 
			
		||||
 *	will lookup the function referenced by this ID on the specified
 | 
			
		||||
 *	pin control device
 | 
			
		||||
 * @group: sometimes a function can map to different pin groups, so this
 | 
			
		||||
 *	selects a certain specific pin group to activate for the function, if
 | 
			
		||||
 *	left as NULL, the first applicable group will be used
 | 
			
		||||
 * @function: a function in the driver to use for this mapping, the driver
 | 
			
		||||
 *	will lookup the function referenced by this ID on the specified
 | 
			
		||||
 *	pin control device
 | 
			
		||||
 */
 | 
			
		||||
struct pinctrl_map {
 | 
			
		||||
	const char *dev_name;
 | 
			
		||||
	const char *name;
 | 
			
		||||
	const char *ctrl_dev_name;
 | 
			
		||||
	const char *function;
 | 
			
		||||
	const char *group;
 | 
			
		||||
	const char *function;
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
/*
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
		Loading…
	
		Reference in a new issue