]> git.itanic.dy.fi Git - linux-stable/commitdiff
dmaengine: dw-edma: Simplify debugfs context CSRs init procedure
authorSerge Semin <Sergey.Semin@baikalelectronics.ru>
Fri, 13 Jan 2023 17:13:57 +0000 (20:13 +0300)
committerLorenzo Pieralisi <lpieralisi@kernel.org>
Fri, 27 Jan 2023 16:15:33 +0000 (17:15 +0100)
DW eDMA v4.70a and older have the read and write channels context CSRs
indirectly accessible, which means CSRs like Channel Control, Xfer size,
SAR, DAR and LLP address are accessed at a fixed MMIO address, with their
reference to the corresponding channel determined by the Viewport CSR. To
have a coherent access to these registers the CSR IOs are supposed to be
protected with a spinlock. DW eDMA v4.80a and newer normally have unrolled
Read/Write channel context registers, with these CSRs directly mapped in
the controller MMIO space.

Both normal and viewport-based registers are exposed via debugfs nodes, and
the original algorithm was based on the unrolled CSRs mapping and
recalculated the viewport addresses when required. This is unscalable (it
only supports a platform with a single eDMA since a base address is
statically preserved) and also needlessly overcomplicated (it loops over
all Rd/Wr context addresses and recalculates the viewport base address on
each debugfs node access).

Simplify the algorithm by adding the channel ID and its direction fields in
the eDMA debugfs node descriptor. These new fields can be used to find a
CSR offset in the channel register space. The DW eDMA debugfs node getter
will also use them to activate the respective context CSRs viewport before
reading data from the specified register. For the unrolled CSR mapping, no
spinlock or viewport activation is needed.

Note: this replaces some REGISTER() uses with CTX_REGISTER(), which avoids
an implicit dependency on a local variable name.  The same problem with the
rest of the macro will be fixed in the next commit.

Link: https://lore.kernel.org/r/20230113171409.30470-16-Sergey.Semin@baikalelectronics.ru
Tested-by: Manivannan Sadhasivam <manivannan.sadhasivam@linaro.org>
Signed-off-by: Serge Semin <Sergey.Semin@baikalelectronics.ru>
Signed-off-by: Lorenzo Pieralisi <lpieralisi@kernel.org>
Signed-off-by: Bjorn Helgaas <bhelgaas@google.com>
Reviewed-by: Manivannan Sadhasivam <manivannan.sadhasivam@linaro.org>
Acked-by: Vinod Koul <vkoul@kernel.org>
drivers/dma/dw-edma/dw-edma-v0-debugfs.c

index 7bb3363b40e4b8a75244534d89a4a34d39a71bc4..fa63d1dd75b2c19911da47133ef3b324580c9243 100644 (file)
 
 #define REGS_ADDR(name) \
        ((void __iomem *)&regs->name)
+
+#define REGS_CH_ADDR(name, _dir, _ch)                                         \
+       ({                                                                     \
+               struct dw_edma_v0_ch_regs __iomem *__ch_regs;                  \
+                                                                              \
+               if ((dw)->chip->mf == EDMA_MF_EDMA_LEGACY)                     \
+                       __ch_regs = &regs->type.legacy.ch;                     \
+               else if (_dir == EDMA_DIR_READ)                                \
+                       __ch_regs = &regs->type.unroll.ch[_ch].rd;             \
+               else                                                           \
+                       __ch_regs = &regs->type.unroll.ch[_ch].wr;             \
+                                                                              \
+               (void __iomem *)&__ch_regs->name;                              \
+       })
+
 #define REGISTER(name) \
        { #name, REGS_ADDR(name) }
 
+#define CTX_REGISTER(name, dir, ch) \
+       { #name, REGS_CH_ADDR(name, dir, ch), dir, ch }
+
 #define WR_REGISTER(name) \
        { #name, REGS_ADDR(wr_##name) }
 #define RD_REGISTER(name) \
 static struct dw_edma                          *dw;
 static struct dw_edma_v0_regs                  __iomem *regs;
 
-static struct {
-       void                                    __iomem *start;
-       void                                    __iomem *end;
-} lim[2][EDMA_V0_MAX_NR_CH];
-
 struct dw_edma_debugfs_entry {
        const char                              *name;
        void __iomem                            *reg;
+       enum dw_edma_dir                        dir;
+       u16                                     ch;
 };
 
 static int dw_edma_debugfs_u32_get(void *data, u64 *val)
@@ -58,33 +73,16 @@ static int dw_edma_debugfs_u32_get(void *data, u64 *val)
 
        if (dw->chip->mf == EDMA_MF_EDMA_LEGACY &&
            reg >= (void __iomem *)&regs->type.legacy.ch) {
-               void __iomem *ptr = &regs->type.legacy.ch;
-               u32 viewport_sel = 0;
                unsigned long flags;
-               u16 ch;
-
-               for (ch = 0; ch < dw->wr_ch_cnt; ch++)
-                       if (lim[0][ch].start >= reg && reg < lim[0][ch].end) {
-                               ptr += (reg - lim[0][ch].start);
-                               goto legacy_sel_wr;
-                       }
-
-               for (ch = 0; ch < dw->rd_ch_cnt; ch++)
-                       if (lim[1][ch].start >= reg && reg < lim[1][ch].end) {
-                               ptr += (reg - lim[1][ch].start);
-                               goto legacy_sel_rd;
-                       }
-
-               return 0;
-legacy_sel_rd:
-               viewport_sel = BIT(31);
-legacy_sel_wr:
-               viewport_sel |= FIELD_PREP(EDMA_V0_VIEWPORT_MASK, ch);
+               u32 viewport_sel;
+
+               viewport_sel = entry->dir == EDMA_DIR_READ ? BIT(31) : 0;
+               viewport_sel |= FIELD_PREP(EDMA_V0_VIEWPORT_MASK, entry->ch);
 
                raw_spin_lock_irqsave(&dw->lock, flags);
 
                writel(viewport_sel, &regs->type.legacy.viewport_sel);
-               *val = readl(ptr);
+               *val = readl(reg);
 
                raw_spin_unlock_irqrestore(&dw->lock, flags);
        } else {
@@ -114,19 +112,19 @@ static void dw_edma_debugfs_create_x32(const struct dw_edma_debugfs_entry ini[],
        }
 }
 
-static void dw_edma_debugfs_regs_ch(struct dw_edma_v0_ch_regs __iomem *regs,
+static void dw_edma_debugfs_regs_ch(enum dw_edma_dir dir, u16 ch,
                                    struct dentry *dent)
 {
-       const struct dw_edma_debugfs_entry debugfs_regs[] = {
-               REGISTER(ch_control1),
-               REGISTER(ch_control2),
-               REGISTER(transfer_size),
-               REGISTER(sar.lsb),
-               REGISTER(sar.msb),
-               REGISTER(dar.lsb),
-               REGISTER(dar.msb),
-               REGISTER(llp.lsb),
-               REGISTER(llp.msb),
+       struct dw_edma_debugfs_entry debugfs_regs[] = {
+               CTX_REGISTER(ch_control1, dir, ch),
+               CTX_REGISTER(ch_control2, dir, ch),
+               CTX_REGISTER(transfer_size, dir, ch),
+               CTX_REGISTER(sar.lsb, dir, ch),
+               CTX_REGISTER(sar.msb, dir, ch),
+               CTX_REGISTER(dar.lsb, dir, ch),
+               CTX_REGISTER(dar.msb, dir, ch),
+               CTX_REGISTER(llp.lsb, dir, ch),
+               CTX_REGISTER(llp.msb, dir, ch),
        };
        int nr_entries;
 
@@ -191,10 +189,7 @@ static void dw_edma_debugfs_regs_wr(struct dentry *dent)
 
                ch_dent = debugfs_create_dir(name, regs_dent);
 
-               dw_edma_debugfs_regs_ch(&regs->type.unroll.ch[i].wr, ch_dent);
-
-               lim[0][i].start = &regs->type.unroll.ch[i].wr;
-               lim[0][i].end = &regs->type.unroll.ch[i].padding_1[0];
+               dw_edma_debugfs_regs_ch(EDMA_DIR_WRITE, i, ch_dent);
        }
 }
 
@@ -256,10 +251,7 @@ static void dw_edma_debugfs_regs_rd(struct dentry *dent)
 
                ch_dent = debugfs_create_dir(name, regs_dent);
 
-               dw_edma_debugfs_regs_ch(&regs->type.unroll.ch[i].rd, ch_dent);
-
-               lim[1][i].start = &regs->type.unroll.ch[i].rd;
-               lim[1][i].end = &regs->type.unroll.ch[i].padding_2[0];
+               dw_edma_debugfs_regs_ch(EDMA_DIR_READ, i, ch_dent);
        }
 }