panel-dpi.c 5.5 KB
Newer Older
1 2 3
/*
 * Generic MIPI DPI Panel Driver
 *
4
 * Copyright (C) 2013 Texas Instruments Incorporated - http://www.ti.com/
5 6 7 8 9 10 11
 * Author: Tomi Valkeinen <tomi.valkeinen@ti.com>
 *
 * 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.
 */

T
Tomi Valkeinen 已提交
12
#include <linux/gpio/consumer.h>
13 14 15
#include <linux/module.h>
#include <linux/platform_device.h>
#include <linux/slab.h>
16
#include <linux/of.h>
17
#include <linux/regulator/consumer.h>
18
#include <linux/backlight.h>
19

20
#include <video/of_display_timing.h>
21

22 23
#include "../dss/omapdss.h"

24 25 26
struct panel_drv_data {
	struct omap_dss_device dssdev;

27
	struct videomode vm;
28

29 30
	struct backlight_device *backlight;

31
	struct gpio_desc *enable_gpio;
32
	struct regulator *vcc_supply;
33 34 35 36
};

#define to_panel_data(p) container_of(p, struct panel_drv_data, dssdev)

37 38
static int panel_dpi_connect(struct omap_dss_device *src,
			     struct omap_dss_device *dst)
39 40 41 42
{
	return 0;
}

43 44
static void panel_dpi_disconnect(struct omap_dss_device *src,
				 struct omap_dss_device *dst)
45 46 47 48 49 50
{
}

static int panel_dpi_enable(struct omap_dss_device *dssdev)
{
	struct panel_drv_data *ddata = to_panel_data(dssdev);
51
	struct omap_dss_device *src = dssdev->src;
52 53 54 55 56 57 58 59
	int r;

	if (!omapdss_device_is_connected(dssdev))
		return -ENODEV;

	if (omapdss_device_is_enabled(dssdev))
		return 0;

60
	src->ops->set_timings(src, &ddata->vm);
61

62
	r = src->ops->enable(src);
63 64 65
	if (r)
		return r;

66 67
	r = regulator_enable(ddata->vcc_supply);
	if (r) {
68
		src->ops->disable(src);
69 70 71
		return r;
	}

72
	gpiod_set_value_cansleep(ddata->enable_gpio, 1);
73
	backlight_enable(ddata->backlight);
74

75 76 77 78 79 80 81 82
	dssdev->state = OMAP_DSS_DISPLAY_ACTIVE;

	return 0;
}

static void panel_dpi_disable(struct omap_dss_device *dssdev)
{
	struct panel_drv_data *ddata = to_panel_data(dssdev);
83
	struct omap_dss_device *src = dssdev->src;
84 85 86 87

	if (!omapdss_device_is_enabled(dssdev))
		return;

88
	backlight_disable(ddata->backlight);
89

90
	gpiod_set_value_cansleep(ddata->enable_gpio, 0);
91
	regulator_disable(ddata->vcc_supply);
92

93
	src->ops->disable(src);
94 95 96 97 98

	dssdev->state = OMAP_DSS_DISPLAY_DISABLED;
}

static void panel_dpi_set_timings(struct omap_dss_device *dssdev,
99
				  struct videomode *vm)
100 101
{
	struct panel_drv_data *ddata = to_panel_data(dssdev);
102
	struct omap_dss_device *src = dssdev->src;
103

104
	ddata->vm = *vm;
105

106
	src->ops->set_timings(src, vm);
107 108 109
}

static void panel_dpi_get_timings(struct omap_dss_device *dssdev,
110
				  struct videomode *vm)
111 112 113
{
	struct panel_drv_data *ddata = to_panel_data(dssdev);

114
	*vm = ddata->vm;
115 116 117
}

static int panel_dpi_check_timings(struct omap_dss_device *dssdev,
118
				   struct videomode *vm)
119
{
120
	struct omap_dss_device *src = dssdev->src;
121

122
	return src->ops->check_timings(src, vm);
123 124
}

125
static const struct omap_dss_driver panel_dpi_ops = {
126 127 128 129 130 131 132 133 134 135 136
	.connect	= panel_dpi_connect,
	.disconnect	= panel_dpi_disconnect,

	.enable		= panel_dpi_enable,
	.disable	= panel_dpi_disable,

	.set_timings	= panel_dpi_set_timings,
	.get_timings	= panel_dpi_get_timings,
	.check_timings	= panel_dpi_check_timings,
};

137 138 139 140 141 142 143 144
static int panel_dpi_probe_of(struct platform_device *pdev)
{
	struct panel_drv_data *ddata = platform_get_drvdata(pdev);
	struct device_node *node = pdev->dev.of_node;
	int r;
	struct display_timing timing;
	struct gpio_desc *gpio;

145 146 147
	gpio = devm_gpiod_get_optional(&pdev->dev, "enable", GPIOD_OUT_LOW);
	if (IS_ERR(gpio))
		return PTR_ERR(gpio);
148

T
Tomi Valkeinen 已提交
149 150
	ddata->enable_gpio = gpio;

151 152 153 154 155 156 157 158 159 160
	/*
	 * Many different panels are supported by this driver and there are
	 * probably very different needs for their reset pins in regards to
	 * timing and order relative to the enable gpio. So for now it's just
	 * ensured that the reset line isn't active.
	 */
	gpio = devm_gpiod_get_optional(&pdev->dev, "reset", GPIOD_OUT_LOW);
	if (IS_ERR(gpio))
		return PTR_ERR(gpio);

161 162 163 164
	ddata->vcc_supply = devm_regulator_get(&pdev->dev, "vcc");
	if (IS_ERR(ddata->vcc_supply))
		return PTR_ERR(ddata->vcc_supply);

165
	ddata->backlight = devm_of_find_backlight(&pdev->dev);
166

167 168
	if (IS_ERR(ddata->backlight))
		return PTR_ERR(ddata->backlight);
169

170 171 172
	r = of_get_display_timing(node, "panel-timing", &timing);
	if (r) {
		dev_err(&pdev->dev, "failed to get video timing\n");
173
		return r;
174 175
	}

176
	videomode_from_timing(&timing, &ddata->vm);
177 178 179 180

	return 0;
}

181 182 183 184 185 186 187 188 189 190 191 192
static int panel_dpi_probe(struct platform_device *pdev)
{
	struct panel_drv_data *ddata;
	struct omap_dss_device *dssdev;
	int r;

	ddata = devm_kzalloc(&pdev->dev, sizeof(*ddata), GFP_KERNEL);
	if (ddata == NULL)
		return -ENOMEM;

	platform_set_drvdata(pdev, ddata);

193 194 195
	r = panel_dpi_probe_of(pdev);
	if (r)
		return r;
196 197 198 199 200 201

	dssdev = &ddata->dssdev;
	dssdev->dev = &pdev->dev;
	dssdev->driver = &panel_dpi_ops;
	dssdev->type = OMAP_DISPLAY_TYPE_DPI;
	dssdev->owner = THIS_MODULE;
202
	dssdev->of_ports = BIT(0);
203

204
	omapdss_display_init(dssdev);
205
	omapdss_device_register(dssdev);
206 207 208 209 210 211 212 213 214

	return 0;
}

static int __exit panel_dpi_remove(struct platform_device *pdev)
{
	struct panel_drv_data *ddata = platform_get_drvdata(pdev);
	struct omap_dss_device *dssdev = &ddata->dssdev;

215
	omapdss_device_unregister(dssdev);
216 217 218 219 220 221

	panel_dpi_disable(dssdev);

	return 0;
}

222 223 224 225 226 227 228
static const struct of_device_id panel_dpi_of_match[] = {
	{ .compatible = "omapdss,panel-dpi", },
	{},
};

MODULE_DEVICE_TABLE(of, panel_dpi_of_match);

229 230 231 232 233
static struct platform_driver panel_dpi_driver = {
	.probe = panel_dpi_probe,
	.remove = __exit_p(panel_dpi_remove),
	.driver = {
		.name = "panel-dpi",
234
		.of_match_table = panel_dpi_of_match,
T
Tomi Valkeinen 已提交
235
		.suppress_bind_attrs = true,
236 237 238 239 240 241 242 243
	},
};

module_platform_driver(panel_dpi_driver);

MODULE_AUTHOR("Tomi Valkeinen <tomi.valkeinen@ti.com>");
MODULE_DESCRIPTION("Generic MIPI DPI Panel Driver");
MODULE_LICENSE("GPL");