SV: omap3isp - H3A auto white balance

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



Hi Laurent,

> Hi Daniel,
> 
> On Tuesday 31 May 2011 12:07:08 Daniel Lundborg wrote:
> 
> [snip]
> 
> > > Any chance you will submit the driver for inclusion in the kernel
?
> > 
> > Yes if there is an interest in it. I can create a patch from your 
> > omap3isp-next-sensors tree if you want.
> 
> That would be nice, thank you.
> 
> --
> Regards,
> 
> Laurent Pinchart

Here's the patch:

--------------------------------------------------------

diff -urNp a/arch/arm/mach-omap2/board-overo.c
b/arch/arm/mach-omap2/board-overo.c
--- a/arch/arm/mach-omap2/board-overo.c	2011-06-01 10:45:32.000000000
+0200
+++ b/arch/arm/mach-omap2/board-overo.c	2011-06-01 10:09:40.000000000
+0200
@@ -28,6 +28,9 @@
 #include <linux/platform_device.h>
 #include <linux/i2c/twl.h>
 #include <linux/regulator/machine.h>
+#include <linux/regulator/fixed.h>
+#include <linux/spi/spi.h>
+#include <linux/interrupt.h>
 
 #include <linux/mtd/mtd.h>
 #include <linux/mtd/nand.h>
@@ -41,10 +44,14 @@
 
 #include <plat/board.h>
 #include <plat/common.h>
+#include <plat/display.h>
+#include <plat/panel-generic-dpi.h>
 #include <mach/gpio.h>
 #include <plat/gpmc.h>
 #include <mach/hardware.h>
 #include <plat/nand.h>
+#include <plat/mcspi.h>
+#include <plat/mux.h>
 #include <plat/usb.h>
 
 #include "mux.h"
@@ -53,17 +60,18 @@
 
 #define OVERO_GPIO_BT_XGATE	15
 #define OVERO_GPIO_W2W_NRESET	16
-#define OVERO_GPIO_PENDOWN	114
-#define OVERO_GPIO_BT_NRESET	164
-#define OVERO_GPIO_USBH_CPEN	168
-#define OVERO_GPIO_USBH_NRESET	183
+#define OVERO_GPIO_PENDOWN 114
+#define OVERO_GPIO_BT_NRESET 164
+#define OVERO_GPIO_USBH_CPEN 168
+#define OVERO_GPIO_USBH_NRESET 183
 
 #define NAND_BLOCK_SIZE SZ_128K
 
-#define OVERO_SMSC911X_CS      5
-#define OVERO_SMSC911X_GPIO    176
-#define OVERO_SMSC911X2_CS     4
-#define OVERO_SMSC911X2_GPIO   65
+#define OVERO_SMSC911X_CS 5
+#define OVERO_SMSC911X_GPIO 176
+#define OVERO_SMSC911X2_CS 4
+#define OVERO_SMSC911X2_GPIO 65
+#define OVERO_GPIO_CAM_RESET 109
 
 #if defined(CONFIG_TOUCHSCREEN_ADS7846) || \
 	defined(CONFIG_TOUCHSCREEN_ADS7846_MODULE)
@@ -124,6 +132,7 @@ static void __init overo_ads7846_init(vo
 static inline void __init overo_ads7846_init(void) { return; }
 #endif
 
+
 #if defined(CONFIG_SMSC911X) || defined(CONFIG_SMSC911X_MODULE)
 
 #include <linux/smsc911x.h>
@@ -233,6 +242,8 @@ static inline void __init overo_init_sms
 static inline void __init overo_init_smsc911x(void) { return; }
 #endif
 
+
+
 static struct mtd_partition overo_nand_partitions[] = {
 	{
 		.name           = "xloader",
@@ -323,6 +334,93 @@ static struct regulator_consumer_supply 
 	.supply			= "vmmc",
 };
 
+#if defined(CONFIG_LEDS_GPIO) || defined(CONFIG_LEDS_GPIO_MODULE)
+#include <linux/leds.h>
+
+static struct gpio_led gpio_leds[] = {
+	{
+		.name			= "overo:red:gpio21",
+		.default_trigger	= "heartbeat",
+		.gpio			= 21,
+		.active_low		= true,
+	},
+	{
+		.name			= "overo:blue:gpio22",
+		.default_trigger	= "none",
+		.gpio			= 22,
+		.active_low		= true,
+	},
+	{
+		.name			= "overo:blue:COM",
+		.default_trigger	= "mmc0",
+		.gpio			= -EINVAL,	/* gets replaced
*/
+		.active_low		= true,
+	},
+};
+
+static struct gpio_led_platform_data gpio_leds_pdata = {
+	.leds		= gpio_leds,
+	.num_leds	= ARRAY_SIZE(gpio_leds),
+};
+
+static struct platform_device gpio_leds_device = {
+	.name	= "leds-gpio",
+	.id	= -1,
+	.dev	= {
+		.platform_data	= &gpio_leds_pdata,
+	},
+};
+
+static void __init overo_init_led(void)
+{
+	platform_device_register(&gpio_leds_device);
+}
+
+#else
+static inline void __init overo_init_led(void) { return; }
+#endif
+
+#if defined(CONFIG_KEYBOARD_GPIO) ||
defined(CONFIG_KEYBOARD_GPIO_MODULE)
+#include <linux/input.h>
+#include <linux/gpio_keys.h>
+
+static struct gpio_keys_button gpio_buttons[] = {
+	{
+		.code			= BTN_0,
+		.gpio			= 23,
+		.desc			= "button0",
+		.wakeup			= 1,
+	},
+	{
+		.code			= BTN_1,
+		.gpio			= 14,
+		.desc			= "button1",
+		.wakeup			= 1,
+	},
+};
+
+static struct gpio_keys_platform_data gpio_keys_pdata = {
+	.buttons	= gpio_buttons,
+	.nbuttons	= ARRAY_SIZE(gpio_buttons),
+};
+
+static struct platform_device gpio_keys_device = {
+	.name	= "gpio-keys",
+	.id	= -1,
+	.dev	= {
+		.platform_data	= &gpio_keys_pdata,
+	},
+};
+
+static void __init overo_init_keys(void)
+{
+	platform_device_register(&gpio_keys_device);
+}
+
+#else
+static inline void __init overo_init_keys(void) { return; }
+#endif
+
 static int overo_twl_gpio_setup(struct device *dev,
 		unsigned gpio, unsigned ngpio)
 {
@@ -330,6 +428,9 @@ static int overo_twl_gpio_setup(struct d
 
 	overo_vmmc1_supply.dev = mmc[0].dev;
 
+	/* TWL4030_GPIO_MAX + 1 == ledB, PMU_STAT (out, active low LED)
*/
+	gpio_leds[2].gpio = gpio + TWL4030_GPIO_MAX + 1;
+
 	return 0;
 }
 
@@ -337,6 +438,7 @@ static struct twl4030_gpio_platform_data
 	.gpio_base	= OMAP_MAX_GPIO_LINES,
 	.irq_base	= TWL4030_GPIO_IRQ_BASE,
 	.irq_end	= TWL4030_GPIO_IRQ_END,
+	.use_leds	= true,
 	.setup		= overo_twl_gpio_setup,
 };
 
@@ -393,6 +495,7 @@ static int __init overo_i2c_init(void)
 			ARRAY_SIZE(overo_i2c_boardinfo));
 	/* i2c2 pins are used for gpio */
 	omap_register_i2c_bus(3, 400, NULL, 0);
+
 	return 0;
 }
 
@@ -415,7 +518,7 @@ static void __init overo_init_irq(void)
 	omap_board_config_size = ARRAY_SIZE(overo_config);
 	omap2_init_common_infrastructure();
 	omap2_init_common_devices(mt46h32m32lf6_sdrc_params,
-				  mt46h32m32lf6_sdrc_params);
+			     mt46h32m32lf6_sdrc_params);
 	omap_init_irq();
 }
 
@@ -436,16 +539,292 @@ static const struct ehci_hcd_omap_platfo
 
 #ifdef CONFIG_OMAP_MUX
 static struct omap_board_mux board_mux[] __initdata = {
+  // SDR/DDR RAM chips
+  OMAP3_MUX(SDRC_CKE0, OMAP_PIN_OUTPUT | OMAP_MUX_MODE0),
+  OMAP3_MUX(SDRC_CKE1, OMAP_PIN_OUTPUT | OMAP_MUX_MODE0),
+  
+  // Camera driver
+  // Camera - Parallel Data
+  OMAP3_MUX(CAM_D0, OMAP_MUX_MODE0 | OMAP_PIN_INPUT),
+  OMAP3_MUX(CAM_D1, OMAP_MUX_MODE0 | OMAP_PIN_INPUT),
+  OMAP3_MUX(CAM_D2, OMAP_MUX_MODE0 | OMAP_PIN_INPUT),
+  OMAP3_MUX(CAM_D3, OMAP_MUX_MODE0 | OMAP_PIN_INPUT),
+  OMAP3_MUX(CAM_D4, OMAP_MUX_MODE0 | OMAP_PIN_INPUT),
+  OMAP3_MUX(CAM_D5, OMAP_MUX_MODE0 | OMAP_PIN_INPUT),
+  OMAP3_MUX(CAM_D6, OMAP_MUX_MODE0 | OMAP_PIN_INPUT),
+  OMAP3_MUX(CAM_D7, OMAP_MUX_MODE0 | OMAP_PIN_INPUT),
+  OMAP3_MUX(CAM_D8, OMAP_MUX_MODE0 | OMAP_PIN_INPUT),
+  OMAP3_MUX(CAM_D9, OMAP_MUX_MODE0 | OMAP_PIN_INPUT),
+  OMAP3_MUX(CAM_PCLK, OMAP_MUX_MODE0 | OMAP_PIN_INPUT),
+  
+  // Camera - HS/VS signals
+  OMAP3_MUX(CAM_HS, OMAP_MUX_MODE0 | OMAP_PIN_INPUT),
+  OMAP3_MUX(CAM_VS, OMAP_MUX_MODE0 | OMAP_PIN_INPUT),
+  
+  // Camera - XCLK
+  OMAP3_MUX(CAM_XCLKA, OMAP_MUX_MODE0 | OMAP_PIN_OUTPUT),
+
+  OMAP3_MUX(CAM_FLD, OMAP_PIN_OUTPUT | OMAP_MUX_MODE2), //
cam_global_reset
+  OMAP3_MUX(CAM_WEN, OMAP_PIN_OUTPUT | OMAP_MUX_MODE2), // cam_shutter
+  OMAP3_MUX(CAM_D10, OMAP_PIN_OUTPUT | OMAP_MUX_MODE4), // gpio_109 =
CAM_RESET
+  
 	{ .reg_offset = OMAP_MUX_TERMINATOR },
 };
+#else
+#define board_mux	NULL
 #endif
 
 static struct omap_musb_board_data musb_board_data = {
 	.interface_type		= MUSB_INTERFACE_ULPI,
+#if defined(CONFIG_USB_MUSB_OTG)
 	.mode			= MUSB_OTG,
+#elif defined(CONFIG_USB_GADGET_MUSB_HDRC)
+	.mode			= MUSB_PERIPHERAL,
+#else
+	.mode			= MUSB_HOST,
+#endif
 	.power			= 100,
 };
 
+static inline int do_gpio_request_out(int gpio, const char *gpio_id)
+{
+  int ret;
+  
+  omap_mux_init_gpio(gpio, OMAP_PIN_OUTPUT);
+  ret = gpio_request(gpio, gpio_id);
+  if (ret < 0)
+    return ret;
+    
+  ret = gpio_direction_output(gpio, 1);
+  if (ret < 0)
+    return ret;
+  
+  gpio_export(gpio, 0);
+  
+  return ret;
+}
+
+static inline int do_gpio_request_in(int gpio, const char *gpio_id)
+{
+  int ret;
+  
+  omap_mux_init_gpio(gpio, OMAP_PIN_INPUT);
+  ret = gpio_request(gpio, gpio_id);
+  if (ret < 0)
+    return ret;
+    
+  ret = gpio_direction_input(gpio);
+  if (ret < 0)
+    return ret;
+  
+  gpio_export(gpio, 0);
+  
+  return ret;
+}
+
+static void overo_init_gpios(void)
+{
+  if (do_gpio_request_out(OVERO_GPIO_W2W_NRESET,
"OVERO_GPIO_W2W_NRESET") < 0 ||
+      do_gpio_request_out(OVERO_GPIO_BT_XGATE, "OVERO_GPIO_BT_XGATE") <
0 ||
+      do_gpio_request_out(OVERO_GPIO_BT_NRESET, "OVERO_GPIO_BT_NRESET")
< 0 ||
+      do_gpio_request_out(OVERO_GPIO_USBH_CPEN, "OVERO_GPIO_USBH_CPEN")
< 0 ||
+      do_gpio_request_out(OVERO_GPIO_CAM_RESET, "OVERO_GPIO_CAM_RESET")
< 0)
+    return;
+
+  gpio_set_value(OVERO_GPIO_W2W_NRESET, 0);
+	udelay(10);
+	gpio_set_value(OVERO_GPIO_W2W_NRESET, 1);
+
+  gpio_set_value(OVERO_GPIO_CAM_RESET, 1); // Set cam reset to 1 (off)
+}
+
+#if defined(CONFIG_VIDEO_OMAP3) && defined(CONFIG_VIDEO_MT9V034)
+
+#include <linux/videodev2.h>
+#include <media/v4l2-subdev.h>
+#include <media/mt9v034.h>
+#include <../drivers/media/video/omap3isp/isp.h>
+#include <../drivers/media/video/omap3isp/ispreg.h>
+#include "devices.h"
+
+#define OVERO_CAMERA_XCLK ISP_XCLK_A
+
+static void print_isp_reg(struct v4l2_subdev *s, const char *name, enum
isp_mem_resources isp_mmio_range, u32 reg_offset)
+{
+  u32 value;
+	struct isp_device *isp = v4l2_dev_to_isp_device(s->v4l2_dev);
+	
+  value = isp_reg_readl(isp, isp_mmio_range, reg_offset);
+  printk(KERN_ALERT "Omap ISP register %s = 0x%x\n", name, value);
+}
+
+static inline void print_isp_regs(struct v4l2_subdev *s, const char
*str)
+{
+	printk(KERN_ALERT "%s\n", str);
+  print_isp_reg(s, "TCTRL_CTRL", OMAP3_ISP_IOMEM_MAIN, ISP_TCTRL_CTRL);
+  print_isp_reg(s, "TCTRL_SHUT_DELAY", OMAP3_ISP_IOMEM_MAIN,
ISP_TCTRL_SHUT_DELAY);
+  print_isp_reg(s, "TCTRL_SHUT_LENGTH", OMAP3_ISP_IOMEM_MAIN,
ISP_TCTRL_SHUT_LENGTH);
+  print_isp_reg(s, "TCTRL_GRESET_LENGTH", OMAP3_ISP_IOMEM_MAIN,
ISP_TCTRL_GRESET_LENGTH);
+  print_isp_reg(s, "TCTRL_FRAME", OMAP3_ISP_IOMEM_MAIN,
ISP_TCTRL_FRAME);
+  print_isp_reg(s, "TCTRL_PSTRB_DELAY", OMAP3_ISP_IOMEM_MAIN,
ISP_TCTRL_PSTRB_DELAY);
+  print_isp_reg(s, "TCTRL_STRB_DELAY", OMAP3_ISP_IOMEM_MAIN,
ISP_TCTRL_STRB_DELAY);
+  print_isp_reg(s, "TCTRL_PSTRB_LENGTH", OMAP3_ISP_IOMEM_MAIN,
ISP_TCTRL_PSTRB_LENGTH);
+  print_isp_reg(s, "TCTRL_STRB_LENGTH", OMAP3_ISP_IOMEM_MAIN,
ISP_TCTRL_STRB_LENGTH);
+  print_isp_reg(s, "TCTRL_TCTRL_GRESET_LENGTH", OMAP3_ISP_IOMEM_MAIN,
ISP_TCTRL_GRESET_LENGTH);
+  print_isp_reg(s, "TCTRL_PSTRB_REPLAY", OMAP3_ISP_IOMEM_MAIN,
ISP_TCTRL_PSTRB_REPLAY);
+  print_isp_reg(s, "CTRL", OMAP3_ISP_IOMEM_MAIN, ISP_CTRL);
+  print_isp_reg(s, "SECURE", OMAP3_ISP_IOMEM_MAIN, ISP_SECURE);
+  print_isp_reg(s, "PID", OMAP3_ISP_IOMEM_CCDC, ISPCCDC_PID);
+  print_isp_reg(s, "SYN_MODE", OMAP3_ISP_IOMEM_CCDC, ISPCCDC_SYN_MODE);
+  print_isp_reg(s, "PCR", OMAP3_ISP_IOMEM_CCDC, ISPCCDC_PCR);
+  print_isp_reg(s, "HD_VD_WID", OMAP3_ISP_IOMEM_CCDC,
ISPCCDC_HD_VD_WID);
+  print_isp_reg(s, "PIX_LINES", OMAP3_ISP_IOMEM_CCDC,
ISPCCDC_PIX_LINES);
+  print_isp_reg(s, "HORZ_INFO", OMAP3_ISP_IOMEM_CCDC,
ISPCCDC_HORZ_INFO);
+  print_isp_reg(s, "VERT_START", OMAP3_ISP_IOMEM_CCDC,
ISPCCDC_VERT_START);
+  print_isp_reg(s, "VERT_LINES", OMAP3_ISP_IOMEM_CCDC,
ISPCCDC_VERT_LINES);
+  print_isp_reg(s, "CULLING", OMAP3_ISP_IOMEM_CCDC, ISPCCDC_CULLING);
+  print_isp_reg(s, "HSIZE_OFF", OMAP3_ISP_IOMEM_CCDC,
ISPCCDC_HSIZE_OFF);
+  print_isp_reg(s, "SDOFST", OMAP3_ISP_IOMEM_CCDC, ISPCCDC_SDOFST);
+  print_isp_reg(s, "SDR_ADDR", OMAP3_ISP_IOMEM_CCDC, ISPCCDC_SDR_ADDR);
+  print_isp_reg(s, "CLAMP", OMAP3_ISP_IOMEM_CCDC, ISPCCDC_CLAMP);
+  print_isp_reg(s, "DCSUB", OMAP3_ISP_IOMEM_CCDC, ISPCCDC_DCSUB);
+  print_isp_reg(s, "COLPTN", OMAP3_ISP_IOMEM_CCDC, ISPCCDC_COLPTN);
+  print_isp_reg(s, "BLKCMP", OMAP3_ISP_IOMEM_CCDC, ISPCCDC_BLKCMP);
+  print_isp_reg(s, "FPC", OMAP3_ISP_IOMEM_CCDC, ISPCCDC_FPC);
+  print_isp_reg(s, "FPC_ADDR", OMAP3_ISP_IOMEM_CCDC, ISPCCDC_FPC_ADDR);
+  print_isp_reg(s, "VDINT", OMAP3_ISP_IOMEM_CCDC, ISPCCDC_VDINT);
+  print_isp_reg(s, "ALAW", OMAP3_ISP_IOMEM_CCDC, ISPCCDC_ALAW);
+  print_isp_reg(s, "REC656IF", OMAP3_ISP_IOMEM_CCDC, ISPCCDC_REC656IF);
+  print_isp_reg(s, "CFG", OMAP3_ISP_IOMEM_CCDC, ISPCCDC_CFG);
+  print_isp_reg(s, "FMTCFG", OMAP3_ISP_IOMEM_CCDC, ISPCCDC_FMTCFG);
+  print_isp_reg(s, "FMT_HORZ", OMAP3_ISP_IOMEM_CCDC, ISPCCDC_FMT_HORZ);
+  print_isp_reg(s, "FMT_VERT", OMAP3_ISP_IOMEM_CCDC, ISPCCDC_FMT_VERT);
+  print_isp_reg(s, "FMT_ADDR0", OMAP3_ISP_IOMEM_CCDC,
ISPCCDC_FMT_ADDR0);
+  print_isp_reg(s, "FMT_ADDR1", OMAP3_ISP_IOMEM_CCDC,
ISPCCDC_FMT_ADDR1);
+  print_isp_reg(s, "FMT_ADDR2", OMAP3_ISP_IOMEM_CCDC,
ISPCCDC_FMT_ADDR2);
+  print_isp_reg(s, "FMT_ADDR3", OMAP3_ISP_IOMEM_CCDC,
ISPCCDC_FMT_ADDR3);
+  print_isp_reg(s, "FMT_ADDR4", OMAP3_ISP_IOMEM_CCDC,
ISPCCDC_FMT_ADDR4);
+  print_isp_reg(s, "FMT_ADDR5", OMAP3_ISP_IOMEM_CCDC,
ISPCCDC_FMT_ADDR5);
+  print_isp_reg(s, "FMT_ADDR6", OMAP3_ISP_IOMEM_CCDC,
ISPCCDC_FMT_ADDR6);
+  print_isp_reg(s, "FMT_ADDR7", OMAP3_ISP_IOMEM_CCDC,
ISPCCDC_FMT_ADDR7);
+  print_isp_reg(s, "PRGEVEN0", OMAP3_ISP_IOMEM_CCDC, ISPCCDC_PRGEVEN0);
+  print_isp_reg(s, "PRGEVEN1", OMAP3_ISP_IOMEM_CCDC, ISPCCDC_PRGEVEN1);
+  print_isp_reg(s, "PRGODD0", OMAP3_ISP_IOMEM_CCDC, ISPCCDC_PRGODD0);
+  print_isp_reg(s, "PRGODD1", OMAP3_ISP_IOMEM_CCDC, ISPCCDC_PRGODD1);
+  print_isp_reg(s, "VP_OUT", OMAP3_ISP_IOMEM_CCDC, ISPCCDC_VP_OUT);
+  print_isp_reg(s, "LSC_CONFIG", OMAP3_ISP_IOMEM_CCDC,
ISPCCDC_LSC_CONFIG);
+  print_isp_reg(s, "LSC_INITIAL", OMAP3_ISP_IOMEM_CCDC,
ISPCCDC_LSC_INITIAL);
+  print_isp_reg(s, "LSC_TABLE_BASE", OMAP3_ISP_IOMEM_CCDC,
ISPCCDC_LSC_TABLE_BASE);
+  print_isp_reg(s, "LSC_TABLE_OFFSET", OMAP3_ISP_IOMEM_CCDC,
ISPCCDC_LSC_TABLE_OFFSET);
+}
+
+void overo_camera_configure(struct v4l2_subdev *subdev)
+{
+	struct isp_device *isp =
v4l2_dev_to_isp_device(subdev->v4l2_dev);
+  
+  isp_reg_clr_set(isp, OMAP3_ISP_IOMEM_MAIN, ISP_TCTRL_CTRL,
~0x9a1b63ff, 0x98036000); // Set CAM_GLOBAL_RESET pin as output, enable
shutter, set DIVC = 216
+  isp_reg_clr(isp, OMAP3_ISP_IOMEM_MAIN, ISP_TCTRL_SHUT_DELAY,
0x01ffffff);  // Set no shutter delay  
+  isp_reg_clr_set(isp, OMAP3_ISP_IOMEM_MAIN, ISP_TCTRL_SHUT_LENGTH,
0x01ffffff, 0x000003e8); // Set shutter signal length to 1000 (=> 1000 *
1/216MHz * 216 = 1 ms)
+  isp_reg_clr_set(isp, OMAP3_ISP_IOMEM_MAIN, ISP_TCTRL_GRESET_LENGTH,
0x01ffffff, 0x000003e8); // Set CAM_GLOBAL_RESET signal length to 1000
(=> 1000 * 1/216MHz * 216 = 1 ms)
+  isp_reg_clr(isp, OMAP3_ISP_IOMEM_CCDC, ISPCCDC_LSC_CONFIG,
0xffffffff);  // Disable LSC
+  isp_reg_clr(isp, OMAP3_ISP_IOMEM_CCDC, ISPCCDC_FMTCFG, 0xffffffff);
// Disable Preview, H3A, HIST
+}
+
+static void overo_camera_take_picture(struct v4l2_subdev *subdev, int
nr_pics, int exposure_time_ms)
+{
+	struct isp_device *isp =
v4l2_dev_to_isp_device(subdev->v4l2_dev);
+  u32 isp_value;
+  
+//print_isp_regs(subdev, "before take_pic");
+
+  for (; nr_pics >= 0; --nr_pics)  // Take a picture again?
+  {
+    isp_reg_clr_set(isp, OMAP3_ISP_IOMEM_MAIN, ISP_TCTRL_CTRL, 0,
0x00e00000);  // Enable shutter (SHUTEN bit = 1)
+    isp_reg_clr_set(isp, OMAP3_ISP_IOMEM_MAIN, ISP_TCTRL_CTRL, 0,
0x20000000);  // Start generation of CAM_GLOBAL_RESET signal (GRESETEN
bit = 1)
+
+    msleep(exposure_time_ms + 2);
+    
+    do
+    {
+      usleep_range(10, 20);
+      isp_value = isp_reg_readl(isp, OMAP3_ISP_IOMEM_CCDC,
ISPCCDC_PCR);
+    }
+    while ((isp_value & 0x2) != 0);  // Wait until PCR is not busy (bit
2)
+
+    usleep_range(1800, 2000);
+  }
+  
+//  print_isp_regs(subdev, "after take_pic");
+}
+
+static void overo_camera_set_clock(struct v4l2_subdev *subdev, unsigned
int rate)
+{
+	struct isp_device *isp =
v4l2_dev_to_isp_device(subdev->v4l2_dev);
+
+	isp->platform_cb.set_xclk(isp, rate, OVERO_CAMERA_XCLK);
+}
+
+static void overo_camera_reset(void)
+{
+  gpio_set_value(OVERO_GPIO_CAM_RESET, 0);
+  
+  msleep(1);
+  
+  gpio_set_value(OVERO_GPIO_CAM_RESET, 1);
+}
+
+
+static struct mt9v034_platform_data overo_mt9v034_platform_data = {
+  .clk_pol = 0,
+	.set_clock = overo_camera_set_clock,
+	.take_picture = overo_camera_take_picture,
+  .configure = overo_camera_configure,
+  .reset = overo_camera_reset,
+};
+
+static struct i2c_board_info overo_camera_i2c_devices[] = {
+	{
+		I2C_BOARD_INFO("mt9v034", 0x48),
+		.platform_data = &overo_mt9v034_platform_data,
+	},
+};
+
+static struct isp_subdev_i2c_board_info overo_camera_i2c_subdevs[] = {
+	{
+		.board_info = &overo_camera_i2c_devices[0],
+		.i2c_adapter_id = 3,
+	},
+	{ NULL, 0, },
+};
+
+static struct isp_v4l2_subdevs_group overo_camera_subdevs[] = {
+	{
+		.subdevs = overo_camera_i2c_subdevs,
+		.interface = ISP_INTERFACE_PARALLEL,
+		.bus = {
+  		.parallel = {
+    		.data_lane_shift = 0,
+    		.clk_pol = 0,
+    		.bridge = ISPCTRL_PAR_BRIDGE_DISABLE,
+		  } 
+		},
+	},
+	{ NULL, 0, },
+};
+
+static struct isp_platform_data overo_isp_platform_data = {
+	.subdevs = overo_camera_subdevs,
+};
+
+void __init overo_camera_init(void)
+{
+	if (omap3_init_camera(&overo_isp_platform_data) < 0)
+		printk(KERN_ALERT "%s: Unable to register camera
platform device\n", __func__);
+}
+#else
+void __init overo_camera_init(void) {}
+#endif  // CONFIG_VIDEO_OMAP3 && CONFIG_MT9V034
+
 static void __init overo_init(void)
 {
 	omap3_mux_init(board_mux, OMAP_PACKAGE_CBB);
@@ -457,48 +836,13 @@ static void __init overo_init(void)
 	usb_ehci_init(&ehci_pdata);
 	overo_ads7846_init();
 	overo_init_smsc911x();
-
-	/* Ensure SDRC pins are mux'd for self-refresh */
-	omap_mux_init_signal("sdrc_cke0", OMAP_PIN_OUTPUT);
-	omap_mux_init_signal("sdrc_cke1", OMAP_PIN_OUTPUT);
-
-	if ((gpio_request(OVERO_GPIO_W2W_NRESET,
-			  "OVERO_GPIO_W2W_NRESET") == 0) &&
-	    (gpio_direction_output(OVERO_GPIO_W2W_NRESET, 1) == 0)) {
-		gpio_export(OVERO_GPIO_W2W_NRESET, 0);
-		gpio_set_value(OVERO_GPIO_W2W_NRESET, 0);
-		udelay(10);
-		gpio_set_value(OVERO_GPIO_W2W_NRESET, 1);
-	} else {
-		printk(KERN_ERR "could not obtain gpio for "
-					"OVERO_GPIO_W2W_NRESET\n");
-	}
-
-	if ((gpio_request(OVERO_GPIO_BT_XGATE, "OVERO_GPIO_BT_XGATE") ==
0) &&
-	    (gpio_direction_output(OVERO_GPIO_BT_XGATE, 0) == 0))
-		gpio_export(OVERO_GPIO_BT_XGATE, 0);
-	else
-		printk(KERN_ERR "could not obtain gpio for
OVERO_GPIO_BT_XGATE\n");
-
-	if ((gpio_request(OVERO_GPIO_BT_NRESET, "OVERO_GPIO_BT_NRESET")
== 0) &&
-	    (gpio_direction_output(OVERO_GPIO_BT_NRESET, 1) == 0)) {
-		gpio_export(OVERO_GPIO_BT_NRESET, 0);
-		gpio_set_value(OVERO_GPIO_BT_NRESET, 0);
-		mdelay(6);
-		gpio_set_value(OVERO_GPIO_BT_NRESET, 1);
-	} else {
-		printk(KERN_ERR "could not obtain gpio for "
-					"OVERO_GPIO_BT_NRESET\n");
-	}
-
-	if ((gpio_request(OVERO_GPIO_USBH_CPEN, "OVERO_GPIO_USBH_CPEN")
== 0) &&
-	    (gpio_direction_output(OVERO_GPIO_USBH_CPEN, 1) == 0))
-		gpio_export(OVERO_GPIO_USBH_CPEN, 0);
-	else
-		printk(KERN_ERR "could not obtain gpio for "
-					"OVERO_GPIO_USBH_CPEN\n");
+	overo_init_led();
+	overo_init_keys();
+	overo_init_gpios();
+  	overo_camera_init();
 }
 
+
 MACHINE_START(OVERO, "Gumstix Overo")
 	.boot_params	= 0x80000100,
 	.map_io		= omap3_map_io,
diff -urNp a/drivers/media/video/Kconfig b/drivers/media/video/Kconfig
--- a/drivers/media/video/Kconfig	2011-06-01 10:46:46.000000000
+0200
+++ b/drivers/media/video/Kconfig	2011-06-01 10:06:01.000000000
+0200
@@ -344,6 +344,13 @@ config VIDEO_MT9V032
 	  This is a Video4Linux2 sensor-level driver for the Micron
 	  MT9V032 752x480 CMOS sensor.
 
+config VIDEO_MT9V034
+	tristate "Aptina MT9V034 sensor support"
+	depends on I2C && VIDEO_V4L2 && VIDEO_V4L2_SUBDEV_API
+	---help---
+	  This is a Video4Linux2 sensor-level driver for the Aptina
+	  MT9V032 752x480 CMOS sensor. Only for snapshot.
+
 config VIDEO_TCM825X
 	tristate "TCM825x camera sensor support"
 	depends on I2C && VIDEO_V4L2
diff -urNp a/drivers/media/video/Makefile b/drivers/media/video/Makefile
--- a/drivers/media/video/Makefile	2011-06-01 10:46:46.000000000
+0200
+++ b/drivers/media/video/Makefile	2011-06-01 10:06:45.000000000
+0200
@@ -67,6 +67,7 @@ obj-$(CONFIG_VIDEO_TCM825X) += tcm825x.o
 obj-$(CONFIG_VIDEO_TVEEPROM) += tveeprom.o
 obj-$(CONFIG_VIDEO_MT9V011) += mt9v011.o
 obj-$(CONFIG_VIDEO_MT9V032) += mt9v032.o
+obj-$(CONFIG_VIDEO_MT9V034) += mt9v034.o
 obj-$(CONFIG_VIDEO_SR030PC30)	+= sr030pc30.o
 obj-$(CONFIG_VIDEO_NOON010PC30)	+= noon010pc30.o
 
diff -urNp a/drivers/media/video/mt9v034.c
b/drivers/media/video/mt9v034.c
--- a/drivers/media/video/mt9v034.c	1970-01-01 01:00:00.000000000
+0100
+++ b/drivers/media/video/mt9v034.c	2011-05-30 15:04:44.000000000
+0200
@@ -0,0 +1,1114 @@
+/*
+ * Driver for MT9V034 CMOS Image Sensor from Micron
+ *
+ * Copyright (C) 2011, Daniel Lundborg <daniel.lundborg@xxxxxxxxx>
+ *
+ * Copyright (C) 2010, Laurent Pinchart
<laurent.pinchart@xxxxxxxxxxxxxxxx>
+ *
+ * Based on the MT9M001 driver,
+ *
+ * Copyright (C) 2008, Guennadi Liakhovetski <kernel@xxxxxxxxxxxxxx>
+ *
+ * This program is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License version 2 as
+ * published by the Free Software Foundation.
+ */
+
+#include <linux/delay.h>
+#include <linux/i2c.h>
+#include <linux/log2.h>
+#include <linux/mutex.h>
+#include <linux/slab.h>
+#include <linux/videodev2.h>
+#include <linux/v4l2-mediabus.h>
+#include <linux/interrupt.h>
+#include <linux/irq.h>
+#include <linux/gpio.h>
+
+#include <media/mt9v034.h>
+#include <media/v4l2-ctrls.h>
+#include <media/v4l2-device.h>
+#include <media/v4l2-subdev.h>
+
+#define MT9V034_CHIP_VERSION                         0x00
+#define		MT9V034_CHIP_ID 0x1324
+#define MT9V034_COLUMN_START_CONTEXT_A               0x01
+#define		MT9V034_COLUMN_START_MIN		1
+#define		MT9V034_COLUMN_START_DEF		2
+#define		MT9V034_COLUMN_START_MAX		752
+#define MT9V034_ROW_START_CONTEXT_A                  0x02
+#define		MT9V034_ROW_START_MIN			4
+#define		MT9V034_ROW_START_DEF			10
+#define		MT9V034_ROW_START_MAX			482
+#define MT9V034_WIN_HEIGHT_CONTEXT_A                 0x03
+#define		MT9V034_WINDOW_HEIGHT_MIN		1
+#define		MT9V034_WINDOW_HEIGHT_DEF		480
+#define		MT9V034_WINDOW_HEIGHT_MAX		480
+#define MT9V034_WIN_WIDTH_CONTEXT_A                  0x04
+#define		MT9V034_WINDOW_WIDTH_MIN		1
+#define		MT9V034_WINDOW_WIDTH_DEF		752
+#define		MT9V034_WINDOW_WIDTH_MAX		752
+#define MT9V034_HORIZONTAL_BLANCING_CONTEXT_A        0x05
+#define		MT9V034_HORIZONTAL_BLANKING_MIN		43
+#define		MT9V034_HORIZONTAL_BLANKING_MAX		1023
+#define MT9V034_VERTICAL_BLANCING_CONTEXT_A          0x06
+#define		MT9V034_VERTICAL_BLANKING_MIN		4
+#define		MT9V034_VERTICAL_BLANKING_MAX		3000
+#define MT9V034_CHIP_CONTROL                         0x07
+#define 	MT9V034_CHIP_CONTROL_MASTER_MODE	(1 << 3)
+#define   MT9V034_CHIP_CONTROL_SNAPSHOT_MODE (3 << 3)
+#define 	MT9V034_CHIP_CONTROL_DOUT_ENABLE	(1 << 7)
+#define 	MT9V034_CHIP_CONTROL_SEQUENTIAL		(1 << 8)
+#define   MT9V034_CHIP_CONTROL_RESERVED (1 << 9)
+#define MT9V034_COARSE_SHUTTER_WIDTH1_CONTEXT_A      0x08
+#define MT9V034_COARSE_SHUTTER_WIDTH2_CONTEXT_A      0x09
+#define MT9V034_SHUTTER_WIDTH_CTRL_CONTEXT_A         0x0a
+#define MT9V034_COARSE_TOTAL_SHUTTER_WIDTH_CONTEXT_A 0x0b
+#define		MT9V034_TOTAL_SHUTTER_WIDTH_MIN_US	1000
+#define		MT9V034_TOTAL_SHUTTER_WIDTH_DEF_US	8000
+#define		MT9V034_TOTAL_SHUTTER_WIDTH_MAX_US	500000
+#define MT9V034_RESET                                0x0c
+#define MT9V034_READ_MODE_CONTEXT_A                  0x0d
+#define		MT9V034_READ_MODE_ROW_BIN_MASK		(3 << 0)
+#define		MT9V034_READ_MODE_ROW_BIN_SHIFT		0
+#define		MT9V034_READ_MODE_COLUMN_BIN_MASK	(3 << 2)
+#define		MT9V034_READ_MODE_COLUMN_BIN_SHIFT	2
+#define		MT9V034_READ_MODE_ROW_FLIP		(1 << 4)
+#define		MT9V034_READ_MODE_COLUMN_FLIP		(1 << 5)
+#define		MT9V034_READ_MODE_DARK_COLUMNS		(1 << 6)
+#define		MT9V034_READ_MODE_DARK_ROWS		(1 << 7)
+#define MT9V034_READ_MODE_CONTEXT_B                  0x0e
+#define MT9V034_SENSOR_TYPE_HDR_ENABLE               0x0f
+#define MT9V034_LED_OUT_CTRL                         0x1b
+#define MT9V034_COMPANDING                           0x1c
+#define MT9V034_VREF_ADC_CTRL                        0x2c
+#define MT9V034_V1_CONTROL_CONTEXT_A                 0x31
+#define MT9V034_V2_CONTROL_CONTEXT_A                 0x32
+#define MT9V034_V3_CONTROL_CONTEXT_A                 0x33
+#define MT9V034_V4_CONTROL_CONTEXT_A                 0x34
+#define MT9V034_ANALOG_GAIN_CONTEXT_A                0x35
+#define		MT9V034_ANALOG_GAIN_MIN			1
+#define		MT9V034_ANALOG_GAIN_DEF			1
+#define		MT9V034_ANALOG_GAIN_MAX			4
+#define MT9V034_ANALOG_GAIN_CONTEXT_B                0x36
+#define MT9V034_V1_CONTROL_CONTEXT_B                 0x39
+#define MT9V034_V2_CONTROL_CONTEXT_B                 0x3a
+#define MT9V034_V3_CONTROL_CONTEXT_B                 0x3b
+#define MT9V034_V4_CONTROL_CONTEXT_B                 0x3c
+#define MT9V034_FRAME_DARK_AVERAGE                   0x42
+#define MT9V034_DARK_AVERAGE_THRESHOLD               0x46
+#define MT9V034_BL_CALIB_CTRL                        0x47
+#define MT9V034_BLACK_LEVEL_CALIBRATION_VALUE        0x48
+#define MT9V034_BL_CALIB_STEP_SIZE                   0x4c
+#define MT9V034_ROW_NOISE_CORR_CONTROL               0x70
+#define MT9V034_ROW_NOISE_CONSTANT                   0x71
+#define MT9V034_PIXCLK_FV_LV_CTRL                    0x72
+#define 	MT9V034_PIXEL_CLOCK_INV_LINE		(1 << 0)
+#define 	MT9V034_PIXEL_CLOCK_INV_FRAME		(1 << 1)
+#define 	MT9V034_PIXEL_CLOCK_XOR_LINE		(1 << 2)
+#define 	MT9V034_PIXEL_CLOCK_CONT_LINE		(1 << 3)
+#define 	MT9V034_PIXEL_CLOCK_INV_PXL_CLK		(1 << 4)
+#define MT9V034_DIGITAL_TEST_PATTERN                 0x7f
+#define 	MT9V034_TEST_PATTERN_DATA_MASK		(1023 << 0)
+#define 	MT9V034_TEST_PATTERN_DATA_SHIFT		0
+#define 	MT9V034_TEST_PATTERN_USE_DATA		(1 << 10)
+#define 	MT9V034_TEST_PATTERN_GRAY_MASK		(3 << 11)
+#define 	MT9V034_TEST_PATTERN_GRAY_NONE		(0 << 11)
+#define 	MT9V034_TEST_PATTERN_GRAY_VERTICAL	(1 << 11)
+#define 	MT9V034_TEST_PATTERN_GRAY_HORIZONTAL	(2 << 11)
+#define 	MT9V034_TEST_PATTERN_GRAY_DIAGONAL	(3 << 11)
+#define 	MT9V034_TEST_PATTERN_ENABLE		(1 << 13)
+#define 	MT9V034_TEST_PATTERN_FLIP		(1 << 14)
+#define MT9V034_TILE_WEIGHT_GAIN_X0_Y0               0x80
+#define MT9V034_TILE_WEIGHT_GAIN_X1_Y0               0x81
+#define MT9V034_TILE_WEIGHT_GAIN_X2_Y0               0x82
+#define MT9V034_TILE_WEIGHT_GAIN_X3_Y0               0x83
+#define MT9V034_TILE_WEIGHT_GAIN_X4_Y0               0x84
+#define MT9V034_TILE_WEIGHT_GAIN_X0_Y1               0x85
+#define MT9V034_TILE_WEIGHT_GAIN_X1_Y1               0x86
+#define MT9V034_TILE_WEIGHT_GAIN_X2_Y1               0x87
+#define MT9V034_TILE_WEIGHT_GAIN_X3_Y1               0x88
+#define MT9V034_TILE_WEIGHT_GAIN_X4_Y1               0x89
+#define MT9V034_TILE_WEIGHT_GAIN_X0_Y2               0x8a
+#define MT9V034_TILE_WEIGHT_GAIN_X1_Y2               0x8b
+#define MT9V034_TILE_WEIGHT_GAIN_X2_Y2               0x8c
+#define MT9V034_TILE_WEIGHT_GAIN_X3_Y2               0x8d
+#define MT9V034_TILE_WEIGHT_GAIN_X4_Y2               0x8e
+#define MT9V034_TILE_WEIGHT_GAIN_X0_Y3               0x8f
+#define MT9V034_TILE_WEIGHT_GAIN_X1_Y3               0x90
+#define MT9V034_TILE_WEIGHT_GAIN_X2_Y3               0x91
+#define MT9V034_TILE_WEIGHT_GAIN_X3_Y3               0x92
+#define MT9V034_TILE_WEIGHT_GAIN_X4_Y3               0x93
+#define MT9V034_TILE_WEIGHT_GAIN_X0_Y4               0x94
+#define MT9V034_TILE_WEIGHT_GAIN_X1_Y4               0x95
+#define MT9V034_TILE_WEIGHT_GAIN_X2_Y4               0x96
+#define MT9V034_TILE_WEIGHT_GAIN_X3_Y4               0x97
+#define MT9V034_TILE_WEIGHT_GAIN_X4_Y4               0x98
+#define MT9V034_TILE_COORD_X0_5                      0x99
+#define MT9V034_TILE_COORD_X1_5                      0x9a
+#define MT9V034_TILE_COORD_X2_5                      0x9b
+#define MT9V034_TILE_COORD_X3_5                      0x9c
+#define MT9V034_TILE_COORD_X4_5                      0x9d
+#define MT9V034_TILE_COORD_X5_5                      0x9e
+#define MT9V034_TILE_COORD_Y0_5                      0x9f
+#define MT9V034_TILE_COORD_Y1_5                      0xa0
+#define MT9V034_TILE_COORD_Y2_5                      0xa1
+#define MT9V034_TILE_COORD_Y3_5                      0xa2
+#define MT9V034_TILE_COORD_Y4_5                      0xa3
+#define MT9V034_TILE_COORD_Y5_5                      0xa4
+#define MT9V034_AEC_AGC_DESIRED_BIN                  0xa5
+#define MT9V034_AEC_UPDATE_FREQUENCY                 0xa6
+#define MT9V034_AEC_LPF                              0xa8
+#define MT9V034_AGC_UPDATE_FREQUENCY                 0xa9
+#define MT9V034_AGC_LPF                              0xaa
+#define MT9V034_MAX_ANALOG_GAIN                      0xab
+#define MT9V034_AEC_MINIMUM_EXPOSURE                 0xac
+#define MT9V034_AEC_MAXIMUM_EXPOSURE                 0xad
+#define MT9V034_BIN_DIFFERENCE_THRESHOLD             0xae
+#define MT9V034_AEC_AGC_ENABLE_A_B                   0xaf
+#define 	MT9V034_AEC_ENABLE			(1 << 0)
+#define 	MT9V034_AGC_ENABLE			(1 << 1)
+#define MT9V034_AEC_AGC_PIX_COUNT                    0xb0
+#define MT9V034_LVDS_MASTER_CTRL                     0xb1
+#define MT9V034_LVDS_SHIFT_CLK_CTRL                  0xb2
+#define MT9V034_LVDS_DATA_CTRL                       0xb3
+#define MT9V034_DATA_STREAM_LATENCY                  0xb4
+#define MT9V034_LVDS_INTERNAL_SYNC                   0xb5
+#define MT9V034_LVDS_PAYLOAD_CONTROL                 0xb6
+#define MT9V034_STEREOSCOP_ERROR_CTRL                0xb7
+#define MT9V034_STEREOSCOP_ERROR_FLAG                0xb8
+#define MT9V034_LVDS_DATA_OUTPUT                     0xb9
+#define MT9V034_AGC_GAIN_OUTPUT                      0xba
+#define MT9V034_AEC_GAIN_OUTPUT                      0xbb
+#define MT9V034_AGC_AEC_CURRENT_BIN                  0xbc
+#define MT9V034_INTERLACE_FIELD_BLANC                0xbf
+#define MT9V034_MON_MODE_CAPTURE_CONTROL             0xc0
+#define MT9V034_ANTI_ECLIPSE_CONTROL                 0xc2
+#define MT9V034_NTSV_FV_LV_CONTROL                   0xc6
+#define MT9V034_NTSC_HORIZ_BLANC_CTRL                0xc7
+#define MT9V034_NTSC_VERT_BLANC_CTRL                 0xc8
+#define MT9V034_COLUMN_START_CONTEXT_B               0xc9
+#define MT9V034_ROW_START_CONTEXT_B                  0xca
+#define MT9V034_WIN_HEIGHT_CONTEXT_B                 0xcb
+#define MT9V034_WIN_WIDTH_CONTEXT_B                  0xcc
+#define MT9V034_HORIZONTAL_BLANCING_CONTEXT_B        0xcd
+#define MT9V034_VERTICAL_BLANCING_CONTEXT_B          0xce
+#define MT9V034_COARSE_SW1_CONTEXT_B                 0xcf
+#define MT9V034_COARSE_SW2_CONTEXT_B                 0xd0
+#define MT9V034_SHUTTER_WIDTH_CTRL_CONTEXT_B         0xd1
+#define MT9V034_COARSE_TOTAL_SHUTTER_WIDTH_CONTEXT_B 0xd2
+#define MT9V034_FINE_SW1_CONTEXT_A                   0xd3
+#define MT9V034_FINE_SW2_CONTEXT_A                   0xd4
+#define MT9V034_FINE_SHUTTER_WIDTH_TOTAL_CONTEXT_A   0xd5
+#define MT9V034_FINE_SW1_CONTEXT_B                   0xd6
+#define MT9V034_FINE_SW2_CONTEXT_B                   0xd7
+#define MT9V034_FINE_SHUTTER_WIDTH_TOTAL_CONTEXT_B   0xd8
+#define MT9V034_MONITOR_MODE                         0xd9
+#define MT9V034_BYTEWISE_ADDR                        0xf0
+#define MT9V034_REGISTER_LOCK                        0xfe
+
+#define MT9V034_PIXEL_ARRAY_HEIGHT		492
+#define MT9V034_PIXEL_ARRAY_WIDTH			782
+#define MT9V034_MAX_PICS_IN_A_ROW 5
+#define MT9V034_EXPOSURE_TIME_USEC 8000
+
+#ifndef DEBUG_PRINTOUTS
+//  #define DEBUG_PRINTOUTS 1
+//  #define DPRINT(s) printk(KERN_ALERT "%s\n", s)
+//#else
+  #define DPRINT(s)
+#endif
+
+struct mt9v034 {
+	struct v4l2_subdev subdev;
+	struct media_pad pad;
+
+	struct v4l2_mbus_framefmt format;
+	struct v4l2_rect crop;
+
+	struct v4l2_ctrl_handler ctrls;
+
+	struct mutex power_lock;
+	int power_count;
+
+	struct mt9v034_platform_data *pdata;
+	u16 chip_control;
+	u16 aec_agc;
+  int isReady;
+  
+  int exposure_time_us;
+  int analog_gain;
+};
+
+struct mt9v034 *g_mt9v034 = 0;  // Global structure so other drivers
can use the mt9v034_takepic function
+
+static int mt9v034_set_chip_control(struct mt9v034 *mt9v034, u16 clear,
u16 set);
+static int mt9v034_read(struct i2c_client *client, const u8 reg);
+
+static struct mt9v034 *to_mt9v034(struct v4l2_subdev *sd)
+{
+	return container_of(sd, struct mt9v034, subdev);
+}
+
+static void mt9v034_takepic(int nr_pics, int wait_ms, struct timespec
*cam_timespec)
+{
+  int nr;
+  int pic_ms;
+  int exp_ms;
+	 struct i2c_client *client;
+   u16 temp;
+
+  if (!g_mt9v034 || !g_mt9v034->isReady)
+    return;
+
+   client = v4l2_get_subdevdata(&g_mt9v034->subdev);    
+  // g_mt9v034->aec_agc = mt9v034_read(client,
MT9V034_AEC_AGC_ENABLE_A_B);
+  // printk(KERN_ALERT "MT9v034 aec_agc = 0x%x.\n",
g_mt9v034->aec_agc);
+  
+  temp = mt9v034_read(client, MT9V034_BL_CALIB_CTRL);
+  printk(KERN_ALERT "MT9V034_BL_CALIB_CTRL = 0x%x.\n", temp);
+  
+  temp = mt9v034_read(client, MT9V034_ROW_NOISE_CORR_CONTROL);
+  printk(KERN_ALERT "MT9V034_ROW_NOISE_CORR_CONTROL = 0x%x.\n", temp);
+    
+  exp_ms = (g_mt9v034->exposure_time_us / 1000);
+  pic_ms = 20 + exp_ms;
+  
+  if (wait_ms > pic_ms)         // Should we wait before taking the
first picture?
+    msleep(wait_ms - pic_ms);   // Sleep the time wanted minus 1
picture time
+  
+  nr = (nr_pics > MT9V034_MAX_PICS_IN_A_ROW)? MT9V034_MAX_PICS_IN_A_ROW
: nr_pics;
+  
+  ktime_get_ts(cam_timespec);
+  
+  // Will take 1 picture more that is overexposed
+  g_mt9v034->pdata->take_picture(&g_mt9v034->subdev, nr,
g_mt9v034->exposure_time_us / 1000); // Take pictures
+}
+EXPORT_SYMBOL_GPL(mt9v034_takepic);
+
+static int mt9v034_read(struct i2c_client *client, const u8 reg)
+{
+	s32 data = i2c_smbus_read_word_data(client, reg);
+	dev_dbg(&client->dev, "%s: read 0x%04x from 0x%02x\n", __func__,
+		swab16(data), reg);
+	return data < 0 ? data : swab16(data);
+}
+
+static int mt9v034_write(struct i2c_client *client, const u8 reg,
+			 const u16 data)
+{
+	dev_dbg(&client->dev, "%s: writing 0x%04x to 0x%02x\n",
__func__, data, reg);
+	return i2c_smbus_write_word_data(client, reg, swab16(data));
+}
+
+static int mt9v034_set_chip_control(struct mt9v034 *mt9v034, u16 clear,
u16 set)
+{
+	struct i2c_client *client =
v4l2_get_subdevdata(&mt9v034->subdev);
+	u16 value;
+	int ret;
+
+  mt9v034->chip_control = mt9v034_read(client, MT9V034_CHIP_CONTROL);
+  value = (mt9v034->chip_control & ~clear) | set;
+	ret = mt9v034_write(client, MT9V034_CHIP_CONTROL, value);
+	if (ret < 0)
+		return ret;
+
+	mt9v034->chip_control = value;
+	return 0;
+}
+
+static int mt9v034_update_aec_agc(struct mt9v034 *mt9v034, u16 which,
int enable)
+{
+	struct i2c_client *client =
v4l2_get_subdevdata(&mt9v034->subdev);
+	u16 value = mt9v034->aec_agc;
+	int ret;
+
+	if (enable)
+		value |= which;
+	else
+		value &= ~which;
+
+	ret = mt9v034_write(client, MT9V034_AEC_AGC_ENABLE_A_B, value);
+	if (ret < 0)
+		return ret;
+
+	mt9v034->aec_agc = value;
+	return 0;
+}
+
+static int mt9v034_configure(struct v4l2_subdev *subdev)
+{
+	struct i2c_client *client = v4l2_get_subdevdata(subdev);
+	struct mt9v034 *mt9v034 = to_mt9v034(subdev);
+	struct v4l2_mbus_framefmt *format = &mt9v034->format;
+	struct v4l2_rect *crop = &mt9v034->crop;
+	unsigned int hratio;
+	unsigned int vratio;
+	int ret;
+  unsigned int exposure_value;
+  unsigned int gain_value;
+  s32 temp;
+  
+  DPRINT("mt9v034_configure");
+
+  ret = mt9v034_set_chip_control(mt9v034,
MT9V034_CHIP_CONTROL_RESERVED, MT9V034_CHIP_CONTROL_SNAPSHOT_MODE);  //
Clear bit 9 for normal operation and set to snapshot mode
+  if (ret < 0)
+    return ret;
+  
+	/* Configure the window size and row/column bin */
+	hratio = DIV_ROUND_CLOSEST(crop->width, format->width);
+	vratio = DIV_ROUND_CLOSEST(crop->height, format->height);
+  
+	ret = mt9v034_write(client, MT9V034_COLUMN_START_CONTEXT_A,
crop->left);
+	if (ret < 0)
+		return ret;
+
+	ret = mt9v034_write(client, MT9V034_ROW_START_CONTEXT_A,
crop->top);
+	if (ret < 0)
+		return ret;
+
+	ret = mt9v034_write(client, MT9V034_WIN_WIDTH_CONTEXT_A,
crop->width);
+	if (ret < 0)
+		return ret;
+
+	ret = mt9v034_write(client, MT9V034_WIN_HEIGHT_CONTEXT_A,
crop->height);
+	if (ret < 0)
+		return ret;
+    
+  ret = mt9v034_write(client, MT9V034_AEC_AGC_ENABLE_A_B,
mt9v034->aec_agc);  // Set AEC (Automatic Exposure Control) and AGC
(Automatic Gain Control)
+  if (ret < 0)
+    return ret;
+
+  if (!mt9v034->aec_agc)  // No automatic exposure or gain control?
+  {
+    exposure_value = (30375 * mt9v034->exposure_time_us / 1000000);
+    
+    ret = mt9v034_write(client,
MT9V034_COARSE_TOTAL_SHUTTER_WIDTH_CONTEXT_A, exposure_value); // Set
"Total shutter width" register so exposure time is set
+    if (ret < 0)
+    {
+      printk(KERN_ALERT "Error setting exposure on MT9V034 camera
sensor.\n");
+      return ret;
+    }
+    
+    gain_value = 16 * mt9v034->analog_gain;
+    
+    ret = mt9v034_write(client,  MT9V034_ANALOG_GAIN_CONTEXT_A,
gain_value); // Set "Analog gain" register so gain is set
+    if (ret < 0)
+    {
+      printk(KERN_ALERT "Error setting analog gain on MT9V034 camera
sensor.\n");
+      return ret;
+    }
+  }
+  
+	ret = mt9v034_write(client, MT9V034_BL_CALIB_CTRL, 1);  //
Override automatic black level correction with programmed values
+	if (ret < 0)
+		return ret;
+    
+	ret = mt9v034_write(client,
MT9V034_BLACK_LEVEL_CALIBRATION_VALUE, 0);  // Disable black calibration
+	if (ret < 0)
+		return ret;
+  
+	ret = mt9v034_write(client, MT9V034_ROW_NOISE_CORR_CONTROL, 0);
// Disable row noise correction
+	if (ret < 0)
+		return ret;
+
+	v4l2_ctrl_handler_setup(&mt9v034->ctrls);
+  
+  mt9v034->isReady = 1;
+  
+  return 0;
+}
+
+static int mt9v034_power_on(struct mt9v034 *mt9v034)
+{
+	if (mt9v034->pdata->set_clock) {
+		mt9v034->pdata->set_clock(&mt9v034->subdev, 24000000);
+	}
+
+  msleep(1);
+
+  mt9v034->pdata->configure(&mt9v034->subdev);
+
+  msleep(1);
+
+  mt9v034->pdata->reset();
+  
+  return mt9v034_configure(&mt9v034->subdev);
+}
+
+static void mt9v034_power_off(struct mt9v034 *mt9v034)
+{
+	if (mt9v034->pdata->set_clock)
+		mt9v034->pdata->set_clock(&mt9v034->subdev, 0);
+    
+  mt9v034->isReady = 0;
+}
+
+static int __mt9v034_set_power(struct mt9v034 *mt9v034, int on)
+{
+	struct i2c_client *client =
v4l2_get_subdevdata(&mt9v034->subdev);
+	int ret;
+
+	if (!on) {
+		mt9v034_power_off(mt9v034);
+    return 0;
+	}
+
+	ret = mt9v034_power_on(mt9v034);
+	if (ret < 0)
+		return ret;
+
+	/* Configure the pixel clock polarity */
+	if (mt9v034->pdata && mt9v034->pdata->clk_pol) {
+		ret  = mt9v034_write(client, MT9V034_PIXCLK_FV_LV_CTRL,
MT9V034_PIXEL_CLOCK_INV_PXL_CLK);
+		if (ret < 0)
+			return ret;
+	}
+
+	return 0;
+}
+
+/*
------------------------------------------------------------------------
-----
+ * V4L2 subdev video operations
+ */
+
+static struct v4l2_mbus_framefmt *
+__mt9v034_get_pad_format(struct mt9v034 *mt9v034, struct v4l2_subdev_fh
*fh,
+			 unsigned int pad, enum
v4l2_subdev_format_whence which)
+{
+	switch (which) {
+	case V4L2_SUBDEV_FORMAT_TRY:
+		return v4l2_subdev_get_try_format(fh, pad);
+	case V4L2_SUBDEV_FORMAT_ACTIVE:
+		return &mt9v034->format;
+	default:
+		return NULL;
+	}
+}
+
+static struct v4l2_rect *
+__mt9v034_get_pad_crop(struct mt9v034 *mt9v034, struct v4l2_subdev_fh
*fh,
+		       unsigned int pad, enum v4l2_subdev_format_whence
which)
+{
+	switch (which) {
+	case V4L2_SUBDEV_FORMAT_TRY:
+		return v4l2_subdev_get_try_crop(fh, pad);
+	case V4L2_SUBDEV_FORMAT_ACTIVE:
+		return &mt9v034->crop;
+	default:
+		return NULL;
+	}
+}
+
+static inline void print_reg(struct mt9v034 *sensor, const char *name,
u8 address)
+{
+	struct i2c_client *client =
v4l2_get_subdevdata(&sensor->subdev);
+  s32 reg_value = i2c_smbus_read_word_data(client, address);
+  printk(KERN_ALERT "%s = 0x%x\n", name, swab16(reg_value));
+}
+
+static inline void mt9v034_print_registers(struct mt9v034 *sensor)
+{
+  print_reg(sensor, "MT9V034_CHIP_VERSION", 0x00);
+  print_reg(sensor, "MT9V034_COLUMN_START_CONTEXT_A", 0x01);
+  print_reg(sensor, "MT9V034_ROW_START_CONTEXT_A", 0x02);
+  print_reg(sensor, "MT9V034_WIN_HEIGHT_CONTEXT_A", 0x03);
+  print_reg(sensor, "MT9V034_WIN_WIDTH_CONTEXT_A", 0x04);
+  print_reg(sensor, "MT9V034_HORIZONTAL_BLANCING_CONTEXT_A", 0x05);
+  print_reg(sensor, "MT9V034_VERTICAL_BLANCING_CONTEXT_A", 0x06);
+  print_reg(sensor, "MT9V034_CHIP_CONTROL", 0x07);
+  print_reg(sensor, "MT9V034_COARSE_SHUTTER_WIDTH1_CONTEXT_A", 0x08);
+  print_reg(sensor, "MT9V034_COARSE_SHUTTER_WIDTH2_CONTEXT_A", 0x09);
+  print_reg(sensor, "MT9V034_SHUTTER_WIDTH_CTRL_CONTEXT_A", 0x0a);
+  print_reg(sensor, "MT9V034_COARSE_TOTAL_SHUTTER_WIDTH_CONTEXT_A",
0x0b);
+  print_reg(sensor, "MT9V034_RESET", 0x0c);
+  print_reg(sensor, "MT9V034_READ_MODE_CONTEXT_A", 0x0d);
+  print_reg(sensor, "MT9V034_READ_MODE_CONTEXT_B", 0x0e);
+  print_reg(sensor, "MT9V034_SENSOR_TYPE_HDR_ENABLE", 0x0f);
+  print_reg(sensor, "MT9V034_LED_OUT_CTRL", 0x1b);
+  print_reg(sensor, "MT9V034_COMPANDING", 0x1c);
+  print_reg(sensor, "MT9V034_VREF_ADC_CTRL", 0x2c);
+  print_reg(sensor, "MT9V034_V1_CONTROL_CONTEXT_A", 0x31);
+  print_reg(sensor, "MT9V034_V2_CONTROL_CONTEXT_A", 0x32);
+  print_reg(sensor, "MT9V034_V3_CONTROL_CONTEXT_A", 0x33);
+  print_reg(sensor, "MT9V034_V4_CONTROL_CONTEXT_A", 0x34);
+  print_reg(sensor, "MT9V034_ANALOG_GAIN_CONTEXT_A", 0x35);
+  print_reg(sensor, "MT9V034_ANALOG_GAIN_CONTEXT_B", 0x36);
+  print_reg(sensor, "MT9V034_V1_CONTROL_CONTEXT_B", 0x39);
+  print_reg(sensor, "MT9V034_V2_CONTROL_CONTEXT_B", 0x3a);
+  print_reg(sensor, "MT9V034_V3_CONTROL_CONTEXT_B", 0x3b);
+  print_reg(sensor, "MT9V034_V4_CONTROL_CONTEXT_B", 0x3c);
+  print_reg(sensor, "MT9V034_FRAME_DARK_AVERAGE", 0x42);
+  print_reg(sensor, "MT9V034_DARK_AVERAGE_THRESHOLD", 0x46);
+  print_reg(sensor, "MT9V034_BL_CALIB_CTRL", 0x47);
+  print_reg(sensor, "MT9V034_BLACK_LEVEL_CALIBRATION_VALUE", 0x48);
+  print_reg(sensor, "MT9V034_BL_CALIB_STEP_SIZE", 0x4c);
+  print_reg(sensor, "MT9V034_ROW_NOISE_CORR_CONTROL", 0x70);
+  print_reg(sensor, "MT9V034_ROW_NOISE_CONSTANT", 0x71);
+  print_reg(sensor, "MT9V034_PIXCLK_FV_LV_CTRL", 0x72);
+  print_reg(sensor, "MT9V034_DIGITAL_TEST_PATTERN", 0x7f);
+  print_reg(sensor, "MT9V034_TILE_WEIGHT_GAIN_X0_Y0", 0x80);
+  print_reg(sensor, "MT9V034_TILE_WEIGHT_GAIN_X1_Y0", 0x81);
+  print_reg(sensor, "MT9V034_TILE_WEIGHT_GAIN_X2_Y0", 0x82);
+  print_reg(sensor, "MT9V034_TILE_WEIGHT_GAIN_X3_Y0", 0x83);
+  print_reg(sensor, "MT9V034_TILE_WEIGHT_GAIN_X4_Y0", 0x84);
+  print_reg(sensor, "MT9V034_TILE_WEIGHT_GAIN_X0_Y1", 0x85);
+  print_reg(sensor, "MT9V034_TILE_WEIGHT_GAIN_X1_Y1", 0x86);
+  print_reg(sensor, "MT9V034_TILE_WEIGHT_GAIN_X2_Y1", 0x87);
+  print_reg(sensor, "MT9V034_TILE_WEIGHT_GAIN_X3_Y1", 0x88);
+  print_reg(sensor, "MT9V034_TILE_WEIGHT_GAIN_X4_Y1", 0x89);
+  print_reg(sensor, "MT9V034_TILE_WEIGHT_GAIN_X0_Y2", 0x8a);
+  print_reg(sensor, "MT9V034_TILE_WEIGHT_GAIN_X1_Y2", 0x8b);
+  print_reg(sensor, "MT9V034_TILE_WEIGHT_GAIN_X2_Y2", 0x8c);
+  print_reg(sensor, "MT9V034_TILE_WEIGHT_GAIN_X3_Y2", 0x8d);
+  print_reg(sensor, "MT9V034_TILE_WEIGHT_GAIN_X4_Y2", 0x8e);
+  print_reg(sensor, "MT9V034_TILE_WEIGHT_GAIN_X0_Y3", 0x8f);
+  print_reg(sensor, "MT9V034_TILE_WEIGHT_GAIN_X1_Y3", 0x90);
+  print_reg(sensor, "MT9V034_TILE_WEIGHT_GAIN_X2_Y3", 0x91);
+  print_reg(sensor, "MT9V034_TILE_WEIGHT_GAIN_X3_Y3", 0x92);
+  print_reg(sensor, "MT9V034_TILE_WEIGHT_GAIN_X4_Y3", 0x93);
+  print_reg(sensor, "MT9V034_TILE_WEIGHT_GAIN_X0_Y4", 0x94);
+  print_reg(sensor, "MT9V034_TILE_WEIGHT_GAIN_X1_Y4", 0x95);
+  print_reg(sensor, "MT9V034_TILE_WEIGHT_GAIN_X2_Y4", 0x96);
+  print_reg(sensor, "MT9V034_TILE_WEIGHT_GAIN_X3_Y4", 0x97);
+  print_reg(sensor, "MT9V034_TILE_WEIGHT_GAIN_X4_Y4", 0x98);
+  print_reg(sensor, "MT9V034_TILE_COORD_X0_5", 0x99);
+  print_reg(sensor, "MT9V034_TILE_COORD_X1_5", 0x9a);
+  print_reg(sensor, "MT9V034_TILE_COORD_X2_5", 0x9b);
+  print_reg(sensor, "MT9V034_TILE_COORD_X3_5", 0x9c);
+  print_reg(sensor, "MT9V034_TILE_COORD_X4_5", 0x9d);
+  print_reg(sensor, "MT9V034_TILE_COORD_X5_5", 0x9e);
+  print_reg(sensor, "MT9V034_TILE_COORD_Y0_5", 0x9f);
+  print_reg(sensor, "MT9V034_TILE_COORD_Y1_5", 0xa0);
+  print_reg(sensor, "MT9V034_TILE_COORD_Y2_5", 0xa1);
+  print_reg(sensor, "MT9V034_TILE_COORD_Y3_5", 0xa2);
+  print_reg(sensor, "MT9V034_TILE_COORD_Y4_5", 0xa3);
+  print_reg(sensor, "MT9V034_TILE_COORD_Y5_5", 0xa4);
+  print_reg(sensor, "MT9V034_AEC_AGC_DESIRED_BIN", 0xa5);
+  print_reg(sensor, "MT9V034_AEC_UPDATE_FREQUENCY", 0xa6);
+  print_reg(sensor, "MT9V034_AEC_LPF", 0xa8);
+  print_reg(sensor, "MT9V034_AGC_UPDATE_FREQUENCY", 0xa9);
+  print_reg(sensor, "MT9V034_AGC_LPF", 0xaa);
+  print_reg(sensor, "MT9V034_MAX_ANALOG_GAIN", 0xab);
+  print_reg(sensor, "MT9V034_AEC_MINIMUM_EXPOSURE", 0xac);
+  print_reg(sensor, "MT9V034_AEC_MAXIMUM_EXPOSURE", 0xad);
+  print_reg(sensor, "MT9V034_BIN_DIFFERENCE_THRESHOLD", 0xae);
+  print_reg(sensor, "MT9V034_AEC_AGC_ENABLE_A_B", 0xaf);
+  print_reg(sensor, "MT9V034_AEC_AGC_PIX_COUNT", 0xb0);
+  print_reg(sensor, "MT9V034_LVDS_MASTER_CTRL", 0xb1);
+  print_reg(sensor, "MT9V034_LVDS_SHIFT_CLK_CTRL", 0xb2);
+  print_reg(sensor, "MT9V034_LVDS_DATA_CTRL", 0xb3);
+  print_reg(sensor, "MT9V034_DATA_STREAM_LATENCY", 0xb4);
+  print_reg(sensor, "MT9V034_LVDS_INTERNAL_SYNC", 0xb5);
+  print_reg(sensor, "MT9V034_LVDS_PAYLOAD_CONTROL", 0xb6);
+  print_reg(sensor, "MT9V034_STEREOSCOP_ERROR_CTRL", 0xb7);
+  print_reg(sensor, "MT9V034_STEREOSCOP_ERROR_FLAG", 0xb8);
+  print_reg(sensor, "MT9V034_LVDS_DATA_OUTPUT", 0xb9);
+  print_reg(sensor, "MT9V034_AGC_GAIN_OUTPUT", 0xba);
+  print_reg(sensor, "MT9V034_AEC_GAIN_OUTPUT", 0xbb);
+  print_reg(sensor, "MT9V034_AGC_AEC_CURRENT_BIN", 0xbc);
+  print_reg(sensor, "MT9V034_INTERLACE_FIELD_BLANC", 0xbf);
+  print_reg(sensor, "MT9V034_MON_MODE_CAPTURE_CONTROL", 0xc0);
+  print_reg(sensor, "MT9V034_ANTI_ECLIPSE_CONTROL", 0xc2);
+  print_reg(sensor, "MT9V034_NTSV_FV_LV_CONTROL", 0xc6);
+  print_reg(sensor, "MT9V034_NTSC_HORIZ_BLANC_CTRL", 0xc7);
+  print_reg(sensor, "MT9V034_NTSC_VERT_BLANC_CTRL", 0xc8);
+  print_reg(sensor, "MT9V034_COLUMN_START_CONTEXT_B", 0xc9);
+  print_reg(sensor, "MT9V034_ROW_START_CONTEXT_B", 0xca);
+  print_reg(sensor, "MT9V034_ROW_WIN_HEIGHT_CONTEXT_B", 0xcb);
+  print_reg(sensor, "MT9V034_WINDOW_WIDTH_CONTEXT_B", 0xcc);
+  print_reg(sensor, "MT9V034_HORIZONTAL_BLANCING_CONTEXT_B", 0xcd);
+  print_reg(sensor, "MT9V034_VERTICAL_BLANCING_CONTEXT_B", 0xce);
+  print_reg(sensor, "MT9V034_COARSE_SW1_CONTEXT_B", 0xcf);
+  print_reg(sensor, "MT9V034_COARSE_SW2_CONTEXT_B", 0xd0);
+  print_reg(sensor, "MT9V034_SHUTTER_WIDTH_CTRL_CONTEXT_B", 0xd1);
+  print_reg(sensor, "MT9V034_COARSE_TOTAL_SHUTTER_WIDTH_CONTEXT_B",
0xd2);
+  print_reg(sensor, "MT9V034_FINE_SW1_CONTEXT_A", 0xd3);
+  print_reg(sensor, "MT9V034_FINE_SW2_CONTEXT_A", 0xd4);
+  print_reg(sensor, "MT9V034_FINE_SHUTTER_WIDTH_TOTAL_CONTEXT_A",
0xd5);
+  print_reg(sensor, "MT9V034_FINE_SW1_CONTEXT_B", 0xd6);
+  print_reg(sensor, "MT9V034_FINE_SW2_CONTEXT_B", 0xd7);
+  print_reg(sensor, "MT9V034_FINE_SHUTTER_WIDTH_TOTAL_CONTEXT_B",
0xd8);
+  print_reg(sensor, "MT9V034_MONITOR_MODE", 0xd9);
+  print_reg(sensor, "MT9V034_BYTEWISE_ADDR", 0xf0);
+  print_reg(sensor, "MT9V034_REGISTER_LOCK", 0xfe);
+}
+
+static int mt9v034_s_stream(struct v4l2_subdev *subdev, int enable)
+{
+	struct mt9v034 *mt9v034 = to_mt9v034(subdev);
+  struct timespec t;
+	int ret;
+
+	DPRINT("mt9v034_s_stream");
+	
+	if (!enable) {
+		return __mt9v034_set_power(mt9v034, 0);
+	}
+
+	ret = __mt9v034_set_power(mt9v034, 1);
+	if (ret < 0)
+		return ret;
+
+//  mt9v034_print_registers(mt9v034);
+
+  msleep(40);
+  
+  mt9v034_takepic(0, 0, &t); // Take 1 picture that has old settings
(automatic gain/exposure). Ignore it.
+  
+	return 0;
+}
+
+static int mt9v034_enum_mbus_code(struct v4l2_subdev *subdev,
+				  struct v4l2_subdev_fh *fh,
+				  struct v4l2_subdev_mbus_code_enum
*code)
+{
+	if (code->index > 0)
+		return -EINVAL;
+
+	code->code = V4L2_MBUS_FMT_SGRBG10_1X10;
+	return 0;
+}
+
+static int mt9v034_enum_frame_size(struct v4l2_subdev *subdev,
+				   struct v4l2_subdev_fh *fh,
+				   struct v4l2_subdev_frame_size_enum
*fse)
+{
+	if (fse->index >= 8 || fse->code != V4L2_MBUS_FMT_SGRBG10_1X10)
+		return -EINVAL;
+
+	fse->min_width = MT9V034_WINDOW_WIDTH_DEF / fse->index;
+	fse->max_width = fse->min_width;
+	fse->min_height = MT9V034_WINDOW_HEIGHT_DEF / fse->index;
+	fse->max_height = fse->min_height;
+
+	return 0;
+}
+
+static int mt9v034_get_format(struct v4l2_subdev *subdev,
+			      struct v4l2_subdev_fh *fh,
+			      struct v4l2_subdev_format *format)
+{
+	struct mt9v034 *mt9v034 = to_mt9v034(subdev);
+
+	format->format = *__mt9v034_get_pad_format(mt9v034, fh,
format->pad, format->which);
+	return 0;
+}
+
+static int mt9v034_set_format(struct v4l2_subdev *subdev,
+			      struct v4l2_subdev_fh *fh,
+			      struct v4l2_subdev_format *format)
+{
+	struct mt9v034 *mt9v034 = to_mt9v034(subdev);
+	struct v4l2_mbus_framefmt *__format;
+	struct v4l2_rect *__crop;
+	unsigned int width;
+	unsigned int height;
+	unsigned int hratio;
+	unsigned int vratio;
+
+	__crop = __mt9v034_get_pad_crop(mt9v034, fh, format->pad,
+					format->which);
+
+	/* Clamp the width and height to avoid dividing by zero. */
+	width = clamp_t(unsigned int, ALIGN(format->format.width, 2),
+			max(__crop->width / 8,
MT9V034_WINDOW_WIDTH_MIN),
+			__crop->width);
+	height = clamp_t(unsigned int, ALIGN(format->format.height, 2),
+			 max(__crop->height / 8,
MT9V034_WINDOW_HEIGHT_MIN),
+			 __crop->height);
+
+	hratio = DIV_ROUND_CLOSEST(__crop->width, width);
+	vratio = DIV_ROUND_CLOSEST(__crop->height, height);
+
+	__format = __mt9v034_get_pad_format(mt9v034, fh, format->pad,
+					    format->which);
+	__format->width = __crop->width / hratio;
+	__format->height = __crop->height / vratio;
+
+	format->format = *__format;
+
+	return 0;
+}
+
+static int mt9v034_get_crop(struct v4l2_subdev *subdev,
+			    struct v4l2_subdev_fh *fh,
+			    struct v4l2_subdev_crop *crop)
+{
+	struct mt9v034 *mt9v034 = to_mt9v034(subdev);
+
+	crop->rect = *__mt9v034_get_pad_crop(mt9v034, fh, crop->pad,
+					     crop->which);
+	return 0;
+}
+
+static int mt9v034_set_crop(struct v4l2_subdev *subdev,
+			    struct v4l2_subdev_fh *fh,
+			    struct v4l2_subdev_crop *crop)
+{
+	struct mt9v034 *mt9v034 = to_mt9v034(subdev);
+	struct v4l2_mbus_framefmt *__format;
+	struct v4l2_rect *__crop;
+	struct v4l2_rect rect;
+
+	/* Clamp the crop rectangle boundaries and align them to a
multiple of 2
+	 * pixels.
+	 */
+	rect.left = clamp(ALIGN(crop->rect.left, 2),
+			  MT9V034_COLUMN_START_MIN,
+			  MT9V034_COLUMN_START_MAX);
+	rect.top = clamp(ALIGN(crop->rect.top, 2),
+			 MT9V034_ROW_START_MIN,
+			 MT9V034_ROW_START_MAX);
+	rect.width = clamp(ALIGN(crop->rect.width, 2),
+			   MT9V034_WINDOW_WIDTH_MIN,
+			   MT9V034_WINDOW_WIDTH_MAX);
+	rect.height = clamp(ALIGN(crop->rect.height, 2),
+			    MT9V034_WINDOW_HEIGHT_MIN,
+			    MT9V034_WINDOW_HEIGHT_MAX);
+	rect.width = min(rect.width, MT9V034_PIXEL_ARRAY_WIDTH -
rect.left);
+	rect.height = min(rect.height, MT9V034_PIXEL_ARRAY_HEIGHT -
rect.top);
+
+	__crop = __mt9v034_get_pad_crop(mt9v034, fh, crop->pad,
crop->which);
+
+	if (rect.width != __crop->width || rect.height !=
__crop->height) {
+		/* Reset the output image size if the crop rectangle
size has
+		 * been modified.
+		 */
+		__format = __mt9v034_get_pad_format(mt9v034, fh,
crop->pad,
+						    crop->which);
+		__format->width = rect.width;
+		__format->height = rect.height;
+	}
+
+	*__crop = rect;
+	crop->rect = rect;
+
+	return 0;
+}
+
+/*
------------------------------------------------------------------------
-----
+ * V4L2 subdev control operations
+ */
+
+#define V4L2_CID_TEST_PATTERN		(V4L2_CID_USER_BASE | 0x1001)
+
+static int mt9v034_s_ctrl(struct v4l2_ctrl *ctrl)
+{ 
+  struct mt9v034 *mt9v034 = container_of(ctrl->handler, struct mt9v034,
ctrls);
+	struct i2c_client *client =
v4l2_get_subdevdata(&mt9v034->subdev);
+	u16 data;
+  int ret = 0;
+
+	switch (ctrl->id) {
+	case V4L2_CID_AUTOGAIN:
+		return mt9v034_update_aec_agc(mt9v034,
MT9V034_AGC_ENABLE, ctrl->val);
+
+	case V4L2_CID_GAIN:
+    mt9v034->analog_gain = ctrl->val;
+    return ret;
+
+	case V4L2_CID_EXPOSURE_AUTO:
+		return mt9v034_update_aec_agc(mt9v034,
MT9V034_AEC_ENABLE, !ctrl->val);
+
+	case V4L2_CID_EXPOSURE:
+    mt9v034->exposure_time_us = ctrl->val;
+		return 0;
+
+	case V4L2_CID_TEST_PATTERN:
+		switch (ctrl->val) {
+		case 0:
+			data = 0;
+			break;
+		case 1:
+			data = MT9V034_TEST_PATTERN_GRAY_VERTICAL
+			     | MT9V034_TEST_PATTERN_ENABLE;
+			break;
+		case 2:
+			data = MT9V034_TEST_PATTERN_GRAY_HORIZONTAL
+			     | MT9V034_TEST_PATTERN_ENABLE;
+			break;
+		case 3:
+			data = MT9V034_TEST_PATTERN_GRAY_DIAGONAL
+			     | MT9V034_TEST_PATTERN_ENABLE;
+			break;
+		default:
+			data = (ctrl->val <<
MT9V034_TEST_PATTERN_DATA_SHIFT)
+			     | MT9V034_TEST_PATTERN_USE_DATA
+			     | MT9V034_TEST_PATTERN_ENABLE
+			     | MT9V034_TEST_PATTERN_FLIP;
+			break;
+		}
+
+		return mt9v034_write(client,
MT9V034_DIGITAL_TEST_PATTERN, data);
+	}
+
+	return 0;
+}
+
+static struct v4l2_ctrl_ops mt9v034_ctrl_ops = {
+	.s_ctrl = mt9v034_s_ctrl,
+};
+
+static const struct v4l2_ctrl_config mt9v034_ctrls[] = {
+	{
+		.ops		= &mt9v034_ctrl_ops,
+		.id		= V4L2_CID_TEST_PATTERN,
+		.type		= V4L2_CTRL_TYPE_INTEGER,
+		.name		= "Test pattern",
+		.min		= 0,
+		.max		= 1023,
+		.step		= 1,
+		.def		= 0,
+		.flags		= 0,
+	},
+};
+
+/*
------------------------------------------------------------------------
-----
+ * V4L2 subdev core operations
+ */
+
+static int mt9v034_set_power(struct v4l2_subdev *subdev, int on)
+{
+	struct mt9v034 *mt9v034 = to_mt9v034(subdev);
+	int ret = 0;
+
+	mutex_lock(&mt9v034->power_lock);
+
+	/* If the power count is modified from 0 to != 0 or from != 0 to
0,
+	 * update the power state.
+	 */
+	if (mt9v034->power_count == !on) {
+		ret = __mt9v034_set_power(mt9v034, !!on);
+		if (ret < 0)
+			goto done;
+	}
+
+	/* Update the power count. */
+	mt9v034->power_count += on ? 1 : -1;
+	WARN_ON(mt9v034->power_count < 0);
+
+done:
+	mutex_unlock(&mt9v034->power_lock);
+	return ret;
+}
+
+/*
------------------------------------------------------------------------
-----
+ * V4L2 subdev internal operations
+ */
+
+static int mt9v034_registered(struct v4l2_subdev *subdev)
+{
+	struct i2c_client *client = v4l2_get_subdevdata(subdev);
+	struct mt9v034 *mt9v034 = to_mt9v034(subdev);
+	s32 data;
+	int ret;
+
+	dev_info(&client->dev, "Probing MT9V034 at address 0x%02x\n",
+			client->addr);
+
+	ret = mt9v034_power_on(mt9v034);
+	if (ret < 0) {
+		dev_err(&client->dev, "MT9V034 power up failed\n");
+		return ret;
+	}
+
+	/* Read and check the sensor version */
+	data = mt9v034_read(client, MT9V034_CHIP_VERSION);
+	if (data != MT9V034_CHIP_ID) {
+		dev_err(&client->dev, "MT9V034 not detected, wrong
version "
+				"0x%04x\n", data);
+		return -ENODEV;
+	}
+
+	mt9v034_power_off(mt9v034);
+
+	dev_info(&client->dev, "MT9V034 detected at address 0x%02x\n",
+			client->addr);
+
+	return ret;
+}
+
+static int mt9v034_open(struct v4l2_subdev *subdev, struct
v4l2_subdev_fh *fh)
+{
+	struct v4l2_mbus_framefmt *format;
+	struct v4l2_rect *crop;
+
+	crop = v4l2_subdev_get_try_crop(fh, 0);
+	crop->left = MT9V034_COLUMN_START_DEF;
+	crop->top = MT9V034_ROW_START_DEF;
+	crop->width = MT9V034_WINDOW_WIDTH_DEF;
+	crop->height = MT9V034_WINDOW_HEIGHT_DEF;
+
+	format = v4l2_subdev_get_try_format(fh, 0);
+	format->code = V4L2_MBUS_FMT_SGRBG10_1X10;
+	format->width = MT9V034_WINDOW_WIDTH_DEF;
+	format->height = MT9V034_WINDOW_HEIGHT_DEF;
+	format->field = V4L2_FIELD_NONE;
+	format->colorspace = V4L2_COLORSPACE_SRGB;
+
+	return mt9v034_set_power(subdev, 1);
+}
+
+static int mt9v034_close(struct v4l2_subdev *subdev, struct
v4l2_subdev_fh *fh)
+{
+	return mt9v034_set_power(subdev, 0);
+}
+
+static struct v4l2_subdev_core_ops mt9v034_subdev_core_ops = {
+	.s_power	= mt9v034_set_power,
+};
+
+static struct v4l2_subdev_video_ops mt9v034_subdev_video_ops = {
+	.s_stream	= mt9v034_s_stream,
+};
+
+static struct v4l2_subdev_pad_ops mt9v034_subdev_pad_ops = {
+	.enum_mbus_code = mt9v034_enum_mbus_code,
+	.enum_frame_size = mt9v034_enum_frame_size,
+	.get_fmt = mt9v034_get_format,
+	.set_fmt = mt9v034_set_format,
+	.get_crop = mt9v034_get_crop,
+	.set_crop = mt9v034_set_crop,
+};
+
+static struct v4l2_subdev_ops mt9v034_subdev_ops = {
+	.core	= &mt9v034_subdev_core_ops,
+	.video	= &mt9v034_subdev_video_ops,
+	.pad	= &mt9v034_subdev_pad_ops,
+};
+
+static const struct v4l2_subdev_internal_ops
mt9v034_subdev_internal_ops = {
+	.registered = mt9v034_registered,
+	.open = mt9v034_open,
+	.close = mt9v034_close,
+};
+
+/*
------------------------------------------------------------------------
-----
+ * Driver initialization and probing
+ */
+
+static int mt9v034_probe(struct i2c_client *client,
+		const struct i2c_device_id *did)
+{
+	struct mt9v034 *mt9v034;
+	unsigned int i;
+	int ret;
+	
+	DPRINT("mt9v034_probe");
+
+	if (!i2c_check_functionality(client->adapter,
I2C_FUNC_SMBUS_WORD_DATA)) {
+		dev_warn(&client->adapter->dev,
+			 "I2C-Adapter doesn't support
I2C_FUNC_SMBUS_WORD\n");
+		return -EIO;
+	}
+
+	mt9v034 = kzalloc(sizeof(*mt9v034), GFP_KERNEL);
+	if (!mt9v034)
+		return -ENOMEM;
+
+	mutex_init(&mt9v034->power_lock);
+	mt9v034->pdata = client->dev.platform_data;
+
+	v4l2_ctrl_handler_init(&mt9v034->ctrls,
ARRAY_SIZE(mt9v034_ctrls) + 4);
+
+	v4l2_ctrl_new_std(&mt9v034->ctrls, &mt9v034_ctrl_ops,
+			  V4L2_CID_AUTOGAIN, 0, 1, 1, 0);
+	v4l2_ctrl_new_std(&mt9v034->ctrls, &mt9v034_ctrl_ops,
+			  V4L2_CID_GAIN, MT9V034_ANALOG_GAIN_MIN,
+			  MT9V034_ANALOG_GAIN_MAX, 1,
MT9V034_ANALOG_GAIN_DEF);
+	v4l2_ctrl_new_std_menu(&mt9v034->ctrls, &mt9v034_ctrl_ops,
+			       V4L2_CID_EXPOSURE_AUTO,
V4L2_EXPOSURE_MANUAL, 0,
+			       V4L2_EXPOSURE_MANUAL);
+	v4l2_ctrl_new_std(&mt9v034->ctrls, &mt9v034_ctrl_ops,
+			  V4L2_CID_EXPOSURE,
MT9V034_TOTAL_SHUTTER_WIDTH_MIN_US,
+			  MT9V034_TOTAL_SHUTTER_WIDTH_MAX_US, 1,
+			  MT9V034_TOTAL_SHUTTER_WIDTH_DEF_US);
+
+	for (i = 0; i < ARRAY_SIZE(mt9v034_ctrls); ++i)
+		v4l2_ctrl_new_custom(&mt9v034->ctrls, &mt9v034_ctrls[i],
NULL);
+
+	mt9v034->subdev.ctrl_handler = &mt9v034->ctrls;
+
+	if (mt9v034->ctrls.error)
+		printk(KERN_INFO "%s: control initialization error
%d\n",
+		       __func__, mt9v034->ctrls.error);
+
+	mt9v034->crop.left = MT9V034_COLUMN_START_DEF;
+	mt9v034->crop.top = MT9V034_ROW_START_DEF;
+	mt9v034->crop.width = MT9V034_WINDOW_WIDTH_DEF;
+	mt9v034->crop.height = MT9V034_WINDOW_HEIGHT_DEF;
+
+	mt9v034->format.code = V4L2_MBUS_FMT_SGRBG10_1X10;
+	
+	mt9v034->format.width = MT9V034_WINDOW_WIDTH_DEF;
+	mt9v034->format.height = MT9V034_WINDOW_HEIGHT_DEF;
+	mt9v034->format.field = V4L2_FIELD_NONE;
+	mt9v034->format.colorspace = V4L2_COLORSPACE_SRGB;
+
+	mt9v034->aec_agc = 0; // Disable AEC and AGC
+  mt9v034->isReady = 0;
+  mt9v034->exposure_time_us = MT9V034_EXPOSURE_TIME_USEC;
+  mt9v034->analog_gain = MT9V034_ANALOG_GAIN_DEF;
+	
+	v4l2_i2c_subdev_init(&mt9v034->subdev, client,
&mt9v034_subdev_ops);
+	mt9v034->subdev.internal_ops = &mt9v034_subdev_internal_ops;
+	mt9v034->subdev.flags |= V4L2_SUBDEV_FL_HAS_DEVNODE;
+
+	mt9v034->pad.flags = MEDIA_PAD_FL_SOURCE;
+	ret = media_entity_init(&mt9v034->subdev.entity, 1,
&mt9v034->pad, 0);
+
+  if (ret < 0)
+    kfree(mt9v034);	
+  else
+    g_mt9v034 = mt9v034;
+    
+  return ret;
+}
+
+static int mt9v034_remove(struct i2c_client *client)
+{
+	struct v4l2_subdev *subdev = i2c_get_clientdata(client);
+	struct mt9v034 *mt9v034 = to_mt9v034(subdev);
+
+	v4l2_device_unregister_subdev(subdev);
+	media_entity_cleanup(&subdev->entity);
+
+	kfree(mt9v034);
+	
+	g_mt9v034 = 0;
+	
+	return 0;
+}
+
+static const struct i2c_device_id mt9v034_id[] = {
+	{ "mt9v034", 0 },
+	{ }
+};
+MODULE_DEVICE_TABLE(i2c, mt9v034_id);
+
+static struct i2c_driver mt9v034_driver = {
+	.driver = {
+		.name = "mt9v034",
+	},
+	.probe		= mt9v034_probe,
+	.remove		= mt9v034_remove,
+	.id_table	= mt9v034_id,
+};
+
+static int __init mt9v034_init(void)
+{
+	return i2c_add_driver(&mt9v034_driver);
+}
+
+static void __exit mt9v034_exit(void)
+{
+	i2c_del_driver(&mt9v034_driver);
+}
+
+module_init(mt9v034_init);
+module_exit(mt9v034_exit);
+
+MODULE_DESCRIPTION("Aptina MT9V034 Camera driver");
+MODULE_AUTHOR("Laurent Pinchart <laurent.pinchart@xxxxxxxxxxxxxxxx>");
+MODULE_LICENSE("GPL");
diff -urNp a/include/media/mt9v034.h b/include/media/mt9v034.h
--- a/include/media/mt9v034.h	1970-01-01 01:00:00.000000000 +0100
+++ b/include/media/mt9v034.h	2011-05-30 15:04:44.000000000 +0200
@@ -0,0 +1,15 @@
+#ifndef _MEDIA_MT9V034_H
+#define _MEDIA_MT9V034_H
+
+struct v4l2_subdev;
+
+struct mt9v034_platform_data {
+  unsigned int clk_pol:1;
+	
+  void (*set_clock)(struct v4l2_subdev *subdev, unsigned int rate);
+  void (*configure)(struct v4l2_subdev *subdev);
+  void (*take_picture)(struct v4l2_subdev *subdev, int nr_pics, int
exposure_time_ms);
+  void (*reset)(void);
+};
+
+#endif
 
--------------------------------------------------------

Hope it comes to some use. :)

Regards,

Daniel Lundborg
--
To unsubscribe from this list: send the line "unsubscribe linux-media" in
the body of a message to majordomo@xxxxxxxxxxxxxxx
More majordomo info at  http://vger.kernel.org/majordomo-info.html


[Index of Archives]     [Linux Input]     [Video for Linux]     [Gstreamer Embedded]     [Mplayer Users]     [Linux USB Devel]     [Linux Audio Users]     [Linux Kernel]     [Linux SCSI]     [Yosemite Backpacking]
  Powered by Linux