diff mbox series

gpio: sim: added configfs option for static base

Message ID 4af4f6c5-d7da-4735-9ef5-ee3c34f7eae6@cyberdanube.com
State New
Headers show
Series gpio: sim: added configfs option for static base | expand

Commit Message

Sebastian Dietz Feb. 28, 2025, 12:46 p.m. UTC
To replicate gpio mappings of systems it is sometimes needed to have
the base at static values.

base is treated as unsigned as there doesn't happen to be a
fwnode_property_read_s32().

Signed-off-by: Sebastian Dietz <s.dietz@cyberdanube.com>
---
 drivers/gpio/gpio-sim.c | 53 +++++++++++++++++++++++++++++++++++++++--
 1 file changed, 51 insertions(+), 2 deletions(-)

Comments

Bartosz Golaszewski March 3, 2025, 1:31 p.m. UTC | #1
On Fri, Feb 28, 2025 at 10:02 PM Sebastian Dietz
<s.dietz@cyberdanube.com> wrote:
>
> On 28.02.25 16:43, Bartosz Golaszewski wrote:
> > On Fri, Feb 28, 2025 at 2:54 PM Sebastian Dietz <s.dietz@cyberdanube.com> wrote:
> >>
> >> On 28.02.25 14:22, Bartosz Golaszewski wrote:
> >>> On Fri, Feb 28, 2025 at 1:46 PM Sebastian Dietz <s.dietz@cyberdanube.com> wrote:
> >>>>
> >>>> To replicate gpio mappings of systems it is sometimes needed to have
> >>>> the base at static values.
> >>>>
> >>>
> >>> Can you give me more info on why you'd need that? Static base is
> >>> largely a legacy and deprecated feature, there's ongoing effort to
> >>> remove it from the kernel.
> >>>
> >>>> base is treated as unsigned as there doesn't happen to be a
> >>>> fwnode_property_read_s32().
> >>>>
> >>>
> >>> Ha! That's interesting, I wonder why that is. We do have signed
> >>> variants for OF-specific properties.
> >>>
> >>> Bart
> >>
> >> We are building digital twins for embedded devices for security research. The
> >> firmware of these devices often export static gpio pins which we simulate
> >> using gpio-sim. With this patch we are able to satisfy these conditions.
> >>
> >> While the feature may be deprecated, i would argue that it makes sense and
> >> fits the nature of a simulator to be able to configure it manually.
> >>
> >> BR,
> >> Sebastian
> >
> > What kind of digital twins? Using qemu? In any case - I really dislike
> > the idea of extending the configfs interface of gpio-sim with an
> > attribute to support an option that we're actively trying to remove
> > from GPIO core. Unless you can give me a really convincing argument, I
> > will allow myself to use my maintainers' right to NAK this one.
> >
> > Bart
>
> Exactly, we are analysing the firmware and re-host it in a qemu instance with a
> newer kernel.
>
> First of all thanks for giving me the oppertunity to pitch the option.
>
> Gpio-sim provides a way to create chips for testing purposes. Some embedded
> device developers still rely on sysfs. While the ABI is certainly obsolete and
> not actively developed, it is still actively maintained - Which means that in
> real-world testing scenarios, it remains relevant. Our experience has shown
> that these firmware images are often build with hardcoded sysfs paths in mind
> and do not use the character device interface.
>

My experience is that even when using drivers without a hardcoded GPIO
base, people tend to rely on specific sysfs paths just because they
tend to remain the same unless some dynamic GPIO expanders come into
play. This is of course dangerous.

> This feature wouldn't encourage new use of static bases - it would just make
> testing existing setups less of a headache.
>

This is not what I mean. We want to remove the "base" attribute from
struct gpio_chip. If we expose it in gpio-sim in its supposedly stable
configfs interface, then we'll implicitly agree to keep it around
forever. I don't want to do it.

> Given that gpio-sim is meant for testing, shouldn't it provide the flexibility
> to test in this scenarios?
>

gpio-sim is for testing the GPIO uAPI interface and any user-space
tools. Maybe you should just simulate whatever device you need
directly in qemu?

Bartosz
diff mbox series

Patch

diff --git a/drivers/gpio/gpio-sim.c b/drivers/gpio/gpio-sim.c
index b6c230fab840..9a57e1d46503 100644
--- a/drivers/gpio/gpio-sim.c
+++ b/drivers/gpio/gpio-sim.c
@@ -38,7 +38,7 @@ 
 #include <linux/types.h>
 
 #define GPIO_SIM_NGPIO_MAX	1024
-#define GPIO_SIM_PROP_MAX	4 /* Max 3 properties + sentinel. */
+#define GPIO_SIM_PROP_MAX	5 /* Max 4 properties + sentinel. */
 #define GPIO_SIM_NUM_ATTRS	3 /* value, pull and sentinel */
 
 static DEFINE_IDA(gpio_sim_ida);
@@ -419,15 +419,21 @@  static int gpio_sim_add_bank(struct fwnode_handle *swnode, struct device *dev)
 	struct gpio_chip *gc;
 	const char *label;
 	u32 num_lines;
+	u32 base;
 	int ret;
 
 	ret = fwnode_property_read_u32(swnode, "ngpios", &num_lines);
 	if (ret)
 		return ret;
 
+
 	if (num_lines > GPIO_SIM_NGPIO_MAX)
 		return -ERANGE;
 
+	ret = fwnode_property_read_u32(swnode, "base", &base);
+	if (ret)
+		return ret;
+
 	ret = fwnode_property_read_string(swnode, "gpio-sim,label", &label);
 	if (ret) {
 		label = devm_kasprintf(dev, GFP_KERNEL, "%s:%pfwP",
@@ -474,7 +480,7 @@  static int gpio_sim_add_bank(struct fwnode_handle *swnode, struct device *dev)
 		return ret;
 
 	gc = &chip->gc;
-	gc->base = -1;
+	gc->base = base;
 	gc->ngpio = num_lines;
 	gc->label = label;
 	gc->owner = THIS_MODULE;
@@ -629,6 +635,9 @@  struct gpio_sim_bank {
 	struct list_head siblings;
 
 	char *label;
+
+	//base is treated as unsigned as there is no read_prop_s32
+	unsigned int base;
 	unsigned int num_lines;
 
 	struct list_head line_list;
@@ -885,6 +894,7 @@  gpio_sim_make_bank_swnode(struct gpio_sim_bank *bank,
 
 	memset(properties, 0, sizeof(properties));
 
+	properties[prop_idx++] = PROPERTY_ENTRY_U32("base", bank->base);
 	properties[prop_idx++] = PROPERTY_ENTRY_U32("ngpios", bank->num_lines);
 
 	if (gpio_sim_bank_has_label(bank))
@@ -1202,10 +1212,48 @@  gpio_sim_bank_config_num_lines_store(struct config_item *item,
 
 CONFIGFS_ATTR(gpio_sim_bank_config_, num_lines);
 
+static ssize_t
+gpio_sim_bank_config_base_show(struct config_item *item, char *page)
+{
+	struct gpio_sim_bank *bank = to_gpio_sim_bank(item);
+	struct gpio_sim_device *dev = gpio_sim_bank_get_device(bank);
+
+	guard(mutex)(&dev->lock);
+
+	return sprintf(page, "%i\n", bank->base);
+}
+
+static ssize_t
+gpio_sim_bank_config_base_store(struct config_item *item,
+				     const char *page, size_t count)
+{
+	struct gpio_sim_bank *bank = to_gpio_sim_bank(item);
+	struct gpio_sim_device *dev = gpio_sim_bank_get_device(bank);
+	unsigned int base;
+	int ret;
+
+	ret = kstrtoint(page, 0, &base);
+	if (ret)
+		return ret;
+
+
+	guard(mutex)(&dev->lock);
+
+	if (gpio_sim_device_is_live(dev))
+		return -EBUSY;
+
+	bank->base = base;
+
+	return count;
+}
+
+CONFIGFS_ATTR(gpio_sim_bank_config_, base);
+
 static struct configfs_attribute *gpio_sim_bank_config_attrs[] = {
 	&gpio_sim_bank_config_attr_chip_name,
 	&gpio_sim_bank_config_attr_label,
 	&gpio_sim_bank_config_attr_num_lines,
+	&gpio_sim_bank_config_attr_base,
 	NULL
 };
 
@@ -1505,6 +1553,7 @@  gpio_sim_device_config_make_bank_group(struct config_group *group,
 	config_group_init_type_name(&bank->group, name,
 				    &gpio_sim_bank_config_group_type);
 	bank->num_lines = 1;
+	bank->base = -1;
 	bank->parent = dev;
 	INIT_LIST_HEAD(&bank->line_list);
 	list_add_tail(&bank->siblings, &dev->bank_list);