Adding support for MOXA ART SoC. Testing port of linux-2.6.32.60-moxart.
[linux-3.6.7-moxart.git] / drivers / media / video / davinci / vpbe_osd.c
blobbba299dbf3967c6e7e708f9631eb9853912d1833
1 /*
2 * Copyright (C) 2007-2010 Texas Instruments Inc
3 * Copyright (C) 2007 MontaVista Software, Inc.
5 * Andy Lowe (alowe@mvista.com), MontaVista Software
6 * - Initial version
7 * Murali Karicheri (mkaricheri@gmail.com), Texas Instruments Ltd.
8 * - ported to sub device interface
10 * This program is free software; you can redistribute it and/or modify
11 * it under the terms of the GNU General Public License as published by
12 * the Free Software Foundation version 2.
14 * This program is distributed in the hope that it will be useful,
15 * but WITHOUT ANY WARRANTY; without even the implied warranty of
16 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
17 * GNU General Public License for more details.
19 * You should have received a copy of the GNU General Public License
20 * along with this program; if not, write to the Free Software
21 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
24 #include <linux/module.h>
25 #include <linux/kernel.h>
26 #include <linux/interrupt.h>
27 #include <linux/platform_device.h>
28 #include <linux/clk.h>
29 #include <linux/slab.h>
31 #include <mach/cputype.h>
32 #include <mach/hardware.h>
34 #include <media/davinci/vpss.h>
35 #include <media/v4l2-device.h>
36 #include <media/davinci/vpbe_types.h>
37 #include <media/davinci/vpbe_osd.h>
39 #include <linux/io.h>
40 #include "vpbe_osd_regs.h"
42 #define MODULE_NAME VPBE_OSD_SUBDEV_NAME
44 /* register access routines */
45 static inline u32 osd_read(struct osd_state *sd, u32 offset)
47 struct osd_state *osd = sd;
49 return readl(osd->osd_base + offset);
52 static inline u32 osd_write(struct osd_state *sd, u32 val, u32 offset)
54 struct osd_state *osd = sd;
56 writel(val, osd->osd_base + offset);
58 return val;
61 static inline u32 osd_set(struct osd_state *sd, u32 mask, u32 offset)
63 struct osd_state *osd = sd;
65 u32 addr = osd->osd_base + offset;
66 u32 val = readl(addr) | mask;
68 writel(val, addr);
70 return val;
73 static inline u32 osd_clear(struct osd_state *sd, u32 mask, u32 offset)
75 struct osd_state *osd = sd;
77 u32 addr = osd->osd_base + offset;
78 u32 val = readl(addr) & ~mask;
80 writel(val, addr);
82 return val;
85 static inline u32 osd_modify(struct osd_state *sd, u32 mask, u32 val,
86 u32 offset)
88 struct osd_state *osd = sd;
90 u32 addr = osd->osd_base + offset;
91 u32 new_val = (readl(addr) & ~mask) | (val & mask);
93 writel(new_val, addr);
95 return new_val;
98 /* define some macros for layer and pixfmt classification */
99 #define is_osd_win(layer) (((layer) == WIN_OSD0) || ((layer) == WIN_OSD1))
100 #define is_vid_win(layer) (((layer) == WIN_VID0) || ((layer) == WIN_VID1))
101 #define is_rgb_pixfmt(pixfmt) \
102 (((pixfmt) == PIXFMT_RGB565) || ((pixfmt) == PIXFMT_RGB888))
103 #define is_yc_pixfmt(pixfmt) \
104 (((pixfmt) == PIXFMT_YCbCrI) || ((pixfmt) == PIXFMT_YCrCbI) || \
105 ((pixfmt) == PIXFMT_NV12))
106 #define MAX_WIN_SIZE OSD_VIDWIN0XP_V0X
107 #define MAX_LINE_LENGTH (OSD_VIDWIN0OFST_V0LO << 5)
110 * _osd_dm6446_vid0_pingpong() - field inversion fix for DM6446
111 * @sd - ptr to struct osd_state
112 * @field_inversion - inversion flag
113 * @fb_base_phys - frame buffer address
114 * @lconfig - ptr to layer config
116 * This routine implements a workaround for the field signal inversion silicon
117 * erratum described in Advisory 1.3.8 for the DM6446. The fb_base_phys and
118 * lconfig parameters apply to the vid0 window. This routine should be called
119 * whenever the vid0 layer configuration or start address is modified, or when
120 * the OSD field inversion setting is modified.
121 * Returns: 1 if the ping-pong buffers need to be toggled in the vsync isr, or
122 * 0 otherwise
124 static int _osd_dm6446_vid0_pingpong(struct osd_state *sd,
125 int field_inversion,
126 unsigned long fb_base_phys,
127 const struct osd_layer_config *lconfig)
129 struct osd_platform_data *pdata;
131 pdata = (struct osd_platform_data *)sd->dev->platform_data;
132 if (pdata->field_inv_wa_enable) {
134 if (!field_inversion || !lconfig->interlaced) {
135 osd_write(sd, fb_base_phys & ~0x1F, OSD_VIDWIN0ADR);
136 osd_write(sd, fb_base_phys & ~0x1F, OSD_PPVWIN0ADR);
137 osd_modify(sd, OSD_MISCCTL_PPSW | OSD_MISCCTL_PPRV, 0,
138 OSD_MISCCTL);
139 return 0;
140 } else {
141 unsigned miscctl = OSD_MISCCTL_PPRV;
143 osd_write(sd,
144 (fb_base_phys & ~0x1F) - lconfig->line_length,
145 OSD_VIDWIN0ADR);
146 osd_write(sd,
147 (fb_base_phys & ~0x1F) + lconfig->line_length,
148 OSD_PPVWIN0ADR);
149 osd_modify(sd,
150 OSD_MISCCTL_PPSW | OSD_MISCCTL_PPRV, miscctl,
151 OSD_MISCCTL);
153 return 1;
157 return 0;
160 static void _osd_set_field_inversion(struct osd_state *sd, int enable)
162 unsigned fsinv = 0;
164 if (enable)
165 fsinv = OSD_MODE_FSINV;
167 osd_modify(sd, OSD_MODE_FSINV, fsinv, OSD_MODE);
170 static void _osd_set_blink_attribute(struct osd_state *sd, int enable,
171 enum osd_blink_interval blink)
173 u32 osdatrmd = 0;
175 if (enable) {
176 osdatrmd |= OSD_OSDATRMD_BLNK;
177 osdatrmd |= blink << OSD_OSDATRMD_BLNKINT_SHIFT;
179 /* caller must ensure that OSD1 is configured in attribute mode */
180 osd_modify(sd, OSD_OSDATRMD_BLNKINT | OSD_OSDATRMD_BLNK, osdatrmd,
181 OSD_OSDATRMD);
184 static void _osd_set_rom_clut(struct osd_state *sd,
185 enum osd_rom_clut rom_clut)
187 if (rom_clut == ROM_CLUT0)
188 osd_clear(sd, OSD_MISCCTL_RSEL, OSD_MISCCTL);
189 else
190 osd_set(sd, OSD_MISCCTL_RSEL, OSD_MISCCTL);
193 static void _osd_set_palette_map(struct osd_state *sd,
194 enum osd_win_layer osdwin,
195 unsigned char pixel_value,
196 unsigned char clut_index,
197 enum osd_pix_format pixfmt)
199 static const int map_2bpp[] = { 0, 5, 10, 15 };
200 static const int map_1bpp[] = { 0, 15 };
201 int bmp_offset;
202 int bmp_shift;
203 int bmp_mask;
204 int bmp_reg;
206 switch (pixfmt) {
207 case PIXFMT_1BPP:
208 bmp_reg = map_1bpp[pixel_value & 0x1];
209 break;
210 case PIXFMT_2BPP:
211 bmp_reg = map_2bpp[pixel_value & 0x3];
212 break;
213 case PIXFMT_4BPP:
214 bmp_reg = pixel_value & 0xf;
215 break;
216 default:
217 return;
220 switch (osdwin) {
221 case OSDWIN_OSD0:
222 bmp_offset = OSD_W0BMP01 + (bmp_reg >> 1) * sizeof(u32);
223 break;
224 case OSDWIN_OSD1:
225 bmp_offset = OSD_W1BMP01 + (bmp_reg >> 1) * sizeof(u32);
226 break;
227 default:
228 return;
231 if (bmp_reg & 1) {
232 bmp_shift = 8;
233 bmp_mask = 0xff << 8;
234 } else {
235 bmp_shift = 0;
236 bmp_mask = 0xff;
239 osd_modify(sd, bmp_mask, clut_index << bmp_shift, bmp_offset);
242 static void _osd_set_rec601_attenuation(struct osd_state *sd,
243 enum osd_win_layer osdwin, int enable)
245 switch (osdwin) {
246 case OSDWIN_OSD0:
247 osd_modify(sd, OSD_OSDWIN0MD_ATN0E,
248 enable ? OSD_OSDWIN0MD_ATN0E : 0,
249 OSD_OSDWIN0MD);
250 if (sd->vpbe_type == VPBE_VERSION_1)
251 osd_modify(sd, OSD_OSDWIN0MD_ATN0E,
252 enable ? OSD_OSDWIN0MD_ATN0E : 0,
253 OSD_OSDWIN0MD);
254 else if ((sd->vpbe_type == VPBE_VERSION_3) ||
255 (sd->vpbe_type == VPBE_VERSION_2))
256 osd_modify(sd, OSD_EXTMODE_ATNOSD0EN,
257 enable ? OSD_EXTMODE_ATNOSD0EN : 0,
258 OSD_EXTMODE);
259 break;
260 case OSDWIN_OSD1:
261 osd_modify(sd, OSD_OSDWIN1MD_ATN1E,
262 enable ? OSD_OSDWIN1MD_ATN1E : 0,
263 OSD_OSDWIN1MD);
264 if (sd->vpbe_type == VPBE_VERSION_1)
265 osd_modify(sd, OSD_OSDWIN1MD_ATN1E,
266 enable ? OSD_OSDWIN1MD_ATN1E : 0,
267 OSD_OSDWIN1MD);
268 else if ((sd->vpbe_type == VPBE_VERSION_3) ||
269 (sd->vpbe_type == VPBE_VERSION_2))
270 osd_modify(sd, OSD_EXTMODE_ATNOSD1EN,
271 enable ? OSD_EXTMODE_ATNOSD1EN : 0,
272 OSD_EXTMODE);
273 break;
277 static void _osd_set_blending_factor(struct osd_state *sd,
278 enum osd_win_layer osdwin,
279 enum osd_blending_factor blend)
281 switch (osdwin) {
282 case OSDWIN_OSD0:
283 osd_modify(sd, OSD_OSDWIN0MD_BLND0,
284 blend << OSD_OSDWIN0MD_BLND0_SHIFT, OSD_OSDWIN0MD);
285 break;
286 case OSDWIN_OSD1:
287 osd_modify(sd, OSD_OSDWIN1MD_BLND1,
288 blend << OSD_OSDWIN1MD_BLND1_SHIFT, OSD_OSDWIN1MD);
289 break;
293 static void _osd_enable_rgb888_pixblend(struct osd_state *sd,
294 enum osd_win_layer osdwin)
297 osd_modify(sd, OSD_MISCCTL_BLDSEL, 0, OSD_MISCCTL);
298 switch (osdwin) {
299 case OSDWIN_OSD0:
300 osd_modify(sd, OSD_EXTMODE_OSD0BLDCHR,
301 OSD_EXTMODE_OSD0BLDCHR, OSD_EXTMODE);
302 break;
303 case OSDWIN_OSD1:
304 osd_modify(sd, OSD_EXTMODE_OSD1BLDCHR,
305 OSD_EXTMODE_OSD1BLDCHR, OSD_EXTMODE);
306 break;
310 static void _osd_enable_color_key(struct osd_state *sd,
311 enum osd_win_layer osdwin,
312 unsigned colorkey,
313 enum osd_pix_format pixfmt)
315 switch (pixfmt) {
316 case PIXFMT_1BPP:
317 case PIXFMT_2BPP:
318 case PIXFMT_4BPP:
319 case PIXFMT_8BPP:
320 if (sd->vpbe_type == VPBE_VERSION_3) {
321 switch (osdwin) {
322 case OSDWIN_OSD0:
323 osd_modify(sd, OSD_TRANSPBMPIDX_BMP0,
324 colorkey <<
325 OSD_TRANSPBMPIDX_BMP0_SHIFT,
326 OSD_TRANSPBMPIDX);
327 break;
328 case OSDWIN_OSD1:
329 osd_modify(sd, OSD_TRANSPBMPIDX_BMP1,
330 colorkey <<
331 OSD_TRANSPBMPIDX_BMP1_SHIFT,
332 OSD_TRANSPBMPIDX);
333 break;
336 break;
337 case PIXFMT_RGB565:
338 if (sd->vpbe_type == VPBE_VERSION_1)
339 osd_write(sd, colorkey & OSD_TRANSPVAL_RGBTRANS,
340 OSD_TRANSPVAL);
341 else if (sd->vpbe_type == VPBE_VERSION_3)
342 osd_write(sd, colorkey & OSD_TRANSPVALL_RGBL,
343 OSD_TRANSPVALL);
344 break;
345 case PIXFMT_YCbCrI:
346 case PIXFMT_YCrCbI:
347 if (sd->vpbe_type == VPBE_VERSION_3)
348 osd_modify(sd, OSD_TRANSPVALU_Y, colorkey,
349 OSD_TRANSPVALU);
350 break;
351 case PIXFMT_RGB888:
352 if (sd->vpbe_type == VPBE_VERSION_3) {
353 osd_write(sd, colorkey & OSD_TRANSPVALL_RGBL,
354 OSD_TRANSPVALL);
355 osd_modify(sd, OSD_TRANSPVALU_RGBU, colorkey >> 16,
356 OSD_TRANSPVALU);
358 break;
359 default:
360 break;
363 switch (osdwin) {
364 case OSDWIN_OSD0:
365 osd_set(sd, OSD_OSDWIN0MD_TE0, OSD_OSDWIN0MD);
366 break;
367 case OSDWIN_OSD1:
368 osd_set(sd, OSD_OSDWIN1MD_TE1, OSD_OSDWIN1MD);
369 break;
373 static void _osd_disable_color_key(struct osd_state *sd,
374 enum osd_win_layer osdwin)
376 switch (osdwin) {
377 case OSDWIN_OSD0:
378 osd_clear(sd, OSD_OSDWIN0MD_TE0, OSD_OSDWIN0MD);
379 break;
380 case OSDWIN_OSD1:
381 osd_clear(sd, OSD_OSDWIN1MD_TE1, OSD_OSDWIN1MD);
382 break;
386 static void _osd_set_osd_clut(struct osd_state *sd,
387 enum osd_win_layer osdwin,
388 enum osd_clut clut)
390 u32 winmd = 0;
392 switch (osdwin) {
393 case OSDWIN_OSD0:
394 if (clut == RAM_CLUT)
395 winmd |= OSD_OSDWIN0MD_CLUTS0;
396 osd_modify(sd, OSD_OSDWIN0MD_CLUTS0, winmd, OSD_OSDWIN0MD);
397 break;
398 case OSDWIN_OSD1:
399 if (clut == RAM_CLUT)
400 winmd |= OSD_OSDWIN1MD_CLUTS1;
401 osd_modify(sd, OSD_OSDWIN1MD_CLUTS1, winmd, OSD_OSDWIN1MD);
402 break;
406 static void _osd_set_zoom(struct osd_state *sd, enum osd_layer layer,
407 enum osd_zoom_factor h_zoom,
408 enum osd_zoom_factor v_zoom)
410 u32 winmd = 0;
412 switch (layer) {
413 case WIN_OSD0:
414 winmd |= (h_zoom << OSD_OSDWIN0MD_OHZ0_SHIFT);
415 winmd |= (v_zoom << OSD_OSDWIN0MD_OVZ0_SHIFT);
416 osd_modify(sd, OSD_OSDWIN0MD_OHZ0 | OSD_OSDWIN0MD_OVZ0, winmd,
417 OSD_OSDWIN0MD);
418 break;
419 case WIN_VID0:
420 winmd |= (h_zoom << OSD_VIDWINMD_VHZ0_SHIFT);
421 winmd |= (v_zoom << OSD_VIDWINMD_VVZ0_SHIFT);
422 osd_modify(sd, OSD_VIDWINMD_VHZ0 | OSD_VIDWINMD_VVZ0, winmd,
423 OSD_VIDWINMD);
424 break;
425 case WIN_OSD1:
426 winmd |= (h_zoom << OSD_OSDWIN1MD_OHZ1_SHIFT);
427 winmd |= (v_zoom << OSD_OSDWIN1MD_OVZ1_SHIFT);
428 osd_modify(sd, OSD_OSDWIN1MD_OHZ1 | OSD_OSDWIN1MD_OVZ1, winmd,
429 OSD_OSDWIN1MD);
430 break;
431 case WIN_VID1:
432 winmd |= (h_zoom << OSD_VIDWINMD_VHZ1_SHIFT);
433 winmd |= (v_zoom << OSD_VIDWINMD_VVZ1_SHIFT);
434 osd_modify(sd, OSD_VIDWINMD_VHZ1 | OSD_VIDWINMD_VVZ1, winmd,
435 OSD_VIDWINMD);
436 break;
440 static void _osd_disable_layer(struct osd_state *sd, enum osd_layer layer)
442 switch (layer) {
443 case WIN_OSD0:
444 osd_clear(sd, OSD_OSDWIN0MD_OACT0, OSD_OSDWIN0MD);
445 break;
446 case WIN_VID0:
447 osd_clear(sd, OSD_VIDWINMD_ACT0, OSD_VIDWINMD);
448 break;
449 case WIN_OSD1:
450 /* disable attribute mode as well as disabling the window */
451 osd_clear(sd, OSD_OSDWIN1MD_OASW | OSD_OSDWIN1MD_OACT1,
452 OSD_OSDWIN1MD);
453 break;
454 case WIN_VID1:
455 osd_clear(sd, OSD_VIDWINMD_ACT1, OSD_VIDWINMD);
456 break;
460 static void osd_disable_layer(struct osd_state *sd, enum osd_layer layer)
462 struct osd_state *osd = sd;
463 struct osd_window_state *win = &osd->win[layer];
464 unsigned long flags;
466 spin_lock_irqsave(&osd->lock, flags);
468 if (!win->is_enabled) {
469 spin_unlock_irqrestore(&osd->lock, flags);
470 return;
472 win->is_enabled = 0;
474 _osd_disable_layer(sd, layer);
476 spin_unlock_irqrestore(&osd->lock, flags);
479 static void _osd_enable_attribute_mode(struct osd_state *sd)
481 /* enable attribute mode for OSD1 */
482 osd_set(sd, OSD_OSDWIN1MD_OASW, OSD_OSDWIN1MD);
485 static void _osd_enable_layer(struct osd_state *sd, enum osd_layer layer)
487 switch (layer) {
488 case WIN_OSD0:
489 osd_set(sd, OSD_OSDWIN0MD_OACT0, OSD_OSDWIN0MD);
490 break;
491 case WIN_VID0:
492 osd_set(sd, OSD_VIDWINMD_ACT0, OSD_VIDWINMD);
493 break;
494 case WIN_OSD1:
495 /* enable OSD1 and disable attribute mode */
496 osd_modify(sd, OSD_OSDWIN1MD_OASW | OSD_OSDWIN1MD_OACT1,
497 OSD_OSDWIN1MD_OACT1, OSD_OSDWIN1MD);
498 break;
499 case WIN_VID1:
500 osd_set(sd, OSD_VIDWINMD_ACT1, OSD_VIDWINMD);
501 break;
505 static int osd_enable_layer(struct osd_state *sd, enum osd_layer layer,
506 int otherwin)
508 struct osd_state *osd = sd;
509 struct osd_window_state *win = &osd->win[layer];
510 struct osd_layer_config *cfg = &win->lconfig;
511 unsigned long flags;
513 spin_lock_irqsave(&osd->lock, flags);
516 * use otherwin flag to know this is the other vid window
517 * in YUV420 mode, if is, skip this check
519 if (!otherwin && (!win->is_allocated ||
520 !win->fb_base_phys ||
521 !cfg->line_length ||
522 !cfg->xsize ||
523 !cfg->ysize)) {
524 spin_unlock_irqrestore(&osd->lock, flags);
525 return -1;
528 if (win->is_enabled) {
529 spin_unlock_irqrestore(&osd->lock, flags);
530 return 0;
532 win->is_enabled = 1;
534 if (cfg->pixfmt != PIXFMT_OSD_ATTR)
535 _osd_enable_layer(sd, layer);
536 else {
537 _osd_enable_attribute_mode(sd);
538 _osd_set_blink_attribute(sd, osd->is_blinking, osd->blink);
541 spin_unlock_irqrestore(&osd->lock, flags);
543 return 0;
546 #define OSD_SRC_ADDR_HIGH4 0x7800000
547 #define OSD_SRC_ADDR_HIGH7 0x7F0000
548 #define OSD_SRCADD_OFSET_SFT 23
549 #define OSD_SRCADD_ADD_SFT 16
550 #define OSD_WINADL_MASK 0xFFFF
551 #define OSD_WINOFST_MASK 0x1000
552 #define VPBE_REG_BASE 0x80000000
554 static void _osd_start_layer(struct osd_state *sd, enum osd_layer layer,
555 unsigned long fb_base_phys,
556 unsigned long cbcr_ofst)
559 if (sd->vpbe_type == VPBE_VERSION_1) {
560 switch (layer) {
561 case WIN_OSD0:
562 osd_write(sd, fb_base_phys & ~0x1F, OSD_OSDWIN0ADR);
563 break;
564 case WIN_VID0:
565 osd_write(sd, fb_base_phys & ~0x1F, OSD_VIDWIN0ADR);
566 break;
567 case WIN_OSD1:
568 osd_write(sd, fb_base_phys & ~0x1F, OSD_OSDWIN1ADR);
569 break;
570 case WIN_VID1:
571 osd_write(sd, fb_base_phys & ~0x1F, OSD_VIDWIN1ADR);
572 break;
574 } else if (sd->vpbe_type == VPBE_VERSION_3) {
575 unsigned long fb_offset_32 =
576 (fb_base_phys - VPBE_REG_BASE) >> 5;
578 switch (layer) {
579 case WIN_OSD0:
580 osd_modify(sd, OSD_OSDWINADH_O0AH,
581 fb_offset_32 >> (OSD_SRCADD_ADD_SFT -
582 OSD_OSDWINADH_O0AH_SHIFT),
583 OSD_OSDWINADH);
584 osd_write(sd, fb_offset_32 & OSD_OSDWIN0ADL_O0AL,
585 OSD_OSDWIN0ADL);
586 break;
587 case WIN_VID0:
588 osd_modify(sd, OSD_VIDWINADH_V0AH,
589 fb_offset_32 >> (OSD_SRCADD_ADD_SFT -
590 OSD_VIDWINADH_V0AH_SHIFT),
591 OSD_VIDWINADH);
592 osd_write(sd, fb_offset_32 & OSD_VIDWIN0ADL_V0AL,
593 OSD_VIDWIN0ADL);
594 break;
595 case WIN_OSD1:
596 osd_modify(sd, OSD_OSDWINADH_O1AH,
597 fb_offset_32 >> (OSD_SRCADD_ADD_SFT -
598 OSD_OSDWINADH_O1AH_SHIFT),
599 OSD_OSDWINADH);
600 osd_write(sd, fb_offset_32 & OSD_OSDWIN1ADL_O1AL,
601 OSD_OSDWIN1ADL);
602 break;
603 case WIN_VID1:
604 osd_modify(sd, OSD_VIDWINADH_V1AH,
605 fb_offset_32 >> (OSD_SRCADD_ADD_SFT -
606 OSD_VIDWINADH_V1AH_SHIFT),
607 OSD_VIDWINADH);
608 osd_write(sd, fb_offset_32 & OSD_VIDWIN1ADL_V1AL,
609 OSD_VIDWIN1ADL);
610 break;
612 } else if (sd->vpbe_type == VPBE_VERSION_2) {
613 struct osd_window_state *win = &sd->win[layer];
614 unsigned long fb_offset_32, cbcr_offset_32;
616 fb_offset_32 = fb_base_phys - VPBE_REG_BASE;
617 if (cbcr_ofst)
618 cbcr_offset_32 = cbcr_ofst;
619 else
620 cbcr_offset_32 = win->lconfig.line_length *
621 win->lconfig.ysize;
622 cbcr_offset_32 += fb_offset_32;
623 fb_offset_32 = fb_offset_32 >> 5;
624 cbcr_offset_32 = cbcr_offset_32 >> 5;
626 * DM365: start address is 27-bit long address b26 - b23 are
627 * in offset register b12 - b9, and * bit 26 has to be '1'
629 if (win->lconfig.pixfmt == PIXFMT_NV12) {
630 switch (layer) {
631 case WIN_VID0:
632 case WIN_VID1:
633 /* Y is in VID0 */
634 osd_modify(sd, OSD_VIDWIN0OFST_V0AH,
635 ((fb_offset_32 & OSD_SRC_ADDR_HIGH4) >>
636 (OSD_SRCADD_OFSET_SFT -
637 OSD_WINOFST_AH_SHIFT)) |
638 OSD_WINOFST_MASK, OSD_VIDWIN0OFST);
639 osd_modify(sd, OSD_VIDWINADH_V0AH,
640 (fb_offset_32 & OSD_SRC_ADDR_HIGH7) >>
641 (OSD_SRCADD_ADD_SFT -
642 OSD_VIDWINADH_V0AH_SHIFT),
643 OSD_VIDWINADH);
644 osd_write(sd, fb_offset_32 & OSD_WINADL_MASK,
645 OSD_VIDWIN0ADL);
646 /* CbCr is in VID1 */
647 osd_modify(sd, OSD_VIDWIN1OFST_V1AH,
648 ((cbcr_offset_32 &
649 OSD_SRC_ADDR_HIGH4) >>
650 (OSD_SRCADD_OFSET_SFT -
651 OSD_WINOFST_AH_SHIFT)) |
652 OSD_WINOFST_MASK, OSD_VIDWIN1OFST);
653 osd_modify(sd, OSD_VIDWINADH_V1AH,
654 (cbcr_offset_32 &
655 OSD_SRC_ADDR_HIGH7) >>
656 (OSD_SRCADD_ADD_SFT -
657 OSD_VIDWINADH_V1AH_SHIFT),
658 OSD_VIDWINADH);
659 osd_write(sd, cbcr_offset_32 & OSD_WINADL_MASK,
660 OSD_VIDWIN1ADL);
661 break;
662 default:
663 break;
667 switch (layer) {
668 case WIN_OSD0:
669 osd_modify(sd, OSD_OSDWIN0OFST_O0AH,
670 ((fb_offset_32 & OSD_SRC_ADDR_HIGH4) >>
671 (OSD_SRCADD_OFSET_SFT -
672 OSD_WINOFST_AH_SHIFT)) | OSD_WINOFST_MASK,
673 OSD_OSDWIN0OFST);
674 osd_modify(sd, OSD_OSDWINADH_O0AH,
675 (fb_offset_32 & OSD_SRC_ADDR_HIGH7) >>
676 (OSD_SRCADD_ADD_SFT -
677 OSD_OSDWINADH_O0AH_SHIFT), OSD_OSDWINADH);
678 osd_write(sd, fb_offset_32 & OSD_WINADL_MASK,
679 OSD_OSDWIN0ADL);
680 break;
681 case WIN_VID0:
682 if (win->lconfig.pixfmt != PIXFMT_NV12) {
683 osd_modify(sd, OSD_VIDWIN0OFST_V0AH,
684 ((fb_offset_32 & OSD_SRC_ADDR_HIGH4) >>
685 (OSD_SRCADD_OFSET_SFT -
686 OSD_WINOFST_AH_SHIFT)) |
687 OSD_WINOFST_MASK, OSD_VIDWIN0OFST);
688 osd_modify(sd, OSD_VIDWINADH_V0AH,
689 (fb_offset_32 & OSD_SRC_ADDR_HIGH7) >>
690 (OSD_SRCADD_ADD_SFT -
691 OSD_VIDWINADH_V0AH_SHIFT),
692 OSD_VIDWINADH);
693 osd_write(sd, fb_offset_32 & OSD_WINADL_MASK,
694 OSD_VIDWIN0ADL);
696 break;
697 case WIN_OSD1:
698 osd_modify(sd, OSD_OSDWIN1OFST_O1AH,
699 ((fb_offset_32 & OSD_SRC_ADDR_HIGH4) >>
700 (OSD_SRCADD_OFSET_SFT -
701 OSD_WINOFST_AH_SHIFT)) | OSD_WINOFST_MASK,
702 OSD_OSDWIN1OFST);
703 osd_modify(sd, OSD_OSDWINADH_O1AH,
704 (fb_offset_32 & OSD_SRC_ADDR_HIGH7) >>
705 (OSD_SRCADD_ADD_SFT -
706 OSD_OSDWINADH_O1AH_SHIFT),
707 OSD_OSDWINADH);
708 osd_write(sd, fb_offset_32 & OSD_WINADL_MASK,
709 OSD_OSDWIN1ADL);
710 break;
711 case WIN_VID1:
712 if (win->lconfig.pixfmt != PIXFMT_NV12) {
713 osd_modify(sd, OSD_VIDWIN1OFST_V1AH,
714 ((fb_offset_32 & OSD_SRC_ADDR_HIGH4) >>
715 (OSD_SRCADD_OFSET_SFT -
716 OSD_WINOFST_AH_SHIFT)) |
717 OSD_WINOFST_MASK, OSD_VIDWIN1OFST);
718 osd_modify(sd, OSD_VIDWINADH_V1AH,
719 (fb_offset_32 & OSD_SRC_ADDR_HIGH7) >>
720 (OSD_SRCADD_ADD_SFT -
721 OSD_VIDWINADH_V1AH_SHIFT),
722 OSD_VIDWINADH);
723 osd_write(sd, fb_offset_32 & OSD_WINADL_MASK,
724 OSD_VIDWIN1ADL);
726 break;
731 static void osd_start_layer(struct osd_state *sd, enum osd_layer layer,
732 unsigned long fb_base_phys,
733 unsigned long cbcr_ofst)
735 struct osd_state *osd = sd;
736 struct osd_window_state *win = &osd->win[layer];
737 struct osd_layer_config *cfg = &win->lconfig;
738 unsigned long flags;
740 spin_lock_irqsave(&osd->lock, flags);
742 win->fb_base_phys = fb_base_phys & ~0x1F;
743 _osd_start_layer(sd, layer, fb_base_phys, cbcr_ofst);
745 if (layer == WIN_VID0) {
746 osd->pingpong =
747 _osd_dm6446_vid0_pingpong(sd, osd->field_inversion,
748 win->fb_base_phys,
749 cfg);
752 spin_unlock_irqrestore(&osd->lock, flags);
755 static void osd_get_layer_config(struct osd_state *sd, enum osd_layer layer,
756 struct osd_layer_config *lconfig)
758 struct osd_state *osd = sd;
759 struct osd_window_state *win = &osd->win[layer];
760 unsigned long flags;
762 spin_lock_irqsave(&osd->lock, flags);
764 *lconfig = win->lconfig;
766 spin_unlock_irqrestore(&osd->lock, flags);
770 * try_layer_config() - Try a specific configuration for the layer
771 * @sd - ptr to struct osd_state
772 * @layer - layer to configure
773 * @lconfig - layer configuration to try
775 * If the requested lconfig is completely rejected and the value of lconfig on
776 * exit is the current lconfig, then try_layer_config() returns 1. Otherwise,
777 * try_layer_config() returns 0. A return value of 0 does not necessarily mean
778 * that the value of lconfig on exit is identical to the value of lconfig on
779 * entry, but merely that it represents a change from the current lconfig.
781 static int try_layer_config(struct osd_state *sd, enum osd_layer layer,
782 struct osd_layer_config *lconfig)
784 struct osd_state *osd = sd;
785 struct osd_window_state *win = &osd->win[layer];
786 int bad_config = 0;
788 /* verify that the pixel format is compatible with the layer */
789 switch (lconfig->pixfmt) {
790 case PIXFMT_1BPP:
791 case PIXFMT_2BPP:
792 case PIXFMT_4BPP:
793 case PIXFMT_8BPP:
794 case PIXFMT_RGB565:
795 if (osd->vpbe_type == VPBE_VERSION_1)
796 bad_config = !is_vid_win(layer);
797 break;
798 case PIXFMT_YCbCrI:
799 case PIXFMT_YCrCbI:
800 bad_config = !is_vid_win(layer);
801 break;
802 case PIXFMT_RGB888:
803 if (osd->vpbe_type == VPBE_VERSION_1)
804 bad_config = !is_vid_win(layer);
805 else if ((osd->vpbe_type == VPBE_VERSION_3) ||
806 (osd->vpbe_type == VPBE_VERSION_2))
807 bad_config = !is_osd_win(layer);
808 break;
809 case PIXFMT_NV12:
810 if (osd->vpbe_type != VPBE_VERSION_2)
811 bad_config = 1;
812 else
813 bad_config = is_osd_win(layer);
814 break;
815 case PIXFMT_OSD_ATTR:
816 bad_config = (layer != WIN_OSD1);
817 break;
818 default:
819 bad_config = 1;
820 break;
822 if (bad_config) {
824 * The requested pixel format is incompatible with the layer,
825 * so keep the current layer configuration.
827 *lconfig = win->lconfig;
828 return bad_config;
831 /* DM6446: */
832 /* only one OSD window at a time can use RGB pixel formats */
833 if ((osd->vpbe_type == VPBE_VERSION_1) &&
834 is_osd_win(layer) && is_rgb_pixfmt(lconfig->pixfmt)) {
835 enum osd_pix_format pixfmt;
836 if (layer == WIN_OSD0)
837 pixfmt = osd->win[WIN_OSD1].lconfig.pixfmt;
838 else
839 pixfmt = osd->win[WIN_OSD0].lconfig.pixfmt;
841 if (is_rgb_pixfmt(pixfmt)) {
843 * The other OSD window is already configured for an
844 * RGB, so keep the current layer configuration.
846 *lconfig = win->lconfig;
847 return 1;
851 /* DM6446: only one video window at a time can use RGB888 */
852 if ((osd->vpbe_type == VPBE_VERSION_1) && is_vid_win(layer) &&
853 lconfig->pixfmt == PIXFMT_RGB888) {
854 enum osd_pix_format pixfmt;
856 if (layer == WIN_VID0)
857 pixfmt = osd->win[WIN_VID1].lconfig.pixfmt;
858 else
859 pixfmt = osd->win[WIN_VID0].lconfig.pixfmt;
861 if (pixfmt == PIXFMT_RGB888) {
863 * The other video window is already configured for
864 * RGB888, so keep the current layer configuration.
866 *lconfig = win->lconfig;
867 return 1;
871 /* window dimensions must be non-zero */
872 if (!lconfig->line_length || !lconfig->xsize || !lconfig->ysize) {
873 *lconfig = win->lconfig;
874 return 1;
877 /* round line_length up to a multiple of 32 */
878 lconfig->line_length = ((lconfig->line_length + 31) / 32) * 32;
879 lconfig->line_length =
880 min(lconfig->line_length, (unsigned)MAX_LINE_LENGTH);
881 lconfig->xsize = min(lconfig->xsize, (unsigned)MAX_WIN_SIZE);
882 lconfig->ysize = min(lconfig->ysize, (unsigned)MAX_WIN_SIZE);
883 lconfig->xpos = min(lconfig->xpos, (unsigned)MAX_WIN_SIZE);
884 lconfig->ypos = min(lconfig->ypos, (unsigned)MAX_WIN_SIZE);
885 lconfig->interlaced = (lconfig->interlaced != 0);
886 if (lconfig->interlaced) {
887 /* ysize and ypos must be even for interlaced displays */
888 lconfig->ysize &= ~1;
889 lconfig->ypos &= ~1;
892 return 0;
895 static void _osd_disable_vid_rgb888(struct osd_state *sd)
898 * The DM6446 supports RGB888 pixel format in a single video window.
899 * This routine disables RGB888 pixel format for both video windows.
900 * The caller must ensure that neither video window is currently
901 * configured for RGB888 pixel format.
903 if (sd->vpbe_type == VPBE_VERSION_1)
904 osd_clear(sd, OSD_MISCCTL_RGBEN, OSD_MISCCTL);
907 static void _osd_enable_vid_rgb888(struct osd_state *sd,
908 enum osd_layer layer)
911 * The DM6446 supports RGB888 pixel format in a single video window.
912 * This routine enables RGB888 pixel format for the specified video
913 * window. The caller must ensure that the other video window is not
914 * currently configured for RGB888 pixel format, as this routine will
915 * disable RGB888 pixel format for the other window.
917 if (sd->vpbe_type == VPBE_VERSION_1) {
918 if (layer == WIN_VID0)
919 osd_modify(sd, OSD_MISCCTL_RGBEN | OSD_MISCCTL_RGBWIN,
920 OSD_MISCCTL_RGBEN, OSD_MISCCTL);
921 else if (layer == WIN_VID1)
922 osd_modify(sd, OSD_MISCCTL_RGBEN | OSD_MISCCTL_RGBWIN,
923 OSD_MISCCTL_RGBEN | OSD_MISCCTL_RGBWIN,
924 OSD_MISCCTL);
928 static void _osd_set_cbcr_order(struct osd_state *sd,
929 enum osd_pix_format pixfmt)
932 * The caller must ensure that all windows using YC pixfmt use the same
933 * Cb/Cr order.
935 if (pixfmt == PIXFMT_YCbCrI)
936 osd_clear(sd, OSD_MODE_CS, OSD_MODE);
937 else if (pixfmt == PIXFMT_YCrCbI)
938 osd_set(sd, OSD_MODE_CS, OSD_MODE);
941 static void _osd_set_layer_config(struct osd_state *sd, enum osd_layer layer,
942 const struct osd_layer_config *lconfig)
944 u32 winmd = 0, winmd_mask = 0, bmw = 0;
946 _osd_set_cbcr_order(sd, lconfig->pixfmt);
948 switch (layer) {
949 case WIN_OSD0:
950 if (sd->vpbe_type == VPBE_VERSION_1) {
951 winmd_mask |= OSD_OSDWIN0MD_RGB0E;
952 if (lconfig->pixfmt == PIXFMT_RGB565)
953 winmd |= OSD_OSDWIN0MD_RGB0E;
954 } else if ((sd->vpbe_type == VPBE_VERSION_3) ||
955 (sd->vpbe_type == VPBE_VERSION_2)) {
956 winmd_mask |= OSD_OSDWIN0MD_BMP0MD;
957 switch (lconfig->pixfmt) {
958 case PIXFMT_RGB565:
959 winmd |= (1 <<
960 OSD_OSDWIN0MD_BMP0MD_SHIFT);
961 break;
962 case PIXFMT_RGB888:
963 winmd |= (2 << OSD_OSDWIN0MD_BMP0MD_SHIFT);
964 _osd_enable_rgb888_pixblend(sd, OSDWIN_OSD0);
965 break;
966 case PIXFMT_YCbCrI:
967 case PIXFMT_YCrCbI:
968 winmd |= (3 << OSD_OSDWIN0MD_BMP0MD_SHIFT);
969 break;
970 default:
971 break;
975 winmd_mask |= OSD_OSDWIN0MD_BMW0 | OSD_OSDWIN0MD_OFF0;
977 switch (lconfig->pixfmt) {
978 case PIXFMT_1BPP:
979 bmw = 0;
980 break;
981 case PIXFMT_2BPP:
982 bmw = 1;
983 break;
984 case PIXFMT_4BPP:
985 bmw = 2;
986 break;
987 case PIXFMT_8BPP:
988 bmw = 3;
989 break;
990 default:
991 break;
993 winmd |= (bmw << OSD_OSDWIN0MD_BMW0_SHIFT);
995 if (lconfig->interlaced)
996 winmd |= OSD_OSDWIN0MD_OFF0;
998 osd_modify(sd, winmd_mask, winmd, OSD_OSDWIN0MD);
999 osd_write(sd, lconfig->line_length >> 5, OSD_OSDWIN0OFST);
1000 osd_write(sd, lconfig->xpos, OSD_OSDWIN0XP);
1001 osd_write(sd, lconfig->xsize, OSD_OSDWIN0XL);
1002 if (lconfig->interlaced) {
1003 osd_write(sd, lconfig->ypos >> 1, OSD_OSDWIN0YP);
1004 osd_write(sd, lconfig->ysize >> 1, OSD_OSDWIN0YL);
1005 } else {
1006 osd_write(sd, lconfig->ypos, OSD_OSDWIN0YP);
1007 osd_write(sd, lconfig->ysize, OSD_OSDWIN0YL);
1009 break;
1010 case WIN_VID0:
1011 winmd_mask |= OSD_VIDWINMD_VFF0;
1012 if (lconfig->interlaced)
1013 winmd |= OSD_VIDWINMD_VFF0;
1015 osd_modify(sd, winmd_mask, winmd, OSD_VIDWINMD);
1016 osd_write(sd, lconfig->line_length >> 5, OSD_VIDWIN0OFST);
1017 osd_write(sd, lconfig->xpos, OSD_VIDWIN0XP);
1018 osd_write(sd, lconfig->xsize, OSD_VIDWIN0XL);
1020 * For YUV420P format the register contents are
1021 * duplicated in both VID registers
1023 if ((sd->vpbe_type == VPBE_VERSION_2) &&
1024 (lconfig->pixfmt == PIXFMT_NV12)) {
1025 /* other window also */
1026 if (lconfig->interlaced) {
1027 winmd_mask |= OSD_VIDWINMD_VFF1;
1028 winmd |= OSD_VIDWINMD_VFF1;
1029 osd_modify(sd, winmd_mask, winmd,
1030 OSD_VIDWINMD);
1033 osd_modify(sd, OSD_MISCCTL_S420D,
1034 OSD_MISCCTL_S420D, OSD_MISCCTL);
1035 osd_write(sd, lconfig->line_length >> 5,
1036 OSD_VIDWIN1OFST);
1037 osd_write(sd, lconfig->xpos, OSD_VIDWIN1XP);
1038 osd_write(sd, lconfig->xsize, OSD_VIDWIN1XL);
1040 * if NV21 pixfmt and line length not 32B
1041 * aligned (e.g. NTSC), Need to set window
1042 * X pixel size to be 32B aligned as well
1044 if (lconfig->xsize % 32) {
1045 osd_write(sd,
1046 ((lconfig->xsize + 31) & ~31),
1047 OSD_VIDWIN1XL);
1048 osd_write(sd,
1049 ((lconfig->xsize + 31) & ~31),
1050 OSD_VIDWIN0XL);
1052 } else if ((sd->vpbe_type == VPBE_VERSION_2) &&
1053 (lconfig->pixfmt != PIXFMT_NV12)) {
1054 osd_modify(sd, OSD_MISCCTL_S420D, ~OSD_MISCCTL_S420D,
1055 OSD_MISCCTL);
1058 if (lconfig->interlaced) {
1059 osd_write(sd, lconfig->ypos >> 1, OSD_VIDWIN0YP);
1060 osd_write(sd, lconfig->ysize >> 1, OSD_VIDWIN0YL);
1061 if ((sd->vpbe_type == VPBE_VERSION_2) &&
1062 lconfig->pixfmt == PIXFMT_NV12) {
1063 osd_write(sd, lconfig->ypos >> 1,
1064 OSD_VIDWIN1YP);
1065 osd_write(sd, lconfig->ysize >> 1,
1066 OSD_VIDWIN1YL);
1068 } else {
1069 osd_write(sd, lconfig->ypos, OSD_VIDWIN0YP);
1070 osd_write(sd, lconfig->ysize, OSD_VIDWIN0YL);
1071 if ((sd->vpbe_type == VPBE_VERSION_2) &&
1072 lconfig->pixfmt == PIXFMT_NV12) {
1073 osd_write(sd, lconfig->ypos, OSD_VIDWIN1YP);
1074 osd_write(sd, lconfig->ysize, OSD_VIDWIN1YL);
1077 break;
1078 case WIN_OSD1:
1080 * The caller must ensure that OSD1 is disabled prior to
1081 * switching from a normal mode to attribute mode or from
1082 * attribute mode to a normal mode.
1084 if (lconfig->pixfmt == PIXFMT_OSD_ATTR) {
1085 if (sd->vpbe_type == VPBE_VERSION_1) {
1086 winmd_mask |= OSD_OSDWIN1MD_ATN1E |
1087 OSD_OSDWIN1MD_RGB1E | OSD_OSDWIN1MD_CLUTS1 |
1088 OSD_OSDWIN1MD_BLND1 | OSD_OSDWIN1MD_TE1;
1089 } else {
1090 winmd_mask |= OSD_OSDWIN1MD_BMP1MD |
1091 OSD_OSDWIN1MD_CLUTS1 | OSD_OSDWIN1MD_BLND1 |
1092 OSD_OSDWIN1MD_TE1;
1094 } else {
1095 if (sd->vpbe_type == VPBE_VERSION_1) {
1096 winmd_mask |= OSD_OSDWIN1MD_RGB1E;
1097 if (lconfig->pixfmt == PIXFMT_RGB565)
1098 winmd |= OSD_OSDWIN1MD_RGB1E;
1099 } else if ((sd->vpbe_type == VPBE_VERSION_3)
1100 || (sd->vpbe_type == VPBE_VERSION_2)) {
1101 winmd_mask |= OSD_OSDWIN1MD_BMP1MD;
1102 switch (lconfig->pixfmt) {
1103 case PIXFMT_RGB565:
1104 winmd |=
1105 (1 << OSD_OSDWIN1MD_BMP1MD_SHIFT);
1106 break;
1107 case PIXFMT_RGB888:
1108 winmd |=
1109 (2 << OSD_OSDWIN1MD_BMP1MD_SHIFT);
1110 _osd_enable_rgb888_pixblend(sd,
1111 OSDWIN_OSD1);
1112 break;
1113 case PIXFMT_YCbCrI:
1114 case PIXFMT_YCrCbI:
1115 winmd |=
1116 (3 << OSD_OSDWIN1MD_BMP1MD_SHIFT);
1117 break;
1118 default:
1119 break;
1123 winmd_mask |= OSD_OSDWIN1MD_BMW1;
1124 switch (lconfig->pixfmt) {
1125 case PIXFMT_1BPP:
1126 bmw = 0;
1127 break;
1128 case PIXFMT_2BPP:
1129 bmw = 1;
1130 break;
1131 case PIXFMT_4BPP:
1132 bmw = 2;
1133 break;
1134 case PIXFMT_8BPP:
1135 bmw = 3;
1136 break;
1137 default:
1138 break;
1140 winmd |= (bmw << OSD_OSDWIN1MD_BMW1_SHIFT);
1143 winmd_mask |= OSD_OSDWIN1MD_OFF1;
1144 if (lconfig->interlaced)
1145 winmd |= OSD_OSDWIN1MD_OFF1;
1147 osd_modify(sd, winmd_mask, winmd, OSD_OSDWIN1MD);
1148 osd_write(sd, lconfig->line_length >> 5, OSD_OSDWIN1OFST);
1149 osd_write(sd, lconfig->xpos, OSD_OSDWIN1XP);
1150 osd_write(sd, lconfig->xsize, OSD_OSDWIN1XL);
1151 if (lconfig->interlaced) {
1152 osd_write(sd, lconfig->ypos >> 1, OSD_OSDWIN1YP);
1153 osd_write(sd, lconfig->ysize >> 1, OSD_OSDWIN1YL);
1154 } else {
1155 osd_write(sd, lconfig->ypos, OSD_OSDWIN1YP);
1156 osd_write(sd, lconfig->ysize, OSD_OSDWIN1YL);
1158 break;
1159 case WIN_VID1:
1160 winmd_mask |= OSD_VIDWINMD_VFF1;
1161 if (lconfig->interlaced)
1162 winmd |= OSD_VIDWINMD_VFF1;
1164 osd_modify(sd, winmd_mask, winmd, OSD_VIDWINMD);
1165 osd_write(sd, lconfig->line_length >> 5, OSD_VIDWIN1OFST);
1166 osd_write(sd, lconfig->xpos, OSD_VIDWIN1XP);
1167 osd_write(sd, lconfig->xsize, OSD_VIDWIN1XL);
1169 * For YUV420P format the register contents are
1170 * duplicated in both VID registers
1172 if (sd->vpbe_type == VPBE_VERSION_2) {
1173 if (lconfig->pixfmt == PIXFMT_NV12) {
1174 /* other window also */
1175 if (lconfig->interlaced) {
1176 winmd_mask |= OSD_VIDWINMD_VFF0;
1177 winmd |= OSD_VIDWINMD_VFF0;
1178 osd_modify(sd, winmd_mask, winmd,
1179 OSD_VIDWINMD);
1181 osd_modify(sd, OSD_MISCCTL_S420D,
1182 OSD_MISCCTL_S420D, OSD_MISCCTL);
1183 osd_write(sd, lconfig->line_length >> 5,
1184 OSD_VIDWIN0OFST);
1185 osd_write(sd, lconfig->xpos, OSD_VIDWIN0XP);
1186 osd_write(sd, lconfig->xsize, OSD_VIDWIN0XL);
1187 } else {
1188 osd_modify(sd, OSD_MISCCTL_S420D,
1189 ~OSD_MISCCTL_S420D, OSD_MISCCTL);
1193 if (lconfig->interlaced) {
1194 osd_write(sd, lconfig->ypos >> 1, OSD_VIDWIN1YP);
1195 osd_write(sd, lconfig->ysize >> 1, OSD_VIDWIN1YL);
1196 if ((sd->vpbe_type == VPBE_VERSION_2) &&
1197 lconfig->pixfmt == PIXFMT_NV12) {
1198 osd_write(sd, lconfig->ypos >> 1,
1199 OSD_VIDWIN0YP);
1200 osd_write(sd, lconfig->ysize >> 1,
1201 OSD_VIDWIN0YL);
1203 } else {
1204 osd_write(sd, lconfig->ypos, OSD_VIDWIN1YP);
1205 osd_write(sd, lconfig->ysize, OSD_VIDWIN1YL);
1206 if ((sd->vpbe_type == VPBE_VERSION_2) &&
1207 lconfig->pixfmt == PIXFMT_NV12) {
1208 osd_write(sd, lconfig->ypos, OSD_VIDWIN0YP);
1209 osd_write(sd, lconfig->ysize, OSD_VIDWIN0YL);
1212 break;
1216 static int osd_set_layer_config(struct osd_state *sd, enum osd_layer layer,
1217 struct osd_layer_config *lconfig)
1219 struct osd_state *osd = sd;
1220 struct osd_window_state *win = &osd->win[layer];
1221 struct osd_layer_config *cfg = &win->lconfig;
1222 unsigned long flags;
1223 int reject_config;
1225 spin_lock_irqsave(&osd->lock, flags);
1227 reject_config = try_layer_config(sd, layer, lconfig);
1228 if (reject_config) {
1229 spin_unlock_irqrestore(&osd->lock, flags);
1230 return reject_config;
1233 /* update the current Cb/Cr order */
1234 if (is_yc_pixfmt(lconfig->pixfmt))
1235 osd->yc_pixfmt = lconfig->pixfmt;
1238 * If we are switching OSD1 from normal mode to attribute mode or from
1239 * attribute mode to normal mode, then we must disable the window.
1241 if (layer == WIN_OSD1) {
1242 if (((lconfig->pixfmt == PIXFMT_OSD_ATTR) &&
1243 (cfg->pixfmt != PIXFMT_OSD_ATTR)) ||
1244 ((lconfig->pixfmt != PIXFMT_OSD_ATTR) &&
1245 (cfg->pixfmt == PIXFMT_OSD_ATTR))) {
1246 win->is_enabled = 0;
1247 _osd_disable_layer(sd, layer);
1251 _osd_set_layer_config(sd, layer, lconfig);
1253 if (layer == WIN_OSD1) {
1254 struct osd_osdwin_state *osdwin_state =
1255 &osd->osdwin[OSDWIN_OSD1];
1257 if ((lconfig->pixfmt != PIXFMT_OSD_ATTR) &&
1258 (cfg->pixfmt == PIXFMT_OSD_ATTR)) {
1260 * We just switched OSD1 from attribute mode to normal
1261 * mode, so we must initialize the CLUT select, the
1262 * blend factor, transparency colorkey enable, and
1263 * attenuation enable (DM6446 only) bits in the
1264 * OSDWIN1MD register.
1266 _osd_set_osd_clut(sd, OSDWIN_OSD1,
1267 osdwin_state->clut);
1268 _osd_set_blending_factor(sd, OSDWIN_OSD1,
1269 osdwin_state->blend);
1270 if (osdwin_state->colorkey_blending) {
1271 _osd_enable_color_key(sd, OSDWIN_OSD1,
1272 osdwin_state->
1273 colorkey,
1274 lconfig->pixfmt);
1275 } else
1276 _osd_disable_color_key(sd, OSDWIN_OSD1);
1277 _osd_set_rec601_attenuation(sd, OSDWIN_OSD1,
1278 osdwin_state->
1279 rec601_attenuation);
1280 } else if ((lconfig->pixfmt == PIXFMT_OSD_ATTR) &&
1281 (cfg->pixfmt != PIXFMT_OSD_ATTR)) {
1283 * We just switched OSD1 from normal mode to attribute
1284 * mode, so we must initialize the blink enable and
1285 * blink interval bits in the OSDATRMD register.
1287 _osd_set_blink_attribute(sd, osd->is_blinking,
1288 osd->blink);
1293 * If we just switched to a 1-, 2-, or 4-bits-per-pixel bitmap format
1294 * then configure a default palette map.
1296 if ((lconfig->pixfmt != cfg->pixfmt) &&
1297 ((lconfig->pixfmt == PIXFMT_1BPP) ||
1298 (lconfig->pixfmt == PIXFMT_2BPP) ||
1299 (lconfig->pixfmt == PIXFMT_4BPP))) {
1300 enum osd_win_layer osdwin =
1301 ((layer == WIN_OSD0) ? OSDWIN_OSD0 : OSDWIN_OSD1);
1302 struct osd_osdwin_state *osdwin_state =
1303 &osd->osdwin[osdwin];
1304 unsigned char clut_index;
1305 unsigned char clut_entries = 0;
1307 switch (lconfig->pixfmt) {
1308 case PIXFMT_1BPP:
1309 clut_entries = 2;
1310 break;
1311 case PIXFMT_2BPP:
1312 clut_entries = 4;
1313 break;
1314 case PIXFMT_4BPP:
1315 clut_entries = 16;
1316 break;
1317 default:
1318 break;
1321 * The default palette map maps the pixel value to the clut
1322 * index, i.e. pixel value 0 maps to clut entry 0, pixel value
1323 * 1 maps to clut entry 1, etc.
1325 for (clut_index = 0; clut_index < 16; clut_index++) {
1326 osdwin_state->palette_map[clut_index] = clut_index;
1327 if (clut_index < clut_entries) {
1328 _osd_set_palette_map(sd, osdwin, clut_index,
1329 clut_index,
1330 lconfig->pixfmt);
1335 *cfg = *lconfig;
1336 /* DM6446: configure the RGB888 enable and window selection */
1337 if (osd->win[WIN_VID0].lconfig.pixfmt == PIXFMT_RGB888)
1338 _osd_enable_vid_rgb888(sd, WIN_VID0);
1339 else if (osd->win[WIN_VID1].lconfig.pixfmt == PIXFMT_RGB888)
1340 _osd_enable_vid_rgb888(sd, WIN_VID1);
1341 else
1342 _osd_disable_vid_rgb888(sd);
1344 if (layer == WIN_VID0) {
1345 osd->pingpong =
1346 _osd_dm6446_vid0_pingpong(sd, osd->field_inversion,
1347 win->fb_base_phys,
1348 cfg);
1351 spin_unlock_irqrestore(&osd->lock, flags);
1353 return 0;
1356 static void osd_init_layer(struct osd_state *sd, enum osd_layer layer)
1358 struct osd_state *osd = sd;
1359 struct osd_window_state *win = &osd->win[layer];
1360 enum osd_win_layer osdwin;
1361 struct osd_osdwin_state *osdwin_state;
1362 struct osd_layer_config *cfg = &win->lconfig;
1363 unsigned long flags;
1365 spin_lock_irqsave(&osd->lock, flags);
1367 win->is_enabled = 0;
1368 _osd_disable_layer(sd, layer);
1370 win->h_zoom = ZOOM_X1;
1371 win->v_zoom = ZOOM_X1;
1372 _osd_set_zoom(sd, layer, win->h_zoom, win->v_zoom);
1374 win->fb_base_phys = 0;
1375 _osd_start_layer(sd, layer, win->fb_base_phys, 0);
1377 cfg->line_length = 0;
1378 cfg->xsize = 0;
1379 cfg->ysize = 0;
1380 cfg->xpos = 0;
1381 cfg->ypos = 0;
1382 cfg->interlaced = 0;
1383 switch (layer) {
1384 case WIN_OSD0:
1385 case WIN_OSD1:
1386 osdwin = (layer == WIN_OSD0) ? OSDWIN_OSD0 : OSDWIN_OSD1;
1387 osdwin_state = &osd->osdwin[osdwin];
1389 * Other code relies on the fact that OSD windows default to a
1390 * bitmap pixel format when they are deallocated, so don't
1391 * change this default pixel format.
1393 cfg->pixfmt = PIXFMT_8BPP;
1394 _osd_set_layer_config(sd, layer, cfg);
1395 osdwin_state->clut = RAM_CLUT;
1396 _osd_set_osd_clut(sd, osdwin, osdwin_state->clut);
1397 osdwin_state->colorkey_blending = 0;
1398 _osd_disable_color_key(sd, osdwin);
1399 osdwin_state->blend = OSD_8_VID_0;
1400 _osd_set_blending_factor(sd, osdwin, osdwin_state->blend);
1401 osdwin_state->rec601_attenuation = 0;
1402 _osd_set_rec601_attenuation(sd, osdwin,
1403 osdwin_state->
1404 rec601_attenuation);
1405 if (osdwin == OSDWIN_OSD1) {
1406 osd->is_blinking = 0;
1407 osd->blink = BLINK_X1;
1409 break;
1410 case WIN_VID0:
1411 case WIN_VID1:
1412 cfg->pixfmt = osd->yc_pixfmt;
1413 _osd_set_layer_config(sd, layer, cfg);
1414 break;
1417 spin_unlock_irqrestore(&osd->lock, flags);
1420 static void osd_release_layer(struct osd_state *sd, enum osd_layer layer)
1422 struct osd_state *osd = sd;
1423 struct osd_window_state *win = &osd->win[layer];
1424 unsigned long flags;
1426 spin_lock_irqsave(&osd->lock, flags);
1428 if (!win->is_allocated) {
1429 spin_unlock_irqrestore(&osd->lock, flags);
1430 return;
1433 spin_unlock_irqrestore(&osd->lock, flags);
1434 osd_init_layer(sd, layer);
1435 spin_lock_irqsave(&osd->lock, flags);
1437 win->is_allocated = 0;
1439 spin_unlock_irqrestore(&osd->lock, flags);
1442 static int osd_request_layer(struct osd_state *sd, enum osd_layer layer)
1444 struct osd_state *osd = sd;
1445 struct osd_window_state *win = &osd->win[layer];
1446 unsigned long flags;
1448 spin_lock_irqsave(&osd->lock, flags);
1450 if (win->is_allocated) {
1451 spin_unlock_irqrestore(&osd->lock, flags);
1452 return -1;
1454 win->is_allocated = 1;
1456 spin_unlock_irqrestore(&osd->lock, flags);
1458 return 0;
1461 static void _osd_init(struct osd_state *sd)
1463 osd_write(sd, 0, OSD_MODE);
1464 osd_write(sd, 0, OSD_VIDWINMD);
1465 osd_write(sd, 0, OSD_OSDWIN0MD);
1466 osd_write(sd, 0, OSD_OSDWIN1MD);
1467 osd_write(sd, 0, OSD_RECTCUR);
1468 osd_write(sd, 0, OSD_MISCCTL);
1469 if (sd->vpbe_type == VPBE_VERSION_3) {
1470 osd_write(sd, 0, OSD_VBNDRY);
1471 osd_write(sd, 0, OSD_EXTMODE);
1472 osd_write(sd, OSD_MISCCTL_DMANG, OSD_MISCCTL);
1476 static void osd_set_left_margin(struct osd_state *sd, u32 val)
1478 osd_write(sd, val, OSD_BASEPX);
1481 static void osd_set_top_margin(struct osd_state *sd, u32 val)
1483 osd_write(sd, val, OSD_BASEPY);
1486 static int osd_initialize(struct osd_state *osd)
1488 if (osd == NULL)
1489 return -ENODEV;
1490 _osd_init(osd);
1492 /* set default Cb/Cr order */
1493 osd->yc_pixfmt = PIXFMT_YCbCrI;
1495 if (osd->vpbe_type == VPBE_VERSION_3) {
1497 * ROM CLUT1 on the DM355 is similar (identical?) to ROM CLUT0
1498 * on the DM6446, so make ROM_CLUT1 the default on the DM355.
1500 osd->rom_clut = ROM_CLUT1;
1503 _osd_set_field_inversion(osd, osd->field_inversion);
1504 _osd_set_rom_clut(osd, osd->rom_clut);
1506 osd_init_layer(osd, WIN_OSD0);
1507 osd_init_layer(osd, WIN_VID0);
1508 osd_init_layer(osd, WIN_OSD1);
1509 osd_init_layer(osd, WIN_VID1);
1511 return 0;
1514 static const struct vpbe_osd_ops osd_ops = {
1515 .initialize = osd_initialize,
1516 .request_layer = osd_request_layer,
1517 .release_layer = osd_release_layer,
1518 .enable_layer = osd_enable_layer,
1519 .disable_layer = osd_disable_layer,
1520 .set_layer_config = osd_set_layer_config,
1521 .get_layer_config = osd_get_layer_config,
1522 .start_layer = osd_start_layer,
1523 .set_left_margin = osd_set_left_margin,
1524 .set_top_margin = osd_set_top_margin,
1527 static int osd_probe(struct platform_device *pdev)
1529 struct osd_platform_data *pdata;
1530 struct osd_state *osd;
1531 struct resource *res;
1532 int ret = 0;
1534 osd = kzalloc(sizeof(struct osd_state), GFP_KERNEL);
1535 if (osd == NULL)
1536 return -ENOMEM;
1538 osd->dev = &pdev->dev;
1539 pdata = (struct osd_platform_data *)pdev->dev.platform_data;
1540 osd->vpbe_type = (enum vpbe_version)pdata->vpbe_type;
1541 if (NULL == pdev->dev.platform_data) {
1542 dev_err(osd->dev, "No platform data defined for OSD"
1543 " sub device\n");
1544 ret = -ENOENT;
1545 goto free_mem;
1548 res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
1549 if (!res) {
1550 dev_err(osd->dev, "Unable to get OSD register address map\n");
1551 ret = -ENODEV;
1552 goto free_mem;
1554 osd->osd_base_phys = res->start;
1555 osd->osd_size = resource_size(res);
1556 if (!request_mem_region(osd->osd_base_phys, osd->osd_size,
1557 MODULE_NAME)) {
1558 dev_err(osd->dev, "Unable to reserve OSD MMIO region\n");
1559 ret = -ENODEV;
1560 goto free_mem;
1562 osd->osd_base = (unsigned long)ioremap_nocache(res->start,
1563 osd->osd_size);
1564 if (!osd->osd_base) {
1565 dev_err(osd->dev, "Unable to map the OSD region\n");
1566 ret = -ENODEV;
1567 goto release_mem_region;
1569 spin_lock_init(&osd->lock);
1570 osd->ops = osd_ops;
1571 platform_set_drvdata(pdev, osd);
1572 dev_notice(osd->dev, "OSD sub device probe success\n");
1573 return ret;
1575 release_mem_region:
1576 release_mem_region(osd->osd_base_phys, osd->osd_size);
1577 free_mem:
1578 kfree(osd);
1579 return ret;
1582 static int osd_remove(struct platform_device *pdev)
1584 struct osd_state *osd = platform_get_drvdata(pdev);
1586 iounmap((void *)osd->osd_base);
1587 release_mem_region(osd->osd_base_phys, osd->osd_size);
1588 kfree(osd);
1589 return 0;
1592 static struct platform_driver osd_driver = {
1593 .probe = osd_probe,
1594 .remove = osd_remove,
1595 .driver = {
1596 .name = MODULE_NAME,
1597 .owner = THIS_MODULE,
1601 module_platform_driver(osd_driver);
1603 MODULE_LICENSE("GPL");
1604 MODULE_DESCRIPTION("DaVinci OSD Manager Driver");
1605 MODULE_AUTHOR("Texas Instruments");