diff options
Diffstat (limited to 'drivers')
-rw-r--r-- | drivers/media/video/Kconfig | 6 | ||||
-rw-r--r-- | drivers/media/video/Makefile | 1 | ||||
-rw-r--r-- | drivers/media/video/ov5640.c | 2135 | ||||
-rw-r--r-- | drivers/media/video/ov5650.c | 1406 | ||||
-rw-r--r-- | drivers/media/video/tegra/sh532u.c | 4 | ||||
-rw-r--r-- | drivers/media/video/videobuf2-dma-nvmap.c | 15 | ||||
-rw-r--r-- | drivers/net/Kconfig | 27 | ||||
-rw-r--r-- | drivers/net/r8169.c | 4 | ||||
-rw-r--r-- | drivers/video/tegra/host/Makefile | 4 | ||||
-rw-r--r-- | drivers/video/tegra/nvmap/nvmap_dev.c | 12 |
10 files changed, 2746 insertions, 868 deletions
diff --git a/drivers/media/video/Kconfig b/drivers/media/video/Kconfig index d092e31dfaa4..0fe628f5912d 100644 --- a/drivers/media/video/Kconfig +++ b/drivers/media/video/Kconfig @@ -872,6 +872,12 @@ config SOC_CAMERA_OV5642 help This is a V4L2 camera driver for the OmniVision OV5642 sensor +config SOC_CAMERA_OV5650 + tristate "ov5650 sensor support" + depends on SOC_CAMERA && I2C + help + This is a V4L2 camera driver for the OmniVision OV5650 sensor + config SOC_CAMERA_OV6650 tristate "ov6650 sensor support" depends on SOC_CAMERA && I2C diff --git a/drivers/media/video/Makefile b/drivers/media/video/Makefile index 27e6eb745ee1..28362059e105 100644 --- a/drivers/media/video/Makefile +++ b/drivers/media/video/Makefile @@ -82,6 +82,7 @@ obj-$(CONFIG_SOC_CAMERA_MT9V022) += mt9v022.o obj-$(CONFIG_SOC_CAMERA_OV2640) += ov2640.o obj-$(CONFIG_SOC_CAMERA_OV5640) += ov5640.o obj-$(CONFIG_SOC_CAMERA_OV5642) += ov5642.o +obj-$(CONFIG_SOC_CAMERA_OV5650) += ov5650.o obj-$(CONFIG_SOC_CAMERA_OV6650) += ov6650.o obj-$(CONFIG_SOC_CAMERA_OV772X) += ov772x.o obj-$(CONFIG_SOC_CAMERA_OV9640) += ov9640.o diff --git a/drivers/media/video/ov5640.c b/drivers/media/video/ov5640.c index 8e45fa3de5d2..04b0443a4d22 100644 --- a/drivers/media/video/ov5640.c +++ b/drivers/media/video/ov5640.c @@ -1,5 +1,5 @@ /* - * Copyright (c) 2012, NVIDIA CORPORATION. All rights reserved. + * Copyright (c) 2012-2013, NVIDIA CORPORATION. All rights reserved. * * This program is free software; you can redistribute it and/or modify it * under the terms and conditions of the GNU General Public License, @@ -14,6 +14,7 @@ * along with this program. If not, see <http://www.gnu.org/licenses/>. */ +#include <linux/delay.h> #include <linux/init.h> #include <linux/module.h> #include <linux/i2c.h> @@ -21,793 +22,1324 @@ #include <media/v4l2-chip-ident.h> #include <media/soc_camera.h> -#define to_ov5640(sd) container_of(sd, struct ov5640_priv, subdev) - -#define OV5640_SYSTEM_CTRL 0x3008 -#define OV5640_CHIP_ID_HI 0x300a -#define OV5640_CHIP_ID_LO 0x300b -#define OV5640_PAD_OUTPUT_ENABLE00 0x3016 -#define OV5640_PAD_OUTPUT_ENABLE01 0x3017 -#define OV5640_PAD_OUTPUT_ENABLE02 0x3018 -#define OV5640_SC_PLL_CTRL0 0x3034 -#define OV5640_SC_PLL_CTRL1 0x3035 -#define OV5640_SC_PLL_CTRL2 0x3036 -#define OV5640_SC_PLL_CTRL3 0x3037 - -/* SCCB Control */ -#define OV5640_SCCB_SYSTEM_CTRL1 0x3103 -#define OV5640_SYSTEM_ROOT_DIVIDER 0x3108 - -/* Timing Control */ -#define OV5640_TIMING_HS_HI 0x3800 -#define OV5640_TIMING_HS_LO 0x3801 -#define OV5640_TIMING_VS_HI 0x3802 -#define OV5640_TIMING_VS_LO 0x3803 -#define OV5640_TIMING_HW_HI 0x3804 -#define OV5640_TIMING_HW_LO 0x3805 -#define OV5640_TIMING_VH_HI 0x3806 -#define OV5640_TIMING_VH_LO 0x3807 -#define OV5640_TIMING_DVPHO_HI 0x3808 -#define OV5640_TIMING_DVPHO_LO 0x3809 -#define OV5640_TIMING_DVPVO_HI 0x380a -#define OV5640_TIMING_DVPVO_LO 0x380b -#define OV5640_TIMING_HTS_HI 0x380c -#define OV5640_TIMING_HTS_LO 0x380d -#define OV5640_TIMING_VTS_HI 0x380e -#define OV5640_TIMING_VTS_LO 0x380f -#define OV5640_TIMING_HOFFSET_HI 0x3810 -#define OV5640_TIMING_HOFFSET_LO 0x3811 -#define OV5640_TIMING_VOFFSET_HI 0x3812 -#define OV5640_TIMING_VOFFSET_LO 0x3813 -#define OV5640_TIMING_X_INC 0x3814 -#define OV5640_TIMING_Y_INC 0x3815 -#define OV5640_TIMING_TC_REG20 0x3820 -#define OV5640_TIMING_TC_REG21 0x3821 - -/* AEC/AGC Power Down Domain Control */ -#define OV5640_AEC_MAX_EXPO_60HZ_HI 0x3a02 -#define OV5640_AEC_MAX_EXPO_60HZ_LO 0x3a03 -#define OV5640_AEC_B50_STEP_HI 0x3a08 -#define OV5640_AEC_B50_STEP_LO 0x3a09 -#define OV5640_AEC_B60_STEP_HI 0x3a0a -#define OV5640_AEC_B60_STEP_LO 0x3a0b -#define OV5640_AEC_CTRL0C 0x3a0c -#define OV5640_AEC_CTRL0D 0x3a0d -#define OV5640_AEC_CTRL0E 0x3a0e -#define OV5640_AEC_CTRL0F 0x3a0f -#define OV5640_AEC_CTRL10 0x3a10 -#define OV5640_AEC_CTRL11 0x3a11 -#define OV5640_AEC_CTRL12 0x3a12 -#define OV5640_AEC_CTRL13 0x3a13 -#define OV5640_AEC_MAX_EXPO_50HZ_HI 0x3a14 -#define OV5640_AEC_MAX_EXPO_50HZ_LO 0x3a15 -#define OV5640_AEC_GAIN_CEILING_HI 0x3a18 -#define OV5640_AEC_GAIN_CEILING_LO 0x3a19 -#define OV5640_AEC_CTRL1B 0x3a1b -#define OV5640_AEC_CTRL1E 0x3a1e -#define OV5640_AEC_CTRL1F 0x3a1f - -/* 50/60Hz Detector Control */ -#define OV5640_5060HZ_CTRL00 0x3c00 -#define OV5640_5060HZ_CTRL01 0x3c01 -#define OV5640_5060HZ_CTRL02 0x3c02 -#define OV5640_5060HZ_CTRL03 0x3c03 -#define OV5640_5060HZ_CTRL04 0x3c04 -#define OV5640_5060HZ_CTRL05 0x3c05 -#define OV5640_LIGHT_METER1_THRESHOLD_HI 0x3c06 -#define OV5640_LIGHT_METER1_THRESHOLD_LO 0x3c07 -#define OV5640_LIGHT_METER2_THRESHOLD_HI 0x3c08 -#define OV5640_LIGHT_METER2_THRESHOLD_LO 0x3c09 -#define OV5640_SAMPLE_NUMBER_HI 0x3c0a -#define OV5640_SAMPLE_NUMBER_LO 0x3c0b - -/* ISP General Controls */ -#define OV5640_ISP_CTRL00 0x5000 -#define OV5640_ISP_CTRL01 0x5001 -#define OV5640_ISP_CTRL37 0x5025 - -/* AWB Control */ -#define OV5640_AWB_CTRL00 0x5180 -#define OV5640_AWB_CTRL01 0x5181 -#define OV5640_AWB_CTRL02 0x5182 -#define OV5640_AWB_CTRL03 0x5183 -#define OV5640_AWB_CTRL04 0x5184 -#define OV5640_AWB_CTRL05 0x5185 -#define OV5640_AWB_CTRL06 0x5186 -#define OV5640_AWB_CTRL07 0x5187 -#define OV5640_AWB_CTRL08 0x5188 -#define OV5640_AWB_CTRL09 0x5189 -#define OV5640_AWB_CTRL10 0x518a -#define OV5640_AWB_CTRL11 0x518b -#define OV5640_AWB_CTRL12 0x518c -#define OV5640_AWB_CTRL13 0x518d -#define OV5640_AWB_CTRL14 0x518e -#define OV5640_AWB_CTRL15 0x518f -#define OV5640_AWB_CTRL16 0x5190 -#define OV5640_AWB_CTRL17 0x5191 -#define OV5640_AWB_CTRL18 0x5192 -#define OV5640_AWB_CTRL19 0x5193 -#define OV5640_AWB_CTRL20 0x5194 -#define OV5640_AWB_CTRL21 0x5195 -#define OV5640_AWB_CTRL22 0x5196 -#define OV5640_AWB_CTRL23 0x5197 -#define OV5640_AWB_CTRL24 0x5198 -#define OV5640_AWB_CTRL25 0x5199 -#define OV5640_AWB_CTRL26 0x519a -#define OV5640_AWB_CTRL27 0x519b -#define OV5640_AWB_CTRL28 0x519c -#define OV5640_AWB_CTRL29 0x519d -#define OV5640_AWB_CTRL30 0x519e - -/* CIP Control */ -#define OV5640_CIP_SHARPENMT_THRESHOLD_1 0x5300 -#define OV5640_CIP_SHARPENMT_THRESHOLD_2 0x5301 -#define OV5640_CIP_SHARPENMT_OFFSET_1 0x5302 -#define OV5640_CIP_SHARPENMT_OFFSET_2 0x5303 -#define OV5640_CIP_DNS_THRESHOLD_1 0x5304 -#define OV5640_CIP_DNS_THRESHOLD_2 0x5305 -#define OV5640_CIP_DNS_OFFSET_1 0x5306 -#define OV5640_CIP_DNS_OFFSET_2 0x5307 -#define OV5640_CIP_CTRL 0x5308 -#define OV5640_CIP_SHARPENTH_THRESHOLD_1 0x5309 -#define OV5640_CIP_SHARPENTH_THRESHOLD_2 0x530a -#define OV5640_CIP_SHARPENTH_OFFSET_1 0x530b -#define OV5640_CIP_SHARPENTH_OFFSET_2 0x530c -#define OV5640_CIP_EDGE_MT_AUTO 0x530d -#define OV5640_CIP_DNS_THRESHOLD_AUTO 0x530e -#define OF5640_CIP_SHARPEN_THRESHOLD_AUTO 0x530f - -/* CMX Control */ -#define OV5640_CMX_CTRL 0x5380 -#define OV5640_CMX1 0x5381 -#define OV5640_CMX2 0x5382 -#define OV5640_CMX3 0x5383 -#define OV5640_CMX4 0x5384 -#define OV5640_CMX5 0x5385 -#define OV5640_CMX6 0x5386 -#define OV5640_CMX7 0x5387 -#define OV5640_CMX8 0x5388 -#define OV5640_CMX9 0x5389 -#define OV5640_CMXSIGN_HI 0x538a -#define OV5640_CMXSIGN_LO 0x538b - -/* Gamma Control */ -#define OV5640_GAMMA_CTRL00 0x5480 -#define OV5640_YST00 0x5481 -#define OV5640_YST01 0x5482 -#define OV5640_YST02 0x5483 -#define OV5640_YST03 0x5484 -#define OV5640_YST04 0x5485 -#define OV5640_YST05 0x5486 -#define OV5640_YST06 0x5487 -#define OV5640_YST07 0x5488 -#define OV5640_YST08 0x5489 -#define OV5640_YST09 0x548a -#define OV5640_YST0A 0x548b -#define OV5640_YST0B 0x548c -#define OV5640_YST0C 0x548d -#define OV5640_YST0D 0x548e -#define OV5640_YST0E 0x548f -#define OV5640_YST0F 0x5490 - -/* SDE Control */ -#define OV5640_SDE_CTRL_0 0x5580 -#define OV5640_SDE_CTRL_1 0x5581 -#define OV5640_SDE_CTRL_2 0x5582 -#define OV5640_SDE_CTRL_3 0x5583 -#define OV5640_SDE_CTRL_4 0x5584 -#define OV5640_SDE_CTRL_5 0x5585 -#define OV5640_SDE_CTRL_6 0x5586 -#define OV5640_SDE_CTRL_7 0x5587 -#define OV5640_SDE_CTRL_8 0x5588 -#define OV5640_SDE_CTRL_9 0x5589 -#define OV5640_SDE_CTRL_10 0x558a -#define OV5640_SDE_CTRL_11 0x558b -#define OV5640_SDE_CTRL_12 0x558c - -/* LENC Control */ -#define OV5640_GMTRX00 0x5800 -#define OV5640_GMTRX01 0x5801 -#define OV5640_GMTRX02 0x5802 -#define OV5640_GMTRX03 0x5803 -#define OV5640_GMTRX04 0x5804 -#define OV5640_GMTRX05 0x5805 -#define OV5640_GMTRX10 0x5806 -#define OV5640_GMTRX11 0x5807 -#define OV5640_GMTRX12 0x5808 -#define OV5640_GMTRX13 0x5809 -#define OV5640_GMTRX14 0x580a -#define OV5640_GMTRX15 0x580b -#define OV5640_GMTRX20 0x580c -#define OV5640_GMTRX21 0x580d -#define OV5640_GMTRX22 0x580e -#define OV5640_GMTRX23 0x580f -#define OV5640_GMTRX24 0x5810 -#define OV5640_GMTRX25 0x5811 -#define OV5640_GMTRX30 0x5812 -#define OV5640_GMTRX31 0x5813 -#define OV5640_GMTRX32 0x5814 -#define OV5640_GMTRX33 0x5815 -#define OV5640_GMTRX34 0x5816 -#define OV5640_GMTRX35 0x5817 -#define OV5640_GMTRX40 0x5818 -#define OV5640_GMTRX41 0x5819 -#define OV5640_GMTRX42 0x581a -#define OV5640_GMTRX43 0x581b -#define OV5640_GMTRX44 0x581c -#define OV5640_GMTRX45 0x581d -#define OV5640_GMTRX50 0x581e -#define OV5640_GMTRX51 0x581f -#define OV5640_GMTRX52 0x5820 -#define OV5640_GMTRX53 0x5821 -#define OV5640_GMTRX54 0x5822 -#define OV5640_GMTRX55 0x5823 -#define OV5640_BRMATRX00 0x5824 -#define OV5640_BRMATRX01 0x5825 -#define OV5640_BRMATRX02 0x5826 -#define OV5640_BRMATRX03 0x5827 -#define OV5640_BRMATRX04 0x5828 -#define OV5640_BRMATRX05 0x5829 -#define OV5640_BRMATRX06 0x582a -#define OV5640_BRMATRX07 0x582b -#define OV5640_BRMATRX08 0x582c -#define OV5640_BRMATRX09 0x582d -#define OV5640_BRMATRX20 0x582e -#define OV5640_BRMATRX21 0x582f -#define OV5640_BRMATRX22 0x5830 -#define OV5640_BRMATRX23 0x5831 -#define OV5640_BRMATRX24 0x5832 -#define OV5640_BRMATRX30 0x5833 -#define OV5640_BRMATRX31 0x5834 -#define OV5640_BRMATRX32 0x5835 -#define OV5640_BRMATRX33 0x5836 -#define OV5640_BRMATRX34 0x5837 -#define OV5640_BRMATRX40 0x5838 -#define OV5640_BRMATRX41 0x5839 -#define OV5640_BRMATRX42 0x583a -#define OV5640_BRMATRX43 0x583b -#define OV5640_BRMATRX44 0x583c -#define OV5640_LENC_BR_OFFSET 0x583d - -#define OV5640_MAX_WIDTH 640 -#define OV5640_MAX_HEIGHT 480 - -/* Misc. structures */ struct ov5640_reg { - u16 reg; - u8 val; + u16 addr; + u16 val; }; -struct ov5640_priv { - struct v4l2_subdev subdev; +#define OV5640_TABLE_WAIT_MS 0 +#define OV5640_TABLE_END 1 - int ident; - u16 chip_id; - u8 revision; - u8 manid; - u8 smiaver; +static struct ov5640_reg mode_2592x1944[] = { + /* PLL Control MIPI bit rate/lane = 672MHz, 16-bit mode. + * Output size: 2608x1948 (0, 0) - (2623, 1951), + * Line Length = 2844, Frame Length = 1968 + */ + {0x3103, 0x11}, + {0x3008, 0x82}, + {OV5640_TABLE_WAIT_MS, 5}, + {0x3008, 0x42}, + {0x3103, 0x03}, + {0x3017, 0x00}, + {0x3018, 0x00}, + {0x3034, 0x18}, + {0x3035, 0x11}, + {0x3036, 0x7d}, + {0x3037, 0x13}, + {0x3108, 0x01}, + {0x3630, 0x36}, + {0x3631, 0x0e}, + {0x3632, 0xe2}, + {0x3633, 0x12}, + {0x3621, 0xe0}, + {0x3704, 0xa0}, + {0x3703, 0x5a}, + {0x3715, 0x78}, + {0x3717, 0x01}, + {0x370b, 0x60}, + {0x3705, 0x1a}, + {0x3905, 0x02}, + {0x3906, 0x10}, + {0x3901, 0x0a}, + {0x3731, 0x12}, + {0x3600, 0x08}, + {0x3601, 0x33}, + {0x302d, 0x60}, + {0x3620, 0x52}, + {0x371b, 0x20}, + {0x471c, 0x50}, + {0x3a13, 0x43}, + {0x3a18, 0x00}, + {0x3a19, 0xf8}, + {0x3635, 0x13}, + {0x3636, 0x03}, + {0x3634, 0x40}, + {0x3622, 0x01}, + {0x3c01, 0x34}, + {0x3c04, 0x28}, + {0x3c05, 0x98}, + {0x3c06, 0x00}, + {0x3c07, 0x07}, + {0x3c08, 0x00}, + {0x3c09, 0x1c}, + {0x3c0a, 0x9c}, + {0x3c0b, 0x40}, + {0x3820, 0x40}, + {0x3821, 0x06}, + {0x3814, 0x11}, + {0x3815, 0x11}, + {0x3800, 0x00}, + {0x3801, 0x00}, + {0x3802, 0x00}, + {0x3803, 0x00}, + {0x3804, 0x0a}, + {0x3805, 0x3f}, + {0x3806, 0x07}, + {0x3807, 0x9f}, + {0x3808, 0x0a}, + {0x3809, 0x30}, + {0x380a, 0x07}, + {0x380b, 0x9c}, + {0x380c, 0x0b}, + {0x380d, 0x1c}, + {0x380e, 0x07}, + {0x380f, 0xb0}, + {0x3810, 0x00}, + {0x3811, 0x08}, + {0x3812, 0x00}, + {0x3813, 0x02}, + {0x3618, 0x04}, + {0x3612, 0x2b}, + {0x3708, 0x64}, + {0x3709, 0x12}, + {0x370c, 0x00}, + {0x3a02, 0x07}, + {0x3a03, 0xb0}, + {0x3a08, 0x01}, + {0x3a09, 0x27}, + {0x3a0a, 0x00}, + {0x3a0b, 0xf6}, + {0x3a0e, 0x06}, + {0x3a0d, 0x08}, + {0x3a14, 0x07}, + {0x3a15, 0xb0}, + {0x4001, 0x02}, + {0x4004, 0x06}, + {0x3000, 0x00}, + {0x3002, 0x1c}, + {0x3004, 0xff}, + {0x3006, 0xc3}, + {0x300e, 0x45}, + {0x302e, 0x08}, + {0x4300, 0x32}, + {0x4800, 0x24}, + {0x4837, 0x0a}, + {0x501f, 0x00}, + {0x440e, 0x00}, + {0x5000, 0xa7}, + {0x5001, 0x83}, + {0x5180, 0xff}, + {0x5181, 0xf2}, + {0x5182, 0x00}, + {0x5183, 0x14}, + {0x5184, 0x25}, + {0x5185, 0x24}, + {0x5186, 0x09}, + {0x5187, 0x09}, + {0x5188, 0x09}, + {0x5189, 0x75}, + {0x518a, 0x54}, + {0x518b, 0xe0}, + {0x518c, 0xb2}, + {0x518d, 0x42}, + {0x518e, 0x3d}, + {0x518f, 0x56}, + {0x5190, 0x46}, + {0x5191, 0xf8}, + {0x5192, 0x04}, + {0x5193, 0x70}, + {0x5194, 0xf0}, + {0x5195, 0xf0}, + {0x5196, 0x03}, + {0x5197, 0x01}, + {0x5198, 0x04}, + {0x5199, 0x12}, + {0x519a, 0x04}, + {0x519b, 0x00}, + {0x519c, 0x06}, + {0x519d, 0x82}, + {0x519e, 0x38}, + {0x5381, 0x1e}, + {0x5382, 0x5b}, + {0x5383, 0x08}, + {0x5384, 0x0a}, + {0x5385, 0x7e}, + {0x5386, 0x88}, + {0x5387, 0x7c}, + {0x5388, 0x6c}, + {0x5389, 0x10}, + {0x538a, 0x01}, + {0x538b, 0x98}, + {0x5300, 0x08}, + {0x5301, 0x30}, + {0x5302, 0x10}, + {0x5303, 0x00}, + {0x5304, 0x08}, + {0x5305, 0x30}, + {0x5306, 0x08}, + {0x5307, 0x16}, + {0x5309, 0x08}, + {0x530a, 0x30}, + {0x530b, 0x04}, + {0x530c, 0x06}, + {0x5480, 0x01}, + {0x5481, 0x08}, + {0x5482, 0x14}, + {0x5483, 0x28}, + {0x5484, 0x51}, + {0x5485, 0x65}, + {0x5486, 0x71}, + {0x5487, 0x7d}, + {0x5488, 0x87}, + {0x5489, 0x91}, + {0x548a, 0x9a}, + {0x548b, 0xaa}, + {0x548c, 0xb8}, + {0x548d, 0xcd}, + {0x548e, 0xdd}, + {0x548f, 0xea}, + {0x5490, 0x1d}, + {0x5580, 0x02}, + {0x5583, 0x40}, + {0x5584, 0x10}, + {0x5589, 0x10}, + {0x558a, 0x00}, + {0x558b, 0xf8}, + {0x5800, 0x23}, + {0x5801, 0x14}, + {0x5802, 0x0f}, + {0x5803, 0x0f}, + {0x5804, 0x12}, + {0x5805, 0x26}, + {0x5806, 0x0c}, + {0x5807, 0x08}, + {0x5808, 0x05}, + {0x5809, 0x05}, + {0x580a, 0x08}, + {0x580b, 0x0d}, + {0x580c, 0x08}, + {0x580d, 0x03}, + {0x580e, 0x00}, + {0x580f, 0x00}, + {0x5810, 0x03}, + {0x5811, 0x09}, + {0x5812, 0x07}, + {0x5813, 0x03}, + {0x5814, 0x00}, + {0x5815, 0x01}, + {0x5816, 0x03}, + {0x5817, 0x08}, + {0x5818, 0x0d}, + {0x5819, 0x08}, + {0x581a, 0x05}, + {0x581b, 0x06}, + {0x581c, 0x08}, + {0x581d, 0x0e}, + {0x581e, 0x29}, + {0x581f, 0x17}, + {0x5820, 0x11}, + {0x5821, 0x11}, + {0x5822, 0x15}, + {0x5823, 0x28}, + {0x5824, 0x46}, + {0x5825, 0x26}, + {0x5826, 0x08}, + {0x5827, 0x26}, + {0x5828, 0x64}, + {0x5829, 0x26}, + {0x582a, 0x24}, + {0x582b, 0x22}, + {0x582c, 0x24}, + {0x582d, 0x24}, + {0x582e, 0x06}, + {0x582f, 0x22}, + {0x5830, 0x40}, + {0x5831, 0x42}, + {0x5832, 0x24}, + {0x5833, 0x26}, + {0x5834, 0x24}, + {0x5835, 0x22}, + {0x5836, 0x22}, + {0x5837, 0x26}, + {0x5838, 0x44}, + {0x5839, 0x24}, + {0x583a, 0x26}, + {0x583b, 0x28}, + {0x583c, 0x42}, + {0x583d, 0xce}, + {0x5025, 0x00}, + {0x3a0f, 0x30}, + {0x3a10, 0x28}, + {0x3a1b, 0x30}, + {0x3a1e, 0x26}, + {0x3a11, 0x60}, + {0x3a1f, 0x14}, + {0x3008, 0x02}, + {OV5640_TABLE_END, 0x0000} +}; - bool flag_vflip; - bool flag_hflip; +static struct ov5640_reg mode_1920x1088[] = { + /* PLL Control MIPI bit rate/lane = 672MHz, 16-bit mode. + * Output size: 1936x1096 (336, 426) - (2287, 1529), + * Line Length = 2500, Frame Length = 1120. + */ + {0x3103, 0x11}, + {0x3008, 0x82}, + {OV5640_TABLE_WAIT_MS, 5}, + {0x3008, 0x42}, + {0x3103, 0x03}, + {0x3017, 0x00}, + {0x3018, 0x00}, + {0x3034, 0x18}, + {0x3035, 0x11}, + {0x3036, 0x54}, + {0x3037, 0x13}, + {0x3108, 0x01}, + {0x3630, 0x36}, + {0x3631, 0x0e}, + {0x3632, 0xe2}, + {0x3633, 0x12}, + {0x3621, 0xe0}, + {0x3704, 0xa0}, + {0x3703, 0x5a}, + {0x3715, 0x78}, + {0x3717, 0x01}, + {0x370b, 0x60}, + {0x3705, 0x1a}, + {0x3905, 0x02}, + {0x3906, 0x10}, + {0x3901, 0x0a}, + {0x3731, 0x12}, + {0x3600, 0x08}, + {0x3601, 0x33}, + {0x302d, 0x60}, + {0x3620, 0x52}, + {0x371b, 0x20}, + {0x471c, 0x50}, + {0x3a13, 0x43}, + {0x3a18, 0x00}, + {0x3a19, 0xf8}, + {0x3635, 0x13}, + {0x3636, 0x03}, + {0x3634, 0x40}, + {0x3622, 0x01}, + {0x3c01, 0x34}, + {0x3c04, 0x28}, + {0x3c05, 0x98}, + {0x3c06, 0x00}, + {0x3c07, 0x07}, + {0x3c08, 0x00}, + {0x3c09, 0x1c}, + {0x3c0a, 0x9c}, + {0x3c0b, 0x40}, + {0x3820, 0x40}, + {0x3821, 0x06}, + {0x3814, 0x11}, + {0x3815, 0x11}, + {0x3800, 0x01}, + {0x3801, 0x50}, + {0x3802, 0x01}, + {0x3803, 0xaa}, + {0x3804, 0x08}, + {0x3805, 0xef}, + {0x3806, 0x05}, + {0x3807, 0xf9}, + {0x3808, 0x07}, + {0x3809, 0x90}, + {0x380a, 0x04}, + {0x380b, 0x48}, + {0x380c, 0x09}, + {0x380d, 0xc4}, + {0x380e, 0x04}, + {0x380f, 0x60}, + {0x3810, 0x00}, + {0x3811, 0x08}, + {0x3812, 0x00}, + {0x3813, 0x04}, + {0x3618, 0x04}, + {0x3612, 0x2b}, + {0x3708, 0x64}, + {0x3709, 0x12}, + {0x370c, 0x00}, + {0x3a02, 0x04}, + {0x3a03, 0x60}, + {0x3a08, 0x01}, + {0x3a09, 0x50}, + {0x3a0a, 0x01}, + {0x3a0b, 0x18}, + {0x3a0e, 0x03}, + {0x3a0d, 0x04}, + {0x3a14, 0x04}, + {0x3a15, 0x60}, + {0x4001, 0x02}, + {0x4004, 0x06}, + {0x3000, 0x00}, + {0x3002, 0x1c}, + {0x3004, 0xff}, + {0x3006, 0xc3}, + {0x300e, 0x45}, + {0x302e, 0x08}, + {0x4300, 0x32}, + {0x501f, 0x00}, + {0x4713, 0x02}, + {0x4407, 0x04}, + {0x440e, 0x00}, + {0x460b, 0x37}, + {0x460c, 0x20}, + {0x4800, 0x24}, + {0x4837, 0x0a}, + {0x3824, 0x04}, + {0x5000, 0xa7}, + {0x5001, 0x83}, + {0x5180, 0xff}, + {0x5181, 0xf2}, + {0x5182, 0x00}, + {0x5183, 0x14}, + {0x5184, 0x25}, + {0x5185, 0x24}, + {0x5186, 0x09}, + {0x5187, 0x09}, + {0x5188, 0x09}, + {0x5189, 0x75}, + {0x518a, 0x54}, + {0x518b, 0xe0}, + {0x518c, 0xb2}, + {0x518d, 0x42}, + {0x518e, 0x3d}, + {0x518f, 0x56}, + {0x5190, 0x46}, + {0x5191, 0xf8}, + {0x5192, 0x04}, + {0x5193, 0x70}, + {0x5194, 0xf0}, + {0x5195, 0xf0}, + {0x5196, 0x03}, + {0x5197, 0x01}, + {0x5198, 0x04}, + {0x5199, 0x12}, + {0x519a, 0x04}, + {0x519b, 0x00}, + {0x519c, 0x06}, + {0x519d, 0x82}, + {0x519e, 0x38}, + {0x5381, 0x1e}, + {0x5382, 0x5b}, + {0x5383, 0x08}, + {0x5384, 0x0a}, + {0x5385, 0x7e}, + {0x5386, 0x88}, + {0x5387, 0x7c}, + {0x5388, 0x6c}, + {0x5389, 0x10}, + {0x538a, 0x01}, + {0x538b, 0x98}, + {0x5300, 0x08}, + {0x5301, 0x30}, + {0x5302, 0x10}, + {0x5303, 0x00}, + {0x5304, 0x08}, + {0x5305, 0x30}, + {0x5306, 0x08}, + {0x5307, 0x16}, + {0x5309, 0x08}, + {0x530a, 0x30}, + {0x530b, 0x04}, + {0x530c, 0x06}, + {0x5480, 0x01}, + {0x5481, 0x08}, + {0x5482, 0x14}, + {0x5483, 0x28}, + {0x5484, 0x51}, + {0x5485, 0x65}, + {0x5486, 0x71}, + {0x5487, 0x7d}, + + {0x5488, 0x87}, + {0x5489, 0x91}, + {0x548a, 0x9a}, + {0x548b, 0xaa}, + {0x548c, 0xb8}, + {0x548d, 0xcd}, + {0x548e, 0xdd}, + {0x548f, 0xea}, + {0x5490, 0x1d}, + {0x5580, 0x02}, + {0x5583, 0x40}, + {0x5584, 0x10}, + {0x5589, 0x10}, + {0x558a, 0x00}, + {0x558b, 0xf8}, + {0x5800, 0x23}, + {0x5801, 0x14}, + {0x5802, 0x0f}, + {0x5803, 0x0f}, + {0x5804, 0x12}, + {0x5805, 0x26}, + {0x5806, 0x0c}, + {0x5807, 0x08}, + {0x5808, 0x05}, + {0x5809, 0x05}, + {0x580a, 0x08}, + {0x580b, 0x0d}, + {0x580c, 0x08}, + {0x580d, 0x03}, + {0x580e, 0x00}, + {0x580f, 0x00}, + {0x5810, 0x03}, + {0x5811, 0x09}, + {0x5812, 0x07}, + {0x5813, 0x03}, + {0x5814, 0x00}, + {0x5815, 0x01}, + {0x5816, 0x03}, + {0x5817, 0x08}, + {0x5818, 0x0d}, + {0x5819, 0x08}, + {0x581a, 0x05}, + {0x581b, 0x06}, + {0x581c, 0x08}, + {0x581d, 0x0e}, + {0x581e, 0x29}, + {0x581f, 0x17}, + {0x5820, 0x11}, + {0x5821, 0x11}, + {0x5822, 0x15}, + {0x5823, 0x28}, + {0x5824, 0x46}, + {0x5825, 0x26}, + {0x5826, 0x08}, + {0x5827, 0x26}, + {0x5828, 0x64}, + {0x5829, 0x26}, + {0x582a, 0x24}, + {0x582b, 0x22}, + {0x582c, 0x24}, + {0x582d, 0x24}, + {0x582e, 0x06}, + {0x582f, 0x22}, + {0x5830, 0x40}, + {0x5831, 0x42}, + {0x5832, 0x24}, + {0x5833, 0x26}, + {0x5834, 0x24}, + {0x5835, 0x22}, + {0x5836, 0x22}, + {0x5837, 0x26}, + {0x5838, 0x44}, + {0x5839, 0x24}, + {0x583a, 0x26}, + {0x583b, 0x28}, + {0x583c, 0x42}, + {0x583d, 0xce}, + {0x5025, 0x00}, + {0x3a0f, 0x30}, + {0x3a10, 0x28}, + {0x3a1b, 0x30}, + {0x3a1e, 0x26}, + {0x3a11, 0x60}, + {0x3a1f, 0x14}, + {0x3008, 0x02}, + {OV5640_TABLE_END, 0x0000} +}; - /* For suspend/resume. */ - struct v4l2_mbus_framefmt current_mf; - bool current_enable; +static struct ov5640_reg mode_1296x972[] = { + /* PLL Control MIPI bit rate/lane = 448MHz, 16-bit mode. + * Output size: 1304x972 (0, 0) - (2623, 1951), + * Line Length = 1886, Frame Length = 990. + */ + {0x3103, 0x11}, + {0x3008, 0x82}, + {OV5640_TABLE_WAIT_MS, 5}, + {0x3008, 0x42}, + {0x3103, 0x03}, + {0x3017, 0x00}, + {0x3018, 0x00}, + {0x3034, 0x18}, + {0x3035, 0x21}, + {0x3036, 0x70}, + {0x3037, 0x13}, + {0x3108, 0x01}, + {0x3630, 0x36}, + {0x3631, 0x0e}, + {0x3632, 0xe2}, + {0x3633, 0x12}, + {0x3621, 0xe0}, + {0x3704, 0xa0}, + {0x3703, 0x5a}, + {0x3715, 0x78}, + {0x3717, 0x01}, + {0x370b, 0x60}, + {0x3705, 0x1a}, + {0x3905, 0x02}, + {0x3906, 0x10}, + {0x3901, 0x0a}, + {0x3731, 0x12}, + {0x3600, 0x08}, + {0x3601, 0x33}, + {0x302d, 0x60}, + {0x3620, 0x52}, + {0x371b, 0x20}, + {0x471c, 0x50}, + {0x3a13, 0x43}, + {0x3a18, 0x00}, + {0x3a19, 0xf8}, + {0x3635, 0x13}, + {0x3636, 0x03}, + {0x3634, 0x40}, + {0x3622, 0x01}, + {0x3c01, 0x34}, + {0x3c04, 0x28}, + {0x3c05, 0x98}, + {0x3c06, 0x00}, + {0x3c07, 0x07}, + {0x3c08, 0x00}, + {0x3c09, 0x1c}, + {0x3c0a, 0x9c}, + {0x3c0b, 0x40}, + {0x3820, 0x41}, + {0x3821, 0x07}, + {0x3814, 0x31}, + {0x3815, 0x31}, + {0x3800, 0x00}, + {0x3801, 0x00}, + {0x3802, 0x00}, + {0x3803, 0x00}, + {0x3804, 0x0a}, + {0x3805, 0x3f}, + {0x3806, 0x07}, + {0x3807, 0x9f}, + {0x3808, 0x05}, + {0x3809, 0x18}, + {0x380a, 0x03}, + {0x380b, 0xcc}, + {0x380c, 0x07}, + {0x380d, 0x5e}, + {0x380e, 0x03}, + {0x380f, 0xde}, + {0x3810, 0x00}, + {0x3811, 0x06}, + {0x3812, 0x00}, + {0x3813, 0x02}, + {0x3618, 0x00}, + {0x3612, 0x29}, + {0x3708, 0x62}, + {0x3709, 0x52}, + {0x370c, 0x03}, + {0x3a02, 0x03}, + {0x3a03, 0xd8}, + {0x3a08, 0x01}, + {0x3a09, 0x27}, + {0x3a0a, 0x00}, + {0x3a0b, 0xf6}, + {0x3a0e, 0x03}, + {0x3a0d, 0x04}, + {0x3a14, 0x03}, + {0x3a15, 0xd8}, + {0x4001, 0x02}, + {0x4004, 0x02}, + {0x3000, 0x00}, + {0x3002, 0x1c}, + {0x3004, 0xff}, + {0x3006, 0xc3}, + {0x300e, 0x45}, + {0x302e, 0x08}, + {0x4300, 0x32}, + {0x501f, 0x00}, + {0x4713, 0x02}, + {0x4407, 0x04}, + {0x440e, 0x00}, + {0x460b, 0x37}, + {0x460c, 0x20}, + {0x4800, 0x24}, + {0x4837, 0x10}, + {0x3824, 0x04}, + {0x5000, 0xa7}, + {0x5001, 0x83}, + {0x5180, 0xff}, + {0x5181, 0xf2}, + {0x5182, 0x00}, + {0x5183, 0x14}, + {0x5184, 0x25}, + {0x5185, 0x24}, + {0x5186, 0x09}, + {0x5187, 0x09}, + {0x5188, 0x09}, + {0x5189, 0x75}, + {0x518a, 0x54}, + {0x518b, 0xe0}, + {0x518c, 0xb2}, + {0x518d, 0x42}, + {0x518e, 0x3d}, + {0x518f, 0x56}, + {0x5190, 0x46}, + {0x5191, 0xf8}, + {0x5192, 0x04}, + {0x5193, 0x70}, + {0x5194, 0xf0}, + {0x5195, 0xf0}, + {0x5196, 0x03}, + {0x5197, 0x01}, + {0x5198, 0x04}, + {0x5199, 0x12}, + {0x519a, 0x04}, + {0x519b, 0x00}, + {0x519c, 0x06}, + {0x519d, 0x82}, + {0x519e, 0x38}, + {0x5381, 0x1e}, + {0x5382, 0x5b}, + {0x5383, 0x08}, + {0x5384, 0x0a}, + {0x5385, 0x7e}, + {0x5386, 0x88}, + {0x5387, 0x7c}, + {0x5388, 0x6c}, + {0x5389, 0x10}, + {0x538a, 0x01}, + {0x538b, 0x98}, + {0x5300, 0x08}, + {0x5301, 0x30}, + {0x5302, 0x10}, + {0x5303, 0x00}, + {0x5304, 0x08}, + {0x5305, 0x30}, + {0x5306, 0x08}, + {0x5307, 0x16}, + {0x5309, 0x08}, + {0x530a, 0x30}, + {0x530b, 0x04}, + {0x530c, 0x06}, + {0x5480, 0x01}, + {0x5481, 0x08}, + {0x5482, 0x14}, + {0x5483, 0x28}, + {0x5484, 0x51}, + {0x5485, 0x65}, + {0x5486, 0x71}, + {0x5487, 0x7d}, + {0x5488, 0x87}, + {0x5489, 0x91}, + {0x548a, 0x9a}, + {0x548b, 0xaa}, + {0x548c, 0xb8}, + {0x548d, 0xcd}, + {0x548e, 0xdd}, + {0x548f, 0xea}, + {0x5490, 0x1d}, + {0x5580, 0x02}, + {0x5583, 0x40}, + {0x5584, 0x10}, + {0x5589, 0x10}, + {0x558a, 0x00}, + {0x558b, 0xf8}, + {0x5800, 0x23}, + {0x5801, 0x14}, + {0x5802, 0x0f}, + {0x5803, 0x0f}, + {0x5804, 0x12}, + {0x5805, 0x26}, + {0x5806, 0x0c}, + {0x5807, 0x08}, + {0x5808, 0x05}, + {0x5809, 0x05}, + {0x580a, 0x08}, + {0x580b, 0x0d}, + {0x580c, 0x08}, + {0x580d, 0x03}, + {0x580e, 0x00}, + {0x580f, 0x00}, + {0x5810, 0x03}, + {0x5811, 0x09}, + {0x5812, 0x07}, + {0x5813, 0x03}, + {0x5814, 0x00}, + {0x5815, 0x01}, + {0x5816, 0x03}, + {0x5817, 0x08}, + {0x5818, 0x0d}, + {0x5819, 0x08}, + {0x581a, 0x05}, + {0x581b, 0x06}, + {0x581c, 0x08}, + {0x581d, 0x0e}, + {0x581e, 0x29}, + {0x581f, 0x17}, + {0x5820, 0x11}, + {0x5821, 0x11}, + {0x5822, 0x15}, + {0x5823, 0x28}, + {0x5824, 0x46}, + {0x5825, 0x26}, + {0x5826, 0x08}, + {0x5827, 0x26}, + {0x5828, 0x64}, + {0x5829, 0x26}, + {0x582a, 0x24}, + {0x582b, 0x22}, + {0x582c, 0x24}, + {0x582d, 0x24}, + {0x582e, 0x06}, + {0x582f, 0x22}, + {0x5830, 0x40}, + {0x5831, 0x42}, + {0x5832, 0x24}, + {0x5833, 0x26}, + {0x5834, 0x24}, + {0x5835, 0x22}, + {0x5836, 0x22}, + {0x5837, 0x26}, + {0x5838, 0x44}, + {0x5839, 0x24}, + {0x583a, 0x26}, + {0x583b, 0x28}, + {0x583c, 0x42}, + {0x583d, 0xce}, + {0x5025, 0x00}, + {0x3a0f, 0x30}, + {0x3a10, 0x28}, + {0x3a1b, 0x30}, + {0x3a1e, 0x26}, + {0x3a11, 0x60}, + {0x3a1f, 0x14}, + {0x3008, 0x02}, + {OV5640_TABLE_END, 0x0000} }; -static const struct ov5640_reg ov5640_defaults[] = { - { OV5640_SCCB_SYSTEM_CTRL1, 0x11}, - { OV5640_SYSTEM_CTRL, 0x82}, - { OV5640_SYSTEM_CTRL, 0x42}, - { OV5640_SCCB_SYSTEM_CTRL1, 0x03}, - { OV5640_PAD_OUTPUT_ENABLE01, 0x00}, - { OV5640_PAD_OUTPUT_ENABLE02, 0x00}, - { OV5640_SC_PLL_CTRL0, 0x18}, - { OV5640_SC_PLL_CTRL1, 0x14}, - { OV5640_SC_PLL_CTRL2, 0x38}, - { OV5640_SC_PLL_CTRL3, 0x13}, - { 0x4800, 0x24}, /* noncontinuous clock */ - { OV5640_SYSTEM_ROOT_DIVIDER, 0x01}, - { 0x3630, 0x36}, - { 0x3631, 0x0e}, - { 0x3632, 0xe2}, - { 0x3633, 0x12}, - { 0x3621, 0xe0}, - { 0x3704, 0xa0}, - { 0x3703, 0x5a}, - { 0x3715, 0x78}, - { 0x3717, 0x01}, - { 0x370b, 0x60}, - { 0x3705, 0x1a}, - { 0x3905, 0x02}, - { 0x3906, 0x10}, - { 0x3901, 0x0a}, - { 0x3731, 0x12}, - { 0x3600, 0x08}, - { 0x3601, 0x33}, - { 0x302d, 0x60}, - { 0x3620, 0x52}, - { 0x371b, 0x20}, - { 0x471c, 0x50}, - { 0x3a13, 0x43}, - { 0x3a18, 0x00}, - { 0x3a19, 0xf8}, - { 0x3635, 0x13}, - { 0x3636, 0x03}, - { 0x3634, 0x40}, - { 0x3622, 0x01}, - { 0x3c01, 0x34}, - { 0x3c04, 0x28}, - { 0x3c05, 0x98}, - { 0x3c06, 0x00}, - { 0x3c07, 0x08}, - { 0x3c08, 0x00}, - { 0x3c09, 0x1c}, - { 0x3c0a, 0x9c}, - { 0x3c0b, 0x40}, - { OV5640_TIMING_TC_REG20, 0x41}, - { OV5640_TIMING_TC_REG21, 0x01}, - { 0x3814, 0x31}, - { 0x3815, 0x31}, - { 0x3800, 0x00}, - { 0x3801, 0x00}, - { 0x3802, 0x00}, - { 0x3803, 0x04}, - { 0x3804, 0x0a}, - { 0x3805, 0x3f}, - { 0x3806, 0x07}, - { 0x3807, 0x9b}, - { 0x3808, 0x02}, - { 0x3809, 0x80}, - { 0x380a, 0x01}, - { 0x380b, 0xe0}, - { 0x380c, 0x07}, - { 0x380d, 0x68}, - { 0x380e, 0x03}, - { 0x380f, 0xd8}, - { 0x3810, 0x00}, - { 0x3811, 0x10}, - { 0x3812, 0x00}, - { 0x3813, 0x06}, - { 0x3618, 0x00}, - { 0x3612, 0x29}, - { 0x3708, 0x64}, - { 0x3709, 0x52}, - { 0x370c, 0x03}, +static struct ov5640_reg mode_640x480[] = { + {0x3103, 0x11}, + {0x3008, 0x82}, + {OV5640_TABLE_WAIT_MS, 5}, + {0x3008, 0x42}, + {0x3103, 0x03}, + {0x3017, 0x00}, + {0x3018, 0x00}, + {0x3034, 0x18}, + {0x3035, 0x14}, + {0x3036, 0x70}, + {0x3037, 0x13}, + {0x4800, 0x24}, /* noncontinuous clock */ + {0x3108, 0x01}, + {0x3630, 0x36}, + {0x3631, 0x0e}, + {0x3632, 0xe2}, + {0x3633, 0x12}, + {0x3621, 0xe0}, + {0x3704, 0xa0}, + {0x3703, 0x5a}, + {0x3715, 0x78}, + {0x3717, 0x01}, + {0x370b, 0x60}, + {0x3705, 0x1a}, + {0x3905, 0x02}, + {0x3906, 0x10}, + {0x3901, 0x0a}, + {0x3731, 0x12}, + {0x3600, 0x08}, + {0x3601, 0x33}, + {0x302d, 0x60}, + {0x3620, 0x52}, + {0x371b, 0x20}, + {0x471c, 0x50}, + {0x3a13, 0x43}, + {0x3a18, 0x00}, + {0x3a19, 0xf8}, + {0x3635, 0x13}, + {0x3636, 0x03}, + {0x3634, 0x40}, + {0x3622, 0x01}, + {0x3c01, 0x34}, + {0x3c04, 0x28}, + {0x3c05, 0x98}, + {0x3c06, 0x00}, + {0x3c07, 0x08}, + {0x3c08, 0x00}, + {0x3c09, 0x1c}, + {0x3c0a, 0x9c}, + {0x3c0b, 0x40}, + {0x3820, 0x41}, + {0x3821, 0x07}, + {0x3814, 0x31}, + {0x3815, 0x31}, + {0x3800, 0x00}, + {0x3801, 0x00}, + {0x3802, 0x00}, + {0x3803, 0x04}, + {0x3804, 0x0a}, + {0x3805, 0x3f}, + {0x3806, 0x07}, + {0x3807, 0x9b}, + {0x3808, 0x02}, + {0x3809, 0x80}, + {0x380a, 0x01}, + {0x380b, 0xe0}, + {0x380c, 0x07}, + {0x380d, 0x68}, + {0x380e, 0x03}, + {0x380f, 0xd8}, + {0x3810, 0x00}, + {0x3811, 0x10}, + {0x3812, 0x00}, + {0x3813, 0x06}, + {0x3618, 0x00}, + {0x3612, 0x29}, + {0x3708, 0x64}, + {0x3709, 0x52}, + {0x370c, 0x03}, /* AEC/AGC Power Down Domain Control */ - { OV5640_AEC_MAX_EXPO_60HZ_HI, 0x03}, - { OV5640_AEC_MAX_EXPO_60HZ_LO, 0xd8}, - { OV5640_AEC_B50_STEP_HI, 0x01}, - { OV5640_AEC_B50_STEP_LO, 0x27}, - { OV5640_AEC_B60_STEP_HI, 0x00}, - { OV5640_AEC_B60_STEP_LO, 0xf6}, - { OV5640_AEC_CTRL0E, 0x03}, - { OV5640_AEC_CTRL0D, 0x04}, - { OV5640_AEC_MAX_EXPO_50HZ_HI, 0x03}, - { OV5640_AEC_MAX_EXPO_50HZ_LO, 0xd8}, - - { 0x4001, 0x02}, - { 0x4004, 0x02}, - { 0x3000, 0x00}, - { 0x3002, 0x1c}, - { 0x3004, 0xff}, - { 0x3006, 0xc3}, - { 0x300e, 0x45}, - { 0x302e, 0x08}, + {0x3a02, 0x03}, + {0x3a03, 0xd8}, + {0x3a08, 0x01}, + {0x3a09, 0x27}, + {0x3a0a, 0x00}, + {0x3a0b, 0xf6}, + {0x3a0e, 0x03}, + {0x3a0d, 0x04}, + {0x3a14, 0x03}, + {0x3a15, 0xd8}, + + {0x4001, 0x02}, + {0x4004, 0x02}, + {0x3000, 0x00}, + {0x3002, 0x1c}, + {0x3004, 0xff}, + {0x3006, 0xc3}, + {0x300e, 0x45}, + {0x302e, 0x08}, /* org:30 bit[3:0] 0x0:YUYV 0x1:YVYU 0x2:UYVY 0x3:VYUY 0xF:UYVY 0x4~0xE:Not-allowed */ - { 0x4300, 0x32}, - { 0x501f, 0x00}, - { 0x4713, 0x03}, - { 0x4407, 0x04}, - { 0x440e, 0x00}, - { 0x460b, 0x35}, - { 0x460c, 0x22}, - { 0x4837, 0x44}, - { 0x3824, 0x02}, - { 0x5000, 0xa7}, - { 0x5001, 0xa3}, + {0x4300, 0x32}, + {0x501f, 0x00}, + {0x4713, 0x03}, + {0x4407, 0x04}, + {0x440e, 0x00}, + {0x460b, 0x35}, + {0x460c, 0x22}, + {0x4837, 0x44}, + {0x3824, 0x02}, + {0x5000, 0xa7}, + {0x5001, 0xa3}, /* AWB Control */ - { OV5640_AWB_CTRL00, 0xff}, { OV5640_AWB_CTRL01, 0xf2}, - { OV5640_AWB_CTRL02, 0x00}, { OV5640_AWB_CTRL03, 0x14}, - { OV5640_AWB_CTRL04, 0x25}, { OV5640_AWB_CTRL05, 0x24}, - { OV5640_AWB_CTRL06, 0x09}, { OV5640_AWB_CTRL07, 0x09}, - { OV5640_AWB_CTRL08, 0x09}, { OV5640_AWB_CTRL09, 0x75}, - { OV5640_AWB_CTRL10, 0x54}, { OV5640_AWB_CTRL11, 0xe0}, - { OV5640_AWB_CTRL12, 0xb2}, { OV5640_AWB_CTRL13, 0x42}, - { OV5640_AWB_CTRL14, 0x3d}, { OV5640_AWB_CTRL15, 0x56}, - { OV5640_AWB_CTRL16, 0x46}, { OV5640_AWB_CTRL17, 0xf8}, - { OV5640_AWB_CTRL18, 0x04}, { OV5640_AWB_CTRL19, 0x70}, - { OV5640_AWB_CTRL20, 0xf0}, { OV5640_AWB_CTRL21, 0xf0}, - { OV5640_AWB_CTRL22, 0x03}, { OV5640_AWB_CTRL23, 0x01}, - { OV5640_AWB_CTRL24, 0x04}, { OV5640_AWB_CTRL25, 0x12}, - { OV5640_AWB_CTRL26, 0x04}, { OV5640_AWB_CTRL27, 0x00}, - { OV5640_AWB_CTRL28, 0x06}, { OV5640_AWB_CTRL29, 0x82}, - { OV5640_AWB_CTRL30, 0x38}, + {0x5180, 0xff}, + {0x5181, 0xf2}, + {0x5182, 0x00}, + {0x5183, 0x14}, + {0x5184, 0x25}, + {0x5185, 0x24}, + {0x5186, 0x09}, + {0x5187, 0x09}, + {0x5188, 0x09}, + {0x5189, 0x75}, + {0x518a, 0x54}, + {0x518b, 0xe0}, + {0x518c, 0xb2}, + {0x518d, 0x42}, + {0x518e, 0x3d}, + {0x518d, 0x56}, + {0x5190, 0x46}, + {0x5191, 0xf8}, + {0x5192, 0x04}, + {0x5193, 0x70}, + {0x5194, 0xf0}, + {0x5195, 0xf0}, + {0x5196, 0x03}, + {0x5197, 0x01}, + {0x5198, 0x04}, + {0x5199, 0x12}, + {0x519a, 0x04}, + {0x519b, 0x00}, + {0x519c, 0x06}, + {0x519d, 0x82}, + {0x519e, 0x38}, /* CMX Control */ - { OV5640_CMX1, 0x1e}, - { OV5640_CMX2, 0x5b}, - { OV5640_CMX3, 0x08}, - { OV5640_CMX4, 0x0a}, - { OV5640_CMX5, 0x7e}, - { OV5640_CMX6, 0x88}, - { OV5640_CMX7, 0x7c}, - { OV5640_CMX8, 0x6c}, - { OV5640_CMX9, 0x10}, - { OV5640_CMXSIGN_HI, 0x01}, - { OV5640_CMXSIGN_LO, 0x98}, + {0x5381, 0x1e}, + {0x5382, 0x5b}, + {0x5383, 0x08}, + {0x5384, 0x0a}, + {0x5385, 0x7e}, + {0x5386, 0x88}, + {0x5387, 0x7c}, + {0x5388, 0x6c}, + {0x5389, 0x10}, + {0x538a, 0x01}, + {0x538b, 0x98}, /* CIP Control */ - { OV5640_CIP_SHARPENMT_THRESHOLD_1, 0x08}, - { OV5640_CIP_SHARPENMT_THRESHOLD_2, 0x30}, - { OV5640_CIP_SHARPENMT_OFFSET_1, 0x10}, - { OV5640_CIP_SHARPENMT_OFFSET_2, 0x00}, - { OV5640_CIP_DNS_THRESHOLD_1, 0x08}, - { OV5640_CIP_DNS_THRESHOLD_2, 0x30}, - { OV5640_CIP_DNS_OFFSET_1, 0x08}, - { OV5640_CIP_DNS_OFFSET_2, 0x16}, - { OV5640_CIP_SHARPENTH_THRESHOLD_1, 0x08}, - { OV5640_CIP_SHARPENTH_THRESHOLD_2, 0x30}, - { OV5640_CIP_SHARPENTH_OFFSET_1, 0x04}, - { OV5640_CIP_SHARPENTH_OFFSET_2, 0x06}, + {0x5300, 0x08}, + {0x5301, 0x30}, + {0x5302, 0x10}, + {0x5303, 0x00}, + {0x5304, 0x08}, + {0x5305, 0x30}, + {0x5306, 0x08}, + {0x5307, 0x16}, + {0x5309, 0x08}, + {0x530a, 0x30}, + {0x530b, 0x04}, + {0x530c, 0x06}, /* Gamma Control */ - { OV5640_GAMMA_CTRL00, 0x01}, - { OV5640_YST00, 0x08}, { OV5640_YST01, 0x14}, - { OV5640_YST02, 0x28}, { OV5640_YST03, 0x51}, - { OV5640_YST04, 0x65}, { OV5640_YST05, 0x71}, - { OV5640_YST06, 0x7d}, { OV5640_YST07, 0x87}, - { OV5640_YST08, 0x91}, { OV5640_YST09, 0x9a}, - { OV5640_YST0A, 0xaa}, { OV5640_YST0B, 0xb8}, - { OV5640_YST0C, 0xcd}, { OV5640_YST0D, 0xdd}, - { OV5640_YST0E, 0xea}, { OV5640_YST0F, 0x1d}, + {0x5480, 0x01}, + {0x5481, 0x08}, + {0x5482, 0x14}, + {0x5483, 0x28}, + {0x5484, 0x51}, + {0x5485, 0x65}, + {0x5486, 0x71}, + {0x5487, 0x7d}, + {0x5488, 0x87}, + {0x5489, 0x91}, + {0x548a, 0x9a}, + {0x548b, 0xaa}, + {0x548c, 0xb8}, + {0x548d, 0xcd}, + {0x548e, 0xdd}, + {0x548f, 0xea}, + {0x5490, 0x1d}, /* SDE Control */ - { OV5640_SDE_CTRL_0, 0x02}, - { OV5640_SDE_CTRL_3, 0x40}, - { OV5640_SDE_CTRL_4, 0x10}, - { OV5640_SDE_CTRL_9, 0x10}, - { OV5640_SDE_CTRL_10, 0x00}, - { OV5640_SDE_CTRL_11, 0xf8}, + {0x5580, 0x02}, + {0x5583, 0x40}, + {0x5584, 0x10}, + {0x5589, 0x10}, + {0x558a, 0x00}, + {0x558b, 0xf8}, /* LENC Control */ - { OV5640_GMTRX00, 0x23}, { OV5640_GMTRX01, 0x14}, - { OV5640_GMTRX02, 0x0f}, { OV5640_GMTRX03, 0x0f}, - { OV5640_GMTRX04, 0x12}, { OV5640_GMTRX05, 0x26}, - { OV5640_GMTRX10, 0x0c}, { OV5640_GMTRX11, 0x08}, - { OV5640_GMTRX12, 0x05}, { OV5640_GMTRX13, 0x05}, - { OV5640_GMTRX14, 0x08}, { OV5640_GMTRX15, 0x0d}, - { OV5640_GMTRX20, 0x08}, { OV5640_GMTRX21, 0x03}, - { OV5640_GMTRX22, 0x00}, { OV5640_GMTRX23, 0x00}, - { OV5640_GMTRX24, 0x03}, { OV5640_GMTRX25, 0x09}, - { OV5640_GMTRX30, 0x07}, { OV5640_GMTRX31, 0x03}, - { OV5640_GMTRX32, 0x00}, { OV5640_GMTRX33, 0x01}, - { OV5640_GMTRX34, 0x03}, { OV5640_GMTRX35, 0x08}, - { OV5640_GMTRX40, 0x0d}, { OV5640_GMTRX41, 0x08}, - { OV5640_GMTRX42, 0x05}, { OV5640_GMTRX43, 0x06}, - { OV5640_GMTRX44, 0x08}, { OV5640_GMTRX45, 0x0e}, - { OV5640_GMTRX50, 0x29}, { OV5640_GMTRX51, 0x17}, - { OV5640_GMTRX52, 0x11}, { OV5640_GMTRX53, 0x11}, - { OV5640_GMTRX54, 0x15}, { OV5640_GMTRX55, 0x28}, - { OV5640_BRMATRX00, 0x46}, { OV5640_BRMATRX01, 0x26}, - { OV5640_BRMATRX02, 0x08}, { OV5640_BRMATRX03, 0x26}, - { OV5640_BRMATRX04, 0x64}, { OV5640_BRMATRX05, 0x26}, - { OV5640_BRMATRX06, 0x24}, { OV5640_BRMATRX07, 0x22}, - { OV5640_BRMATRX08, 0x24}, { OV5640_BRMATRX09, 0x24}, - { OV5640_BRMATRX20, 0x06}, { OV5640_BRMATRX21, 0x22}, - { OV5640_BRMATRX22, 0x40}, { OV5640_BRMATRX23, 0x42}, - { OV5640_BRMATRX24, 0x24}, { OV5640_BRMATRX30, 0x26}, - { OV5640_BRMATRX31, 0x24}, { OV5640_BRMATRX32, 0x22}, - { OV5640_BRMATRX33, 0x22}, { OV5640_BRMATRX34, 0x26}, - { OV5640_BRMATRX40, 0x44}, { OV5640_BRMATRX41, 0x24}, - { OV5640_BRMATRX42, 0x26}, { OV5640_BRMATRX43, 0x28}, - { OV5640_BRMATRX44, 0x42}, { OV5640_LENC_BR_OFFSET, 0xce}, - - { OV5640_ISP_CTRL37, 0x00}, - { OV5640_AEC_CTRL0F, 0x30}, - { OV5640_AEC_CTRL10, 0x28}, - { OV5640_AEC_CTRL1B, 0x30}, - { OV5640_AEC_CTRL1E, 0x26}, - { OV5640_AEC_CTRL11, 0x60}, - { OV5640_AEC_CTRL1F, 0x14}, - { OV5640_SYSTEM_CTRL, 0x02}, + {0x5800, 0x23}, + {0x5801, 0x14}, + {0x5802, 0x0f}, + {0x5803, 0x0f}, + {0x5804, 0x12}, + {0x5805, 0x26}, + {0x5806, 0x0c}, + {0x5807, 0x08}, + {0x5808, 0x05}, + {0x5809, 0x05}, + {0x580a, 0x08}, + {0x580b, 0x0d}, + {0x580c, 0x08}, + {0x580d, 0x03}, + {0x580e, 0x00}, + {0x580f, 0x00}, + {0x5810, 0x03}, + {0x5811, 0x09}, + {0x5812, 0x07}, + {0x5813, 0x03}, + {0x5814, 0x00}, + {0x5815, 0x01}, + {0x5816, 0x03}, + {0x5817, 0x08}, + {0x5818, 0x0d}, + {0x5819, 0x08}, + {0x581a, 0x05}, + {0x581b, 0x06}, + {0x581c, 0x08}, + {0x581d, 0x0e}, + {0x581e, 0x29}, + {0x581f, 0x17}, + {0x5820, 0x11}, + {0x5821, 0x11}, + {0x5822, 0x15}, + {0x5823, 0x28}, + {0x5824, 0x46}, + {0x5825, 0x26}, + {0x5826, 0x08}, + {0x5827, 0x26}, + {0x5828, 0x64}, + {0x5829, 0x26}, + {0x582a, 0x24}, + {0x582b, 0x22}, + {0x582c, 0x24}, + {0x582d, 0x24}, + {0x582e, 0x06}, + {0x582f, 0x22}, + {0x5830, 0x40}, + {0x5831, 0x42}, + {0x5832, 0x24}, + {0x5833, 0x26}, + {0x5834, 0x24}, + {0x5835, 0x22}, + {0x5836, 0x22}, + {0x5837, 0x26}, + {0x5838, 0x44}, + {0x5839, 0x24}, + {0x583a, 0x26}, + {0x583b, 0x28}, + {0x583c, 0x42}, + {0x583d, 0xce}, + + {0x5025, 0x00}, + {0x3a0f, 0x30}, + {0x3a10, 0x28}, + {0x3a1b, 0x30}, + {0x3a1e, 0x26}, + {0x3a11, 0x60}, + {0x3a1f, 0x14}, + {0x3008, 0x02}, + {OV5640_TABLE_END, 0x0000}, +}; + +enum { + OV5640_MODE_640x480, + OV5640_MODE_1296x972, + OV5640_MODE_1920x1088, + OV5640_MODE_2592x1944, + OV5640_SIZE_LAST, +}; + +static struct ov5640_reg *mode_table[] = { + [OV5640_MODE_640x480] = mode_640x480, + [OV5640_MODE_1296x972] = mode_1296x972, + [OV5640_MODE_1920x1088] = mode_1920x1088, + [OV5640_MODE_2592x1944] = mode_2592x1944, +}; + +static int test_pattern; +module_param(test_pattern, int, 0644); + +static struct ov5640_reg tp_cbars[] = { + {0x503D, 0x80}, + {0x503E, 0x00}, + {0x5046, 0x01}, + {OV5640_TABLE_END, 0x0000} +}; + +#define to_ov5640(sd) container_of(sd, struct ov5640_priv, subdev) + +#define SIZEOF_I2C_TRANSBUF 32 + +struct ov5640_priv { + struct v4l2_subdev subdev; + struct v4l2_mbus_framefmt mf; + + int ident; + u16 chip_id; + u8 revision; + + int mode; + + struct i2c_client *client; + u8 i2c_trans_buf[SIZEOF_I2C_TRANSBUF]; }; static enum v4l2_mbus_pixelcode ov5640_codes[] = { - V4L2_MBUS_FMT_YUYV8_2X8, + V4L2_MBUS_FMT_UYVY8_2X8, }; -static const struct v4l2_queryctrl ov5640_controls[] = { - { - .id = V4L2_CID_VFLIP, - .type = V4L2_CTRL_TYPE_BOOLEAN, - .name = "Flip Vertically", - .minimum = 0, - .maximum = 1, - .step = 1, - .default_value = 0, - }, - { - .id = V4L2_CID_HFLIP, - .type = V4L2_CTRL_TYPE_BOOLEAN, - .name = "Flip Horizontally", - .minimum = 0, - .maximum = 1, - .step = 1, - .default_value = 0, - }, +static const struct v4l2_frmsize_discrete ov5640_frmsizes[OV5640_SIZE_LAST] = { + {640, 480}, + {1296, 972}, + {1920, 1088}, + {2592, 1944}, }; -/* read a register */ -static int ov5640_reg_read(struct i2c_client *client, u16 reg, u8 *val) +static int ov5640_find_mode(u32 width, u32 height) { - int ret; - unsigned char data[2] = { reg >> 8, reg & 0xff }; + int i; - ret = i2c_master_send(client, data, 2); - if (ret < 2) { - dev_err(&client->dev, "%s: i2c read error, reg: %x\n", - __func__, reg); - return ret < 0 ? ret : -EIO; + for (i = 0; i < OV5640_SIZE_LAST; i++) { + if ((ov5640_frmsizes[i].width >= width) && + (ov5640_frmsizes[i].height >= height)) + break; } - ret = i2c_master_recv(client, val, 1); - if (ret < 1) { - dev_err(&client->dev, "%s: i2c read error, reg: %x\n", - __func__, reg); - return ret < 0 ? ret : -EIO; - } + /* If not found, select biggest */ + if (i >= OV5640_SIZE_LAST) + i = OV5640_SIZE_LAST - 1; - return 0; + return i; } -/* write a register */ -static int ov5640_reg_write(struct i2c_client *client, u16 reg, u8 val) +static int ov5640_read_reg(struct i2c_client *client, u16 addr, u8 *val) { - int ret; - unsigned char data[3] = { reg >> 8, reg & 0xff, val }; + int err; + struct i2c_msg msg[2]; + unsigned char data[3]; - ret = i2c_master_send(client, data, 3); - if (ret < 3) { - dev_err(&client->dev, "%s: i2c write error, reg: %x\n", - __func__, reg); - return ret < 0 ? ret : -EIO; - } + if (!client->adapter) + return -ENODEV; - return 0; -} + msg[0].addr = client->addr; + msg[0].flags = 0; + msg[0].len = 2; + msg[0].buf = data; + /* high byte goes out first */ + data[0] = (u8) (addr >> 8); + data[1] = (u8) (addr & 0xff); -/* Read a register, alter its bits, write it back */ -static int ov5640_reg_rmw(struct i2c_client *client, u16 reg, u8 set, u8 unset) -{ - u8 val; - int ret; + msg[1].addr = client->addr; + msg[1].flags = I2C_M_RD; + msg[1].len = 1; + msg[1].buf = data + 2; - ret = ov5640_reg_read(client, reg, &val); - if (ret < 0) { - dev_err(&client->dev, - "[Read]-Modify-Write of register 0x%04x failed!\n", - reg); - return ret; - } + err = i2c_transfer(client->adapter, msg, 2); - val |= set; - val &= ~unset; + if (err != 2) + return -EINVAL; - ret = ov5640_reg_write(client, reg, val); - if (ret < 0) { - dev_err(&client->dev, - "Read-Modify-[Write] of register 0x%04x failed!\n", - reg); - return ret; - } + *val = data[2]; return 0; } -static int ov5640_reg_write_array(struct i2c_client *client, - const struct ov5640_reg *regarray, - int regarraylen) +static int ov5640_write_reg(struct i2c_client *client, u16 addr, u8 value) { - int i; - int ret; + int count; + struct i2c_msg msg[1]; + unsigned char data[4]; - for (i = 0; i < regarraylen; i++) { - ret = ov5640_reg_write(client, - regarray[i].reg, regarray[i].val); - if (ret < 0) - return ret; - } + if (!client->adapter) + return -ENODEV; - return 0; + data[0] = addr; + data[1] = (u8) (addr & 0xff); + data[2] = value; + + msg[0].addr = client->addr; + msg[0].flags = 0; + msg[0].len = 3; + msg[0].buf = data; + + count = i2c_transfer(client->adapter, msg, ARRAY_SIZE(msg)); + if (count == ARRAY_SIZE(msg)) + return 0; + dev_err(&client->dev, + "ov5840: i2c transfer failed, addr: %x, value: %02x\n", + addr, (u32)value); + return -EIO; } -/* Start/Stop streaming from the device */ -static int ov5640_s_stream(struct v4l2_subdev *sd, int enable) +static int ov5640_write_bulk_reg(struct i2c_client *client, u8 *data, int len) { - struct i2c_client *client = v4l2_get_subdevdata(sd); - struct ov5640_priv *priv = to_ov5640(sd); - int ret; + int err; + struct i2c_msg msg; - /* Program orientation register. */ - if (priv->flag_vflip) - ret = ov5640_reg_rmw(client, OV5640_TIMING_TC_REG20, 0x2, 0); - else - ret = ov5640_reg_rmw(client, OV5640_TIMING_TC_REG20, 0, 0x2); - if (ret < 0) - return ret; + if (!client->adapter) + return -ENODEV; - if (priv->flag_hflip) - ret = ov5640_reg_rmw(client, OV5640_TIMING_TC_REG21, 0x2, 0); - else - ret = ov5640_reg_rmw(client, OV5640_TIMING_TC_REG21, 0, 0x2); - if (ret < 0) - return ret; + msg.addr = client->addr; + msg.flags = 0; + msg.len = len; + msg.buf = data; - if (!enable) { - /* Software Reset */ - ret = ov5640_reg_write(client, OV5640_SYSTEM_CTRL, 0x82); - if (!ret) - /* Setting Streaming to Standby */ - ret = ov5640_reg_write(client, OV5640_SYSTEM_CTRL, - 0x42); - } + err = i2c_transfer(client->adapter, &msg, 1); + if (err == 1) + return 0; - priv->current_enable = enable; + dev_err(&client->dev, "ov5640: i2c transfer failed at %x\n", + (int)data[0] << 8 | data[1]); - return ret; + return err; } -/* Alter bus settings on camera side */ -static int ov5640_set_bus_param(struct soc_camera_device *icd, - unsigned long flags) +static int ov5640_write_table(struct ov5640_priv *priv, + struct ov5640_reg table[]) { + int err; + struct ov5640_reg *next, *n_next; + u8 *b_ptr = priv->i2c_trans_buf; + unsigned int buf_filled = 0; + u16 val; + + for (next = table; next->addr != OV5640_TABLE_END; next++) { + if (next->addr == OV5640_TABLE_WAIT_MS) { + msleep(next->val); + continue; + } + + val = next->val; + + if (!buf_filled) { + b_ptr = priv->i2c_trans_buf; + *b_ptr++ = next->addr >> 8; + *b_ptr++ = next->addr & 0xff; + buf_filled = 2; + } + *b_ptr++ = val; + buf_filled++; + + n_next = next + 1; + if (n_next->addr != OV5640_TABLE_END && + n_next->addr != OV5640_TABLE_WAIT_MS && + buf_filled < SIZEOF_I2C_TRANSBUF && + n_next->addr == next->addr + 1) { + continue; + } + + err = ov5640_write_bulk_reg(priv->client, + priv->i2c_trans_buf, buf_filled); + if (err) + return err; + + buf_filled = 0; + } return 0; } -/* Request bus settings on camera side */ -static unsigned long ov5640_query_bus_param(struct soc_camera_device *icd) -{ - struct soc_camera_link *icl = to_soc_camera_link(icd); - - unsigned long flags = SOCAM_PCLK_SAMPLE_RISING | SOCAM_MASTER | - SOCAM_VSYNC_ACTIVE_HIGH | SOCAM_HSYNC_ACTIVE_HIGH | - SOCAM_DATA_ACTIVE_HIGH | SOCAM_DATAWIDTH_8; - - return soc_camera_apply_sensor_flags(icl, flags); -} - -/* select nearest higher resolution for capture */ -static void ov5640_res_roundup(u32 *width, u32 *height) +static void ov5640_set_default_fmt(struct ov5640_priv *priv) { - /* Width must be a multiple of 4 pixels. */ - *width = ALIGN(*width, 4); + struct v4l2_mbus_framefmt *mf = &priv->mf; - /* Max resolution is 1280x720 (720p). */ - if (*width > OV5640_MAX_WIDTH) - *width = OV5640_MAX_WIDTH; - - if (*height > OV5640_MAX_HEIGHT) - *height = OV5640_MAX_HEIGHT; + mf->width = ov5640_frmsizes[OV5640_MODE_2592x1944].width; + mf->height = ov5640_frmsizes[OV5640_MODE_2592x1944].height; + mf->code = V4L2_MBUS_FMT_UYVY8_2X8; + mf->field = V4L2_FIELD_NONE; + mf->colorspace = V4L2_COLORSPACE_SRGB; } -/* Setup registers according to resolution and color encoding */ -static int ov5640_set_res(struct i2c_client *client, u32 width, u32 height) +/* Start/Stop streaming from the device */ +static int ov5640_s_stream(struct v4l2_subdev *sd, int enable) { - /* Note, this stuff is bogus. It's just copied from ov9740.c. */ -#if 0 - u32 x_start; - u32 y_start; - u32 x_end; - u32 y_end; - bool scaling = 0; - u32 scale_input_x; - u32 scale_input_y; - int ret; - - if ((width != OV5640_MAX_WIDTH) || (height != OV5640_MAX_HEIGHT)) - scaling = 1; + struct ov5640_priv *priv = to_ov5640(sd); + int ret = 0; - /* - * Try to use as much of the sensor area as possible when supporting - * smaller resolutions. Depending on the aspect ratio of the - * chosen resolution, we can either use the full width of the sensor, - * or the full height of the sensor (or both if the aspect ratio is - * the same as 1280x720. - */ - if ((OV5640_MAX_WIDTH * height) > (OV5640_MAX_HEIGHT * width)) { - scale_input_x = (OV5640_MAX_HEIGHT * width) / height; - scale_input_y = OV5640_MAX_HEIGHT; - } else { - scale_input_x = OV5640_MAX_WIDTH; - scale_input_y = (OV5640_MAX_WIDTH * height) / width; + if (!enable) { + ov5640_set_default_fmt(priv); + return 0; } - /* These describe the area of the sensor to use. */ - x_start = (OV5640_MAX_WIDTH - scale_input_x) / 2; - y_start = (OV5640_MAX_HEIGHT - scale_input_y) / 2; - x_end = x_start + scale_input_x - 1; - y_end = y_start + scale_input_y - 1; - - ret = ov5640_reg_write(client, OV5640_X_ADDR_START_HI, x_start >> 8); - if (ret) - goto done; - ret = ov5640_reg_write(client, OV5640_X_ADDR_START_LO, x_start & 0xff); - if (ret) - goto done; - ret = ov5640_reg_write(client, OV5640_Y_ADDR_START_HI, y_start >> 8); + ret = ov5640_write_table(priv, mode_table[priv->mode]); if (ret) - goto done; - ret = ov5640_reg_write(client, OV5640_Y_ADDR_START_LO, y_start & 0xff); - if (ret) - goto done; + return ret; - ret = ov5640_reg_write(client, OV5640_X_ADDR_END_HI, x_end >> 8); - if (ret) - goto done; - ret = ov5640_reg_write(client, OV5640_X_ADDR_END_LO, x_end & 0xff); - if (ret) - goto done; - ret = ov5640_reg_write(client, OV5640_Y_ADDR_END_HI, y_end >> 8); - if (ret) - goto done; - ret = ov5640_reg_write(client, OV5640_Y_ADDR_END_LO, y_end & 0xff); - if (ret) - goto done; + if (test_pattern == 1) { + ret = ov5640_write_table(priv, tp_cbars); + if (ret) + return ret; + } - ret = ov5640_reg_write(client, OV5640_X_OUTPUT_SIZE_HI, width >> 8); - if (ret) - goto done; - ret = ov5640_reg_write(client, OV5640_X_OUTPUT_SIZE_LO, width & 0xff); - if (ret) - goto done; - ret = ov5640_reg_write(client, OV5640_Y_OUTPUT_SIZE_HI, height >> 8); - if (ret) - goto done; - ret = ov5640_reg_write(client, OV5640_Y_OUTPUT_SIZE_LO, height & 0xff); - if (ret) - goto done; + return ret; +} - ret = ov5640_reg_write(client, OV5640_ISP_CTRL1E, scale_input_x >> 8); - if (ret) - goto done; - ret = ov5640_reg_write(client, OV5640_ISP_CTRL1F, scale_input_x & 0xff); - if (ret) - goto done; - ret = ov5640_reg_write(client, OV5640_ISP_CTRL20, scale_input_y >> 8); - if (ret) - goto done; - ret = ov5640_reg_write(client, OV5640_ISP_CTRL21, scale_input_y & 0xff); - if (ret) - goto done; +static int ov5640_try_fmt(struct v4l2_subdev *sd, + struct v4l2_mbus_framefmt *mf) +{ + int mode; - ret = ov5640_reg_write(client, OV5640_VFIFO_READ_START_HI, - (scale_input_x - width) >> 8); - if (ret) - goto done; - ret = ov5640_reg_write(client, OV5640_VFIFO_READ_START_LO, - (scale_input_x - width) & 0xff); - if (ret) - goto done; + mode = ov5640_find_mode(mf->width, mf->height); + mf->width = ov5640_frmsizes[mode].width; + mf->height = ov5640_frmsizes[mode].height; - ret = ov5640_reg_write(client, OV5640_ISP_CTRL00, 0xff); - if (ret) - goto done; - ret = ov5640_reg_write(client, OV5640_ISP_CTRL01, 0xef | - (scaling << 4)); - if (ret) - goto done; - ret = ov5640_reg_write(client, OV5640_ISP_CTRL03, 0xff); + mf->field = V4L2_FIELD_NONE; + mf->code = V4L2_MBUS_FMT_UYVY8_2X8; + mf->colorspace = V4L2_COLORSPACE_SRGB; -done: - return ret; -#endif return 0; } @@ -815,47 +1347,16 @@ done: static int ov5640_s_fmt(struct v4l2_subdev *sd, struct v4l2_mbus_framefmt *mf) { - struct i2c_client *client = v4l2_get_subdevdata(sd); struct ov5640_priv *priv = to_ov5640(sd); - enum v4l2_colorspace cspace; - enum v4l2_mbus_pixelcode code = mf->code; int ret; - ov5640_res_roundup(&mf->width, &mf->height); - - switch (code) { - case V4L2_MBUS_FMT_YUYV8_2X8: - cspace = V4L2_COLORSPACE_SRGB; - break; - default: - return -EINVAL; - } - - ret = ov5640_reg_write_array(client, ov5640_defaults, - ARRAY_SIZE(ov5640_defaults)); - if (ret < 0) - return ret; - - ret = ov5640_set_res(client, mf->width, mf->height); + ret = ov5640_try_fmt(sd, mf); if (ret < 0) return ret; - mf->code = code; - mf->colorspace = cspace; + priv->mode = ov5640_find_mode(mf->width, mf->height); - memcpy(&priv->current_mf, mf, sizeof(struct v4l2_mbus_framefmt)); - - return ret; -} - -static int ov5640_try_fmt(struct v4l2_subdev *sd, - struct v4l2_mbus_framefmt *mf) -{ - ov5640_res_roundup(&mf->width, &mf->height); - - mf->field = V4L2_FIELD_NONE; - mf->code = V4L2_MBUS_FMT_YUYV8_2X8; - mf->colorspace = V4L2_COLORSPACE_SRGB; + memcpy(&priv->mf, mf, sizeof(struct v4l2_mbus_framefmt)); return 0; } @@ -875,8 +1376,8 @@ static int ov5640_cropcap(struct v4l2_subdev *sd, struct v4l2_cropcap *a) { a->bounds.left = 0; a->bounds.top = 0; - a->bounds.width = OV5640_MAX_WIDTH; - a->bounds.height = OV5640_MAX_HEIGHT; + a->bounds.width = ov5640_frmsizes[OV5640_MODE_2592x1944].width; + a->bounds.height = ov5640_frmsizes[OV5640_MODE_2592x1944].height; a->defrect = a->bounds; a->type = V4L2_BUF_TYPE_VIDEO_CAPTURE; a->pixelaspect.numerator = 1; @@ -889,51 +1390,13 @@ static int ov5640_g_crop(struct v4l2_subdev *sd, struct v4l2_crop *a) { a->c.left = 0; a->c.top = 0; - a->c.width = OV5640_MAX_WIDTH; - a->c.height = OV5640_MAX_HEIGHT; + a->c.width = ov5640_frmsizes[OV5640_MODE_2592x1944].width; + a->c.height = ov5640_frmsizes[OV5640_MODE_2592x1944].height; a->type = V4L2_BUF_TYPE_VIDEO_CAPTURE; return 0; } -/* Get status of additional camera capabilities */ -static int ov5640_g_ctrl(struct v4l2_subdev *sd, struct v4l2_control *ctrl) -{ - struct ov5640_priv *priv = to_ov5640(sd); - - switch (ctrl->id) { - case V4L2_CID_VFLIP: - ctrl->value = priv->flag_vflip; - break; - case V4L2_CID_HFLIP: - ctrl->value = priv->flag_hflip; - break; - default: - return -EINVAL; - } - - return 0; -} - -/* Set status of additional camera capabilities */ -static int ov5640_s_ctrl(struct v4l2_subdev *sd, struct v4l2_control *ctrl) -{ - struct ov5640_priv *priv = to_ov5640(sd); - - switch (ctrl->id) { - case V4L2_CID_VFLIP: - priv->flag_vflip = ctrl->value; - break; - case V4L2_CID_HFLIP: - priv->flag_hflip = ctrl->value; - break; - default: - return -EINVAL; - } - - return 0; -} - /* Get chip identification */ static int ov5640_g_chip_ident(struct v4l2_subdev *sd, struct v4l2_dbg_chip_ident *id) @@ -946,24 +1409,6 @@ static int ov5640_g_chip_ident(struct v4l2_subdev *sd, return 0; } -static int ov5640_s_power(struct v4l2_subdev *sd, int on) -{ - struct ov5640_priv *priv = to_ov5640(sd); - - if (!priv->current_enable) - return 0; - - if (on) { - ov5640_s_fmt(sd, &priv->current_mf); - ov5640_s_stream(sd, priv->current_enable); - } else { - ov5640_s_stream(sd, 0); - priv->current_enable = true; - } - - return 0; -} - #ifdef CONFIG_VIDEO_ADV_DEBUG static int ov5640_get_register(struct v4l2_subdev *sd, struct v4l2_dbg_register *reg) @@ -977,7 +1422,7 @@ static int ov5640_get_register(struct v4l2_subdev *sd, reg->size = 2; - ret = ov5640_reg_read(client, reg->reg, &val); + ret = ov5640_read_reg(client, reg->reg, &val); if (ret) return ret; @@ -994,53 +1439,32 @@ static int ov5640_set_register(struct v4l2_subdev *sd, if (reg->reg & ~0xffff || reg->val & ~0xff) return -EINVAL; - return ov5640_reg_write(client, reg->reg, reg->val); + return ov5640_write_reg(client, reg->reg, reg->val); } #endif -static int ov5640_video_probe(struct soc_camera_device *icd, - struct i2c_client *client) +/* Alter bus settings on camera side */ +static int ov5640_set_bus_param(struct soc_camera_device *icd, + unsigned long flags) { - struct v4l2_subdev *sd = i2c_get_clientdata(client); - struct ov5640_priv *priv = to_ov5640(sd); - u8 chip_id_hi, chip_id_lo; - int ret; - - /* We must have a parent by now. And it cannot be a wrong one. */ - BUG_ON(!icd->parent || - to_soc_camera_host(icd->parent)->nr != icd->iface); - - /* - * check and show product ID and manufacturer ID - */ - ret = ov5640_reg_read(client, OV5640_CHIP_ID_HI, &chip_id_hi); - if (ret < 0) - goto err; - - ret = ov5640_reg_read(client, OV5640_CHIP_ID_LO, &chip_id_lo); - if (ret < 0) - goto err; - - priv->chip_id = (chip_id_hi << 8) | chip_id_lo; - - if (priv->chip_id != 0x5640) { - ret = -ENODEV; - goto err; - } + return 0; +} - priv->ident = V4L2_IDENT_OV5640; +/* Request bus settings on camera side */ +static unsigned long ov5640_query_bus_param(struct soc_camera_device *icd) +{ + struct soc_camera_link *icl = to_soc_camera_link(icd); - dev_info(&client->dev, "Chip ID 0x%04x\n", priv->chip_id); + unsigned long flags = SOCAM_PCLK_SAMPLE_RISING | SOCAM_MASTER | + SOCAM_VSYNC_ACTIVE_HIGH | SOCAM_HSYNC_ACTIVE_HIGH | + SOCAM_DATA_ACTIVE_HIGH | SOCAM_DATAWIDTH_8; -err: - return ret; + return soc_camera_apply_sensor_flags(icl, flags); } static struct soc_camera_ops ov5640_ops = { .set_bus_param = ov5640_set_bus_param, .query_bus_param = ov5640_query_bus_param, - .controls = ov5640_controls, - .num_controls = ARRAY_SIZE(ov5640_controls), }; static struct v4l2_subdev_video_ops ov5640_video_ops = { @@ -1053,10 +1477,7 @@ static struct v4l2_subdev_video_ops ov5640_video_ops = { }; static struct v4l2_subdev_core_ops ov5640_core_ops = { - .g_ctrl = ov5640_g_ctrl, - .s_ctrl = ov5640_s_ctrl, .g_chip_ident = ov5640_g_chip_ident, - .s_power = ov5640_s_power, #ifdef CONFIG_VIDEO_ADV_DEBUG .g_register = ov5640_get_register, .s_register = ov5640_set_register, @@ -1077,6 +1498,7 @@ static int ov5640_probe(struct i2c_client *client, struct ov5640_priv *priv; struct soc_camera_device *icd = client->dev.platform_data; struct soc_camera_link *icl; + u8 chip_id_hi, chip_id_lo; int ret; if (!icd) { @@ -1090,31 +1512,52 @@ static int ov5640_probe(struct i2c_client *client, return -EINVAL; } - priv = kzalloc(sizeof(struct ov5640_priv), GFP_KERNEL); + priv = devm_kzalloc(&client->dev, sizeof(struct ov5640_priv), + GFP_KERNEL); if (!priv) { dev_err(&client->dev, "Failed to allocate private data!\n"); return -ENOMEM; } v4l2_i2c_subdev_init(&priv->subdev, client, &ov5640_subdev_ops); - icd->ops = &ov5640_ops; - ret = ov5640_video_probe(icd, client); + priv->client = client; + + priv->ident = V4L2_IDENT_OV5640; + + /* + * check and show product ID and manufacturer ID + */ + ret = ov5640_read_reg(client, 0x300A, &chip_id_hi); if (ret < 0) { - icd->ops = NULL; - kfree(priv); + dev_err(&client->dev, "Failure to read Chip ID (high byte)\n"); + return ret; } + ret = ov5640_read_reg(client, 0x300B, &chip_id_lo); + if (ret < 0) { + dev_err(&client->dev, "Failure to read Chip ID (low byte)\n"); + return ret; + } + + priv->chip_id = (chip_id_hi << 8) | chip_id_lo; + + if (priv->chip_id != 0x5640) { + dev_err(&client->dev, "Chip ID: %x not supported!\n", + priv->chip_id); + ret = -ENODEV; + return ret; + } + + ov5640_set_default_fmt(priv); + + dev_info(&client->dev, "Chip ID 0x%04x\n", priv->chip_id); return ret; } static int ov5640_remove(struct i2c_client *client) { - struct ov5640_priv *priv = i2c_get_clientdata(client); - - kfree(priv); - return 0; } diff --git a/drivers/media/video/ov5650.c b/drivers/media/video/ov5650.c new file mode 100644 index 000000000000..9b47d66536ef --- /dev/null +++ b/drivers/media/video/ov5650.c @@ -0,0 +1,1406 @@ +/* + * OmniVision OV5650 sensor driver + * + * Copyright (c) 2013, NVIDIA CORPORATION. All rights reserved. + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License as + * published by the Free Software Foundation version 2. + * + * This program is distributed "as is" WITHOUT ANY WARRANTY of any + * kind, whether express or implied; without even the implied warranty + * of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + */ + +#include <linux/videodev2.h> +#include <linux/slab.h> +#include <linux/i2c.h> +#include <linux/log2.h> +#include <linux/delay.h> +#include <linux/module.h> + +#include <media/v4l2-device.h> +#include <media/v4l2-subdev.h> +#include <media/v4l2-ctrls.h> +#include <media/v4l2-chip-ident.h> +#include <media/soc_camera.h> + +#include <media/ov5650.h> + +#define SIZEOF_I2C_TRANSBUF 32 + +struct ov5650_priv { + struct v4l2_subdev subdev; + struct v4l2_mbus_framefmt mf; + const struct ov5650_platform_data *pdata; + + int ident; + u16 chip_id; + u8 revision; + + int mode; + + struct i2c_client *client; + u8 i2c_trans_buf[SIZEOF_I2C_TRANSBUF]; +}; + +static struct ov5650_priv *to_ov5650(const struct v4l2_subdev *sd) +{ + return container_of(sd, struct ov5650_priv, subdev); +} + +/** + * struct ov5650_reg - ov5650 register format + * @addr: 16-bit offset to register + * @val: 8/16/32-bit register value + * + * Define a structure for OV5650 register initialization values + */ +struct ov5650_reg { + u16 addr; + u16 val; +}; + +#define OV5650_TABLE_WAIT_MS 0 +#define OV5650_TABLE_END 1 +#define OV5650_MAX_RETRIES 3 + +static struct ov5650_reg tp_none_seq[] = { + {0x5046, 0x00}, + {OV5650_TABLE_END, 0x0000} +}; + +static struct ov5650_reg tp_cbars_seq[] = { + {0x503D, 0xC0}, + {0x503E, 0x00}, + {0x5046, 0x01}, + {OV5650_TABLE_END, 0x0000} +}; + +static struct ov5650_reg tp_checker_seq[] = { + {0x503D, 0xC0}, + {0x503E, 0x0A}, + {0x5046, 0x01}, + {OV5650_TABLE_END, 0x0000} +}; + +static struct ov5650_reg *test_pattern_modes[] = { + tp_none_seq, + tp_cbars_seq, + tp_checker_seq, +}; + +enum { + TP_NONE = 1, + TP_COLORBARS = 2, + TP_CHECKER = 3, +}; + +static int test_mode = 2; +module_param(test_mode, int, 0644); + +static struct ov5650_reg reset_seq[] = { + {0x3008, 0x82}, + {OV5650_TABLE_WAIT_MS, 5}, + {0x3008, 0x42}, + {OV5650_TABLE_WAIT_MS, 5}, + {OV5650_TABLE_END, 0x0000}, +}; + +static struct ov5650_reg mode_start[] = { + {0x3103, 0x93}, + {0x3017, 0xff}, + {0x3018, 0xfc}, + + {0x3600, 0x50}, + {0x3601, 0x0d}, + {0x3604, 0x50}, + {0x3605, 0x04}, + {0x3606, 0x3f}, + {0x3612, 0x1a}, + {0x3630, 0x22}, + {0x3631, 0x22}, + {0x3702, 0x3a}, + {0x3704, 0x18}, + {0x3705, 0xda}, + {0x3706, 0x41}, + {0x370a, 0x80}, + {0x370b, 0x40}, + {0x370e, 0x00}, + {0x3710, 0x28}, + {0x3712, 0x13}, + {0x3830, 0x50}, + {0x3a18, 0x00}, + {0x3a19, 0xf8}, + {0x3a00, 0x38}, + + + {0x3603, 0xa7}, + {0x3615, 0x50}, + {0x3620, 0x56}, + {0x3810, 0x00}, + {0x3836, 0x00}, + {0x3a1a, 0x06}, + {0x4000, 0x01}, + {0x401c, 0x48}, + {0x401d, 0x08}, + {0x5000, 0x06}, + {0x5001, 0x00}, + {0x5002, 0x00}, + {0x503d, 0x00}, + {0x5046, 0x00}, + + {0x300f, 0x8f}, + + {0x3010, 0x10}, + {0x3011, 0x14}, + {0x3012, 0x02}, + {0x3815, 0x82}, + {0x3503, 0x00}, + {0x3613, 0x44}, + {OV5650_TABLE_END, 0x0}, +}; + +static struct ov5650_reg mode_2592x1944[] = { + {0x3621, 0x2f}, + + {0x3632, 0x55}, + {0x3703, 0xe6}, + {0x370c, 0xa0}, + {0x370d, 0x04}, + {0x3713, 0x2f}, + {0x3800, 0x02}, + {0x3801, 0x58}, + {0x3802, 0x00}, + {0x3803, 0x0c}, + {0x3804, 0x0a}, + {0x3805, 0x20}, + {0x3806, 0x07}, + {0x3807, 0xa0}, + {0x3808, 0x0a}, + + {0x3809, 0x20}, + + {0x380a, 0x07}, + + {0x380b, 0xa0}, + + {0x380c, 0x0c}, + + {0x380d, 0xb4}, + + {0x380e, 0x07}, + + {0x380f, 0xb0}, + + {0x3818, 0xc0}, + {0x381a, 0x3c}, + {0x3a0d, 0x06}, + {0x3c01, 0x00}, + {0x3007, 0x3f}, + {0x5059, 0x80}, + {0x3003, 0x03}, + {0x3500, 0x00}, + {0x3501, 0x7a}, + + {0x3502, 0xd0}, + + {0x350a, 0x00}, + {0x350b, 0x00}, + {0x401d, 0x08}, + {0x4801, 0x0f}, + {0x300e, 0x0c}, + {0x4803, 0x50}, + {0x4800, 0x34}, + {OV5650_TABLE_END, 0x0000} +}; + +static struct ov5650_reg mode_1296x972[] = { + {0x3621, 0xaf}, + + {0x3632, 0x5a}, + {0x3703, 0xb0}, + {0x370c, 0xc5}, + {0x370d, 0x42}, + {0x3713, 0x2f}, + {0x3800, 0x03}, + {0x3801, 0x3c}, + {0x3802, 0x00}, + {0x3803, 0x62}, + {0x3804, 0x05}, + {0x3805, 0x10}, + {0x3806, 0x03}, + {0x3807, 0xd0}, + {0x3808, 0x05}, + + {0x3809, 0x10}, + + {0x380a, 0x03}, + + {0x380b, 0xd0}, + + {0x380c, 0x08}, + + {0x380d, 0xa8}, + + {0x380e, 0x05}, + + {0x380f, 0xa4}, + + {0x3818, 0xc1}, + {0x381a, 0x00}, + {0x3a0d, 0x08}, + {0x3c01, 0x00}, + {0x3007, 0x3b}, + {0x5059, 0x80}, + {0x3003, 0x03}, + {0x3500, 0x00}, + + {0x3501, 0x5a}, + {0x3502, 0x10}, + {0x350a, 0x00}, + {0x350b, 0x10}, + {0x401d, 0x08}, + {0x4801, 0x0f}, + {0x300e, 0x0c}, + {0x4803, 0x50}, + {0x4800, 0x34}, + {OV5650_TABLE_END, 0x0000} +}; + +static struct ov5650_reg mode_2080x1164[] = { + {0x3103, 0x93}, + {0x3007, 0x3b}, + {0x3017, 0xff}, + {0x3018, 0xfc}, + + {0x3600, 0x54}, + {0x3601, 0x05}, + {0x3603, 0xa7}, + {0x3604, 0x40}, + {0x3605, 0x04}, + {0x3606, 0x3f}, + {0x3612, 0x1a}, + {0x3613, 0x44}, + {0x3615, 0x52}, + {0x3620, 0x56}, + {0x3623, 0x01}, + {0x3630, 0x22}, + {0x3631, 0x36}, + {0x3632, 0x5f}, + {0x3633, 0x24}, + + {0x3702, 0x3a}, + {0x3704, 0x18}, + {0x3706, 0x41}, + {0x370b, 0x40}, + {0x370e, 0x00}, + {0x3710, 0x28}, + {0x3711, 0x24}, + {0x3712, 0x13}, + + {0x3810, 0x00}, + {0x3815, 0x82}, + {0x3830, 0x50}, + {0x3836, 0x00}, + + {0x3a1a, 0x06}, + {0x3a18, 0x00}, + {0x3a19, 0xf8}, + {0x3a00, 0x38}, + + {0x3a0d, 0x06}, + {0x3c01, 0x34}, + + {0x401f, 0x03}, + {0x4000, 0x05}, + {0x401d, 0x08}, + {0x4001, 0x02}, + + {0x5001, 0x00}, + {0x5002, 0x00}, + {0x503d, 0x00}, + {0x5046, 0x00}, + + {0x300f, 0x8f}, + + {0x3010, 0x10}, + {0x3011, 0x14}, + {0x3012, 0x02}, + {0x3503, 0x00}, + + + {0x3621, 0x2f}, + + {0x3703, 0xe6}, + {0x370c, 0x00}, + {0x370d, 0x04}, + {0x3713, 0x22}, + {0x3714, 0x27}, + {0x3705, 0xda}, + {0x370a, 0x80}, + + {0x3800, 0x02}, + {0x3801, 0x12}, + {0x3802, 0x00}, + {0x3803, 0x0a}, + {0x3804, 0x08}, + {0x3805, 0x20}, + {0x3806, 0x04}, + {0x3807, 0x92}, + {0x3808, 0x08}, + + {0x3809, 0x20}, + + {0x380a, 0x04}, + + {0x380b, 0x92}, + + {0x380c, 0x0a}, + + {0x380d, 0x96}, + + {0x380e, 0x04}, + + {0x380f, 0x9e}, + + {0x3818, 0xc0}, + {0x381a, 0x3c}, + {0x381c, 0x31}, + {0x381d, 0x8e}, + {0x381e, 0x04}, + {0x381f, 0x92}, + {0x3820, 0x04}, + {0x3821, 0x19}, + {0x3824, 0x01}, + {0x3827, 0x0a}, + {0x401c, 0x46}, + + {0x3003, 0x03}, + {0x3500, 0x00}, + {0x3501, 0x49}, + {0x3502, 0xa0}, + {0x350a, 0x00}, + {0x350b, 0x00}, + {0x4801, 0x0f}, + {0x300e, 0x0c}, + {0x4803, 0x50}, + {0x4800, 0x34}, + + {OV5650_TABLE_END, 0x0000} +}; + +static struct ov5650_reg mode_1920x1080[] = { + {0x3103, 0x93}, + {0x3007, 0x3b}, + {0x3017, 0xff}, + {0x3018, 0xfc}, + + {0x3600, 0x54}, + {0x3601, 0x05}, + {0x3603, 0xa7}, + {0x3604, 0x40}, + {0x3605, 0x04}, + {0x3606, 0x3f}, + {0x3612, 0x1a}, + {0x3613, 0x44}, + {0x3615, 0x52}, + {0x3620, 0x56}, + {0x3623, 0x01}, + {0x3630, 0x22}, + {0x3631, 0x36}, + {0x3632, 0x5f}, + {0x3633, 0x24}, + + {0x3702, 0x3a}, + {0x3704, 0x18}, + {0x3706, 0x41}, + {0x370b, 0x40}, + {0x370e, 0x00}, + {0x3710, 0x28}, + {0x3711, 0x24}, + {0x3712, 0x13}, + + {0x3810, 0x00}, + {0x3815, 0x82}, + + {0x3830, 0x50}, + {0x3836, 0x00}, + + {0x3a1a, 0x06}, + {0x3a18, 0x00}, + {0x3a19, 0xf8}, + {0x3a00, 0x38}, + {0x3a0d, 0x06}, + {0x3c01, 0x34}, + + {0x401f, 0x03}, + {0x4000, 0x05}, + {0x401d, 0x08}, + {0x4001, 0x02}, + + {0x5001, 0x00}, + {0x5002, 0x00}, + {0x503d, 0x00}, + {0x5046, 0x00}, + + {0x300f, 0x8f}, + {0x3010, 0x10}, + {0x3011, 0x14}, + {0x3012, 0x02}, + {0x3503, 0x00}, + + {0x3621, 0x2f}, + {0x3703, 0xe6}, + {0x370c, 0x00}, + {0x370d, 0x04}, + {0x3713, 0x22}, + {0x3714, 0x27}, + {0x3705, 0xda}, + {0x370a, 0x80}, + + {0x3800, 0x02}, + {0x3801, 0x94}, + {0x3802, 0x00}, + {0x3803, 0x00}, + {0x3804, 0x07}, + {0x3805, 0x80}, + {0x3806, 0x04}, + {0x3807, 0x40}, + {0x3808, 0x07}, + {0x3809, 0x80}, + {0x380a, 0x04}, + {0x380b, 0x40}, + {0x380c, 0x0a}, + {0x380d, 0x84}, + {0x380e, 0x04}, + {0x380f, 0xa4}, + {0x3818, 0xc0}, + {0x381a, 0x3c}, + {0x381c, 0x31}, + {0x381d, 0xa4}, + {0x381e, 0x04}, + {0x381f, 0x60}, + {0x3820, 0x03}, + {0x3821, 0x1a}, + {0x3824, 0x01}, + {0x3827, 0x0a}, + {0x401c, 0x46}, + + {0x3003, 0x03}, + {0x3500, 0x00}, + {0x3501, 0x49}, + {0x3502, 0xa0}, + {0x350a, 0x00}, + {0x350b, 0x00}, + {0x4801, 0x0f}, + {0x300e, 0x0c}, + {0x4803, 0x50}, + {0x4800, 0x34}, + + {OV5650_TABLE_END, 0x0000} +}; + +static struct ov5650_reg mode_1280x720[] = { + {0x3103, 0x93}, + {0x3b07, 0x0c}, + {0x3017, 0xff}, + {0x3018, 0xfc}, + {0x3706, 0x41}, + {0x3613, 0xc4}, + {0x370d, 0x42}, + {0x3703, 0x9a}, + {0x3630, 0x22}, + {0x3605, 0x04}, + {0x3606, 0x3f}, + {0x3712, 0x13}, + {0x370e, 0x00}, + {0x370b, 0x40}, + {0x3600, 0x54}, + {0x3601, 0x05}, + {0x3713, 0x22}, + {0x3714, 0x27}, + {0x3631, 0x22}, + {0x3612, 0x1a}, + {0x3604, 0x40}, + {0x3705, 0xdb}, + {0x370a, 0x81}, + {0x370c, 0x00}, + {0x3710, 0x28}, + {0x3702, 0x3a}, + {0x3704, 0x18}, + {0x3a18, 0x00}, + {0x3a19, 0xf8}, + {0x3a00, 0x38}, + {0x3800, 0x02}, + {0x3801, 0x54}, + {0x3803, 0x0c}, + {0x380c, 0x0c}, + {0x380d, 0xb4}, + {0x380e, 0x07}, + {0x380f, 0xb0}, + {0x3830, 0x50}, + {0x3a08, 0x12}, + {0x3a09, 0x70}, + {0x3a0a, 0x0f}, + {0x3a0b, 0x60}, + {0x3a0d, 0x06}, + {0x3a0e, 0x06}, + {0x3a13, 0x54}, + {0x3815, 0x82}, + {0x5059, 0x80}, + {0x3615, 0x52}, + {0x505a, 0x0a}, + {0x505b, 0x2e}, + {0x3713, 0x92}, + {0x3714, 0x17}, + {0x3804, 0x05}, + {0x3805, 0x00}, + {0x3806, 0x02}, + {0x3807, 0xd0}, + {0x3808, 0x05}, + {0x3809, 0x00}, + {0x380a, 0x02}, + {0x380b, 0xd0}, + {0x380c, 0x08}, + {0x380d, 0x72}, + {0x380e, 0x02}, + {0x380f, 0xe4}, + {0x3815, 0x81}, + {0x381c, 0x10}, + {0x381d, 0x82}, + {0x381e, 0x05}, + {0x381f, 0xc0}, + {0x3821, 0x20}, + {0x3824, 0x23}, + {0x3825, 0x2c}, + {0x3826, 0x00}, + {0x3827, 0x0c}, + {0x3a08, 0x1b}, + {0x3a09, 0xc0}, + {0x3a0a, 0x17}, + {0x3a0b, 0x20}, + {0x3a0d, 0x01}, + {0x3a0e, 0x01}, + {0x3a1a, 0x06}, + {0x3503, 0x00}, + {0x3623, 0x01}, + {0x3633, 0x24}, + {0x3c01, 0x34}, + {0x3c04, 0x28}, + {0x3c05, 0x98}, + {0x3c07, 0x07}, + {0x3c09, 0xc2}, + {0x4000, 0x05}, + {0x401d, 0x28}, + {0x4001, 0x02}, + {0x401c, 0x42}, + {0x5046, 0x09}, + {0x3810, 0x40}, + {0x3836, 0x41}, + {0x505f, 0x04}, + {0x5000, 0xfe}, + {0x5001, 0x01}, + {0x5002, 0x00}, + {0x503d, 0x00}, /* bit[7]=1 enable test_pattern */ + {0x5901, 0x00}, + {0x585a, 0x01}, + {0x585b, 0x2c}, + {0x585c, 0x01}, + {0x585d, 0x93}, + {0x585e, 0x01}, + {0x585f, 0x90}, + {0x5860, 0x01}, + {0x5861, 0x0d}, + {0x5180, 0xc0}, + {0x5184, 0x00}, + {0x470a, 0x00}, + {0x470b, 0x00}, + {0x470c, 0x00}, + {0x300f, 0x8e}, + {0x3603, 0xa7}, + {0x3632, 0x55}, + {0x3620, 0x56}, + {0x3621, 0xaf}, + {0x3818, 0xc1}, + {0x3631, 0x36}, + {0x3632, 0x5f}, + {0x3711, 0x24}, + {0x401f, 0x03}, + {0x3008, 0x02}, + + {0x3011, 0x14}, + {0x3007, 0x3B}, + {0x4801, 0x0f}, + {0x3003, 0x03}, + {0x300e, 0x0c}, + {0x4803, 0x50}, + {0x4800, 0x04}, /* bit[5]=0 as CSI continuous clock */ + {0x300f, 0x8f}, + {0x3010, 0x10}, + {0x3815, 0x82}, + {0x3003, 0x01}, + + {OV5650_TABLE_END, 0x0000} +}; + +static struct ov5650_reg mode_1264x704[] = { + {0x3600, 0x54}, + {0x3601, 0x05}, + {0x3604, 0x40}, + {0x3705, 0xdb}, + {0x370a, 0x81}, + {0x3615, 0x52}, + {0x3810, 0x40}, + {0x3836, 0x41}, + {0x4000, 0x05}, + {0x401c, 0x42}, + {0x401d, 0x08}, + {0x5046, 0x09}, + {0x3010, 0x00}, + {0x3503, 0x00}, + {0x3613, 0xc4}, + + {0x3621, 0xaf}, + + {0x3632, 0x55}, + {0x3703, 0x9a}, + {0x370c, 0x00}, + {0x370d, 0x42}, + {0x3713, 0x22}, + {0x3800, 0x02}, + {0x3801, 0x54}, + {0x3802, 0x00}, + {0x3803, 0x0c}, + {0x3804, 0x05}, + {0x3805, 0x00}, + {0x3806, 0x02}, + {0x3807, 0xd0}, + {0x3808, 0x05}, + + {0x3809, 0x00}, + + {0x380a, 0x02}, + + {0x380b, 0xd0}, + + {0x380c, 0x08}, + + {0x380d, 0x72}, + + {0x380e, 0x02}, + + {0x380f, 0xe4}, + + {0x3818, 0xc1}, + {0x381a, 0x3c}, + {0x3a0d, 0x06}, + {0x3c01, 0x34}, + {0x3007, 0x3b}, + {0x5059, 0x80}, + {0x3003, 0x03}, + {0x3500, 0x04}, + {0x3501, 0xa5}, + + {0x3502, 0x10}, + + {0x350a, 0x00}, + {0x350b, 0x00}, + {0x4801, 0x0f}, + {0x300e, 0x0c}, + {0x4803, 0x50}, + {0x4800, 0x24}, + {0x300f, 0x8b}, + + {0x3711, 0x24}, + {0x3713, 0x92}, + {0x3714, 0x17}, + {0x381c, 0x10}, + {0x381d, 0x82}, + {0x381e, 0x05}, + {0x381f, 0xc0}, + {0x3821, 0x20}, + {0x3824, 0x23}, + {0x3825, 0x2c}, + {0x3826, 0x00}, + {0x3827, 0x0c}, + {0x3623, 0x01}, + {0x3633, 0x24}, + {0x3632, 0x5f}, + {0x401f, 0x03}, + + {OV5650_TABLE_END, 0x0000} +}; + +static struct ov5650_reg mode_320x240[] = { + {0x3103, 0x93}, + {0x3b07, 0x0c}, + {0x3017, 0xff}, + {0x3018, 0xfc}, + {0x3706, 0x41}, + {0x3613, 0xc4}, + {0x370d, 0x42}, + {0x3703, 0x9a}, + {0x3630, 0x22}, + {0x3605, 0x04}, + {0x3606, 0x3f}, + {0x3712, 0x13}, + {0x370e, 0x00}, + {0x370b, 0x40}, + {0x3600, 0x54}, + {0x3601, 0x05}, + {0x3713, 0x22}, + {0x3714, 0x27}, + {0x3631, 0x22}, + {0x3612, 0x1a}, + {0x3604, 0x40}, + {0x3705, 0xdc}, + {0x370a, 0x83}, + {0x370c, 0xc8}, + {0x3710, 0x28}, + {0x3702, 0x3a}, + {0x3704, 0x18}, + {0x3a18, 0x00}, + {0x3a19, 0xf8}, + {0x3a00, 0x38}, + {0x3800, 0x02}, + {0x3801, 0x54}, + {0x3803, 0x0c}, + {0x380c, 0x0c}, + {0x380d, 0xb4}, + {0x380e, 0x07}, + {0x380f, 0xb0}, + {0x3830, 0x50}, + {0x3a08, 0x12}, + {0x3a09, 0x70}, + {0x3a0a, 0x0f}, + {0x3a0b, 0x60}, + {0x3a0d, 0x06}, + {0x3a0e, 0x06}, + {0x3a13, 0x54}, + {0x3815, 0x82}, + {0x5059, 0x80}, + {0x3615, 0x52}, + {0x505a, 0x0a}, + {0x505b, 0x2e}, + {0x3713, 0x92}, + {0x3714, 0x17}, + {0x3803, 0x0a}, + {0x3804, 0x05}, + {0x3805, 0x00}, + {0x3806, 0x01}, + {0x3807, 0x00}, + {0x3808, 0x01}, + {0x3809, 0x40}, + {0x380a, 0x01}, + {0x380b, 0x00}, + {0x380c, 0x0a}, + + {0x380d, 0x04}, + + {0x380e, 0x01}, + + {0x380f, 0x38}, + + {0x3500, 0x00}, + {0x3501, 0x13}, + {0x3502, 0x80}, + {0x350b, 0x7f}, + + {0x3815, 0x81}, + {0x3824, 0x23}, + {0x3825, 0x20}, + {0x3826, 0x00}, + {0x3827, 0x08}, + {0x370d, 0xc2}, + {0x3a08, 0x17}, + {0x3a09, 0x64}, + {0x3a0a, 0x13}, + {0x3a0b, 0x80}, + {0x3a00, 0x58}, + {0x3a1a, 0x06}, + {0x3503, 0x00}, + {0x3623, 0x01}, + {0x3633, 0x24}, + {0x3c01, 0x34}, + {0x3c04, 0x28}, + {0x3c05, 0x98}, + {0x3c07, 0x07}, + {0x3c09, 0xc2}, + {0x4000, 0x05}, + {0x401d, 0x08}, + {0x4001, 0x02}, + {0x401c, 0x42}, + {0x5046, 0x09}, + {0x3810, 0x40}, + {0x3836, 0x41}, + {0x505f, 0x04}, + {0x5001, 0x00}, + {0x5002, 0x02}, + {0x503d, 0x00}, + {0x5901, 0x08}, + {0x585a, 0x01}, + {0x585b, 0x2c}, + {0x585c, 0x01}, + {0x585d, 0x93}, + {0x585e, 0x01}, + {0x585f, 0x90}, + {0x5860, 0x01}, + {0x5861, 0x0d}, + {0x5180, 0xc0}, + {0x5184, 0x00}, + {0x470a, 0x00}, + {0x470b, 0x00}, + {0x470c, 0x00}, + {0x300f, 0x8e}, + {0x3603, 0xa7}, + {0x3632, 0x55}, + {0x3620, 0x56}, + {0x3621, 0xaf}, + {0x3818, 0xc3}, + {0x3631, 0x36}, + {0x3632, 0x5f}, + {0x3711, 0x24}, + {0x401f, 0x03}, + + {0x3011, 0x14}, + {0x3007, 0x3B}, + {0x300f, 0x8f}, + {0x4801, 0x0f}, + {0x3003, 0x03}, + {0x300e, 0x0c}, + {0x3010, 0x15}, + {0x4803, 0x50}, + {0x4800, 0x24}, + {0x4837, 0x40}, + {0x3815, 0x82}, + + {OV5650_TABLE_END, 0x0000} +}; + +static struct ov5650_reg mode_end[] = { + {0x3212, 0x00}, + {0x3003, 0x01}, + {0x3212, 0x10}, + {0x3212, 0xa0}, + {0x3008, 0x02}, + + {OV5650_TABLE_END, 0x0000} +}; + +enum { + OV5650_MODE_2592x1944, + OV5650_MODE_2080x1164, + OV5650_MODE_1920x1080, + OV5650_MODE_1296x972, + OV5650_MODE_1280x720, + OV5650_MODE_1264x704, + OV5650_MODE_320x240, + OV5650_MODE_INVALID +}; + +static struct ov5650_reg *mode_table[] = { + [OV5650_MODE_2592x1944] = mode_2592x1944, + [OV5650_MODE_2080x1164] = mode_2080x1164, + [OV5650_MODE_1920x1080] = mode_1920x1080, + [OV5650_MODE_1296x972] = mode_1296x972, + [OV5650_MODE_1280x720] = mode_1280x720, + [OV5650_MODE_1264x704] = mode_1264x704, + [OV5650_MODE_320x240] = mode_320x240 +}; + +static const struct v4l2_frmsize_discrete ov5650_frmsizes[] = { + {2592, 1944}, + {2080, 1164}, + {1920, 1080}, + {1296, 972}, + {1280, 720}, + {1264, 704}, + {320, 240}, +}; + +static int ov5650_find_mode(u32 width, u32 height) +{ + if (width == 2592 && height == 1944) + return OV5650_MODE_2592x1944; + else if (width == 2080 && height == 1164) + return OV5650_MODE_2080x1164; + else if (width == 1920 && height == 1080) + return OV5650_MODE_1920x1080; + else if (width == 1296 && height == 972) + return OV5650_MODE_1296x972; + else if (width == 1280 && height == 720) + return OV5650_MODE_1280x720; + else if (width == 1264 && height == 704) + return OV5650_MODE_1264x704; + else if (width == 320 && height == 240) + return OV5650_MODE_320x240; + else { + pr_err("ov5650: %dx%d is not supported\n", width, height); + return OV5650_MODE_2592x1944; + } +} + +/** + * ov5650_reg_read - Read a value from a register in an ov5650 sensor device + * @client: i2c driver client structure + * @reg: register address / offset + * @val: stores the value that gets read + * + * Read a value from a register in an ov5650 sensor device. + * The value is returned in 'val'. + * Returns zero if successful, or non-zero otherwise. + */ +static int ov5650_reg_read(struct i2c_client *client, u16 reg, u8 *val) +{ + int ret; + u8 data[2] = {0}; + struct i2c_msg msg = { + .addr = client->addr, + .flags = 0, + .len = 2, + .buf = data, + }; + + data[0] = (u8)(reg >> 8); + data[1] = (u8)(reg & 0xff); + + ret = i2c_transfer(client->adapter, &msg, 1); + if (ret < 0) + goto err; + + msg.flags = I2C_M_RD; + msg.len = 1; + ret = i2c_transfer(client->adapter, &msg, 1); + if (ret < 0) + goto err; + + *val = data[0]; + return 0; + +err: + dev_err(&client->dev, "Failed reading register 0x%02x!\n", reg); + return ret; +} + +/** + * Write a value to a register in ov5650 sensor device. + * @client: i2c driver client structure. + * @reg: Address of the register to read value from. + * @val: Value to be written to a specific register. + * Returns zero if successful, or non-zero otherwise. + */ +static int ov5650_reg_write(struct i2c_client *client, u16 reg, u8 val) +{ + int ret; + unsigned char data[3] = { (u8)(reg >> 8), (u8)(reg & 0xff), val }; + struct i2c_msg msg = { + .addr = client->addr, + .flags = 0, + .len = 3, + .buf = data, + }; + + ret = i2c_transfer(client->adapter, &msg, 1); + if (ret < 0) { + dev_err(&client->dev, "Failed writing register 0x%02x!\n", reg); + return ret; + } + + return 0; +} + +static int ov5650_write_bulk_reg(struct ov5650_priv *priv, int len) +{ + struct i2c_client *client = priv->client; + u8 *data = priv->i2c_trans_buf; + int err; + struct i2c_msg msg; + + if (!client->adapter) + return -ENODEV; + + msg.addr = client->addr; + msg.flags = 0; + msg.len = len; + msg.buf = data; + + err = i2c_transfer(client->adapter, &msg, 1); + if (err != 1) { + dev_err(&client->dev, "I2C bulk transfer failed at %x\n", + (int)data[0] << 8 | data[1]); + return err; + } + + return 0; +} +static int ov5650_write_table(struct ov5650_priv *priv, + const struct ov5650_reg table[], + const struct ov5650_reg override_list[], + int num_override_regs) +{ + int err; + const struct ov5650_reg *next, *n_next; + u8 *b_ptr = priv->i2c_trans_buf; + unsigned int buf_filled = 0; + unsigned int i; + u16 val; + + for (next = table; next->addr != OV5650_TABLE_END; next++) { + if (next->addr == OV5650_TABLE_WAIT_MS) { + msleep(next->val); + continue; + } + + val = next->val; + /* When an override list is passed in, replace the reg */ + /* value to write if the reg is in the list */ + if (override_list) { + for (i = 0; i < num_override_regs; i++) { + if (next->addr == override_list[i].addr) { + val = override_list[i].val; + break; + } + } + } + + if (!buf_filled) { + b_ptr = priv->i2c_trans_buf; + *b_ptr++ = next->addr >> 8; + *b_ptr++ = next->addr & 0xff; + buf_filled = 2; + } + *b_ptr++ = val; + buf_filled++; + + n_next = next + 1; + if (n_next->addr != OV5650_TABLE_END && + n_next->addr != OV5650_TABLE_WAIT_MS && + buf_filled < SIZEOF_I2C_TRANSBUF && + n_next->addr == next->addr + 1) { + continue; + } + + err = ov5650_write_bulk_reg(priv, buf_filled); + if (err) + return err; + + buf_filled = 0; + } + return 0; +} + +/* ----------------------------------------------------------------------------- + * V4L2 subdev internal operations + */ + +static void ov5650_set_default_fmt(struct ov5650_priv *priv) +{ + struct v4l2_mbus_framefmt *mf = &priv->mf; + + mf->width = ov5650_frmsizes[OV5650_MODE_2592x1944].width; + mf->height = ov5650_frmsizes[OV5650_MODE_2592x1944].height; + mf->code = V4L2_MBUS_FMT_SBGGR10_1X10; + mf->field = V4L2_FIELD_NONE; + mf->colorspace = V4L2_COLORSPACE_SRGB; +} + +static int ov5650_s_stream(struct v4l2_subdev *sd, int enable) +{ + struct ov5650_priv *priv = to_ov5650(sd); + struct ov5650_reg reg_list[6]; + int ret = 0; + + if (enable) { + ret = ov5650_write_table(priv, reset_seq, NULL, 0); + if (ret) + return ret; + + ret = ov5650_write_table(priv, mode_start, NULL, 0); + if (ret) + return ret; + + ret = ov5650_write_table(priv, mode_table[priv->mode], + reg_list, 6); + if (ret) + return ret; + + ret = ov5650_write_table(priv, mode_end, NULL, 0); + if (ret) + return ret; + + switch (test_mode) { + case TP_NONE: + ret = ov5650_write_table(priv, tp_none_seq, NULL, 0); + break; + case TP_COLORBARS: + ret = ov5650_write_table(priv, tp_cbars_seq, NULL, 0); + break; + case TP_CHECKER: + ret = ov5650_write_table(priv, tp_checker_seq, NULL, 0); + break; + } + if (ret) + return ret; + + } else + ov5650_set_default_fmt(priv); + + return ret; +} + +/* Alter bus settings on camera side */ +static int ov5650_set_bus_param(struct soc_camera_device *icd, + unsigned long flags) +{ + return 0; +} + +/* Request bus settings on camera side */ +static unsigned long ov5650_query_bus_param(struct soc_camera_device *icd) +{ + struct soc_camera_link *icl = to_soc_camera_link(icd); + + unsigned long flags = SOCAM_PCLK_SAMPLE_RISING | SOCAM_MASTER | + SOCAM_VSYNC_ACTIVE_HIGH | SOCAM_HSYNC_ACTIVE_HIGH | + SOCAM_DATA_ACTIVE_HIGH | SOCAM_DATAWIDTH_10; + + return soc_camera_apply_sensor_flags(icl, flags); +} + +static int ov5650_s_fmt(struct v4l2_subdev *sd, + struct v4l2_mbus_framefmt *mf) +{ + struct ov5650_priv *priv = to_ov5650(sd); + + priv->mode = ov5650_find_mode(mf->width, mf->height); + memcpy(&priv->mf, mf, sizeof(struct v4l2_mbus_framefmt)); + + return 0; +} + +static int ov5650_s_power(struct v4l2_subdev *sd, int on) +{ + struct ov5650_priv *priv = to_ov5650(sd); + + if (on) { + ov5650_s_fmt(sd, &priv->mf); + ov5650_s_stream(sd, 1); + } else + ov5650_s_stream(sd, 0); + + return 0; +} + +static int ov5650_try_fmt(struct v4l2_subdev *sd, + struct v4l2_mbus_framefmt *mf) +{ + int mode; + + mode = ov5650_find_mode(mf->width, mf->height); + mf->width = ov5650_frmsizes[mode].width; + mf->height = ov5650_frmsizes[mode].height; + + if (mf->code != V4L2_MBUS_FMT_SBGGR8_1X8 && + mf->code != V4L2_MBUS_FMT_SBGGR10_1X10) + mf->code = V4L2_MBUS_FMT_SBGGR10_1X10; + + mf->field = V4L2_FIELD_NONE; + mf->colorspace = V4L2_COLORSPACE_SRGB; + + ov5650_s_fmt(sd, mf); + + return 0; +} + +static int ov5650_enum_fmt(struct v4l2_subdev *sd, unsigned int index, + enum v4l2_mbus_pixelcode *code) +{ + if (index >= 2) + return -EINVAL; + + switch (index) { + case 0: + *code = V4L2_MBUS_FMT_SBGGR10_1X10; + break; + case 1: + *code = V4L2_MBUS_FMT_SBGGR8_1X8; + break; + } + + return 0; +} + +static int ov5650_cropcap(struct v4l2_subdev *sd, struct v4l2_cropcap *a) +{ + a->bounds.left = 0; + a->bounds.top = 0; + a->bounds.width = ov5650_frmsizes[OV5650_MODE_2592x1944].width; + a->bounds.height = ov5650_frmsizes[OV5650_MODE_2592x1944].height; + a->defrect = a->bounds; + a->type = V4L2_BUF_TYPE_VIDEO_CAPTURE; + a->pixelaspect.numerator = 1; + a->pixelaspect.denominator = 1; + + return 0; +} + +static int ov5650_g_crop(struct v4l2_subdev *sd, struct v4l2_crop *a) +{ + a->c.left = 0; + a->c.top = 0; + a->c.width = ov5650_frmsizes[OV5650_MODE_2592x1944].width; + a->c.height = ov5650_frmsizes[OV5650_MODE_2592x1944].height; + a->type = V4L2_BUF_TYPE_VIDEO_CAPTURE; + + return 0; +} + +/* Get chip identification */ +static int ov5650_g_chip_ident(struct v4l2_subdev *sd, + struct v4l2_dbg_chip_ident *id) +{ + struct ov5650_priv *priv = to_ov5650(sd); + + id->ident = priv->ident; + id->revision = priv->revision; + + return 0; +} + +static struct soc_camera_ops ov5650_ops = { + .set_bus_param = ov5650_set_bus_param, + .query_bus_param = ov5650_query_bus_param, +}; + +static struct v4l2_subdev_video_ops ov5650_video_ops = { + .s_stream = ov5650_s_stream, + .s_mbus_fmt = ov5650_s_fmt, + .try_mbus_fmt = ov5650_try_fmt, + .enum_mbus_fmt = ov5650_enum_fmt, + .cropcap = ov5650_cropcap, + .g_crop = ov5650_g_crop, +}; + +static struct v4l2_subdev_core_ops ov5650_core_ops = { + .g_chip_ident = ov5650_g_chip_ident, + .s_power = ov5650_s_power, +}; + +static struct v4l2_subdev_ops ov5650_subdev_ops = { + .core = &ov5650_core_ops, + .video = &ov5650_video_ops, +}; + +static int ov5650_probe(struct i2c_client *client, + const struct i2c_device_id *did) +{ + struct ov5650_priv *priv; + struct soc_camera_device *icd = client->dev.platform_data; + struct soc_camera_link *icl; + u8 chipid[2]; + int ret; + + /* Checking soc-camera interface */ + if (!icd) { + dev_err(&client->dev, "Missing soc-camera data!\n"); + return -EINVAL; + } + + icl = to_soc_camera_link(icd); + if (!icl) { + dev_err(&client->dev, "Missing platform_data for driver\n"); + return -EINVAL; + } + + /* Register OV5650 soc_camera device interface */ + priv = kzalloc(sizeof(struct ov5650_priv), GFP_KERNEL); + if (!priv) { + dev_err(&client->dev, "Failed to allocate private data!\n"); + return -ENOMEM; + } + + v4l2_i2c_subdev_init(&priv->subdev, client, &ov5650_subdev_ops); + icd->ops = &ov5650_ops; + + priv->subdev.flags |= V4L2_SUBDEV_FL_HAS_DEVNODE; + + priv->ident = V4L2_IDENT_OV5650; + + ret = ov5650_reg_read(client, 0x300A, &chipid[0]); + if (ret) { + dev_err(&client->dev, "Failure to read Chip ID (high byte)\n"); + goto err; + } + + ret = ov5650_reg_read(client, 0x300B, &chipid[1]); + if (ret) { + dev_err(&client->dev, "Failure to read Chip ID (low byte)\n"); + goto err; + } + + if ((chipid[0] != 0x56) || ((chipid[1] & 0x51) != chipid[1])) { + dev_err(&client->dev, "Chip ID: %x%x not supported!\n", + chipid[0], chipid[1]); + ret = -ENODEV; + goto err; + } + + priv->chip_id = (chipid[0] << 8) | chipid[1]; + priv->revision = (chipid[1] == 0x50) ? 0x1A : 0x1B; + + priv->client = client; + + ov5650_set_default_fmt(priv); + + dev_info(&client->dev, "Detected a OV%x chip, revision %x\n", + priv->chip_id, priv->revision); + + return 0; + +err: + kfree(priv); + + return ret; +} + +static int ov5650_remove(struct i2c_client *client) +{ + struct ov5650_priv *priv = i2c_get_clientdata(client); + + kfree(priv); + return 0; +} + +static const struct i2c_device_id ov5650_id[] = { + { "ov5650", 0 }, + { } +}; +MODULE_DEVICE_TABLE(i2c, ov5650_id); + +static struct i2c_driver ov5650_i2c_driver = { + .driver = { + .name = "ov5650", + }, + .probe = ov5650_probe, + .remove = ov5650_remove, + .id_table = ov5650_id, +}; + +static int __init ov5650_module_init(void) +{ + return i2c_add_driver(&ov5650_i2c_driver); +} + +static void __exit ov5650_module_exit(void) +{ + i2c_del_driver(&ov5650_i2c_driver); +} + +module_init(ov5650_module_init); +module_exit(ov5650_module_exit); + +MODULE_DESCRIPTION("OmniVision OV5650 Camera driver"); +MODULE_AUTHOR("Bryan Wu <pengw@nvidia.com>"); +MODULE_LICENSE("GPL v2"); diff --git a/drivers/media/video/tegra/sh532u.c b/drivers/media/video/tegra/sh532u.c index e41dd1f4afb1..b9f9b91d5a45 100644 --- a/drivers/media/video/tegra/sh532u.c +++ b/drivers/media/video/tegra/sh532u.c @@ -1,7 +1,7 @@ /* * SH532U focuser driver. * - * Copyright (C) 2011-2012 NVIDIA Corporation. + * Copyright (C) 2011-2013 NVIDIA Corporation. * * 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 @@ -386,7 +386,7 @@ static void sh532u_gpio_exit(struct sh532u_info *info) { unsigned i; - for (i = 0; i <= ARRAY_SIZE(sh532u_gpios); i++) { + for (i = 0; i < ARRAY_SIZE(sh532u_gpios); i++) { if (info->gpio[i].flag && info->gpio[i].own) { gpio_free(info->gpio[i].gpio); info->gpio[i].own = false; diff --git a/drivers/media/video/videobuf2-dma-nvmap.c b/drivers/media/video/videobuf2-dma-nvmap.c index 8c3c36c9922f..4ba8fed0cc02 100644 --- a/drivers/media/video/videobuf2-dma-nvmap.c +++ b/drivers/media/video/videobuf2-dma-nvmap.c @@ -1,5 +1,5 @@ /* - * Copyright (c) 2012, NVIDIA CORPORATION. All rights reserved. + * Copyright (c) 2012-2013, NVIDIA CORPORATION. All rights reserved. * * This program is free software; you can redistribute it and/or modify it * under the terms and conditions of the GNU General Public License, @@ -82,6 +82,8 @@ static void *vb2_dma_nvmap_alloc(void *alloc_ctx, unsigned long size) buf->handler.put = vb2_dma_nvmap_put; buf->handler.arg = buf; + *((unsigned long *)buf->vaddr) = (unsigned long)buf->nvmap_ref->handle; + atomic_inc(&buf->refcount); return buf; @@ -135,12 +137,12 @@ static int vb2_dma_nvmap_mmap(void *buf_priv, struct vm_area_struct *vma) { struct vb2_dc_buf *buf = buf_priv; unsigned long vm_start, paddr; - void * vaddr; + void *vaddr; int size; int ret; if (!buf) { - printk(KERN_ERR "No buffer to map\n"); + pr_err("No buffer to map\n"); return -EINVAL; } @@ -155,7 +157,8 @@ static int vb2_dma_nvmap_mmap(void *buf_priv, struct vm_area_struct *vma) ret = remap_pfn_range(vma, vm_start, paddr >> PAGE_SHIFT, PAGE_SIZE, vma->vm_page_prot); if (ret) { - printk(KERN_ERR "Remapping memory failed, error: %d\n", ret); + pr_err("Remapping memory failed, error: %d\n", + ret); return ret; } pr_debug("%s: mapped paddr 0x%08lx at 0x%08lx, size %ld\n", @@ -185,8 +188,8 @@ static void *vb2_dma_nvmap_get_userptr(void *alloc_ctx, unsigned long vaddr, ret = vb2_get_contig_userptr(vaddr, size, &vma, &paddr); if (ret) { - printk(KERN_ERR "Failed acquiring VMA for vaddr 0x%08lx\n", - vaddr); + pr_err("Failed acquiring VMA for vaddr 0x%08lx\n", + vaddr); kfree(buf); return ERR_PTR(ret); } diff --git a/drivers/net/Kconfig b/drivers/net/Kconfig index 41b279abbd5c..1c3fac524474 100644 --- a/drivers/net/Kconfig +++ b/drivers/net/Kconfig @@ -2237,18 +2237,27 @@ config R8169 To compile this driver as a module, choose M here: the module will be called r8169. This is recommended. -config SB1250_MAC - tristate "SB1250 Gigabit Ethernet support" - depends on SIBYTE_SB1xxx_SOC - select PHYLIB +config R8169 + tristate "Realtek 8169 gigabit ethernet support" + depends on PCI + select CRC32 + select MII ---help--- - This driver supports Gigabit Ethernet interfaces based on the - Broadcom SiByte family of System-On-a-Chip parts. They include - the BCM1120, BCM1125, BCM1125H, BCM1250, BCM1255, BCM1280, BCM1455 - and BCM1480 chips. + Say Y here if you have a Realtek 8169 PCI Gigabit Ethernet adapter. To compile this driver as a module, choose M here: the module - will be called sb1250-mac. + will be called r8169. This is recommended. + +config R8169_FW_LOAD + bool "Load firmware for Realtek 8169" + depends on R8169 + select FW_LOADER + default y + ---help--- + This enables runtime loading of optional firmware by the Realtek 8169 + driver. Some environments do not require firmware, or prefer not to delay + the boot process when firmware is not available. Say N here to disable + firmware loading. If unsure, say Y. config SIS190 tristate "SiS190/SiS191 gigabit ethernet support" diff --git a/drivers/net/r8169.c b/drivers/net/r8169.c index 6d657cabb951..a7ff1005ea64 100644 --- a/drivers/net/r8169.c +++ b/drivers/net/r8169.c @@ -1327,7 +1327,11 @@ static int rtl8169_set_wol(struct net_device *dev, struct ethtool_wolinfo *wol) static const char *rtl_lookup_firmware_name(struct rtl8169_private *tp) { +#ifdef CONFIG_R8169_FW_LOAD return rtl_chip_infos[tp->mac_version].fw_name; +#else + return NULL; +#endif } static void rtl8169_get_drvinfo(struct net_device *dev, diff --git a/drivers/video/tegra/host/Makefile b/drivers/video/tegra/host/Makefile index 89a10faefaba..c2608218c811 100644 --- a/drivers/video/tegra/host/Makefile +++ b/drivers/video/tegra/host/Makefile @@ -22,11 +22,9 @@ obj-$(CONFIG_TEGRA_GRHOST) += t20/ obj-$(CONFIG_TEGRA_GRHOST) += t30/ obj-$(CONFIG_TEGRA_GRHOST) += gr2d/ obj-$(CONFIG_TEGRA_GRHOST) += isp/ -ifneq ($(CONFIG_VIDEO_TEGRA),y) -ifneq ($(CONFIG_VIDEO_TEGRA),m) +ifeq ($(CONFIG_TEGRA_CAMERA),y) obj-$(CONFIG_TEGRA_GRHOST) += vi/ endif -endif obj-$(CONFIG_TEGRA_GRHOST) += nvhost.o obj-$(CONFIG_TEGRA_GRHOST_USE_NVMAP) += nvmap.o diff --git a/drivers/video/tegra/nvmap/nvmap_dev.c b/drivers/video/tegra/nvmap/nvmap_dev.c index 6a3330562b7b..3ea7bd2ea797 100644 --- a/drivers/video/tegra/nvmap/nvmap_dev.c +++ b/drivers/video/tegra/nvmap/nvmap_dev.c @@ -1,10 +1,10 @@ /* + * Copyright (c) 2011-2013, NVIDIA CORPORATION. All rights reserved. + * * drivers/video/tegra/nvmap/nvmap_dev.c * * User-space interface to nvmap * - * Copyright (c) 2011-2012, NVIDIA Corporation. - * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or @@ -232,6 +232,14 @@ struct nvmap_handle *nvmap_get_handle_id(struct nvmap_client *client, struct nvmap_handle_ref *ref; struct nvmap_handle *h = NULL; + /* Allow the handle to be accessed by other (non-owner) + clients only if the owner is "videobuf2-dma-nvmap", + which is a V4L2 capture kernel module. This handle can + be accessed by the "user" client for rendering */ + if (!strcmp(((struct nvmap_handle *)id)->owner->name, + "videobuf2-dma-nvmap")) + client = ((struct nvmap_handle *)id)->owner; + nvmap_ref_lock(client); ref = _nvmap_validate_id_locked(client, id); if (ref) |