blob: 7ebaa40d3b56b611a9478190f504c0b0f1357921 [file] [log] [blame]
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001/*
2 * linux/drivers/video/omap2/dss/manager.c
3 *
4 * Copyright (C) 2009 Nokia Corporation
5 * Author: Tomi Valkeinen <tomi.valkeinen@nokia.com>
6 *
7 * Some code and ideas taken from drivers/video/omap/ driver
8 * by Imre Deak.
9 *
10 * This program is free software; you can redistribute it and/or modify it
11 * under the terms of the GNU General Public License version 2 as published by
12 * the Free Software Foundation.
13 *
14 * This program is distributed in the hope that it will be useful, but WITHOUT
15 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
16 * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
17 * more details.
18 *
19 * You should have received a copy of the GNU General Public License along with
20 * this program. If not, see <http://www.gnu.org/licenses/>.
21 */
22
23#define DSS_SUBSYS_NAME "MANAGER"
24
25#include <linux/kernel.h>
Tejun Heo5a0e3ad2010-03-24 17:04:11 +090026#include <linux/slab.h>
Tomi Valkeineneed07e02009-08-07 13:43:20 +030027#include <linux/module.h>
28#include <linux/platform_device.h>
29#include <linux/spinlock.h>
30#include <linux/jiffies.h>
31
Tomi Valkeinena0b38cc2011-05-11 14:05:07 +030032#include <video/omapdss.h>
Tomi Valkeineneed07e02009-08-07 13:43:20 +030033#include <plat/cpu.h>
34
35#include "dss.h"
Archit Tanejaa0acb552010-09-15 19:20:00 +053036#include "dss_features.h"
Tomi Valkeineneed07e02009-08-07 13:43:20 +030037
38static int num_managers;
39static struct list_head manager_list;
40
41static ssize_t manager_name_show(struct omap_overlay_manager *mgr, char *buf)
42{
43 return snprintf(buf, PAGE_SIZE, "%s\n", mgr->name);
44}
45
46static ssize_t manager_display_show(struct omap_overlay_manager *mgr, char *buf)
47{
48 return snprintf(buf, PAGE_SIZE, "%s\n",
49 mgr->device ? mgr->device->name : "<none>");
50}
51
52static ssize_t manager_display_store(struct omap_overlay_manager *mgr,
53 const char *buf, size_t size)
54{
55 int r = 0;
56 size_t len = size;
57 struct omap_dss_device *dssdev = NULL;
58
59 int match(struct omap_dss_device *dssdev, void *data)
60 {
61 const char *str = data;
62 return sysfs_streq(dssdev->name, str);
63 }
64
65 if (buf[size-1] == '\n')
66 --len;
67
68 if (len > 0)
69 dssdev = omap_dss_find_device((void *)buf, match);
70
71 if (len > 0 && dssdev == NULL)
72 return -EINVAL;
73
74 if (dssdev)
75 DSSDBG("display %s found\n", dssdev->name);
76
77 if (mgr->device) {
78 r = mgr->unset_device(mgr);
79 if (r) {
80 DSSERR("failed to unset display\n");
81 goto put_device;
82 }
83 }
84
85 if (dssdev) {
86 r = mgr->set_device(mgr, dssdev);
87 if (r) {
88 DSSERR("failed to set manager\n");
89 goto put_device;
90 }
91
92 r = mgr->apply(mgr);
93 if (r) {
94 DSSERR("failed to apply dispc config\n");
95 goto put_device;
96 }
97 }
98
99put_device:
100 if (dssdev)
101 omap_dss_put_device(dssdev);
102
103 return r ? r : size;
104}
105
106static ssize_t manager_default_color_show(struct omap_overlay_manager *mgr,
107 char *buf)
108{
109 return snprintf(buf, PAGE_SIZE, "%d\n", mgr->info.default_color);
110}
111
112static ssize_t manager_default_color_store(struct omap_overlay_manager *mgr,
113 const char *buf, size_t size)
114{
115 struct omap_overlay_manager_info info;
116 u32 color;
117 int r;
118
119 if (sscanf(buf, "%d", &color) != 1)
120 return -EINVAL;
121
122 mgr->get_manager_info(mgr, &info);
123
124 info.default_color = color;
125
126 r = mgr->set_manager_info(mgr, &info);
127 if (r)
128 return r;
129
130 r = mgr->apply(mgr);
131 if (r)
132 return r;
133
134 return size;
135}
136
137static const char *trans_key_type_str[] = {
138 "gfx-destination",
139 "video-source",
140};
141
142static ssize_t manager_trans_key_type_show(struct omap_overlay_manager *mgr,
143 char *buf)
144{
145 enum omap_dss_trans_key_type key_type;
146
147 key_type = mgr->info.trans_key_type;
148 BUG_ON(key_type >= ARRAY_SIZE(trans_key_type_str));
149
150 return snprintf(buf, PAGE_SIZE, "%s\n", trans_key_type_str[key_type]);
151}
152
153static ssize_t manager_trans_key_type_store(struct omap_overlay_manager *mgr,
154 const char *buf, size_t size)
155{
156 enum omap_dss_trans_key_type key_type;
157 struct omap_overlay_manager_info info;
158 int r;
159
160 for (key_type = OMAP_DSS_COLOR_KEY_GFX_DST;
161 key_type < ARRAY_SIZE(trans_key_type_str); key_type++) {
162 if (sysfs_streq(buf, trans_key_type_str[key_type]))
163 break;
164 }
165
166 if (key_type == ARRAY_SIZE(trans_key_type_str))
167 return -EINVAL;
168
169 mgr->get_manager_info(mgr, &info);
170
171 info.trans_key_type = key_type;
172
173 r = mgr->set_manager_info(mgr, &info);
174 if (r)
175 return r;
176
177 r = mgr->apply(mgr);
178 if (r)
179 return r;
180
181 return size;
182}
183
184static ssize_t manager_trans_key_value_show(struct omap_overlay_manager *mgr,
185 char *buf)
186{
187 return snprintf(buf, PAGE_SIZE, "%d\n", mgr->info.trans_key);
188}
189
190static ssize_t manager_trans_key_value_store(struct omap_overlay_manager *mgr,
191 const char *buf, size_t size)
192{
193 struct omap_overlay_manager_info info;
194 u32 key_value;
195 int r;
196
197 if (sscanf(buf, "%d", &key_value) != 1)
198 return -EINVAL;
199
200 mgr->get_manager_info(mgr, &info);
201
202 info.trans_key = key_value;
203
204 r = mgr->set_manager_info(mgr, &info);
205 if (r)
206 return r;
207
208 r = mgr->apply(mgr);
209 if (r)
210 return r;
211
212 return size;
213}
214
215static ssize_t manager_trans_key_enabled_show(struct omap_overlay_manager *mgr,
216 char *buf)
217{
218 return snprintf(buf, PAGE_SIZE, "%d\n", mgr->info.trans_enabled);
219}
220
221static ssize_t manager_trans_key_enabled_store(struct omap_overlay_manager *mgr,
222 const char *buf, size_t size)
223{
224 struct omap_overlay_manager_info info;
225 int enable;
226 int r;
227
228 if (sscanf(buf, "%d", &enable) != 1)
229 return -EINVAL;
230
231 mgr->get_manager_info(mgr, &info);
232
233 info.trans_enabled = enable ? true : false;
234
235 r = mgr->set_manager_info(mgr, &info);
236 if (r)
237 return r;
238
239 r = mgr->apply(mgr);
240 if (r)
241 return r;
242
243 return size;
244}
245
246static ssize_t manager_alpha_blending_enabled_show(
247 struct omap_overlay_manager *mgr, char *buf)
248{
249 return snprintf(buf, PAGE_SIZE, "%d\n", mgr->info.alpha_enabled);
250}
251
252static ssize_t manager_alpha_blending_enabled_store(
253 struct omap_overlay_manager *mgr,
254 const char *buf, size_t size)
255{
256 struct omap_overlay_manager_info info;
257 int enable;
258 int r;
259
260 if (sscanf(buf, "%d", &enable) != 1)
261 return -EINVAL;
262
263 mgr->get_manager_info(mgr, &info);
264
265 info.alpha_enabled = enable ? true : false;
266
267 r = mgr->set_manager_info(mgr, &info);
268 if (r)
269 return r;
270
271 r = mgr->apply(mgr);
272 if (r)
273 return r;
274
275 return size;
276}
277
Tomi Valkeinen3c07cae2011-06-21 09:34:30 +0300278static ssize_t manager_cpr_enable_show(struct omap_overlay_manager *mgr,
279 char *buf)
280{
281 return snprintf(buf, PAGE_SIZE, "%d\n", mgr->info.cpr_enable);
282}
283
284static ssize_t manager_cpr_enable_store(struct omap_overlay_manager *mgr,
285 const char *buf, size_t size)
286{
287 struct omap_overlay_manager_info info;
288 int v;
289 int r;
290 bool enable;
291
292 if (!dss_has_feature(FEAT_CPR))
293 return -ENODEV;
294
295 r = kstrtoint(buf, 0, &v);
296 if (r)
297 return r;
298
299 enable = !!v;
300
301 mgr->get_manager_info(mgr, &info);
302
303 if (info.cpr_enable == enable)
304 return size;
305
306 info.cpr_enable = enable;
307
308 r = mgr->set_manager_info(mgr, &info);
309 if (r)
310 return r;
311
312 r = mgr->apply(mgr);
313 if (r)
314 return r;
315
316 return size;
317}
318
319static ssize_t manager_cpr_coef_show(struct omap_overlay_manager *mgr,
320 char *buf)
321{
322 struct omap_overlay_manager_info info;
323
324 mgr->get_manager_info(mgr, &info);
325
326 return snprintf(buf, PAGE_SIZE,
327 "%d %d %d %d %d %d %d %d %d\n",
328 info.cpr_coefs.rr,
329 info.cpr_coefs.rg,
330 info.cpr_coefs.rb,
331 info.cpr_coefs.gr,
332 info.cpr_coefs.gg,
333 info.cpr_coefs.gb,
334 info.cpr_coefs.br,
335 info.cpr_coefs.bg,
336 info.cpr_coefs.bb);
337}
338
339static ssize_t manager_cpr_coef_store(struct omap_overlay_manager *mgr,
340 const char *buf, size_t size)
341{
342 struct omap_overlay_manager_info info;
343 struct omap_dss_cpr_coefs coefs;
344 int r, i;
345 s16 *arr;
346
347 if (!dss_has_feature(FEAT_CPR))
348 return -ENODEV;
349
350 if (sscanf(buf, "%hd %hd %hd %hd %hd %hd %hd %hd %hd",
351 &coefs.rr, &coefs.rg, &coefs.rb,
352 &coefs.gr, &coefs.gg, &coefs.gb,
353 &coefs.br, &coefs.bg, &coefs.bb) != 9)
354 return -EINVAL;
355
356 arr = (s16[]){ coefs.rr, coefs.rg, coefs.rb,
357 coefs.gr, coefs.gg, coefs.gb,
358 coefs.br, coefs.bg, coefs.bb };
359
360 for (i = 0; i < 9; ++i) {
361 if (arr[i] < -512 || arr[i] > 511)
362 return -EINVAL;
363 }
364
365 mgr->get_manager_info(mgr, &info);
366
367 info.cpr_coefs = coefs;
368
369 r = mgr->set_manager_info(mgr, &info);
370 if (r)
371 return r;
372
373 r = mgr->apply(mgr);
374 if (r)
375 return r;
376
377 return size;
378}
379
Tomi Valkeineneed07e02009-08-07 13:43:20 +0300380struct manager_attribute {
381 struct attribute attr;
382 ssize_t (*show)(struct omap_overlay_manager *, char *);
383 ssize_t (*store)(struct omap_overlay_manager *, const char *, size_t);
384};
385
386#define MANAGER_ATTR(_name, _mode, _show, _store) \
387 struct manager_attribute manager_attr_##_name = \
388 __ATTR(_name, _mode, _show, _store)
389
390static MANAGER_ATTR(name, S_IRUGO, manager_name_show, NULL);
391static MANAGER_ATTR(display, S_IRUGO|S_IWUSR,
392 manager_display_show, manager_display_store);
393static MANAGER_ATTR(default_color, S_IRUGO|S_IWUSR,
394 manager_default_color_show, manager_default_color_store);
395static MANAGER_ATTR(trans_key_type, S_IRUGO|S_IWUSR,
396 manager_trans_key_type_show, manager_trans_key_type_store);
397static MANAGER_ATTR(trans_key_value, S_IRUGO|S_IWUSR,
398 manager_trans_key_value_show, manager_trans_key_value_store);
399static MANAGER_ATTR(trans_key_enabled, S_IRUGO|S_IWUSR,
400 manager_trans_key_enabled_show,
401 manager_trans_key_enabled_store);
402static MANAGER_ATTR(alpha_blending_enabled, S_IRUGO|S_IWUSR,
403 manager_alpha_blending_enabled_show,
404 manager_alpha_blending_enabled_store);
Tomi Valkeinen3c07cae2011-06-21 09:34:30 +0300405static MANAGER_ATTR(cpr_enable, S_IRUGO|S_IWUSR,
406 manager_cpr_enable_show,
407 manager_cpr_enable_store);
408static MANAGER_ATTR(cpr_coef, S_IRUGO|S_IWUSR,
409 manager_cpr_coef_show,
410 manager_cpr_coef_store);
Tomi Valkeineneed07e02009-08-07 13:43:20 +0300411
412
413static struct attribute *manager_sysfs_attrs[] = {
414 &manager_attr_name.attr,
415 &manager_attr_display.attr,
416 &manager_attr_default_color.attr,
417 &manager_attr_trans_key_type.attr,
418 &manager_attr_trans_key_value.attr,
419 &manager_attr_trans_key_enabled.attr,
420 &manager_attr_alpha_blending_enabled.attr,
Tomi Valkeinen3c07cae2011-06-21 09:34:30 +0300421 &manager_attr_cpr_enable.attr,
422 &manager_attr_cpr_coef.attr,
Tomi Valkeineneed07e02009-08-07 13:43:20 +0300423 NULL
424};
425
426static ssize_t manager_attr_show(struct kobject *kobj, struct attribute *attr,
427 char *buf)
428{
429 struct omap_overlay_manager *manager;
430 struct manager_attribute *manager_attr;
431
432 manager = container_of(kobj, struct omap_overlay_manager, kobj);
433 manager_attr = container_of(attr, struct manager_attribute, attr);
434
435 if (!manager_attr->show)
436 return -ENOENT;
437
438 return manager_attr->show(manager, buf);
439}
440
441static ssize_t manager_attr_store(struct kobject *kobj, struct attribute *attr,
442 const char *buf, size_t size)
443{
444 struct omap_overlay_manager *manager;
445 struct manager_attribute *manager_attr;
446
447 manager = container_of(kobj, struct omap_overlay_manager, kobj);
448 manager_attr = container_of(attr, struct manager_attribute, attr);
449
450 if (!manager_attr->store)
451 return -ENOENT;
452
453 return manager_attr->store(manager, buf, size);
454}
455
Emese Revfy52cf25d2010-01-19 02:58:23 +0100456static const struct sysfs_ops manager_sysfs_ops = {
Tomi Valkeineneed07e02009-08-07 13:43:20 +0300457 .show = manager_attr_show,
458 .store = manager_attr_store,
459};
460
461static struct kobj_type manager_ktype = {
462 .sysfs_ops = &manager_sysfs_ops,
463 .default_attrs = manager_sysfs_attrs,
464};
465
466/*
467 * We have 4 levels of cache for the dispc settings. First two are in SW and
468 * the latter two in HW.
469 *
470 * +--------------------+
471 * |overlay/manager_info|
472 * +--------------------+
473 * v
474 * apply()
475 * v
476 * +--------------------+
477 * | dss_cache |
478 * +--------------------+
479 * v
480 * configure()
481 * v
482 * +--------------------+
483 * | shadow registers |
484 * +--------------------+
485 * v
486 * VFP or lcd/digit_enable
487 * v
488 * +--------------------+
489 * | registers |
490 * +--------------------+
491 */
492
493struct overlay_cache_data {
494 /* If true, cache changed, but not written to shadow registers. Set
495 * in apply(), cleared when registers written. */
496 bool dirty;
497 /* If true, shadow registers contain changed values not yet in real
498 * registers. Set when writing to shadow registers, cleared at
499 * VSYNC/EVSYNC */
500 bool shadow_dirty;
501
502 bool enabled;
503
Nishant Kamat4df9d102011-06-20 20:39:28 +0530504 struct omap_overlay_info info;
Tomi Valkeineneed07e02009-08-07 13:43:20 +0300505
506 enum omap_channel channel;
507 bool replication;
508 bool ilace;
509
510 enum omap_burst_size burst_size;
511 u32 fifo_low;
512 u32 fifo_high;
Tomi Valkeineneed07e02009-08-07 13:43:20 +0300513};
514
515struct manager_cache_data {
516 /* If true, cache changed, but not written to shadow registers. Set
517 * in apply(), cleared when registers written. */
518 bool dirty;
519 /* If true, shadow registers contain changed values not yet in real
520 * registers. Set when writing to shadow registers, cleared at
521 * VSYNC/EVSYNC */
522 bool shadow_dirty;
523
Nishant Kamat4df9d102011-06-20 20:39:28 +0530524 struct omap_overlay_manager_info info;
Tomi Valkeineneed07e02009-08-07 13:43:20 +0300525
Tomi Valkeineneed07e02009-08-07 13:43:20 +0300526 bool manual_update;
527 bool do_manual_update;
528
529 /* manual update region */
530 u16 x, y, w, h;
Tomi Valkeinen26a8c252010-06-09 15:31:34 +0300531
532 /* enlarge the update area if the update area contains scaled
533 * overlays */
534 bool enlarge_update_area;
Tomi Valkeineneed07e02009-08-07 13:43:20 +0300535};
536
537static struct {
538 spinlock_t lock;
Archit Tanejaa0acb552010-09-15 19:20:00 +0530539 struct overlay_cache_data overlay_cache[MAX_DSS_OVERLAYS];
540 struct manager_cache_data manager_cache[MAX_DSS_MANAGERS];
Tomi Valkeineneed07e02009-08-07 13:43:20 +0300541
542 bool irq_enabled;
543} dss_cache;
544
545
546
547static int omap_dss_set_device(struct omap_overlay_manager *mgr,
548 struct omap_dss_device *dssdev)
549{
550 int i;
551 int r;
552
553 if (dssdev->manager) {
554 DSSERR("display '%s' already has a manager '%s'\n",
555 dssdev->name, dssdev->manager->name);
556 return -EINVAL;
557 }
558
559 if ((mgr->supported_displays & dssdev->type) == 0) {
560 DSSERR("display '%s' does not support manager '%s'\n",
561 dssdev->name, mgr->name);
562 return -EINVAL;
563 }
564
565 for (i = 0; i < mgr->num_overlays; i++) {
566 struct omap_overlay *ovl = mgr->overlays[i];
567
568 if (ovl->manager != mgr || !ovl->info.enabled)
569 continue;
570
571 r = dss_check_overlay(ovl, dssdev);
572 if (r)
573 return r;
574 }
575
576 dssdev->manager = mgr;
577 mgr->device = dssdev;
578 mgr->device_changed = true;
579
580 return 0;
581}
582
583static int omap_dss_unset_device(struct omap_overlay_manager *mgr)
584{
585 if (!mgr->device) {
586 DSSERR("failed to unset display, display not set.\n");
587 return -EINVAL;
588 }
589
590 mgr->device->manager = NULL;
591 mgr->device = NULL;
592 mgr->device_changed = true;
593
594 return 0;
595}
596
Tomi Valkeinen3f71cbe2010-01-08 17:06:04 +0200597static int dss_mgr_wait_for_vsync(struct omap_overlay_manager *mgr)
598{
599 unsigned long timeout = msecs_to_jiffies(500);
600 u32 irq;
601
Sumit Semwal18faa1b2010-12-02 11:27:14 +0000602 if (mgr->device->type == OMAP_DISPLAY_TYPE_VENC) {
Tomi Valkeinen3f71cbe2010-01-08 17:06:04 +0200603 irq = DISPC_IRQ_EVSYNC_ODD;
Mythri P Kb1196012011-03-08 17:15:54 +0530604 } else if (mgr->device->type == OMAP_DISPLAY_TYPE_HDMI) {
605 irq = DISPC_IRQ_EVSYNC_EVEN;
Sumit Semwal18faa1b2010-12-02 11:27:14 +0000606 } else {
607 if (mgr->id == OMAP_DSS_CHANNEL_LCD)
608 irq = DISPC_IRQ_VSYNC;
609 else
610 irq = DISPC_IRQ_VSYNC2;
611 }
Tomi Valkeinen3f71cbe2010-01-08 17:06:04 +0200612 return omap_dispc_wait_for_irq_interruptible_timeout(irq, timeout);
613}
614
Tomi Valkeineneed07e02009-08-07 13:43:20 +0300615static int dss_mgr_wait_for_go(struct omap_overlay_manager *mgr)
616{
617 unsigned long timeout = msecs_to_jiffies(500);
618 struct manager_cache_data *mc;
Tomi Valkeineneed07e02009-08-07 13:43:20 +0300619 u32 irq;
620 int r;
621 int i;
Tomi Valkeinen446f7bf2010-01-11 16:12:31 +0200622 struct omap_dss_device *dssdev = mgr->device;
Tomi Valkeineneed07e02009-08-07 13:43:20 +0300623
Ville Syrjäläa74b2602010-03-04 16:03:56 +0200624 if (!dssdev || dssdev->state != OMAP_DSS_DISPLAY_ACTIVE)
Tomi Valkeineneed07e02009-08-07 13:43:20 +0300625 return 0;
626
Tomi Valkeinen8cff88c2011-04-30 14:09:53 +0300627 if (dssdev->caps & OMAP_DSS_DISPLAY_CAP_MANUAL_UPDATE)
628 return 0;
629
Mythri P Kb1196012011-03-08 17:15:54 +0530630 if (dssdev->type == OMAP_DISPLAY_TYPE_VENC
631 || dssdev->type == OMAP_DISPLAY_TYPE_HDMI) {
Tomi Valkeineneed07e02009-08-07 13:43:20 +0300632 irq = DISPC_IRQ_EVSYNC_ODD | DISPC_IRQ_EVSYNC_EVEN;
Tomi Valkeineneed07e02009-08-07 13:43:20 +0300633 } else {
Tomi Valkeinen8cff88c2011-04-30 14:09:53 +0300634 irq = (dssdev->manager->id == OMAP_DSS_CHANNEL_LCD) ?
635 DISPC_IRQ_VSYNC : DISPC_IRQ_VSYNC2;
Tomi Valkeineneed07e02009-08-07 13:43:20 +0300636 }
637
638 mc = &dss_cache.manager_cache[mgr->id];
639 i = 0;
640 while (1) {
641 unsigned long flags;
642 bool shadow_dirty, dirty;
643
644 spin_lock_irqsave(&dss_cache.lock, flags);
645 dirty = mc->dirty;
646 shadow_dirty = mc->shadow_dirty;
647 spin_unlock_irqrestore(&dss_cache.lock, flags);
648
649 if (!dirty && !shadow_dirty) {
650 r = 0;
651 break;
652 }
653
654 /* 4 iterations is the worst case:
655 * 1 - initial iteration, dirty = true (between VFP and VSYNC)
656 * 2 - first VSYNC, dirty = true
657 * 3 - dirty = false, shadow_dirty = true
658 * 4 - shadow_dirty = false */
659 if (i++ == 3) {
660 DSSERR("mgr(%d)->wait_for_go() not finishing\n",
661 mgr->id);
662 r = 0;
663 break;
664 }
665
666 r = omap_dispc_wait_for_irq_interruptible_timeout(irq, timeout);
667 if (r == -ERESTARTSYS)
668 break;
669
670 if (r) {
671 DSSERR("mgr(%d)->wait_for_go() timeout\n", mgr->id);
672 break;
673 }
674 }
675
676 return r;
677}
678
679int dss_mgr_wait_for_go_ovl(struct omap_overlay *ovl)
680{
681 unsigned long timeout = msecs_to_jiffies(500);
Tomi Valkeineneed07e02009-08-07 13:43:20 +0300682 struct overlay_cache_data *oc;
683 struct omap_dss_device *dssdev;
684 u32 irq;
685 int r;
686 int i;
687
Ville Syrjäläa74b2602010-03-04 16:03:56 +0200688 if (!ovl->manager)
Tomi Valkeineneed07e02009-08-07 13:43:20 +0300689 return 0;
690
691 dssdev = ovl->manager->device;
692
Ville Syrjäläa74b2602010-03-04 16:03:56 +0200693 if (!dssdev || dssdev->state != OMAP_DSS_DISPLAY_ACTIVE)
694 return 0;
695
Tomi Valkeinen8cff88c2011-04-30 14:09:53 +0300696 if (dssdev->caps & OMAP_DSS_DISPLAY_CAP_MANUAL_UPDATE)
697 return 0;
698
Mythri P Kb1196012011-03-08 17:15:54 +0530699 if (dssdev->type == OMAP_DISPLAY_TYPE_VENC
700 || dssdev->type == OMAP_DISPLAY_TYPE_HDMI) {
Tomi Valkeineneed07e02009-08-07 13:43:20 +0300701 irq = DISPC_IRQ_EVSYNC_ODD | DISPC_IRQ_EVSYNC_EVEN;
Tomi Valkeineneed07e02009-08-07 13:43:20 +0300702 } else {
Tomi Valkeinen8cff88c2011-04-30 14:09:53 +0300703 irq = (dssdev->manager->id == OMAP_DSS_CHANNEL_LCD) ?
704 DISPC_IRQ_VSYNC : DISPC_IRQ_VSYNC2;
Tomi Valkeineneed07e02009-08-07 13:43:20 +0300705 }
706
707 oc = &dss_cache.overlay_cache[ovl->id];
708 i = 0;
709 while (1) {
710 unsigned long flags;
711 bool shadow_dirty, dirty;
712
713 spin_lock_irqsave(&dss_cache.lock, flags);
714 dirty = oc->dirty;
715 shadow_dirty = oc->shadow_dirty;
716 spin_unlock_irqrestore(&dss_cache.lock, flags);
717
718 if (!dirty && !shadow_dirty) {
719 r = 0;
720 break;
721 }
722
723 /* 4 iterations is the worst case:
724 * 1 - initial iteration, dirty = true (between VFP and VSYNC)
725 * 2 - first VSYNC, dirty = true
726 * 3 - dirty = false, shadow_dirty = true
727 * 4 - shadow_dirty = false */
728 if (i++ == 3) {
729 DSSERR("ovl(%d)->wait_for_go() not finishing\n",
730 ovl->id);
731 r = 0;
732 break;
733 }
734
735 r = omap_dispc_wait_for_irq_interruptible_timeout(irq, timeout);
736 if (r == -ERESTARTSYS)
737 break;
738
739 if (r) {
740 DSSERR("ovl(%d)->wait_for_go() timeout\n", ovl->id);
741 break;
742 }
743 }
744
745 return r;
746}
747
748static int overlay_enabled(struct omap_overlay *ovl)
749{
750 return ovl->info.enabled && ovl->manager && ovl->manager->device;
751}
752
753/* Is rect1 a subset of rect2? */
754static bool rectangle_subset(int x1, int y1, int w1, int h1,
755 int x2, int y2, int w2, int h2)
756{
757 if (x1 < x2 || y1 < y2)
758 return false;
759
760 if (x1 + w1 > x2 + w2)
761 return false;
762
763 if (y1 + h1 > y2 + h2)
764 return false;
765
766 return true;
767}
768
769/* Do rect1 and rect2 overlap? */
770static bool rectangle_intersects(int x1, int y1, int w1, int h1,
771 int x2, int y2, int w2, int h2)
772{
773 if (x1 >= x2 + w2)
774 return false;
775
776 if (x2 >= x1 + w1)
777 return false;
778
779 if (y1 >= y2 + h2)
780 return false;
781
782 if (y2 >= y1 + h1)
783 return false;
784
785 return true;
786}
787
788static bool dispc_is_overlay_scaled(struct overlay_cache_data *oc)
789{
Nishant Kamat4df9d102011-06-20 20:39:28 +0530790 struct omap_overlay_info *oi = &oc->info;
791
792 if (oi->out_width != 0 && oi->width != oi->out_width)
Tomi Valkeineneed07e02009-08-07 13:43:20 +0300793 return true;
794
Nishant Kamat4df9d102011-06-20 20:39:28 +0530795 if (oi->out_height != 0 && oi->height != oi->out_height)
Tomi Valkeineneed07e02009-08-07 13:43:20 +0300796 return true;
797
798 return false;
799}
800
801static int configure_overlay(enum omap_plane plane)
802{
803 struct overlay_cache_data *c;
804 struct manager_cache_data *mc;
Nishant Kamat4df9d102011-06-20 20:39:28 +0530805 struct omap_overlay_info *oi;
806 struct omap_overlay_manager_info *mi;
Tomi Valkeineneed07e02009-08-07 13:43:20 +0300807 u16 outw, outh;
808 u16 x, y, w, h;
809 u32 paddr;
810 int r;
Tomi Valkeinen26a8c252010-06-09 15:31:34 +0300811 u16 orig_w, orig_h, orig_outw, orig_outh;
Tomi Valkeineneed07e02009-08-07 13:43:20 +0300812
813 DSSDBGF("%d", plane);
814
815 c = &dss_cache.overlay_cache[plane];
Nishant Kamat4df9d102011-06-20 20:39:28 +0530816 oi = &c->info;
Tomi Valkeineneed07e02009-08-07 13:43:20 +0300817
818 if (!c->enabled) {
819 dispc_enable_plane(plane, 0);
820 return 0;
821 }
822
823 mc = &dss_cache.manager_cache[c->channel];
Nishant Kamat4df9d102011-06-20 20:39:28 +0530824 mi = &mc->info;
Tomi Valkeineneed07e02009-08-07 13:43:20 +0300825
Nishant Kamat4df9d102011-06-20 20:39:28 +0530826 x = oi->pos_x;
827 y = oi->pos_y;
828 w = oi->width;
829 h = oi->height;
830 outw = oi->out_width == 0 ? oi->width : oi->out_width;
831 outh = oi->out_height == 0 ? oi->height : oi->out_height;
832 paddr = oi->paddr;
Tomi Valkeineneed07e02009-08-07 13:43:20 +0300833
Tomi Valkeinen26a8c252010-06-09 15:31:34 +0300834 orig_w = w;
835 orig_h = h;
836 orig_outw = outw;
837 orig_outh = outh;
838
Tomi Valkeinen8cff88c2011-04-30 14:09:53 +0300839 if (mc->manual_update && mc->do_manual_update) {
Tomi Valkeineneed07e02009-08-07 13:43:20 +0300840 unsigned bpp;
Tomi Valkeinen26a8c252010-06-09 15:31:34 +0300841 unsigned scale_x_m = w, scale_x_d = outw;
842 unsigned scale_y_m = h, scale_y_d = outh;
843
Tomi Valkeineneed07e02009-08-07 13:43:20 +0300844 /* If the overlay is outside the update region, disable it */
845 if (!rectangle_intersects(mc->x, mc->y, mc->w, mc->h,
846 x, y, outw, outh)) {
847 dispc_enable_plane(plane, 0);
848 return 0;
849 }
850
Nishant Kamat4df9d102011-06-20 20:39:28 +0530851 switch (oi->color_mode) {
Amber Jainf20e4222011-05-19 19:47:50 +0530852 case OMAP_DSS_COLOR_NV12:
853 bpp = 8;
854 break;
Tomi Valkeineneed07e02009-08-07 13:43:20 +0300855 case OMAP_DSS_COLOR_RGB16:
856 case OMAP_DSS_COLOR_ARGB16:
857 case OMAP_DSS_COLOR_YUV2:
858 case OMAP_DSS_COLOR_UYVY:
Amber Jainf20e4222011-05-19 19:47:50 +0530859 case OMAP_DSS_COLOR_RGBA16:
860 case OMAP_DSS_COLOR_RGBX16:
861 case OMAP_DSS_COLOR_ARGB16_1555:
862 case OMAP_DSS_COLOR_XRGB16_1555:
Tomi Valkeineneed07e02009-08-07 13:43:20 +0300863 bpp = 16;
864 break;
865
866 case OMAP_DSS_COLOR_RGB24P:
867 bpp = 24;
868 break;
869
870 case OMAP_DSS_COLOR_RGB24U:
871 case OMAP_DSS_COLOR_ARGB32:
872 case OMAP_DSS_COLOR_RGBA32:
873 case OMAP_DSS_COLOR_RGBX32:
874 bpp = 32;
875 break;
876
877 default:
878 BUG();
879 }
880
Nishant Kamat4df9d102011-06-20 20:39:28 +0530881 if (mc->x > oi->pos_x) {
Tomi Valkeinen26a8c252010-06-09 15:31:34 +0300882 x = 0;
Nishant Kamat4df9d102011-06-20 20:39:28 +0530883 outw -= (mc->x - oi->pos_x);
884 paddr += (mc->x - oi->pos_x) *
Tomi Valkeinen26a8c252010-06-09 15:31:34 +0300885 scale_x_m / scale_x_d * bpp / 8;
Tomi Valkeineneed07e02009-08-07 13:43:20 +0300886 } else {
Nishant Kamat4df9d102011-06-20 20:39:28 +0530887 x = oi->pos_x - mc->x;
Tomi Valkeineneed07e02009-08-07 13:43:20 +0300888 }
Tomi Valkeinen26a8c252010-06-09 15:31:34 +0300889
Nishant Kamat4df9d102011-06-20 20:39:28 +0530890 if (mc->y > oi->pos_y) {
Tomi Valkeinen26a8c252010-06-09 15:31:34 +0300891 y = 0;
Nishant Kamat4df9d102011-06-20 20:39:28 +0530892 outh -= (mc->y - oi->pos_y);
893 paddr += (mc->y - oi->pos_y) *
Tomi Valkeinen26a8c252010-06-09 15:31:34 +0300894 scale_y_m / scale_y_d *
Nishant Kamat4df9d102011-06-20 20:39:28 +0530895 oi->screen_width * bpp / 8;
Tomi Valkeinen26a8c252010-06-09 15:31:34 +0300896 } else {
Nishant Kamat4df9d102011-06-20 20:39:28 +0530897 y = oi->pos_y - mc->y;
Tomi Valkeinen26a8c252010-06-09 15:31:34 +0300898 }
899
900 if (mc->w < (x + outw))
901 outw -= (x + outw) - (mc->w);
902
903 if (mc->h < (y + outh))
904 outh -= (y + outh) - (mc->h);
905
906 w = w * outw / orig_outw;
907 h = h * outh / orig_outh;
Tomi Valkeinenf55fdcf2010-06-03 16:27:46 +0300908
909 /* YUV mode overlay's input width has to be even and the
910 * algorithm above may adjust the width to be odd.
911 *
912 * Here we adjust the width if needed, preferring to increase
913 * the width if the original width was bigger.
914 */
915 if ((w & 1) &&
Nishant Kamat4df9d102011-06-20 20:39:28 +0530916 (oi->color_mode == OMAP_DSS_COLOR_YUV2 ||
917 oi->color_mode == OMAP_DSS_COLOR_UYVY)) {
Tomi Valkeinenf55fdcf2010-06-03 16:27:46 +0300918 if (orig_w > w)
919 w += 1;
920 else
921 w -= 1;
922 }
Tomi Valkeineneed07e02009-08-07 13:43:20 +0300923 }
924
925 r = dispc_setup_plane(plane,
926 paddr,
Nishant Kamat4df9d102011-06-20 20:39:28 +0530927 oi->screen_width,
Tomi Valkeineneed07e02009-08-07 13:43:20 +0300928 x, y,
929 w, h,
930 outw, outh,
Nishant Kamat4df9d102011-06-20 20:39:28 +0530931 oi->color_mode,
Tomi Valkeineneed07e02009-08-07 13:43:20 +0300932 c->ilace,
Nishant Kamat4df9d102011-06-20 20:39:28 +0530933 oi->rotation_type,
934 oi->rotation,
935 oi->mirror,
936 oi->global_alpha,
937 oi->pre_mult_alpha,
Amber Jain0d66cbb2011-05-19 19:47:54 +0530938 c->channel,
Nishant Kamat4df9d102011-06-20 20:39:28 +0530939 oi->p_uv_addr);
Tomi Valkeineneed07e02009-08-07 13:43:20 +0300940
941 if (r) {
942 /* this shouldn't happen */
943 DSSERR("dispc_setup_plane failed for ovl %d\n", plane);
944 dispc_enable_plane(plane, 0);
945 return r;
946 }
947
948 dispc_enable_replication(plane, c->replication);
949
950 dispc_set_burst_size(plane, c->burst_size);
951 dispc_setup_plane_fifo(plane, c->fifo_low, c->fifo_high);
952
953 dispc_enable_plane(plane, 1);
954
955 return 0;
956}
957
958static void configure_manager(enum omap_channel channel)
959{
Nishant Kamat4df9d102011-06-20 20:39:28 +0530960 struct omap_overlay_manager_info *mi;
Tomi Valkeineneed07e02009-08-07 13:43:20 +0300961
962 DSSDBGF("%d", channel);
963
Nishant Kamat4df9d102011-06-20 20:39:28 +0530964 /* picking info from the cache */
965 mi = &dss_cache.manager_cache[channel].info;
Tomi Valkeineneed07e02009-08-07 13:43:20 +0300966
Nishant Kamat4df9d102011-06-20 20:39:28 +0530967 dispc_set_default_color(channel, mi->default_color);
968 dispc_set_trans_key(channel, mi->trans_key_type, mi->trans_key);
969 dispc_enable_trans_key(channel, mi->trans_enabled);
970 dispc_enable_alpha_blending(channel, mi->alpha_enabled);
Tomi Valkeinen3c07cae2011-06-21 09:34:30 +0300971 if (dss_has_feature(FEAT_CPR)) {
972 dispc_enable_cpr(channel, mi->cpr_enable);
973 dispc_set_cpr_coef(channel, &mi->cpr_coefs);
974 }
Tomi Valkeineneed07e02009-08-07 13:43:20 +0300975}
976
977/* configure_dispc() tries to write values from cache to shadow registers.
978 * It writes only to those managers/overlays that are not busy.
979 * returns 0 if everything could be written to shadow registers.
980 * returns 1 if not everything could be written to shadow registers. */
981static int configure_dispc(void)
982{
983 struct overlay_cache_data *oc;
984 struct manager_cache_data *mc;
Archit Tanejaa0acb552010-09-15 19:20:00 +0530985 const int num_ovls = dss_feat_get_num_ovls();
986 const int num_mgrs = dss_feat_get_num_mgrs();
Tomi Valkeineneed07e02009-08-07 13:43:20 +0300987 int i;
988 int r;
Archit Tanejaa0acb552010-09-15 19:20:00 +0530989 bool mgr_busy[MAX_DSS_MANAGERS];
990 bool mgr_go[MAX_DSS_MANAGERS];
Tomi Valkeineneed07e02009-08-07 13:43:20 +0300991 bool busy;
992
993 r = 0;
994 busy = false;
995
Sumit Semwal18faa1b2010-12-02 11:27:14 +0000996 for (i = 0; i < num_mgrs; i++) {
997 mgr_busy[i] = dispc_go_busy(i);
998 mgr_go[i] = false;
999 }
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001000
1001 /* Commit overlay settings */
1002 for (i = 0; i < num_ovls; ++i) {
1003 oc = &dss_cache.overlay_cache[i];
1004 mc = &dss_cache.manager_cache[oc->channel];
1005
1006 if (!oc->dirty)
1007 continue;
1008
Tomi Valkeinen8cff88c2011-04-30 14:09:53 +03001009 if (mc->manual_update && !mc->do_manual_update)
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001010 continue;
1011
1012 if (mgr_busy[oc->channel]) {
1013 busy = true;
1014 continue;
1015 }
1016
1017 r = configure_overlay(i);
1018 if (r)
1019 DSSERR("configure_overlay %d failed\n", i);
1020
1021 oc->dirty = false;
1022 oc->shadow_dirty = true;
1023 mgr_go[oc->channel] = true;
1024 }
1025
1026 /* Commit manager settings */
1027 for (i = 0; i < num_mgrs; ++i) {
1028 mc = &dss_cache.manager_cache[i];
1029
1030 if (!mc->dirty)
1031 continue;
1032
1033 if (mc->manual_update && !mc->do_manual_update)
1034 continue;
1035
1036 if (mgr_busy[i]) {
1037 busy = true;
1038 continue;
1039 }
1040
1041 configure_manager(i);
1042 mc->dirty = false;
1043 mc->shadow_dirty = true;
1044 mgr_go[i] = true;
1045 }
1046
1047 /* set GO */
1048 for (i = 0; i < num_mgrs; ++i) {
1049 mc = &dss_cache.manager_cache[i];
1050
1051 if (!mgr_go[i])
1052 continue;
1053
1054 /* We don't need GO with manual update display. LCD iface will
1055 * always be turned off after frame, and new settings will be
1056 * taken in to use at next update */
Tomi Valkeinen8cff88c2011-04-30 14:09:53 +03001057 if (!mc->manual_update)
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001058 dispc_go(i);
1059 }
1060
1061 if (busy)
1062 r = 1;
1063 else
1064 r = 0;
1065
1066 return r;
1067}
1068
Tomi Valkeinenf49a9512010-03-26 16:26:37 +02001069/* Make the coordinates even. There are some strange problems with OMAP and
1070 * partial DSI update when the update widths are odd. */
1071static void make_even(u16 *x, u16 *w)
1072{
1073 u16 x1, x2;
1074
1075 x1 = *x;
1076 x2 = *x + *w;
1077
1078 x1 &= ~1;
1079 x2 = ALIGN(x2, 2);
1080
1081 *x = x1;
1082 *w = x2 - x1;
1083}
1084
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001085/* Configure dispc for partial update. Return possibly modified update
1086 * area */
1087void dss_setup_partial_planes(struct omap_dss_device *dssdev,
Tomi Valkeinen26a8c252010-06-09 15:31:34 +03001088 u16 *xi, u16 *yi, u16 *wi, u16 *hi, bool enlarge_update_area)
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001089{
1090 struct overlay_cache_data *oc;
1091 struct manager_cache_data *mc;
Nishant Kamat4df9d102011-06-20 20:39:28 +05301092 struct omap_overlay_info *oi;
Archit Tanejaa0acb552010-09-15 19:20:00 +05301093 const int num_ovls = dss_feat_get_num_ovls();
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001094 struct omap_overlay_manager *mgr;
1095 int i;
1096 u16 x, y, w, h;
1097 unsigned long flags;
Tomi Valkeinen8cab90f2010-06-09 15:09:30 +03001098 bool area_changed;
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001099
1100 x = *xi;
1101 y = *yi;
1102 w = *wi;
1103 h = *hi;
1104
1105 DSSDBG("dispc_setup_partial_planes %d,%d %dx%d\n",
1106 *xi, *yi, *wi, *hi);
1107
1108 mgr = dssdev->manager;
1109
1110 if (!mgr) {
1111 DSSDBG("no manager\n");
1112 return;
1113 }
1114
Tomi Valkeinenf49a9512010-03-26 16:26:37 +02001115 make_even(&x, &w);
1116
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001117 spin_lock_irqsave(&dss_cache.lock, flags);
1118
Tomi Valkeinen8cab90f2010-06-09 15:09:30 +03001119 /*
1120 * Execute the outer loop until the inner loop has completed
1121 * once without increasing the update area. This will ensure that
1122 * all scaled overlays end up completely within the update area.
1123 */
1124 do {
1125 area_changed = false;
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001126
Tomi Valkeinen8cab90f2010-06-09 15:09:30 +03001127 /* We need to show the whole overlay if it is scaled. So look
1128 * for those, and make the update area larger if found.
1129 * Also mark the overlay cache dirty */
1130 for (i = 0; i < num_ovls; ++i) {
1131 unsigned x1, y1, x2, y2;
1132 unsigned outw, outh;
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001133
Tomi Valkeinen8cab90f2010-06-09 15:09:30 +03001134 oc = &dss_cache.overlay_cache[i];
Nishant Kamat4df9d102011-06-20 20:39:28 +05301135 oi = &oc->info;
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001136
Tomi Valkeinen8cab90f2010-06-09 15:09:30 +03001137 if (oc->channel != mgr->id)
1138 continue;
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001139
Tomi Valkeinen8cab90f2010-06-09 15:09:30 +03001140 oc->dirty = true;
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001141
Tomi Valkeinen26a8c252010-06-09 15:31:34 +03001142 if (!enlarge_update_area)
1143 continue;
1144
Tomi Valkeinen8cab90f2010-06-09 15:09:30 +03001145 if (!oc->enabled)
1146 continue;
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001147
Tomi Valkeinen8cab90f2010-06-09 15:09:30 +03001148 if (!dispc_is_overlay_scaled(oc))
1149 continue;
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001150
Nishant Kamat4df9d102011-06-20 20:39:28 +05301151 outw = oi->out_width == 0 ?
1152 oi->width : oi->out_width;
1153 outh = oi->out_height == 0 ?
1154 oi->height : oi->out_height;
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001155
Tomi Valkeinen8cab90f2010-06-09 15:09:30 +03001156 /* is the overlay outside the update region? */
1157 if (!rectangle_intersects(x, y, w, h,
Nishant Kamat4df9d102011-06-20 20:39:28 +05301158 oi->pos_x, oi->pos_y,
Tomi Valkeinen8cab90f2010-06-09 15:09:30 +03001159 outw, outh))
1160 continue;
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001161
Tomi Valkeinen8cab90f2010-06-09 15:09:30 +03001162 /* if the overlay totally inside the update region? */
Nishant Kamat4df9d102011-06-20 20:39:28 +05301163 if (rectangle_subset(oi->pos_x, oi->pos_y, outw, outh,
Tomi Valkeinen8cab90f2010-06-09 15:09:30 +03001164 x, y, w, h))
1165 continue;
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001166
Nishant Kamat4df9d102011-06-20 20:39:28 +05301167 if (x > oi->pos_x)
1168 x1 = oi->pos_x;
Tomi Valkeinen8cab90f2010-06-09 15:09:30 +03001169 else
1170 x1 = x;
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001171
Nishant Kamat4df9d102011-06-20 20:39:28 +05301172 if (y > oi->pos_y)
1173 y1 = oi->pos_y;
Tomi Valkeinen8cab90f2010-06-09 15:09:30 +03001174 else
1175 y1 = y;
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001176
Nishant Kamat4df9d102011-06-20 20:39:28 +05301177 if ((x + w) < (oi->pos_x + outw))
1178 x2 = oi->pos_x + outw;
Tomi Valkeinen8cab90f2010-06-09 15:09:30 +03001179 else
1180 x2 = x + w;
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001181
Nishant Kamat4df9d102011-06-20 20:39:28 +05301182 if ((y + h) < (oi->pos_y + outh))
1183 y2 = oi->pos_y + outh;
Tomi Valkeinen8cab90f2010-06-09 15:09:30 +03001184 else
1185 y2 = y + h;
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001186
Tomi Valkeinen8cab90f2010-06-09 15:09:30 +03001187 x = x1;
1188 y = y1;
1189 w = x2 - x1;
1190 h = y2 - y1;
Tomi Valkeinenf49a9512010-03-26 16:26:37 +02001191
Tomi Valkeinen8cab90f2010-06-09 15:09:30 +03001192 make_even(&x, &w);
1193
1194 DSSDBG("changing upd area due to ovl(%d) "
1195 "scaling %d,%d %dx%d\n",
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001196 i, x, y, w, h);
Tomi Valkeinen8cab90f2010-06-09 15:09:30 +03001197
1198 area_changed = true;
1199 }
1200 } while (area_changed);
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001201
1202 mc = &dss_cache.manager_cache[mgr->id];
1203 mc->do_manual_update = true;
Tomi Valkeinen26a8c252010-06-09 15:31:34 +03001204 mc->enlarge_update_area = enlarge_update_area;
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001205 mc->x = x;
1206 mc->y = y;
1207 mc->w = w;
1208 mc->h = h;
1209
1210 configure_dispc();
1211
1212 mc->do_manual_update = false;
1213
1214 spin_unlock_irqrestore(&dss_cache.lock, flags);
1215
1216 *xi = x;
1217 *yi = y;
1218 *wi = w;
1219 *hi = h;
1220}
1221
1222void dss_start_update(struct omap_dss_device *dssdev)
1223{
1224 struct manager_cache_data *mc;
1225 struct overlay_cache_data *oc;
Archit Tanejaa0acb552010-09-15 19:20:00 +05301226 const int num_ovls = dss_feat_get_num_ovls();
1227 const int num_mgrs = dss_feat_get_num_mgrs();
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001228 struct omap_overlay_manager *mgr;
1229 int i;
1230
1231 mgr = dssdev->manager;
1232
1233 for (i = 0; i < num_ovls; ++i) {
1234 oc = &dss_cache.overlay_cache[i];
1235 if (oc->channel != mgr->id)
1236 continue;
1237
1238 oc->shadow_dirty = false;
1239 }
1240
1241 for (i = 0; i < num_mgrs; ++i) {
1242 mc = &dss_cache.manager_cache[i];
1243 if (mgr->id != i)
1244 continue;
1245
1246 mc->shadow_dirty = false;
1247 }
1248
Tomi Valkeinena2faee82010-01-08 17:14:53 +02001249 dssdev->manager->enable(dssdev->manager);
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001250}
1251
1252static void dss_apply_irq_handler(void *data, u32 mask)
1253{
1254 struct manager_cache_data *mc;
1255 struct overlay_cache_data *oc;
Archit Tanejaa0acb552010-09-15 19:20:00 +05301256 const int num_ovls = dss_feat_get_num_ovls();
1257 const int num_mgrs = dss_feat_get_num_mgrs();
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001258 int i, r;
Archit Tanejaa0acb552010-09-15 19:20:00 +05301259 bool mgr_busy[MAX_DSS_MANAGERS];
Sumit Semwal18faa1b2010-12-02 11:27:14 +00001260 u32 irq_mask;
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001261
Sumit Semwal18faa1b2010-12-02 11:27:14 +00001262 for (i = 0; i < num_mgrs; i++)
1263 mgr_busy[i] = dispc_go_busy(i);
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001264
1265 spin_lock(&dss_cache.lock);
1266
1267 for (i = 0; i < num_ovls; ++i) {
1268 oc = &dss_cache.overlay_cache[i];
1269 if (!mgr_busy[oc->channel])
1270 oc->shadow_dirty = false;
1271 }
1272
1273 for (i = 0; i < num_mgrs; ++i) {
1274 mc = &dss_cache.manager_cache[i];
1275 if (!mgr_busy[i])
1276 mc->shadow_dirty = false;
1277 }
1278
1279 r = configure_dispc();
1280 if (r == 1)
1281 goto end;
1282
1283 /* re-read busy flags */
Sumit Semwal18faa1b2010-12-02 11:27:14 +00001284 for (i = 0; i < num_mgrs; i++)
1285 mgr_busy[i] = dispc_go_busy(i);
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001286
1287 /* keep running as long as there are busy managers, so that
1288 * we can collect overlay-applied information */
1289 for (i = 0; i < num_mgrs; ++i) {
1290 if (mgr_busy[i])
1291 goto end;
1292 }
1293
Sumit Semwal18faa1b2010-12-02 11:27:14 +00001294 irq_mask = DISPC_IRQ_VSYNC | DISPC_IRQ_EVSYNC_ODD |
1295 DISPC_IRQ_EVSYNC_EVEN;
1296 if (dss_has_feature(FEAT_MGR_LCD2))
1297 irq_mask |= DISPC_IRQ_VSYNC2;
1298
1299 omap_dispc_unregister_isr(dss_apply_irq_handler, NULL, irq_mask);
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001300 dss_cache.irq_enabled = false;
1301
1302end:
1303 spin_unlock(&dss_cache.lock);
1304}
1305
1306static int omap_dss_mgr_apply(struct omap_overlay_manager *mgr)
1307{
1308 struct overlay_cache_data *oc;
1309 struct manager_cache_data *mc;
1310 int i;
1311 struct omap_overlay *ovl;
1312 int num_planes_enabled = 0;
1313 bool use_fifomerge;
1314 unsigned long flags;
1315 int r;
1316
1317 DSSDBG("omap_dss_mgr_apply(%s)\n", mgr->name);
1318
1319 spin_lock_irqsave(&dss_cache.lock, flags);
1320
1321 /* Configure overlays */
1322 for (i = 0; i < omap_dss_get_num_overlays(); ++i) {
1323 struct omap_dss_device *dssdev;
1324
1325 ovl = omap_dss_get_overlay(i);
1326
1327 if (!(ovl->caps & OMAP_DSS_OVL_CAP_DISPC))
1328 continue;
1329
1330 oc = &dss_cache.overlay_cache[ovl->id];
1331
1332 if (!overlay_enabled(ovl)) {
1333 if (oc->enabled) {
1334 oc->enabled = false;
1335 oc->dirty = true;
1336 }
1337 continue;
1338 }
1339
1340 if (!ovl->info_dirty) {
1341 if (oc->enabled)
1342 ++num_planes_enabled;
1343 continue;
1344 }
1345
1346 dssdev = ovl->manager->device;
1347
1348 if (dss_check_overlay(ovl, dssdev)) {
1349 if (oc->enabled) {
1350 oc->enabled = false;
1351 oc->dirty = true;
1352 }
1353 continue;
1354 }
1355
1356 ovl->info_dirty = false;
1357 oc->dirty = true;
Nishant Kamat4df9d102011-06-20 20:39:28 +05301358 oc->info = ovl->info;
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001359
1360 oc->replication =
1361 dss_use_replication(dssdev, ovl->info.color_mode);
1362
1363 oc->ilace = dssdev->type == OMAP_DISPLAY_TYPE_VENC;
1364
1365 oc->channel = ovl->manager->id;
1366
1367 oc->enabled = true;
1368
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001369 ++num_planes_enabled;
1370 }
1371
1372 /* Configure managers */
1373 list_for_each_entry(mgr, &manager_list, list) {
1374 struct omap_dss_device *dssdev;
1375
1376 if (!(mgr->caps & OMAP_DSS_OVL_MGR_CAP_DISPC))
1377 continue;
1378
1379 mc = &dss_cache.manager_cache[mgr->id];
1380
1381 if (mgr->device_changed) {
1382 mgr->device_changed = false;
1383 mgr->info_dirty = true;
1384 }
1385
1386 if (!mgr->info_dirty)
1387 continue;
1388
1389 if (!mgr->device)
1390 continue;
1391
1392 dssdev = mgr->device;
1393
1394 mgr->info_dirty = false;
1395 mc->dirty = true;
Nishant Kamat4df9d102011-06-20 20:39:28 +05301396 mc->info = mgr->info;
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001397
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001398 mc->manual_update =
Tomi Valkeinen8cff88c2011-04-30 14:09:53 +03001399 dssdev->caps & OMAP_DSS_DISPLAY_CAP_MANUAL_UPDATE;
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001400 }
1401
1402 /* XXX TODO: Try to get fifomerge working. The problem is that it
1403 * affects both managers, not individually but at the same time. This
1404 * means the change has to be well synchronized. I guess the proper way
1405 * is to have a two step process for fifo merge:
1406 * fifomerge enable:
1407 * 1. disable other planes, leaving one plane enabled
1408 * 2. wait until the planes are disabled on HW
1409 * 3. config merged fifo thresholds, enable fifomerge
1410 * fifomerge disable:
1411 * 1. config unmerged fifo thresholds, disable fifomerge
1412 * 2. wait until fifo changes are in HW
1413 * 3. enable planes
1414 */
1415 use_fifomerge = false;
1416
1417 /* Configure overlay fifos */
1418 for (i = 0; i < omap_dss_get_num_overlays(); ++i) {
1419 struct omap_dss_device *dssdev;
1420 u32 size;
1421
1422 ovl = omap_dss_get_overlay(i);
1423
1424 if (!(ovl->caps & OMAP_DSS_OVL_CAP_DISPC))
1425 continue;
1426
1427 oc = &dss_cache.overlay_cache[ovl->id];
1428
1429 if (!oc->enabled)
1430 continue;
1431
1432 dssdev = ovl->manager->device;
1433
1434 size = dispc_get_plane_fifo_size(ovl->id);
1435 if (use_fifomerge)
1436 size *= 3;
1437
1438 switch (dssdev->type) {
1439 case OMAP_DISPLAY_TYPE_DPI:
1440 case OMAP_DISPLAY_TYPE_DBI:
1441 case OMAP_DISPLAY_TYPE_SDI:
1442 case OMAP_DISPLAY_TYPE_VENC:
Mythri P Kb1196012011-03-08 17:15:54 +05301443 case OMAP_DISPLAY_TYPE_HDMI:
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001444 default_get_overlay_fifo_thresholds(ovl->id, size,
1445 &oc->burst_size, &oc->fifo_low,
1446 &oc->fifo_high);
1447 break;
1448#ifdef CONFIG_OMAP2_DSS_DSI
1449 case OMAP_DISPLAY_TYPE_DSI:
1450 dsi_get_overlay_fifo_thresholds(ovl->id, size,
1451 &oc->burst_size, &oc->fifo_low,
1452 &oc->fifo_high);
1453 break;
1454#endif
1455 default:
1456 BUG();
1457 }
1458 }
1459
1460 r = 0;
Archit Taneja6af9cd12011-01-31 16:27:44 +00001461 dss_clk_enable(DSS_CLK_ICK | DSS_CLK_FCK);
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001462 if (!dss_cache.irq_enabled) {
Sumit Semwal18faa1b2010-12-02 11:27:14 +00001463 u32 mask;
1464
1465 mask = DISPC_IRQ_VSYNC | DISPC_IRQ_EVSYNC_ODD |
1466 DISPC_IRQ_EVSYNC_EVEN;
1467 if (dss_has_feature(FEAT_MGR_LCD2))
1468 mask |= DISPC_IRQ_VSYNC2;
1469
1470 r = omap_dispc_register_isr(dss_apply_irq_handler, NULL, mask);
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001471 dss_cache.irq_enabled = true;
1472 }
1473 configure_dispc();
Archit Taneja6af9cd12011-01-31 16:27:44 +00001474 dss_clk_disable(DSS_CLK_ICK | DSS_CLK_FCK);
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001475
1476 spin_unlock_irqrestore(&dss_cache.lock, flags);
1477
1478 return r;
1479}
1480
1481static int dss_check_manager(struct omap_overlay_manager *mgr)
1482{
1483 /* OMAP supports only graphics source transparency color key and alpha
1484 * blending simultaneously. See TRM 15.4.2.4.2.2 Alpha Mode */
1485
1486 if (mgr->info.alpha_enabled && mgr->info.trans_enabled &&
1487 mgr->info.trans_key_type != OMAP_DSS_COLOR_KEY_GFX_DST)
1488 return -EINVAL;
1489
1490 return 0;
1491}
1492
1493static int omap_dss_mgr_set_info(struct omap_overlay_manager *mgr,
1494 struct omap_overlay_manager_info *info)
1495{
1496 int r;
1497 struct omap_overlay_manager_info old_info;
1498
1499 old_info = mgr->info;
1500 mgr->info = *info;
1501
1502 r = dss_check_manager(mgr);
1503 if (r) {
1504 mgr->info = old_info;
1505 return r;
1506 }
1507
1508 mgr->info_dirty = true;
1509
1510 return 0;
1511}
1512
1513static void omap_dss_mgr_get_info(struct omap_overlay_manager *mgr,
1514 struct omap_overlay_manager_info *info)
1515{
1516 *info = mgr->info;
1517}
1518
Tomi Valkeinena2faee82010-01-08 17:14:53 +02001519static int dss_mgr_enable(struct omap_overlay_manager *mgr)
1520{
1521 dispc_enable_channel(mgr->id, 1);
1522 return 0;
1523}
1524
1525static int dss_mgr_disable(struct omap_overlay_manager *mgr)
1526{
1527 dispc_enable_channel(mgr->id, 0);
1528 return 0;
1529}
1530
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001531static void omap_dss_add_overlay_manager(struct omap_overlay_manager *manager)
1532{
1533 ++num_managers;
1534 list_add_tail(&manager->list, &manager_list);
1535}
1536
1537int dss_init_overlay_managers(struct platform_device *pdev)
1538{
1539 int i, r;
1540
1541 spin_lock_init(&dss_cache.lock);
1542
1543 INIT_LIST_HEAD(&manager_list);
1544
1545 num_managers = 0;
1546
Archit Tanejaa0acb552010-09-15 19:20:00 +05301547 for (i = 0; i < dss_feat_get_num_mgrs(); ++i) {
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001548 struct omap_overlay_manager *mgr;
1549 mgr = kzalloc(sizeof(*mgr), GFP_KERNEL);
1550
1551 BUG_ON(mgr == NULL);
1552
1553 switch (i) {
1554 case 0:
1555 mgr->name = "lcd";
1556 mgr->id = OMAP_DSS_CHANNEL_LCD;
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001557 break;
1558 case 1:
1559 mgr->name = "tv";
1560 mgr->id = OMAP_DSS_CHANNEL_DIGIT;
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001561 break;
Sumit Semwal18faa1b2010-12-02 11:27:14 +00001562 case 2:
1563 mgr->name = "lcd2";
1564 mgr->id = OMAP_DSS_CHANNEL_LCD2;
1565 break;
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001566 }
1567
1568 mgr->set_device = &omap_dss_set_device;
1569 mgr->unset_device = &omap_dss_unset_device;
1570 mgr->apply = &omap_dss_mgr_apply;
1571 mgr->set_manager_info = &omap_dss_mgr_set_info;
1572 mgr->get_manager_info = &omap_dss_mgr_get_info;
1573 mgr->wait_for_go = &dss_mgr_wait_for_go;
Tomi Valkeinen3f71cbe2010-01-08 17:06:04 +02001574 mgr->wait_for_vsync = &dss_mgr_wait_for_vsync;
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001575
Tomi Valkeinena2faee82010-01-08 17:14:53 +02001576 mgr->enable = &dss_mgr_enable;
1577 mgr->disable = &dss_mgr_disable;
1578
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001579 mgr->caps = OMAP_DSS_OVL_MGR_CAP_DISPC;
Archit Tanejaa0acb552010-09-15 19:20:00 +05301580 mgr->supported_displays =
1581 dss_feat_get_supported_displays(mgr->id);
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001582
1583 dss_overlay_setup_dispc_manager(mgr);
1584
1585 omap_dss_add_overlay_manager(mgr);
1586
1587 r = kobject_init_and_add(&mgr->kobj, &manager_ktype,
1588 &pdev->dev.kobj, "manager%d", i);
1589
1590 if (r) {
1591 DSSERR("failed to create sysfs file\n");
1592 continue;
1593 }
1594 }
1595
1596#ifdef L4_EXAMPLE
1597 {
1598 int omap_dss_mgr_apply_l4(struct omap_overlay_manager *mgr)
1599 {
1600 DSSDBG("omap_dss_mgr_apply_l4(%s)\n", mgr->name);
1601
1602 return 0;
1603 }
1604
1605 struct omap_overlay_manager *mgr;
1606 mgr = kzalloc(sizeof(*mgr), GFP_KERNEL);
1607
1608 BUG_ON(mgr == NULL);
1609
1610 mgr->name = "l4";
1611 mgr->supported_displays =
1612 OMAP_DISPLAY_TYPE_DBI | OMAP_DISPLAY_TYPE_DSI;
1613
1614 mgr->set_device = &omap_dss_set_device;
1615 mgr->unset_device = &omap_dss_unset_device;
1616 mgr->apply = &omap_dss_mgr_apply_l4;
1617 mgr->set_manager_info = &omap_dss_mgr_set_info;
1618 mgr->get_manager_info = &omap_dss_mgr_get_info;
1619
1620 dss_overlay_setup_l4_manager(mgr);
1621
1622 omap_dss_add_overlay_manager(mgr);
1623
1624 r = kobject_init_and_add(&mgr->kobj, &manager_ktype,
1625 &pdev->dev.kobj, "managerl4");
1626
1627 if (r)
1628 DSSERR("failed to create sysfs file\n");
1629 }
1630#endif
1631
1632 return 0;
1633}
1634
1635void dss_uninit_overlay_managers(struct platform_device *pdev)
1636{
1637 struct omap_overlay_manager *mgr;
1638
1639 while (!list_empty(&manager_list)) {
1640 mgr = list_first_entry(&manager_list,
1641 struct omap_overlay_manager, list);
1642 list_del(&mgr->list);
1643 kobject_del(&mgr->kobj);
1644 kobject_put(&mgr->kobj);
1645 kfree(mgr);
1646 }
1647
1648 num_managers = 0;
1649}
1650
1651int omap_dss_get_num_overlay_managers(void)
1652{
1653 return num_managers;
1654}
1655EXPORT_SYMBOL(omap_dss_get_num_overlay_managers);
1656
1657struct omap_overlay_manager *omap_dss_get_overlay_manager(int num)
1658{
1659 int i = 0;
1660 struct omap_overlay_manager *mgr;
1661
1662 list_for_each_entry(mgr, &manager_list, list) {
1663 if (i++ == num)
1664 return mgr;
1665 }
1666
1667 return NULL;
1668}
1669EXPORT_SYMBOL(omap_dss_get_overlay_manager);
1670