Hi Laurent, Am Donnerstag, den 22.11.2012, 22:45 +0100 schrieb Laurent Pinchart: > From: Laurent Pinchart <laurent.pinchart+renesas@xxxxxxxxxxxxxxxx> > > Signed-off-by: Laurent Pinchart <laurent.pinchart@xxxxxxxxxxxxxxxx> > --- > drivers/video/display/Kconfig | 13 +++ > drivers/video/display/Makefile | 1 + > drivers/video/display/panel-dpi.c | 147 +++++++++++++++++++++++++++++++++++++ > include/video/panel-dpi.h | 24 ++++++ > 4 files changed, 185 insertions(+), 0 deletions(-) > create mode 100644 drivers/video/display/panel-dpi.c > create mode 100644 include/video/panel-dpi.h > > diff --git a/drivers/video/display/Kconfig b/drivers/video/display/Kconfig > index 1d533e7..0f9b990 100644 > --- a/drivers/video/display/Kconfig > +++ b/drivers/video/display/Kconfig > @@ -2,3 +2,16 @@ menuconfig DISPLAY_CORE > tristate "Display Core" > ---help--- > Support common display framework for graphics devices. > + > +if DISPLAY_CORE > + > +config DISPLAY_PANEL_DPI > + tristate "DPI (Parallel) Display Panels" > + ---help--- > + Support for simple digital (parallel) pixel interface panels. Those > + panels receive pixel data through a parallel bus and have no control > + bus. I have tried this driver together with the imx parallel-display with the added patch below for device tree support. > + If you are in doubt, say N. > + > +endif # DISPLAY_CORE > diff --git a/drivers/video/display/Makefile b/drivers/video/display/Makefile > index bd93496..47978d4 100644 > --- a/drivers/video/display/Makefile > +++ b/drivers/video/display/Makefile > @@ -1 +1,2 @@ > obj-$(CONFIG_DISPLAY_CORE) += display-core.o > +obj-$(CONFIG_DISPLAY_PANEL_DPI) += panel-dpi.o > diff --git a/drivers/video/display/panel-dpi.c b/drivers/video/display/panel-dpi.c > new file mode 100644 > index 0000000..c56197a > --- /dev/null > +++ b/drivers/video/display/panel-dpi.c > @@ -0,0 +1,147 @@ > +/* > + * DPI Display Panel > + * > + * Copyright (C) 2012 Renesas Solutions Corp. > + * > + * Contacts: Laurent Pinchart <laurent.pinchart@xxxxxxxxxxxxxxxx> > + * > + * This program is free software; you can redistribute it and/or modify > + * it under the terms of the GNU General Public License version 2 as > + * published by the Free Software Foundation. > + */ > + > +#include <linux/kernel.h> > +#include <linux/module.h> > +#include <linux/platform_device.h> > +#include <linux/slab.h> > + > +#include <video/display.h> > +#include <video/panel-dpi.h> > + > +struct panel_dpi { > + struct display_entity entity; > + const struct panel_dpi_platform_data *pdata; > +}; > + > +#define to_panel_dpi(p) container_of(p, struct panel_dpi, entity) > + > +static const struct display_entity_interface_params panel_dpi_params = { > + .type = DISPLAY_ENTITY_INTERFACE_DPI, > +}; > + > +static int panel_dpi_set_state(struct display_entity *entity, > + enum display_entity_state state) > +{ > + switch (state) { > + case DISPLAY_ENTITY_STATE_OFF: > + case DISPLAY_ENTITY_STATE_STANDBY: > + display_entity_set_stream(entity->source, > + DISPLAY_ENTITY_STREAM_STOPPED); > + break; > + > + case DISPLAY_ENTITY_STATE_ON: > + display_entity_set_stream(entity->source, > + DISPLAY_ENTITY_STREAM_CONTINUOUS); > + break; > + } > + > + return 0; > +} > + > +static int panel_dpi_get_modes(struct display_entity *entity, > + const struct videomode **modes) > +{ > + struct panel_dpi *panel = to_panel_dpi(entity); > + > + *modes = panel->pdata->mode; > + return 1; > +} > + > +static int panel_dpi_get_size(struct display_entity *entity, > + unsigned int *width, unsigned int *height) > +{ > + struct panel_dpi *panel = to_panel_dpi(entity); > + > + *width = panel->pdata->width; > + *height = panel->pdata->height; > + return 0; > +} > + > +static int panel_dpi_get_params(struct display_entity *entity, > + struct display_entity_interface_params *params) > +{ > + *params = panel_dpi_params; > + return 0; > +} > + > +static const struct display_entity_control_ops panel_dpi_control_ops = { > + .set_state = panel_dpi_set_state, > + .get_modes = panel_dpi_get_modes, > + .get_size = panel_dpi_get_size, > + .get_params = panel_dpi_get_params, > +}; > + > +static void panel_dpi_release(struct display_entity *entity) > +{ > + struct panel_dpi *panel = to_panel_dpi(entity); > + > + kfree(panel); > +} > + > +static int panel_dpi_remove(struct platform_device *pdev) > +{ > + struct panel_dpi *panel = platform_get_drvdata(pdev); > + > + platform_set_drvdata(pdev, NULL); > + display_entity_unregister(&panel->entity); > + > + return 0; > +} > + > +static int __devinit panel_dpi_probe(struct platform_device *pdev) > +{ > + const struct panel_dpi_platform_data *pdata = pdev->dev.platform_data; > + struct panel_dpi *panel; > + int ret; > + > + if (pdata == NULL) > + return -ENODEV; > + > + panel = kzalloc(sizeof(*panel), GFP_KERNEL); > + if (panel == NULL) > + return -ENOMEM; > + > + panel->pdata = pdata; > + panel->entity.dev = &pdev->dev; > + panel->entity.release = panel_dpi_release; I don't understand this. Shouldn't the panel be allocated with devm_kzalloc and display_entity_register make sure that this driver cannot be unbound instead? What if we call in sequence on this device's entity: display_entity_get(entity); display_entity_release(entity); /* here struct panel_dpi gets freed */ display_entity_get(entity); display_entity_release(entity); > + panel->entity.ops.ctrl = &panel_dpi_control_ops; > + > + ret = display_entity_register(&panel->entity); > + if (ret < 0) { > + kfree(panel); > + return ret; > + } > + > + platform_set_drvdata(pdev, panel); > + > + return 0; > +} > + > +static const struct dev_pm_ops panel_dpi_dev_pm_ops = { > +}; > + > +static struct platform_driver panel_dpi_driver = { > + .probe = panel_dpi_probe, > + .remove = panel_dpi_remove, > + .driver = { > + .name = "panel_dpi", > + .owner = THIS_MODULE, > + .pm = &panel_dpi_dev_pm_ops, > + }, > +}; > + > +module_platform_driver(panel_dpi_driver); > + > +MODULE_AUTHOR("Laurent Pinchart <laurent.pinchart@xxxxxxxxxxxxxxxx>"); > +MODULE_DESCRIPTION("DPI Display Panel"); > +MODULE_LICENSE("GPL"); > diff --git a/include/video/panel-dpi.h b/include/video/panel-dpi.h > new file mode 100644 > index 0000000..0547b4a > --- /dev/null > +++ b/include/video/panel-dpi.h > @@ -0,0 +1,24 @@ > +/* > + * DPI Display Panel > + * > + * Copyright (C) 2012 Renesas Solutions Corp. > + * > + * Contacts: Laurent Pinchart <laurent.pinchart@xxxxxxxxxxxxxxxx> > + * > + * This program is free software; you can redistribute it and/or modify > + * it under the terms of the GNU General Public License version 2 as > + * published by the Free Software Foundation. > + */ > + > +#ifndef __PANEL_DPI_H__ > +#define __PANEL_DPI_H__ > + > +#include <linux/videomode.h> > + > +struct panel_dpi_platform_data { > + unsigned long width; /* Panel width in mm */ > + unsigned long height; /* Panel height in mm */ > + const struct videomode *mode; > +}; > + > +#endif /* __PANEL_DPI_H__ */ From: Philipp Zabel <p.zabel@xxxxxxxxxxxxxx> Date: Thu, 29 Nov 2012 19:18:30 +0100 Subject: [PATCH] video: panel: Add device tree support to the DPI panel driver Signed-off-by: Philipp Zabel <p.zabel@xxxxxxxxxxxxxx> --- drivers/video/display/panel-dpi.c | 56 ++++++++++++++++++++++++++++++++++--- 1 file changed, 52 insertions(+), 4 deletions(-) diff --git a/drivers/video/display/panel-dpi.c b/drivers/video/display/panel-dpi.c index c56197a..5a7dd8e 100644 --- a/drivers/video/display/panel-dpi.c +++ b/drivers/video/display/panel-dpi.c @@ -14,6 +14,8 @@ #include <linux/module.h> #include <linux/platform_device.h> #include <linux/slab.h> +#include <linux/of.h> +#include <linux/of_videomode.h> #include <video/display.h> #include <video/panel-dpi.h> @@ -98,20 +100,60 @@ static int panel_dpi_remove(struct platform_device *pdev) return 0; } -static int __devinit panel_dpi_probe(struct platform_device *pdev) +static int __devinit panel_dpi_parse_dt(struct device *dev, + struct panel_dpi *panel) { - const struct panel_dpi_platform_data *pdata = pdev->dev.platform_data; - struct panel_dpi *panel; + struct device_node *np = dev->of_node; + struct panel_dpi_platform_data *pdata; + struct videomode *vm; + u32 width, height; int ret; + if (!np) + return -ENODEV; + + pdata = devm_kzalloc(dev, sizeof(*pdata), GFP_KERNEL); if (pdata == NULL) + return -ENOMEM; + + vm = devm_kzalloc(dev, sizeof(*vm), GFP_KERNEL); + if (vm == NULL) + return -ENOMEM; + + ret = of_get_videomode(np, vm, 0); + if (ret < 0) return -ENODEV; + of_property_read_u32(np, "width", &width); + of_property_read_u32(np, "height", &height); + + pdata->mode = vm; + pdata->width = width; + pdata->height = height; + panel->pdata = pdata; + + return 0; +} + +static int __devinit panel_dpi_probe(struct platform_device *pdev) +{ + const struct panel_dpi_platform_data *pdata = pdev->dev.platform_data; + struct panel_dpi *panel; + int ret; + panel = kzalloc(sizeof(*panel), GFP_KERNEL); if (panel == NULL) return -ENOMEM; - panel->pdata = pdata; + if (pdata) { + panel->pdata = pdata; + } else { + ret = panel_dpi_parse_dt(&pdev->dev, panel); + if (ret < 0) { + kfree(panel); + return ret; + } + } panel->entity.dev = &pdev->dev; panel->entity.release = panel_dpi_release; panel->entity.ops.ctrl = &panel_dpi_control_ops; @@ -130,11 +172,17 @@ static int __devinit panel_dpi_probe(struct platform_device *pdev) static const struct dev_pm_ops panel_dpi_dev_pm_ops = { }; +static const struct of_device_id panel_dpi_dt_ids[] = { + { .compatible = "dpi-panel", }, + { } +}; + static struct platform_driver panel_dpi_driver = { .probe = panel_dpi_probe, .remove = panel_dpi_remove, .driver = { .name = "panel_dpi", + .of_match_table = panel_dpi_dt_ids, .owner = THIS_MODULE, .pm = &panel_dpi_dev_pm_ops, }, -- 1.7.10.4 regards Philipp -- To unsubscribe from this list: send the line "unsubscribe linux-media" in the body of a message to majordomo@xxxxxxxxxxxxxxx More majordomo info at http://vger.kernel.org/majordomo-info.html