Re: [PATCH 2/3] drm/tegra: Support setting the EMC clock

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 






On Mon, May 26, 2014 at 2:07 AM, Lucas Stach <l.stach@xxxxxxxxxxxxxx> wrote:
Am Freitag, den 23.05.2014, 18:58 -0700 schrieb Stéphane Marchesin:
> The current code doesn't enable the EMC clock, without which the
> display cannot function, so let's enable this clock. We also need a
> bit of code to pick the right frequency for the EMC clock depending
> on the current video mode settings.
>
That's not the right way to do it. The DRM driver has no business
controlling the EMC clock directly. This should be done through a real
EMC driver plus some kind of bus QoS, where DC is just one client.

I thought about it but didn't see another consumer in upstream kernels. Who are the other consumers of EMC?

Stéphane


Regards,
Lucas

> Signed-off-by: Stéphane Marchesin <marcheu@xxxxxxxxxxxx>
> ---
>  drivers/gpu/drm/tegra/dc.c  | 61 ++++++++++++++++++++++++++++++++++++++++++++-
>  drivers/gpu/drm/tegra/drm.h |  1 +
>  2 files changed, 61 insertions(+), 1 deletion(-)
>
> diff --git a/drivers/gpu/drm/tegra/dc.c b/drivers/gpu/drm/tegra/dc.c
> index edb871d..f398dfb 100644
> --- a/drivers/gpu/drm/tegra/dc.c
> +++ b/drivers/gpu/drm/tegra/dc.c
> @@ -325,6 +325,9 @@ static void tegra_crtc_disable(struct drm_crtc *crtc)
>       }
>
>       drm_vblank_off(drm, dc->pipe);
> +
> +     if (dc->emc_clk)
> +             clk_set_rate(dc->emc_clk, 0);
>  }
>
>  static bool tegra_crtc_mode_fixup(struct drm_crtc *crtc,
> @@ -640,6 +643,50 @@ unsigned int tegra_dc_format(uint32_t format)
>       return WIN_COLOR_DEPTH_B8G8R8A8;
>  }
>
> +static unsigned long tegra_emc_bw_to_freq_req(unsigned long bw)
> +{
> +     int bytes_per_emc_clock;
> +
> +     if (of_machine_is_compatible("nvidia,tegra124"))
> +             bytes_per_emc_clock = 16;
> +     else
> +             bytes_per_emc_clock = 8;
> +
> +     return (bw + bytes_per_emc_clock - 1) / bytes_per_emc_clock;
> +}
> +
> +#define EMC_FREQ_CUTOFF_USE_130_PERCENT 100000000UL
> +#define EMC_FREQ_CUTOFF_USE_140_PERCENT 50000000UL
> +
> +static int tegra_dc_program_bandwidth(struct tegra_dc *dc,
> +                                   struct drm_display_mode *mode,
> +                                   struct tegra_dc_window *window)
> +{
> +     unsigned long bandwidth = mode->clock * window->bits_per_pixel / 8;
> +     unsigned long freq;
> +     struct clk *emc_master;
> +
> +     if (!dc->emc_clk)
> +             return 0;
> +
> +     emc_master = clk_get_parent(dc->emc_clk);
> +     freq = tegra_emc_bw_to_freq_req(bandwidth) * 1000;
> +     freq = clk_round_rate(emc_master, freq);
> +
> +     /* XXX: Add safety margins for DVFS */
> +
> +     if (freq < EMC_FREQ_CUTOFF_USE_140_PERCENT)
> +             bandwidth += 4 * bandwidth / 10;
> +     else if (freq < EMC_FREQ_CUTOFF_USE_130_PERCENT)
> +             bandwidth += 3 * bandwidth / 10;
> +     else
> +             bandwidth += bandwidth / 10;
> +
> +     freq = tegra_emc_bw_to_freq_req(bandwidth) * 1000;
> +
> +     return clk_set_rate(dc->emc_clk, freq);
> +}
> +
>  static int tegra_crtc_mode_set(struct drm_crtc *crtc,
>                              struct drm_display_mode *mode,
>                              struct drm_display_mode *adjusted,
> @@ -691,7 +738,11 @@ static int tegra_crtc_mode_set(struct drm_crtc *crtc,
>       if (err < 0)
>               dev_err(dc->dev, "failed to enable root plane\n");
>
> -     return 0;
> +     err = tegra_dc_program_bandwidth(dc, mode, &window);
> +     if (err)
> +             dev_err(dc->dev, "failed to program the EMC clock\n");
> +
> +     return err;
>  }
>
>  static int tegra_crtc_mode_set_base(struct drm_crtc *crtc, int x, int y,
> @@ -1260,6 +1311,12 @@ static int tegra_dc_probe(struct platform_device *pdev)
>       if (err < 0)
>               return err;
>
> +     dc->emc_clk = devm_clk_get(&pdev->dev, "emc");
> +     if (IS_ERR(dc->emc_clk))
> +             dc->emc_clk = NULL;
> +     else
> +             clk_prepare_enable(dc->emc_clk);
> +
>       regs = platform_get_resource(pdev, IORESOURCE_MEM, 0);
>       dc->regs = devm_ioremap_resource(&pdev->dev, regs);
>       if (IS_ERR(dc->regs))
> @@ -1312,6 +1369,8 @@ static int tegra_dc_remove(struct platform_device *pdev)
>       }
>
>       clk_disable_unprepare(dc->clk);
> +     if (dc->emc_clk)
> +             clk_disable_unprepare(dc->emc_clk);
>
>       return 0;
>  }
> diff --git a/drivers/gpu/drm/tegra/drm.h b/drivers/gpu/drm/tegra/drm.h
> index 6753598..30d91c0 100644
> --- a/drivers/gpu/drm/tegra/drm.h
> +++ b/drivers/gpu/drm/tegra/drm.h
> @@ -101,6 +101,7 @@ struct tegra_dc {
>
>       struct clk *clk;
>       struct reset_control *rst;
> +     struct clk *emc_clk;
>       void __iomem *regs;
>       int irq;
>

--
Pengutronix e.K.             | Lucas Stach                 |
Industrial Linux Solutions   | http://www.pengutronix.de/  |

_______________________________________________
dri-devel mailing list
dri-devel@xxxxxxxxxxxxxxxxxxxxx
http://lists.freedesktop.org/mailman/listinfo/dri-devel

_______________________________________________
dri-devel mailing list
dri-devel@xxxxxxxxxxxxxxxxxxxxx
http://lists.freedesktop.org/mailman/listinfo/dri-devel

[Index of Archives]     [Linux DRI Users]     [Linux Intel Graphics]     [Linux USB Devel]     [Video for Linux]     [Linux Audio Users]     [Yosemite News]     [Linux Kernel]     [Linux SCSI]     [XFree86]     [Linux USB Devel]     [Video for Linux]     [Linux Audio Users]     [Linux Kernel]     [Linux SCSI]     [XFree86]
  Powered by Linux