On Mon, May 26, 2014 at 2:35 AM, Thierry Reding <treding at nvidia.com> wrote:
> On Mon, May 26, 2014 at 11:28:42AM +0200, St?phane Marchesin wrote: > > > > > > > > On Mon, May 26, 2014 at 2:07 AM, Lucas Stach <l.stach at pengutronix.de > <mailto:l.stach at pengutronix.de>> 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? > > There are no other EMC consumers upstream at the moment. Some recent > discussions also indicate that it's unlikely that EMC scaling will be > implemented using shared clocks upstream. > > See here for the full description: > > https://lkml.org/lkml/2014/5/13/469 So if keeping the EMC clock private is a no-go, and shared clocks are also a no-go, should I just make a separate one-off driver just for EMC and call into that? St?phane > > > Also adding linux-tegra to Cc. I like to keep that list in the loop for > patches that touch the Tegra DRM driver. That's especially useful if > other APIs are involved (such as clocks here). > > Thierry > > > > > St?phane > > > > > > Regards, > > Lucas > > > > > Signed-off-by: St?phane Marchesin <marcheu at chromium.org<mailto: > marcheu at chromium.org>> > > > --- > > > 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 at lists.freedesktop.org<mailto:dri-devel at > > lists.freedesktop.org> > > http://lists.freedesktop.org/mailman/listinfo/dri-devel > > > -------------- next part -------------- An HTML attachment was scrubbed... URL: <http://lists.freedesktop.org/archives/dri-devel/attachments/20140526/b63abaa3/attachment-0001.html>