diff options
Diffstat (limited to 'linux')
13 files changed, 630 insertions, 196 deletions
| diff --git a/linux/drivers/media/video/gspca/m5602/Makefile b/linux/drivers/media/video/gspca/m5602/Makefile index 9fa3644f4..bf7a19a1e 100644 --- a/linux/drivers/media/video/gspca/m5602/Makefile +++ b/linux/drivers/media/video/gspca/m5602/Makefile @@ -2,9 +2,10 @@ obj-$(CONFIG_USB_M5602) += gspca_m5602.o  gspca_m5602-objs := m5602_core.o \  		    m5602_ov9650.o \ +		    m5602_ov7660.o \  		    m5602_mt9m111.o \  		    m5602_po1030.o \  		    m5602_s5k83a.o \  		    m5602_s5k4aa.o -EXTRA_CFLAGS += -Idrivers/media/video/gspca
\ No newline at end of file +EXTRA_CFLAGS += -Idrivers/media/video/gspca diff --git a/linux/drivers/media/video/gspca/m5602/m5602_core.c b/linux/drivers/media/video/gspca/m5602/m5602_core.c index 93302f31a..36bdcda84 100644 --- a/linux/drivers/media/video/gspca/m5602/m5602_core.c +++ b/linux/drivers/media/video/gspca/m5602/m5602_core.c @@ -17,6 +17,7 @@   */  #include "m5602_ov9650.h" +#include "m5602_ov7660.h"  #include "m5602_mt9m111.h"  #include "m5602_po1030.h"  #include "m5602_s5k83a.h" @@ -84,10 +85,11 @@ int m5602_wait_for_i2c(struct sd *sd)  {  	int err;  	u8 data; +  	do {  		err = m5602_read_bridge(sd, M5602_XB_I2C_STATUS, &data);  	} while ((data & I2C_BUSY) && !err); -	return (err < 0) ? err : 0; +	return err;  }  int m5602_read_sensor(struct sd *sd, const u8 address, @@ -111,18 +113,16 @@ int m5602_read_sensor(struct sd *sd, const u8 address,  	if (err < 0)  		return err; +	/* Sensors with registers that only are one byte width are differently read */ +	/* FIXME: This works with the ov9650, but has issues with the po1030 */  	if (sd->sensor->i2c_regW == 1) { -		err = m5602_write_bridge(sd, M5602_XB_I2C_CTRL, len); +		err = m5602_write_bridge(sd, M5602_XB_I2C_CTRL, 1);  		if (err < 0)  			return err;  		err = m5602_write_bridge(sd, M5602_XB_I2C_CTRL, 0x08); -		if (err < 0) -			return err;  	} else {  		err = m5602_write_bridge(sd, M5602_XB_I2C_CTRL, 0x18 + len); -		if (err < 0) -			return err;  	}  	for (i = 0; (i < len) && !err; i++) { @@ -218,6 +218,11 @@ static int m5602_probe_sensor(struct sd *sd)  	if (!sd->sensor->probe(sd))  		return 0; +	/* Try the ov7660 */ +	sd->sensor = &ov7660; +	if (!sd->sensor->probe(sd)) +		return 0; +  	/* Try the s5k83a */  	sd->sensor = &s5k83a;  	if (!sd->sensor->probe(sd)) @@ -421,8 +426,9 @@ MODULE_DESCRIPTION(DRIVER_DESC);  MODULE_LICENSE("GPL");  module_param(force_sensor, int, S_IRUGO | S_IWUSR);  MODULE_PARM_DESC(force_sensor, -		"force detection of sensor, " -		"1 = OV9650, 2 = S5K83A, 3 = S5K4AA, 4 = MT9M111, 5 = PO1030"); +		"forces detection of a sensor, " +		"1 = OV9650, 2 = S5K83A, 3 = S5K4AA, " +		"4 = MT9M111, 5 = PO1030, 6 = OV7660");  module_param(dump_bridge, bool, S_IRUGO | S_IWUSR);  MODULE_PARM_DESC(dump_bridge, "Dumps all usb bridge registers at startup"); diff --git a/linux/drivers/media/video/gspca/m5602/m5602_mt9m111.c b/linux/drivers/media/video/gspca/m5602/m5602_mt9m111.c index 241108c78..a9780dfc7 100644 --- a/linux/drivers/media/video/gspca/m5602/m5602_mt9m111.c +++ b/linux/drivers/media/video/gspca/m5602/m5602_mt9m111.c @@ -36,6 +36,7 @@ static int mt9m111_get_red_balance(struct gspca_dev *gspca_dev, __s32 *val);  static int mt9m111_set_red_balance(struct gspca_dev *gspca_dev, __s32 val);  static struct v4l2_pix_format mt9m111_modes[] = { +#if 0  	{  		320,  		240, @@ -45,7 +46,9 @@ static struct v4l2_pix_format mt9m111_modes[] = {  		.bytesperline = 320,  		.colorspace = V4L2_COLORSPACE_SRGB,  		.priv = 0 -	}, { +	}, +#endif +	{  		640,  		480,  		V4L2_PIX_FMT_SBGGR8, @@ -67,7 +70,7 @@ const static struct ctrl mt9m111_ctrls[] = {  			.minimum        = 0,  			.maximum        = 1,  			.step           = 1, -			.default_value  = 1 +			.default_value  = 0  		},  		.set = mt9m111_set_vflip,  		.get = mt9m111_get_vflip @@ -81,7 +84,7 @@ const static struct ctrl mt9m111_ctrls[] = {  			.minimum        = 0,  			.maximum        = 1,  			.step           = 1, -			.default_value  = 1 +			.default_value  = 0  		},  		.set = mt9m111_set_hflip,  		.get = mt9m111_get_hflip @@ -95,7 +98,7 @@ const static struct ctrl mt9m111_ctrls[] = {  			.minimum        = 0,  			.maximum        = (INITIAL_MAX_GAIN - 1) * 2 * 2 * 2,  			.step           = 1, -			.default_value  = DEFAULT_GAIN, +			.default_value  = MT9M111_DEFAULT_GAIN,  			.flags          = V4L2_CTRL_FLAG_SLIDER  		},  		.set = mt9m111_set_gain, @@ -391,6 +394,9 @@ static int mt9m111_set_vflip(struct gspca_dev *gspca_dev, __s32 val)  	sensor_settings[VFLIP_IDX] = val; +	/* The mt9m111 is flipped by default */ +	val = !val; +  	/* Set the correct page map */  	err = m5602_write_sensor(sd, MT9M111_PAGE_MAP, data, 2);  	if (err < 0) @@ -427,6 +433,10 @@ static int mt9m111_set_hflip(struct gspca_dev *gspca_dev, __s32 val)  	PDEBUG(D_V4L2, "Set horizontal flip to %d", val);  	sensor_settings[HFLIP_IDX] = val; + +	/* The mt9m111 is flipped by default */ +	val = !val; +  	/* Set the correct page map */  	err = m5602_write_sensor(sd, MT9M111_PAGE_MAP, data, 2);  	if (err < 0) diff --git a/linux/drivers/media/video/gspca/m5602/m5602_mt9m111.h b/linux/drivers/media/video/gspca/m5602/m5602_mt9m111.h index 716aba523..e71ddb786 100644 --- a/linux/drivers/media/video/gspca/m5602/m5602_mt9m111.h +++ b/linux/drivers/media/video/gspca/m5602/m5602_mt9m111.h @@ -97,7 +97,7 @@  #define MT9M111_2D_DEFECT_CORRECTION_ENABLE	(1 << 0)  #define INITIAL_MAX_GAIN			64 -#define DEFAULT_GAIN 				283 +#define MT9M111_DEFAULT_GAIN 			283  #define MT9M111_GREEN_GAIN_DEFAULT		0x20  #define MT9M111_BLUE_GAIN_DEFAULT		0x20  #define MT9M111_RED_GAIN_DEFAULT		0x20 diff --git a/linux/drivers/media/video/gspca/m5602/m5602_ov7660.c b/linux/drivers/media/video/gspca/m5602/m5602_ov7660.c new file mode 100644 index 000000000..7aafeb7cf --- /dev/null +++ b/linux/drivers/media/video/gspca/m5602/m5602_ov7660.c @@ -0,0 +1,227 @@ +/* + * Driver for the ov7660 sensor + * + * Copyright (C) 2009 Erik Andrén + * Copyright (C) 2007 Ilyes Gouta. Based on the m5603x Linux Driver Project. + * Copyright (C) 2005 m5603x Linux Driver Project <m5602@x3ng.com.br> + * + * Portions of code to USB interface and ALi driver software, + * Copyright (c) 2006 Willem Duinker + * v4l2 interface modeled after the V4L2 driver + * for SN9C10x PC Camera Controllers + * + * 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. + * + */ + +#include "m5602_ov7660.h" + +static int ov7660_get_gain(struct gspca_dev *gspca_dev, __s32 *val); +static int ov7660_set_gain(struct gspca_dev *gspca_dev, __s32 val); + +const static struct ctrl ov7660_ctrls[] = { +#define GAIN_IDX 1 +	{ +		{ +			.id		= V4L2_CID_GAIN, +			.type		= V4L2_CTRL_TYPE_INTEGER, +			.name		= "gain", +			.minimum	= 0x00, +			.maximum	= 0xff, +			.step		= 0x1, +			.default_value	= OV7660_DEFAULT_GAIN, +			.flags		= V4L2_CTRL_FLAG_SLIDER +		}, +		.set = ov7660_set_gain, +		.get = ov7660_get_gain +	}, +}; + +static struct v4l2_pix_format ov7660_modes[] = { +	{ +		640, +		480, +		V4L2_PIX_FMT_SBGGR8, +		V4L2_FIELD_NONE, +		.sizeimage = +			640 * 480, +		.bytesperline = 640, +		.colorspace = V4L2_COLORSPACE_SRGB, +		.priv = 0 +	} +}; + +static void ov7660_dump_registers(struct sd *sd); + +int ov7660_probe(struct sd *sd) +{ +	int err = 0, i; +	u8 prod_id = 0, ver_id = 0; + +	s32 *sensor_settings; + +	if (force_sensor) { +		if (force_sensor == OV7660_SENSOR) { +			info("Forcing an %s sensor", ov7660.name); +			goto sensor_found; +		} +		/* If we want to force another sensor, +		don't try to probe this one */ +		return -ENODEV; +	} + +	/* Do the preinit */ +	for (i = 0; i < ARRAY_SIZE(preinit_ov7660) && !err; i++) { +		u8 data[2]; + +		if (preinit_ov7660[i][0] == BRIDGE) { +			err = m5602_write_bridge(sd, +				preinit_ov7660[i][1], +				preinit_ov7660[i][2]); +		} else { +			data[0] = preinit_ov7660[i][2]; +			err = m5602_write_sensor(sd, +				preinit_ov7660[i][1], data, 1); +		} +	} +	if (err < 0) +		return err; + +	if (m5602_read_sensor(sd, OV7660_PID, &prod_id, 1)) +		return -ENODEV; + +	if (m5602_read_sensor(sd, OV7660_VER, &ver_id, 1)) +		return -ENODEV; + +	info("Sensor reported 0x%x%x", prod_id, ver_id); + +	if ((prod_id == 0x76) && (ver_id == 0x60)) { +		info("Detected a ov7660 sensor"); +		goto sensor_found; +	} +	return -ENODEV; + +sensor_found: +	sensor_settings = kmalloc( +		ARRAY_SIZE(ov7660_ctrls) * sizeof(s32), GFP_KERNEL); +	if (!sensor_settings) +		return -ENOMEM; + +	sd->gspca_dev.cam.cam_mode = ov7660_modes; +	sd->gspca_dev.cam.nmodes = ARRAY_SIZE(ov7660_modes); +	sd->desc->ctrls = ov7660_ctrls; +	sd->desc->nctrls = ARRAY_SIZE(ov7660_ctrls); + +	for (i = 0; i < ARRAY_SIZE(ov7660_ctrls); i++) +		sensor_settings[i] = ov7660_ctrls[i].qctrl.default_value; +	sd->sensor_priv = sensor_settings; + +	return 0; +} + +int ov7660_init(struct sd *sd) +{ +	int i, err = 0; +	s32 *sensor_settings = sd->sensor_priv; + +	/* Init the sensor */ +	for (i = 0; i < ARRAY_SIZE(init_ov7660); i++) { +		u8 data[2]; + +		if (init_ov7660[i][0] == BRIDGE) { +			err = m5602_write_bridge(sd, +				init_ov7660[i][1], +				init_ov7660[i][2]); +		} else { +			data[0] = init_ov7660[i][2]; +			err = m5602_write_sensor(sd, +					init_ov7660[i][1], data, 1); +		} +	} + +	if (dump_sensor) +		ov7660_dump_registers(sd); + +	err = ov7660_set_gain(&sd->gspca_dev, sensor_settings[GAIN_IDX]); +	if (err < 0) +		return err; + +	return err; +} + +int ov7660_start(struct sd *sd) +{ +	return 0; +} + +int ov7660_stop(struct sd *sd) +{ +	return 0; +} + +void ov7660_disconnect(struct sd *sd) +{ +	ov7660_stop(sd); + +	sd->sensor = NULL; +	kfree(sd->sensor_priv); +} + +static int ov7660_get_gain(struct gspca_dev *gspca_dev, __s32 *val) +{ +	struct sd *sd = (struct sd *) gspca_dev; +	s32 *sensor_settings = sd->sensor_priv; + +	*val = sensor_settings[GAIN_IDX]; +	PDEBUG(D_V4L2, "Read gain %d", *val); +	return 0; +} + +static int ov7660_set_gain(struct gspca_dev *gspca_dev, __s32 val) +{ +	int err; +	u8 i2c_data; +	struct sd *sd = (struct sd *) gspca_dev; +	s32 *sensor_settings = sd->sensor_priv; + +	PDEBUG(D_V4L2, "Setting gain to %d", val); + +	sensor_settings[GAIN_IDX] = val; + +	err = m5602_write_sensor(sd, OV7660_GAIN, &i2c_data, 1); +	return err; +} + +static void ov7660_dump_registers(struct sd *sd) +{ +	int address; +	info("Dumping the ov7660 register state"); +	for (address = 0; address < 0xa9; address++) { +		u8 value; +		m5602_read_sensor(sd, address, &value, 1); +		info("register 0x%x contains 0x%x", +		     address, value); +	} + +	info("ov7660 register state dump complete"); + +	info("Probing for which registers that are read/write"); +	for (address = 0; address < 0xff; address++) { +		u8 old_value, ctrl_value; +		u8 test_value[2] = {0xff, 0xff}; + +		m5602_read_sensor(sd, address, &old_value, 1); +		m5602_write_sensor(sd, address, test_value, 1); +		m5602_read_sensor(sd, address, &ctrl_value, 1); + +		if (ctrl_value == test_value[0]) +			info("register 0x%x is writeable", address); +		else +			info("register 0x%x is read only", address); + +		/* Restore original value */ +		m5602_write_sensor(sd, address, &old_value, 1); +	} +} diff --git a/linux/drivers/media/video/gspca/m5602/m5602_ov7660.h b/linux/drivers/media/video/gspca/m5602/m5602_ov7660.h new file mode 100644 index 000000000..3f2c169a9 --- /dev/null +++ b/linux/drivers/media/video/gspca/m5602/m5602_ov7660.h @@ -0,0 +1,279 @@ +/* + * Driver for the ov7660 sensor + * + * Copyright (C) 2009 Erik Andrén + * Copyright (C) 2007 Ilyes Gouta. Based on the m5603x Linux Driver Project. + * Copyright (C) 2005 m5603x Linux Driver Project <m5602@x3ng.com.br> + * + * Portions of code to USB interface and ALi driver software, + * Copyright (c) 2006 Willem Duinker + * v4l2 interface modeled after the V4L2 driver + * for SN9C10x PC Camera Controllers + * + * 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. + * + */ + +#ifndef M5602_OV7660_H_ +#define M5602_OV7660_H_ + +#include "m5602_sensor.h" + +#define OV7660_GAIN		0x00 +#define OV7660_BLUE_GAIN	0x01 +#define OV7660_RED_GAIN		0x02 +#define OV7660_VREF		0x03 +#define OV7660_COM1		0x04 +#define OV7660_BAVE		0x05 +#define OV7660_GEAVE		0x06 +#define OV7660_AECHH		0x07 +#define OV7660_RAVE		0x08 +#define OV7660_COM2		0x09 +#define OV7660_PID		0x0a +#define OV7660_VER		0x0b +#define OV7660_COM3		0x0c +#define OV7660_COM4		0x0d +#define OV7660_COM5		0x0e +#define OV7660_COM6		0x0f +#define OV7660_AECH		0x10 +#define OV7660_CLKRC		0x11 +#define OV7660_COM7		0x12 +#define OV7660_COM8		0x13 +#define OV7660_COM9		0x14 +#define OV7660_COM10		0x15 +#define OV7660_RSVD16		0x16 +#define OV7660_HSTART		0x17 +#define OV7660_HSTOP		0x18 +#define OV7660_VSTART		0x19 +#define OV7660_VSTOP		0x1a +#define OV7660_PSHFT		0x1b +#define OV7660_MIDH		0x1c +#define OV7660_MIDL		0x1d +#define OV7660_MVFP		0x1e +#define OV7660_LAEC		0x1f +#define OV7660_BOS		0x20 +#define OV7660_GBOS		0x21 +#define OV7660_GROS		0x22 +#define OV7660_ROS		0x23 +#define OV7660_AEW		0x24 +#define OV7660_AEB		0x25 +#define OV7660_VPT		0x26 +#define OV7660_BBIAS		0x27 +#define OV7660_GbBIAS		0x28 +#define OV7660_RSVD29		0x29 +#define OV7660_RBIAS		0x2c +#define OV7660_HREF		0x32 +#define OV7660_ADC		0x37 +#define OV7660_OFON 		0x39 +#define OV7660_TSLB 		0x3a +#define OV7660_COM12 		0x3c +#define OV7660_COM13 		0x3d +#define OV7660_LCC1		0x62 +#define OV7660_LCC2		0x63 +#define OV7660_LCC3		0x64 +#define OV7660_LCC4		0x65 +#define OV7660_LCC5		0x66 +#define OV7660_HV 		0x69 +#define OV7660_RSVDA1 		0xa1 + +#define OV7660_DEFAULT_GAIN		0x0e +#define OV7660_DEFAULT_RED_GAIN	0x80 +#define OV7660_DEFAULT_BLUE_GAIN 	0x80 +#define OV7660_DEFAULT_SATURATION	0x00 +#define OV7660_DEFAULT_EXPOSURE	0x20 + +/* Kernel module parameters */ +extern int force_sensor; +extern int dump_sensor; + +int ov7660_probe(struct sd *sd); +int ov7660_init(struct sd *sd); +int ov7660_start(struct sd *sd); +int ov7660_stop(struct sd *sd); +void ov7660_disconnect(struct sd *sd); + +const static struct m5602_sensor ov7660 = { +	.name = "ov7660", +	.i2c_slave_id = 0x42, +	.i2c_regW = 1, +	.probe = ov7660_probe, +	.init = ov7660_init, +	.start = ov7660_start, +	.stop = ov7660_stop, +	.disconnect = ov7660_disconnect, +}; + +static const unsigned char preinit_ov7660[][4] = +{ +	{BRIDGE, M5602_XB_MCU_CLK_DIV, 0x02}, +	{BRIDGE, M5602_XB_MCU_CLK_CTRL, 0xb0}, +	{BRIDGE, M5602_XB_SEN_CLK_DIV, 0x00}, +	{BRIDGE, M5602_XB_SEN_CLK_CTRL, 0xb0}, +	{BRIDGE, M5602_XB_ADC_CTRL, 0xc0}, +	{BRIDGE, M5602_XB_SENSOR_TYPE, 0x0d}, +	{BRIDGE, M5602_XB_SENSOR_CTRL, 0x00}, +	{BRIDGE, M5602_XB_GPIO_DIR, 0x03}, +	{BRIDGE, M5602_XB_GPIO_DIR, 0x03}, +	{BRIDGE, M5602_XB_ADC_CTRL, 0xc0}, +	{BRIDGE, M5602_XB_SENSOR_TYPE, 0x0c}, + +	{SENSOR, OV7660_OFON, 0x0c}, +	{SENSOR, OV7660_COM2, 0x11}, +	{SENSOR, OV7660_COM7, 0x05}, + +	{BRIDGE, M5602_XB_GPIO_DIR, 0x01}, +	{BRIDGE, M5602_XB_GPIO_DAT, 0x04}, +	{BRIDGE, M5602_XB_GPIO_EN_H, 0x06}, +	{BRIDGE, M5602_XB_GPIO_DIR_H, 0x06}, +	{BRIDGE, M5602_XB_GPIO_DAT_H, 0x00}, +	{BRIDGE, M5602_XB_SEN_CLK_DIV, 0x08}, +	{BRIDGE, M5602_XB_SEN_CLK_CTRL, 0xb0}, +	{BRIDGE, M5602_XB_SEN_CLK_DIV, 0x00}, +	{BRIDGE, M5602_XB_SEN_CLK_CTRL, 0xb0}, +	{BRIDGE, M5602_XB_ADC_CTRL, 0xc0}, +	{BRIDGE, M5602_XB_SENSOR_TYPE, 0x0c}, +	{BRIDGE, M5602_XB_GPIO_DIR, 0x05}, +	{BRIDGE, M5602_XB_GPIO_DAT, 0x00}, +	{BRIDGE, M5602_XB_GPIO_EN_H, 0x06}, +	{BRIDGE, M5602_XB_GPIO_EN_L, 0x00} +}; + +static const unsigned char init_ov7660[][4] = +{ +	{BRIDGE, M5602_XB_MCU_CLK_DIV, 0x02}, +	{BRIDGE, M5602_XB_MCU_CLK_CTRL, 0xb0}, +	{BRIDGE, M5602_XB_SEN_CLK_DIV, 0x00}, +	{BRIDGE, M5602_XB_SEN_CLK_CTRL, 0xb0}, +	{BRIDGE, M5602_XB_ADC_CTRL, 0xc0}, +	{BRIDGE, M5602_XB_SENSOR_TYPE, 0x0d}, +	{BRIDGE, M5602_XB_SENSOR_CTRL, 0x00}, +	{BRIDGE, M5602_XB_GPIO_DIR, 0x03}, +	{BRIDGE, M5602_XB_GPIO_DIR, 0x03}, +	{BRIDGE, M5602_XB_ADC_CTRL, 0xc0}, +	{BRIDGE, M5602_XB_SENSOR_TYPE, 0x0c}, + +	{SENSOR, OV7660_OFON, 0x0c}, +	{SENSOR, OV7660_COM2, 0x11}, +	{SENSOR, OV7660_COM7, 0x05}, + +	{BRIDGE, M5602_XB_GPIO_DIR, 0x01}, +	{BRIDGE, M5602_XB_GPIO_DAT, 0x04}, +	{BRIDGE, M5602_XB_GPIO_EN_H, 0x06}, +	{BRIDGE, M5602_XB_GPIO_DIR_H, 0x06}, +	{BRIDGE, M5602_XB_GPIO_DAT_H, 0x00}, +	{BRIDGE, M5602_XB_SEN_CLK_DIV, 0x08}, +	{BRIDGE, M5602_XB_SEN_CLK_CTRL, 0xb0}, +	{BRIDGE, M5602_XB_SEN_CLK_DIV, 0x00}, +	{BRIDGE, M5602_XB_SEN_CLK_CTRL, 0xb0}, +	{BRIDGE, M5602_XB_ADC_CTRL, 0xc0}, +	{BRIDGE, M5602_XB_SENSOR_TYPE, 0x0c}, +	{BRIDGE, M5602_XB_GPIO_DIR, 0x05}, +	{BRIDGE, M5602_XB_GPIO_DAT, 0x00}, +	{BRIDGE, M5602_XB_GPIO_EN_H, 0x06}, +	{BRIDGE, M5602_XB_GPIO_EN_L, 0x00}, + +	{BRIDGE, M5602_XB_SEN_CLK_DIV, 0x02}, +	{BRIDGE, M5602_XB_SEN_CLK_CTRL, 0xb0}, + +	{SENSOR, OV7660_AECH, OV7660_DEFAULT_EXPOSURE}, +	{SENSOR, OV7660_COM1, 0x00}, + +	{BRIDGE, M5602_XB_GPIO_DIR, 0x01}, +	{BRIDGE, M5602_XB_GPIO_DAT, 0x04}, +	{BRIDGE, M5602_XB_GPIO_EN_H, 0x06}, +	{BRIDGE, M5602_XB_GPIO_DIR_H, 0x06}, +	{BRIDGE, M5602_XB_GPIO_DAT_H, 0x00}, +	{BRIDGE, M5602_XB_SEN_CLK_DIV, 0x08}, +	{BRIDGE, M5602_XB_SEN_CLK_CTRL, 0xb0}, + +	{BRIDGE, M5602_XB_SEN_CLK_DIV, 0x00}, +	{BRIDGE, M5602_XB_SEN_CLK_CTRL, 0xb0}, +	{BRIDGE, M5602_XB_ADC_CTRL, 0xc0}, +	{BRIDGE, M5602_XB_SENSOR_TYPE, 0x0c}, +	{BRIDGE, M5602_XB_GPIO_DIR, 0x05}, +	{BRIDGE, M5602_XB_GPIO_DAT, 0x00}, +	{BRIDGE, M5602_XB_GPIO_EN_H, 0x06}, +	{BRIDGE, M5602_XB_GPIO_EN_L, 0x00}, + +	{SENSOR, OV7660_COM7, 0x80}, +	{SENSOR, OV7660_CLKRC, 0x80}, +	{SENSOR, OV7660_BLUE_GAIN, 0x80}, +	{SENSOR, OV7660_RED_GAIN, 0x80}, +	{SENSOR, OV7660_COM9, 0x4c}, +	{SENSOR, OV7660_OFON, 0x43}, +	{SENSOR, OV7660_COM12, 0x28}, +	{SENSOR, OV7660_COM8, 0x00}, +	{SENSOR, OV7660_COM10, 0x40}, +	{SENSOR, OV7660_HSTART, 0x0c}, +	{SENSOR, OV7660_HSTOP, 0x61}, +	{SENSOR, OV7660_HREF, 0xa4}, +	{SENSOR, OV7660_PSHFT, 0x0b}, +	{SENSOR, OV7660_VSTART, 0x01}, +	{SENSOR, OV7660_VSTOP, 0x7a}, +	{SENSOR, OV7660_VREF, 0x00}, +	{SENSOR, OV7660_COM7, 0x05}, +	{SENSOR, OV7660_COM6, 0x4b}, +	{SENSOR, OV7660_BBIAS, 0x98}, +	{SENSOR, OV7660_GbBIAS, 0x98}, +	{SENSOR, OV7660_RSVD29, 0x98}, +	{SENSOR, OV7660_RBIAS, 0x98}, +	{SENSOR, OV7660_COM1, 0x00}, +	{SENSOR, OV7660_AECH, 0x00}, +	{SENSOR, OV7660_AECHH, 0x00}, +	{SENSOR, OV7660_ADC, 0x04}, +	{SENSOR, OV7660_COM13, 0x00}, +	{SENSOR, OV7660_RSVDA1, 0x23}, +	{SENSOR, OV7660_TSLB, 0x0d}, +	{SENSOR, OV7660_HV, 0x80}, +	{SENSOR, OV7660_LCC1, 0x00}, +	{SENSOR, OV7660_LCC2, 0x00}, +	{SENSOR, OV7660_LCC3, 0x10}, +	{SENSOR, OV7660_LCC4, 0x40}, +	{SENSOR, OV7660_LCC5, 0x01}, + +	{BRIDGE, M5602_XB_SEN_CLK_DIV, 0x06}, +	{BRIDGE, M5602_XB_SEN_CLK_CTRL, 0xb0}, +	{BRIDGE, M5602_XB_ADC_CTRL, 0xc0}, +	{BRIDGE, M5602_XB_SENSOR_TYPE, 0x0c}, +	{BRIDGE, M5602_XB_LINE_OF_FRAME_H, 0x81}, +	{BRIDGE, M5602_XB_PIX_OF_LINE_H, 0x82}, +	{BRIDGE, M5602_XB_SIG_INI, 0x01}, +	{BRIDGE, M5602_XB_VSYNC_PARA, 0x00}, +	{BRIDGE, M5602_XB_VSYNC_PARA, 0x08}, +	{BRIDGE, M5602_XB_VSYNC_PARA, 0x00}, +	{BRIDGE, M5602_XB_VSYNC_PARA, 0x00}, +	{BRIDGE, M5602_XB_VSYNC_PARA, 0x01}, +	{BRIDGE, M5602_XB_VSYNC_PARA, 0xe0}, /* 480 */ +	{BRIDGE, M5602_XB_VSYNC_PARA, 0x00}, +	{BRIDGE, M5602_XB_VSYNC_PARA, 0x00}, +	{BRIDGE, M5602_XB_SIG_INI, 0x00}, +	{BRIDGE, M5602_XB_SIG_INI, 0x02}, +	{BRIDGE, M5602_XB_VSYNC_PARA, 0x00}, +	{BRIDGE, M5602_XB_VSYNC_PARA, 0x27}, /* 39 */ +	{BRIDGE, M5602_XB_VSYNC_PARA, 0x02}, +	{BRIDGE, M5602_XB_VSYNC_PARA, 0xa7}, /* 679 */ +	{BRIDGE, M5602_XB_SIG_INI, 0x00}, + +	{BRIDGE, M5602_XB_SEN_CLK_DIV, 0x00}, +	{BRIDGE, M5602_XB_SEN_CLK_CTRL, 0xb0}, + +	{SENSOR, OV7660_AECH, 0x20}, +	{SENSOR, OV7660_COM1, 0x00}, +	{SENSOR, OV7660_OFON, 0x0c}, +	{SENSOR, OV7660_COM2, 0x11}, +	{SENSOR, OV7660_COM7, 0x05}, +	{SENSOR, OV7660_BLUE_GAIN, 0x80}, +	{SENSOR, OV7660_RED_GAIN, 0x80}, + +	{BRIDGE, M5602_XB_GPIO_DIR, 0x01}, +	{BRIDGE, M5602_XB_GPIO_DAT, 0x04}, +	{BRIDGE, M5602_XB_GPIO_EN_H, 0x06}, +	{BRIDGE, M5602_XB_GPIO_DIR_H, 0x06}, +	{BRIDGE, M5602_XB_GPIO_DAT_H, 0x00}, +	{BRIDGE, M5602_XB_SEN_CLK_DIV, 0x08}, +	{BRIDGE, M5602_XB_SEN_CLK_CTRL, 0xb0} +}; + +#endif diff --git a/linux/drivers/media/video/gspca/m5602/m5602_ov9650.c b/linux/drivers/media/video/gspca/m5602/m5602_ov9650.c index f0e335533..fcaf207c4 100644 --- a/linux/drivers/media/video/gspca/m5602/m5602_ov9650.c +++ b/linux/drivers/media/video/gspca/m5602/m5602_ov9650.c @@ -47,6 +47,14 @@ static  #endif  	struct dmi_system_id ov9650_flip_dmi_table[] = {  	{ +		.ident = "ASUS A6VA", +		.matches = { +			DMI_MATCH(DMI_SYS_VENDOR, "ASUSTeK Computer Inc."), +			DMI_MATCH(DMI_PRODUCT_NAME, "A6VA") +		} +	}, +	{ +  		.ident = "ASUS A6VC",  		.matches = {  			DMI_MATCH(DMI_SYS_VENDOR, "ASUSTeK Computer Inc."), diff --git a/linux/drivers/media/video/gspca/m5602/m5602_po1030.c b/linux/drivers/media/video/gspca/m5602/m5602_po1030.c index 840a3ca53..7706bd29f 100644 --- a/linux/drivers/media/video/gspca/m5602/m5602_po1030.c +++ b/linux/drivers/media/video/gspca/m5602/m5602_po1030.c @@ -42,6 +42,7 @@ static int po1030_get_auto_exposure(struct gspca_dev *gspca_dev,  					 __s32 *val);  static struct v4l2_pix_format po1030_modes[] = { +#if 0  	{  		320,  		240, @@ -51,7 +52,9 @@ static struct v4l2_pix_format po1030_modes[] = {  		.bytesperline = 320,  		.colorspace = V4L2_COLORSPACE_SRGB,  		.priv = 2 -	}, { +	}, +#endif +	{  		640,  		480,  		V4L2_PIX_FMT_SBGGR8, diff --git a/linux/drivers/media/video/gspca/m5602/m5602_s5k4aa.c b/linux/drivers/media/video/gspca/m5602/m5602_s5k4aa.c index 06da4b3ed..3ec22ebd2 100644 --- a/linux/drivers/media/video/gspca/m5602/m5602_s5k4aa.c +++ b/linux/drivers/media/video/gspca/m5602/m5602_s5k4aa.c @@ -420,18 +420,21 @@ static int s5k4aa_set_vflip(struct gspca_dev *gspca_dev, __s32 val)  	if (err < 0)  		return err; -	if (dmi_check_system(s5k4aa_vflip_dmi_table)) +	if (dmi_check_system(s5k4aa_vflip_dmi_table)) {  		val = !val; +		data = (data & 0x3f) | +		       (!sensor_settings[HFLIP_IDX] << 6) | +		       ((val & 0x01) << 7); +	} else { +		data = (data & 0x3f) | +		(sensor_settings[HFLIP_IDX] << 6) | +		((val & 0x01) << 7); +	} -	data = ((data & ~S5K4AA_RM_V_FLIP) -			| ((val & 0x01) << 7));  	err = m5602_write_sensor(sd, S5K4AA_READ_MODE, &data, 1);  	if (err < 0)  		return err; -	if (dmi_check_system(s5k4aa_vflip_dmi_table)) -		val = !val; -  	if (val) {  		err = m5602_read_sensor(sd, S5K4AA_ROWSTART_LO, &data, 1);  		if (err < 0) @@ -447,7 +450,6 @@ static int s5k4aa_set_vflip(struct gspca_dev *gspca_dev, __s32 val)  		data--;  		err = m5602_write_sensor(sd, S5K4AA_ROWSTART_LO, &data, 1);  	} -  	return err;  } @@ -483,6 +485,17 @@ static int s5k4aa_set_hflip(struct gspca_dev *gspca_dev, __s32 val)  	if (err < 0)  		return err; +	if (dmi_check_system(s5k4aa_vflip_dmi_table)) { +		val = !val; +		data = (data & 0x3f) | +		       (!sensor_settings[VFLIP_IDX] << 7) | +		       ((val & 0x01) << 6); +	} else { +		data = (data & 0x3f) | +		       (sensor_settings[VFLIP_IDX] << 7) | +		       ((val & 0x01) << 6); +	} +  	data = ((data & ~S5K4AA_RM_H_FLIP) | ((val & 0x01) << 6));  	err = m5602_write_sensor(sd, S5K4AA_READ_MODE, &data, 1);  	if (err < 0) @@ -503,7 +516,6 @@ static int s5k4aa_set_hflip(struct gspca_dev *gspca_dev, __s32 val)  		data--;  		err = m5602_write_sensor(sd, S5K4AA_COLSTART_LO, &data, 1);  	} -  	return err;  } diff --git a/linux/drivers/media/video/gspca/m5602/m5602_s5k4aa.h b/linux/drivers/media/video/gspca/m5602/m5602_s5k4aa.h index 2349174ad..c8d909a1f 100644 --- a/linux/drivers/media/video/gspca/m5602/m5602_s5k4aa.h +++ b/linux/drivers/media/video/gspca/m5602/m5602_s5k4aa.h @@ -175,79 +175,8 @@ static const unsigned char init_s5k4aa[][4] =  	{SENSOR, S5K4AA_PAGE_MAP, 0x02, 0x00},  	{SENSOR, 0x0c, 0x05, 0x00},  	{SENSOR, 0x02, 0x0e, 0x00}, -	{SENSOR, 0x11, 0x00, 0x00}, -	{SENSOR, 0x12, 0x00, 0x00}, -	{SENSOR, S5K4AA_PAGE_MAP, 0x02, 0x00},  	{SENSOR, S5K4AA_READ_MODE, 0xa0, 0x00},  	{SENSOR, 0x37, 0x00, 0x00}, -	{SENSOR, S5K4AA_ROWSTART_HI, 0x00, 0x00}, -	{SENSOR, S5K4AA_ROWSTART_LO, 0x2a, 0x00}, -	{SENSOR, S5K4AA_COLSTART_HI, 0x00, 0x00}, -	{SENSOR, S5K4AA_COLSTART_LO, 0x0b, 0x00}, -	{SENSOR, S5K4AA_WINDOW_HEIGHT_HI, 0x03, 0x00}, -	{SENSOR, S5K4AA_WINDOW_HEIGHT_LO, 0xc4, 0x00}, -	{SENSOR, S5K4AA_WINDOW_WIDTH_HI, 0x05, 0x00}, -	{SENSOR, S5K4AA_WINDOW_WIDTH_LO, 0x08, 0x00}, -	{SENSOR, S5K4AA_H_BLANK_HI__, 0x00, 0x00}, -	{SENSOR, S5K4AA_H_BLANK_LO__, 0x48, 0x00}, -	{SENSOR, S5K4AA_EXPOSURE_HI, 0x00, 0x00}, -	{SENSOR, S5K4AA_EXPOSURE_LO, 0x43, 0x00}, -	{SENSOR, 0x11, 0x04, 0x00}, -	{SENSOR, 0x12, 0xc3, 0x00}, -	{SENSOR, S5K4AA_PAGE_MAP, 0x02, 0x00}, - -	{BRIDGE, M5602_XB_SEN_CLK_DIV, 0x06, 0x00}, -	{BRIDGE, M5602_XB_SEN_CLK_CTRL, 0xb0, 0x00}, -	{BRIDGE, M5602_XB_ADC_CTRL, 0xc0, 0x00}, -	{BRIDGE, M5602_XB_SENSOR_TYPE, 0x08, 0x00}, -	{BRIDGE, M5602_XB_LINE_OF_FRAME_H, 0x81, 0x00}, -	{BRIDGE, M5602_XB_PIX_OF_LINE_H, 0x82, 0x00}, -	{BRIDGE, M5602_XB_SIG_INI, 0x01, 0x00}, -	{BRIDGE, M5602_XB_VSYNC_PARA, 0x00, 0x00}, -	{BRIDGE, M5602_XB_VSYNC_PARA, 0x00, 0x00}, -	{BRIDGE, M5602_XB_VSYNC_PARA, 0x00, 0x00}, -	{BRIDGE, M5602_XB_VSYNC_PARA, 0x00, 0x00}, -	/* VSYNC_PARA, VSYNC_PARA : img height 480 = 0x01e0 */ -	{BRIDGE, M5602_XB_VSYNC_PARA, 0x01, 0x00}, -	{BRIDGE, M5602_XB_VSYNC_PARA, 0xe0, 0x00}, -	{BRIDGE, M5602_XB_VSYNC_PARA, 0x00, 0x00}, -	{BRIDGE, M5602_XB_VSYNC_PARA, 0x00, 0x00}, -	{BRIDGE, M5602_XB_SIG_INI, 0x00, 0x00}, -	{BRIDGE, M5602_XB_SIG_INI, 0x02, 0x00}, -	{BRIDGE, M5602_XB_HSYNC_PARA, 0x00, 0x00}, -	{BRIDGE, M5602_XB_HSYNC_PARA, 0x00, 0x00}, -	/* HSYNC_PARA, HSYNC_PARA : img width 640 = 0x0280 */ -	{BRIDGE, M5602_XB_HSYNC_PARA, 0x02, 0x00}, -	{BRIDGE, M5602_XB_HSYNC_PARA, 0x80, 0x00}, -	{BRIDGE, M5602_XB_SIG_INI, 0x00, 0x00}, -	{BRIDGE, M5602_XB_SEN_CLK_DIV, 0x00, 0x00}, -	{BRIDGE, M5602_XB_SEN_CLK_CTRL, 0xa0, 0x00}, /* 48 MHz */ - -	{SENSOR, S5K4AA_PAGE_MAP, 0x02, 0x00}, -	{SENSOR, S5K4AA_READ_MODE, S5K4AA_RM_H_FLIP | S5K4AA_RM_ROW_SKIP_2X -		| S5K4AA_RM_COL_SKIP_2X, 0x00}, -	/* 0x37 : Fix image stability when light is too bright and improves -	 * image quality in 640x480, but worsens it in 1280x1024 */ -	{SENSOR, 0x37, 0x01, 0x00}, -	/* ROWSTART_HI, ROWSTART_LO : 10 + (1024-960)/2 = 42 = 0x002a */ -	{SENSOR, S5K4AA_ROWSTART_HI, 0x00, 0x00}, -	{SENSOR, S5K4AA_ROWSTART_LO, 0x2a, 0x00}, -	{SENSOR, S5K4AA_COLSTART_HI, 0x00, 0x00}, -	{SENSOR, S5K4AA_COLSTART_LO, 0x0c, 0x00}, -	/* window_height_hi, window_height_lo : 960 = 0x03c0 */ -	{SENSOR, S5K4AA_WINDOW_HEIGHT_HI, 0x03, 0x00}, -	{SENSOR, S5K4AA_WINDOW_HEIGHT_LO, 0xc0, 0x00}, -	/* window_width_hi, window_width_lo : 1280 = 0x0500 */ -	{SENSOR, S5K4AA_WINDOW_WIDTH_HI, 0x05, 0x00}, -	{SENSOR, S5K4AA_WINDOW_WIDTH_LO, 0x00, 0x00}, -	{SENSOR, S5K4AA_H_BLANK_HI__, 0x00, 0x00}, -	{SENSOR, S5K4AA_H_BLANK_LO__, 0xa8, 0x00}, /* helps to sync... */ -	{SENSOR, S5K4AA_EXPOSURE_HI, 0x01, 0x00}, -	{SENSOR, S5K4AA_EXPOSURE_LO, 0x00, 0x00}, -	{SENSOR, 0x11, 0x04, 0x00}, -	{SENSOR, 0x12, 0xc3, 0x00}, -	{SENSOR, S5K4AA_PAGE_MAP, 0x02, 0x00}, -	{SENSOR, 0x02, 0x0e, 0x00},  };  static const unsigned char VGA_s5k4aa[][4] = diff --git a/linux/drivers/media/video/gspca/m5602/m5602_s5k83a.c b/linux/drivers/media/video/gspca/m5602/m5602_s5k83a.c index e1529afd4..942030fd0 100644 --- a/linux/drivers/media/video/gspca/m5602/m5602_s5k83a.c +++ b/linux/drivers/media/video/gspca/m5602/m5602_s5k83a.c @@ -124,7 +124,8 @@ const static struct ctrl s5k83a_ctrls[] = {  static void s5k83a_dump_registers(struct sd *sd);  static int s5k83a_get_rotation(struct sd *sd, u8 *reg_data);  static int s5k83a_set_led_indication(struct sd *sd, u8 val); -int s5k83a_set_flip_real(struct gspca_dev *gspca_dev, __s32 vflip, __s32 hflip); +static int s5k83a_set_flip_real(struct gspca_dev *gspca_dev, +				__s32 vflip, __s32 hflip);  int s5k83a_probe(struct sd *sd)  { @@ -198,6 +199,8 @@ sensor_found:  int s5k83a_init(struct sd *sd)  {  	int i, err = 0; +	s32 *sensor_settings = +			((struct s5k83a_priv *) sd->sensor_priv)->settings;  	for (i = 0; i < ARRAY_SIZE(init_s5k83a) && !err; i++) {  		u8 data[2] = {0x00, 0x00}; @@ -230,7 +233,27 @@ int s5k83a_init(struct sd *sd)  	if (dump_sensor)  		s5k83a_dump_registers(sd); -	return (err < 0) ? err : 0; +	err = s5k83a_set_gain(&sd->gspca_dev, sensor_settings[GAIN_IDX]); +	if (err < 0) +		return err; + +	err = s5k83a_set_brightness(&sd->gspca_dev, +				     sensor_settings[BRIGHTNESS_IDX]); +	if (err < 0) +		return err; + +	err = s5k83a_set_exposure(&sd->gspca_dev, +				   sensor_settings[EXPOSURE_IDX]); +	if (err < 0) +		return err; + +	err = s5k83a_set_hflip(&sd->gspca_dev, sensor_settings[HFLIP_IDX]); +	if (err < 0) +		return err; + +	err = s5k83a_set_vflip(&sd->gspca_dev, sensor_settings[VFLIP_IDX]); + +	return err;  }  static int rotation_thread_function(void *data) @@ -277,14 +300,29 @@ static int rotation_thread_function(void *data)  int s5k83a_start(struct sd *sd)  { +	int i, err = 0;  	struct s5k83a_priv *sens_priv = sd->sensor_priv;  	/* Create another thread, polling the GPIO ports of the camera to check  	   if it got rotated. This is how the windows driver does it so we have  	   to assume that there is no better way of accomplishing this */ -	sens_priv->rotation_thread = kthread_create(rotation_thread_function, sd, "rotation thread"); +	sens_priv->rotation_thread = kthread_create(rotation_thread_function, +						    sd, "rotation thread");  	wake_up_process(sens_priv->rotation_thread); +	/* Preinit the sensor */ +	for (i = 0; i < ARRAY_SIZE(start_s5k83a) && !err; i++) { +		u8 data[2] = {start_s5k83a[i][2], start_s5k83a[i][3]}; +		if (start_s5k83a[i][0] == SENSOR) +			err = m5602_write_sensor(sd, start_s5k83a[i][1], +				data, 2); +		else +			err = m5602_write_bridge(sd, start_s5k83a[i][1], +				data[0]); +	} +	if (err < 0) +		return err; +  	return s5k83a_set_led_indication(sd, 1);  } @@ -402,7 +440,8 @@ static int s5k83a_get_vflip(struct gspca_dev *gspca_dev, __s32 *val)  	return 0;  } -int s5k83a_set_flip_real(struct gspca_dev *gspca_dev, __s32 vflip, __s32 hflip) +static int s5k83a_set_flip_real(struct gspca_dev *gspca_dev, +				__s32 vflip, __s32 hflip)  {  	int err;  	u8 data[1]; @@ -505,7 +544,7 @@ static int s5k83a_set_led_indication(struct sd *sd, u8 val)  	err = m5602_write_bridge(sd, M5602_XB_GPIO_DAT, data[0]); -	return (err < 0) ? err : 0; +	return err;  }  /* Get camera rotation on Acer notebooks */ diff --git a/linux/drivers/media/video/gspca/m5602/m5602_s5k83a.h b/linux/drivers/media/video/gspca/m5602/m5602_s5k83a.h index 0697f8a99..7814b078a 100644 --- a/linux/drivers/media/video/gspca/m5602/m5602_s5k83a.h +++ b/linux/drivers/media/video/gspca/m5602/m5602_s5k83a.h @@ -110,32 +110,8 @@ static const unsigned char preinit_s5k83a[][4] =  */  static const unsigned char init_s5k83a[][4] =  { -	{SENSOR, S5K83A_PAGE_MAP, 0x04, 0x00}, -	{SENSOR, 0xaf, 0x01, 0x00}, -	{SENSOR, S5K83A_PAGE_MAP, 0x00, 0x00}, -	{SENSOR, 0x7b, 0xff, 0x00}, -	{SENSOR, S5K83A_PAGE_MAP, 0x05, 0x00}, -	{SENSOR, 0x01, 0x50, 0x00}, -	{SENSOR, 0x12, 0x20, 0x00}, -	{SENSOR, 0x17, 0x40, 0x00}, -	{SENSOR, 0x1c, 0x00, 0x00}, -	{SENSOR, 0x02, 0x70, 0x00}, -	{SENSOR, 0x03, 0x0b, 0x00}, -	{SENSOR, 0x04, 0xf0, 0x00}, -	{SENSOR, 0x05, 0x0b, 0x00}, - -	{SENSOR, 0x06, 0x71, 0x00}, -	{SENSOR, 0x07, 0xe8, 0x00}, -	{SENSOR, 0x08, 0x02, 0x00}, -	{SENSOR, 0x09, 0x88, 0x00}, -	{SENSOR, 0x14, 0x00, 0x00}, -	{SENSOR, 0x15, 0x20, 0x00}, -	{SENSOR, 0x19, 0x00, 0x00}, -	{SENSOR, 0x1a, 0x98, 0x00}, -	{SENSOR, 0x0f, 0x02, 0x00}, -	{SENSOR, 0x10, 0xe5, 0x00}, - -	{BRIDGE, M5602_XB_SIG_INI, 0x00, 0x00}, +	/* The following sequence is useless after a clean boot +	   but is necessary after resume from suspend */  	{BRIDGE, M5602_XB_GPIO_DIR, 0x1d, 0x00},  	{BRIDGE, M5602_XB_GPIO_DAT, 0x08, 0x00},  	{BRIDGE, M5602_XB_GPIO_EN_H, 0x3f, 0x00}, @@ -155,7 +131,7 @@ static const unsigned char init_s5k83a[][4] =  	{BRIDGE, M5602_XB_SEN_CLK_DIV, 0x00, 0x00},  	{BRIDGE, M5602_XB_SEN_CLK_CTRL, 0xf0, 0x00},  	{BRIDGE, M5602_XB_GPIO_DIR, 0x1d, 0x00}, -	{BRIDGE, M5602_XB_GPIO_DAT, 0x1c, 0x00}, +	{BRIDGE, M5602_XB_GPIO_DAT, 0x08, 0x00},  	{BRIDGE, M5602_XB_GPIO_EN_H, 0x06, 0x00},  	{BRIDGE, M5602_XB_GPIO_DIR_H, 0x06, 0x00},  	{BRIDGE, M5602_XB_GPIO_DAT_H, 0x00, 0x00}, @@ -164,8 +140,7 @@ static const unsigned char init_s5k83a[][4] =  	{SENSOR, S5K83A_PAGE_MAP, 0x04, 0x00},  	{SENSOR, 0xaf, 0x01, 0x00}, -	{SENSOR, S5K83A_PAGE_MAP, 0x05, 0x00}, -	/* ff ( init value )is very dark) || 71 and f0 better */ +	{SENSOR, S5K83A_PAGE_MAP, 0x00, 0x00},  	{SENSOR, 0x7b, 0xff, 0x00},  	{SENSOR, S5K83A_PAGE_MAP, 0x05, 0x00},  	{SENSOR, 0x01, 0x50, 0x00}, @@ -173,66 +148,26 @@ static const unsigned char init_s5k83a[][4] =  	{SENSOR, 0x17, 0x40, 0x00},  	{SENSOR, 0x1c, 0x00, 0x00},  	{SENSOR, 0x02, 0x70, 0x00}, -	/* some values like 0x10 give a blue-purple image */  	{SENSOR, 0x03, 0x0b, 0x00},  	{SENSOR, 0x04, 0xf0, 0x00},  	{SENSOR, 0x05, 0x0b, 0x00}, -  	{SENSOR, 0x06, 0x71, 0x00}, -	{SENSOR, 0x07, 0xe8, 0x00}, +	{SENSOR, 0x07, 0xe8, 0x00}, /* 488 */  	{SENSOR, 0x08, 0x02, 0x00}, -	{SENSOR, 0x09, 0x88, 0x00}, +	{SENSOR, 0x09, 0x88, 0x00}, /* 648 */  	{SENSOR, 0x14, 0x00, 0x00}, -	{SENSOR, 0x15, 0x20, 0x00}, +	{SENSOR, 0x15, 0x20, 0x00}, /* 32 */  	{SENSOR, 0x19, 0x00, 0x00}, -	{SENSOR, 0x1a, 0x98, 0x00}, +	{SENSOR, 0x1a, 0x98, 0x00}, /* 152 */  	{SENSOR, 0x0f, 0x02, 0x00}, -	{SENSOR, 0x10, 0xe5, 0x00}, - -	/* The following sequence is useless after a clean boot -	   but is necessary after resume from suspend */ -	{BRIDGE, M5602_XB_GPIO_DIR, 0x1d, 0x00}, -	{BRIDGE, M5602_XB_GPIO_DAT, 0x08, 0x00}, -	{BRIDGE, M5602_XB_GPIO_EN_H, 0x3f, 0x00}, -	{BRIDGE, M5602_XB_GPIO_DIR_H, 0x3f, 0x00}, -	{BRIDGE, M5602_XB_GPIO_DAT_H, 0x00, 0x00}, -	{BRIDGE, M5602_XB_GPIO_EN_L, 0xff, 0x00}, -	{BRIDGE, M5602_XB_GPIO_DIR_L, 0xff, 0x00}, -	{BRIDGE, M5602_XB_GPIO_DAT_L, 0x00, 0x00}, -	{BRIDGE, M5602_XB_SEN_CLK_DIV, 0xb0, 0x00}, -	{BRIDGE, M5602_XB_SEN_CLK_CTRL, 0x80, 0x00}, -	{BRIDGE, M5602_XB_SEN_CLK_DIV, 0x00, 0x00}, -	{BRIDGE, M5602_XB_SEN_CLK_CTRL, 0xb0, 0x00}, -	{BRIDGE, M5602_XB_ADC_CTRL, 0xc0, 0x00}, -	{BRIDGE, M5602_XB_SENSOR_TYPE, 0x09, 0x00}, -	{BRIDGE, M5602_XB_MCU_CLK_DIV, 0x02, 0x00}, -	{BRIDGE, M5602_XB_MCU_CLK_CTRL, 0xb0, 0x00}, -	{BRIDGE, M5602_XB_SEN_CLK_DIV, 0x00, 0x00}, -	{BRIDGE, M5602_XB_SEN_CLK_CTRL, 0xf0, 0x00}, -	{BRIDGE, M5602_XB_GPIO_DIR, 0x1d, 0x00}, -	{BRIDGE, M5602_XB_GPIO_DAT, 0x08, 0x00}, -	{BRIDGE, M5602_XB_GPIO_EN_H, 0x06, 0x00}, -	{BRIDGE, M5602_XB_GPIO_DIR_H, 0x06, 0x00}, -	{BRIDGE, M5602_XB_GPIO_DAT_H, 0x00, 0x00}, -	{BRIDGE, M5602_XB_GPIO_EN_L, 0x00, 0x00}, -	{BRIDGE, M5602_XB_I2C_CLK_DIV, 0x20, 0x00}, - -	{SENSOR, S5K83A_PAGE_MAP, 0x04, 0x00}, -	{SENSOR, 0xaf, 0x01, 0x00}, -	{SENSOR, S5K83A_PAGE_MAP, 0x00, 0x00}, -	{SENSOR, 0x7b, 0xff, 0x00}, -	{SENSOR, S5K83A_PAGE_MAP, 0x05, 0x00}, -	{SENSOR, 0x01, 0x50, 0x00}, -	{SENSOR, 0x12, 0x20, 0x00}, -	{SENSOR, 0x17, 0x40, 0x00}, -	{SENSOR, S5K83A_GAIN, 0x0f, 0x00}, -	{SENSOR, 0x1c, 0x00, 0x00}, -	{SENSOR, 0x02, 0x70, 0x00}, -	{SENSOR, 0x03, 0x0b, 0x00}, -	{SENSOR, 0x04, 0xf0, 0x00}, -	{SENSOR, 0x05, 0x0b, 0x00}, -	{SENSOR, S5K83A_PAGE_MAP, 0x05, 0x00}, +	{SENSOR, 0x10, 0xe5, 0x00}, /* 741 */ +	/* normal colors +	(this is value after boot, but after tries can be different) */ +	{SENSOR, 0x00, 0x06, 0x00}, +}; +static const unsigned char start_s5k83a[][4] = +{  	{BRIDGE, M5602_XB_SEN_CLK_DIV, 0x06, 0x00},  	{BRIDGE, M5602_XB_SEN_CLK_CTRL, 0xb0, 0x00},  	{BRIDGE, M5602_XB_ADC_CTRL, 0xc0, 0x00}, @@ -257,22 +192,6 @@ static const unsigned char init_s5k83a[][4] =  	{BRIDGE, M5602_XB_SIG_INI, 0x00, 0x00},  	{BRIDGE, M5602_XB_SEN_CLK_DIV, 0x00, 0x00},  	{BRIDGE, M5602_XB_SEN_CLK_CTRL, 0xb0, 0x00}, - -	{SENSOR, 0x06, 0x71, 0x00}, -	{SENSOR, 0x07, 0xe8, 0x00}, -	{SENSOR, 0x08, 0x02, 0x00}, -	{SENSOR, 0x09, 0x88, 0x00}, -	{SENSOR, 0x14, 0x00, 0x00}, -	{SENSOR, 0x15, 0x20, 0x00}, -	{SENSOR, 0x19, 0x00, 0x00}, -	{SENSOR, 0x1a, 0x98, 0x00}, -	{SENSOR, 0x0f, 0x02, 0x00}, -	{SENSOR, 0x10, 0xe5, 0x00}, -	{SENSOR, S5K83A_PAGE_MAP, 0x05, 0x00}, - -	/* normal colors -	   (this is value after boot, but after tries can be different) */ -	{SENSOR, 0x00, 0x06, 0x00},  };  #endif diff --git a/linux/drivers/media/video/gspca/m5602/m5602_sensor.h b/linux/drivers/media/video/gspca/m5602/m5602_sensor.h index c3a72117b..edff4f1f5 100644 --- a/linux/drivers/media/video/gspca/m5602/m5602_sensor.h +++ b/linux/drivers/media/video/gspca/m5602/m5602_sensor.h @@ -30,7 +30,8 @@ enum sensors {  	S5K83A_SENSOR	= 2,  	S5K4AA_SENSOR	= 3,  	MT9M111_SENSOR	= 4, -	PO1030_SENSOR	= 5 +	PO1030_SENSOR	= 5, +	OV7660_SENSOR   = 6,  };  /* Enumerates all possible instruction types */ | 
