blob: 5fb8063e2559eefc892d86aeec0fa8ee86555109 [file] [log] [blame]
Bryan Wua9a62b62010-11-17 13:34:33 +00001/*
2 * Generic DPI Panels support
3 *
4 * Copyright (C) 2010 Canonical Ltd.
5 * Author: Bryan Wu <bryan.wu@canonical.com>
6 *
Bryan Wuac1427e2010-11-18 09:45:59 +08007 * LCD panel driver for Sharp LQ043T1DG01
8 *
9 * Copyright (C) 2009 Texas Instruments Inc
10 * Author: Vaibhav Hiremath <hvaibhav@ti.com>
11 *
12 * LCD panel driver for Toppoly TDO35S
13 *
14 * Copyright (C) 2009 CompuLab, Ltd.
15 * Author: Mike Rapoport <mike@compulab.co.il>
16 *
Bryan Wua9a62b62010-11-17 13:34:33 +000017 * Copyright (C) 2008 Nokia Corporation
18 * Author: Tomi Valkeinen <tomi.valkeinen@nokia.com>
19 *
20 * This program is free software; you can redistribute it and/or modify it
21 * under the terms of the GNU General Public License version 2 as published by
22 * the Free Software Foundation.
23 *
24 * This program is distributed in the hope that it will be useful, but WITHOUT
25 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
26 * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
27 * more details.
28 *
29 * You should have received a copy of the GNU General Public License along with
30 * this program. If not, see <http://www.gnu.org/licenses/>.
31 */
32
33#include <linux/module.h>
34#include <linux/delay.h>
35#include <linux/slab.h>
Tomi Valkeinena0b38cc2011-05-11 14:05:07 +030036#include <video/omapdss.h>
Bryan Wua9a62b62010-11-17 13:34:33 +000037
Archit Tanejaa0d8dde2013-02-12 16:46:46 +053038#include <video/omap-panel-data.h>
Bryan Wua9a62b62010-11-17 13:34:33 +000039
40struct panel_config {
41 struct omap_video_timings timings;
42
Bryan Wua9a62b62010-11-17 13:34:33 +000043 int power_on_delay;
44 int power_off_delay;
45
46 /*
47 * Used to match device to panel configuration
48 * when use generic panel driver
49 */
50 const char *name;
51};
52
53/* Panel configurations */
54static struct panel_config generic_dpi_panels[] = {
Bryan Wua9a62b62010-11-17 13:34:33 +000055 /* Sharp LQ043T1DG01 */
56 {
57 {
58 .x_res = 480,
59 .y_res = 272,
60
61 .pixel_clock = 9000,
62
63 .hsw = 42,
64 .hfp = 3,
65 .hbp = 2,
66
67 .vsw = 11,
68 .vfp = 3,
69 .vbp = 2,
Archit Tanejaa8d5e412012-06-25 12:26:38 +053070
71 .vsync_level = OMAPDSS_SIG_ACTIVE_LOW,
72 .hsync_level = OMAPDSS_SIG_ACTIVE_LOW,
73 .data_pclk_edge = OMAPDSS_DRIVE_SIG_RISING_EDGE,
74 .de_level = OMAPDSS_SIG_ACTIVE_LOW,
75 .sync_pclk_edge = OMAPDSS_DRIVE_SIG_OPPOSITE_EDGES,
Bryan Wua9a62b62010-11-17 13:34:33 +000076 },
Bryan Wua9a62b62010-11-17 13:34:33 +000077 .power_on_delay = 50,
78 .power_off_delay = 100,
79 .name = "sharp_lq",
80 },
81
82 /* Sharp LS037V7DW01 */
83 {
84 {
85 .x_res = 480,
86 .y_res = 640,
87
88 .pixel_clock = 19200,
89
90 .hsw = 2,
91 .hfp = 1,
92 .hbp = 28,
93
94 .vsw = 1,
95 .vfp = 1,
96 .vbp = 1,
Archit Tanejaa8d5e412012-06-25 12:26:38 +053097
98 .vsync_level = OMAPDSS_SIG_ACTIVE_LOW,
99 .hsync_level = OMAPDSS_SIG_ACTIVE_LOW,
100 .data_pclk_edge = OMAPDSS_DRIVE_SIG_RISING_EDGE,
101 .de_level = OMAPDSS_SIG_ACTIVE_HIGH,
102 .sync_pclk_edge = OMAPDSS_DRIVE_SIG_OPPOSITE_EDGES,
Bryan Wua9a62b62010-11-17 13:34:33 +0000103 },
Bryan Wua9a62b62010-11-17 13:34:33 +0000104 .power_on_delay = 50,
105 .power_off_delay = 100,
106 .name = "sharp_ls",
107 },
108
109 /* Toppoly TDO35S */
110 {
111 {
112 .x_res = 480,
113 .y_res = 640,
114
115 .pixel_clock = 26000,
116
117 .hfp = 104,
118 .hsw = 8,
119 .hbp = 8,
120
121 .vfp = 4,
122 .vsw = 2,
123 .vbp = 2,
Archit Tanejaa8d5e412012-06-25 12:26:38 +0530124
125 .vsync_level = OMAPDSS_SIG_ACTIVE_LOW,
126 .hsync_level = OMAPDSS_SIG_ACTIVE_LOW,
127 .data_pclk_edge = OMAPDSS_DRIVE_SIG_FALLING_EDGE,
128 .de_level = OMAPDSS_SIG_ACTIVE_HIGH,
129 .sync_pclk_edge = OMAPDSS_DRIVE_SIG_FALLING_EDGE,
Bryan Wua9a62b62010-11-17 13:34:33 +0000130 },
Bryan Wua9a62b62010-11-17 13:34:33 +0000131 .power_on_delay = 0,
132 .power_off_delay = 0,
133 .name = "toppoly_tdo35s",
134 },
Tomi Valkeinen09dc89c2011-02-22 16:39:47 +0200135
136 /* Samsung LTE430WQ-F0C */
137 {
138 {
139 .x_res = 480,
140 .y_res = 272,
141
142 .pixel_clock = 9200,
143
144 .hfp = 8,
145 .hsw = 41,
146 .hbp = 45 - 41,
147
148 .vfp = 4,
149 .vsw = 10,
150 .vbp = 12 - 10,
Archit Tanejaa8d5e412012-06-25 12:26:38 +0530151
152 .vsync_level = OMAPDSS_SIG_ACTIVE_LOW,
153 .hsync_level = OMAPDSS_SIG_ACTIVE_LOW,
154 .data_pclk_edge = OMAPDSS_DRIVE_SIG_RISING_EDGE,
155 .de_level = OMAPDSS_SIG_ACTIVE_HIGH,
156 .sync_pclk_edge = OMAPDSS_DRIVE_SIG_OPPOSITE_EDGES,
Tomi Valkeinen09dc89c2011-02-22 16:39:47 +0200157 },
Tomi Valkeinen09dc89c2011-02-22 16:39:47 +0200158 .power_on_delay = 0,
159 .power_off_delay = 0,
160 .name = "samsung_lte430wq_f0c",
161 },
Enric Balletbo i Serra50ca19d2011-05-03 09:07:36 +0200162
163 /* Seiko 70WVW1TZ3Z3 */
164 {
165 {
166 .x_res = 800,
167 .y_res = 480,
168
169 .pixel_clock = 33000,
170
171 .hsw = 128,
172 .hfp = 10,
173 .hbp = 10,
174
175 .vsw = 2,
176 .vfp = 4,
177 .vbp = 11,
Archit Tanejaa8d5e412012-06-25 12:26:38 +0530178
179 .vsync_level = OMAPDSS_SIG_ACTIVE_LOW,
180 .hsync_level = OMAPDSS_SIG_ACTIVE_LOW,
181 .data_pclk_edge = OMAPDSS_DRIVE_SIG_RISING_EDGE,
182 .de_level = OMAPDSS_SIG_ACTIVE_HIGH,
183 .sync_pclk_edge = OMAPDSS_DRIVE_SIG_OPPOSITE_EDGES,
Enric Balletbo i Serra50ca19d2011-05-03 09:07:36 +0200184 },
Enric Balletbo i Serra50ca19d2011-05-03 09:07:36 +0200185 .power_on_delay = 0,
186 .power_off_delay = 0,
187 .name = "seiko_70wvw1tz3",
188 },
Enric Balletbo i Serraa0ce4cc2011-05-03 09:07:37 +0200189
190 /* Powertip PH480272T */
191 {
192 {
193 .x_res = 480,
194 .y_res = 272,
195
196 .pixel_clock = 9000,
197
198 .hsw = 40,
199 .hfp = 2,
200 .hbp = 2,
201
202 .vsw = 10,
203 .vfp = 2,
204 .vbp = 2,
Archit Tanejaa8d5e412012-06-25 12:26:38 +0530205
206 .vsync_level = OMAPDSS_SIG_ACTIVE_LOW,
207 .hsync_level = OMAPDSS_SIG_ACTIVE_LOW,
208 .data_pclk_edge = OMAPDSS_DRIVE_SIG_RISING_EDGE,
209 .de_level = OMAPDSS_SIG_ACTIVE_LOW,
210 .sync_pclk_edge = OMAPDSS_DRIVE_SIG_OPPOSITE_EDGES,
Enric Balletbo i Serraa0ce4cc2011-05-03 09:07:37 +0200211 },
Enric Balletbo i Serraa0ce4cc2011-05-03 09:07:37 +0200212 .power_on_delay = 0,
213 .power_off_delay = 0,
214 .name = "powertip_ph480272t",
215 },
Thomas Weber1e497c42011-09-01 15:05:07 +0200216
217 /* Innolux AT070TN83 */
218 {
219 {
220 .x_res = 800,
221 .y_res = 480,
222
223 .pixel_clock = 40000,
224
225 .hsw = 48,
226 .hfp = 1,
227 .hbp = 1,
228
229 .vsw = 3,
230 .vfp = 12,
231 .vbp = 25,
Archit Tanejaa8d5e412012-06-25 12:26:38 +0530232
233 .vsync_level = OMAPDSS_SIG_ACTIVE_LOW,
234 .hsync_level = OMAPDSS_SIG_ACTIVE_LOW,
235 .data_pclk_edge = OMAPDSS_DRIVE_SIG_RISING_EDGE,
236 .de_level = OMAPDSS_SIG_ACTIVE_HIGH,
237 .sync_pclk_edge = OMAPDSS_DRIVE_SIG_OPPOSITE_EDGES,
Thomas Weber1e497c42011-09-01 15:05:07 +0200238 },
Thomas Weber1e497c42011-09-01 15:05:07 +0200239 .power_on_delay = 0,
240 .power_off_delay = 0,
241 .name = "innolux_at070tn83",
242 },
Tomi Valkeinen3fddbf52011-04-19 19:23:05 +0300243
244 /* NEC NL2432DR22-11B */
245 {
246 {
247 .x_res = 240,
248 .y_res = 320,
249
250 .pixel_clock = 5400,
251
252 .hsw = 3,
253 .hfp = 3,
254 .hbp = 39,
255
256 .vsw = 1,
257 .vfp = 2,
258 .vbp = 7,
Archit Tanejaa8d5e412012-06-25 12:26:38 +0530259
260 .vsync_level = OMAPDSS_SIG_ACTIVE_LOW,
261 .hsync_level = OMAPDSS_SIG_ACTIVE_LOW,
262 .data_pclk_edge = OMAPDSS_DRIVE_SIG_RISING_EDGE,
263 .de_level = OMAPDSS_SIG_ACTIVE_HIGH,
264 .sync_pclk_edge = OMAPDSS_DRIVE_SIG_OPPOSITE_EDGES,
Tomi Valkeinen3fddbf52011-04-19 19:23:05 +0300265 },
Tomi Valkeinen3fddbf52011-04-19 19:23:05 +0300266 .name = "nec_nl2432dr22-11b",
267 },
Tomi Valkeinen8a41ba92011-04-19 19:59:58 +0300268
269 /* Unknown panel used in OMAP H4 */
270 {
271 {
272 .x_res = 240,
273 .y_res = 320,
274
275 .pixel_clock = 6250,
276
277 .hsw = 15,
278 .hfp = 15,
279 .hbp = 60,
280
281 .vsw = 1,
282 .vfp = 1,
283 .vbp = 1,
Archit Tanejaa8d5e412012-06-25 12:26:38 +0530284
285 .vsync_level = OMAPDSS_SIG_ACTIVE_HIGH,
286 .hsync_level = OMAPDSS_SIG_ACTIVE_HIGH,
287 .data_pclk_edge = OMAPDSS_DRIVE_SIG_RISING_EDGE,
288 .de_level = OMAPDSS_SIG_ACTIVE_HIGH,
289 .sync_pclk_edge = OMAPDSS_DRIVE_SIG_OPPOSITE_EDGES,
Tomi Valkeinen8a41ba92011-04-19 19:59:58 +0300290 },
Tomi Valkeinen8a41ba92011-04-19 19:59:58 +0300291 .name = "h4",
292 },
Tomi Valkeinen8d49fe72011-04-20 10:47:28 +0300293
Ilya Yanoka47161a2011-11-20 19:15:39 +0100294 /* FocalTech ETM070003DH6 */
295 {
296 {
297 .x_res = 800,
298 .y_res = 480,
299
300 .pixel_clock = 28000,
301
302 .hsw = 48,
303 .hfp = 40,
304 .hbp = 40,
305
306 .vsw = 3,
307 .vfp = 13,
308 .vbp = 29,
Archit Tanejaa8d5e412012-06-25 12:26:38 +0530309
310 .vsync_level = OMAPDSS_SIG_ACTIVE_LOW,
311 .hsync_level = OMAPDSS_SIG_ACTIVE_LOW,
312 .data_pclk_edge = OMAPDSS_DRIVE_SIG_RISING_EDGE,
313 .de_level = OMAPDSS_SIG_ACTIVE_HIGH,
314 .sync_pclk_edge = OMAPDSS_DRIVE_SIG_OPPOSITE_EDGES,
Ilya Yanoka47161a2011-11-20 19:15:39 +0100315 },
Ilya Yanoka47161a2011-11-20 19:15:39 +0100316 .name = "focaltech_etm070003dh6",
317 },
Daniel Mack62941ec2011-12-20 10:54:32 +0100318
319 /* Microtips Technologies - UMSH-8173MD */
320 {
321 {
322 .x_res = 800,
323 .y_res = 480,
324
325 .pixel_clock = 34560,
326
327 .hsw = 13,
328 .hfp = 101,
329 .hbp = 101,
330
331 .vsw = 23,
332 .vfp = 1,
333 .vbp = 1,
Archit Tanejaa8d5e412012-06-25 12:26:38 +0530334
335 .vsync_level = OMAPDSS_SIG_ACTIVE_LOW,
336 .hsync_level = OMAPDSS_SIG_ACTIVE_LOW,
337 .data_pclk_edge = OMAPDSS_DRIVE_SIG_FALLING_EDGE,
338 .de_level = OMAPDSS_SIG_ACTIVE_HIGH,
339 .sync_pclk_edge = OMAPDSS_DRIVE_SIG_OPPOSITE_EDGES,
Daniel Mack62941ec2011-12-20 10:54:32 +0100340 },
Daniel Mack62941ec2011-12-20 10:54:32 +0100341 .power_on_delay = 0,
342 .power_off_delay = 0,
343 .name = "microtips_umsh_8173md",
344 },
Ilya Yanok6acea9c2011-12-26 23:56:41 +0100345
346 /* OrtusTech COM43H4M10XTC */
347 {
348 {
349 .x_res = 480,
350 .y_res = 272,
351
352 .pixel_clock = 8000,
353
354 .hsw = 41,
355 .hfp = 8,
356 .hbp = 4,
357
358 .vsw = 10,
359 .vfp = 4,
360 .vbp = 2,
Archit Tanejaa8d5e412012-06-25 12:26:38 +0530361
362 .vsync_level = OMAPDSS_SIG_ACTIVE_HIGH,
363 .hsync_level = OMAPDSS_SIG_ACTIVE_HIGH,
364 .data_pclk_edge = OMAPDSS_DRIVE_SIG_RISING_EDGE,
365 .de_level = OMAPDSS_SIG_ACTIVE_HIGH,
366 .sync_pclk_edge = OMAPDSS_DRIVE_SIG_OPPOSITE_EDGES,
Ilya Yanok6acea9c2011-12-26 23:56:41 +0100367 },
Ilya Yanok6acea9c2011-12-26 23:56:41 +0100368 .name = "ortustech_com43h4m10xtc",
369 },
Yegor Yefremov992ee642012-02-08 11:11:08 +0100370
371 /* Innolux AT080TN52 */
372 {
373 {
374 .x_res = 800,
375 .y_res = 600,
376
377 .pixel_clock = 41142,
378
379 .hsw = 20,
380 .hfp = 210,
381 .hbp = 46,
382
383 .vsw = 10,
384 .vfp = 12,
385 .vbp = 23,
Archit Tanejaa8d5e412012-06-25 12:26:38 +0530386
387 .vsync_level = OMAPDSS_SIG_ACTIVE_LOW,
388 .hsync_level = OMAPDSS_SIG_ACTIVE_LOW,
389 .data_pclk_edge = OMAPDSS_DRIVE_SIG_RISING_EDGE,
390 .de_level = OMAPDSS_SIG_ACTIVE_LOW,
391 .sync_pclk_edge = OMAPDSS_DRIVE_SIG_OPPOSITE_EDGES,
Yegor Yefremov992ee642012-02-08 11:11:08 +0100392 },
Yegor Yefremov992ee642012-02-08 11:11:08 +0100393 .name = "innolux_at080tn52",
394 },
Thomas Weber697bacb2012-02-17 13:41:58 +0100395
396 /* Mitsubishi AA084SB01 */
397 {
398 {
399 .x_res = 800,
400 .y_res = 600,
401 .pixel_clock = 40000,
402
403 .hsw = 1,
404 .hfp = 254,
405 .hbp = 1,
406
407 .vsw = 1,
408 .vfp = 26,
409 .vbp = 1,
Archit Tanejaa8d5e412012-06-25 12:26:38 +0530410
411 .vsync_level = OMAPDSS_SIG_ACTIVE_HIGH,
412 .hsync_level = OMAPDSS_SIG_ACTIVE_HIGH,
413 .data_pclk_edge = OMAPDSS_DRIVE_SIG_RISING_EDGE,
414 .de_level = OMAPDSS_SIG_ACTIVE_HIGH,
415 .sync_pclk_edge = OMAPDSS_DRIVE_SIG_OPPOSITE_EDGES,
Thomas Weber697bacb2012-02-17 13:41:58 +0100416 },
Thomas Weber697bacb2012-02-17 13:41:58 +0100417 .name = "mitsubishi_aa084sb01",
418 },
Thomas Webere97374c2012-02-17 13:41:59 +0100419 /* EDT ET0500G0DH6 */
420 {
421 {
422 .x_res = 800,
423 .y_res = 480,
424 .pixel_clock = 33260,
425
426 .hsw = 128,
427 .hfp = 216,
428 .hbp = 40,
429
430 .vsw = 2,
431 .vfp = 35,
432 .vbp = 10,
Archit Tanejaa8d5e412012-06-25 12:26:38 +0530433
434 .vsync_level = OMAPDSS_SIG_ACTIVE_HIGH,
435 .hsync_level = OMAPDSS_SIG_ACTIVE_HIGH,
436 .data_pclk_edge = OMAPDSS_DRIVE_SIG_RISING_EDGE,
437 .de_level = OMAPDSS_SIG_ACTIVE_HIGH,
438 .sync_pclk_edge = OMAPDSS_DRIVE_SIG_OPPOSITE_EDGES,
Thomas Webere97374c2012-02-17 13:41:59 +0100439 },
Thomas Webere97374c2012-02-17 13:41:59 +0100440 .name = "edt_et0500g0dh6",
441 },
Jan Weitzel3dd606e2012-04-20 11:41:22 +0200442
443 /* Prime-View PD050VL1 */
444 {
445 {
446 .x_res = 640,
447 .y_res = 480,
448
449 .pixel_clock = 25000,
450
451 .hsw = 96,
452 .hfp = 18,
453 .hbp = 46,
454
455 .vsw = 2,
456 .vfp = 10,
457 .vbp = 33,
Archit Tanejaa8d5e412012-06-25 12:26:38 +0530458
459 .vsync_level = OMAPDSS_SIG_ACTIVE_LOW,
460 .hsync_level = OMAPDSS_SIG_ACTIVE_LOW,
461 .data_pclk_edge = OMAPDSS_DRIVE_SIG_FALLING_EDGE,
462 .de_level = OMAPDSS_SIG_ACTIVE_HIGH,
463 .sync_pclk_edge = OMAPDSS_DRIVE_SIG_OPPOSITE_EDGES,
Jan Weitzel3dd606e2012-04-20 11:41:22 +0200464 },
Jan Weitzel3dd606e2012-04-20 11:41:22 +0200465 .name = "primeview_pd050vl1",
466 },
467
468 /* Prime-View PM070WL4 */
469 {
470 {
471 .x_res = 800,
472 .y_res = 480,
473
474 .pixel_clock = 32000,
475
476 .hsw = 128,
477 .hfp = 42,
478 .hbp = 86,
479
480 .vsw = 2,
481 .vfp = 10,
482 .vbp = 33,
Archit Tanejaa8d5e412012-06-25 12:26:38 +0530483
484 .vsync_level = OMAPDSS_SIG_ACTIVE_LOW,
485 .hsync_level = OMAPDSS_SIG_ACTIVE_LOW,
486 .data_pclk_edge = OMAPDSS_DRIVE_SIG_FALLING_EDGE,
487 .de_level = OMAPDSS_SIG_ACTIVE_HIGH,
488 .sync_pclk_edge = OMAPDSS_DRIVE_SIG_OPPOSITE_EDGES,
Jan Weitzel3dd606e2012-04-20 11:41:22 +0200489 },
Jan Weitzel3dd606e2012-04-20 11:41:22 +0200490 .name = "primeview_pm070wl4",
491 },
492
493 /* Prime-View PD104SLF */
494 {
495 {
496 .x_res = 800,
497 .y_res = 600,
498
499 .pixel_clock = 40000,
500
501 .hsw = 128,
502 .hfp = 42,
503 .hbp = 86,
504
505 .vsw = 4,
506 .vfp = 1,
507 .vbp = 23,
Archit Tanejaa8d5e412012-06-25 12:26:38 +0530508
509 .vsync_level = OMAPDSS_SIG_ACTIVE_LOW,
510 .hsync_level = OMAPDSS_SIG_ACTIVE_LOW,
511 .data_pclk_edge = OMAPDSS_DRIVE_SIG_FALLING_EDGE,
512 .de_level = OMAPDSS_SIG_ACTIVE_HIGH,
513 .sync_pclk_edge = OMAPDSS_DRIVE_SIG_OPPOSITE_EDGES,
Jan Weitzel3dd606e2012-04-20 11:41:22 +0200514 },
Jan Weitzel3dd606e2012-04-20 11:41:22 +0200515 .name = "primeview_pd104slf",
516 },
Bryan Wua9a62b62010-11-17 13:34:33 +0000517};
518
519struct panel_drv_data {
520
521 struct omap_dss_device *dssdev;
522
523 struct panel_config *panel_config;
Archit Tanejae19d6592012-08-08 14:20:30 +0530524
525 struct mutex lock;
Bryan Wua9a62b62010-11-17 13:34:33 +0000526};
527
528static inline struct panel_generic_dpi_data
529*get_panel_data(const struct omap_dss_device *dssdev)
530{
531 return (struct panel_generic_dpi_data *) dssdev->data;
532}
533
534static int generic_dpi_panel_power_on(struct omap_dss_device *dssdev)
535{
536 int r;
537 struct panel_generic_dpi_data *panel_data = get_panel_data(dssdev);
538 struct panel_drv_data *drv_data = dev_get_drvdata(&dssdev->dev);
539 struct panel_config *panel_config = drv_data->panel_config;
540
541 if (dssdev->state == OMAP_DSS_DISPLAY_ACTIVE)
542 return 0;
543
Archit Tanejac4991442012-08-08 14:28:54 +0530544 omapdss_dpi_set_timings(dssdev, &dssdev->panel.timings);
Archit Tanejac6b393d2012-07-06 15:30:52 +0530545 omapdss_dpi_set_data_lines(dssdev, dssdev->phy.dpi.data_lines);
Archit Tanejac4991442012-08-08 14:28:54 +0530546
Bryan Wua9a62b62010-11-17 13:34:33 +0000547 r = omapdss_dpi_display_enable(dssdev);
548 if (r)
549 goto err0;
550
551 /* wait couple of vsyncs until enabling the LCD */
552 if (panel_config->power_on_delay)
553 msleep(panel_config->power_on_delay);
554
555 if (panel_data->platform_enable) {
556 r = panel_data->platform_enable(dssdev);
557 if (r)
558 goto err1;
559 }
560
561 return 0;
562err1:
563 omapdss_dpi_display_disable(dssdev);
564err0:
565 return r;
566}
567
568static void generic_dpi_panel_power_off(struct omap_dss_device *dssdev)
569{
570 struct panel_generic_dpi_data *panel_data = get_panel_data(dssdev);
571 struct panel_drv_data *drv_data = dev_get_drvdata(&dssdev->dev);
572 struct panel_config *panel_config = drv_data->panel_config;
573
574 if (dssdev->state != OMAP_DSS_DISPLAY_ACTIVE)
575 return;
576
577 if (panel_data->platform_disable)
578 panel_data->platform_disable(dssdev);
579
580 /* wait couple of vsyncs after disabling the LCD */
581 if (panel_config->power_off_delay)
582 msleep(panel_config->power_off_delay);
583
584 omapdss_dpi_display_disable(dssdev);
585}
586
587static int generic_dpi_panel_probe(struct omap_dss_device *dssdev)
588{
589 struct panel_generic_dpi_data *panel_data = get_panel_data(dssdev);
590 struct panel_config *panel_config = NULL;
591 struct panel_drv_data *drv_data = NULL;
592 int i;
593
594 dev_dbg(&dssdev->dev, "probe\n");
595
596 if (!panel_data || !panel_data->name)
597 return -EINVAL;
598
599 for (i = 0; i < ARRAY_SIZE(generic_dpi_panels); i++) {
600 if (strcmp(panel_data->name, generic_dpi_panels[i].name) == 0) {
601 panel_config = &generic_dpi_panels[i];
602 break;
603 }
604 }
605
606 if (!panel_config)
607 return -EINVAL;
608
Bryan Wua9a62b62010-11-17 13:34:33 +0000609 dssdev->panel.timings = panel_config->timings;
Bryan Wua9a62b62010-11-17 13:34:33 +0000610
Archit Tanejaf5e484d2012-09-14 15:17:01 +0300611 drv_data = devm_kzalloc(&dssdev->dev, sizeof(*drv_data), GFP_KERNEL);
Bryan Wua9a62b62010-11-17 13:34:33 +0000612 if (!drv_data)
613 return -ENOMEM;
614
615 drv_data->dssdev = dssdev;
616 drv_data->panel_config = panel_config;
617
Archit Tanejae19d6592012-08-08 14:20:30 +0530618 mutex_init(&drv_data->lock);
619
Bryan Wua9a62b62010-11-17 13:34:33 +0000620 dev_set_drvdata(&dssdev->dev, drv_data);
621
622 return 0;
623}
624
Tomi Valkeinen14e4d782011-03-31 12:03:51 +0300625static void __exit generic_dpi_panel_remove(struct omap_dss_device *dssdev)
Bryan Wua9a62b62010-11-17 13:34:33 +0000626{
Bryan Wua9a62b62010-11-17 13:34:33 +0000627 dev_dbg(&dssdev->dev, "remove\n");
628
Bryan Wua9a62b62010-11-17 13:34:33 +0000629 dev_set_drvdata(&dssdev->dev, NULL);
630}
631
632static int generic_dpi_panel_enable(struct omap_dss_device *dssdev)
633{
Archit Tanejae19d6592012-08-08 14:20:30 +0530634 struct panel_drv_data *drv_data = dev_get_drvdata(&dssdev->dev);
635 int r;
636
637 mutex_lock(&drv_data->lock);
Bryan Wua9a62b62010-11-17 13:34:33 +0000638
639 r = generic_dpi_panel_power_on(dssdev);
640 if (r)
Archit Tanejae19d6592012-08-08 14:20:30 +0530641 goto err;
Bryan Wua9a62b62010-11-17 13:34:33 +0000642
643 dssdev->state = OMAP_DSS_DISPLAY_ACTIVE;
Archit Tanejae19d6592012-08-08 14:20:30 +0530644err:
645 mutex_unlock(&drv_data->lock);
Bryan Wua9a62b62010-11-17 13:34:33 +0000646
Archit Tanejae19d6592012-08-08 14:20:30 +0530647 return r;
Bryan Wua9a62b62010-11-17 13:34:33 +0000648}
649
650static void generic_dpi_panel_disable(struct omap_dss_device *dssdev)
651{
Archit Tanejae19d6592012-08-08 14:20:30 +0530652 struct panel_drv_data *drv_data = dev_get_drvdata(&dssdev->dev);
653
654 mutex_lock(&drv_data->lock);
655
Bryan Wua9a62b62010-11-17 13:34:33 +0000656 generic_dpi_panel_power_off(dssdev);
657
658 dssdev->state = OMAP_DSS_DISPLAY_DISABLED;
Archit Tanejae19d6592012-08-08 14:20:30 +0530659
660 mutex_unlock(&drv_data->lock);
Bryan Wua9a62b62010-11-17 13:34:33 +0000661}
662
Bryan Wua9a62b62010-11-17 13:34:33 +0000663static void generic_dpi_panel_set_timings(struct omap_dss_device *dssdev,
664 struct omap_video_timings *timings)
665{
Archit Tanejae19d6592012-08-08 14:20:30 +0530666 struct panel_drv_data *drv_data = dev_get_drvdata(&dssdev->dev);
667
668 mutex_lock(&drv_data->lock);
669
Archit Tanejac4991442012-08-08 14:28:54 +0530670 omapdss_dpi_set_timings(dssdev, timings);
Archit Tanejae19d6592012-08-08 14:20:30 +0530671
Archit Tanejabdcae3c2012-08-08 14:29:48 +0530672 dssdev->panel.timings = *timings;
673
Archit Tanejae19d6592012-08-08 14:20:30 +0530674 mutex_unlock(&drv_data->lock);
675}
676
677static void generic_dpi_panel_get_timings(struct omap_dss_device *dssdev,
678 struct omap_video_timings *timings)
679{
680 struct panel_drv_data *drv_data = dev_get_drvdata(&dssdev->dev);
681
682 mutex_lock(&drv_data->lock);
683
684 *timings = dssdev->panel.timings;
685
686 mutex_unlock(&drv_data->lock);
Bryan Wua9a62b62010-11-17 13:34:33 +0000687}
688
Bryan Wua9a62b62010-11-17 13:34:33 +0000689static int generic_dpi_panel_check_timings(struct omap_dss_device *dssdev,
690 struct omap_video_timings *timings)
691{
Archit Tanejae19d6592012-08-08 14:20:30 +0530692 struct panel_drv_data *drv_data = dev_get_drvdata(&dssdev->dev);
693 int r;
694
695 mutex_lock(&drv_data->lock);
696
697 r = dpi_check_timings(dssdev, timings);
698
699 mutex_unlock(&drv_data->lock);
700
701 return r;
Bryan Wua9a62b62010-11-17 13:34:33 +0000702}
703
704static struct omap_dss_driver dpi_driver = {
705 .probe = generic_dpi_panel_probe,
Tomi Valkeinen14e4d782011-03-31 12:03:51 +0300706 .remove = __exit_p(generic_dpi_panel_remove),
Bryan Wua9a62b62010-11-17 13:34:33 +0000707
708 .enable = generic_dpi_panel_enable,
709 .disable = generic_dpi_panel_disable,
Bryan Wua9a62b62010-11-17 13:34:33 +0000710
711 .set_timings = generic_dpi_panel_set_timings,
Archit Tanejae19d6592012-08-08 14:20:30 +0530712 .get_timings = generic_dpi_panel_get_timings,
Bryan Wua9a62b62010-11-17 13:34:33 +0000713 .check_timings = generic_dpi_panel_check_timings,
714
715 .driver = {
716 .name = "generic_dpi_panel",
717 .owner = THIS_MODULE,
718 },
719};
720
721static int __init generic_dpi_panel_drv_init(void)
722{
723 return omap_dss_register_driver(&dpi_driver);
724}
725
726static void __exit generic_dpi_panel_drv_exit(void)
727{
728 omap_dss_unregister_driver(&dpi_driver);
729}
730
731module_init(generic_dpi_panel_drv_init);
732module_exit(generic_dpi_panel_drv_exit);
733MODULE_LICENSE("GPL");