drm/panel: simple: Add display timing support
The simple panel driver's ->get_modes() implementation calculates the display mode list from the typical timings and the ->get_timings() implementation returns the timings to the connected encoder for mode validation and fixup. Signed-off-by: Philipp Zabel <p.zabel@pengutronix.de> [treding@nvidia.com: select VIDEOMODE_HELPERS] Signed-off-by: Thierry Reding <treding@nvidia.com>
This commit is contained in:
parent
2938931f37
commit
a5d3e62514
|
@ -11,6 +11,7 @@ config DRM_PANEL_SIMPLE
|
||||||
tristate "support for simple panels"
|
tristate "support for simple panels"
|
||||||
depends on OF
|
depends on OF
|
||||||
depends on BACKLIGHT_CLASS_DEVICE
|
depends on BACKLIGHT_CLASS_DEVICE
|
||||||
|
select VIDEOMODE_HELPERS
|
||||||
help
|
help
|
||||||
DRM panel driver for dumb panels that need at most a regulator and
|
DRM panel driver for dumb panels that need at most a regulator and
|
||||||
a GPIO to be powered up. Optionally a backlight can be attached so
|
a GPIO to be powered up. Optionally a backlight can be attached so
|
||||||
|
|
|
@ -33,9 +33,14 @@
|
||||||
#include <drm/drm_mipi_dsi.h>
|
#include <drm/drm_mipi_dsi.h>
|
||||||
#include <drm/drm_panel.h>
|
#include <drm/drm_panel.h>
|
||||||
|
|
||||||
|
#include <video/display_timing.h>
|
||||||
|
#include <video/videomode.h>
|
||||||
|
|
||||||
struct panel_desc {
|
struct panel_desc {
|
||||||
const struct drm_display_mode *modes;
|
const struct drm_display_mode *modes;
|
||||||
unsigned int num_modes;
|
unsigned int num_modes;
|
||||||
|
const struct display_timing *timings;
|
||||||
|
unsigned int num_timings;
|
||||||
|
|
||||||
unsigned int bpc;
|
unsigned int bpc;
|
||||||
|
|
||||||
|
@ -94,6 +99,25 @@ static int panel_simple_get_fixed_modes(struct panel_simple *panel)
|
||||||
if (!panel->desc)
|
if (!panel->desc)
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
|
for (i = 0; i < panel->desc->num_timings; i++) {
|
||||||
|
const struct display_timing *dt = &panel->desc->timings[i];
|
||||||
|
struct videomode vm;
|
||||||
|
|
||||||
|
videomode_from_timing(dt, &vm);
|
||||||
|
mode = drm_mode_create(drm);
|
||||||
|
if (!mode) {
|
||||||
|
dev_err(drm->dev, "failed to add mode %ux%u\n",
|
||||||
|
dt->hactive.typ, dt->vactive.typ);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
drm_display_mode_from_videomode(&vm, mode);
|
||||||
|
drm_mode_set_name(mode);
|
||||||
|
|
||||||
|
drm_mode_probed_add(connector, mode);
|
||||||
|
num++;
|
||||||
|
}
|
||||||
|
|
||||||
for (i = 0; i < panel->desc->num_modes; i++) {
|
for (i = 0; i < panel->desc->num_modes; i++) {
|
||||||
const struct drm_display_mode *m = &panel->desc->modes[i];
|
const struct drm_display_mode *m = &panel->desc->modes[i];
|
||||||
|
|
||||||
|
@ -226,12 +250,30 @@ static int panel_simple_get_modes(struct drm_panel *panel)
|
||||||
return num;
|
return num;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int panel_simple_get_timings(struct drm_panel *panel,
|
||||||
|
unsigned int num_timings,
|
||||||
|
struct display_timing *timings)
|
||||||
|
{
|
||||||
|
struct panel_simple *p = to_panel_simple(panel);
|
||||||
|
unsigned int i;
|
||||||
|
|
||||||
|
if (p->desc->num_timings < num_timings)
|
||||||
|
num_timings = p->desc->num_timings;
|
||||||
|
|
||||||
|
if (timings)
|
||||||
|
for (i = 0; i < num_timings; i++)
|
||||||
|
timings[i] = p->desc->timings[i];
|
||||||
|
|
||||||
|
return p->desc->num_timings;
|
||||||
|
}
|
||||||
|
|
||||||
static const struct drm_panel_funcs panel_simple_funcs = {
|
static const struct drm_panel_funcs panel_simple_funcs = {
|
||||||
.disable = panel_simple_disable,
|
.disable = panel_simple_disable,
|
||||||
.unprepare = panel_simple_unprepare,
|
.unprepare = panel_simple_unprepare,
|
||||||
.prepare = panel_simple_prepare,
|
.prepare = panel_simple_prepare,
|
||||||
.enable = panel_simple_enable,
|
.enable = panel_simple_enable,
|
||||||
.get_modes = panel_simple_get_modes,
|
.get_modes = panel_simple_get_modes,
|
||||||
|
.get_timings = panel_simple_get_timings,
|
||||||
};
|
};
|
||||||
|
|
||||||
static int panel_simple_probe(struct device *dev, const struct panel_desc *desc)
|
static int panel_simple_probe(struct device *dev, const struct panel_desc *desc)
|
||||||
|
|
Loading…
Reference in New Issue