xref: /openbmc/linux/drivers/media/usb/gspca/stv06xx/stv06xx_vv6410.c (revision 4ed91d48259d9ddd378424d008f2e6559f7e78f8)
1 /*
2  * Copyright (c) 2001 Jean-Fredric Clere, Nikolas Zimmermann, Georg Acher
3  *		      Mark Cave-Ayland, Carlo E Prelz, Dick Streefland
4  * Copyright (c) 2002, 2003 Tuukka Toivonen
5  * Copyright (c) 2008 Erik Andrén
6  *
7  * This program is free software; you can redistribute it and/or modify
8  * it under the terms of the GNU General Public License as published by
9  * the Free Software Foundation; either version 2 of the License, or
10  * (at your option) any later version.
11  *
12  * This program is distributed in the hope that it will be useful,
13  * but WITHOUT ANY WARRANTY; without even the implied warranty of
14  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
15  * GNU General Public License for more details.
16  *
17  * P/N 861037:      Sensor HDCS1000        ASIC STV0600
18  * P/N 861050-0010: Sensor HDCS1000        ASIC STV0600
19  * P/N 861050-0020: Sensor Photobit PB100  ASIC STV0600-1 - QuickCam Express
20  * P/N 861055:      Sensor ST VV6410       ASIC STV0610   - LEGO cam
21  * P/N 861075-0040: Sensor HDCS1000        ASIC
22  * P/N 961179-0700: Sensor ST VV6410       ASIC STV0602   - Dexxa WebCam USB
23  * P/N 861040-0000: Sensor ST VV6410       ASIC STV0610   - QuickCam Web
24  */
25 
26 #define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
27 
28 #include "stv06xx_vv6410.h"
29 
30 static struct v4l2_pix_format vv6410_mode[] = {
31 	{
32 		356,
33 		292,
34 		V4L2_PIX_FMT_SGRBG8,
35 		V4L2_FIELD_NONE,
36 		.sizeimage = 356 * 292,
37 		.bytesperline = 356,
38 		.colorspace = V4L2_COLORSPACE_SRGB,
39 		.priv = 0
40 	}
41 };
42 
43 static int vv6410_s_ctrl(struct v4l2_ctrl *ctrl)
44 {
45 	struct gspca_dev *gspca_dev =
46 		container_of(ctrl->handler, struct gspca_dev, ctrl_handler);
47 	int err = -EINVAL;
48 
49 	switch (ctrl->id) {
50 	case V4L2_CID_HFLIP:
51 		if (!gspca_dev->streaming)
52 			return 0;
53 		err = vv6410_set_hflip(gspca_dev, ctrl->val);
54 		break;
55 	case V4L2_CID_VFLIP:
56 		if (!gspca_dev->streaming)
57 			return 0;
58 		err = vv6410_set_vflip(gspca_dev, ctrl->val);
59 		break;
60 	case V4L2_CID_GAIN:
61 		err = vv6410_set_analog_gain(gspca_dev, ctrl->val);
62 		break;
63 	case V4L2_CID_EXPOSURE:
64 		err = vv6410_set_exposure(gspca_dev, ctrl->val);
65 		break;
66 	}
67 	return err;
68 }
69 
70 static const struct v4l2_ctrl_ops vv6410_ctrl_ops = {
71 	.s_ctrl = vv6410_s_ctrl,
72 };
73 
74 static int vv6410_probe(struct sd *sd)
75 {
76 	u16 data;
77 	int err;
78 
79 	err = stv06xx_read_sensor(sd, VV6410_DEVICEH, &data);
80 	if (err < 0)
81 		return -ENODEV;
82 
83 	if (data != 0x19)
84 		return -ENODEV;
85 
86 	pr_info("vv6410 sensor detected\n");
87 
88 	sd->gspca_dev.cam.cam_mode = vv6410_mode;
89 	sd->gspca_dev.cam.nmodes = ARRAY_SIZE(vv6410_mode);
90 	return 0;
91 }
92 
93 static int vv6410_init_controls(struct sd *sd)
94 {
95 	struct v4l2_ctrl_handler *hdl = &sd->gspca_dev.ctrl_handler;
96 
97 	v4l2_ctrl_handler_init(hdl, 2);
98 	/* Disable the hardware VFLIP and HFLIP as we currently lack a
99 	   mechanism to adjust the image offset in such a way that
100 	   we don't need to renegotiate the announced format */
101 	/* v4l2_ctrl_new_std(hdl, &vv6410_ctrl_ops, */
102 	/*		V4L2_CID_HFLIP, 0, 1, 1, 0); */
103 	/* v4l2_ctrl_new_std(hdl, &vv6410_ctrl_ops, */
104 	/*		V4L2_CID_VFLIP, 0, 1, 1, 0); */
105 	v4l2_ctrl_new_std(hdl, &vv6410_ctrl_ops,
106 			V4L2_CID_EXPOSURE, 0, 32768, 1, 20000);
107 	v4l2_ctrl_new_std(hdl, &vv6410_ctrl_ops,
108 			V4L2_CID_GAIN, 0, 15, 1, 10);
109 	return hdl->error;
110 }
111 
112 static int vv6410_init(struct sd *sd)
113 {
114 	int err = 0, i;
115 
116 	for (i = 0; i < ARRAY_SIZE(stv_bridge_init); i++)
117 		stv06xx_write_bridge(sd, stv_bridge_init[i].addr, stv_bridge_init[i].data);
118 
119 	err = stv06xx_write_sensor_bytes(sd, (u8 *) vv6410_sensor_init,
120 					 ARRAY_SIZE(vv6410_sensor_init));
121 	return (err < 0) ? err : 0;
122 }
123 
124 static int vv6410_start(struct sd *sd)
125 {
126 	int err;
127 	struct gspca_dev *gspca_dev = (struct gspca_dev *)sd;
128 	struct cam *cam = &sd->gspca_dev.cam;
129 	u32 priv = cam->cam_mode[sd->gspca_dev.curr_mode].priv;
130 
131 	if (priv & VV6410_SUBSAMPLE) {
132 		PDEBUG(D_CONF, "Enabling subsampling");
133 		stv06xx_write_bridge(sd, STV_Y_CTRL, 0x02);
134 		stv06xx_write_bridge(sd, STV_X_CTRL, 0x06);
135 
136 		stv06xx_write_bridge(sd, STV_SCAN_RATE, 0x10);
137 	} else {
138 		stv06xx_write_bridge(sd, STV_Y_CTRL, 0x01);
139 		stv06xx_write_bridge(sd, STV_X_CTRL, 0x0a);
140 		stv06xx_write_bridge(sd, STV_SCAN_RATE, 0x00);
141 
142 	}
143 
144 	/* Turn on LED */
145 	err = stv06xx_write_bridge(sd, STV_LED_CTRL, LED_ON);
146 	if (err < 0)
147 		return err;
148 
149 	err = stv06xx_write_sensor(sd, VV6410_SETUP0, 0);
150 	if (err < 0)
151 		return err;
152 
153 	PDEBUG(D_STREAM, "Starting stream");
154 
155 	return 0;
156 }
157 
158 static int vv6410_stop(struct sd *sd)
159 {
160 	struct gspca_dev *gspca_dev = (struct gspca_dev *)sd;
161 	int err;
162 
163 	/* Turn off LED */
164 	err = stv06xx_write_bridge(sd, STV_LED_CTRL, LED_OFF);
165 	if (err < 0)
166 		return err;
167 
168 	err = stv06xx_write_sensor(sd, VV6410_SETUP0, VV6410_LOW_POWER_MODE);
169 	if (err < 0)
170 		return err;
171 
172 	PDEBUG(D_STREAM, "Halting stream");
173 
174 	return 0;
175 }
176 
177 static int vv6410_dump(struct sd *sd)
178 {
179 	u8 i;
180 	int err = 0;
181 
182 	pr_info("Dumping all vv6410 sensor registers\n");
183 	for (i = 0; i < 0xff && !err; i++) {
184 		u16 data;
185 		err = stv06xx_read_sensor(sd, i, &data);
186 		pr_info("Register 0x%x contained 0x%x\n", i, data);
187 	}
188 	return (err < 0) ? err : 0;
189 }
190 
191 static int vv6410_set_hflip(struct gspca_dev *gspca_dev, __s32 val)
192 {
193 	int err;
194 	u16 i2c_data;
195 	struct sd *sd = (struct sd *) gspca_dev;
196 
197 	err = stv06xx_read_sensor(sd, VV6410_DATAFORMAT, &i2c_data);
198 	if (err < 0)
199 		return err;
200 
201 	if (val)
202 		i2c_data |= VV6410_HFLIP;
203 	else
204 		i2c_data &= ~VV6410_HFLIP;
205 
206 	PDEBUG(D_CONF, "Set horizontal flip to %d", val);
207 	err = stv06xx_write_sensor(sd, VV6410_DATAFORMAT, i2c_data);
208 
209 	return (err < 0) ? err : 0;
210 }
211 
212 static int vv6410_set_vflip(struct gspca_dev *gspca_dev, __s32 val)
213 {
214 	int err;
215 	u16 i2c_data;
216 	struct sd *sd = (struct sd *) gspca_dev;
217 
218 	err = stv06xx_read_sensor(sd, VV6410_DATAFORMAT, &i2c_data);
219 	if (err < 0)
220 		return err;
221 
222 	if (val)
223 		i2c_data |= VV6410_VFLIP;
224 	else
225 		i2c_data &= ~VV6410_VFLIP;
226 
227 	PDEBUG(D_CONF, "Set vertical flip to %d", val);
228 	err = stv06xx_write_sensor(sd, VV6410_DATAFORMAT, i2c_data);
229 
230 	return (err < 0) ? err : 0;
231 }
232 
233 static int vv6410_set_analog_gain(struct gspca_dev *gspca_dev, __s32 val)
234 {
235 	int err;
236 	struct sd *sd = (struct sd *) gspca_dev;
237 
238 	PDEBUG(D_CONF, "Set analog gain to %d", val);
239 	err = stv06xx_write_sensor(sd, VV6410_ANALOGGAIN, 0xf0 | (val & 0xf));
240 
241 	return (err < 0) ? err : 0;
242 }
243 
244 static int vv6410_set_exposure(struct gspca_dev *gspca_dev, __s32 val)
245 {
246 	int err;
247 	struct sd *sd = (struct sd *) gspca_dev;
248 	unsigned int fine, coarse;
249 
250 	val = (val * val >> 14) + val / 4;
251 
252 	fine = val % VV6410_CIF_LINELENGTH;
253 	coarse = min(512, val / VV6410_CIF_LINELENGTH);
254 
255 	PDEBUG(D_CONF, "Set coarse exposure to %d, fine exposure to %d",
256 	       coarse, fine);
257 
258 	err = stv06xx_write_sensor(sd, VV6410_FINEH, fine >> 8);
259 	if (err < 0)
260 		goto out;
261 
262 	err = stv06xx_write_sensor(sd, VV6410_FINEL, fine & 0xff);
263 	if (err < 0)
264 		goto out;
265 
266 	err = stv06xx_write_sensor(sd, VV6410_COARSEH, coarse >> 8);
267 	if (err < 0)
268 		goto out;
269 
270 	err = stv06xx_write_sensor(sd, VV6410_COARSEL, coarse & 0xff);
271 
272 out:
273 	return err;
274 }
275