diff options
author | Theodore Kilgore <kilgota@auburn.edu> | 2009-03-25 06:13:13 -0400 |
---|---|---|
committer | Mauro Carvalho Chehab <mchehab@redhat.com> | 2009-03-30 11:43:40 -0400 |
commit | 14a19c0a2254ba58ed7559e072456ab94c9a2d3c (patch) | |
tree | 5aa420f6240fdbcc83b763cfcd1dd8ff1d528a95 /drivers/media/video | |
parent | 58e2ded892cf05023bf533936cadc9d264e8a20f (diff) |
V4L/DVB (11213): gspca - sq905c: New subdriver.
The code in the new sq905c.c is based upon the structure of the code in
gspca/sq905.c, and upon the code in libgphoto2/camlibs/digigr8, which supports
the same set of cameras in stillcam mode. I am a co-author of gspca/sq905.c and
I am the sole author of libgphoto2/camlibs/digigr8, which is licensed under the
LGPL. I hereby give myself permission to use my own code from libgphoto2 in
gspca/sq905c.c.
Signed-off-by: Theodore Kilgore <kilgota@auburn.edu>
Signed-off-by: Jean-Francois Moine <moinejf@free.fr>
Signed-off-by: Mauro Carvalho Chehab <mchehab@redhat.com>
Diffstat (limited to 'drivers/media/video')
-rw-r--r-- | drivers/media/video/gspca/Kconfig | 9 | ||||
-rw-r--r-- | drivers/media/video/gspca/Makefile | 2 | ||||
-rw-r--r-- | drivers/media/video/gspca/sq905c.c | 328 |
3 files changed, 339 insertions, 0 deletions
diff --git a/drivers/media/video/gspca/Kconfig b/drivers/media/video/gspca/Kconfig index a0f05ef5ca70..578dc4ffc965 100644 --- a/drivers/media/video/gspca/Kconfig +++ b/drivers/media/video/gspca/Kconfig | |||
@@ -185,6 +185,15 @@ config USB_GSPCA_SQ905 | |||
185 | To compile this driver as a module, choose M here: the | 185 | To compile this driver as a module, choose M here: the |
186 | module will be called gspca_sq905. | 186 | module will be called gspca_sq905. |
187 | 187 | ||
188 | config USB_GSPCA_SQ905C | ||
189 | tristate "SQ Technologies SQ905C based USB Camera Driver" | ||
190 | depends on VIDEO_V4L2 && USB_GSPCA | ||
191 | help | ||
192 | Say Y here if you want support for cameras based on the SQ905C chip. | ||
193 | |||
194 | To compile this driver as a module, choose M here: the | ||
195 | module will be called gspca_sq905c. | ||
196 | |||
188 | config USB_GSPCA_STK014 | 197 | config USB_GSPCA_STK014 |
189 | tristate "Syntek DV4000 (STK014) USB Camera Driver" | 198 | tristate "Syntek DV4000 (STK014) USB Camera Driver" |
190 | depends on VIDEO_V4L2 && USB_GSPCA | 199 | depends on VIDEO_V4L2 && USB_GSPCA |
diff --git a/drivers/media/video/gspca/Makefile b/drivers/media/video/gspca/Makefile index b6ec61185736..8a6643e8eb96 100644 --- a/drivers/media/video/gspca/Makefile +++ b/drivers/media/video/gspca/Makefile | |||
@@ -17,6 +17,7 @@ obj-$(CONFIG_USB_GSPCA_SPCA506) += gspca_spca506.o | |||
17 | obj-$(CONFIG_USB_GSPCA_SPCA508) += gspca_spca508.o | 17 | obj-$(CONFIG_USB_GSPCA_SPCA508) += gspca_spca508.o |
18 | obj-$(CONFIG_USB_GSPCA_SPCA561) += gspca_spca561.o | 18 | obj-$(CONFIG_USB_GSPCA_SPCA561) += gspca_spca561.o |
19 | obj-$(CONFIG_USB_GSPCA_SQ905) += gspca_sq905.o | 19 | obj-$(CONFIG_USB_GSPCA_SQ905) += gspca_sq905.o |
20 | obj-$(CONFIG_USB_GSPCA_SQ905C) += gspca_sq905c.o | ||
20 | obj-$(CONFIG_USB_GSPCA_SUNPLUS) += gspca_sunplus.o | 21 | obj-$(CONFIG_USB_GSPCA_SUNPLUS) += gspca_sunplus.o |
21 | obj-$(CONFIG_USB_GSPCA_STK014) += gspca_stk014.o | 22 | obj-$(CONFIG_USB_GSPCA_STK014) += gspca_stk014.o |
22 | obj-$(CONFIG_USB_GSPCA_T613) += gspca_t613.o | 23 | obj-$(CONFIG_USB_GSPCA_T613) += gspca_t613.o |
@@ -43,6 +44,7 @@ gspca_spca506-objs := spca506.o | |||
43 | gspca_spca508-objs := spca508.o | 44 | gspca_spca508-objs := spca508.o |
44 | gspca_spca561-objs := spca561.o | 45 | gspca_spca561-objs := spca561.o |
45 | gspca_sq905-objs := sq905.o | 46 | gspca_sq905-objs := sq905.o |
47 | gspca_sq905c-objs := sq905c.o | ||
46 | gspca_stk014-objs := stk014.o | 48 | gspca_stk014-objs := stk014.o |
47 | gspca_sunplus-objs := sunplus.o | 49 | gspca_sunplus-objs := sunplus.o |
48 | gspca_t613-objs := t613.o | 50 | gspca_t613-objs := t613.o |
diff --git a/drivers/media/video/gspca/sq905c.c b/drivers/media/video/gspca/sq905c.c new file mode 100644 index 000000000000..0bcb74a1b143 --- /dev/null +++ b/drivers/media/video/gspca/sq905c.c | |||
@@ -0,0 +1,328 @@ | |||
1 | /* | ||
2 | * SQ905C subdriver | ||
3 | * | ||
4 | * Copyright (C) 2009 Theodore Kilgore | ||
5 | * | ||
6 | * This program is free software; you can redistribute it and/or modify | ||
7 | * it under the terms of the GNU General Public License as published by | ||
8 | * the Free Software Foundation; either version 2 of the License, or | ||
9 | * any later version. | ||
10 | * | ||
11 | * This program is distributed in the hope that it will be useful, | ||
12 | * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||
13 | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | ||
14 | * GNU General Public License for more details. | ||
15 | * | ||
16 | * You should have received a copy of the GNU General Public License | ||
17 | * along with this program; if not, write to the Free Software | ||
18 | * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA | ||
19 | */ | ||
20 | |||
21 | /* | ||
22 | * | ||
23 | * This driver uses work done in | ||
24 | * libgphoto2/camlibs/digigr8, Copyright (C) Theodore Kilgore. | ||
25 | * | ||
26 | * This driver has also used as a base the sq905c driver | ||
27 | * and may contain code fragments from it. | ||
28 | */ | ||
29 | |||
30 | #define MODULE_NAME "sq905c" | ||
31 | |||
32 | #include <linux/workqueue.h> | ||
33 | #include "gspca.h" | ||
34 | |||
35 | MODULE_AUTHOR("Theodore Kilgore <kilgota@auburn.edu>"); | ||
36 | MODULE_DESCRIPTION("GSPCA/SQ905C USB Camera Driver"); | ||
37 | MODULE_LICENSE("GPL"); | ||
38 | |||
39 | /* Default timeouts, in ms */ | ||
40 | #define SQ905C_CMD_TIMEOUT 500 | ||
41 | #define SQ905C_DATA_TIMEOUT 1000 | ||
42 | |||
43 | /* Maximum transfer size to use. */ | ||
44 | #define SQ905C_MAX_TRANSFER 0x8000 | ||
45 | |||
46 | #define FRAME_HEADER_LEN 0x50 | ||
47 | |||
48 | /* Commands. These go in the "value" slot. */ | ||
49 | #define SQ905C_CLEAR 0xa0 /* clear everything */ | ||
50 | #define SQ905C_CAPTURE_LOW 0xa040 /* Starts capture at 160x120 */ | ||
51 | #define SQ905C_CAPTURE_MED 0x1440 /* Starts capture at 320x240 */ | ||
52 | #define SQ905C_CAPTURE_HI 0x2840 /* Starts capture at 320x240 */ | ||
53 | |||
54 | /* For capture, this must go in the "index" slot. */ | ||
55 | #define SQ905C_CAPTURE_INDEX 0x110f | ||
56 | |||
57 | /* Structure to hold all of our device specific stuff */ | ||
58 | struct sd { | ||
59 | struct gspca_dev gspca_dev; /* !! must be the first item */ | ||
60 | const struct v4l2_pix_format *cap_mode; | ||
61 | /* Driver stuff */ | ||
62 | struct work_struct work_struct; | ||
63 | struct workqueue_struct *work_thread; | ||
64 | }; | ||
65 | |||
66 | /* | ||
67 | * Most of these cameras will do 640x480 and 320x240. 160x120 works | ||
68 | * in theory but gives very poor output. Therefore, not supported. | ||
69 | * The 0x2770:0x9050 cameras have max resolution of 320x240. | ||
70 | */ | ||
71 | static struct v4l2_pix_format sq905c_mode[] = { | ||
72 | { 320, 240, V4L2_PIX_FMT_SQ905C, V4L2_FIELD_NONE, | ||
73 | .bytesperline = 320, | ||
74 | .sizeimage = 320 * 240, | ||
75 | .colorspace = V4L2_COLORSPACE_SRGB, | ||
76 | .priv = 0}, | ||
77 | { 640, 480, V4L2_PIX_FMT_SQ905C, V4L2_FIELD_NONE, | ||
78 | .bytesperline = 640, | ||
79 | .sizeimage = 640 * 480, | ||
80 | .colorspace = V4L2_COLORSPACE_SRGB, | ||
81 | .priv = 0} | ||
82 | }; | ||
83 | |||
84 | /* Send a command to the camera. */ | ||
85 | static int sq905c_command(struct gspca_dev *gspca_dev, u16 command, u16 index) | ||
86 | { | ||
87 | int ret; | ||
88 | |||
89 | ret = usb_control_msg(gspca_dev->dev, | ||
90 | usb_sndctrlpipe(gspca_dev->dev, 0), | ||
91 | USB_REQ_SYNCH_FRAME, /* request */ | ||
92 | USB_DIR_OUT | USB_TYPE_VENDOR | USB_RECIP_DEVICE, | ||
93 | command, index, NULL, 0, | ||
94 | SQ905C_CMD_TIMEOUT); | ||
95 | if (ret < 0) { | ||
96 | PDEBUG(D_ERR, "%s: usb_control_msg failed (%d)", | ||
97 | __func__, ret); | ||
98 | return ret; | ||
99 | } | ||
100 | |||
101 | return 0; | ||
102 | } | ||
103 | |||
104 | /* This function is called as a workqueue function and runs whenever the camera | ||
105 | * is streaming data. Because it is a workqueue function it is allowed to sleep | ||
106 | * so we can use synchronous USB calls. To avoid possible collisions with other | ||
107 | * threads attempting to use the camera's USB interface the gspca usb_lock is | ||
108 | * used when performing the one USB control operation inside the workqueue, | ||
109 | * which tells the camera to close the stream. In practice the only thing | ||
110 | * which needs to be protected against is the usb_set_interface call that | ||
111 | * gspca makes during stream_off. Otherwise the camera doesn't provide any | ||
112 | * controls that the user could try to change. | ||
113 | */ | ||
114 | static void sq905c_dostream(struct work_struct *work) | ||
115 | { | ||
116 | struct sd *dev = container_of(work, struct sd, work_struct); | ||
117 | struct gspca_dev *gspca_dev = &dev->gspca_dev; | ||
118 | struct gspca_frame *frame; | ||
119 | int bytes_left; /* bytes remaining in current frame. */ | ||
120 | int data_len; /* size to use for the next read. */ | ||
121 | int act_len; | ||
122 | int discarding = 0; /* true if we failed to get space for frame. */ | ||
123 | int packet_type; | ||
124 | int ret; | ||
125 | u8 *buffer; | ||
126 | |||
127 | buffer = kmalloc(SQ905C_MAX_TRANSFER, GFP_KERNEL | GFP_DMA); | ||
128 | if (!buffer) { | ||
129 | PDEBUG(D_ERR, "Couldn't allocate USB buffer"); | ||
130 | goto quit_stream; | ||
131 | } | ||
132 | |||
133 | while (gspca_dev->present && gspca_dev->streaming) { | ||
134 | if (!gspca_dev->present) | ||
135 | goto quit_stream; | ||
136 | /* Request the header, which tells the size to download */ | ||
137 | ret = usb_bulk_msg(gspca_dev->dev, | ||
138 | usb_rcvbulkpipe(gspca_dev->dev, 0x81), | ||
139 | buffer, FRAME_HEADER_LEN, &act_len, | ||
140 | SQ905C_DATA_TIMEOUT); | ||
141 | PDEBUG(D_STREAM, | ||
142 | "Got %d bytes out of %d for header", | ||
143 | act_len, FRAME_HEADER_LEN); | ||
144 | if (ret < 0 || act_len < FRAME_HEADER_LEN) | ||
145 | goto quit_stream; | ||
146 | /* size is read from 4 bytes starting 0x40, little endian */ | ||
147 | bytes_left = buffer[0x40]|(buffer[0x41]<<8)|(buffer[0x42]<<16) | ||
148 | |(buffer[0x43]<<24); | ||
149 | PDEBUG(D_STREAM, "bytes_left = 0x%x", bytes_left); | ||
150 | /* We keep the header. It has other information, too. */ | ||
151 | packet_type = FIRST_PACKET; | ||
152 | frame = gspca_get_i_frame(gspca_dev); | ||
153 | if (frame && !discarding) { | ||
154 | gspca_frame_add(gspca_dev, packet_type, | ||
155 | frame, buffer, FRAME_HEADER_LEN); | ||
156 | } else | ||
157 | discarding = 1; | ||
158 | while (bytes_left > 0) { | ||
159 | data_len = bytes_left > SQ905C_MAX_TRANSFER ? | ||
160 | SQ905C_MAX_TRANSFER : bytes_left; | ||
161 | if (!gspca_dev->present) | ||
162 | goto quit_stream; | ||
163 | ret = usb_bulk_msg(gspca_dev->dev, | ||
164 | usb_rcvbulkpipe(gspca_dev->dev, 0x81), | ||
165 | buffer, data_len, &act_len, | ||
166 | SQ905C_DATA_TIMEOUT); | ||
167 | if (ret < 0 || act_len < data_len) | ||
168 | goto quit_stream; | ||
169 | PDEBUG(D_STREAM, | ||
170 | "Got %d bytes out of %d for frame", | ||
171 | data_len, bytes_left); | ||
172 | bytes_left -= data_len; | ||
173 | if (bytes_left == 0) | ||
174 | packet_type = LAST_PACKET; | ||
175 | else | ||
176 | packet_type = INTER_PACKET; | ||
177 | frame = gspca_get_i_frame(gspca_dev); | ||
178 | if (frame && !discarding) | ||
179 | gspca_frame_add(gspca_dev, packet_type, | ||
180 | frame, buffer, data_len); | ||
181 | else | ||
182 | discarding = 1; | ||
183 | } | ||
184 | } | ||
185 | quit_stream: | ||
186 | mutex_lock(&gspca_dev->usb_lock); | ||
187 | if (gspca_dev->present) | ||
188 | sq905c_command(gspca_dev, SQ905C_CLEAR, 0); | ||
189 | mutex_unlock(&gspca_dev->usb_lock); | ||
190 | kfree(buffer); | ||
191 | } | ||
192 | |||
193 | /* This function is called at probe time just before sd_init */ | ||
194 | static int sd_config(struct gspca_dev *gspca_dev, | ||
195 | const struct usb_device_id *id) | ||
196 | { | ||
197 | struct cam *cam = &gspca_dev->cam; | ||
198 | struct sd *dev = (struct sd *) gspca_dev; | ||
199 | |||
200 | PDEBUG(D_PROBE, | ||
201 | "SQ9050 camera detected" | ||
202 | " (vid/pid 0x%04X:0x%04X)", id->idVendor, id->idProduct); | ||
203 | cam->cam_mode = sq905c_mode; | ||
204 | cam->nmodes = 2; | ||
205 | if (id->idProduct == 0x9050) | ||
206 | cam->nmodes = 1; | ||
207 | /* We don't use the buffer gspca allocates so make it small. */ | ||
208 | cam->bulk_size = 32; | ||
209 | INIT_WORK(&dev->work_struct, sq905c_dostream); | ||
210 | return 0; | ||
211 | } | ||
212 | |||
213 | /* called on streamoff with alt==0 and on disconnect */ | ||
214 | /* the usb_lock is held at entry - restore on exit */ | ||
215 | static void sd_stop0(struct gspca_dev *gspca_dev) | ||
216 | { | ||
217 | struct sd *dev = (struct sd *) gspca_dev; | ||
218 | |||
219 | /* wait for the work queue to terminate */ | ||
220 | mutex_unlock(&gspca_dev->usb_lock); | ||
221 | /* This waits for sq905c_dostream to finish */ | ||
222 | destroy_workqueue(dev->work_thread); | ||
223 | dev->work_thread = NULL; | ||
224 | mutex_lock(&gspca_dev->usb_lock); | ||
225 | } | ||
226 | |||
227 | /* this function is called at probe and resume time */ | ||
228 | static int sd_init(struct gspca_dev *gspca_dev) | ||
229 | { | ||
230 | int ret; | ||
231 | |||
232 | /* connect to the camera and reset it. */ | ||
233 | ret = sq905c_command(gspca_dev, SQ905C_CLEAR, 0); | ||
234 | return ret; | ||
235 | } | ||
236 | |||
237 | /* Set up for getting frames. */ | ||
238 | static int sd_start(struct gspca_dev *gspca_dev) | ||
239 | { | ||
240 | struct sd *dev = (struct sd *) gspca_dev; | ||
241 | int ret; | ||
242 | |||
243 | dev->cap_mode = gspca_dev->cam.cam_mode; | ||
244 | /* "Open the shutter" and set size, to start capture */ | ||
245 | switch (gspca_dev->width) { | ||
246 | case 640: | ||
247 | PDEBUG(D_STREAM, "Start streaming at high resolution"); | ||
248 | dev->cap_mode++; | ||
249 | ret = sq905c_command(gspca_dev, SQ905C_CAPTURE_HI, | ||
250 | SQ905C_CAPTURE_INDEX); | ||
251 | break; | ||
252 | default: /* 320 */ | ||
253 | PDEBUG(D_STREAM, "Start streaming at medium resolution"); | ||
254 | ret = sq905c_command(gspca_dev, SQ905C_CAPTURE_MED, | ||
255 | SQ905C_CAPTURE_INDEX); | ||
256 | } | ||
257 | |||
258 | if (ret < 0) { | ||
259 | PDEBUG(D_ERR, "Start streaming command failed"); | ||
260 | return ret; | ||
261 | } | ||
262 | /* Start the workqueue function to do the streaming */ | ||
263 | dev->work_thread = create_singlethread_workqueue(MODULE_NAME); | ||
264 | queue_work(dev->work_thread, &dev->work_struct); | ||
265 | |||
266 | return 0; | ||
267 | } | ||
268 | |||
269 | /* Table of supported USB devices */ | ||
270 | static const __devinitdata struct usb_device_id device_table[] = { | ||
271 | {USB_DEVICE(0x2770, 0x905c)}, | ||
272 | {USB_DEVICE(0x2770, 0x9050)}, | ||
273 | {USB_DEVICE(0x2770, 0x913d)}, | ||
274 | {} | ||
275 | }; | ||
276 | |||
277 | MODULE_DEVICE_TABLE(usb, device_table); | ||
278 | |||
279 | /* sub-driver description */ | ||
280 | static const struct sd_desc sd_desc = { | ||
281 | .name = MODULE_NAME, | ||
282 | .config = sd_config, | ||
283 | .init = sd_init, | ||
284 | .start = sd_start, | ||
285 | .stop0 = sd_stop0, | ||
286 | }; | ||
287 | |||
288 | /* -- device connect -- */ | ||
289 | static int sd_probe(struct usb_interface *intf, | ||
290 | const struct usb_device_id *id) | ||
291 | { | ||
292 | return gspca_dev_probe(intf, id, | ||
293 | &sd_desc, | ||
294 | sizeof(struct sd), | ||
295 | THIS_MODULE); | ||
296 | } | ||
297 | |||
298 | static struct usb_driver sd_driver = { | ||
299 | .name = MODULE_NAME, | ||
300 | .id_table = device_table, | ||
301 | .probe = sd_probe, | ||
302 | .disconnect = gspca_disconnect, | ||
303 | #ifdef CONFIG_PM | ||
304 | .suspend = gspca_suspend, | ||
305 | .resume = gspca_resume, | ||
306 | #endif | ||
307 | }; | ||
308 | |||
309 | /* -- module insert / remove -- */ | ||
310 | static int __init sd_mod_init(void) | ||
311 | { | ||
312 | int ret; | ||
313 | |||
314 | ret = usb_register(&sd_driver); | ||
315 | if (ret < 0) | ||
316 | return ret; | ||
317 | PDEBUG(D_PROBE, "registered"); | ||
318 | return 0; | ||
319 | } | ||
320 | |||
321 | static void __exit sd_mod_exit(void) | ||
322 | { | ||
323 | usb_deregister(&sd_driver); | ||
324 | PDEBUG(D_PROBE, "deregistered"); | ||
325 | } | ||
326 | |||
327 | module_init(sd_mod_init); | ||
328 | module_exit(sd_mod_exit); | ||