Message ID | 20250203031213.399914-3-koichiro.den@canonical.com |
---|---|
State | New |
Headers | show |
Series | Introduce configfs-based interface for gpio-aggregator | expand |
Hi Den-san, On Mon, 3 Feb 2025 at 04:12, Koichiro Den <koichiro.den@canonical.com> wrote: > The existing sysfs 'new_device' interface has several limitations: > * No way to determine when GPIO aggregator creation is complete. > * No way to retrieve errors when creating a GPIO aggregator. > * No way to trace a GPIO line of an aggregator back to its > corresponding physical device. > * The 'new_device' echo does not indicate which virtual gpiochip<N> > was created. > * No way to assign names to GPIO lines exported through an aggregator. > > Introduce the new configfs interface for gpio-aggregator to address > these limitations. It provides a more streamlined, modern, and > extensible configuration method. For backward compatibility, the > 'new_device' interface and its behaviour is retained for now. > > This commit implements minimal functionalities: > > /config/gpio-aggregator/<name-of-your-choice>/ > /config/gpio-aggregator/<name-of-your-choice>/live > /config/gpio-aggregator/<name-of-your-choice>/<lineY>/ > /config/gpio-aggregator/<name-of-your-choice>/<lineY>/key > /config/gpio-aggregator/<name-of-your-choice>/<lineY>/offset > > Basic setup flow is: > 1. Create a directory for a GPIO aggregator. > 2. Create subdirectories for each line you want to instantiate. > 3. In each line directory, configure the key and offset. > The key/offset semantics are as follows: > * If offset is >= 0: > - key specifies the name of the chip this GPIO belongs to > - offset specifies the line offset within that chip. > * If offset is <0: > - key needs to specify the GPIO line name. > 4. Return to the aggregator's root directory and write '1' to the live > attribute. > > For example, the command in the existing kernel doc: > > echo 'e6052000.gpio 19 e6050000.gpio 20-21' > new_device > > is equivalent to: > > mkdir /sys/kernel/config/gpio-aggregator/<custom-name> > # Change <custom-name> to name of your choice (e.g. "aggr0") > cd /sys/kernel/config/gpio-aggregator/<custom-name> > mkdir line0 line1 line2 # Only "line<Y>" naming allowed. > echo e6052000.gpio > line0/key > echo 19 > line0/offset > echo e6050000.gpio > line1/key > echo 20 > line1/offset > echo e6050000.gpio > line2/key > echo 21 > line2/offset > echo 1 > live > > Signed-off-by: Koichiro Den <koichiro.den@canonical.com> Thanks for your patch! > --- a/drivers/gpio/gpio-aggregator.c > +++ b/drivers/gpio/gpio-aggregator.c > @@ -9,10 +9,14 @@ > > #include <linux/bitmap.h> > #include <linux/bitops.h> > +#include <linux/completion.h> > +#include <linux/configfs.h> Using configfs requires CONFIG_CONFIGFS_FS. So either GPIO_AGGREGATOR should select CONFIGFS_FS, or all configfs-related code should be protected by checks for CONFIG_CONFIGFS_FS. Since we want to encourage people to use the new API, I think the former is preferred. > #include <linux/ctype.h> > #include <linux/delay.h> > #include <linux/idr.h> > #include <linux/kernel.h> > +#include <linux/list.h> > +#include <linux/lockdep.h> > #include <linux/mod_devicetable.h> > #include <linux/module.h> > #include <linux/mutex.h> > @@ -34,11 +38,39 @@ > */ > > struct gpio_aggregator { > + struct config_group group; > struct gpiod_lookup_table *lookups; > struct platform_device *pdev; > + struct mutex lock; > + int id; > + > + /* Synchronize with probe */ > + struct notifier_block bus_notifier; > + struct completion probe_completion; > + bool driver_bound; > + > + /* List of gpio_aggregator_line. Always added in order */ > + struct list_head list_head; > + > char args[]; > }; > > +struct gpio_aggregator_line { > + struct config_group group; > + struct gpio_aggregator *parent; > + struct list_head entry; > + > + /* Line index within the aggregator device */ > + int idx; unsigned int > + > + /* GPIO chip label or line name */ > + char *key; > + /* Can be negative to indicate lookup by line name */ > + int offset; > + > + enum gpio_lookup_flags flags; > +}; > + > static DEFINE_MUTEX(gpio_aggregator_lock); /* protects idr */ > static DEFINE_IDR(gpio_aggregator_idr); > +static bool aggr_is_active(struct gpio_aggregator *aggr) > +{ > + lockdep_assert_held(&aggr->lock); > + > + return !!aggr->pdev && platform_get_drvdata(aggr->pdev); No need for "!!". > +} > +static void aggr_line_add(struct gpio_aggregator *aggr, > + struct gpio_aggregator_line *line) > +{ > + struct gpio_aggregator_line *tmp; > + > + lockdep_assert_held(&aggr->lock); > + > + list_for_each_entry(tmp, &aggr->list_head, entry) { > + if (WARN_ON(tmp->idx == line->idx)) Can this really happen? > + return; > + if (tmp->idx > line->idx) { > + list_add_tail(&line->entry, &tmp->entry); > + return; > + } > + } > + list_add_tail(&line->entry, &aggr->list_head); > +} > +static void aggr_lockup_configfs(struct gpio_aggregator *aggr, bool lock) > +{ > + struct configfs_subsystem *subsys = aggr->group.cg_subsys; > + struct gpio_aggregator_line *line; > + > + /* > + * The device only needs to depend on leaf lines. This is > + * sufficient to lock up all the configfs entries that the > + * instantiated, alive device depends on. > + */ > + list_for_each_entry(line, &aggr->list_head, entry) { > + if (lock) > + WARN_ON(configfs_depend_item_unlocked( > + subsys, &line->group.cg_item)); Can this happen? I am also worried if this can be triggered by the user, combined with panic_on_warn. > + else > + configfs_undepend_item_unlocked( > + &line->group.cg_item); > + } > +} > + > +static ssize_t > +gpio_aggr_line_key_show(struct config_item *item, char *page) > +{ > + struct gpio_aggregator_line *line = to_gpio_aggregator_line(item); > + struct gpio_aggregator *aggr = line->parent; > + > + guard(mutex)(&aggr->lock); > + > + return sprintf(page, "%s\n", line->key ?: ""); Please use sysfs_emit() instead (everywhere). > +} > +static ssize_t > +gpio_aggr_device_live_store(struct config_item *item, const char *page, > + size_t count) > +{ > + struct gpio_aggregator *aggr = to_gpio_aggregator(item); > + int ret = 0; > + bool live; > + > + ret = kstrtobool(page, &live); > + if (ret) > + return ret; > + > + if (live) > + aggr_lockup_configfs(aggr, true); > + > + scoped_guard(mutex, &aggr->lock) { > + if (live == aggr_is_active(aggr)) > + ret = -EPERM; > + else if (live) > + ret = aggr_activate(aggr); > + else > + aggr_deactivate(aggr); > + } > + > + /* > + * Undepend is required only if device disablement (live == 0) s/Undepend/Lock-up/? > + * succeeds or if device enablement (live == 1) fails. > + */ > + if (live == !!ret) > + aggr_lockup_configfs(aggr, false); > + > + return ret ?: count; > +} > +static struct config_group * > +gpio_aggr_make_group(struct config_group *group, const char *name) > +{ > + /* arg space is unneeded */ > + struct gpio_aggregator *aggr = kzalloc(sizeof(*aggr), GFP_KERNEL); > + if (!aggr) > + return ERR_PTR(-ENOMEM); > + > + mutex_lock(&gpio_aggregator_lock); > + aggr->id = idr_alloc(&gpio_aggregator_idr, aggr, 0, 0, GFP_KERNEL); > + mutex_unlock(&gpio_aggregator_lock); > + > + if (aggr->id < 0) { > + kfree(aggr); > + return ERR_PTR(aggr->id); > + } The above more or less duplicates the existing code in new_device_store(). > + > + INIT_LIST_HEAD(&aggr->list_head); > + mutex_init(&aggr->lock); > + config_group_init_type_name(&aggr->group, name, &gpio_aggr_device_type); > + init_completion(&aggr->probe_completion); > + > + return &aggr->group; > +} Gr{oetje,eeting}s, Geert
On Wed, Feb 12, 2025 at 02:48:12PM GMT, Geert Uytterhoeven wrote: > Hi Den-san, > > On Mon, 3 Feb 2025 at 04:12, Koichiro Den <koichiro.den@canonical.com> wrote: > > The existing sysfs 'new_device' interface has several limitations: > > * No way to determine when GPIO aggregator creation is complete. > > * No way to retrieve errors when creating a GPIO aggregator. > > * No way to trace a GPIO line of an aggregator back to its > > corresponding physical device. > > * The 'new_device' echo does not indicate which virtual gpiochip<N> > > was created. > > * No way to assign names to GPIO lines exported through an aggregator. > > > > Introduce the new configfs interface for gpio-aggregator to address > > these limitations. It provides a more streamlined, modern, and > > extensible configuration method. For backward compatibility, the > > 'new_device' interface and its behaviour is retained for now. > > > > This commit implements minimal functionalities: > > > > /config/gpio-aggregator/<name-of-your-choice>/ > > /config/gpio-aggregator/<name-of-your-choice>/live > > /config/gpio-aggregator/<name-of-your-choice>/<lineY>/ > > /config/gpio-aggregator/<name-of-your-choice>/<lineY>/key > > /config/gpio-aggregator/<name-of-your-choice>/<lineY>/offset > > > > Basic setup flow is: > > 1. Create a directory for a GPIO aggregator. > > 2. Create subdirectories for each line you want to instantiate. > > 3. In each line directory, configure the key and offset. > > The key/offset semantics are as follows: > > * If offset is >= 0: > > - key specifies the name of the chip this GPIO belongs to > > - offset specifies the line offset within that chip. > > * If offset is <0: > > - key needs to specify the GPIO line name. > > 4. Return to the aggregator's root directory and write '1' to the live > > attribute. > > > > For example, the command in the existing kernel doc: > > > > echo 'e6052000.gpio 19 e6050000.gpio 20-21' > new_device > > > > is equivalent to: > > > > mkdir /sys/kernel/config/gpio-aggregator/<custom-name> > > # Change <custom-name> to name of your choice (e.g. "aggr0") > > cd /sys/kernel/config/gpio-aggregator/<custom-name> > > mkdir line0 line1 line2 # Only "line<Y>" naming allowed. > > echo e6052000.gpio > line0/key > > echo 19 > line0/offset > > echo e6050000.gpio > line1/key > > echo 20 > line1/offset > > echo e6050000.gpio > line2/key > > echo 21 > line2/offset > > echo 1 > live > > > > Signed-off-by: Koichiro Den <koichiro.den@canonical.com> > > Thanks for your patch! > > > --- a/drivers/gpio/gpio-aggregator.c > > +++ b/drivers/gpio/gpio-aggregator.c > > @@ -9,10 +9,14 @@ > > > > #include <linux/bitmap.h> > > #include <linux/bitops.h> > > +#include <linux/completion.h> > > +#include <linux/configfs.h> > > Using configfs requires CONFIG_CONFIGFS_FS. > So either GPIO_AGGREGATOR should select CONFIGFS_FS, or > all configfs-related code should be protected by checks for > CONFIG_CONFIGFS_FS. Since we want to encourage people to use the new > API, I think the former is preferred. Indeed. I had mentioned this in the response to Bartosz here: https://lore.kernel.org/all/dmy4mvxut3l5kqds2b2fnnes5ukr73spddwgrbkeoqrb5p5wir@hkq6ltr7d6dt/ I agree with the former. > > > #include <linux/ctype.h> > > #include <linux/delay.h> > > #include <linux/idr.h> > > #include <linux/kernel.h> > > +#include <linux/list.h> > > +#include <linux/lockdep.h> > > #include <linux/mod_devicetable.h> > > #include <linux/module.h> > > #include <linux/mutex.h> > > @@ -34,11 +38,39 @@ > > */ > > > > struct gpio_aggregator { > > + struct config_group group; > > struct gpiod_lookup_table *lookups; > > struct platform_device *pdev; > > + struct mutex lock; > > + int id; > > + > > + /* Synchronize with probe */ > > + struct notifier_block bus_notifier; > > + struct completion probe_completion; > > + bool driver_bound; > > + > > + /* List of gpio_aggregator_line. Always added in order */ > > + struct list_head list_head; > > + > > char args[]; > > }; > > > > +struct gpio_aggregator_line { > > + struct config_group group; > > + struct gpio_aggregator *parent; > > + struct list_head entry; > > + > > + /* Line index within the aggregator device */ > > + int idx; > > unsigned int Thanks. I'll fix this in v3. > > > + > > + /* GPIO chip label or line name */ > > + char *key; > > + /* Can be negative to indicate lookup by line name */ > > + int offset; > > + > > + enum gpio_lookup_flags flags; > > +}; > > + > > static DEFINE_MUTEX(gpio_aggregator_lock); /* protects idr */ > > static DEFINE_IDR(gpio_aggregator_idr); > > > +static bool aggr_is_active(struct gpio_aggregator *aggr) > > +{ > > + lockdep_assert_held(&aggr->lock); > > + > > + return !!aggr->pdev && platform_get_drvdata(aggr->pdev); > > No need for "!!". I'll fix this in v3. > > > +} > > > +static void aggr_line_add(struct gpio_aggregator *aggr, > > + struct gpio_aggregator_line *line) > > +{ > > + struct gpio_aggregator_line *tmp; > > + > > + lockdep_assert_held(&aggr->lock); > > + > > + list_for_each_entry(tmp, &aggr->list_head, entry) { > > + if (WARN_ON(tmp->idx == line->idx)) > > Can this really happen? I don't think so. It was just a safeguard for the future to help us notice when something very bad would happen, caused by changes on codebase. So let me drop it in v3. > > > + return; > > + if (tmp->idx > line->idx) { > > + list_add_tail(&line->entry, &tmp->entry); > > + return; > > + } > > + } > > + list_add_tail(&line->entry, &aggr->list_head); > > +} > > > +static void aggr_lockup_configfs(struct gpio_aggregator *aggr, bool lock) > > +{ > > + struct configfs_subsystem *subsys = aggr->group.cg_subsys; > > + struct gpio_aggregator_line *line; > > + > > + /* > > + * The device only needs to depend on leaf lines. This is > > + * sufficient to lock up all the configfs entries that the > > + * instantiated, alive device depends on. > > + */ > > + list_for_each_entry(line, &aggr->list_head, entry) { > > + if (lock) > > + WARN_ON(configfs_depend_item_unlocked( > > + subsys, &line->group.cg_item)); > > Can this happen? > I am also worried if this can be triggered by the user, combined > with panic_on_warn. I don't think so. This was just a safeguard for the future. > > > + else > > + configfs_undepend_item_unlocked( > > + &line->group.cg_item); > > + } > > +} > > + > > +static ssize_t > > +gpio_aggr_line_key_show(struct config_item *item, char *page) > > +{ > > + struct gpio_aggregator_line *line = to_gpio_aggregator_line(item); > > + struct gpio_aggregator *aggr = line->parent; > > + > > + guard(mutex)(&aggr->lock); > > + > > + return sprintf(page, "%s\n", line->key ?: ""); > > Please use sysfs_emit() instead (everywhere). > Thanks for pointing it out. I'll fix all of them in v3. > > > +} > > > +static ssize_t > > +gpio_aggr_device_live_store(struct config_item *item, const char *page, > > + size_t count) > > +{ > > + struct gpio_aggregator *aggr = to_gpio_aggregator(item); > > + int ret = 0; > > + bool live; > > + > > + ret = kstrtobool(page, &live); > > + if (ret) > > + return ret; > > + > > + if (live) > > + aggr_lockup_configfs(aggr, true); > > + > > + scoped_guard(mutex, &aggr->lock) { > > + if (live == aggr_is_active(aggr)) > > + ret = -EPERM; > > + else if (live) > > + ret = aggr_activate(aggr); > > + else > > + aggr_deactivate(aggr); > > + } > > + > > + /* > > + * Undepend is required only if device disablement (live == 0) > > s/Undepend/Lock-up/? I must admit that I couldn't find a best suitable antonym to 'depend'. IMO Lock-up sounds a bit misleading. How about 'Unlock'? > > > + * succeeds or if device enablement (live == 1) fails. > > + */ > > + if (live == !!ret) > > + aggr_lockup_configfs(aggr, false); > > + > > + return ret ?: count; > > +} > > > +static struct config_group * > > +gpio_aggr_make_group(struct config_group *group, const char *name) > > +{ > > + /* arg space is unneeded */ > > + struct gpio_aggregator *aggr = kzalloc(sizeof(*aggr), GFP_KERNEL); > > + if (!aggr) > > + return ERR_PTR(-ENOMEM); > > + > > + mutex_lock(&gpio_aggregator_lock); > > + aggr->id = idr_alloc(&gpio_aggregator_idr, aggr, 0, 0, GFP_KERNEL); > > + mutex_unlock(&gpio_aggregator_lock); > > + > > + if (aggr->id < 0) { > > + kfree(aggr); > > + return ERR_PTR(aggr->id); > > + } > > The above more or less duplicates the existing code in > new_device_store(). I'll factor out the common part and add new funcs gpio_alloc()/gpio_free(). Please let me know if any objections. > > > + > > + INIT_LIST_HEAD(&aggr->list_head); > > + mutex_init(&aggr->lock); > > + config_group_init_type_name(&aggr->group, name, &gpio_aggr_device_type); > > + init_completion(&aggr->probe_completion); > > + > > + return &aggr->group; > > +} > > Gr{oetje,eeting}s, Thanks for the review. Koichiro > > Geert > > -- > Geert Uytterhoeven -- There's lots of Linux beyond ia32 -- geert@linux-m68k.org > > In personal conversations with technical people, I call myself a hacker. But > when I'm talking to journalists I just say "programmer" or something like that. > -- Linus Torvalds
Hi Den-san, On Thu, 13 Feb 2025 at 15:13, Koichiro Den <koichiro.den@canonical.com> wrote: > On Wed, Feb 12, 2025 at 02:48:12PM GMT, Geert Uytterhoeven wrote: > > On Mon, 3 Feb 2025 at 04:12, Koichiro Den <koichiro.den@canonical.com> wrote: > > > The existing sysfs 'new_device' interface has several limitations: > > > * No way to determine when GPIO aggregator creation is complete. > > > * No way to retrieve errors when creating a GPIO aggregator. > > > * No way to trace a GPIO line of an aggregator back to its > > > corresponding physical device. > > > * The 'new_device' echo does not indicate which virtual gpiochip<N> > > > was created. > > > * No way to assign names to GPIO lines exported through an aggregator. > > > > > > Introduce the new configfs interface for gpio-aggregator to address > > > these limitations. It provides a more streamlined, modern, and > > > extensible configuration method. For backward compatibility, the > > > 'new_device' interface and its behaviour is retained for now. > > > > > > This commit implements minimal functionalities: > > > > > > /config/gpio-aggregator/<name-of-your-choice>/ > > > /config/gpio-aggregator/<name-of-your-choice>/live > > > /config/gpio-aggregator/<name-of-your-choice>/<lineY>/ > > > /config/gpio-aggregator/<name-of-your-choice>/<lineY>/key > > > /config/gpio-aggregator/<name-of-your-choice>/<lineY>/offset > > > > > > Basic setup flow is: > > > 1. Create a directory for a GPIO aggregator. > > > 2. Create subdirectories for each line you want to instantiate. > > > 3. In each line directory, configure the key and offset. > > > The key/offset semantics are as follows: > > > * If offset is >= 0: > > > - key specifies the name of the chip this GPIO belongs to > > > - offset specifies the line offset within that chip. > > > * If offset is <0: > > > - key needs to specify the GPIO line name. > > > 4. Return to the aggregator's root directory and write '1' to the live > > > attribute. > > > > > > For example, the command in the existing kernel doc: > > > > > > echo 'e6052000.gpio 19 e6050000.gpio 20-21' > new_device > > > > > > is equivalent to: > > > > > > mkdir /sys/kernel/config/gpio-aggregator/<custom-name> > > > # Change <custom-name> to name of your choice (e.g. "aggr0") > > > cd /sys/kernel/config/gpio-aggregator/<custom-name> > > > mkdir line0 line1 line2 # Only "line<Y>" naming allowed. > > > echo e6052000.gpio > line0/key > > > echo 19 > line0/offset > > > echo e6050000.gpio > line1/key > > > echo 20 > line1/offset > > > echo e6050000.gpio > line2/key > > > echo 21 > line2/offset > > > echo 1 > live > > > > > > Signed-off-by: Koichiro Den <koichiro.den@canonical.com> > > > --- a/drivers/gpio/gpio-aggregator.c > > > +++ b/drivers/gpio/gpio-aggregator.c > > > + /* > > > + * Undepend is required only if device disablement (live == 0) > > > > s/Undepend/Lock-up/? > > I must admit that I couldn't find a best suitable antonym to 'depend'. > IMO Lock-up sounds a bit misleading. How about 'Unlock'? I wrote "Lock-up" to match the "lockup" in aggr_lockup_configfs() below. But now I understand why you wrote "Undepend": when passed "false", aggr_lockup_configfs() calls configfs_undepend_item_unlocked(), so "Undepend" is fine. > > > + * succeeds or if device enablement (live == 1) fails. > > > + */ > > > + if (live == !!ret) > > > + aggr_lockup_configfs(aggr, false); > > > + > > > + return ret ?: count; > > > +} > > > +static struct config_group * > > > +gpio_aggr_make_group(struct config_group *group, const char *name) > > > +{ > > > + /* arg space is unneeded */ > > > + struct gpio_aggregator *aggr = kzalloc(sizeof(*aggr), GFP_KERNEL); > > > + if (!aggr) > > > + return ERR_PTR(-ENOMEM); > > > + > > > + mutex_lock(&gpio_aggregator_lock); > > > + aggr->id = idr_alloc(&gpio_aggregator_idr, aggr, 0, 0, GFP_KERNEL); > > > + mutex_unlock(&gpio_aggregator_lock); > > > + > > > + if (aggr->id < 0) { > > > + kfree(aggr); > > > + return ERR_PTR(aggr->id); > > > + } > > > > The above more or less duplicates the existing code in > > new_device_store(). > > I'll factor out the common part and add new funcs gpio_alloc()/gpio_free(). > Please let me know if any objections. Thanks, that would be fine! Gr{oetje,eeting}s, Geert
diff --git a/drivers/gpio/gpio-aggregator.c b/drivers/gpio/gpio-aggregator.c index 570cd1ff8cc2..c63cf3067ce7 100644 --- a/drivers/gpio/gpio-aggregator.c +++ b/drivers/gpio/gpio-aggregator.c @@ -9,10 +9,14 @@ #include <linux/bitmap.h> #include <linux/bitops.h> +#include <linux/completion.h> +#include <linux/configfs.h> #include <linux/ctype.h> #include <linux/delay.h> #include <linux/idr.h> #include <linux/kernel.h> +#include <linux/list.h> +#include <linux/lockdep.h> #include <linux/mod_devicetable.h> #include <linux/module.h> #include <linux/mutex.h> @@ -34,11 +38,39 @@ */ struct gpio_aggregator { + struct config_group group; struct gpiod_lookup_table *lookups; struct platform_device *pdev; + struct mutex lock; + int id; + + /* Synchronize with probe */ + struct notifier_block bus_notifier; + struct completion probe_completion; + bool driver_bound; + + /* List of gpio_aggregator_line. Always added in order */ + struct list_head list_head; + char args[]; }; +struct gpio_aggregator_line { + struct config_group group; + struct gpio_aggregator *parent; + struct list_head entry; + + /* Line index within the aggregator device */ + int idx; + + /* GPIO chip label or line name */ + char *key; + /* Can be negative to indicate lookup by line name */ + int offset; + + enum gpio_lookup_flags flags; +}; + static DEFINE_MUTEX(gpio_aggregator_lock); /* protects idr */ static DEFINE_IDR(gpio_aggregator_idr); @@ -61,6 +93,97 @@ static int aggr_add_gpio(struct gpio_aggregator *aggr, const char *key, return 0; } +static int aggr_notifier_call(struct notifier_block *nb, unsigned long action, + void *data) +{ + struct gpio_aggregator *aggr; + struct device *dev = data; + + aggr = container_of(nb, struct gpio_aggregator, bus_notifier); + if (!device_match_name(dev, aggr->lookups->dev_id)) + return NOTIFY_DONE; + + switch (action) { + case BUS_NOTIFY_BOUND_DRIVER: + aggr->driver_bound = true; + break; + case BUS_NOTIFY_DRIVER_NOT_BOUND: + aggr->driver_bound = false; + break; + default: + return NOTIFY_DONE; + } + + complete(&aggr->probe_completion); + return NOTIFY_OK; +} + +static bool aggr_is_active(struct gpio_aggregator *aggr) +{ + lockdep_assert_held(&aggr->lock); + + return !!aggr->pdev && platform_get_drvdata(aggr->pdev); +} + +static size_t aggr_count_lines(struct gpio_aggregator *aggr) +{ + lockdep_assert_held(&aggr->lock); + + return list_count_nodes(&aggr->list_head); +} + +static struct gpio_aggregator_line *aggr_line_alloc( + struct gpio_aggregator *parent, int idx, char *key, int offset) +{ + struct gpio_aggregator_line *line; + + line = kzalloc(sizeof(*line), GFP_KERNEL); + if (!line) + return ERR_PTR(-ENOMEM); + + if (key) { + line->key = kstrdup(key, GFP_KERNEL); + if (!line->key) { + kfree(line); + return ERR_PTR(-ENOMEM); + } + } + + line->flags = GPIO_LOOKUP_FLAGS_DEFAULT; + line->parent = parent; + line->idx = idx; + line->offset = offset; + INIT_LIST_HEAD(&line->entry); + + return line; +} + +static void aggr_line_add(struct gpio_aggregator *aggr, + struct gpio_aggregator_line *line) +{ + struct gpio_aggregator_line *tmp; + + lockdep_assert_held(&aggr->lock); + + list_for_each_entry(tmp, &aggr->list_head, entry) { + if (WARN_ON(tmp->idx == line->idx)) + return; + if (tmp->idx > line->idx) { + list_add_tail(&line->entry, &tmp->entry); + return; + } + } + list_add_tail(&line->entry, &aggr->list_head); +} + +static void aggr_line_del(struct gpio_aggregator *aggr, + struct gpio_aggregator_line *line) +{ + lockdep_assert_held(&aggr->lock); + + list_del(&line->entry); +} + /* * GPIO Forwarder @@ -385,6 +508,411 @@ static struct gpiochip_fwd *gpiochip_fwd_create(struct device *dev, } +/* + * Configfs interface + */ + +static struct gpio_aggregator * +to_gpio_aggregator(struct config_item *item) +{ + struct config_group *group = to_config_group(item); + + return container_of(group, struct gpio_aggregator, group); +} + +static struct gpio_aggregator_line * +to_gpio_aggregator_line(struct config_item *item) +{ + struct config_group *group = to_config_group(item); + + return container_of(group, struct gpio_aggregator_line, group); +} + +static int aggr_activate(struct gpio_aggregator *aggr) +{ + struct platform_device_info pdevinfo; + struct gpio_aggregator_line *line; + struct platform_device *pdev; + unsigned int n = 0; + int ret = 0; + + if (aggr_count_lines(aggr) == 0) + return -EINVAL; + + aggr->lookups = kzalloc(struct_size(aggr->lookups, table, 1), + GFP_KERNEL); + if (!aggr->lookups) + return -ENOMEM; + + memset(&pdevinfo, 0, sizeof(pdevinfo)); + pdevinfo.name = DRV_NAME; + pdevinfo.id = aggr->id; + + /* The list is always sorted as new elements are inserted in order. */ + list_for_each_entry(line, &aggr->list_head, entry) { + /* + * - Either GPIO chip label or line name must be configured + * (i.e. line->key must be non-NULL) + * - Line directories must be named with sequential numeric + * suffixes starting from 0. (i.e. ./line0, ./line1, ...) + */ + if (!line->key || line->idx != n) { + ret = -EINVAL; + goto err_remove_lookups; + } + + if (line->offset < 0) + ret = aggr_add_gpio(aggr, line->key, U16_MAX, &n); + else + ret = aggr_add_gpio(aggr, line->key, line->offset, &n); + if (ret) + goto err_remove_lookups; + } + + aggr->lookups->dev_id = kasprintf(GFP_KERNEL, "%s.%d", DRV_NAME, aggr->id); + if (!aggr->lookups->dev_id) { + ret = -ENOMEM; + goto err_remove_lookups; + } + + gpiod_add_lookup_table(aggr->lookups); + + reinit_completion(&aggr->probe_completion); + aggr->driver_bound = false; + aggr->bus_notifier.notifier_call = aggr_notifier_call; + bus_register_notifier(&platform_bus_type, &aggr->bus_notifier); + + pdev = platform_device_register_full(&pdevinfo); + if (IS_ERR(pdev)) { + ret = PTR_ERR(pdev); + bus_unregister_notifier(&platform_bus_type, &aggr->bus_notifier); + goto err_remove_lookup_table; + } + + wait_for_completion(&aggr->probe_completion); + bus_unregister_notifier(&platform_bus_type, &aggr->bus_notifier); + + if (!aggr->driver_bound) { + ret = -ENXIO; + goto err_unregister_pdev; + } + + aggr->pdev = pdev; + return 0; + +err_unregister_pdev: + platform_device_unregister(pdev); +err_remove_lookup_table: + kfree(aggr->lookups->dev_id); + gpiod_remove_lookup_table(aggr->lookups); +err_remove_lookups: + kfree(aggr->lookups); + + return ret; +} + +static void aggr_deactivate(struct gpio_aggregator *aggr) +{ + platform_device_unregister(aggr->pdev); + gpiod_remove_lookup_table(aggr->lookups); + kfree(aggr->lookups->dev_id); + kfree(aggr->lookups); + aggr->pdev = NULL; +} + +static void aggr_lockup_configfs(struct gpio_aggregator *aggr, bool lock) +{ + struct configfs_subsystem *subsys = aggr->group.cg_subsys; + struct gpio_aggregator_line *line; + + /* + * The device only needs to depend on leaf lines. This is + * sufficient to lock up all the configfs entries that the + * instantiated, alive device depends on. + */ + list_for_each_entry(line, &aggr->list_head, entry) { + if (lock) + WARN_ON(configfs_depend_item_unlocked( + subsys, &line->group.cg_item)); + else + configfs_undepend_item_unlocked( + &line->group.cg_item); + } +} + +static ssize_t +gpio_aggr_line_key_show(struct config_item *item, char *page) +{ + struct gpio_aggregator_line *line = to_gpio_aggregator_line(item); + struct gpio_aggregator *aggr = line->parent; + + guard(mutex)(&aggr->lock); + + return sprintf(page, "%s\n", line->key ?: ""); +} + +static ssize_t +gpio_aggr_line_key_store(struct config_item *item, const char *page, + size_t count) +{ + struct gpio_aggregator_line *line = to_gpio_aggregator_line(item); + struct gpio_aggregator *aggr = line->parent; + + char *key __free(kfree) = kstrndup(skip_spaces(page), count, + GFP_KERNEL); + if (!key) + return -ENOMEM; + + strim(key); + + guard(mutex)(&aggr->lock); + + if (aggr_is_active(aggr)) + return -EBUSY; + + kfree(line->key); + line->key = no_free_ptr(key); + + return count; +} + +CONFIGFS_ATTR(gpio_aggr_line_, key); + +static ssize_t +gpio_aggr_line_offset_show(struct config_item *item, char *page) +{ + struct gpio_aggregator_line *line = to_gpio_aggregator_line(item); + struct gpio_aggregator *aggr = line->parent; + unsigned int offset; + + scoped_guard(mutex, &aggr->lock) + offset = line->offset; + + return sprintf(page, "%d\n", offset); +} + +static ssize_t +gpio_aggr_line_offset_store(struct config_item *item, const char *page, + size_t count) +{ + struct gpio_aggregator_line *line = to_gpio_aggregator_line(item); + struct gpio_aggregator *aggr = line->parent; + int offset, ret; + + ret = kstrtoint(page, 0, &offset); + if (ret) + return ret; + + /* + * Negative number here means: 'key' represents a line name to lookup. + * Non-negative means: 'key' represents the label of the chip with + * the 'offset' value representing the line within that chip. + * + * GPIOLIB uses the U16_MAX value to indicate lookup by line name so + * the greatest offset we can accept is (U16_MAX - 1). + */ + if (offset > (U16_MAX - 1)) + return -EINVAL; + + guard(mutex)(&aggr->lock); + + if (aggr_is_active(aggr)) + return -EBUSY; + + line->offset = offset; + + return count; +} + +CONFIGFS_ATTR(gpio_aggr_line_, offset); + +static struct configfs_attribute *gpio_aggr_line_attrs[] = { + &gpio_aggr_line_attr_key, + &gpio_aggr_line_attr_offset, + NULL +}; + +static ssize_t +gpio_aggr_device_live_show(struct config_item *item, char *page) +{ + struct gpio_aggregator *aggr = to_gpio_aggregator(item); + bool active; + + scoped_guard(mutex, &aggr->lock) + active = aggr_is_active(aggr); + + return sprintf(page, "%c\n", active ? '1' : '0'); +} + +static ssize_t +gpio_aggr_device_live_store(struct config_item *item, const char *page, + size_t count) +{ + struct gpio_aggregator *aggr = to_gpio_aggregator(item); + int ret = 0; + bool live; + + ret = kstrtobool(page, &live); + if (ret) + return ret; + + if (live) + aggr_lockup_configfs(aggr, true); + + scoped_guard(mutex, &aggr->lock) { + if (live == aggr_is_active(aggr)) + ret = -EPERM; + else if (live) + ret = aggr_activate(aggr); + else + aggr_deactivate(aggr); + } + + /* + * Undepend is required only if device disablement (live == 0) + * succeeds or if device enablement (live == 1) fails. + */ + if (live == !!ret) + aggr_lockup_configfs(aggr, false); + + return ret ?: count; +} + +CONFIGFS_ATTR(gpio_aggr_device_, live); + +static struct configfs_attribute *gpio_aggr_device_attrs[] = { + &gpio_aggr_device_attr_live, + NULL +}; + +static void +gpio_aggr_line_release(struct config_item *item) +{ + struct gpio_aggregator_line *line = to_gpio_aggregator_line(item); + struct gpio_aggregator *aggr = line->parent; + + guard(mutex)(&aggr->lock); + + aggr_line_del(aggr, line); + kfree(line->key); + kfree(line); +} + +static struct configfs_item_operations gpio_aggr_line_item_ops = { + .release = gpio_aggr_line_release, +}; + +static const struct config_item_type gpio_aggr_line_type = { + .ct_item_ops = &gpio_aggr_line_item_ops, + .ct_attrs = gpio_aggr_line_attrs, + .ct_owner = THIS_MODULE, +}; + +static void gpio_aggr_device_release(struct config_item *item) +{ + struct gpio_aggregator *aggr = to_gpio_aggregator(item); + + guard(mutex)(&aggr->lock); + + if (aggr_is_active(aggr)) + aggr_deactivate(aggr); + + mutex_destroy(&aggr->lock); + idr_remove(&gpio_aggregator_idr, aggr->id); + kfree(aggr); +} + +static struct configfs_item_operations gpio_aggr_device_item_ops = { + .release = gpio_aggr_device_release, +}; + +static struct config_group * +gpio_aggr_device_make_group(struct config_group *group, const char *name) +{ + struct gpio_aggregator *aggr = to_gpio_aggregator(&group->cg_item); + struct gpio_aggregator_line *line; + unsigned int idx; + int ret, nchar; + + ret = sscanf(name, "line%u%n", &idx, &nchar); + if (ret != 1 || nchar != strlen(name)) + return ERR_PTR(-EINVAL); + + guard(mutex)(&aggr->lock); + + if (aggr_is_active(aggr)) + return ERR_PTR(-EBUSY); + + list_for_each_entry(line, &aggr->list_head, entry) + if (line->idx == idx) + return ERR_PTR(-EINVAL); + + line = aggr_line_alloc(aggr, idx, NULL, -1); + if (!line) + return ERR_PTR(-ENOMEM); + + config_group_init_type_name(&line->group, name, &gpio_aggr_line_type); + + aggr_line_add(aggr, line); + + return &line->group; +} + +static struct configfs_group_operations gpio_aggr_device_group_ops = { + .make_group = gpio_aggr_device_make_group, +}; + +static const struct config_item_type gpio_aggr_device_type = { + .ct_group_ops = &gpio_aggr_device_group_ops, + .ct_item_ops = &gpio_aggr_device_item_ops, + .ct_attrs = gpio_aggr_device_attrs, + .ct_owner = THIS_MODULE, +}; + +static struct config_group * +gpio_aggr_make_group(struct config_group *group, const char *name) +{ + /* arg space is unneeded */ + struct gpio_aggregator *aggr = kzalloc(sizeof(*aggr), GFP_KERNEL); + if (!aggr) + return ERR_PTR(-ENOMEM); + + mutex_lock(&gpio_aggregator_lock); + aggr->id = idr_alloc(&gpio_aggregator_idr, aggr, 0, 0, GFP_KERNEL); + mutex_unlock(&gpio_aggregator_lock); + + if (aggr->id < 0) { + kfree(aggr); + return ERR_PTR(aggr->id); + } + + INIT_LIST_HEAD(&aggr->list_head); + mutex_init(&aggr->lock); + config_group_init_type_name(&aggr->group, name, &gpio_aggr_device_type); + init_completion(&aggr->probe_completion); + + return &aggr->group; +} + +static struct configfs_group_operations gpio_aggr_group_ops = { + .make_group = gpio_aggr_make_group, +}; + +static const struct config_item_type gpio_aggr_type = { + .ct_group_ops = &gpio_aggr_group_ops, + .ct_owner = THIS_MODULE, +}; + +static struct configfs_subsystem gpio_aggr_subsys = { + .su_group = { + .cg_item = { + .ci_namebuf = DRV_NAME, + .ci_type = &gpio_aggr_type, + }, + }, +}; + + /* * Sysfs interface */ @@ -622,12 +1150,31 @@ static void __exit gpio_aggregator_remove_all(void) static int __init gpio_aggregator_init(void) { - return platform_driver_register(&gpio_aggregator_driver); + int ret = 0; + + ret = platform_driver_register(&gpio_aggregator_driver); + if (ret) { + pr_err("Failed to register the platform driver: %d\n", ret); + return ret; + } + + config_group_init(&gpio_aggr_subsys.su_group); + mutex_init(&gpio_aggr_subsys.su_mutex); + ret = configfs_register_subsystem(&gpio_aggr_subsys); + if (ret) { + pr_err("Failed to register the '%s' configfs subsystem: %d\n", + gpio_aggr_subsys.su_group.cg_item.ci_namebuf, ret); + mutex_destroy(&gpio_aggr_subsys.su_mutex); + platform_driver_unregister(&gpio_aggregator_driver); + } + + return ret; } module_init(gpio_aggregator_init); static void __exit gpio_aggregator_exit(void) { + configfs_unregister_subsystem(&gpio_aggr_subsys); gpio_aggregator_remove_all(); platform_driver_unregister(&gpio_aggregator_driver); }
The existing sysfs 'new_device' interface has several limitations: * No way to determine when GPIO aggregator creation is complete. * No way to retrieve errors when creating a GPIO aggregator. * No way to trace a GPIO line of an aggregator back to its corresponding physical device. * The 'new_device' echo does not indicate which virtual gpiochip<N> was created. * No way to assign names to GPIO lines exported through an aggregator. Introduce the new configfs interface for gpio-aggregator to address these limitations. It provides a more streamlined, modern, and extensible configuration method. For backward compatibility, the 'new_device' interface and its behaviour is retained for now. This commit implements minimal functionalities: /config/gpio-aggregator/<name-of-your-choice>/ /config/gpio-aggregator/<name-of-your-choice>/live /config/gpio-aggregator/<name-of-your-choice>/<lineY>/ /config/gpio-aggregator/<name-of-your-choice>/<lineY>/key /config/gpio-aggregator/<name-of-your-choice>/<lineY>/offset Basic setup flow is: 1. Create a directory for a GPIO aggregator. 2. Create subdirectories for each line you want to instantiate. 3. In each line directory, configure the key and offset. The key/offset semantics are as follows: * If offset is >= 0: - key specifies the name of the chip this GPIO belongs to - offset specifies the line offset within that chip. * If offset is <0: - key needs to specify the GPIO line name. 4. Return to the aggregator's root directory and write '1' to the live attribute. For example, the command in the existing kernel doc: echo 'e6052000.gpio 19 e6050000.gpio 20-21' > new_device is equivalent to: mkdir /sys/kernel/config/gpio-aggregator/<custom-name> # Change <custom-name> to name of your choice (e.g. "aggr0") cd /sys/kernel/config/gpio-aggregator/<custom-name> mkdir line0 line1 line2 # Only "line<Y>" naming allowed. echo e6052000.gpio > line0/key echo 19 > line0/offset echo e6050000.gpio > line1/key echo 20 > line1/offset echo e6050000.gpio > line2/key echo 21 > line2/offset echo 1 > live Signed-off-by: Koichiro Den <koichiro.den@canonical.com> --- drivers/gpio/gpio-aggregator.c | 549 ++++++++++++++++++++++++++++++++- 1 file changed, 548 insertions(+), 1 deletion(-)