Commit 174d8e52 authored by Paul Cercueil's avatar Paul Cercueil

drm/ingenic: Alloc F0 and F1 DMA descriptors at once

Instead of calling dmam_alloc_coherent() once for each 4-bit DMA
hardware descriptor, we can have them both in a physical memory page, as
long as they are aligned to 16 bytes. This reduces memory consumption,
and will make it easier to add more DMA descriptors in the future.

Note that the old code would not create the F0 descriptor on SoCs that
don't support multiple planes. We don't care, because:
- we don't use more memory by allocating two descriptors instead of a
  single one;
- the only SoC that does not support multiple planes (JZ4740) still has
  two independent DMA channels, for an unknown reason.
Signed-off-by: default avatarPaul Cercueil <paul@crapouillou.net>
Reviewed-by: default avatarSam Ravnborg <sam@ravnborg.org>
Link: https://patchwork.freedesktop.org/patch/msgid/20200926170501.1109197-4-paul@crapouillou.net
parent 1677d31c
...@@ -45,7 +45,12 @@ struct ingenic_dma_hwdesc { ...@@ -45,7 +45,12 @@ struct ingenic_dma_hwdesc {
u32 addr; u32 addr;
u32 id; u32 id;
u32 cmd; u32 cmd;
} __packed; } __aligned(16);
struct ingenic_dma_hwdescs {
struct ingenic_dma_hwdesc hwdesc_f0;
struct ingenic_dma_hwdesc hwdesc_f1;
};
struct jz_soc_info { struct jz_soc_info {
bool needs_dev_clk; bool needs_dev_clk;
...@@ -68,8 +73,8 @@ struct ingenic_drm { ...@@ -68,8 +73,8 @@ struct ingenic_drm {
struct clk *lcd_clk, *pix_clk; struct clk *lcd_clk, *pix_clk;
const struct jz_soc_info *soc_info; const struct jz_soc_info *soc_info;
struct ingenic_dma_hwdesc *dma_hwdesc_f0, *dma_hwdesc_f1; struct ingenic_dma_hwdescs *dma_hwdescs;
dma_addr_t dma_hwdesc_phys_f0, dma_hwdesc_phys_f1; dma_addr_t dma_hwdescs_phys;
bool panel_is_sharp; bool panel_is_sharp;
bool no_vblank; bool no_vblank;
...@@ -546,9 +551,9 @@ static void ingenic_drm_plane_atomic_update(struct drm_plane *plane, ...@@ -546,9 +551,9 @@ static void ingenic_drm_plane_atomic_update(struct drm_plane *plane,
cpp = state->fb->format->cpp[0]; cpp = state->fb->format->cpp[0];
if (priv->soc_info->has_osd && plane->type == DRM_PLANE_TYPE_OVERLAY) if (priv->soc_info->has_osd && plane->type == DRM_PLANE_TYPE_OVERLAY)
hwdesc = priv->dma_hwdesc_f0; hwdesc = &priv->dma_hwdescs->hwdesc_f0;
else else
hwdesc = priv->dma_hwdesc_f1; hwdesc = &priv->dma_hwdescs->hwdesc_f1;
hwdesc->addr = addr; hwdesc->addr = addr;
hwdesc->cmd = JZ_LCD_CMD_EOF_IRQ | (width * height * cpp / 4); hwdesc->cmd = JZ_LCD_CMD_EOF_IRQ | (width * height * cpp / 4);
...@@ -856,6 +861,7 @@ static int ingenic_drm_bind(struct device *dev, bool has_components) ...@@ -856,6 +861,7 @@ static int ingenic_drm_bind(struct device *dev, bool has_components)
void __iomem *base; void __iomem *base;
long parent_rate; long parent_rate;
unsigned int i, clone_mask = 0; unsigned int i, clone_mask = 0;
dma_addr_t dma_hwdesc_phys_f0, dma_hwdesc_phys_f1;
int ret, irq; int ret, irq;
soc_info = of_device_get_match_data(dev); soc_info = of_device_get_match_data(dev);
...@@ -930,26 +936,25 @@ static int ingenic_drm_bind(struct device *dev, bool has_components) ...@@ -930,26 +936,25 @@ static int ingenic_drm_bind(struct device *dev, bool has_components)
return PTR_ERR(priv->pix_clk); return PTR_ERR(priv->pix_clk);
} }
priv->dma_hwdesc_f1 = dmam_alloc_coherent(dev, sizeof(*priv->dma_hwdesc_f1), priv->dma_hwdescs = dmam_alloc_coherent(dev,
&priv->dma_hwdesc_phys_f1, sizeof(*priv->dma_hwdescs),
&priv->dma_hwdescs_phys,
GFP_KERNEL); GFP_KERNEL);
if (!priv->dma_hwdesc_f1) if (!priv->dma_hwdescs)
return -ENOMEM; return -ENOMEM;
priv->dma_hwdesc_f1->next = priv->dma_hwdesc_phys_f1;
priv->dma_hwdesc_f1->id = 0xf1;
if (priv->soc_info->has_osd) { /* Configure DMA hwdesc for foreground0 plane */
priv->dma_hwdesc_f0 = dmam_alloc_coherent(dev, dma_hwdesc_phys_f0 = priv->dma_hwdescs_phys
sizeof(*priv->dma_hwdesc_f0), + offsetof(struct ingenic_dma_hwdescs, hwdesc_f0);
&priv->dma_hwdesc_phys_f0, priv->dma_hwdescs->hwdesc_f0.next = dma_hwdesc_phys_f0;
GFP_KERNEL); priv->dma_hwdescs->hwdesc_f0.id = 0xf0;
if (!priv->dma_hwdesc_f0)
return -ENOMEM;
priv->dma_hwdesc_f0->next = priv->dma_hwdesc_phys_f0; /* Configure DMA hwdesc for foreground1 plane */
priv->dma_hwdesc_f0->id = 0xf0; dma_hwdesc_phys_f1 = priv->dma_hwdescs_phys
} + offsetof(struct ingenic_dma_hwdescs, hwdesc_f1);
priv->dma_hwdescs->hwdesc_f1.next = dma_hwdesc_phys_f1;
priv->dma_hwdescs->hwdesc_f1.id = 0xf1;
if (soc_info->has_osd) if (soc_info->has_osd)
priv->ipu_plane = drm_plane_from_index(drm, 0); priv->ipu_plane = drm_plane_from_index(drm, 0);
...@@ -1101,8 +1106,8 @@ static int ingenic_drm_bind(struct device *dev, bool has_components) ...@@ -1101,8 +1106,8 @@ static int ingenic_drm_bind(struct device *dev, bool has_components)
} }
/* Set address of our DMA descriptor chain */ /* Set address of our DMA descriptor chain */
regmap_write(priv->map, JZ_REG_LCD_DA0, priv->dma_hwdesc_phys_f0); regmap_write(priv->map, JZ_REG_LCD_DA0, dma_hwdesc_phys_f0);
regmap_write(priv->map, JZ_REG_LCD_DA1, priv->dma_hwdesc_phys_f1); regmap_write(priv->map, JZ_REG_LCD_DA1, dma_hwdesc_phys_f1);
/* Enable OSD if available */ /* Enable OSD if available */
if (soc_info->has_osd) if (soc_info->has_osd)
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment