]> git.proxmox.com Git - mirror_ubuntu-artful-kernel.git/commitdiff
Merge tag 'fbdev-updates-for-3.7' of git://github.com/schandinat/linux-2.6
authorLinus Torvalds <torvalds@linux-foundation.org>
Fri, 12 Oct 2012 01:21:02 +0000 (10:21 +0900)
committerLinus Torvalds <torvalds@linux-foundation.org>
Fri, 12 Oct 2012 01:21:02 +0000 (10:21 +0900)
Pull fbdev updates from Florian Tobias Schandinat:
 "This includes:
   - large updates for OMAP
     - basic OMAP5 DSS support for DPI and DSI outputs
     - large cleanups and restructuring
   - some update to Exynos and da8xx-fb
   - removal of the pnx4008 driver (arch removed)
   - various other small patches"

Fix up some trivial conflicts (mostly just include line changes, but
also some due to the renaming of the deferred work functions by Tejun).

* tag 'fbdev-updates-for-3.7' of git://github.com/schandinat/linux-2.6: (193 commits)
  gbefb: fix compile error
  video: mark nuc900fb_map_video_memory as __devinit
  video/mx3fb: set .owner to prevent module unloading while being used
  video: exynos_dp: use clk_prepare_enable and clk_disable_unprepare
  drivers/video/exynos/exynos_mipi_dsi.c: fix error return code
  drivers/video/savage/savagefb_driver.c: fix error return code
  video: s3c-fb: use clk_prepare_enable and clk_disable_unprepare
  da8xx-fb: save and restore LCDC context across suspend/resume cycle
  da8xx-fb: add pm_runtime support
  video/udlfb: fix line counting in fb_write
  OMAPDSS: add missing include for string.h
  OMAPDSS: DISPC: Configure color conversion coefficients for writeback
  OMAPDSS: DISPC: Add manager like functions for writeback
  OMAPDSS: DISPC: Configure writeback FIFOs
  OMAPDSS: DISPC: Configure writeback specific parameters in dispc_wb_setup()
  OMAPDSS: DISPC: Configure overlay-like parameters in dispc_wb_setup
  OMAPDSS: DISPC: Add function to set channel in for writeback
  OMAPDSS: DISPC: Don't set chroma resampling bit for writeback
  OMAPDSS: DISPC: Downscale chroma if plane is writeback
  OMAPDSS: DISPC: Configure input and output sizes for writeback
  ...

46 files changed:
1  2 
arch/arm/mach-exynos/mach-nuri.c
arch/arm/mach-exynos/mach-origen.c
arch/arm/mach-exynos/mach-smdk4x12.c
arch/arm/mach-exynos/mach-smdkv310.c
arch/arm/mach-exynos/mach-universal_c210.c
arch/arm/mach-omap2/board-4430sdp.c
arch/arm/mach-omap2/board-omap4panda.c
arch/arm/mach-omap2/display.c
arch/arm/mach-omap2/twl-common.c
arch/arm/mach-s3c24xx/mach-smdk2416.c
arch/arm/mach-s3c64xx/mach-anw6410.c
arch/arm/mach-s3c64xx/mach-crag6410.c
arch/arm/mach-s3c64xx/mach-hmt.c
arch/arm/mach-s3c64xx/mach-mini6410.c
arch/arm/mach-s3c64xx/mach-ncp.c
arch/arm/mach-s3c64xx/mach-real6410.c
arch/arm/mach-s3c64xx/mach-smdk6410.c
arch/arm/mach-s5p64x0/mach-smdk6440.c
arch/arm/mach-s5p64x0/mach-smdk6450.c
arch/arm/mach-s5pc100/mach-smdkc100.c
arch/arm/mach-s5pv210/mach-goni.c
arch/arm/mach-s5pv210/mach-smdkv210.c
drivers/gpu/drm/exynos/exynos_drm_fimd.c
drivers/media/platform/omap/omap_vout.c
drivers/staging/omapdrm/omap_drv.c
drivers/video/Kconfig
drivers/video/Makefile
drivers/video/da8xx-fb.c
drivers/video/ep93xx-fb.c
drivers/video/exynos/exynos_mipi_dsi.c
drivers/video/gbefb.c
drivers/video/imxfb.c
drivers/video/jz4740_fb.c
drivers/video/msm/mddi.c
drivers/video/msm/mddi_client_nt35399.c
drivers/video/msm/mdp.c
drivers/video/msm/mdp_hw.h
drivers/video/mx3fb.c
drivers/video/nuc900fb.c
drivers/video/omap2/displays/panel-taal.c
drivers/video/omap2/dss/dispc.c
drivers/video/omap2/dss/dsi.c
drivers/video/omap2/omapfb/omapfb-main.c
drivers/video/ps3fb.c
drivers/video/udlfb.c
drivers/video/vermilion/vermilion.c

Simple merge
Simple merge
index 7a265d1a82d3e6e172799fe2dfc52d16b4217f7c,ee069b47ef262d9d68d279ff05d9098451d59a53..730f1ac65928ed832a52fdb2cf0e8b80f2ae4d60
  #include <plat/devs.h>
  #include <plat/fb.h>
  #include <plat/gpio-cfg.h>
 -#include <plat/iic.h>
 +#include <linux/platform_data/i2c-s3c2410.h>
  #include <plat/keypad.h>
  #include <plat/mfc.h>
- #include <plat/regs-fb.h>
  #include <plat/regs-serial.h>
  #include <plat/sdhci.h>
  
Simple merge
Simple merge
Simple merge
index d83ae4540b9bceefe58fb041dfdeae2eda6a3a54,9bf8a89e78c620a69578139f7d18c89963a9a741..635e109f5ad352e97e04dfad00c11e8728c72c98
@@@ -237,8 -251,17 +237,12 @@@ void __init omap3_pmic_get_config(struc
  
  #if defined(CONFIG_ARCH_OMAP4)
  static struct twl4030_usb_data omap4_usb_pdata = {
 -      .phy_init       = omap4430_phy_init,
 -      .phy_exit       = omap4430_phy_exit,
 -      .phy_power      = omap4430_phy_power,
 -      .phy_set_clock  = omap4430_phy_set_clk,
 -      .phy_suspend    = omap4430_phy_suspend,
  };
  
+ static struct regulator_consumer_supply omap4_vdda_hdmi_dac_supplies[] = {
+       REGULATOR_SUPPLY("vdda_hdmi_dac", "omapdss_hdmi"),
+ };
  static struct regulator_init_data omap4_vdac_idata = {
        .constraints = {
                .min_uV                 = 1800000,
index db2787aa1e5e7fb5ee61dff9d5c62a861d1d76aa,96478f05fc003096b5875a8429389077cffb6843..f30d7fccbfeedd4e2a22e00703b3933231b03157
  #include <plat/clock.h>
  #include <plat/devs.h>
  #include <plat/cpu.h>
 -#include <plat/nand.h>
 +#include <linux/platform_data/mtd-nand-s3c2410.h>
  #include <plat/sdhci.h>
 -#include <plat/udc.h>
 +#include <linux/platform_data/usb-s3c2410_udc.h>
  #include <linux/platform_data/s3c-hsudc.h>
  
- #include <plat/regs-fb-v4.h>
  #include <plat/fb.h>
  
  #include <plat/common-smdk.h>
index 15c58dfc4584f655b630b45d58798155551a4639,f7a108bafd4cf6928b466c080f4e1c2c79fc2149..99e82ac81b694ce69e27c7e5c8b3a53fc8cef473
@@@ -42,9 -43,8 +43,8 @@@
  #include <asm/mach-types.h>
  
  #include <plat/regs-serial.h>
 -#include <plat/iic.h>
 +#include <linux/platform_data/i2c-s3c2410.h>
  #include <plat/fb.h>
- #include <plat/regs-fb-v4.h>
  
  #include <plat/clock.h>
  #include <plat/devs.h>
Simple merge
Simple merge
index 09311cc40115d7dbf25c40dfe98bd45c5ea0a289,96ebcc5944c3860a45dabd065eeed85e1952cb5c..07c349cca33329fad68fbb253cfa5915e8783237
  #include <plat/cpu.h>
  #include <plat/devs.h>
  #include <plat/fb.h>
 -#include <plat/nand.h>
 +#include <linux/platform_data/mtd-nand-s3c2410.h>
  #include <plat/regs-serial.h>
 -#include <plat/ts.h>
 +#include <linux/platform_data/touchscreen-s3c2410.h>
- #include <plat/regs-fb-v4.h>
  
  #include <video/platform_lcd.h>
+ #include <video/samsung_fimd.h>
  
  #include "common.h"
  
Simple merge
index 6daca203e72b3eaea40d0792b0b8fd972918606d,76d3ffbe9a92f59fa9c6bbd1871e2aabdcde0652..7476f7c722ab18d86d7625caa0b5c6e7d41c3b1f
  #include <plat/cpu.h>
  #include <plat/devs.h>
  #include <plat/fb.h>
 -#include <plat/nand.h>
 +#include <linux/platform_data/mtd-nand-s3c2410.h>
  #include <plat/regs-serial.h>
 -#include <plat/ts.h>
 +#include <linux/platform_data/touchscreen-s3c2410.h>
- #include <plat/regs-fb-v4.h>
  
  #include <video/platform_lcd.h>
+ #include <video/samsung_fimd.h>
  
  #include "common.h"
  
index 2547a8846472180016b0bd24e42f4eb53b009b20,d1378d76e1510ec2861b3826cd4c1663ae79a360..da1a771a29e9f1493ce192d35fde34f6fa23ed01
  #include <plat/devs.h>
  #include <plat/cpu.h>
  #include <plat/adc.h>
 -#include <plat/ts.h>
 +#include <linux/platform_data/touchscreen-s3c2410.h>
  #include <plat/keypad.h>
  #include <plat/backlight.h>
- #include <plat/regs-fb-v4.h>
  
  #include "common.h"
  
Simple merge
Simple merge
index 5d2c0934928b8b85f63fc525247fad486ef8da2a,c3dcafa559b116ec4a400690910b623f697b329b..dba7384a87bde67d48256dd3e0a95cc61735be55
  #include <plat/devs.h>
  #include <plat/cpu.h>
  #include <plat/fb.h>
 -#include <plat/iic.h>
 -#include <plat/ata.h>
 +#include <linux/platform_data/i2c-s3c2410.h>
 +#include <linux/platform_data/ata-samsung_cf.h>
  #include <plat/adc.h>
  #include <plat/keypad.h>
 -#include <plat/ts.h>
 -#include <plat/audio.h>
 +#include <linux/platform_data/touchscreen-s3c2410.h>
 +#include <linux/platform_data/asoc-s3c.h>
  #include <plat/backlight.h>
- #include <plat/regs-fb-v4.h>
  
  #include "common.h"
  
Simple merge
Simple merge
index 134016f0e660dce4e76ad9cd8e13b184ea49d563,0000000000000000000000000000000000000000..a3b1a34c896db9257000f6c1a3e8fb8bb7cd7df0
mode 100644,000000..100644
--- /dev/null
@@@ -1,2291 -1,0 +1,2316 @@@
-               if (!ovl->manager || !ovl->manager->device)
 +/*
 + * omap_vout.c
 + *
 + * Copyright (C) 2005-2010 Texas Instruments.
 + *
 + * This file is licensed under the terms of the GNU General Public License
 + * version 2. This program is licensed "as is" without any warranty of any
 + * kind, whether express or implied.
 + *
 + * Leveraged code from the OMAP2 camera driver
 + * Video-for-Linux (Version 2) camera capture driver for
 + * the OMAP24xx camera controller.
 + *
 + * Author: Andy Lowe (source@mvista.com)
 + *
 + * Copyright (C) 2004 MontaVista Software, Inc.
 + * Copyright (C) 2010 Texas Instruments.
 + *
 + * History:
 + * 20-APR-2006 Khasim         Modified VRFB based Rotation,
 + *                            The image data is always read from 0 degree
 + *                            view and written
 + *                            to the virtual space of desired rotation angle
 + * 4-DEC-2006  Jian           Changed to support better memory management
 + *
 + * 17-Nov-2008 Hardik         Changed driver to use video_ioctl2
 + *
 + * 23-Feb-2010 Vaibhav H      Modified to use new DSS2 interface
 + *
 + */
 +
 +#include <linux/init.h>
 +#include <linux/module.h>
 +#include <linux/vmalloc.h>
 +#include <linux/sched.h>
 +#include <linux/types.h>
 +#include <linux/platform_device.h>
 +#include <linux/irq.h>
 +#include <linux/videodev2.h>
 +#include <linux/dma-mapping.h>
 +#include <linux/slab.h>
 +
 +#include <media/videobuf-dma-contig.h>
 +#include <media/v4l2-device.h>
 +#include <media/v4l2-ioctl.h>
 +
 +#include <plat/cpu.h>
 +#include <plat/dma.h>
 +#include <plat/vrfb.h>
 +#include <video/omapdss.h>
 +
 +#include "omap_voutlib.h"
 +#include "omap_voutdef.h"
 +#include "omap_vout_vrfb.h"
 +
 +MODULE_AUTHOR("Texas Instruments");
 +MODULE_DESCRIPTION("OMAP Video for Linux Video out driver");
 +MODULE_LICENSE("GPL");
 +
 +/* Driver Configuration macros */
 +#define VOUT_NAME             "omap_vout"
 +
 +enum omap_vout_channels {
 +      OMAP_VIDEO1,
 +      OMAP_VIDEO2,
 +};
 +
 +static struct videobuf_queue_ops video_vbq_ops;
 +/* Variables configurable through module params*/
 +static u32 video1_numbuffers = 3;
 +static u32 video2_numbuffers = 3;
 +static u32 video1_bufsize = OMAP_VOUT_MAX_BUF_SIZE;
 +static u32 video2_bufsize = OMAP_VOUT_MAX_BUF_SIZE;
 +static bool vid1_static_vrfb_alloc;
 +static bool vid2_static_vrfb_alloc;
 +static bool debug;
 +
 +/* Module parameters */
 +module_param(video1_numbuffers, uint, S_IRUGO);
 +MODULE_PARM_DESC(video1_numbuffers,
 +      "Number of buffers to be allocated at init time for Video1 device.");
 +
 +module_param(video2_numbuffers, uint, S_IRUGO);
 +MODULE_PARM_DESC(video2_numbuffers,
 +      "Number of buffers to be allocated at init time for Video2 device.");
 +
 +module_param(video1_bufsize, uint, S_IRUGO);
 +MODULE_PARM_DESC(video1_bufsize,
 +      "Size of the buffer to be allocated for video1 device");
 +
 +module_param(video2_bufsize, uint, S_IRUGO);
 +MODULE_PARM_DESC(video2_bufsize,
 +      "Size of the buffer to be allocated for video2 device");
 +
 +module_param(vid1_static_vrfb_alloc, bool, S_IRUGO);
 +MODULE_PARM_DESC(vid1_static_vrfb_alloc,
 +      "Static allocation of the VRFB buffer for video1 device");
 +
 +module_param(vid2_static_vrfb_alloc, bool, S_IRUGO);
 +MODULE_PARM_DESC(vid2_static_vrfb_alloc,
 +      "Static allocation of the VRFB buffer for video2 device");
 +
 +module_param(debug, bool, S_IRUGO);
 +MODULE_PARM_DESC(debug, "Debug level (0-1)");
 +
 +/* list of image formats supported by OMAP2 video pipelines */
 +static const struct v4l2_fmtdesc omap_formats[] = {
 +      {
 +              /* Note:  V4L2 defines RGB565 as:
 +               *
 +               *      Byte 0                    Byte 1
 +               *      g2 g1 g0 r4 r3 r2 r1 r0   b4 b3 b2 b1 b0 g5 g4 g3
 +               *
 +               * We interpret RGB565 as:
 +               *
 +               *      Byte 0                    Byte 1
 +               *      g2 g1 g0 b4 b3 b2 b1 b0   r4 r3 r2 r1 r0 g5 g4 g3
 +               */
 +              .description = "RGB565, le",
 +              .pixelformat = V4L2_PIX_FMT_RGB565,
 +      },
 +      {
 +              /* Note:  V4L2 defines RGB32 as: RGB-8-8-8-8  we use
 +               *  this for RGB24 unpack mode, the last 8 bits are ignored
 +               * */
 +              .description = "RGB32, le",
 +              .pixelformat = V4L2_PIX_FMT_RGB32,
 +      },
 +      {
 +              /* Note:  V4L2 defines RGB24 as: RGB-8-8-8  we use
 +               *        this for RGB24 packed mode
 +               *
 +               */
 +              .description = "RGB24, le",
 +              .pixelformat = V4L2_PIX_FMT_RGB24,
 +      },
 +      {
 +              .description = "YUYV (YUV 4:2:2), packed",
 +              .pixelformat = V4L2_PIX_FMT_YUYV,
 +      },
 +      {
 +              .description = "UYVY, packed",
 +              .pixelformat = V4L2_PIX_FMT_UYVY,
 +      },
 +};
 +
 +#define NUM_OUTPUT_FORMATS (ARRAY_SIZE(omap_formats))
 +
 +/*
 + * Try format
 + */
 +static int omap_vout_try_format(struct v4l2_pix_format *pix)
 +{
 +      int ifmt, bpp = 0;
 +
 +      pix->height = clamp(pix->height, (u32)VID_MIN_HEIGHT,
 +                                              (u32)VID_MAX_HEIGHT);
 +      pix->width = clamp(pix->width, (u32)VID_MIN_WIDTH, (u32)VID_MAX_WIDTH);
 +
 +      for (ifmt = 0; ifmt < NUM_OUTPUT_FORMATS; ifmt++) {
 +              if (pix->pixelformat == omap_formats[ifmt].pixelformat)
 +                      break;
 +      }
 +
 +      if (ifmt == NUM_OUTPUT_FORMATS)
 +              ifmt = 0;
 +
 +      pix->pixelformat = omap_formats[ifmt].pixelformat;
 +      pix->field = V4L2_FIELD_ANY;
 +      pix->priv = 0;
 +
 +      switch (pix->pixelformat) {
 +      case V4L2_PIX_FMT_YUYV:
 +      case V4L2_PIX_FMT_UYVY:
 +      default:
 +              pix->colorspace = V4L2_COLORSPACE_JPEG;
 +              bpp = YUYV_BPP;
 +              break;
 +      case V4L2_PIX_FMT_RGB565:
 +      case V4L2_PIX_FMT_RGB565X:
 +              pix->colorspace = V4L2_COLORSPACE_SRGB;
 +              bpp = RGB565_BPP;
 +              break;
 +      case V4L2_PIX_FMT_RGB24:
 +              pix->colorspace = V4L2_COLORSPACE_SRGB;
 +              bpp = RGB24_BPP;
 +              break;
 +      case V4L2_PIX_FMT_RGB32:
 +      case V4L2_PIX_FMT_BGR32:
 +              pix->colorspace = V4L2_COLORSPACE_SRGB;
 +              bpp = RGB32_BPP;
 +              break;
 +      }
 +      pix->bytesperline = pix->width * bpp;
 +      pix->sizeimage = pix->bytesperline * pix->height;
 +
 +      return bpp;
 +}
 +
 +/*
 + * omap_vout_uservirt_to_phys: This inline function is used to convert user
 + * space virtual address to physical address.
 + */
 +static u32 omap_vout_uservirt_to_phys(u32 virtp)
 +{
 +      unsigned long physp = 0;
 +      struct vm_area_struct *vma;
 +      struct mm_struct *mm = current->mm;
 +
 +      vma = find_vma(mm, virtp);
 +      /* For kernel direct-mapped memory, take the easy way */
 +      if (virtp >= PAGE_OFFSET) {
 +              physp = virt_to_phys((void *) virtp);
 +      } else if (vma && (vma->vm_flags & VM_IO) && vma->vm_pgoff) {
 +              /* this will catch, kernel-allocated, mmaped-to-usermode
 +                 addresses */
 +              physp = (vma->vm_pgoff << PAGE_SHIFT) + (virtp - vma->vm_start);
 +      } else {
 +              /* otherwise, use get_user_pages() for general userland pages */
 +              int res, nr_pages = 1;
 +              struct page *pages;
 +              down_read(&current->mm->mmap_sem);
 +
 +              res = get_user_pages(current, current->mm, virtp, nr_pages, 1,
 +                              0, &pages, NULL);
 +              up_read(&current->mm->mmap_sem);
 +
 +              if (res == nr_pages) {
 +                      physp =  __pa(page_address(&pages[0]) +
 +                                      (virtp & ~PAGE_MASK));
 +              } else {
 +                      printk(KERN_WARNING VOUT_NAME
 +                                      "get_user_pages failed\n");
 +                      return 0;
 +              }
 +      }
 +
 +      return physp;
 +}
 +
 +/*
 + * Free the V4L2 buffers
 + */
 +void omap_vout_free_buffers(struct omap_vout_device *vout)
 +{
 +      int i, numbuffers;
 +
 +      /* Allocate memory for the buffers */
 +      numbuffers = (vout->vid) ?  video2_numbuffers : video1_numbuffers;
 +      vout->buffer_size = (vout->vid) ? video2_bufsize : video1_bufsize;
 +
 +      for (i = 0; i < numbuffers; i++) {
 +              omap_vout_free_buffer(vout->buf_virt_addr[i],
 +                              vout->buffer_size);
 +              vout->buf_phy_addr[i] = 0;
 +              vout->buf_virt_addr[i] = 0;
 +      }
 +}
 +
 +/*
 + * Convert V4L2 rotation to DSS rotation
 + *    V4L2 understand 0, 90, 180, 270.
 + *    Convert to 0, 1, 2 and 3 respectively for DSS
 + */
 +static int v4l2_rot_to_dss_rot(int v4l2_rotation,
 +                      enum dss_rotation *rotation, bool mirror)
 +{
 +      int ret = 0;
 +
 +      switch (v4l2_rotation) {
 +      case 90:
 +              *rotation = dss_rotation_90_degree;
 +              break;
 +      case 180:
 +              *rotation = dss_rotation_180_degree;
 +              break;
 +      case 270:
 +              *rotation = dss_rotation_270_degree;
 +              break;
 +      case 0:
 +              *rotation = dss_rotation_0_degree;
 +              break;
 +      default:
 +              ret = -EINVAL;
 +      }
 +      return ret;
 +}
 +
 +static int omap_vout_calculate_offset(struct omap_vout_device *vout)
 +{
 +      struct omapvideo_info *ovid;
 +      struct v4l2_rect *crop = &vout->crop;
 +      struct v4l2_pix_format *pix = &vout->pix;
 +      int *cropped_offset = &vout->cropped_offset;
 +      int ps = 2, line_length = 0;
 +
 +      ovid = &vout->vid_info;
 +
 +      if (ovid->rotation_type == VOUT_ROT_VRFB) {
 +              omap_vout_calculate_vrfb_offset(vout);
 +      } else {
 +              vout->line_length = line_length = pix->width;
 +
 +              if (V4L2_PIX_FMT_YUYV == pix->pixelformat ||
 +                      V4L2_PIX_FMT_UYVY == pix->pixelformat)
 +                      ps = 2;
 +              else if (V4L2_PIX_FMT_RGB32 == pix->pixelformat)
 +                      ps = 4;
 +              else if (V4L2_PIX_FMT_RGB24 == pix->pixelformat)
 +                      ps = 3;
 +
 +              vout->ps = ps;
 +
 +              *cropped_offset = (line_length * ps) *
 +                      crop->top + crop->left * ps;
 +      }
 +
 +      v4l2_dbg(1, debug, &vout->vid_dev->v4l2_dev, "%s Offset:%x\n",
 +                      __func__, vout->cropped_offset);
 +
 +      return 0;
 +}
 +
 +/*
 + * Convert V4L2 pixel format to DSS pixel format
 + */
 +static int video_mode_to_dss_mode(struct omap_vout_device *vout)
 +{
 +      struct omap_overlay *ovl;
 +      struct omapvideo_info *ovid;
 +      struct v4l2_pix_format *pix = &vout->pix;
 +      enum omap_color_mode mode;
 +
 +      ovid = &vout->vid_info;
 +      ovl = ovid->overlays[0];
 +
 +      switch (pix->pixelformat) {
 +      case 0:
 +              break;
 +      case V4L2_PIX_FMT_YUYV:
 +              mode = OMAP_DSS_COLOR_YUV2;
 +              break;
 +      case V4L2_PIX_FMT_UYVY:
 +              mode = OMAP_DSS_COLOR_UYVY;
 +              break;
 +      case V4L2_PIX_FMT_RGB565:
 +              mode = OMAP_DSS_COLOR_RGB16;
 +              break;
 +      case V4L2_PIX_FMT_RGB24:
 +              mode = OMAP_DSS_COLOR_RGB24P;
 +              break;
 +      case V4L2_PIX_FMT_RGB32:
 +              mode = (ovl->id == OMAP_DSS_VIDEO1) ?
 +                      OMAP_DSS_COLOR_RGB24U : OMAP_DSS_COLOR_ARGB32;
 +              break;
 +      case V4L2_PIX_FMT_BGR32:
 +              mode = OMAP_DSS_COLOR_RGBX32;
 +              break;
 +      default:
 +              mode = -EINVAL;
 +      }
 +      return mode;
 +}
 +
 +/*
 + * Setup the overlay
 + */
 +static int omapvid_setup_overlay(struct omap_vout_device *vout,
 +              struct omap_overlay *ovl, int posx, int posy, int outw,
 +              int outh, u32 addr)
 +{
 +      int ret = 0;
 +      struct omap_overlay_info info;
 +      int cropheight, cropwidth, pixheight, pixwidth;
 +
 +      if ((ovl->caps & OMAP_DSS_OVL_CAP_SCALE) == 0 &&
 +                      (outw != vout->pix.width || outh != vout->pix.height)) {
 +              ret = -EINVAL;
 +              goto setup_ovl_err;
 +      }
 +
 +      vout->dss_mode = video_mode_to_dss_mode(vout);
 +      if (vout->dss_mode == -EINVAL) {
 +              ret = -EINVAL;
 +              goto setup_ovl_err;
 +      }
 +
 +      /* Setup the input plane parameters according to
 +       * rotation value selected.
 +       */
 +      if (is_rotation_90_or_270(vout)) {
 +              cropheight = vout->crop.width;
 +              cropwidth = vout->crop.height;
 +              pixheight = vout->pix.width;
 +              pixwidth = vout->pix.height;
 +      } else {
 +              cropheight = vout->crop.height;
 +              cropwidth = vout->crop.width;
 +              pixheight = vout->pix.height;
 +              pixwidth = vout->pix.width;
 +      }
 +
 +      ovl->get_overlay_info(ovl, &info);
 +      info.paddr = addr;
 +      info.width = cropwidth;
 +      info.height = cropheight;
 +      info.color_mode = vout->dss_mode;
 +      info.mirror = vout->mirror;
 +      info.pos_x = posx;
 +      info.pos_y = posy;
 +      info.out_width = outw;
 +      info.out_height = outh;
 +      info.global_alpha = vout->win.global_alpha;
 +      if (!is_rotation_enabled(vout)) {
 +              info.rotation = 0;
 +              info.rotation_type = OMAP_DSS_ROT_DMA;
 +              info.screen_width = pixwidth;
 +      } else {
 +              info.rotation = vout->rotation;
 +              info.rotation_type = OMAP_DSS_ROT_VRFB;
 +              info.screen_width = 2048;
 +      }
 +
 +      v4l2_dbg(1, debug, &vout->vid_dev->v4l2_dev,
 +              "%s enable=%d addr=%x width=%d\n height=%d color_mode=%d\n"
 +              "rotation=%d mirror=%d posx=%d posy=%d out_width = %d \n"
 +              "out_height=%d rotation_type=%d screen_width=%d\n",
 +              __func__, ovl->is_enabled(ovl), info.paddr, info.width, info.height,
 +              info.color_mode, info.rotation, info.mirror, info.pos_x,
 +              info.pos_y, info.out_width, info.out_height, info.rotation_type,
 +              info.screen_width);
 +
 +      ret = ovl->set_overlay_info(ovl, &info);
 +      if (ret)
 +              goto setup_ovl_err;
 +
 +      return 0;
 +
 +setup_ovl_err:
 +      v4l2_warn(&vout->vid_dev->v4l2_dev, "setup_overlay failed\n");
 +      return ret;
 +}
 +
 +/*
 + * Initialize the overlay structure
 + */
 +static int omapvid_init(struct omap_vout_device *vout, u32 addr)
 +{
 +      int ret = 0, i;
 +      struct v4l2_window *win;
 +      struct omap_overlay *ovl;
 +      int posx, posy, outw, outh, temp;
 +      struct omap_video_timings *timing;
 +      struct omapvideo_info *ovid = &vout->vid_info;
 +
 +      win = &vout->win;
 +      for (i = 0; i < ovid->num_overlays; i++) {
++              struct omap_dss_device *dssdev;
++
 +              ovl = ovid->overlays[i];
-               timing = &ovl->manager->device->panel.timings;
++              dssdev = ovl->get_device(ovl);
++
++              if (!dssdev)
 +                      return -EINVAL;
 +
-               if (!ovl->manager || !ovl->manager->device)
++              timing = &dssdev->panel.timings;
 +
 +              outw = win->w.width;
 +              outh = win->w.height;
 +              switch (vout->rotation) {
 +              case dss_rotation_90_degree:
 +                      /* Invert the height and width for 90
 +                       * and 270 degree rotation
 +                       */
 +                      temp = outw;
 +                      outw = outh;
 +                      outh = temp;
 +                      posy = (timing->y_res - win->w.width) - win->w.left;
 +                      posx = win->w.top;
 +                      break;
 +
 +              case dss_rotation_180_degree:
 +                      posx = (timing->x_res - win->w.width) - win->w.left;
 +                      posy = (timing->y_res - win->w.height) - win->w.top;
 +                      break;
 +
 +              case dss_rotation_270_degree:
 +                      temp = outw;
 +                      outw = outh;
 +                      outh = temp;
 +                      posy = win->w.left;
 +                      posx = (timing->x_res - win->w.height) - win->w.top;
 +                      break;
 +
 +              default:
 +                      posx = win->w.left;
 +                      posy = win->w.top;
 +                      break;
 +              }
 +
 +              ret = omapvid_setup_overlay(vout, ovl, posx, posy,
 +                              outw, outh, addr);
 +              if (ret)
 +                      goto omapvid_init_err;
 +      }
 +      return 0;
 +
 +omapvid_init_err:
 +      v4l2_warn(&vout->vid_dev->v4l2_dev, "apply_changes failed\n");
 +      return ret;
 +}
 +
 +/*
 + * Apply the changes set the go bit of DSS
 + */
 +static int omapvid_apply_changes(struct omap_vout_device *vout)
 +{
 +      int i;
 +      struct omap_overlay *ovl;
 +      struct omapvideo_info *ovid = &vout->vid_info;
 +
 +      for (i = 0; i < ovid->num_overlays; i++) {
++              struct omap_dss_device *dssdev;
++
 +              ovl = ovid->overlays[i];
-       /* get the display device attached to the overlay */
-       if (!ovl->manager || !ovl->manager->device)
-               return;
++              dssdev = ovl->get_device(ovl);
++              if (!dssdev)
 +                      return -EINVAL;
 +              ovl->manager->apply(ovl->manager);
 +      }
 +
 +      return 0;
 +}
 +
 +static int omapvid_handle_interlace_display(struct omap_vout_device *vout,
 +              unsigned int irqstatus, struct timeval timevalue)
 +{
 +      u32 fid;
 +
 +      if (vout->first_int) {
 +              vout->first_int = 0;
 +              goto err;
 +      }
 +
 +      if (irqstatus & DISPC_IRQ_EVSYNC_ODD)
 +              fid = 1;
 +      else if (irqstatus & DISPC_IRQ_EVSYNC_EVEN)
 +              fid = 0;
 +      else
 +              goto err;
 +
 +      vout->field_id ^= 1;
 +      if (fid != vout->field_id) {
 +              if (fid == 0)
 +                      vout->field_id = fid;
 +      } else if (0 == fid) {
 +              if (vout->cur_frm == vout->next_frm)
 +                      goto err;
 +
 +              vout->cur_frm->ts = timevalue;
 +              vout->cur_frm->state = VIDEOBUF_DONE;
 +              wake_up_interruptible(&vout->cur_frm->done);
 +              vout->cur_frm = vout->next_frm;
 +      } else {
 +              if (list_empty(&vout->dma_queue) ||
 +                              (vout->cur_frm != vout->next_frm))
 +                      goto err;
 +      }
 +
 +      return vout->field_id;
 +err:
 +      return 0;
 +}
 +
 +static void omap_vout_isr(void *arg, unsigned int irqstatus)
 +{
 +      int ret, fid, mgr_id;
 +      u32 addr, irq;
 +      struct omap_overlay *ovl;
 +      struct timeval timevalue;
 +      struct omapvideo_info *ovid;
 +      struct omap_dss_device *cur_display;
 +      struct omap_vout_device *vout = (struct omap_vout_device *)arg;
 +
 +      if (!vout->streaming)
 +              return;
 +
 +      ovid = &vout->vid_info;
 +      ovl = ovid->overlays[0];
-       cur_display = ovl->manager->device;
 +
 +      mgr_id = ovl->manager->id;
-               if (ovl->manager && ovl->manager->device)
++
++      /* get the display device attached to the overlay */
++      cur_display = ovl->get_device(ovl);
++
++      if (!cur_display)
++              return;
 +
 +      spin_lock(&vout->vbq_lock);
 +      do_gettimeofday(&timevalue);
 +
 +      switch (cur_display->type) {
 +      case OMAP_DISPLAY_TYPE_DSI:
 +      case OMAP_DISPLAY_TYPE_DPI:
 +              if (mgr_id == OMAP_DSS_CHANNEL_LCD)
 +                      irq = DISPC_IRQ_VSYNC;
 +              else if (mgr_id == OMAP_DSS_CHANNEL_LCD2)
 +                      irq = DISPC_IRQ_VSYNC2;
 +              else
 +                      goto vout_isr_err;
 +
 +              if (!(irqstatus & irq))
 +                      goto vout_isr_err;
 +              break;
 +      case OMAP_DISPLAY_TYPE_VENC:
 +              fid = omapvid_handle_interlace_display(vout, irqstatus,
 +                              timevalue);
 +              if (!fid)
 +                      goto vout_isr_err;
 +              break;
 +      case OMAP_DISPLAY_TYPE_HDMI:
 +              if (!(irqstatus & DISPC_IRQ_EVSYNC_EVEN))
 +                      goto vout_isr_err;
 +              break;
 +      default:
 +              goto vout_isr_err;
 +      }
 +
 +      if (!vout->first_int && (vout->cur_frm != vout->next_frm)) {
 +              vout->cur_frm->ts = timevalue;
 +              vout->cur_frm->state = VIDEOBUF_DONE;
 +              wake_up_interruptible(&vout->cur_frm->done);
 +              vout->cur_frm = vout->next_frm;
 +      }
 +
 +      vout->first_int = 0;
 +      if (list_empty(&vout->dma_queue))
 +              goto vout_isr_err;
 +
 +      vout->next_frm = list_entry(vout->dma_queue.next,
 +                      struct videobuf_buffer, queue);
 +      list_del(&vout->next_frm->queue);
 +
 +      vout->next_frm->state = VIDEOBUF_ACTIVE;
 +
 +      addr = (unsigned long) vout->queued_buf_addr[vout->next_frm->i]
 +              + vout->cropped_offset;
 +
 +      /* First save the configuration in ovelray structure */
 +      ret = omapvid_init(vout, addr);
 +      if (ret)
 +              printk(KERN_ERR VOUT_NAME
 +                      "failed to set overlay info\n");
 +      /* Enable the pipeline and set the Go bit */
 +      ret = omapvid_apply_changes(vout);
 +      if (ret)
 +              printk(KERN_ERR VOUT_NAME "failed to change mode\n");
 +
 +vout_isr_err:
 +      spin_unlock(&vout->vbq_lock);
 +}
 +
 +/* Video buffer call backs */
 +
 +/*
 + * Buffer setup function is called by videobuf layer when REQBUF ioctl is
 + * called. This is used to setup buffers and return size and count of
 + * buffers allocated. After the call to this buffer, videobuf layer will
 + * setup buffer queue depending on the size and count of buffers
 + */
 +static int omap_vout_buffer_setup(struct videobuf_queue *q, unsigned int *count,
 +                        unsigned int *size)
 +{
 +      int startindex = 0, i, j;
 +      u32 phy_addr = 0, virt_addr = 0;
 +      struct omap_vout_device *vout = q->priv_data;
 +      struct omapvideo_info *ovid = &vout->vid_info;
 +      int vid_max_buf_size;
 +
 +      if (!vout)
 +              return -EINVAL;
 +
 +      vid_max_buf_size = vout->vid == OMAP_VIDEO1 ? video1_bufsize :
 +              video2_bufsize;
 +
 +      if (V4L2_BUF_TYPE_VIDEO_OUTPUT != q->type)
 +              return -EINVAL;
 +
 +      startindex = (vout->vid == OMAP_VIDEO1) ?
 +              video1_numbuffers : video2_numbuffers;
 +      if (V4L2_MEMORY_MMAP == vout->memory && *count < startindex)
 +              *count = startindex;
 +
 +      if (ovid->rotation_type == VOUT_ROT_VRFB) {
 +              if (omap_vout_vrfb_buffer_setup(vout, count, startindex))
 +                      return -ENOMEM;
 +      }
 +
 +      if (V4L2_MEMORY_MMAP != vout->memory)
 +              return 0;
 +
 +      /* Now allocated the V4L2 buffers */
 +      *size = PAGE_ALIGN(vout->pix.width * vout->pix.height * vout->bpp);
 +      startindex = (vout->vid == OMAP_VIDEO1) ?
 +              video1_numbuffers : video2_numbuffers;
 +
 +      /* Check the size of the buffer */
 +      if (*size > vid_max_buf_size) {
 +              v4l2_err(&vout->vid_dev->v4l2_dev,
 +                              "buffer allocation mismatch [%u] [%u]\n",
 +                              *size, vout->buffer_size);
 +              return -ENOMEM;
 +      }
 +
 +      for (i = startindex; i < *count; i++) {
 +              vout->buffer_size = *size;
 +
 +              virt_addr = omap_vout_alloc_buffer(vout->buffer_size,
 +                              &phy_addr);
 +              if (!virt_addr) {
 +                      if (ovid->rotation_type == VOUT_ROT_NONE) {
 +                              break;
 +                      } else {
 +                              if (!is_rotation_enabled(vout))
 +                                      break;
 +                      /* Free the VRFB buffers if no space for V4L2 buffers */
 +                      for (j = i; j < *count; j++) {
 +                              omap_vout_free_buffer(
 +                                              vout->smsshado_virt_addr[j],
 +                                              vout->smsshado_size);
 +                              vout->smsshado_virt_addr[j] = 0;
 +                              vout->smsshado_phy_addr[j] = 0;
 +                              }
 +                      }
 +              }
 +              vout->buf_virt_addr[i] = virt_addr;
 +              vout->buf_phy_addr[i] = phy_addr;
 +      }
 +      *count = vout->buffer_allocated = i;
 +
 +      return 0;
 +}
 +
 +/*
 + * Free the V4L2 buffers additionally allocated than default
 + * number of buffers
 + */
 +static void omap_vout_free_extra_buffers(struct omap_vout_device *vout)
 +{
 +      int num_buffers = 0, i;
 +
 +      num_buffers = (vout->vid == OMAP_VIDEO1) ?
 +              video1_numbuffers : video2_numbuffers;
 +
 +      for (i = num_buffers; i < vout->buffer_allocated; i++) {
 +              if (vout->buf_virt_addr[i])
 +                      omap_vout_free_buffer(vout->buf_virt_addr[i],
 +                                      vout->buffer_size);
 +
 +              vout->buf_virt_addr[i] = 0;
 +              vout->buf_phy_addr[i] = 0;
 +      }
 +      vout->buffer_allocated = num_buffers;
 +}
 +
 +/*
 + * This function will be called when VIDIOC_QBUF ioctl is called.
 + * It prepare buffers before give out for the display. This function
 + * converts user space virtual address into physical address if userptr memory
 + * exchange mechanism is used. If rotation is enabled, it copies entire
 + * buffer into VRFB memory space before giving it to the DSS.
 + */
 +static int omap_vout_buffer_prepare(struct videobuf_queue *q,
 +                      struct videobuf_buffer *vb,
 +                      enum v4l2_field field)
 +{
 +      struct omap_vout_device *vout = q->priv_data;
 +      struct omapvideo_info *ovid = &vout->vid_info;
 +
 +      if (VIDEOBUF_NEEDS_INIT == vb->state) {
 +              vb->width = vout->pix.width;
 +              vb->height = vout->pix.height;
 +              vb->size = vb->width * vb->height * vout->bpp;
 +              vb->field = field;
 +      }
 +      vb->state = VIDEOBUF_PREPARED;
 +      /* if user pointer memory mechanism is used, get the physical
 +       * address of the buffer
 +       */
 +      if (V4L2_MEMORY_USERPTR == vb->memory) {
 +              if (0 == vb->baddr)
 +                      return -EINVAL;
 +              /* Physical address */
 +              vout->queued_buf_addr[vb->i] = (u8 *)
 +                      omap_vout_uservirt_to_phys(vb->baddr);
 +      } else {
 +              u32 addr, dma_addr;
 +              unsigned long size;
 +
 +              addr = (unsigned long) vout->buf_virt_addr[vb->i];
 +              size = (unsigned long) vb->size;
 +
 +              dma_addr = dma_map_single(vout->vid_dev->v4l2_dev.dev, (void *) addr,
 +                              size, DMA_TO_DEVICE);
 +              if (dma_mapping_error(vout->vid_dev->v4l2_dev.dev, dma_addr))
 +                      v4l2_err(&vout->vid_dev->v4l2_dev, "dma_map_single failed\n");
 +
 +              vout->queued_buf_addr[vb->i] = (u8 *)vout->buf_phy_addr[vb->i];
 +      }
 +
 +      if (ovid->rotation_type == VOUT_ROT_VRFB)
 +              return omap_vout_prepare_vrfb(vout, vb);
 +      else
 +              return 0;
 +}
 +
 +/*
 + * Buffer queue function will be called from the videobuf layer when _QBUF
 + * ioctl is called. It is used to enqueue buffer, which is ready to be
 + * displayed.
 + */
 +static void omap_vout_buffer_queue(struct videobuf_queue *q,
 +                        struct videobuf_buffer *vb)
 +{
 +      struct omap_vout_device *vout = q->priv_data;
 +
 +      /* Driver is also maintainig a queue. So enqueue buffer in the driver
 +       * queue */
 +      list_add_tail(&vb->queue, &vout->dma_queue);
 +
 +      vb->state = VIDEOBUF_QUEUED;
 +}
 +
 +/*
 + * Buffer release function is called from videobuf layer to release buffer
 + * which are already allocated
 + */
 +static void omap_vout_buffer_release(struct videobuf_queue *q,
 +                          struct videobuf_buffer *vb)
 +{
 +      struct omap_vout_device *vout = q->priv_data;
 +
 +      vb->state = VIDEOBUF_NEEDS_INIT;
 +
 +      if (V4L2_MEMORY_MMAP != vout->memory)
 +              return;
 +}
 +
 +/*
 + *  File operations
 + */
 +static unsigned int omap_vout_poll(struct file *file,
 +                                 struct poll_table_struct *wait)
 +{
 +      struct omap_vout_device *vout = file->private_data;
 +      struct videobuf_queue *q = &vout->vbq;
 +
 +      return videobuf_poll_stream(file, q, wait);
 +}
 +
 +static void omap_vout_vm_open(struct vm_area_struct *vma)
 +{
 +      struct omap_vout_device *vout = vma->vm_private_data;
 +
 +      v4l2_dbg(1, debug, &vout->vid_dev->v4l2_dev,
 +              "vm_open [vma=%08lx-%08lx]\n", vma->vm_start, vma->vm_end);
 +      vout->mmap_count++;
 +}
 +
 +static void omap_vout_vm_close(struct vm_area_struct *vma)
 +{
 +      struct omap_vout_device *vout = vma->vm_private_data;
 +
 +      v4l2_dbg(1, debug, &vout->vid_dev->v4l2_dev,
 +              "vm_close [vma=%08lx-%08lx]\n", vma->vm_start, vma->vm_end);
 +      vout->mmap_count--;
 +}
 +
 +static struct vm_operations_struct omap_vout_vm_ops = {
 +      .open   = omap_vout_vm_open,
 +      .close  = omap_vout_vm_close,
 +};
 +
 +static int omap_vout_mmap(struct file *file, struct vm_area_struct *vma)
 +{
 +      int i;
 +      void *pos;
 +      unsigned long start = vma->vm_start;
 +      unsigned long size = (vma->vm_end - vma->vm_start);
 +      struct omap_vout_device *vout = file->private_data;
 +      struct videobuf_queue *q = &vout->vbq;
 +
 +      v4l2_dbg(1, debug, &vout->vid_dev->v4l2_dev,
 +                      " %s pgoff=0x%lx, start=0x%lx, end=0x%lx\n", __func__,
 +                      vma->vm_pgoff, vma->vm_start, vma->vm_end);
 +
 +      /* look for the buffer to map */
 +      for (i = 0; i < VIDEO_MAX_FRAME; i++) {
 +              if (NULL == q->bufs[i])
 +                      continue;
 +              if (V4L2_MEMORY_MMAP != q->bufs[i]->memory)
 +                      continue;
 +              if (q->bufs[i]->boff == (vma->vm_pgoff << PAGE_SHIFT))
 +                      break;
 +      }
 +
 +      if (VIDEO_MAX_FRAME == i) {
 +              v4l2_dbg(1, debug, &vout->vid_dev->v4l2_dev,
 +                              "offset invalid [offset=0x%lx]\n",
 +                              (vma->vm_pgoff << PAGE_SHIFT));
 +              return -EINVAL;
 +      }
 +      /* Check the size of the buffer */
 +      if (size > vout->buffer_size) {
 +              v4l2_err(&vout->vid_dev->v4l2_dev,
 +                              "insufficient memory [%lu] [%u]\n",
 +                              size, vout->buffer_size);
 +              return -ENOMEM;
 +      }
 +
 +      q->bufs[i]->baddr = vma->vm_start;
 +
 +      vma->vm_flags |= VM_DONTEXPAND | VM_DONTDUMP;
 +      vma->vm_page_prot = pgprot_writecombine(vma->vm_page_prot);
 +      vma->vm_ops = &omap_vout_vm_ops;
 +      vma->vm_private_data = (void *) vout;
 +      pos = (void *)vout->buf_virt_addr[i];
 +      vma->vm_pgoff = virt_to_phys((void *)pos) >> PAGE_SHIFT;
 +      while (size > 0) {
 +              unsigned long pfn;
 +              pfn = virt_to_phys((void *) pos) >> PAGE_SHIFT;
 +              if (remap_pfn_range(vma, start, pfn, PAGE_SIZE, PAGE_SHARED))
 +                      return -EAGAIN;
 +              start += PAGE_SIZE;
 +              pos += PAGE_SIZE;
 +              size -= PAGE_SIZE;
 +      }
 +      vout->mmap_count++;
 +      v4l2_dbg(1, debug, &vout->vid_dev->v4l2_dev, "Exiting %s\n", __func__);
 +
 +      return 0;
 +}
 +
 +static int omap_vout_release(struct file *file)
 +{
 +      unsigned int ret, i;
 +      struct videobuf_queue *q;
 +      struct omapvideo_info *ovid;
 +      struct omap_vout_device *vout = file->private_data;
 +
 +      v4l2_dbg(1, debug, &vout->vid_dev->v4l2_dev, "Entering %s\n", __func__);
 +      ovid = &vout->vid_info;
 +
 +      if (!vout)
 +              return 0;
 +
 +      q = &vout->vbq;
 +      /* Disable all the overlay managers connected with this interface */
 +      for (i = 0; i < ovid->num_overlays; i++) {
 +              struct omap_overlay *ovl = ovid->overlays[i];
-       if (!ovl->manager || !ovl->manager->device)
++              struct omap_dss_device *dssdev = ovl->get_device(ovl);
++
++              if (dssdev)
 +                      ovl->disable(ovl);
 +      }
 +      /* Turn off the pipeline */
 +      ret = omapvid_apply_changes(vout);
 +      if (ret)
 +              v4l2_warn(&vout->vid_dev->v4l2_dev,
 +                              "Unable to apply changes\n");
 +
 +      /* Free all buffers */
 +      omap_vout_free_extra_buffers(vout);
 +
 +      /* Free the VRFB buffers only if they are allocated
 +       * during reqbufs.  Don't free if init time allocated
 +       */
 +      if (ovid->rotation_type == VOUT_ROT_VRFB) {
 +              if (!vout->vrfb_static_allocation)
 +                      omap_vout_free_vrfb_buffers(vout);
 +      }
 +      videobuf_mmap_free(q);
 +
 +      /* Even if apply changes fails we should continue
 +         freeing allocated memory */
 +      if (vout->streaming) {
 +              u32 mask = 0;
 +
 +              mask = DISPC_IRQ_VSYNC | DISPC_IRQ_EVSYNC_EVEN |
 +                      DISPC_IRQ_EVSYNC_ODD | DISPC_IRQ_VSYNC2;
 +              omap_dispc_unregister_isr(omap_vout_isr, vout, mask);
 +              vout->streaming = 0;
 +
 +              videobuf_streamoff(q);
 +              videobuf_queue_cancel(q);
 +      }
 +
 +      if (vout->mmap_count != 0)
 +              vout->mmap_count = 0;
 +
 +      vout->opened -= 1;
 +      file->private_data = NULL;
 +
 +      if (vout->buffer_allocated)
 +              videobuf_mmap_free(q);
 +
 +      v4l2_dbg(1, debug, &vout->vid_dev->v4l2_dev, "Exiting %s\n", __func__);
 +      return ret;
 +}
 +
 +static int omap_vout_open(struct file *file)
 +{
 +      struct videobuf_queue *q;
 +      struct omap_vout_device *vout = NULL;
 +
 +      vout = video_drvdata(file);
 +      v4l2_dbg(1, debug, &vout->vid_dev->v4l2_dev, "Entering %s\n", __func__);
 +
 +      if (vout == NULL)
 +              return -ENODEV;
 +
 +      /* for now, we only support single open */
 +      if (vout->opened)
 +              return -EBUSY;
 +
 +      vout->opened += 1;
 +
 +      file->private_data = vout;
 +      vout->type = V4L2_BUF_TYPE_VIDEO_OUTPUT;
 +
 +      q = &vout->vbq;
 +      video_vbq_ops.buf_setup = omap_vout_buffer_setup;
 +      video_vbq_ops.buf_prepare = omap_vout_buffer_prepare;
 +      video_vbq_ops.buf_release = omap_vout_buffer_release;
 +      video_vbq_ops.buf_queue = omap_vout_buffer_queue;
 +      spin_lock_init(&vout->vbq_lock);
 +
 +      videobuf_queue_dma_contig_init(q, &video_vbq_ops, q->dev,
 +                      &vout->vbq_lock, vout->type, V4L2_FIELD_NONE,
 +                      sizeof(struct videobuf_buffer), vout, NULL);
 +
 +      v4l2_dbg(1, debug, &vout->vid_dev->v4l2_dev, "Exiting %s\n", __func__);
 +      return 0;
 +}
 +
 +/*
 + * V4L2 ioctls
 + */
 +static int vidioc_querycap(struct file *file, void *fh,
 +              struct v4l2_capability *cap)
 +{
 +      struct omap_vout_device *vout = fh;
 +
 +      strlcpy(cap->driver, VOUT_NAME, sizeof(cap->driver));
 +      strlcpy(cap->card, vout->vfd->name, sizeof(cap->card));
 +      cap->bus_info[0] = '\0';
 +      cap->capabilities = V4L2_CAP_STREAMING | V4L2_CAP_VIDEO_OUTPUT |
 +              V4L2_CAP_VIDEO_OUTPUT_OVERLAY;
 +
 +      return 0;
 +}
 +
 +static int vidioc_enum_fmt_vid_out(struct file *file, void *fh,
 +                      struct v4l2_fmtdesc *fmt)
 +{
 +      int index = fmt->index;
 +
 +      if (index >= NUM_OUTPUT_FORMATS)
 +              return -EINVAL;
 +
 +      fmt->flags = omap_formats[index].flags;
 +      strlcpy(fmt->description, omap_formats[index].description,
 +                      sizeof(fmt->description));
 +      fmt->pixelformat = omap_formats[index].pixelformat;
 +
 +      return 0;
 +}
 +
 +static int vidioc_g_fmt_vid_out(struct file *file, void *fh,
 +                      struct v4l2_format *f)
 +{
 +      struct omap_vout_device *vout = fh;
 +
 +      f->fmt.pix = vout->pix;
 +      return 0;
 +
 +}
 +
 +static int vidioc_try_fmt_vid_out(struct file *file, void *fh,
 +                      struct v4l2_format *f)
 +{
 +      struct omap_overlay *ovl;
 +      struct omapvideo_info *ovid;
 +      struct omap_video_timings *timing;
 +      struct omap_vout_device *vout = fh;
++      struct omap_dss_device *dssdev;
 +
 +      ovid = &vout->vid_info;
 +      ovl = ovid->overlays[0];
++      /* get the display device attached to the overlay */
++      dssdev = ovl->get_device(ovl);
 +
-       /* get the display device attached to the overlay */
-       timing = &ovl->manager->device->panel.timings;
++      if (!dssdev)
 +              return -EINVAL;
-       if (!ovl->manager || !ovl->manager->device) {
++
++      timing = &dssdev->panel.timings;
 +
 +      vout->fbuf.fmt.height = timing->y_res;
 +      vout->fbuf.fmt.width = timing->x_res;
 +
 +      omap_vout_try_format(&f->fmt.pix);
 +      return 0;
 +}
 +
 +static int vidioc_s_fmt_vid_out(struct file *file, void *fh,
 +                      struct v4l2_format *f)
 +{
 +      int ret, bpp;
 +      struct omap_overlay *ovl;
 +      struct omapvideo_info *ovid;
 +      struct omap_video_timings *timing;
 +      struct omap_vout_device *vout = fh;
++      struct omap_dss_device *dssdev;
 +
 +      if (vout->streaming)
 +              return -EBUSY;
 +
 +      mutex_lock(&vout->lock);
 +
 +      ovid = &vout->vid_info;
 +      ovl = ovid->overlays[0];
++      dssdev = ovl->get_device(ovl);
 +
 +      /* get the display device attached to the overlay */
-       timing = &ovl->manager->device->panel.timings;
++      if (!dssdev) {
 +              ret = -EINVAL;
 +              goto s_fmt_vid_out_exit;
 +      }
-       if (!ovl->manager || !ovl->manager->device) {
++      timing = &dssdev->panel.timings;
 +
 +      /* We dont support RGB24-packed mode if vrfb rotation
 +       * is enabled*/
 +      if ((is_rotation_enabled(vout)) &&
 +                      f->fmt.pix.pixelformat == V4L2_PIX_FMT_RGB24) {
 +              ret = -EINVAL;
 +              goto s_fmt_vid_out_exit;
 +      }
 +
 +      /* get the framebuffer parameters */
 +
 +      if (is_rotation_90_or_270(vout)) {
 +              vout->fbuf.fmt.height = timing->x_res;
 +              vout->fbuf.fmt.width = timing->y_res;
 +      } else {
 +              vout->fbuf.fmt.height = timing->y_res;
 +              vout->fbuf.fmt.width = timing->x_res;
 +      }
 +
 +      /* change to samller size is OK */
 +
 +      bpp = omap_vout_try_format(&f->fmt.pix);
 +      f->fmt.pix.sizeimage = f->fmt.pix.width * f->fmt.pix.height * bpp;
 +
 +      /* try & set the new output format */
 +      vout->bpp = bpp;
 +      vout->pix = f->fmt.pix;
 +      vout->vrfb_bpp = 1;
 +
 +      /* If YUYV then vrfb bpp is 2, for  others its 1 */
 +      if (V4L2_PIX_FMT_YUYV == vout->pix.pixelformat ||
 +                      V4L2_PIX_FMT_UYVY == vout->pix.pixelformat)
 +              vout->vrfb_bpp = 2;
 +
 +      /* set default crop and win */
 +      omap_vout_new_format(&vout->pix, &vout->fbuf, &vout->crop, &vout->win);
 +
 +      /* Save the changes in the overlay strcuture */
 +      ret = omapvid_init(vout, 0);
 +      if (ret) {
 +              v4l2_err(&vout->vid_dev->v4l2_dev, "failed to change mode\n");
 +              goto s_fmt_vid_out_exit;
 +      }
 +
 +      ret = 0;
 +
 +s_fmt_vid_out_exit:
 +      mutex_unlock(&vout->lock);
 +      return ret;
 +}
 +
 +static int vidioc_try_fmt_vid_overlay(struct file *file, void *fh,
 +                      struct v4l2_format *f)
 +{
 +      int ret = 0;
 +      struct omap_vout_device *vout = fh;
 +      struct omap_overlay *ovl;
 +      struct omapvideo_info *ovid;
 +      struct v4l2_window *win = &f->fmt.win;
 +
 +      ovid = &vout->vid_info;
 +      ovl = ovid->overlays[0];
 +
 +      ret = omap_vout_try_window(&vout->fbuf, win);
 +
 +      if (!ret) {
 +              if ((ovl->caps & OMAP_DSS_OVL_CAP_GLOBAL_ALPHA) == 0)
 +                      win->global_alpha = 255;
 +              else
 +                      win->global_alpha = f->fmt.win.global_alpha;
 +      }
 +
 +      return ret;
 +}
 +
 +static int vidioc_s_fmt_vid_overlay(struct file *file, void *fh,
 +                      struct v4l2_format *f)
 +{
 +      int ret = 0;
 +      struct omap_overlay *ovl;
 +      struct omapvideo_info *ovid;
 +      struct omap_vout_device *vout = fh;
 +      struct v4l2_window *win = &f->fmt.win;
 +
 +      mutex_lock(&vout->lock);
 +      ovid = &vout->vid_info;
 +      ovl = ovid->overlays[0];
 +
 +      ret = omap_vout_new_window(&vout->crop, &vout->win, &vout->fbuf, win);
 +      if (!ret) {
 +              /* Video1 plane does not support global alpha on OMAP3 */
 +              if ((ovl->caps & OMAP_DSS_OVL_CAP_GLOBAL_ALPHA) == 0)
 +                      vout->win.global_alpha = 255;
 +              else
 +                      vout->win.global_alpha = f->fmt.win.global_alpha;
 +
 +              vout->win.chromakey = f->fmt.win.chromakey;
 +      }
 +      mutex_unlock(&vout->lock);
 +      return ret;
 +}
 +
 +static int vidioc_enum_fmt_vid_overlay(struct file *file, void *fh,
 +                      struct v4l2_fmtdesc *fmt)
 +{
 +      int index = fmt->index;
 +
 +      if (index >= NUM_OUTPUT_FORMATS)
 +              return -EINVAL;
 +
 +      fmt->flags = omap_formats[index].flags;
 +      strlcpy(fmt->description, omap_formats[index].description,
 +                      sizeof(fmt->description));
 +      fmt->pixelformat = omap_formats[index].pixelformat;
 +      return 0;
 +}
 +
 +static int vidioc_g_fmt_vid_overlay(struct file *file, void *fh,
 +                      struct v4l2_format *f)
 +{
 +      u32 key_value =  0;
 +      struct omap_overlay *ovl;
 +      struct omapvideo_info *ovid;
 +      struct omap_vout_device *vout = fh;
 +      struct omap_overlay_manager_info info;
 +      struct v4l2_window *win = &f->fmt.win;
 +
 +      ovid = &vout->vid_info;
 +      ovl = ovid->overlays[0];
 +
 +      win->w = vout->win.w;
 +      win->field = vout->win.field;
 +      win->global_alpha = vout->win.global_alpha;
 +
 +      if (ovl->manager && ovl->manager->get_manager_info) {
 +              ovl->manager->get_manager_info(ovl->manager, &info);
 +              key_value = info.trans_key;
 +      }
 +      win->chromakey = key_value;
 +      return 0;
 +}
 +
 +static int vidioc_cropcap(struct file *file, void *fh,
 +              struct v4l2_cropcap *cropcap)
 +{
 +      struct omap_vout_device *vout = fh;
 +      struct v4l2_pix_format *pix = &vout->pix;
 +
 +      if (cropcap->type != V4L2_BUF_TYPE_VIDEO_OUTPUT)
 +              return -EINVAL;
 +
 +      /* Width and height are always even */
 +      cropcap->bounds.width = pix->width & ~1;
 +      cropcap->bounds.height = pix->height & ~1;
 +
 +      omap_vout_default_crop(&vout->pix, &vout->fbuf, &cropcap->defrect);
 +      cropcap->pixelaspect.numerator = 1;
 +      cropcap->pixelaspect.denominator = 1;
 +      return 0;
 +}
 +
 +static int vidioc_g_crop(struct file *file, void *fh, struct v4l2_crop *crop)
 +{
 +      struct omap_vout_device *vout = fh;
 +
 +      if (crop->type != V4L2_BUF_TYPE_VIDEO_OUTPUT)
 +              return -EINVAL;
 +      crop->c = vout->crop;
 +      return 0;
 +}
 +
 +static int vidioc_s_crop(struct file *file, void *fh, const struct v4l2_crop *crop)
 +{
 +      int ret = -EINVAL;
 +      struct omap_vout_device *vout = fh;
 +      struct omapvideo_info *ovid;
 +      struct omap_overlay *ovl;
 +      struct omap_video_timings *timing;
++      struct omap_dss_device *dssdev;
 +
 +      if (vout->streaming)
 +              return -EBUSY;
 +
 +      mutex_lock(&vout->lock);
 +      ovid = &vout->vid_info;
 +      ovl = ovid->overlays[0];
++      /* get the display device attached to the overlay */
++      dssdev = ovl->get_device(ovl);
 +
-       /* get the display device attached to the overlay */
-       timing = &ovl->manager->device->panel.timings;
++      if (!dssdev) {
 +              ret = -EINVAL;
 +              goto s_crop_err;
 +      }
-               if (ovl->manager && ovl->manager->device) {
++
++      timing = &dssdev->panel.timings;
 +
 +      if (is_rotation_90_or_270(vout)) {
 +              vout->fbuf.fmt.height = timing->x_res;
 +              vout->fbuf.fmt.width = timing->y_res;
 +      } else {
 +              vout->fbuf.fmt.height = timing->y_res;
 +              vout->fbuf.fmt.width = timing->x_res;
 +      }
 +
 +      if (crop->type == V4L2_BUF_TYPE_VIDEO_OUTPUT)
 +              ret = omap_vout_new_crop(&vout->pix, &vout->crop, &vout->win,
 +                              &vout->fbuf, &crop->c);
 +
 +s_crop_err:
 +      mutex_unlock(&vout->lock);
 +      return ret;
 +}
 +
 +static int vidioc_queryctrl(struct file *file, void *fh,
 +              struct v4l2_queryctrl *ctrl)
 +{
 +      int ret = 0;
 +
 +      switch (ctrl->id) {
 +      case V4L2_CID_ROTATE:
 +              ret = v4l2_ctrl_query_fill(ctrl, 0, 270, 90, 0);
 +              break;
 +      case V4L2_CID_BG_COLOR:
 +              ret = v4l2_ctrl_query_fill(ctrl, 0, 0xFFFFFF, 1, 0);
 +              break;
 +      case V4L2_CID_VFLIP:
 +              ret = v4l2_ctrl_query_fill(ctrl, 0, 1, 1, 0);
 +              break;
 +      default:
 +              ctrl->name[0] = '\0';
 +              ret = -EINVAL;
 +      }
 +      return ret;
 +}
 +
 +static int vidioc_g_ctrl(struct file *file, void *fh, struct v4l2_control *ctrl)
 +{
 +      int ret = 0;
 +      struct omap_vout_device *vout = fh;
 +
 +      switch (ctrl->id) {
 +      case V4L2_CID_ROTATE:
 +              ctrl->value = vout->control[0].value;
 +              break;
 +      case V4L2_CID_BG_COLOR:
 +      {
 +              struct omap_overlay_manager_info info;
 +              struct omap_overlay *ovl;
 +
 +              ovl = vout->vid_info.overlays[0];
 +              if (!ovl->manager || !ovl->manager->get_manager_info) {
 +                      ret = -EINVAL;
 +                      break;
 +              }
 +
 +              ovl->manager->get_manager_info(ovl->manager, &info);
 +              ctrl->value = info.default_color;
 +              break;
 +      }
 +      case V4L2_CID_VFLIP:
 +              ctrl->value = vout->control[2].value;
 +              break;
 +      default:
 +              ret = -EINVAL;
 +      }
 +      return ret;
 +}
 +
 +static int vidioc_s_ctrl(struct file *file, void *fh, struct v4l2_control *a)
 +{
 +      int ret = 0;
 +      struct omap_vout_device *vout = fh;
 +
 +      switch (a->id) {
 +      case V4L2_CID_ROTATE:
 +      {
 +              struct omapvideo_info *ovid;
 +              int rotation = a->value;
 +
 +              ovid = &vout->vid_info;
 +
 +              mutex_lock(&vout->lock);
 +              if (rotation && ovid->rotation_type == VOUT_ROT_NONE) {
 +                      mutex_unlock(&vout->lock);
 +                      ret = -ERANGE;
 +                      break;
 +              }
 +
 +              if (rotation && vout->pix.pixelformat == V4L2_PIX_FMT_RGB24) {
 +                      mutex_unlock(&vout->lock);
 +                      ret = -EINVAL;
 +                      break;
 +              }
 +
 +              if (v4l2_rot_to_dss_rot(rotation, &vout->rotation,
 +                                                      vout->mirror)) {
 +                      mutex_unlock(&vout->lock);
 +                      ret = -EINVAL;
 +                      break;
 +              }
 +
 +              vout->control[0].value = rotation;
 +              mutex_unlock(&vout->lock);
 +              break;
 +      }
 +      case V4L2_CID_BG_COLOR:
 +      {
 +              struct omap_overlay *ovl;
 +              unsigned int  color = a->value;
 +              struct omap_overlay_manager_info info;
 +
 +              ovl = vout->vid_info.overlays[0];
 +
 +              mutex_lock(&vout->lock);
 +              if (!ovl->manager || !ovl->manager->get_manager_info) {
 +                      mutex_unlock(&vout->lock);
 +                      ret = -EINVAL;
 +                      break;
 +              }
 +
 +              ovl->manager->get_manager_info(ovl->manager, &info);
 +              info.default_color = color;
 +              if (ovl->manager->set_manager_info(ovl->manager, &info)) {
 +                      mutex_unlock(&vout->lock);
 +                      ret = -EINVAL;
 +                      break;
 +              }
 +
 +              vout->control[1].value = color;
 +              mutex_unlock(&vout->lock);
 +              break;
 +      }
 +      case V4L2_CID_VFLIP:
 +      {
 +              struct omap_overlay *ovl;
 +              struct omapvideo_info *ovid;
 +              unsigned int  mirror = a->value;
 +
 +              ovid = &vout->vid_info;
 +              ovl = ovid->overlays[0];
 +
 +              mutex_lock(&vout->lock);
 +              if (mirror && ovid->rotation_type == VOUT_ROT_NONE) {
 +                      mutex_unlock(&vout->lock);
 +                      ret = -ERANGE;
 +                      break;
 +              }
 +
 +              if (mirror  && vout->pix.pixelformat == V4L2_PIX_FMT_RGB24) {
 +                      mutex_unlock(&vout->lock);
 +                      ret = -EINVAL;
 +                      break;
 +              }
 +              vout->mirror = mirror;
 +              vout->control[2].value = mirror;
 +              mutex_unlock(&vout->lock);
 +              break;
 +      }
 +      default:
 +              ret = -EINVAL;
 +      }
 +      return ret;
 +}
 +
 +static int vidioc_reqbufs(struct file *file, void *fh,
 +                      struct v4l2_requestbuffers *req)
 +{
 +      int ret = 0;
 +      unsigned int i, num_buffers = 0;
 +      struct omap_vout_device *vout = fh;
 +      struct videobuf_queue *q = &vout->vbq;
 +
 +      if ((req->type != V4L2_BUF_TYPE_VIDEO_OUTPUT) || (req->count < 0))
 +              return -EINVAL;
 +      /* if memory is not mmp or userptr
 +         return error */
 +      if ((V4L2_MEMORY_MMAP != req->memory) &&
 +                      (V4L2_MEMORY_USERPTR != req->memory))
 +              return -EINVAL;
 +
 +      mutex_lock(&vout->lock);
 +      /* Cannot be requested when streaming is on */
 +      if (vout->streaming) {
 +              ret = -EBUSY;
 +              goto reqbuf_err;
 +      }
 +
 +      /* If buffers are already allocated free them */
 +      if (q->bufs[0] && (V4L2_MEMORY_MMAP == q->bufs[0]->memory)) {
 +              if (vout->mmap_count) {
 +                      ret = -EBUSY;
 +                      goto reqbuf_err;
 +              }
 +              num_buffers = (vout->vid == OMAP_VIDEO1) ?
 +                      video1_numbuffers : video2_numbuffers;
 +              for (i = num_buffers; i < vout->buffer_allocated; i++) {
 +                      omap_vout_free_buffer(vout->buf_virt_addr[i],
 +                                      vout->buffer_size);
 +                      vout->buf_virt_addr[i] = 0;
 +                      vout->buf_phy_addr[i] = 0;
 +              }
 +              vout->buffer_allocated = num_buffers;
 +              videobuf_mmap_free(q);
 +      } else if (q->bufs[0] && (V4L2_MEMORY_USERPTR == q->bufs[0]->memory)) {
 +              if (vout->buffer_allocated) {
 +                      videobuf_mmap_free(q);
 +                      for (i = 0; i < vout->buffer_allocated; i++) {
 +                              kfree(q->bufs[i]);
 +                              q->bufs[i] = NULL;
 +                      }
 +                      vout->buffer_allocated = 0;
 +              }
 +      }
 +
 +      /*store the memory type in data structure */
 +      vout->memory = req->memory;
 +
 +      INIT_LIST_HEAD(&vout->dma_queue);
 +
 +      /* call videobuf_reqbufs api */
 +      ret = videobuf_reqbufs(q, req);
 +      if (ret < 0)
 +              goto reqbuf_err;
 +
 +      vout->buffer_allocated = req->count;
 +
 +reqbuf_err:
 +      mutex_unlock(&vout->lock);
 +      return ret;
 +}
 +
 +static int vidioc_querybuf(struct file *file, void *fh,
 +                      struct v4l2_buffer *b)
 +{
 +      struct omap_vout_device *vout = fh;
 +
 +      return videobuf_querybuf(&vout->vbq, b);
 +}
 +
 +static int vidioc_qbuf(struct file *file, void *fh,
 +                      struct v4l2_buffer *buffer)
 +{
 +      struct omap_vout_device *vout = fh;
 +      struct videobuf_queue *q = &vout->vbq;
 +
 +      if ((V4L2_BUF_TYPE_VIDEO_OUTPUT != buffer->type) ||
 +                      (buffer->index >= vout->buffer_allocated) ||
 +                      (q->bufs[buffer->index]->memory != buffer->memory)) {
 +              return -EINVAL;
 +      }
 +      if (V4L2_MEMORY_USERPTR == buffer->memory) {
 +              if ((buffer->length < vout->pix.sizeimage) ||
 +                              (0 == buffer->m.userptr)) {
 +                      return -EINVAL;
 +              }
 +      }
 +
 +      if ((is_rotation_enabled(vout)) &&
 +                      vout->vrfb_dma_tx.req_status == DMA_CHAN_NOT_ALLOTED) {
 +              v4l2_warn(&vout->vid_dev->v4l2_dev,
 +                              "DMA Channel not allocated for Rotation\n");
 +              return -EINVAL;
 +      }
 +
 +      return videobuf_qbuf(q, buffer);
 +}
 +
 +static int vidioc_dqbuf(struct file *file, void *fh, struct v4l2_buffer *b)
 +{
 +      struct omap_vout_device *vout = fh;
 +      struct videobuf_queue *q = &vout->vbq;
 +
 +      int ret;
 +      u32 addr;
 +      unsigned long size;
 +      struct videobuf_buffer *vb;
 +
 +      vb = q->bufs[b->index];
 +
 +      if (!vout->streaming)
 +              return -EINVAL;
 +
 +      if (file->f_flags & O_NONBLOCK)
 +              /* Call videobuf_dqbuf for non blocking mode */
 +              ret = videobuf_dqbuf(q, (struct v4l2_buffer *)b, 1);
 +      else
 +              /* Call videobuf_dqbuf for  blocking mode */
 +              ret = videobuf_dqbuf(q, (struct v4l2_buffer *)b, 0);
 +
 +      addr = (unsigned long) vout->buf_phy_addr[vb->i];
 +      size = (unsigned long) vb->size;
 +      dma_unmap_single(vout->vid_dev->v4l2_dev.dev,  addr,
 +                              size, DMA_TO_DEVICE);
 +      return ret;
 +}
 +
 +static int vidioc_streamon(struct file *file, void *fh, enum v4l2_buf_type i)
 +{
 +      int ret = 0, j;
 +      u32 addr = 0, mask = 0;
 +      struct omap_vout_device *vout = fh;
 +      struct videobuf_queue *q = &vout->vbq;
 +      struct omapvideo_info *ovid = &vout->vid_info;
 +
 +      mutex_lock(&vout->lock);
 +
 +      if (vout->streaming) {
 +              ret = -EBUSY;
 +              goto streamon_err;
 +      }
 +
 +      ret = videobuf_streamon(q);
 +      if (ret)
 +              goto streamon_err;
 +
 +      if (list_empty(&vout->dma_queue)) {
 +              ret = -EIO;
 +              goto streamon_err1;
 +      }
 +
 +      /* Get the next frame from the buffer queue */
 +      vout->next_frm = vout->cur_frm = list_entry(vout->dma_queue.next,
 +                      struct videobuf_buffer, queue);
 +      /* Remove buffer from the buffer queue */
 +      list_del(&vout->cur_frm->queue);
 +      /* Mark state of the current frame to active */
 +      vout->cur_frm->state = VIDEOBUF_ACTIVE;
 +      /* Initialize field_id and started member */
 +      vout->field_id = 0;
 +
 +      /* set flag here. Next QBUF will start DMA */
 +      vout->streaming = 1;
 +
 +      vout->first_int = 1;
 +
 +      if (omap_vout_calculate_offset(vout)) {
 +              ret = -EINVAL;
 +              goto streamon_err1;
 +      }
 +      addr = (unsigned long) vout->queued_buf_addr[vout->cur_frm->i]
 +              + vout->cropped_offset;
 +
 +      mask = DISPC_IRQ_VSYNC | DISPC_IRQ_EVSYNC_EVEN | DISPC_IRQ_EVSYNC_ODD
 +              | DISPC_IRQ_VSYNC2;
 +
 +      omap_dispc_register_isr(omap_vout_isr, vout, mask);
 +
 +      for (j = 0; j < ovid->num_overlays; j++) {
 +              struct omap_overlay *ovl = ovid->overlays[j];
 +
-               if (ovl->manager && ovl->manager->device) {
++              if (ovl->get_device(ovl)) {
 +                      struct omap_overlay_info info;
 +                      ovl->get_overlay_info(ovl, &info);
 +                      info.paddr = addr;
 +                      if (ovl->set_overlay_info(ovl, &info)) {
 +                              ret = -EINVAL;
 +                              goto streamon_err1;
 +                      }
 +              }
 +      }
 +
 +      /* First save the configuration in ovelray structure */
 +      ret = omapvid_init(vout, addr);
 +      if (ret)
 +              v4l2_err(&vout->vid_dev->v4l2_dev,
 +                              "failed to set overlay info\n");
 +      /* Enable the pipeline and set the Go bit */
 +      ret = omapvid_apply_changes(vout);
 +      if (ret)
 +              v4l2_err(&vout->vid_dev->v4l2_dev, "failed to change mode\n");
 +
 +      for (j = 0; j < ovid->num_overlays; j++) {
 +              struct omap_overlay *ovl = ovid->overlays[j];
++              struct omap_dss_device *dssdev = ovl->get_device(ovl);
 +
-               if (ovl->manager && ovl->manager->device)
++              if (dssdev) {
 +                      ret = ovl->enable(ovl);
 +                      if (ret)
 +                              goto streamon_err1;
 +              }
 +      }
 +
 +      ret = 0;
 +
 +streamon_err1:
 +      if (ret)
 +              ret = videobuf_streamoff(q);
 +streamon_err:
 +      mutex_unlock(&vout->lock);
 +      return ret;
 +}
 +
 +static int vidioc_streamoff(struct file *file, void *fh, enum v4l2_buf_type i)
 +{
 +      u32 mask = 0;
 +      int ret = 0, j;
 +      struct omap_vout_device *vout = fh;
 +      struct omapvideo_info *ovid = &vout->vid_info;
 +
 +      if (!vout->streaming)
 +              return -EINVAL;
 +
 +      vout->streaming = 0;
 +      mask = DISPC_IRQ_VSYNC | DISPC_IRQ_EVSYNC_EVEN | DISPC_IRQ_EVSYNC_ODD
 +              | DISPC_IRQ_VSYNC2;
 +
 +      omap_dispc_unregister_isr(omap_vout_isr, vout, mask);
 +
 +      for (j = 0; j < ovid->num_overlays; j++) {
 +              struct omap_overlay *ovl = ovid->overlays[j];
++              struct omap_dss_device *dssdev = ovl->get_device(ovl);
 +
-       struct omap_dss_device *display =
-               vout->vid_info.overlays[0]->manager->device;
++              if (dssdev)
 +                      ovl->disable(ovl);
 +      }
 +
 +      /* Turn of the pipeline */
 +      ret = omapvid_apply_changes(vout);
 +      if (ret)
 +              v4l2_err(&vout->vid_dev->v4l2_dev, "failed to change mode in"
 +                              " streamoff\n");
 +
 +      INIT_LIST_HEAD(&vout->dma_queue);
 +      ret = videobuf_streamoff(&vout->vbq);
 +
 +      return ret;
 +}
 +
 +static int vidioc_s_fbuf(struct file *file, void *fh,
 +                              const struct v4l2_framebuffer *a)
 +{
 +      int enable = 0;
 +      struct omap_overlay *ovl;
 +      struct omapvideo_info *ovid;
 +      struct omap_vout_device *vout = fh;
 +      struct omap_overlay_manager_info info;
 +      enum omap_dss_trans_key_type key_type = OMAP_DSS_COLOR_KEY_GFX_DST;
 +
 +      ovid = &vout->vid_info;
 +      ovl = ovid->overlays[0];
 +
 +      /* OMAP DSS doesn't support Source and Destination color
 +         key together */
 +      if ((a->flags & V4L2_FBUF_FLAG_SRC_CHROMAKEY) &&
 +                      (a->flags & V4L2_FBUF_FLAG_CHROMAKEY))
 +              return -EINVAL;
 +      /* OMAP DSS Doesn't support the Destination color key
 +         and alpha blending together */
 +      if ((a->flags & V4L2_FBUF_FLAG_CHROMAKEY) &&
 +                      (a->flags & V4L2_FBUF_FLAG_LOCAL_ALPHA))
 +              return -EINVAL;
 +
 +      if ((a->flags & V4L2_FBUF_FLAG_SRC_CHROMAKEY)) {
 +              vout->fbuf.flags |= V4L2_FBUF_FLAG_SRC_CHROMAKEY;
 +              key_type =  OMAP_DSS_COLOR_KEY_VID_SRC;
 +      } else
 +              vout->fbuf.flags &= ~V4L2_FBUF_FLAG_SRC_CHROMAKEY;
 +
 +      if ((a->flags & V4L2_FBUF_FLAG_CHROMAKEY)) {
 +              vout->fbuf.flags |= V4L2_FBUF_FLAG_CHROMAKEY;
 +              key_type =  OMAP_DSS_COLOR_KEY_GFX_DST;
 +      } else
 +              vout->fbuf.flags &=  ~V4L2_FBUF_FLAG_CHROMAKEY;
 +
 +      if (a->flags & (V4L2_FBUF_FLAG_CHROMAKEY |
 +                              V4L2_FBUF_FLAG_SRC_CHROMAKEY))
 +              enable = 1;
 +      else
 +              enable = 0;
 +      if (ovl->manager && ovl->manager->get_manager_info &&
 +                      ovl->manager->set_manager_info) {
 +
 +              ovl->manager->get_manager_info(ovl->manager, &info);
 +              info.trans_enabled = enable;
 +              info.trans_key_type = key_type;
 +              info.trans_key = vout->win.chromakey;
 +
 +              if (ovl->manager->set_manager_info(ovl->manager, &info))
 +                      return -EINVAL;
 +      }
 +      if (a->flags & V4L2_FBUF_FLAG_LOCAL_ALPHA) {
 +              vout->fbuf.flags |= V4L2_FBUF_FLAG_LOCAL_ALPHA;
 +              enable = 1;
 +      } else {
 +              vout->fbuf.flags &= ~V4L2_FBUF_FLAG_LOCAL_ALPHA;
 +              enable = 0;
 +      }
 +      if (ovl->manager && ovl->manager->get_manager_info &&
 +                      ovl->manager->set_manager_info) {
 +              ovl->manager->get_manager_info(ovl->manager, &info);
 +              /* enable this only if there is no zorder cap */
 +              if ((ovl->caps & OMAP_DSS_OVL_CAP_ZORDER) == 0)
 +                      info.partial_alpha_enabled = enable;
 +              if (ovl->manager->set_manager_info(ovl->manager, &info))
 +                      return -EINVAL;
 +      }
 +
 +      return 0;
 +}
 +
 +static int vidioc_g_fbuf(struct file *file, void *fh,
 +              struct v4l2_framebuffer *a)
 +{
 +      struct omap_overlay *ovl;
 +      struct omapvideo_info *ovid;
 +      struct omap_vout_device *vout = fh;
 +      struct omap_overlay_manager_info info;
 +
 +      ovid = &vout->vid_info;
 +      ovl = ovid->overlays[0];
 +
 +      /* The video overlay must stay within the framebuffer and can't be
 +         positioned independently. */
 +      a->flags = V4L2_FBUF_FLAG_OVERLAY;
 +      a->capability = V4L2_FBUF_CAP_LOCAL_ALPHA | V4L2_FBUF_CAP_CHROMAKEY
 +              | V4L2_FBUF_CAP_SRC_CHROMAKEY;
 +
 +      if (ovl->manager && ovl->manager->get_manager_info) {
 +              ovl->manager->get_manager_info(ovl->manager, &info);
 +              if (info.trans_key_type == OMAP_DSS_COLOR_KEY_VID_SRC)
 +                      a->flags |= V4L2_FBUF_FLAG_SRC_CHROMAKEY;
 +              if (info.trans_key_type == OMAP_DSS_COLOR_KEY_GFX_DST)
 +                      a->flags |= V4L2_FBUF_FLAG_CHROMAKEY;
 +      }
 +      if (ovl->manager && ovl->manager->get_manager_info) {
 +              ovl->manager->get_manager_info(ovl->manager, &info);
 +              if (info.partial_alpha_enabled)
 +                      a->flags |= V4L2_FBUF_FLAG_LOCAL_ALPHA;
 +      }
 +
 +      return 0;
 +}
 +
 +static const struct v4l2_ioctl_ops vout_ioctl_ops = {
 +      .vidioc_querycap                        = vidioc_querycap,
 +      .vidioc_enum_fmt_vid_out                = vidioc_enum_fmt_vid_out,
 +      .vidioc_g_fmt_vid_out                   = vidioc_g_fmt_vid_out,
 +      .vidioc_try_fmt_vid_out                 = vidioc_try_fmt_vid_out,
 +      .vidioc_s_fmt_vid_out                   = vidioc_s_fmt_vid_out,
 +      .vidioc_queryctrl                       = vidioc_queryctrl,
 +      .vidioc_g_ctrl                          = vidioc_g_ctrl,
 +      .vidioc_s_fbuf                          = vidioc_s_fbuf,
 +      .vidioc_g_fbuf                          = vidioc_g_fbuf,
 +      .vidioc_s_ctrl                          = vidioc_s_ctrl,
 +      .vidioc_try_fmt_vid_overlay             = vidioc_try_fmt_vid_overlay,
 +      .vidioc_s_fmt_vid_overlay               = vidioc_s_fmt_vid_overlay,
 +      .vidioc_enum_fmt_vid_overlay            = vidioc_enum_fmt_vid_overlay,
 +      .vidioc_g_fmt_vid_overlay               = vidioc_g_fmt_vid_overlay,
 +      .vidioc_cropcap                         = vidioc_cropcap,
 +      .vidioc_g_crop                          = vidioc_g_crop,
 +      .vidioc_s_crop                          = vidioc_s_crop,
 +      .vidioc_reqbufs                         = vidioc_reqbufs,
 +      .vidioc_querybuf                        = vidioc_querybuf,
 +      .vidioc_qbuf                            = vidioc_qbuf,
 +      .vidioc_dqbuf                           = vidioc_dqbuf,
 +      .vidioc_streamon                        = vidioc_streamon,
 +      .vidioc_streamoff                       = vidioc_streamoff,
 +};
 +
 +static const struct v4l2_file_operations omap_vout_fops = {
 +      .owner          = THIS_MODULE,
 +      .poll           = omap_vout_poll,
 +      .unlocked_ioctl = video_ioctl2,
 +      .mmap           = omap_vout_mmap,
 +      .open           = omap_vout_open,
 +      .release        = omap_vout_release,
 +};
 +
 +/* Init functions used during driver initialization */
 +/* Initial setup of video_data */
 +static int __init omap_vout_setup_video_data(struct omap_vout_device *vout)
 +{
 +      struct video_device *vfd;
 +      struct v4l2_pix_format *pix;
 +      struct v4l2_control *control;
-               if (ovl->manager && ovl->manager->device) {
-                       def_display = ovl->manager->device;
++      struct omap_overlay *ovl = vout->vid_info.overlays[0];
++      struct omap_dss_device *display = ovl->get_device(ovl);
 +
 +      /* set the default pix */
 +      pix = &vout->pix;
 +
 +      /* Set the default picture of QVGA  */
 +      pix->width = QQVGA_WIDTH;
 +      pix->height = QQVGA_HEIGHT;
 +
 +      /* Default pixel format is RGB 5-6-5 */
 +      pix->pixelformat = V4L2_PIX_FMT_RGB565;
 +      pix->field = V4L2_FIELD_ANY;
 +      pix->bytesperline = pix->width * 2;
 +      pix->sizeimage = pix->bytesperline * pix->height;
 +      pix->priv = 0;
 +      pix->colorspace = V4L2_COLORSPACE_JPEG;
 +
 +      vout->bpp = RGB565_BPP;
 +      vout->fbuf.fmt.width  =  display->panel.timings.x_res;
 +      vout->fbuf.fmt.height =  display->panel.timings.y_res;
 +
 +      /* Set the data structures for the overlay parameters*/
 +      vout->win.global_alpha = 255;
 +      vout->fbuf.flags = 0;
 +      vout->fbuf.capability = V4L2_FBUF_CAP_LOCAL_ALPHA |
 +              V4L2_FBUF_CAP_SRC_CHROMAKEY | V4L2_FBUF_CAP_CHROMAKEY;
 +      vout->win.chromakey = 0;
 +
 +      omap_vout_new_format(pix, &vout->fbuf, &vout->crop, &vout->win);
 +
 +      /*Initialize the control variables for
 +        rotation, flipping and background color. */
 +      control = vout->control;
 +      control[0].id = V4L2_CID_ROTATE;
 +      control[0].value = 0;
 +      vout->rotation = 0;
 +      vout->mirror = 0;
 +      vout->control[2].id = V4L2_CID_HFLIP;
 +      vout->control[2].value = 0;
 +      if (vout->vid_info.rotation_type == VOUT_ROT_VRFB)
 +              vout->vrfb_bpp = 2;
 +
 +      control[1].id = V4L2_CID_BG_COLOR;
 +      control[1].value = 0;
 +
 +      /* initialize the video_device struct */
 +      vfd = vout->vfd = video_device_alloc();
 +
 +      if (!vfd) {
 +              printk(KERN_ERR VOUT_NAME ": could not allocate"
 +                              " video device struct\n");
 +              return -ENOMEM;
 +      }
 +      vfd->release = video_device_release;
 +      vfd->ioctl_ops = &vout_ioctl_ops;
 +
 +      strlcpy(vfd->name, VOUT_NAME, sizeof(vfd->name));
 +
 +      vfd->fops = &omap_vout_fops;
 +      vfd->v4l2_dev = &vout->vid_dev->v4l2_dev;
 +      vfd->vfl_dir = VFL_DIR_TX;
 +      mutex_init(&vout->lock);
 +
 +      vfd->minor = -1;
 +      return 0;
 +
 +}
 +
 +/* Setup video buffers */
 +static int __init omap_vout_setup_video_bufs(struct platform_device *pdev,
 +              int vid_num)
 +{
 +      u32 numbuffers;
 +      int ret = 0, i;
 +      struct omapvideo_info *ovid;
 +      struct omap_vout_device *vout;
 +      struct v4l2_device *v4l2_dev = platform_get_drvdata(pdev);
 +      struct omap2video_device *vid_dev =
 +              container_of(v4l2_dev, struct omap2video_device, v4l2_dev);
 +
 +      vout = vid_dev->vouts[vid_num];
 +      ovid = &vout->vid_info;
 +
 +      numbuffers = (vid_num == 0) ? video1_numbuffers : video2_numbuffers;
 +      vout->buffer_size = (vid_num == 0) ? video1_bufsize : video2_bufsize;
 +      dev_info(&pdev->dev, "Buffer Size = %d\n", vout->buffer_size);
 +
 +      for (i = 0; i < numbuffers; i++) {
 +              vout->buf_virt_addr[i] =
 +                      omap_vout_alloc_buffer(vout->buffer_size,
 +                                      (u32 *) &vout->buf_phy_addr[i]);
 +              if (!vout->buf_virt_addr[i]) {
 +                      numbuffers = i;
 +                      ret = -ENOMEM;
 +                      goto free_buffers;
 +              }
 +      }
 +
 +      vout->cropped_offset = 0;
 +
 +      if (ovid->rotation_type == VOUT_ROT_VRFB) {
 +              int static_vrfb_allocation = (vid_num == 0) ?
 +                      vid1_static_vrfb_alloc : vid2_static_vrfb_alloc;
 +              ret = omap_vout_setup_vrfb_bufs(pdev, vid_num,
 +                              static_vrfb_allocation);
 +      }
 +
 +      return ret;
 +
 +free_buffers:
 +      for (i = 0; i < numbuffers; i++) {
 +              omap_vout_free_buffer(vout->buf_virt_addr[i],
 +                                              vout->buffer_size);
 +              vout->buf_virt_addr[i] = 0;
 +              vout->buf_phy_addr[i] = 0;
 +      }
 +      return ret;
 +
 +}
 +
 +/* Create video out devices */
 +static int __init omap_vout_create_video_devices(struct platform_device *pdev)
 +{
 +      int ret = 0, k;
 +      struct omap_vout_device *vout;
 +      struct video_device *vfd = NULL;
 +      struct v4l2_device *v4l2_dev = platform_get_drvdata(pdev);
 +      struct omap2video_device *vid_dev = container_of(v4l2_dev,
 +                      struct omap2video_device, v4l2_dev);
 +
 +      for (k = 0; k < pdev->num_resources; k++) {
 +
 +              vout = kzalloc(sizeof(struct omap_vout_device), GFP_KERNEL);
 +              if (!vout) {
 +                      dev_err(&pdev->dev, ": could not allocate memory\n");
 +                      return -ENOMEM;
 +              }
 +
 +              vout->vid = k;
 +              vid_dev->vouts[k] = vout;
 +              vout->vid_dev = vid_dev;
 +              /* Select video2 if only 1 overlay is controlled by V4L2 */
 +              if (pdev->num_resources == 1)
 +                      vout->vid_info.overlays[0] = vid_dev->overlays[k + 2];
 +              else
 +                      /* Else select video1 and video2 one by one. */
 +                      vout->vid_info.overlays[0] = vid_dev->overlays[k + 1];
 +              vout->vid_info.num_overlays = 1;
 +              vout->vid_info.id = k + 1;
 +
 +              /* Set VRFB as rotation_type for omap2 and omap3 */
 +              if (cpu_is_omap24xx() || cpu_is_omap34xx())
 +                      vout->vid_info.rotation_type = VOUT_ROT_VRFB;
 +
 +              /* Setup the default configuration for the video devices
 +               */
 +              if (omap_vout_setup_video_data(vout) != 0) {
 +                      ret = -ENOMEM;
 +                      goto error;
 +              }
 +
 +              /* Allocate default number of buffers for the video streaming
 +               * and reserve the VRFB space for rotation
 +               */
 +              if (omap_vout_setup_video_bufs(pdev, k) != 0) {
 +                      ret = -ENOMEM;
 +                      goto error1;
 +              }
 +
 +              /* Register the Video device with V4L2
 +               */
 +              vfd = vout->vfd;
 +              if (video_register_device(vfd, VFL_TYPE_GRABBER, -1) < 0) {
 +                      dev_err(&pdev->dev, ": Could not register "
 +                                      "Video for Linux device\n");
 +                      vfd->minor = -1;
 +                      ret = -ENODEV;
 +                      goto error2;
 +              }
 +              video_set_drvdata(vfd, vout);
 +
 +              /* Configure the overlay structure */
 +              ret = omapvid_init(vid_dev->vouts[k], 0);
 +              if (!ret)
 +                      goto success;
 +
 +error2:
 +              if (vout->vid_info.rotation_type == VOUT_ROT_VRFB)
 +                      omap_vout_release_vrfb(vout);
 +              omap_vout_free_buffers(vout);
 +error1:
 +              video_device_release(vfd);
 +error:
 +              kfree(vout);
 +              return ret;
 +
 +success:
 +              dev_info(&pdev->dev, ": registered and initialized"
 +                              " video device %d\n", vfd->minor);
 +              if (k == (pdev->num_resources - 1))
 +                      return 0;
 +      }
 +
 +      return -ENODEV;
 +}
 +/* Driver functions */
 +static void omap_vout_cleanup_device(struct omap_vout_device *vout)
 +{
 +      struct video_device *vfd;
 +      struct omapvideo_info *ovid;
 +
 +      if (!vout)
 +              return;
 +
 +      vfd = vout->vfd;
 +      ovid = &vout->vid_info;
 +      if (vfd) {
 +              if (!video_is_registered(vfd)) {
 +                      /*
 +                       * The device was never registered, so release the
 +                       * video_device struct directly.
 +                       */
 +                      video_device_release(vfd);
 +              } else {
 +                      /*
 +                       * The unregister function will release the video_device
 +                       * struct as well as unregistering it.
 +                       */
 +                      video_unregister_device(vfd);
 +              }
 +      }
 +      if (ovid->rotation_type == VOUT_ROT_VRFB) {
 +              omap_vout_release_vrfb(vout);
 +              /* Free the VRFB buffer if allocated
 +               * init time
 +               */
 +              if (vout->vrfb_static_allocation)
 +                      omap_vout_free_vrfb_buffers(vout);
 +      }
 +      omap_vout_free_buffers(vout);
 +
 +      kfree(vout);
 +}
 +
 +static int omap_vout_remove(struct platform_device *pdev)
 +{
 +      int k;
 +      struct v4l2_device *v4l2_dev = platform_get_drvdata(pdev);
 +      struct omap2video_device *vid_dev = container_of(v4l2_dev, struct
 +                      omap2video_device, v4l2_dev);
 +
 +      v4l2_device_unregister(v4l2_dev);
 +      for (k = 0; k < pdev->num_resources; k++)
 +              omap_vout_cleanup_device(vid_dev->vouts[k]);
 +
 +      for (k = 0; k < vid_dev->num_displays; k++) {
 +              if (vid_dev->displays[k]->state != OMAP_DSS_DISPLAY_DISABLED)
 +                      vid_dev->displays[k]->driver->disable(vid_dev->displays[k]);
 +
 +              omap_dss_put_device(vid_dev->displays[k]);
 +      }
 +      kfree(vid_dev);
 +      return 0;
 +}
 +
 +static int __init omap_vout_probe(struct platform_device *pdev)
 +{
 +      int ret = 0, i;
 +      struct omap_overlay *ovl;
 +      struct omap_dss_device *dssdev = NULL;
 +      struct omap_dss_device *def_display;
 +      struct omap2video_device *vid_dev = NULL;
 +
 +      if (pdev->num_resources == 0) {
 +              dev_err(&pdev->dev, "probed for an unknown device\n");
 +              return -ENODEV;
 +      }
 +
 +      vid_dev = kzalloc(sizeof(struct omap2video_device), GFP_KERNEL);
 +      if (vid_dev == NULL)
 +              return -ENOMEM;
 +
 +      vid_dev->num_displays = 0;
 +      for_each_dss_dev(dssdev) {
 +              omap_dss_get_device(dssdev);
 +
 +              if (!dssdev->driver) {
 +                      dev_warn(&pdev->dev, "no driver for display: %s\n",
 +                                      dssdev->name);
 +                      omap_dss_put_device(dssdev);
 +                      continue;
 +              }
 +
 +              vid_dev->displays[vid_dev->num_displays++] = dssdev;
 +      }
 +
 +      if (vid_dev->num_displays == 0) {
 +              dev_err(&pdev->dev, "no displays\n");
 +              ret = -EINVAL;
 +              goto probe_err0;
 +      }
 +
 +      vid_dev->num_overlays = omap_dss_get_num_overlays();
 +      for (i = 0; i < vid_dev->num_overlays; i++)
 +              vid_dev->overlays[i] = omap_dss_get_overlay(i);
 +
 +      vid_dev->num_managers = omap_dss_get_num_overlay_managers();
 +      for (i = 0; i < vid_dev->num_managers; i++)
 +              vid_dev->managers[i] = omap_dss_get_overlay_manager(i);
 +
 +      /* Get the Video1 overlay and video2 overlay.
 +       * Setup the Display attached to that overlays
 +       */
 +      for (i = 1; i < vid_dev->num_overlays; i++) {
 +              ovl = omap_dss_get_overlay(i);
-               if (ovl->manager && ovl->manager->device)
-                       def_display = ovl->manager->device;
++              dssdev = ovl->get_device(ovl);
++
++              if (dssdev) {
++                      def_display = dssdev;
 +              } else {
 +                      dev_warn(&pdev->dev, "cannot find display\n");
 +                      def_display = NULL;
 +              }
 +              if (def_display) {
 +                      struct omap_dss_driver *dssdrv = def_display->driver;
 +
 +                      ret = dssdrv->enable(def_display);
 +                      if (ret) {
 +                              /* Here we are not considering a error
 +                               *  as display may be enabled by frame
 +                               *  buffer driver
 +                               */
 +                              dev_warn(&pdev->dev,
 +                                      "'%s' Display already enabled\n",
 +                                      def_display->name);
 +                      }
 +              }
 +      }
 +
 +      if (v4l2_device_register(&pdev->dev, &vid_dev->v4l2_dev) < 0) {
 +              dev_err(&pdev->dev, "v4l2_device_register failed\n");
 +              ret = -ENODEV;
 +              goto probe_err1;
 +      }
 +
 +      ret = omap_vout_create_video_devices(pdev);
 +      if (ret)
 +              goto probe_err2;
 +
 +      for (i = 0; i < vid_dev->num_displays; i++) {
 +              struct omap_dss_device *display = vid_dev->displays[i];
 +
 +              if (display->driver->update)
 +                      display->driver->update(display, 0, 0,
 +                                      display->panel.timings.x_res,
 +                                      display->panel.timings.y_res);
 +      }
 +      return 0;
 +
 +probe_err2:
 +      v4l2_device_unregister(&vid_dev->v4l2_dev);
 +probe_err1:
 +      for (i = 1; i < vid_dev->num_overlays; i++) {
 +              def_display = NULL;
 +              ovl = omap_dss_get_overlay(i);
++              dssdev = ovl->get_device(ovl);
++
++              if (dssdev)
++                      def_display = dssdev;
 +
 +              if (def_display && def_display->driver)
 +                      def_display->driver->disable(def_display);
 +      }
 +probe_err0:
 +      kfree(vid_dev);
 +      return ret;
 +}
 +
 +static struct platform_driver omap_vout_driver = {
 +      .driver = {
 +              .name = VOUT_NAME,
 +      },
 +      .remove = omap_vout_remove,
 +};
 +
 +static int __init omap_vout_init(void)
 +{
 +      if (platform_driver_probe(&omap_vout_driver, omap_vout_probe) != 0) {
 +              printk(KERN_ERR VOUT_NAME ":Could not register Video driver\n");
 +              return -EINVAL;
 +      }
 +      return 0;
 +}
 +
 +static void omap_vout_cleanup(void)
 +{
 +      platform_driver_unregister(&omap_vout_driver);
 +}
 +
 +late_initcall(omap_vout_init);
 +module_exit(omap_vout_cleanup);
Simple merge
Simple merge
Simple merge
index 113d43a16f546b95cce777ba62f2a5d5d0e3370c,65a11ef87e1c9b2a1fb4d6dc7e83559a165002a4..80665f66ac1ab844b283bb432a39f0591908d84d
@@@ -948,10 -1038,10 +1038,10 @@@ static int __devexit fb_remove(struct p
                dma_free_coherent(NULL, par->vram_size, par->vram_virt,
                                  par->vram_phys);
                free_irq(par->irq, par);
-               clk_disable(par->lcdc_clk);
-               clk_put(par->lcdc_clk);
+               pm_runtime_put_sync(&dev->dev);
+               pm_runtime_disable(&dev->dev);
                framebuffer_release(info);
 -              iounmap((void __iomem *)da8xx_fb_reg_base);
 +              iounmap(da8xx_fb_reg_base);
                release_mem_region(lcdc_regs->start, resource_size(lcdc_regs));
  
        }
@@@ -1385,14 -1477,12 +1477,12 @@@ err_release_fb_mem
  err_release_fb:
        framebuffer_release(da8xx_fb_info);
  
- err_clk_disable:
-       clk_disable(fb_clk);
- err_clk_put:
-       clk_put(fb_clk);
+ err_pm_runtime_disable:
+       pm_runtime_put_sync(&device->dev);
+       pm_runtime_disable(&device->dev);
  
  err_ioremap:
 -      iounmap((void __iomem *)da8xx_fb_reg_base);
 +      iounmap(da8xx_fb_reg_base);
  
  err_request_mem:
        release_mem_region(lcdc_regs->start, len);
Simple merge
Simple merge
Simple merge
Simple merge
index bf73f048006183da73d75377faddce1d143917d2,d43e178fae273ead19b8707baf3890b47c84ac18..35ac9e8bee63f13a30133382dca3a8f43aa7a25d
  #include <linux/clk.h>
  #include <linux/io.h>
  #include <linux/sched.h>
- #include <mach/msm_iomap.h>
- #include <mach/irqs.h>
- #include <mach/board.h>
 -#include <mach/msm_fb.h>
 +#include <linux/platform_data/video-msm_fb.h>
  #include "mddi_hw.h"
  
  #define FLAG_DISABLE_HIBERNATION 0x0001
index 2e0f3bab61143410971a45f3686f0796823bdd21,752eff7de3cefd91de8487155d68ee4c1fa905e5..f2566c19e71cb10442e42552e9c492b5a3ba1db8
@@@ -25,8 -25,7 +25,7 @@@
  #include <linux/major.h>
  #include <linux/slab.h>
  
- #include <mach/msm_iomap.h>
 -#include <mach/msm_fb.h>
 +#include <linux/platform_data/video-msm_fb.h>
  #include <linux/platform_device.h>
  #include <linux/export.h>
  
index a0bacf581b32896deb3186168c08808791e733fa,2a841374263056a17e2754c4317c8f896d7efc2d..35848d7410015f60ca5c64272557091cb5dc39a8
@@@ -15,8 -15,7 +15,7 @@@
  #ifndef _MDP_HW_H_
  #define _MDP_HW_H_
  
- #include <mach/msm_iomap.h>
 -#include <mach/msm_fb.h>
 +#include <linux/platform_data/video-msm_fb.h>
  
  struct mdp_info {
        struct mdp_device mdp_dev;
Simple merge
Simple merge
index 6b5e6e0e202f253cada3b80ddf5d14db9b4b0bd7,4cf94161ff5101a74fcf0cdb165ce8c9f3aa1f2e..f2f644680ca8dfe2262fac39c42fcd50766982f0
@@@ -900,29 -925,50 +925,50 @@@ static int taal_probe(struct omap_dss_d
  
        atomic_set(&td->do_update, 0);
  
-       td->workqueue = create_singlethread_workqueue("taal_esd");
-       if (td->workqueue == NULL) {
-               dev_err(&dssdev->dev, "can't create ESD workqueue\n");
-               r = -ENOMEM;
-               goto err_wq;
+       if (gpio_is_valid(td->reset_gpio)) {
+               r = devm_gpio_request_one(&dssdev->dev, td->reset_gpio,
+                               GPIOF_OUT_INIT_LOW, "taal rst");
+               if (r) {
+                       dev_err(&dssdev->dev, "failed to request reset gpio\n");
+                       return r;
+               }
        }
-       INIT_DEFERRABLE_WORK(&td->esd_work, taal_esd_work);
-       INIT_DELAYED_WORK(&td->ulps_work, taal_ulps_work);
  
-       dev_set_drvdata(&dssdev->dev, td);
+       if (gpio_is_valid(td->ext_te_gpio)) {
+               r = devm_gpio_request_one(&dssdev->dev, td->ext_te_gpio,
+                               GPIOF_IN, "taal irq");
+               if (r) {
+                       dev_err(&dssdev->dev, "GPIO request failed\n");
+                       return r;
+               }
+               r = devm_request_irq(&dssdev->dev, gpio_to_irq(td->ext_te_gpio),
+                               taal_te_isr,
+                               IRQF_TRIGGER_RISING,
+                               "taal vsync", dssdev);
  
-       if (gpio_is_valid(panel_data->reset_gpio)) {
-               r = gpio_request_one(panel_data->reset_gpio, GPIOF_OUT_INIT_LOW,
-                               "taal rst");
                if (r) {
-                       dev_err(&dssdev->dev, "failed to request reset gpio\n");
-                       goto err_rst_gpio;
+                       dev_err(&dssdev->dev, "IRQ request failed\n");
+                       return r;
                }
 -              INIT_DELAYED_WORK_DEFERRABLE(&td->te_timeout_work,
++              INIT_DEFERRABLE_WORK(&td->te_timeout_work,
+                                       taal_te_timeout_work_callback);
+               dev_dbg(&dssdev->dev, "Using GPIO TE\n");
        }
  
 -      INIT_DELAYED_WORK_DEFERRABLE(&td->esd_work, taal_esd_work);
+       td->workqueue = create_singlethread_workqueue("taal_esd");
+       if (td->workqueue == NULL) {
+               dev_err(&dssdev->dev, "can't create ESD workqueue\n");
+               return -ENOMEM;
+       }
++      INIT_DEFERRABLE_WORK(&td->esd_work, taal_esd_work);
+       INIT_DELAYED_WORK(&td->ulps_work, taal_ulps_work);
        taal_hw_reset(dssdev);
  
-       if (panel_data->use_dsi_backlight) {
+       if (td->use_dsi_backlight) {
                memset(&props, 0, sizeof(struct backlight_properties));
                props.max_brightness = 255;
  
index ee9e29639dcc4323c5d1d87b31390da0a1ad5373,a173a9481a23f54a34321f9cc250f4226828812b..b43477a5fae869e5e8981faaea48f51c15827e49
@@@ -37,9 -37,6 +37,8 @@@
  #include <linux/platform_device.h>
  #include <linux/pm_runtime.h>
  
- #include <plat/clock.h>
 +#include <plat/cpu.h>
 +
  #include <video/omapdss.h>
  
  #include "dss.h"
Simple merge
Simple merge
Simple merge
Simple merge