]> git.ipfire.org Git - thirdparty/linux.git/commitdiff
gpio: rockchip: teardown bugs and resource leaks
authorMarco Scardovi <scardracs@disroot.org>
Tue, 26 May 2026 17:02:46 +0000 (19:02 +0200)
committerBartosz Golaszewski <bartosz.golaszewski@oss.qualcomm.com>
Thu, 28 May 2026 13:23:40 +0000 (15:23 +0200)
Address several teardown issues and resource leaks in the driver's remove
path and error handling:

1. Debounce clock reference leak: The debounce clock (bank->db_clk) is
   obtained using of_clk_get() which increments the clock's reference
   count, but clk_put() is never called. Register a devm action to
   cleanly release it on unbind. Note that of_clk_get(..., 1) remains
   necessary over devm_clk_get() because the DT binding does not define
   clock-names, precluding name-based lookup.

2. Unregistered chained IRQ handler: The chained IRQ handler is not
   disconnected in remove(). If a stray interrupt fires after the driver
   is removed, the kernel attempts to execute a stale handler, leading
   to a panic. Fix this by clearing the handler in remove().

3. IRQ domain leak: The linear IRQ domain and its generic chips are
   allocated manually during probe but never removed. Remove the IRQ
   domain during driver teardown to free the associated generic chips
   and mappings.

Fixes: 936ee2675eee ("gpio/rockchip: add driver for rockchip gpio")
Assisted-by: Antigravity:gemini-3.5-flash
Signed-off-by: Marco Scardovi <scardracs@disroot.org>
Link: https://patch.msgid.link/20260526171050.12785-3-scardracs@disroot.org
[Bartosz: don't emit an error message on devres allocation failure]
Signed-off-by: Bartosz Golaszewski <bartosz.golaszewski@oss.qualcomm.com>
drivers/gpio/gpio-rockchip.c

index 33580093a4e76a53e7fb25ff0fa53bba66da71b1..bc97d5d5d3296fe06ba0d478ea92855d9d0d0336 100644 (file)
@@ -638,10 +638,17 @@ fail:
        return ret;
 }
 
+static void rockchip_clk_put(void *data)
+{
+       struct clk *clk = data;
+
+       clk_put(clk);
+}
+
 static int rockchip_get_bank_data(struct rockchip_pin_bank *bank)
 {
        struct resource res;
-       int id = 0;
+       int id = 0, ret;
 
        if (of_address_to_resource(bank->of_node, 0, &res)) {
                dev_err(bank->dev, "cannot find IO resource for bank\n");
@@ -673,6 +680,11 @@ static int rockchip_get_bank_data(struct rockchip_pin_bank *bank)
                        dev_err(bank->dev, "cannot find debounce clk\n");
                        return -EINVAL;
                }
+
+               ret = devm_add_action_or_reset(bank->dev, rockchip_clk_put,
+                                              bank->db_clk);
+               if (ret)
+                       return ret;
                break;
        case GPIO_TYPE_V1:
                bank->gpio_regs = &gpio_regs_v1;
@@ -789,6 +801,9 @@ static void rockchip_gpio_remove(struct platform_device *pdev)
 {
        struct rockchip_pin_bank *bank = platform_get_drvdata(pdev);
 
+       irq_set_chained_handler_and_data(bank->irq, NULL, NULL);
+       if (bank->domain)
+               irq_domain_remove(bank->domain);
        gpiochip_remove(&bank->gpio_chip);
 }