regmap-irq: Add handle_mask_sync() callback
authorWilliam Breathitt Gray <william.gray@linaro.org>
Tue, 22 Nov 2022 07:10:59 +0000 (02:10 -0500)
committerMark Brown <broonie@kernel.org>
Fri, 9 Dec 2022 17:39:33 +0000 (17:39 +0000)
Provide a public callback handle_mask_sync() that drivers can use when
they have more complex IRQ masking logic. The default implementation is
regmap_irq_handle_mask_sync(), used if the chip doesn't provide its own
callback.

Cc: Mark Brown <broonie@kernel.org>
Signed-off-by: William Breathitt Gray <william.gray@linaro.org>
Link: https://lore.kernel.org/r/e083474b3d467a86e6cb53da8072de4515bd6276.1669100542.git.william.gray@linaro.org
Signed-off-by: Mark Brown <broonie@kernel.org>
drivers/base/regmap/regmap-irq.c
include/linux/regmap.h

index 4ef9488..968681f 100644 (file)
@@ -115,12 +115,20 @@ static void regmap_irq_sync_unlock(struct irq_data *data)
         */
        for (i = 0; i < d->chip->num_regs; i++) {
                if (d->mask_base) {
-                       reg = d->get_irq_reg(d, d->mask_base, i);
-                       ret = regmap_update_bits(d->map, reg,
-                                       d->mask_buf_def[i], d->mask_buf[i]);
-                       if (ret)
-                               dev_err(d->map->dev, "Failed to sync masks in %x\n",
-                                       reg);
+                       if (d->chip->handle_mask_sync)
+                               d->chip->handle_mask_sync(d->map, i,
+                                                         d->mask_buf_def[i],
+                                                         d->mask_buf[i],
+                                                         d->chip->irq_drv_data);
+                       else {
+                               reg = d->get_irq_reg(d, d->mask_base, i);
+                               ret = regmap_update_bits(d->map, reg,
+                                               d->mask_buf_def[i],
+                                               d->mask_buf[i]);
+                               if (ret)
+                                       dev_err(d->map->dev, "Failed to sync masks in %x\n",
+                                               reg);
+                       }
                }
 
                if (d->unmask_base) {
@@ -917,13 +925,23 @@ int regmap_add_irq_chip_fwnode(struct fwnode_handle *fwnode,
                d->mask_buf[i] = d->mask_buf_def[i];
 
                if (d->mask_base) {
-                       reg = d->get_irq_reg(d, d->mask_base, i);
-                       ret = regmap_update_bits(d->map, reg,
-                                       d->mask_buf_def[i], d->mask_buf[i]);
-                       if (ret) {
-                               dev_err(map->dev, "Failed to set masks in 0x%x: %d\n",
-                                       reg, ret);
-                               goto err_alloc;
+                       if (chip->handle_mask_sync) {
+                               ret = chip->handle_mask_sync(d->map, i,
+                                                            d->mask_buf_def[i],
+                                                            d->mask_buf[i],
+                                                            chip->irq_drv_data);
+                               if (ret)
+                                       goto err_alloc;
+                       } else {
+                               reg = d->get_irq_reg(d, d->mask_base, i);
+                               ret = regmap_update_bits(d->map, reg,
+                                               d->mask_buf_def[i],
+                                               d->mask_buf[i]);
+                               if (ret) {
+                                       dev_err(map->dev, "Failed to set masks in 0x%x: %d\n",
+                                               reg, ret);
+                                       goto err_alloc;
+                               }
                        }
                }
 
index 5a56fc5..a3bc695 100644 (file)
@@ -1580,6 +1580,8 @@ struct regmap_irq_chip_data;
  *                  before regmap_irq_handler process the interrupts.
  * @handle_post_irq: Driver specific callback to handle interrupt from device
  *                  after handling the interrupts in regmap_irq_handler().
+ * @handle_mask_sync: Callback used to handle IRQ mask syncs. The index will be
+ *                   in the range [0, num_regs)
  * @set_type_virt:   Driver specific callback to extend regmap_irq_set_type()
  *                  and configure virt regs. Deprecated, use @set_type_config
  *                  callback and config registers instead.
@@ -1641,6 +1643,9 @@ struct regmap_irq_chip {
 
        int (*handle_pre_irq)(void *irq_drv_data);
        int (*handle_post_irq)(void *irq_drv_data);
+       int (*handle_mask_sync)(struct regmap *map, int index,
+                               unsigned int mask_buf_def,
+                               unsigned int mask_buf, void *irq_drv_data);
        int (*set_type_virt)(unsigned int **buf, unsigned int type,
                             unsigned long hwirq, int reg);
        int (*set_type_config)(unsigned int **buf, unsigned int type,