blob: 63674b054d35992d46ed52bb579408dc6f171f04 [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
Tomi Valkeineneed07e02009-08-07 13:43:20 +0300510 u32 fifo_low;
511 u32 fifo_high;
Tomi Valkeineneed07e02009-08-07 13:43:20 +0300512};
513
514struct manager_cache_data {
515 /* If true, cache changed, but not written to shadow registers. Set
516 * in apply(), cleared when registers written. */
517 bool dirty;
518 /* If true, shadow registers contain changed values not yet in real
519 * registers. Set when writing to shadow registers, cleared at
520 * VSYNC/EVSYNC */
521 bool shadow_dirty;
522
Nishant Kamat4df9d102011-06-20 20:39:28 +0530523 struct omap_overlay_manager_info info;
Tomi Valkeineneed07e02009-08-07 13:43:20 +0300524
Tomi Valkeineneed07e02009-08-07 13:43:20 +0300525 bool manual_update;
526 bool do_manual_update;
527
528 /* manual update region */
529 u16 x, y, w, h;
Tomi Valkeinen26a8c252010-06-09 15:31:34 +0300530
531 /* enlarge the update area if the update area contains scaled
532 * overlays */
533 bool enlarge_update_area;
Tomi Valkeineneed07e02009-08-07 13:43:20 +0300534};
535
536static struct {
537 spinlock_t lock;
Archit Tanejaa0acb552010-09-15 19:20:00 +0530538 struct overlay_cache_data overlay_cache[MAX_DSS_OVERLAYS];
539 struct manager_cache_data manager_cache[MAX_DSS_MANAGERS];
Tomi Valkeineneed07e02009-08-07 13:43:20 +0300540
541 bool irq_enabled;
542} dss_cache;
543
544
545
546static int omap_dss_set_device(struct omap_overlay_manager *mgr,
547 struct omap_dss_device *dssdev)
548{
549 int i;
550 int r;
551
552 if (dssdev->manager) {
553 DSSERR("display '%s' already has a manager '%s'\n",
554 dssdev->name, dssdev->manager->name);
555 return -EINVAL;
556 }
557
558 if ((mgr->supported_displays & dssdev->type) == 0) {
559 DSSERR("display '%s' does not support manager '%s'\n",
560 dssdev->name, mgr->name);
561 return -EINVAL;
562 }
563
564 for (i = 0; i < mgr->num_overlays; i++) {
565 struct omap_overlay *ovl = mgr->overlays[i];
566
567 if (ovl->manager != mgr || !ovl->info.enabled)
568 continue;
569
570 r = dss_check_overlay(ovl, dssdev);
571 if (r)
572 return r;
573 }
574
575 dssdev->manager = mgr;
576 mgr->device = dssdev;
577 mgr->device_changed = true;
578
579 return 0;
580}
581
582static int omap_dss_unset_device(struct omap_overlay_manager *mgr)
583{
584 if (!mgr->device) {
585 DSSERR("failed to unset display, display not set.\n");
586 return -EINVAL;
587 }
588
Daniel Morsing0f770b42011-08-03 22:10:51 +0200589 /*
590 * Don't allow currently enabled displays to have the overlay manager
591 * pulled out from underneath them
592 */
593 if (mgr->device->state != OMAP_DSS_DISPLAY_DISABLED)
594 return -EINVAL;
595
Tomi Valkeineneed07e02009-08-07 13:43:20 +0300596 mgr->device->manager = NULL;
597 mgr->device = NULL;
598 mgr->device_changed = true;
599
600 return 0;
601}
602
Tomi Valkeinen3f71cbe2010-01-08 17:06:04 +0200603static int dss_mgr_wait_for_vsync(struct omap_overlay_manager *mgr)
604{
605 unsigned long timeout = msecs_to_jiffies(500);
606 u32 irq;
607
Sumit Semwal18faa1b2010-12-02 11:27:14 +0000608 if (mgr->device->type == OMAP_DISPLAY_TYPE_VENC) {
Tomi Valkeinen3f71cbe2010-01-08 17:06:04 +0200609 irq = DISPC_IRQ_EVSYNC_ODD;
Mythri P Kb1196012011-03-08 17:15:54 +0530610 } else if (mgr->device->type == OMAP_DISPLAY_TYPE_HDMI) {
611 irq = DISPC_IRQ_EVSYNC_EVEN;
Sumit Semwal18faa1b2010-12-02 11:27:14 +0000612 } else {
613 if (mgr->id == OMAP_DSS_CHANNEL_LCD)
614 irq = DISPC_IRQ_VSYNC;
615 else
616 irq = DISPC_IRQ_VSYNC2;
617 }
Tomi Valkeinen3f71cbe2010-01-08 17:06:04 +0200618 return omap_dispc_wait_for_irq_interruptible_timeout(irq, timeout);
619}
620
Tomi Valkeineneed07e02009-08-07 13:43:20 +0300621static int dss_mgr_wait_for_go(struct omap_overlay_manager *mgr)
622{
623 unsigned long timeout = msecs_to_jiffies(500);
624 struct manager_cache_data *mc;
Tomi Valkeineneed07e02009-08-07 13:43:20 +0300625 u32 irq;
626 int r;
627 int i;
Tomi Valkeinen446f7bf2010-01-11 16:12:31 +0200628 struct omap_dss_device *dssdev = mgr->device;
Tomi Valkeineneed07e02009-08-07 13:43:20 +0300629
Ville Syrjäläa74b2602010-03-04 16:03:56 +0200630 if (!dssdev || dssdev->state != OMAP_DSS_DISPLAY_ACTIVE)
Tomi Valkeineneed07e02009-08-07 13:43:20 +0300631 return 0;
632
Tomi Valkeinen8cff88c2011-04-30 14:09:53 +0300633 if (dssdev->caps & OMAP_DSS_DISPLAY_CAP_MANUAL_UPDATE)
634 return 0;
635
Mythri P Kb1196012011-03-08 17:15:54 +0530636 if (dssdev->type == OMAP_DISPLAY_TYPE_VENC
637 || dssdev->type == OMAP_DISPLAY_TYPE_HDMI) {
Tomi Valkeineneed07e02009-08-07 13:43:20 +0300638 irq = DISPC_IRQ_EVSYNC_ODD | DISPC_IRQ_EVSYNC_EVEN;
Tomi Valkeineneed07e02009-08-07 13:43:20 +0300639 } else {
Tomi Valkeinen8cff88c2011-04-30 14:09:53 +0300640 irq = (dssdev->manager->id == OMAP_DSS_CHANNEL_LCD) ?
641 DISPC_IRQ_VSYNC : DISPC_IRQ_VSYNC2;
Tomi Valkeineneed07e02009-08-07 13:43:20 +0300642 }
643
644 mc = &dss_cache.manager_cache[mgr->id];
645 i = 0;
646 while (1) {
647 unsigned long flags;
648 bool shadow_dirty, dirty;
649
650 spin_lock_irqsave(&dss_cache.lock, flags);
651 dirty = mc->dirty;
652 shadow_dirty = mc->shadow_dirty;
653 spin_unlock_irqrestore(&dss_cache.lock, flags);
654
655 if (!dirty && !shadow_dirty) {
656 r = 0;
657 break;
658 }
659
660 /* 4 iterations is the worst case:
661 * 1 - initial iteration, dirty = true (between VFP and VSYNC)
662 * 2 - first VSYNC, dirty = true
663 * 3 - dirty = false, shadow_dirty = true
664 * 4 - shadow_dirty = false */
665 if (i++ == 3) {
666 DSSERR("mgr(%d)->wait_for_go() not finishing\n",
667 mgr->id);
668 r = 0;
669 break;
670 }
671
672 r = omap_dispc_wait_for_irq_interruptible_timeout(irq, timeout);
673 if (r == -ERESTARTSYS)
674 break;
675
676 if (r) {
677 DSSERR("mgr(%d)->wait_for_go() timeout\n", mgr->id);
678 break;
679 }
680 }
681
682 return r;
683}
684
685int dss_mgr_wait_for_go_ovl(struct omap_overlay *ovl)
686{
687 unsigned long timeout = msecs_to_jiffies(500);
Tomi Valkeineneed07e02009-08-07 13:43:20 +0300688 struct overlay_cache_data *oc;
689 struct omap_dss_device *dssdev;
690 u32 irq;
691 int r;
692 int i;
693
Ville Syrjäläa74b2602010-03-04 16:03:56 +0200694 if (!ovl->manager)
Tomi Valkeineneed07e02009-08-07 13:43:20 +0300695 return 0;
696
697 dssdev = ovl->manager->device;
698
Ville Syrjäläa74b2602010-03-04 16:03:56 +0200699 if (!dssdev || dssdev->state != OMAP_DSS_DISPLAY_ACTIVE)
700 return 0;
701
Tomi Valkeinen8cff88c2011-04-30 14:09:53 +0300702 if (dssdev->caps & OMAP_DSS_DISPLAY_CAP_MANUAL_UPDATE)
703 return 0;
704
Mythri P Kb1196012011-03-08 17:15:54 +0530705 if (dssdev->type == OMAP_DISPLAY_TYPE_VENC
706 || dssdev->type == OMAP_DISPLAY_TYPE_HDMI) {
Tomi Valkeineneed07e02009-08-07 13:43:20 +0300707 irq = DISPC_IRQ_EVSYNC_ODD | DISPC_IRQ_EVSYNC_EVEN;
Tomi Valkeineneed07e02009-08-07 13:43:20 +0300708 } else {
Tomi Valkeinen8cff88c2011-04-30 14:09:53 +0300709 irq = (dssdev->manager->id == OMAP_DSS_CHANNEL_LCD) ?
710 DISPC_IRQ_VSYNC : DISPC_IRQ_VSYNC2;
Tomi Valkeineneed07e02009-08-07 13:43:20 +0300711 }
712
713 oc = &dss_cache.overlay_cache[ovl->id];
714 i = 0;
715 while (1) {
716 unsigned long flags;
717 bool shadow_dirty, dirty;
718
719 spin_lock_irqsave(&dss_cache.lock, flags);
720 dirty = oc->dirty;
721 shadow_dirty = oc->shadow_dirty;
722 spin_unlock_irqrestore(&dss_cache.lock, flags);
723
724 if (!dirty && !shadow_dirty) {
725 r = 0;
726 break;
727 }
728
729 /* 4 iterations is the worst case:
730 * 1 - initial iteration, dirty = true (between VFP and VSYNC)
731 * 2 - first VSYNC, dirty = true
732 * 3 - dirty = false, shadow_dirty = true
733 * 4 - shadow_dirty = false */
734 if (i++ == 3) {
735 DSSERR("ovl(%d)->wait_for_go() not finishing\n",
736 ovl->id);
737 r = 0;
738 break;
739 }
740
741 r = omap_dispc_wait_for_irq_interruptible_timeout(irq, timeout);
742 if (r == -ERESTARTSYS)
743 break;
744
745 if (r) {
746 DSSERR("ovl(%d)->wait_for_go() timeout\n", ovl->id);
747 break;
748 }
749 }
750
751 return r;
752}
753
754static int overlay_enabled(struct omap_overlay *ovl)
755{
756 return ovl->info.enabled && ovl->manager && ovl->manager->device;
757}
758
759/* Is rect1 a subset of rect2? */
760static bool rectangle_subset(int x1, int y1, int w1, int h1,
761 int x2, int y2, int w2, int h2)
762{
763 if (x1 < x2 || y1 < y2)
764 return false;
765
766 if (x1 + w1 > x2 + w2)
767 return false;
768
769 if (y1 + h1 > y2 + h2)
770 return false;
771
772 return true;
773}
774
775/* Do rect1 and rect2 overlap? */
776static bool rectangle_intersects(int x1, int y1, int w1, int h1,
777 int x2, int y2, int w2, int h2)
778{
779 if (x1 >= x2 + w2)
780 return false;
781
782 if (x2 >= x1 + w1)
783 return false;
784
785 if (y1 >= y2 + h2)
786 return false;
787
788 if (y2 >= y1 + h1)
789 return false;
790
791 return true;
792}
793
794static bool dispc_is_overlay_scaled(struct overlay_cache_data *oc)
795{
Nishant Kamat4df9d102011-06-20 20:39:28 +0530796 struct omap_overlay_info *oi = &oc->info;
797
798 if (oi->out_width != 0 && oi->width != oi->out_width)
Tomi Valkeineneed07e02009-08-07 13:43:20 +0300799 return true;
800
Nishant Kamat4df9d102011-06-20 20:39:28 +0530801 if (oi->out_height != 0 && oi->height != oi->out_height)
Tomi Valkeineneed07e02009-08-07 13:43:20 +0300802 return true;
803
804 return false;
805}
806
807static int configure_overlay(enum omap_plane plane)
808{
809 struct overlay_cache_data *c;
810 struct manager_cache_data *mc;
Nishant Kamat4df9d102011-06-20 20:39:28 +0530811 struct omap_overlay_info *oi;
812 struct omap_overlay_manager_info *mi;
Tomi Valkeineneed07e02009-08-07 13:43:20 +0300813 u16 outw, outh;
814 u16 x, y, w, h;
815 u32 paddr;
816 int r;
Tomi Valkeinen26a8c252010-06-09 15:31:34 +0300817 u16 orig_w, orig_h, orig_outw, orig_outh;
Tomi Valkeineneed07e02009-08-07 13:43:20 +0300818
819 DSSDBGF("%d", plane);
820
821 c = &dss_cache.overlay_cache[plane];
Nishant Kamat4df9d102011-06-20 20:39:28 +0530822 oi = &c->info;
Tomi Valkeineneed07e02009-08-07 13:43:20 +0300823
824 if (!c->enabled) {
825 dispc_enable_plane(plane, 0);
826 return 0;
827 }
828
829 mc = &dss_cache.manager_cache[c->channel];
Nishant Kamat4df9d102011-06-20 20:39:28 +0530830 mi = &mc->info;
Tomi Valkeineneed07e02009-08-07 13:43:20 +0300831
Nishant Kamat4df9d102011-06-20 20:39:28 +0530832 x = oi->pos_x;
833 y = oi->pos_y;
834 w = oi->width;
835 h = oi->height;
836 outw = oi->out_width == 0 ? oi->width : oi->out_width;
837 outh = oi->out_height == 0 ? oi->height : oi->out_height;
838 paddr = oi->paddr;
Tomi Valkeineneed07e02009-08-07 13:43:20 +0300839
Tomi Valkeinen26a8c252010-06-09 15:31:34 +0300840 orig_w = w;
841 orig_h = h;
842 orig_outw = outw;
843 orig_outh = outh;
844
Tomi Valkeinen8cff88c2011-04-30 14:09:53 +0300845 if (mc->manual_update && mc->do_manual_update) {
Tomi Valkeineneed07e02009-08-07 13:43:20 +0300846 unsigned bpp;
Tomi Valkeinen26a8c252010-06-09 15:31:34 +0300847 unsigned scale_x_m = w, scale_x_d = outw;
848 unsigned scale_y_m = h, scale_y_d = outh;
849
Tomi Valkeineneed07e02009-08-07 13:43:20 +0300850 /* If the overlay is outside the update region, disable it */
851 if (!rectangle_intersects(mc->x, mc->y, mc->w, mc->h,
852 x, y, outw, outh)) {
853 dispc_enable_plane(plane, 0);
854 return 0;
855 }
856
Nishant Kamat4df9d102011-06-20 20:39:28 +0530857 switch (oi->color_mode) {
Amber Jainf20e4222011-05-19 19:47:50 +0530858 case OMAP_DSS_COLOR_NV12:
859 bpp = 8;
860 break;
Tomi Valkeineneed07e02009-08-07 13:43:20 +0300861 case OMAP_DSS_COLOR_RGB16:
862 case OMAP_DSS_COLOR_ARGB16:
863 case OMAP_DSS_COLOR_YUV2:
864 case OMAP_DSS_COLOR_UYVY:
Amber Jainf20e4222011-05-19 19:47:50 +0530865 case OMAP_DSS_COLOR_RGBA16:
866 case OMAP_DSS_COLOR_RGBX16:
867 case OMAP_DSS_COLOR_ARGB16_1555:
868 case OMAP_DSS_COLOR_XRGB16_1555:
Tomi Valkeineneed07e02009-08-07 13:43:20 +0300869 bpp = 16;
870 break;
871
872 case OMAP_DSS_COLOR_RGB24P:
873 bpp = 24;
874 break;
875
876 case OMAP_DSS_COLOR_RGB24U:
877 case OMAP_DSS_COLOR_ARGB32:
878 case OMAP_DSS_COLOR_RGBA32:
879 case OMAP_DSS_COLOR_RGBX32:
880 bpp = 32;
881 break;
882
883 default:
884 BUG();
885 }
886
Nishant Kamat4df9d102011-06-20 20:39:28 +0530887 if (mc->x > oi->pos_x) {
Tomi Valkeinen26a8c252010-06-09 15:31:34 +0300888 x = 0;
Nishant Kamat4df9d102011-06-20 20:39:28 +0530889 outw -= (mc->x - oi->pos_x);
890 paddr += (mc->x - oi->pos_x) *
Tomi Valkeinen26a8c252010-06-09 15:31:34 +0300891 scale_x_m / scale_x_d * bpp / 8;
Tomi Valkeineneed07e02009-08-07 13:43:20 +0300892 } else {
Nishant Kamat4df9d102011-06-20 20:39:28 +0530893 x = oi->pos_x - mc->x;
Tomi Valkeineneed07e02009-08-07 13:43:20 +0300894 }
Tomi Valkeinen26a8c252010-06-09 15:31:34 +0300895
Nishant Kamat4df9d102011-06-20 20:39:28 +0530896 if (mc->y > oi->pos_y) {
Tomi Valkeinen26a8c252010-06-09 15:31:34 +0300897 y = 0;
Nishant Kamat4df9d102011-06-20 20:39:28 +0530898 outh -= (mc->y - oi->pos_y);
899 paddr += (mc->y - oi->pos_y) *
Tomi Valkeinen26a8c252010-06-09 15:31:34 +0300900 scale_y_m / scale_y_d *
Nishant Kamat4df9d102011-06-20 20:39:28 +0530901 oi->screen_width * bpp / 8;
Tomi Valkeinen26a8c252010-06-09 15:31:34 +0300902 } else {
Nishant Kamat4df9d102011-06-20 20:39:28 +0530903 y = oi->pos_y - mc->y;
Tomi Valkeinen26a8c252010-06-09 15:31:34 +0300904 }
905
906 if (mc->w < (x + outw))
907 outw -= (x + outw) - (mc->w);
908
909 if (mc->h < (y + outh))
910 outh -= (y + outh) - (mc->h);
911
912 w = w * outw / orig_outw;
913 h = h * outh / orig_outh;
Tomi Valkeinenf55fdcf2010-06-03 16:27:46 +0300914
915 /* YUV mode overlay's input width has to be even and the
916 * algorithm above may adjust the width to be odd.
917 *
918 * Here we adjust the width if needed, preferring to increase
919 * the width if the original width was bigger.
920 */
921 if ((w & 1) &&
Nishant Kamat4df9d102011-06-20 20:39:28 +0530922 (oi->color_mode == OMAP_DSS_COLOR_YUV2 ||
923 oi->color_mode == OMAP_DSS_COLOR_UYVY)) {
Tomi Valkeinenf55fdcf2010-06-03 16:27:46 +0300924 if (orig_w > w)
925 w += 1;
926 else
927 w -= 1;
928 }
Tomi Valkeineneed07e02009-08-07 13:43:20 +0300929 }
930
931 r = dispc_setup_plane(plane,
932 paddr,
Nishant Kamat4df9d102011-06-20 20:39:28 +0530933 oi->screen_width,
Tomi Valkeineneed07e02009-08-07 13:43:20 +0300934 x, y,
935 w, h,
936 outw, outh,
Nishant Kamat4df9d102011-06-20 20:39:28 +0530937 oi->color_mode,
Tomi Valkeineneed07e02009-08-07 13:43:20 +0300938 c->ilace,
Nishant Kamat4df9d102011-06-20 20:39:28 +0530939 oi->rotation_type,
940 oi->rotation,
941 oi->mirror,
942 oi->global_alpha,
943 oi->pre_mult_alpha,
Amber Jain0d66cbb2011-05-19 19:47:54 +0530944 c->channel,
Nishant Kamat4df9d102011-06-20 20:39:28 +0530945 oi->p_uv_addr);
Tomi Valkeineneed07e02009-08-07 13:43:20 +0300946
947 if (r) {
948 /* this shouldn't happen */
949 DSSERR("dispc_setup_plane failed for ovl %d\n", plane);
950 dispc_enable_plane(plane, 0);
951 return r;
952 }
953
954 dispc_enable_replication(plane, c->replication);
955
Tomi Valkeinen5ed8cf52011-06-21 09:35:36 +0300956 dispc_set_fifo_threshold(plane, c->fifo_low, c->fifo_high);
Tomi Valkeineneed07e02009-08-07 13:43:20 +0300957
958 dispc_enable_plane(plane, 1);
959
960 return 0;
961}
962
963static void configure_manager(enum omap_channel channel)
964{
Nishant Kamat4df9d102011-06-20 20:39:28 +0530965 struct omap_overlay_manager_info *mi;
Tomi Valkeineneed07e02009-08-07 13:43:20 +0300966
967 DSSDBGF("%d", channel);
968
Nishant Kamat4df9d102011-06-20 20:39:28 +0530969 /* picking info from the cache */
970 mi = &dss_cache.manager_cache[channel].info;
Tomi Valkeineneed07e02009-08-07 13:43:20 +0300971
Nishant Kamat4df9d102011-06-20 20:39:28 +0530972 dispc_set_default_color(channel, mi->default_color);
973 dispc_set_trans_key(channel, mi->trans_key_type, mi->trans_key);
974 dispc_enable_trans_key(channel, mi->trans_enabled);
975 dispc_enable_alpha_blending(channel, mi->alpha_enabled);
Tomi Valkeinen3c07cae2011-06-21 09:34:30 +0300976 if (dss_has_feature(FEAT_CPR)) {
977 dispc_enable_cpr(channel, mi->cpr_enable);
978 dispc_set_cpr_coef(channel, &mi->cpr_coefs);
979 }
Tomi Valkeineneed07e02009-08-07 13:43:20 +0300980}
981
982/* configure_dispc() tries to write values from cache to shadow registers.
983 * It writes only to those managers/overlays that are not busy.
984 * returns 0 if everything could be written to shadow registers.
985 * returns 1 if not everything could be written to shadow registers. */
986static int configure_dispc(void)
987{
988 struct overlay_cache_data *oc;
989 struct manager_cache_data *mc;
Archit Tanejaa0acb552010-09-15 19:20:00 +0530990 const int num_ovls = dss_feat_get_num_ovls();
991 const int num_mgrs = dss_feat_get_num_mgrs();
Tomi Valkeineneed07e02009-08-07 13:43:20 +0300992 int i;
993 int r;
Archit Tanejaa0acb552010-09-15 19:20:00 +0530994 bool mgr_busy[MAX_DSS_MANAGERS];
995 bool mgr_go[MAX_DSS_MANAGERS];
Tomi Valkeineneed07e02009-08-07 13:43:20 +0300996 bool busy;
997
998 r = 0;
999 busy = false;
1000
Sumit Semwal18faa1b2010-12-02 11:27:14 +00001001 for (i = 0; i < num_mgrs; i++) {
1002 mgr_busy[i] = dispc_go_busy(i);
1003 mgr_go[i] = false;
1004 }
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001005
1006 /* Commit overlay settings */
1007 for (i = 0; i < num_ovls; ++i) {
1008 oc = &dss_cache.overlay_cache[i];
1009 mc = &dss_cache.manager_cache[oc->channel];
1010
1011 if (!oc->dirty)
1012 continue;
1013
Tomi Valkeinen8cff88c2011-04-30 14:09:53 +03001014 if (mc->manual_update && !mc->do_manual_update)
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001015 continue;
1016
1017 if (mgr_busy[oc->channel]) {
1018 busy = true;
1019 continue;
1020 }
1021
1022 r = configure_overlay(i);
1023 if (r)
1024 DSSERR("configure_overlay %d failed\n", i);
1025
1026 oc->dirty = false;
1027 oc->shadow_dirty = true;
1028 mgr_go[oc->channel] = true;
1029 }
1030
1031 /* Commit manager settings */
1032 for (i = 0; i < num_mgrs; ++i) {
1033 mc = &dss_cache.manager_cache[i];
1034
1035 if (!mc->dirty)
1036 continue;
1037
1038 if (mc->manual_update && !mc->do_manual_update)
1039 continue;
1040
1041 if (mgr_busy[i]) {
1042 busy = true;
1043 continue;
1044 }
1045
1046 configure_manager(i);
1047 mc->dirty = false;
1048 mc->shadow_dirty = true;
1049 mgr_go[i] = true;
1050 }
1051
1052 /* set GO */
1053 for (i = 0; i < num_mgrs; ++i) {
1054 mc = &dss_cache.manager_cache[i];
1055
1056 if (!mgr_go[i])
1057 continue;
1058
1059 /* We don't need GO with manual update display. LCD iface will
1060 * always be turned off after frame, and new settings will be
1061 * taken in to use at next update */
Tomi Valkeinen8cff88c2011-04-30 14:09:53 +03001062 if (!mc->manual_update)
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001063 dispc_go(i);
1064 }
1065
1066 if (busy)
1067 r = 1;
1068 else
1069 r = 0;
1070
1071 return r;
1072}
1073
Tomi Valkeinenf49a9512010-03-26 16:26:37 +02001074/* Make the coordinates even. There are some strange problems with OMAP and
1075 * partial DSI update when the update widths are odd. */
1076static void make_even(u16 *x, u16 *w)
1077{
1078 u16 x1, x2;
1079
1080 x1 = *x;
1081 x2 = *x + *w;
1082
1083 x1 &= ~1;
1084 x2 = ALIGN(x2, 2);
1085
1086 *x = x1;
1087 *w = x2 - x1;
1088}
1089
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001090/* Configure dispc for partial update. Return possibly modified update
1091 * area */
1092void dss_setup_partial_planes(struct omap_dss_device *dssdev,
Tomi Valkeinen26a8c252010-06-09 15:31:34 +03001093 u16 *xi, u16 *yi, u16 *wi, u16 *hi, bool enlarge_update_area)
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001094{
1095 struct overlay_cache_data *oc;
1096 struct manager_cache_data *mc;
Nishant Kamat4df9d102011-06-20 20:39:28 +05301097 struct omap_overlay_info *oi;
Archit Tanejaa0acb552010-09-15 19:20:00 +05301098 const int num_ovls = dss_feat_get_num_ovls();
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001099 struct omap_overlay_manager *mgr;
1100 int i;
1101 u16 x, y, w, h;
1102 unsigned long flags;
Tomi Valkeinen8cab90f2010-06-09 15:09:30 +03001103 bool area_changed;
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001104
1105 x = *xi;
1106 y = *yi;
1107 w = *wi;
1108 h = *hi;
1109
1110 DSSDBG("dispc_setup_partial_planes %d,%d %dx%d\n",
1111 *xi, *yi, *wi, *hi);
1112
1113 mgr = dssdev->manager;
1114
1115 if (!mgr) {
1116 DSSDBG("no manager\n");
1117 return;
1118 }
1119
Tomi Valkeinenf49a9512010-03-26 16:26:37 +02001120 make_even(&x, &w);
1121
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001122 spin_lock_irqsave(&dss_cache.lock, flags);
1123
Tomi Valkeinen8cab90f2010-06-09 15:09:30 +03001124 /*
1125 * Execute the outer loop until the inner loop has completed
1126 * once without increasing the update area. This will ensure that
1127 * all scaled overlays end up completely within the update area.
1128 */
1129 do {
1130 area_changed = false;
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001131
Tomi Valkeinen8cab90f2010-06-09 15:09:30 +03001132 /* We need to show the whole overlay if it is scaled. So look
1133 * for those, and make the update area larger if found.
1134 * Also mark the overlay cache dirty */
1135 for (i = 0; i < num_ovls; ++i) {
1136 unsigned x1, y1, x2, y2;
1137 unsigned outw, outh;
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001138
Tomi Valkeinen8cab90f2010-06-09 15:09:30 +03001139 oc = &dss_cache.overlay_cache[i];
Nishant Kamat4df9d102011-06-20 20:39:28 +05301140 oi = &oc->info;
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001141
Tomi Valkeinen8cab90f2010-06-09 15:09:30 +03001142 if (oc->channel != mgr->id)
1143 continue;
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001144
Tomi Valkeinen8cab90f2010-06-09 15:09:30 +03001145 oc->dirty = true;
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001146
Tomi Valkeinen26a8c252010-06-09 15:31:34 +03001147 if (!enlarge_update_area)
1148 continue;
1149
Tomi Valkeinen8cab90f2010-06-09 15:09:30 +03001150 if (!oc->enabled)
1151 continue;
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001152
Tomi Valkeinen8cab90f2010-06-09 15:09:30 +03001153 if (!dispc_is_overlay_scaled(oc))
1154 continue;
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001155
Nishant Kamat4df9d102011-06-20 20:39:28 +05301156 outw = oi->out_width == 0 ?
1157 oi->width : oi->out_width;
1158 outh = oi->out_height == 0 ?
1159 oi->height : oi->out_height;
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001160
Tomi Valkeinen8cab90f2010-06-09 15:09:30 +03001161 /* is the overlay outside the update region? */
1162 if (!rectangle_intersects(x, y, w, h,
Nishant Kamat4df9d102011-06-20 20:39:28 +05301163 oi->pos_x, oi->pos_y,
Tomi Valkeinen8cab90f2010-06-09 15:09:30 +03001164 outw, outh))
1165 continue;
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001166
Tomi Valkeinen8cab90f2010-06-09 15:09:30 +03001167 /* if the overlay totally inside the update region? */
Nishant Kamat4df9d102011-06-20 20:39:28 +05301168 if (rectangle_subset(oi->pos_x, oi->pos_y, outw, outh,
Tomi Valkeinen8cab90f2010-06-09 15:09:30 +03001169 x, y, w, h))
1170 continue;
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001171
Nishant Kamat4df9d102011-06-20 20:39:28 +05301172 if (x > oi->pos_x)
1173 x1 = oi->pos_x;
Tomi Valkeinen8cab90f2010-06-09 15:09:30 +03001174 else
1175 x1 = x;
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001176
Nishant Kamat4df9d102011-06-20 20:39:28 +05301177 if (y > oi->pos_y)
1178 y1 = oi->pos_y;
Tomi Valkeinen8cab90f2010-06-09 15:09:30 +03001179 else
1180 y1 = y;
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001181
Nishant Kamat4df9d102011-06-20 20:39:28 +05301182 if ((x + w) < (oi->pos_x + outw))
1183 x2 = oi->pos_x + outw;
Tomi Valkeinen8cab90f2010-06-09 15:09:30 +03001184 else
1185 x2 = x + w;
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001186
Nishant Kamat4df9d102011-06-20 20:39:28 +05301187 if ((y + h) < (oi->pos_y + outh))
1188 y2 = oi->pos_y + outh;
Tomi Valkeinen8cab90f2010-06-09 15:09:30 +03001189 else
1190 y2 = y + h;
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001191
Tomi Valkeinen8cab90f2010-06-09 15:09:30 +03001192 x = x1;
1193 y = y1;
1194 w = x2 - x1;
1195 h = y2 - y1;
Tomi Valkeinenf49a9512010-03-26 16:26:37 +02001196
Tomi Valkeinen8cab90f2010-06-09 15:09:30 +03001197 make_even(&x, &w);
1198
1199 DSSDBG("changing upd area due to ovl(%d) "
1200 "scaling %d,%d %dx%d\n",
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001201 i, x, y, w, h);
Tomi Valkeinen8cab90f2010-06-09 15:09:30 +03001202
1203 area_changed = true;
1204 }
1205 } while (area_changed);
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001206
1207 mc = &dss_cache.manager_cache[mgr->id];
1208 mc->do_manual_update = true;
Tomi Valkeinen26a8c252010-06-09 15:31:34 +03001209 mc->enlarge_update_area = enlarge_update_area;
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001210 mc->x = x;
1211 mc->y = y;
1212 mc->w = w;
1213 mc->h = h;
1214
1215 configure_dispc();
1216
1217 mc->do_manual_update = false;
1218
1219 spin_unlock_irqrestore(&dss_cache.lock, flags);
1220
1221 *xi = x;
1222 *yi = y;
1223 *wi = w;
1224 *hi = h;
1225}
1226
1227void dss_start_update(struct omap_dss_device *dssdev)
1228{
1229 struct manager_cache_data *mc;
1230 struct overlay_cache_data *oc;
Archit Tanejaa0acb552010-09-15 19:20:00 +05301231 const int num_ovls = dss_feat_get_num_ovls();
1232 const int num_mgrs = dss_feat_get_num_mgrs();
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001233 struct omap_overlay_manager *mgr;
1234 int i;
1235
1236 mgr = dssdev->manager;
1237
1238 for (i = 0; i < num_ovls; ++i) {
1239 oc = &dss_cache.overlay_cache[i];
1240 if (oc->channel != mgr->id)
1241 continue;
1242
1243 oc->shadow_dirty = false;
1244 }
1245
1246 for (i = 0; i < num_mgrs; ++i) {
1247 mc = &dss_cache.manager_cache[i];
1248 if (mgr->id != i)
1249 continue;
1250
1251 mc->shadow_dirty = false;
1252 }
1253
Tomi Valkeinena2faee82010-01-08 17:14:53 +02001254 dssdev->manager->enable(dssdev->manager);
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001255}
1256
1257static void dss_apply_irq_handler(void *data, u32 mask)
1258{
1259 struct manager_cache_data *mc;
1260 struct overlay_cache_data *oc;
Archit Tanejaa0acb552010-09-15 19:20:00 +05301261 const int num_ovls = dss_feat_get_num_ovls();
1262 const int num_mgrs = dss_feat_get_num_mgrs();
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001263 int i, r;
Archit Tanejaa0acb552010-09-15 19:20:00 +05301264 bool mgr_busy[MAX_DSS_MANAGERS];
Sumit Semwal18faa1b2010-12-02 11:27:14 +00001265 u32 irq_mask;
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001266
Sumit Semwal18faa1b2010-12-02 11:27:14 +00001267 for (i = 0; i < num_mgrs; i++)
1268 mgr_busy[i] = dispc_go_busy(i);
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001269
1270 spin_lock(&dss_cache.lock);
1271
1272 for (i = 0; i < num_ovls; ++i) {
1273 oc = &dss_cache.overlay_cache[i];
1274 if (!mgr_busy[oc->channel])
1275 oc->shadow_dirty = false;
1276 }
1277
1278 for (i = 0; i < num_mgrs; ++i) {
1279 mc = &dss_cache.manager_cache[i];
1280 if (!mgr_busy[i])
1281 mc->shadow_dirty = false;
1282 }
1283
1284 r = configure_dispc();
1285 if (r == 1)
1286 goto end;
1287
1288 /* re-read busy flags */
Sumit Semwal18faa1b2010-12-02 11:27:14 +00001289 for (i = 0; i < num_mgrs; i++)
1290 mgr_busy[i] = dispc_go_busy(i);
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001291
1292 /* keep running as long as there are busy managers, so that
1293 * we can collect overlay-applied information */
1294 for (i = 0; i < num_mgrs; ++i) {
1295 if (mgr_busy[i])
1296 goto end;
1297 }
1298
Sumit Semwal18faa1b2010-12-02 11:27:14 +00001299 irq_mask = DISPC_IRQ_VSYNC | DISPC_IRQ_EVSYNC_ODD |
1300 DISPC_IRQ_EVSYNC_EVEN;
1301 if (dss_has_feature(FEAT_MGR_LCD2))
1302 irq_mask |= DISPC_IRQ_VSYNC2;
1303
1304 omap_dispc_unregister_isr(dss_apply_irq_handler, NULL, irq_mask);
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001305 dss_cache.irq_enabled = false;
1306
1307end:
1308 spin_unlock(&dss_cache.lock);
1309}
1310
1311static int omap_dss_mgr_apply(struct omap_overlay_manager *mgr)
1312{
1313 struct overlay_cache_data *oc;
1314 struct manager_cache_data *mc;
1315 int i;
1316 struct omap_overlay *ovl;
1317 int num_planes_enabled = 0;
1318 bool use_fifomerge;
1319 unsigned long flags;
1320 int r;
1321
1322 DSSDBG("omap_dss_mgr_apply(%s)\n", mgr->name);
1323
Tomi Valkeinen4fbafaf2011-05-27 10:52:19 +03001324 r = dispc_runtime_get();
1325 if (r)
1326 return r;
1327
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001328 spin_lock_irqsave(&dss_cache.lock, flags);
1329
1330 /* Configure overlays */
1331 for (i = 0; i < omap_dss_get_num_overlays(); ++i) {
1332 struct omap_dss_device *dssdev;
1333
1334 ovl = omap_dss_get_overlay(i);
1335
1336 if (!(ovl->caps & OMAP_DSS_OVL_CAP_DISPC))
1337 continue;
1338
1339 oc = &dss_cache.overlay_cache[ovl->id];
1340
1341 if (!overlay_enabled(ovl)) {
1342 if (oc->enabled) {
1343 oc->enabled = false;
1344 oc->dirty = true;
1345 }
1346 continue;
1347 }
1348
1349 if (!ovl->info_dirty) {
1350 if (oc->enabled)
1351 ++num_planes_enabled;
1352 continue;
1353 }
1354
1355 dssdev = ovl->manager->device;
1356
1357 if (dss_check_overlay(ovl, dssdev)) {
1358 if (oc->enabled) {
1359 oc->enabled = false;
1360 oc->dirty = true;
1361 }
1362 continue;
1363 }
1364
1365 ovl->info_dirty = false;
1366 oc->dirty = true;
Nishant Kamat4df9d102011-06-20 20:39:28 +05301367 oc->info = ovl->info;
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001368
1369 oc->replication =
1370 dss_use_replication(dssdev, ovl->info.color_mode);
1371
1372 oc->ilace = dssdev->type == OMAP_DISPLAY_TYPE_VENC;
1373
1374 oc->channel = ovl->manager->id;
1375
1376 oc->enabled = true;
1377
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001378 ++num_planes_enabled;
1379 }
1380
1381 /* Configure managers */
1382 list_for_each_entry(mgr, &manager_list, list) {
1383 struct omap_dss_device *dssdev;
1384
1385 if (!(mgr->caps & OMAP_DSS_OVL_MGR_CAP_DISPC))
1386 continue;
1387
1388 mc = &dss_cache.manager_cache[mgr->id];
1389
1390 if (mgr->device_changed) {
1391 mgr->device_changed = false;
1392 mgr->info_dirty = true;
1393 }
1394
1395 if (!mgr->info_dirty)
1396 continue;
1397
1398 if (!mgr->device)
1399 continue;
1400
1401 dssdev = mgr->device;
1402
1403 mgr->info_dirty = false;
1404 mc->dirty = true;
Nishant Kamat4df9d102011-06-20 20:39:28 +05301405 mc->info = mgr->info;
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001406
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001407 mc->manual_update =
Tomi Valkeinen8cff88c2011-04-30 14:09:53 +03001408 dssdev->caps & OMAP_DSS_DISPLAY_CAP_MANUAL_UPDATE;
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001409 }
1410
1411 /* XXX TODO: Try to get fifomerge working. The problem is that it
1412 * affects both managers, not individually but at the same time. This
1413 * means the change has to be well synchronized. I guess the proper way
1414 * is to have a two step process for fifo merge:
1415 * fifomerge enable:
1416 * 1. disable other planes, leaving one plane enabled
1417 * 2. wait until the planes are disabled on HW
1418 * 3. config merged fifo thresholds, enable fifomerge
1419 * fifomerge disable:
1420 * 1. config unmerged fifo thresholds, disable fifomerge
1421 * 2. wait until fifo changes are in HW
1422 * 3. enable planes
1423 */
1424 use_fifomerge = false;
1425
1426 /* Configure overlay fifos */
1427 for (i = 0; i < omap_dss_get_num_overlays(); ++i) {
1428 struct omap_dss_device *dssdev;
Tomi Valkeinen5ed8cf52011-06-21 09:35:36 +03001429 u32 size, burst_size;
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001430
1431 ovl = omap_dss_get_overlay(i);
1432
1433 if (!(ovl->caps & OMAP_DSS_OVL_CAP_DISPC))
1434 continue;
1435
1436 oc = &dss_cache.overlay_cache[ovl->id];
1437
1438 if (!oc->enabled)
1439 continue;
1440
1441 dssdev = ovl->manager->device;
1442
1443 size = dispc_get_plane_fifo_size(ovl->id);
1444 if (use_fifomerge)
1445 size *= 3;
1446
Tomi Valkeinen5ed8cf52011-06-21 09:35:36 +03001447 burst_size = dispc_get_burst_size(ovl->id);
1448
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001449 switch (dssdev->type) {
1450 case OMAP_DISPLAY_TYPE_DPI:
1451 case OMAP_DISPLAY_TYPE_DBI:
1452 case OMAP_DISPLAY_TYPE_SDI:
1453 case OMAP_DISPLAY_TYPE_VENC:
Mythri P Kb1196012011-03-08 17:15:54 +05301454 case OMAP_DISPLAY_TYPE_HDMI:
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001455 default_get_overlay_fifo_thresholds(ovl->id, size,
Tomi Valkeinen5ed8cf52011-06-21 09:35:36 +03001456 burst_size, &oc->fifo_low,
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001457 &oc->fifo_high);
1458 break;
1459#ifdef CONFIG_OMAP2_DSS_DSI
1460 case OMAP_DISPLAY_TYPE_DSI:
1461 dsi_get_overlay_fifo_thresholds(ovl->id, size,
Tomi Valkeinen5ed8cf52011-06-21 09:35:36 +03001462 burst_size, &oc->fifo_low,
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001463 &oc->fifo_high);
1464 break;
1465#endif
1466 default:
1467 BUG();
1468 }
1469 }
1470
1471 r = 0;
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001472 if (!dss_cache.irq_enabled) {
Sumit Semwal18faa1b2010-12-02 11:27:14 +00001473 u32 mask;
1474
1475 mask = DISPC_IRQ_VSYNC | DISPC_IRQ_EVSYNC_ODD |
1476 DISPC_IRQ_EVSYNC_EVEN;
1477 if (dss_has_feature(FEAT_MGR_LCD2))
1478 mask |= DISPC_IRQ_VSYNC2;
1479
1480 r = omap_dispc_register_isr(dss_apply_irq_handler, NULL, mask);
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001481 dss_cache.irq_enabled = true;
1482 }
1483 configure_dispc();
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001484
1485 spin_unlock_irqrestore(&dss_cache.lock, flags);
1486
Tomi Valkeinen4fbafaf2011-05-27 10:52:19 +03001487 dispc_runtime_put();
1488
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001489 return r;
1490}
1491
1492static int dss_check_manager(struct omap_overlay_manager *mgr)
1493{
1494 /* OMAP supports only graphics source transparency color key and alpha
1495 * blending simultaneously. See TRM 15.4.2.4.2.2 Alpha Mode */
1496
1497 if (mgr->info.alpha_enabled && mgr->info.trans_enabled &&
1498 mgr->info.trans_key_type != OMAP_DSS_COLOR_KEY_GFX_DST)
1499 return -EINVAL;
1500
1501 return 0;
1502}
1503
1504static int omap_dss_mgr_set_info(struct omap_overlay_manager *mgr,
1505 struct omap_overlay_manager_info *info)
1506{
1507 int r;
1508 struct omap_overlay_manager_info old_info;
1509
1510 old_info = mgr->info;
1511 mgr->info = *info;
1512
1513 r = dss_check_manager(mgr);
1514 if (r) {
1515 mgr->info = old_info;
1516 return r;
1517 }
1518
1519 mgr->info_dirty = true;
1520
1521 return 0;
1522}
1523
1524static void omap_dss_mgr_get_info(struct omap_overlay_manager *mgr,
1525 struct omap_overlay_manager_info *info)
1526{
1527 *info = mgr->info;
1528}
1529
Tomi Valkeinena2faee82010-01-08 17:14:53 +02001530static int dss_mgr_enable(struct omap_overlay_manager *mgr)
1531{
1532 dispc_enable_channel(mgr->id, 1);
1533 return 0;
1534}
1535
1536static int dss_mgr_disable(struct omap_overlay_manager *mgr)
1537{
1538 dispc_enable_channel(mgr->id, 0);
1539 return 0;
1540}
1541
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001542static void omap_dss_add_overlay_manager(struct omap_overlay_manager *manager)
1543{
1544 ++num_managers;
1545 list_add_tail(&manager->list, &manager_list);
1546}
1547
1548int dss_init_overlay_managers(struct platform_device *pdev)
1549{
1550 int i, r;
1551
1552 spin_lock_init(&dss_cache.lock);
1553
1554 INIT_LIST_HEAD(&manager_list);
1555
1556 num_managers = 0;
1557
Archit Tanejaa0acb552010-09-15 19:20:00 +05301558 for (i = 0; i < dss_feat_get_num_mgrs(); ++i) {
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001559 struct omap_overlay_manager *mgr;
1560 mgr = kzalloc(sizeof(*mgr), GFP_KERNEL);
1561
1562 BUG_ON(mgr == NULL);
1563
1564 switch (i) {
1565 case 0:
1566 mgr->name = "lcd";
1567 mgr->id = OMAP_DSS_CHANNEL_LCD;
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001568 break;
1569 case 1:
1570 mgr->name = "tv";
1571 mgr->id = OMAP_DSS_CHANNEL_DIGIT;
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001572 break;
Sumit Semwal18faa1b2010-12-02 11:27:14 +00001573 case 2:
1574 mgr->name = "lcd2";
1575 mgr->id = OMAP_DSS_CHANNEL_LCD2;
1576 break;
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001577 }
1578
1579 mgr->set_device = &omap_dss_set_device;
1580 mgr->unset_device = &omap_dss_unset_device;
1581 mgr->apply = &omap_dss_mgr_apply;
1582 mgr->set_manager_info = &omap_dss_mgr_set_info;
1583 mgr->get_manager_info = &omap_dss_mgr_get_info;
1584 mgr->wait_for_go = &dss_mgr_wait_for_go;
Tomi Valkeinen3f71cbe2010-01-08 17:06:04 +02001585 mgr->wait_for_vsync = &dss_mgr_wait_for_vsync;
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001586
Tomi Valkeinena2faee82010-01-08 17:14:53 +02001587 mgr->enable = &dss_mgr_enable;
1588 mgr->disable = &dss_mgr_disable;
1589
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001590 mgr->caps = OMAP_DSS_OVL_MGR_CAP_DISPC;
Archit Tanejaa0acb552010-09-15 19:20:00 +05301591 mgr->supported_displays =
1592 dss_feat_get_supported_displays(mgr->id);
Tomi Valkeineneed07e02009-08-07 13:43:20 +03001593
1594 dss_overlay_setup_dispc_manager(mgr);
1595
1596 omap_dss_add_overlay_manager(mgr);
1597
1598 r = kobject_init_and_add(&mgr->kobj, &manager_ktype,
1599 &pdev->dev.kobj, "manager%d", i);
1600
1601 if (r) {
1602 DSSERR("failed to create sysfs file\n");
1603 continue;
1604 }
1605 }
1606
1607#ifdef L4_EXAMPLE
1608 {
1609 int omap_dss_mgr_apply_l4(struct omap_overlay_manager *mgr)
1610 {
1611 DSSDBG("omap_dss_mgr_apply_l4(%s)\n", mgr->name);
1612
1613 return 0;
1614 }
1615
1616 struct omap_overlay_manager *mgr;
1617 mgr = kzalloc(sizeof(*mgr), GFP_KERNEL);
1618
1619 BUG_ON(mgr == NULL);
1620
1621 mgr->name = "l4";
1622 mgr->supported_displays =
1623 OMAP_DISPLAY_TYPE_DBI | OMAP_DISPLAY_TYPE_DSI;
1624
1625 mgr->set_device = &omap_dss_set_device;
1626 mgr->unset_device = &omap_dss_unset_device;
1627 mgr->apply = &omap_dss_mgr_apply_l4;
1628 mgr->set_manager_info = &omap_dss_mgr_set_info;
1629 mgr->get_manager_info = &omap_dss_mgr_get_info;
1630
1631 dss_overlay_setup_l4_manager(mgr);
1632
1633 omap_dss_add_overlay_manager(mgr);
1634
1635 r = kobject_init_and_add(&mgr->kobj, &manager_ktype,
1636 &pdev->dev.kobj, "managerl4");
1637
1638 if (r)
1639 DSSERR("failed to create sysfs file\n");
1640 }
1641#endif
1642
1643 return 0;
1644}
1645
1646void dss_uninit_overlay_managers(struct platform_device *pdev)
1647{
1648 struct omap_overlay_manager *mgr;
1649
1650 while (!list_empty(&manager_list)) {
1651 mgr = list_first_entry(&manager_list,
1652 struct omap_overlay_manager, list);
1653 list_del(&mgr->list);
1654 kobject_del(&mgr->kobj);
1655 kobject_put(&mgr->kobj);
1656 kfree(mgr);
1657 }
1658
1659 num_managers = 0;
1660}
1661
1662int omap_dss_get_num_overlay_managers(void)
1663{
1664 return num_managers;
1665}
1666EXPORT_SYMBOL(omap_dss_get_num_overlay_managers);
1667
1668struct omap_overlay_manager *omap_dss_get_overlay_manager(int num)
1669{
1670 int i = 0;
1671 struct omap_overlay_manager *mgr;
1672
1673 list_for_each_entry(mgr, &manager_list, list) {
1674 if (i++ == num)
1675 return mgr;
1676 }
1677
1678 return NULL;
1679}
1680EXPORT_SYMBOL(omap_dss_get_overlay_manager);
1681