1 diff -urN -X /home/spse/dontdiff linux-2.4.19-pre8/Documentation/Configure.help linux-2.4.19-pre8-konicawc/Documentation/Configure.help
2 --- linux-2.4.19-pre8/Documentation/Configure.help Sat May 11 01:02:03 2002
3 +++ linux-2.4.19-pre8-konicawc/Documentation/Configure.help Sat May 11 01:19:34 2002
4 @@ -13685,6 +13685,21 @@
5 The module will be called vicam.o. If you want to compile it as a
6 module, say M here and read <file:Documentation/modules.txt>.
10 + Say Y here if you want support for webcams based on a Konica
11 + chipset. This is known to work with the Intel YC76 webcam.
13 + This driver uses the Video For Linux API. You must enable
14 + (Y or M in config) Video For Linux (under Character Devices)
15 + to use this driver. Information on this API and pointers to
16 + "v4l" programs may be found on the WWW at
17 + <http://roadrunner.swansea.uk.linux.org/v4l.shtml>.
19 + This code is also available as a module ( = code which can be
20 + inserted in and removed from the running kernel whenever you want).
21 + The module will be called konicawc.o. If you want to compile it as
22 + a module, say M here and read <file:Documentation/modules.txt>.
24 Pegasus/Pegasus II based USB-Ethernet device support
26 diff -urN -X /home/spse/dontdiff linux-2.4.19-pre8/drivers/usb/Config.in linux-2.4.19-pre8-konicawc/drivers/usb/Config.in
27 --- linux-2.4.19-pre8/drivers/usb/Config.in Sat May 11 01:02:11 2002
28 +++ linux-2.4.19-pre8-konicawc/drivers/usb/Config.in Sat May 11 01:18:02 2002
30 dep_tristate ' USB 3com HomeConnect (aka vicam) support (EXPERIMENTAL)' CONFIG_USB_VICAM $CONFIG_USB $CONFIG_VIDEO_DEV $CONFIG_EXPERIMENTAL
31 dep_tristate ' D-Link USB FM radio support (EXPERIMENTAL)' CONFIG_USB_DSBR $CONFIG_USB $CONFIG_VIDEO_DEV $CONFIG_EXPERIMENTAL
32 dep_tristate ' DABUSB driver' CONFIG_USB_DABUSB $CONFIG_USB
33 + dep_tristate ' USB Konica Webcam support' CONFIG_USB_KONICAWC $CONFIG_USB $CONFIG_VIDEO_DEV
36 comment 'USB Network adaptors'
37 diff -urN -X /home/spse/dontdiff linux-2.4.19-pre8/drivers/usb/Makefile linux-2.4.19-pre8-konicawc/drivers/usb/Makefile
38 --- linux-2.4.19-pre8/drivers/usb/Makefile Sat May 11 01:02:11 2002
39 +++ linux-2.4.19-pre8-konicawc/drivers/usb/Makefile Sat May 11 01:36:40 2002
42 # Objects that export symbols.
44 -export-objs := hcd.o usb.o ov511.o pwc-uncompress.o
45 +export-objs := hcd.o usb.o ov511.o pwc-uncompress.o usbvideo.o
50 obj-$(CONFIG_USB_AUDIO) += audio.o
51 obj-$(CONFIG_USB_EMI26) += emi26.o
52 obj-$(CONFIG_USB_IBMCAM) += ibmcam.o usbvideo.o ultracam.o
53 +obj-$(CONFIG_USB_KONICAWC) += konicawc.o usbvideo.o
54 obj-$(CONFIG_USB_PWC) += pwc.o
55 obj-$(CONFIG_USB_DC2XX) += dc2xx.o
56 obj-$(CONFIG_USB_MDC800) += mdc800.o
57 diff -urN -X /home/spse/dontdiff linux-2.4.19-pre8/drivers/usb/ibmcam.c linux-2.4.19-pre8-konicawc/drivers/usb/ibmcam.c
58 --- linux-2.4.19-pre8/drivers/usb/ibmcam.c Sat May 11 01:02:11 2002
59 +++ linux-2.4.19-pre8-konicawc/drivers/usb/ibmcam.c Sat May 11 01:34:30 2002
60 @@ -3899,6 +3899,17 @@
65 +static struct usb_device_id id_table[] = {
66 + { USB_DEVICE_VER(IBMCAM_VENDOR_ID, IBMCAM_PRODUCT_ID, 0x0002, 0x0002) }, /* Model 1 */
67 + { USB_DEVICE_VER(IBMCAM_VENDOR_ID, IBMCAM_PRODUCT_ID, 0x030a, 0x030a) }, /* Model 2 */
68 + { USB_DEVICE_VER(IBMCAM_VENDOR_ID, IBMCAM_PRODUCT_ID, 0x0301, 0x0301) }, /* Model 3 */
69 + { USB_DEVICE_VER(IBMCAM_VENDOR_ID, NETCAM_PRODUCT_ID, 0x030a, 0x030a) }, /* Model 4 */
70 + { USB_DEVICE_VER(IBMCAM_VENDOR_ID, VEO_800C_PRODUCT_ID, 0x030a, 0x030a) }, /* Model 2 */
71 + { USB_DEVICE_VER(IBMCAM_VENDOR_ID, VEO_800D_PRODUCT_ID, 0x030a, 0x030a) }, /* Model 4 */
72 + { } /* Terminating entry */
87 static void __exit ibmcam_cleanup(void)
88 @@ -3934,15 +3946,6 @@
89 usbvideo_Deregister(&cams);
92 -static __devinitdata struct usb_device_id id_table[] = {
93 - { USB_DEVICE_VER(IBMCAM_VENDOR_ID, IBMCAM_PRODUCT_ID, 0x0002, 0x0002) }, /* Model 1 */
94 - { USB_DEVICE_VER(IBMCAM_VENDOR_ID, IBMCAM_PRODUCT_ID, 0x030a, 0x030a) }, /* Model 2 */
95 - { USB_DEVICE_VER(IBMCAM_VENDOR_ID, IBMCAM_PRODUCT_ID, 0x0301, 0x0301) }, /* Model 3 */
96 - { USB_DEVICE_VER(IBMCAM_VENDOR_ID, NETCAM_PRODUCT_ID, 0x030a, 0x030a) }, /* Model 4 */
97 - { USB_DEVICE_VER(IBMCAM_VENDOR_ID, VEO_800C_PRODUCT_ID, 0x030a, 0x030a) }, /* Model 2 */
98 - { USB_DEVICE_VER(IBMCAM_VENDOR_ID, VEO_800D_PRODUCT_ID, 0x030a, 0x030a) }, /* Model 4 */
99 - { } /* Terminating entry */
101 MODULE_DEVICE_TABLE(usb, id_table);
103 module_init(ibmcam_init);
104 diff -urN -X /home/spse/dontdiff linux-2.4.19-pre8/drivers/usb/konicawc.c linux-2.4.19-pre8-konicawc/drivers/usb/konicawc.c
105 --- linux-2.4.19-pre8/drivers/usb/konicawc.c Thu Jan 1 01:00:00 1970
106 +++ linux-2.4.19-pre8-konicawc/drivers/usb/konicawc.c Sat May 11 03:18:42 2002
109 + * konicawc.c - konica webcam driver
111 + * Author: Simon Evans <spse@secret.org.uk>
113 + * Copyright (C) 2002 Simon Evans
117 + * Driver for USB webcams based on Konica chipset. This
118 + * chipset is used in Intel YC76 camera.
122 +#include <linux/kernel.h>
123 +#include <linux/module.h>
124 +#include <linux/init.h>
128 +#include "usbvideo.h"
130 +#define MAX_BRIGHTNESS 108
131 +#define MAX_CONTRAST 108
132 +#define MAX_SATURATION 108
133 +#define MAX_SHARPNESS 108
134 +#define MAX_WHITEBAL 372
138 +#define MAX_CAMERAS 1
140 +#define DRIVER_VERSION "v1.3"
141 +#define DRIVER_DESC "Konica Webcam driver"
144 + SetWhitebal = 0x01,
145 + SetBrightness = 0x02,
146 + SetSharpness = 0x03,
147 + SetContrast = 0x04,
148 + SetSaturation = 0x05,
160 +#define MAX_FRAME_SIZE SIZE_320X240
162 +static usbvideo_t *cams;
164 +/* Some default values for inital camera settings,
165 + can be set by modprobe */
168 +static enum frame_sizes size;
169 +static int speed = 6; /* Speed (fps) 0 (slowest) to 6 (fastest) */
170 +static int brightness = MAX_BRIGHTNESS/2;
171 +static int contrast = MAX_CONTRAST/2;
172 +static int saturation = MAX_SATURATION/2;
173 +static int sharpness = MAX_SHARPNESS/2;
174 +static int whitebal = 3*(MAX_WHITEBAL/4);
176 +static int spd_to_iface[] = { 1, 0, 3, 2, 4, 5, 6 };
178 +/* These FPS speeds are from the windows config box. They are
179 + * indexed on size (0-2) and speed (0-6). Divide by 3 to get the
183 +static int spd_to_fps[][7] = { { 24, 40, 48, 60, 72, 80, 100 },
184 + { 24, 40, 48, 60, 72, 80, 100 },
185 + { 18, 30, 36, 45, 54, 60, 75 },
186 + { 6, 10, 12, 15, 18, 21, 25 } };
194 +static struct cam_size camera_sizes[] = { { 160, 120, 0x7 },
197 + { 320, 240, 0x5 } };
200 + u8 brightness; /* camera uses 0 - 9, x11 for real value */
201 + u8 contrast; /* as above */
202 + u8 saturation; /* as above */
203 + u8 sharpness; /* as above */
204 + u8 white_bal; /* 0 - 33, x11 for real value */
205 + u8 speed; /* Stored as 0 - 6, used as index in spd_to_* (above) */
206 + u8 size; /* Frame Size */
209 + struct urb *sts_urb[USBVIDEO_NUMSBUF];
210 + u8 sts_buf[USBVIDEO_NUMSBUF][FRAMES_PER_DESC];
211 + struct urb *last_data_urb;
213 + int cur_frame_size; /* number of bytes in current frame size */
214 + int maxline; /* number of lines per frame */
215 + int yplanesz; /* Number of bytes in the Y plane */
216 + unsigned int skip_frame:2;
217 + unsigned int buttonsts:1;
221 +#define konicawc_set_misc(uvd, req, value, index) konicawc_ctrl_msg(uvd, USB_DIR_OUT, req, value, index, NULL, 0)
222 +#define konicawc_get_misc(uvd, req, value, index, buf, sz) konicawc_ctrl_msg(uvd, USB_DIR_IN, req, value, index, buf, sz)
223 +#define konicawc_set_value(uvd, value, index) konicawc_ctrl_msg(uvd, USB_DIR_OUT, 2, value, index, NULL, 0)
226 +static int konicawc_ctrl_msg(uvd_t *uvd, u8 dir, u8 request, u16 value, u16 index, void *buf, int len)
228 + int retval = usb_control_msg(uvd->dev,
229 + dir ? usb_rcvctrlpipe(uvd->dev, 0) : usb_sndctrlpipe(uvd->dev, 0),
230 + request, 0x40 | dir, value, index, buf, len, HZ);
231 + return retval < 0 ? retval : 0;
235 +static void konicawc_set_camera_size(uvd_t *uvd)
237 + struct konicawc *cam = (struct konicawc *)uvd->user_data;
239 + konicawc_set_misc(uvd, 0x2, camera_sizes[cam->size].cmd, 0x08);
240 + cam->width = camera_sizes[cam->size].width;
241 + cam->height = camera_sizes[cam->size].height;
242 + cam->yplanesz = cam->height * cam->width;
243 + cam->cur_frame_size = (cam->yplanesz * 3) / 2;
244 + cam->maxline = cam->yplanesz / 256;
245 + uvd->videosize = VIDEOSIZE(cam->width, cam->height);
249 +static int konicawc_setup_on_open(uvd_t *uvd)
251 + struct konicawc *cam = (struct konicawc *)uvd->user_data;
253 + konicawc_set_misc(uvd, 0x2, 0, 0x0b);
254 + dbg("setting brightness to %d (%d)", cam->brightness,
255 + cam->brightness * 11);
256 + konicawc_set_value(uvd, cam->brightness, SetBrightness);
257 + dbg("setting white balance to %d (%d)", cam->white_bal,
258 + cam->white_bal * 11);
259 + konicawc_set_value(uvd, cam->white_bal, SetWhitebal);
260 + dbg("setting contrast to %d (%d)", cam->contrast,
261 + cam->contrast * 11);
262 + konicawc_set_value(uvd, cam->contrast, SetContrast);
263 + dbg("setting saturation to %d (%d)", cam->saturation,
264 + cam->saturation * 11);
265 + konicawc_set_value(uvd, cam->saturation, SetSaturation);
266 + dbg("setting sharpness to %d (%d)", cam->sharpness,
267 + cam->sharpness * 11);
268 + konicawc_set_value(uvd, cam->sharpness, SetSharpness);
269 + konicawc_set_camera_size(uvd);
270 + konicawc_set_misc(uvd, 0x2, 1, 0x0b);
271 + cam->lastframe = -1;
272 + cam->skip_frame = 2;
273 + cam->buttonsts = 0;
278 +static void konicawc_adjust_picture(uvd_t *uvd)
280 + struct konicawc *cam = (struct konicawc *)uvd->user_data;
282 + konicawc_set_misc(uvd, 0x2, 0, 0x0b);
283 + dbg("new brightness: %d", uvd->vpic.brightness);
284 + uvd->vpic.brightness = (uvd->vpic.brightness > MAX_BRIGHTNESS) ? MAX_BRIGHTNESS : uvd->vpic.brightness;
285 + if(cam->brightness != uvd->vpic.brightness / 11) {
286 + cam->brightness = uvd->vpic.brightness / 11;
287 + dbg("setting brightness to %d (%d)", cam->brightness,
288 + cam->brightness * 11);
289 + konicawc_set_value(uvd, cam->brightness, SetBrightness);
292 + dbg("new contrast: %d", uvd->vpic.contrast);
293 + uvd->vpic.contrast = (uvd->vpic.contrast > MAX_CONTRAST) ? MAX_CONTRAST : uvd->vpic.contrast;
294 + if(cam->contrast != uvd->vpic.contrast / 11) {
295 + cam->contrast = uvd->vpic.contrast / 11;
296 + dbg("setting contrast to %d (%d)", cam->contrast,
297 + cam->contrast * 11);
298 + konicawc_set_value(uvd, cam->contrast, SetContrast);
300 + konicawc_set_misc(uvd, 0x2, 1, 0x0b);
304 +static int konicawc_compress_iso(uvd_t *uvd, struct urb *dataurb, struct urb *stsurb)
308 + unsigned char *status = stsurb->transfer_buffer;
309 + int keep = 0, discard = 0, bad = 0;
310 + struct konicawc *cam = (struct konicawc *)uvd->user_data;
312 + for (i = 0; i < dataurb->number_of_packets; i++) {
313 + int button = cam->buttonsts;
315 + int n = dataurb->iso_frame_desc[i].actual_length;
316 + int st = dataurb->iso_frame_desc[i].status;
317 + cdata = dataurb->transfer_buffer +
318 + dataurb->iso_frame_desc[i].offset;
320 + /* Detect and ignore errored packets */
323 + err("Data error: packet=%d. len=%d. status=%d.",
325 + uvd->stats.iso_err_count++;
329 + /* Detect and ignore empty packets */
331 + uvd->stats.iso_skip_count++;
335 + /* See what the status data said about the packet */
336 + sts = *(status+stsurb->iso_frame_desc[i].offset);
338 + /* sts: 0x80-0xff: frame start with frame number (ie 0-7f)
340 + * bit 0 0: keep packet
341 + * 1: drop packet (padding data)
343 + * bit 4 0 button not clicked
345 + * button is used to `take a picture' (in software)
349 + button = sts & 0x40;
353 + /* work out the button status, but dont do
354 + anything with it for now */
356 + if(button != cam->buttonsts) {
357 + dbg("button: %sclicked", button ? "" : "un");
358 + cam->buttonsts = button;
361 + if(sts == 0x01) { /* drop frame */
366 + if((sts > 0x01) && (sts < 0x80)) {
367 + info("unknown status %2.2x", sts);
373 + if(sts & 0x80) { /* frame start */
374 + unsigned char marker[] = { 0, 0xff, 0, 0x00 };
376 + if(cam->skip_frame == 2) {
381 + /* Do we have enough space for this frame? */
383 + dbg("RingQueue: need %d have %d", cam->cur_frame_size + 4, RingQueue_GetFreeSpace(&uvd->dp));
384 + if(RingQueue_GetFreeSpace(&uvd->dp) < cam->cur_frame_size + 4) {
385 + info("Dropping frame");
386 + cam->skip_frame = 1;
389 + cam->skip_frame = 0;
392 + cam->skip_frame = 0;
395 + dbg("Adding Marker packet = %d, frame = %2.2x",
397 + marker[3] = sts & 0x7F;
398 + if(RingQueue_GetFreeSpace(&uvd->dp) < (4+n)) {
399 + err("RingQueue Full! (cant add header) - Dropping frame");
400 + cam->skip_frame = 1;
402 + RingQueue_Enqueue(&uvd->dp, marker, 4);
407 + if(cam->skip_frame)
410 + totlen += n; /* Little local accounting */
412 + dbg("Adding packet %d, bytes = %d", i, n);
413 + if(RingQueue_GetFreeSpace(&uvd->dp) < n) {
414 + err("RingQueue Full! (want %d got %d) - Dropping frame", n,
415 + RingQueue_GetFreeSpace(&uvd->dp));
416 + cam->skip_frame = 1;
418 + RingQueue_Enqueue(&uvd->dp, cdata, n);
422 + dbg("finished: keep = %d discard = %d bad = %d added %d bytes",
423 + keep, discard, bad, totlen);
429 +static void konicawc_isoc_irq(struct urb *urb)
432 + uvd_t *uvd = urb->context;
433 + struct konicawc *cam = (struct konicawc *)uvd->user_data;
435 + /* We don't want to do anything if we are about to be removed! */
436 + if (!CAMERA_IS_OPERATIONAL(uvd))
440 + if(urb->status == -EOVERFLOW) {
441 + dbg("isoc_irq: got babble");
442 + } else if(urb->status == -ENOENT || urb->status == -EINPROGRESS) {
443 + dbg("isoc_irq: URB unlinked");
445 + dbg("isoc_irq: status %d", urb->status);
450 + if (urb->actual_length > 32) {
451 + cam->last_data_urb = urb;
455 + if (!uvd->streaming) {
457 + info("Not streaming, but interrupt!");
461 + uvd->stats.urb_count++;
462 + if (urb->actual_length <= 0)
463 + goto urb_done_with;
465 + /* Copy the data received into ring queue */
466 + if(cam->last_data_urb) {
467 + len = konicawc_compress_iso(uvd, cam->last_data_urb, urb);
468 + for (i = 0; i < FRAMES_PER_DESC; i++) {
469 + cam->last_data_urb->iso_frame_desc[i].status = 0;
470 + cam->last_data_urb->iso_frame_desc[i].actual_length = 0;
472 + cam->last_data_urb = NULL;
474 + uvd->stats.urb_length = len;
475 + uvd->stats.data_count += len;
477 + if(RingQueue_GetLength(&uvd->dp) >= 384)
478 + RingQueue_WakeUpInterruptible(&uvd->dp);
482 + for (i = 0; i < FRAMES_PER_DESC; i++) {
483 + urb->iso_frame_desc[i].status = 0;
484 + urb->iso_frame_desc[i].actual_length = 0;
490 +static int konicawc_start_data(uvd_t *uvd)
492 + struct usb_device *dev = uvd->dev;
494 + struct konicawc *cam = (struct konicawc *)uvd->user_data;
496 + struct usb_interface_descriptor *interface;
498 + interface = &dev->actconfig->interface[uvd->iface].altsetting[spd_to_iface[cam->speed]];
499 + pktsz = interface->endpoint[1].wMaxPacketSize;
500 + dbg("pktsz = %d", pktsz);
501 + if (!CAMERA_IS_OPERATIONAL(uvd)) {
502 + err("Camera is not operational");
505 + uvd->curframe = -1;
507 + /* Alternate interface 1 is is the biggest frame size */
508 + i = usb_set_interface(dev, uvd->iface, uvd->ifaceAltActive);
510 + err("usb_set_interface error");
511 + uvd->last_error = i;
515 + /* We double buffer the Iso lists */
516 + for (i=0; i < USBVIDEO_NUMSBUF; i++) {
518 + struct urb *urb = uvd->sbuf[i].urb;
520 + urb->context = uvd;
521 + urb->pipe = usb_rcvisocpipe(dev, uvd->video_endp);
522 + urb->transfer_flags = USB_ISO_ASAP;
523 + urb->transfer_buffer = uvd->sbuf[i].data;
524 + urb->complete = konicawc_isoc_irq;
525 + urb->number_of_packets = FRAMES_PER_DESC;
526 + urb->transfer_buffer_length = pktsz * FRAMES_PER_DESC;
527 + for (j=k=0; j < FRAMES_PER_DESC; j++, k += pktsz) {
528 + urb->iso_frame_desc[j].offset = k;
529 + urb->iso_frame_desc[j].length = pktsz;
532 + urb = cam->sts_urb[i];
534 + urb->context = uvd;
535 + urb->pipe = usb_rcvisocpipe(dev, uvd->video_endp-1);
536 + urb->transfer_flags = USB_ISO_ASAP;
537 + urb->transfer_buffer = cam->sts_buf[i];
538 + urb->complete = konicawc_isoc_irq;
539 + urb->number_of_packets = FRAMES_PER_DESC;
540 + urb->transfer_buffer_length = FRAMES_PER_DESC;
541 + for (j=0; j < FRAMES_PER_DESC; j++) {
542 + urb->iso_frame_desc[j].offset = j;
543 + urb->iso_frame_desc[j].length = 1;
547 + cam->last_data_urb = NULL;
549 + /* Link URBs into a ring so that they invoke each other infinitely */
550 + for (i=0; i < USBVIDEO_NUMSBUF; i++) {
551 + if ((i+1) < USBVIDEO_NUMSBUF) {
552 + cam->sts_urb[i]->next = uvd->sbuf[i].urb;
553 + uvd->sbuf[i].urb->next = cam->sts_urb[i+1];
555 + cam->sts_urb[i]->next = uvd->sbuf[i].urb;
556 + uvd->sbuf[i].urb->next = cam->sts_urb[0];
560 + /* Submit all URBs */
561 + for (i=0; i < USBVIDEO_NUMSBUF; i++) {
562 + errFlag = usb_submit_urb(uvd->sbuf[i].urb);
564 + err ("usb_submit_isoc(%d) ret %d", i, errFlag);
566 + errFlag = usb_submit_urb(cam->sts_urb[i]);
568 + err("usb_submit_isoc(%d) ret %d", i, errFlag);
571 + uvd->streaming = 1;
573 + dbg("streaming=1 video_endp=$%02x", uvd->video_endp);
578 +static void konicawc_stop_data(uvd_t *uvd)
581 + struct konicawc *cam;
583 + if ((uvd == NULL) || (!uvd->streaming) || (uvd->dev == NULL))
586 + cam = (struct konicawc *)uvd->user_data;
587 + cam->last_data_urb = NULL;
589 + /* Unschedule all of the iso td's */
590 + for (i=0; i < USBVIDEO_NUMSBUF; i++) {
591 + j = usb_unlink_urb(uvd->sbuf[i].urb);
593 + err("usb_unlink_urb() error %d.", j);
595 + j = usb_unlink_urb(cam->sts_urb[i]);
597 + err("usb_unlink_urb() error %d.", j);
600 + uvd->streaming = 0;
602 + if (!uvd->remove_pending) {
603 + /* Set packet size to 0 */
604 + j = usb_set_interface(uvd->dev, uvd->iface, uvd->ifaceAltInactive);
606 + err("usb_set_interface() error %d.", j);
607 + uvd->last_error = j;
613 +static void konicawc_process_isoc(uvd_t *uvd, usbvideo_frame_t *frame)
615 + struct konicawc *cam = (struct konicawc *)uvd->user_data;
616 + int maxline = cam->maxline;
617 + int yplanesz = cam->yplanesz;
619 + assert(frame != NULL);
622 + dbg("maxline = %d yplanesz = %d", maxline, yplanesz);
625 + dbg("Frame state = %d", frame->scanstate);
627 + if(frame->scanstate == ScanState_Scanning) {
632 + dbg("Searching for marker, queue len = %d", RingQueue_GetLength(&uvd->dp));
633 + while(RingQueue_GetLength(&uvd->dp) >= 4) {
634 + if ((RING_QUEUE_PEEK(&uvd->dp, 0) == 0x00) &&
635 + (RING_QUEUE_PEEK(&uvd->dp, 1) == 0xff) &&
636 + (RING_QUEUE_PEEK(&uvd->dp, 2) == 0x00) &&
637 + (RING_QUEUE_PEEK(&uvd->dp, 3) < 0x80)) {
638 + curframe = RING_QUEUE_PEEK(&uvd->dp, 3);
639 + if(cam->lastframe != -1) {
640 + fdrops = (0x80 + curframe - cam->lastframe) & 0x7F;
643 + info("Dropped %d frames (%d -> %d)", fdrops,
644 + cam->lastframe, curframe);
647 + cam->lastframe = curframe;
648 + frame->curline = 0;
649 + frame->scanstate = ScanState_Lines;
650 + RING_QUEUE_DEQUEUE_BYTES(&uvd->dp, 4);
653 + RING_QUEUE_DEQUEUE_BYTES(&uvd->dp, 1);
657 + dbg("dropped %d bytes looking for new frame", drop);
660 + if(frame->scanstate == ScanState_Scanning)
663 + /* Try to move data from queue into frame buffer
664 + * We get data in blocks of 384 bytes made up of:
665 + * 256 Y, 64 U, 64 V.
666 + * This needs to be written out as a Y plane, a U plane and a V plane.
669 + while ( frame->curline < maxline && (RingQueue_GetLength(&uvd->dp) >= 384)) {
671 + RingQueue_Dequeue(&uvd->dp, frame->data + (frame->curline * 256), 256);
673 + RingQueue_Dequeue(&uvd->dp, frame->data + yplanesz + (frame->curline * 64), 64);
675 + RingQueue_Dequeue(&uvd->dp, frame->data + (5 * yplanesz)/4 + (frame->curline * 64), 64);
676 + frame->seqRead_Length += 384;
679 + /* See if we filled the frame */
680 + if (frame->curline == maxline) {
682 + dbg("got whole frame");
684 + frame->frameState = FrameState_Done_Hold;
685 + frame->curline = 0;
686 + uvd->curframe = -1;
687 + uvd->stats.frame_num++;
692 +static int konicawc_find_fps(int size, int fps)
697 + dbg("konica_find_fps: size = %d fps = %d", size, fps);
698 + if(fps <= spd_to_fps[size][0])
701 + if(fps >= spd_to_fps[size][MAX_SPEED])
704 + for(i = 0; i < MAX_SPEED; i++) {
705 + if((fps >= spd_to_fps[size][i]) && (fps <= spd_to_fps[size][i+1])) {
706 + dbg("fps %d between %d and %d", fps, i, i+1);
707 + if( (fps - spd_to_fps[size][i]) < (spd_to_fps[size][i+1] - fps))
713 + return MAX_SPEED+1;
717 +static int konicawc_set_video_mode(uvd_t *uvd, int x, int y, int fps)
719 + struct konicawc *cam = (struct konicawc *)uvd->user_data;
720 + int newspeed = cam->speed;
723 + if(x > 0 && y > 0) {
725 + dbg("trying to find size %d,%d", x, y);
726 + for(newsize = 0; newsize <= MAX_FRAME_SIZE; newsize++) {
727 + if((camera_sizes[newsize].width == x) && (camera_sizes[newsize].height == y))
731 + newsize = cam->size;
734 + if(newsize > MAX_FRAME_SIZE) {
735 + dbg("couldnt find size %d,%d", x, y);
740 + dbg("trying to set fps to %d", fps);
741 + newspeed = konicawc_find_fps(newsize, fps);
742 + dbg("find_fps returned %d (%d)", newspeed, spd_to_fps[newsize][newspeed]);
745 + if(newspeed > MAX_SPEED)
748 + dbg("setting size to %d speed to %d", newsize, newspeed);
749 + if((newsize == cam->size) && (newspeed == cam->speed)) {
750 + dbg("Nothing to do");
753 + info("setting to %dx%d @ %d fps", camera_sizes[newsize].width,
754 + camera_sizes[newsize].height, spd_to_fps[newsize][newspeed]/3);
756 + konicawc_stop_data(uvd);
757 + konicawc_set_misc(uvd, 0x2, 0, 0x0b);
758 + uvd->ifaceAltActive = spd_to_iface[newspeed];
759 + dbg("new interface = %d", uvd->ifaceAltActive);
760 + cam->speed = newspeed;
762 + if(cam->size != newsize) {
763 + cam->size = newsize;
764 + konicawc_set_camera_size(uvd);
767 + /* Flush the input queue and clear any current frame in progress */
769 + RingQueue_Flush(&uvd->dp);
770 + cam->skip_frame = 2;
771 + cam->lastframe = -1;
772 + if(uvd->curframe != -1) {
773 + uvd->frame[uvd->curframe].curline = 0;
774 + uvd->frame[uvd->curframe].seqRead_Length = 0;
775 + uvd->frame[uvd->curframe].seqRead_Index = 0;
778 + konicawc_set_misc(uvd, 0x2, 1, 0x0b);
779 + konicawc_start_data(uvd);
784 +static int konicawc_calculate_fps(uvd_t *uvd)
786 + struct konicawc *cam = uvd->user_data;
787 + return spd_to_fps[cam->size][cam->speed]/3;
791 +static void konicawc_configure_video(uvd_t *uvd)
793 + struct konicawc *cam = (struct konicawc *)uvd->user_data;
796 + memset(&uvd->vpic, 0, sizeof(uvd->vpic));
797 + memset(&uvd->vpic_old, 0x55, sizeof(uvd->vpic_old));
799 + RESTRICT_TO_RANGE(brightness, 0, MAX_BRIGHTNESS);
800 + RESTRICT_TO_RANGE(contrast, 0, MAX_CONTRAST);
801 + RESTRICT_TO_RANGE(saturation, 0, MAX_SATURATION);
802 + RESTRICT_TO_RANGE(sharpness, 0, MAX_SHARPNESS);
803 + RESTRICT_TO_RANGE(whitebal, 0, MAX_WHITEBAL);
805 + cam->brightness = brightness / 11;
806 + cam->contrast = contrast / 11;
807 + cam->saturation = saturation / 11;
808 + cam->sharpness = sharpness / 11;
809 + cam->white_bal = whitebal / 11;
811 + uvd->vpic.colour = 108;
812 + uvd->vpic.hue = 108;
813 + uvd->vpic.brightness = brightness;
814 + uvd->vpic.contrast = contrast;
815 + uvd->vpic.whiteness = whitebal;
816 + uvd->vpic.depth = 6;
817 + uvd->vpic.palette = VIDEO_PALETTE_YUV420P;
819 + memset(&uvd->vcap, 0, sizeof(uvd->vcap));
820 + strcpy(uvd->vcap.name, "Konica Webcam");
821 + uvd->vcap.type = VID_TYPE_CAPTURE;
822 + uvd->vcap.channels = 1;
823 + uvd->vcap.audios = 0;
824 + uvd->vcap.minwidth = camera_sizes[SIZE_160X120].width;
825 + uvd->vcap.minheight = camera_sizes[SIZE_160X120].height;
826 + uvd->vcap.maxwidth = camera_sizes[SIZE_320X240].width;
827 + uvd->vcap.maxheight = camera_sizes[SIZE_320X240].height;
829 + memset(&uvd->vchan, 0, sizeof(uvd->vchan));
830 + uvd->vchan.flags = 0 ;
831 + uvd->vchan.tuners = 0;
832 + uvd->vchan.channel = 0;
833 + uvd->vchan.type = VIDEO_TYPE_CAMERA;
834 + strcpy(uvd->vchan.name, "Camera");
836 + /* Talk to device */
837 + dbg("device init");
838 + if(!konicawc_get_misc(uvd, 0x3, 0, 0x10, buf, 2))
839 + dbg("3,10 -> %2.2x %2.2x", buf[0], buf[1]);
840 + if(!konicawc_get_misc(uvd, 0x3, 0, 0x10, buf, 2))
841 + dbg("3,10 -> %2.2x %2.2x", buf[0], buf[1]);
842 + if(konicawc_set_misc(uvd, 0x2, 0, 0xd))
843 + dbg("2,0,d failed");
844 + dbg("setting initial values");
848 +static void *konicawc_probe(struct usb_device *dev, unsigned int ifnum, const struct usb_device_id *devid)
852 + int actInterface=-1, inactInterface=-1, maxPS=0;
853 + unsigned char video_ep = 0;
856 + dbg("konicawc_probe(%p,%u.)", dev, ifnum);
858 + /* We don't handle multi-config cameras */
859 + if (dev->descriptor.bNumConfigurations != 1)
862 + info("Konica Webcam (rev. 0x%04x)", dev->descriptor.bcdDevice);
863 + RESTRICT_TO_RANGE(speed, 0, MAX_SPEED);
865 + /* Validate found interface: must have one ISO endpoint */
866 + nas = dev->actconfig->interface[ifnum].num_altsetting;
868 + err("Incorrect number of alternate settings (%d) for this camera!", nas);
871 + /* Validate all alternate settings */
872 + for (i=0; i < nas; i++) {
873 + const struct usb_interface_descriptor *interface;
874 + const struct usb_endpoint_descriptor *endpoint;
876 + interface = &dev->actconfig->interface[ifnum].altsetting[i];
877 + if (interface->bNumEndpoints != 2) {
878 + err("Interface %d. has %u. endpoints!",
879 + ifnum, (unsigned)(interface->bNumEndpoints));
882 + endpoint = &interface->endpoint[1];
883 + dbg("found endpoint: addr: 0x%2.2x maxps = 0x%4.4x",
884 + endpoint->bEndpointAddress, endpoint->wMaxPacketSize);
886 + video_ep = endpoint->bEndpointAddress;
887 + else if (video_ep != endpoint->bEndpointAddress) {
888 + err("Alternate settings have different endpoint addresses!");
891 + if ((endpoint->bmAttributes & 0x03) != 0x01) {
892 + err("Interface %d. has non-ISO endpoint!", ifnum);
895 + if ((endpoint->bEndpointAddress & 0x80) == 0) {
896 + err("Interface %d. has ISO OUT endpoint!", ifnum);
899 + if (endpoint->wMaxPacketSize == 0) {
900 + if (inactInterface < 0)
901 + inactInterface = i;
903 + err("More than one inactive alt. setting!");
907 + if (i == spd_to_iface[speed]) {
908 + /* This one is the requested one */
912 + if(endpoint->wMaxPacketSize > maxPS)
913 + maxPS = endpoint->wMaxPacketSize;
915 + if(actInterface == -1) {
916 + err("Cant find required endpoint");
920 + dbg("Selecting requested active setting=%d. maxPS=%d.", actInterface, maxPS);
922 + /* Code below may sleep, need to lock module while we are here */
924 + uvd = usbvideo_AllocateDevice(cams);
926 + struct konicawc *cam = (struct konicawc *)(uvd->user_data);
927 + /* Here uvd is a fully allocated uvd_t object */
928 + for(i = 0; i < USBVIDEO_NUMSBUF; i++) {
929 + cam->sts_urb[i] = usb_alloc_urb(FRAMES_PER_DESC);
930 + if(cam->sts_urb[i] == NULL) {
932 + usb_free_urb(cam->sts_urb[i]);
934 + err("cant allocate urbs");
938 + cam->speed = speed;
939 + RESTRICT_TO_RANGE(size, SIZE_160X120, SIZE_320X240);
940 + cam->width = camera_sizes[size].width;
941 + cam->height = camera_sizes[size].height;
945 + uvd->debug = debug;
947 + uvd->iface = ifnum;
948 + uvd->ifaceAltInactive = inactInterface;
949 + uvd->ifaceAltActive = actInterface;
950 + uvd->video_endp = video_ep;
951 + uvd->iso_packet_len = maxPS;
952 + uvd->paletteBits = 1L << VIDEO_PALETTE_YUV420P;
953 + uvd->defaultPalette = VIDEO_PALETTE_YUV420P;
954 + uvd->canvas = VIDEOSIZE(320, 240);
955 + uvd->videosize = VIDEOSIZE(cam->width, cam->height);
957 + /* Initialize konicawc specific data */
958 + konicawc_configure_video(uvd);
960 + i = usbvideo_RegisterVideoDevice(uvd);
961 + uvd->max_frame_size = (320 * 240 * 3)/2;
963 + err("usbvideo_RegisterVideoDevice() failed.");
972 +static void konicawc_free_uvd(uvd_t *uvd)
975 + struct konicawc *cam = (struct konicawc *)uvd->user_data;
977 + for (i=0; i < USBVIDEO_NUMSBUF; i++) {
978 + usb_free_urb(cam->sts_urb[i]);
979 + cam->sts_urb[i] = NULL;
984 +static struct usb_device_id id_table[] = {
985 + { USB_DEVICE(0x04c8, 0x0720) }, /* Intel YC 76 */
986 + { } /* Terminating entry */
990 +static int __init konicawc_init(void)
992 + usbvideo_cb_t cbTbl;
993 + info(DRIVER_DESC " " DRIVER_VERSION);
994 + memset(&cbTbl, 0, sizeof(cbTbl));
995 + cbTbl.probe = konicawc_probe;
996 + cbTbl.setupOnOpen = konicawc_setup_on_open;
997 + cbTbl.processData = konicawc_process_isoc;
998 + cbTbl.getFPS = konicawc_calculate_fps;
999 + cbTbl.setVideoMode = konicawc_set_video_mode;
1000 + cbTbl.startDataPump = konicawc_start_data;
1001 + cbTbl.stopDataPump = konicawc_stop_data;
1002 + cbTbl.adjustPicture = konicawc_adjust_picture;
1003 + cbTbl.userFree = konicawc_free_uvd;
1004 + return usbvideo_register(
1007 + sizeof(struct konicawc),
1015 +static void __exit konicawc_cleanup(void)
1017 + usbvideo_Deregister(&cams);
1021 +MODULE_DEVICE_TABLE(usb, id_table);
1023 +MODULE_LICENSE("GPL");
1024 +MODULE_AUTHOR("Simon Evans <spse@secret.org.uk>");
1025 +MODULE_DESCRIPTION(DRIVER_DESC);
1026 +MODULE_PARM(speed, "i");
1027 +MODULE_PARM_DESC(speed, "Initial speed: 0 (slowest) - 6 (fastest)");
1028 +MODULE_PARM(size, "i");
1029 +MODULE_PARM_DESC(size, "Initial Size 0: 160x120 1: 160x136 2: 176x144 3: 320x240");
1030 +MODULE_PARM(brightness, "i");
1031 +MODULE_PARM_DESC(brightness, "Initial brightness 0 - 108");
1032 +MODULE_PARM(contrast, "i");
1033 +MODULE_PARM_DESC(contrast, "Initial contrast 0 - 108");
1034 +MODULE_PARM(saturation, "i");
1035 +MODULE_PARM_DESC(saturation, "Initial saturation 0 - 108");
1036 +MODULE_PARM(sharpness, "i");
1037 +MODULE_PARM_DESC(sharpness, "Initial brightness 0 - 108");
1038 +MODULE_PARM(whitebal, "i");
1039 +MODULE_PARM_DESC(whitebal, "Initial white balance 0 - 363");
1040 +MODULE_PARM(debug, "i");
1041 +MODULE_PARM_DESC(debug, "Debug level: 0-9 (default=0)");
1042 +module_init(konicawc_init);
1043 +module_exit(konicawc_cleanup);
1044 diff -urN -X /home/spse/dontdiff linux-2.4.19-pre8/drivers/usb/ultracam.c linux-2.4.19-pre8-konicawc/drivers/usb/ultracam.c
1045 --- linux-2.4.19-pre8/drivers/usb/ultracam.c Wed Oct 17 22:34:06 2001
1046 +++ linux-2.4.19-pre8-konicawc/drivers/usb/ultracam.c Sat May 11 01:34:30 2002
1047 @@ -659,6 +659,12 @@
1052 +static struct usb_device_id id_table[] = {
1053 + { USB_DEVICE(ULTRACAM_VENDOR_ID, ULTRACAM_PRODUCT_ID) },
1054 + { } /* Terminating entry */
1069 static void __exit ultracam_cleanup(void)
1070 @@ -690,16 +697,7 @@
1071 usbvideo_Deregister(&cams);
1074 -#if defined(usb_device_id_ver)
1076 -static __devinitdata struct usb_device_id id_table[] = {
1077 - { USB_DEVICE(ULTRACAM_VENDOR_ID, ULTRACAM_PRODUCT_ID) },
1078 - { } /* Terminating entry */
1080 MODULE_DEVICE_TABLE(usb, id_table);
1083 -#endif /* defined(usb_device_id_ver) */
1084 MODULE_LICENSE("GPL");
1086 module_init(ultracam_init);
1087 diff -urN -X /home/spse/dontdiff linux-2.4.19-pre8/drivers/usb/usbvideo.c linux-2.4.19-pre8-konicawc/drivers/usb/usbvideo.c
1088 --- linux-2.4.19-pre8/drivers/usb/usbvideo.c Thu Oct 11 07:42:46 2001
1089 +++ linux-2.4.19-pre8-konicawc/drivers/usb/usbvideo.c Sat May 11 03:11:30 2002
1090 @@ -159,9 +159,21 @@
1092 void RingQueue_Allocate(RingQueue_t *rq, int rqLen)
1094 + /* Make sure the requested size is a power of 2 and
1095 + round up if necessary. This allows index wrapping
1096 + using masks rather than modulo */
1104 + if(rqLen != 1 << (i-1))
1108 + rq->ri = rq->wi = 0;
1109 rq->queue = usbvideo_rvmalloc(rq->length);
1110 assert(rq->queue != NULL);
1112 @@ -185,12 +197,33 @@
1114 int RingQueue_Dequeue(RingQueue_t *rq, unsigned char *dst, int len)
1120 assert(dst != NULL);
1121 - for (i=0; i < len; i++) {
1122 - dst[i] = rq->queue[rq->ri];
1123 - RING_QUEUE_DEQUEUE_BYTES(rq,1);
1126 + rql = RingQueue_GetLength(rq);
1130 + /* Clip requested length to available data */
1135 + if(rq->ri > rq->wi) {
1136 + /* Read data from tail */
1137 + int read = (toread < (rq->length - rq->ri)) ? toread : rq->length - rq->ri;
1138 + memcpy(dst, rq->queue + rq->ri, read);
1141 + rq->ri = (rq->ri + read) & (rq->length-1);
1144 + /* Read data from head */
1145 + memcpy(dst, rq->queue + rq->ri, toread);
1146 + rq->ri = (rq->ri + toread) & (rq->length-1);
1154 - memmove(rq->queue + rq->wi, cdata, m);
1155 + memcpy(rq->queue + rq->wi, cdata, m);
1156 RING_QUEUE_ADVANCE_INDEX(rq, wi, m);
1159 @@ -225,22 +258,6 @@
1163 -int RingQueue_GetLength(const RingQueue_t *rq)
1167 - assert(rq != NULL);
1176 - return wi + (rq->length - ri);
1179 void RingQueue_InterruptibleSleepOn(RingQueue_t *rq)
1182 @@ -254,6 +271,14 @@
1183 wake_up_interruptible(&rq->wqh);
1186 +void RingQueue_Flush(RingQueue_t *rq)
1188 + assert(rq != NULL);
1195 * usbvideo_VideosizeToString()
1198 q_used = RingQueue_GetLength(&uvd->dp);
1199 if ((uvd->dp.ri + q_used) >= uvd->dp.length) {
1200 u_hi = uvd->dp.length;
1201 - u_lo = (q_used + uvd->dp.ri) % uvd->dp.length;
1202 + u_lo = (q_used + uvd->dp.ri) & (uvd->dp.length-1);
1204 u_hi = (q_used + uvd->dp.ri);
1207 const int num_extra,
1208 const char *driverName,
1209 const usbvideo_cb_t *cbTbl,
1210 - struct module *md )
1211 + struct module *md,
1212 + const struct usb_device_id *id_table)
1214 static const char proc[] = "usbvideo_register";
1216 @@ -791,6 +817,10 @@
1217 cams->cb.getFrame = usbvideo_GetFrame;
1218 if (cams->cb.disconnect == NULL)
1219 cams->cb.disconnect = usbvideo_Disconnect;
1220 + if (cams->cb.startDataPump == NULL)
1221 + cams->cb.startDataPump = usbvideo_StartDataPump;
1222 + if (cams->cb.stopDataPump == NULL)
1223 + cams->cb.stopDataPump = usbvideo_StopDataPump;
1226 * If both /proc fs callbacks are NULL then we assume that the driver
1228 cams->usbdrv.name = cams->drvName;
1229 cams->usbdrv.probe = cams->cb.probe;
1230 cams->usbdrv.disconnect = cams->cb.disconnect;
1231 + cams->usbdrv.id_table = id_table;
1234 if (cams->uses_procfs) {
1236 uvd->remove_pending = 1; /* Now all ISO data will be ignored */
1238 /* At this time we ask to cancel outstanding URBs */
1239 - usbvideo_StopDataPump(uvd);
1240 + GET_CALLBACK(uvd, stopDataPump)(uvd);
1242 for (i=0; i < USBVIDEO_NUMSBUF; i++)
1243 usb_free_urb(uvd->sbuf[i].urb);
1244 @@ -1192,8 +1223,7 @@
1246 if (!CAMERA_IS_OPERATIONAL(uvd))
1249 - if (size > (((2 * uvd->max_frame_size) + PAGE_SIZE - 1) & ~(PAGE_SIZE - 1)))
1250 + if (size > (((USBVIDEO_NUMFRAMES * uvd->max_frame_size) + PAGE_SIZE - 1) & ~(PAGE_SIZE - 1)))
1253 pos = (unsigned long) uvd->fbuf;
1254 @@ -1254,7 +1284,7 @@
1255 /* Allocate memory for the frame buffers */
1256 uvd->fbuf_size = USBVIDEO_NUMFRAMES * uvd->max_frame_size;
1257 uvd->fbuf = usbvideo_rvmalloc(uvd->fbuf_size);
1258 - RingQueue_Allocate(&uvd->dp, 128*1024); /* FIXME #define */
1259 + RingQueue_Allocate(&uvd->dp, RING_QUEUE_SIZE);
1260 if ((uvd->fbuf == NULL) ||
1261 (!RingQueue_IsAllocated(&uvd->dp))) {
1262 err("%s: Failed to allocate fbuf or dp", proc);
1263 @@ -1299,7 +1329,7 @@
1265 /* Start data pump if we have valid endpoint */
1266 if (uvd->video_endp != 0)
1267 - errCode = usbvideo_StartDataPump(uvd);
1268 + errCode = GET_CALLBACK(uvd, startDataPump)(uvd);
1270 if (VALID_CALLBACK(uvd, setupOnOpen)) {
1272 @@ -1350,7 +1380,7 @@
1273 info("%s($%p)", proc, dev);
1276 - usbvideo_StopDataPump(uvd);
1277 + GET_CALLBACK(uvd, stopDataPump)(uvd);
1278 usbvideo_rvfree(uvd->fbuf, uvd->fbuf_size);
1280 RingQueue_Free(&uvd->dp);
1281 @@ -1458,8 +1488,8 @@
1285 - vw.width = VIDEOSIZE_X(uvd->canvas);
1286 - vw.height = VIDEOSIZE_Y(uvd->canvas);
1287 + vw.width = VIDEOSIZE_X(uvd->videosize);
1288 + vw.height = VIDEOSIZE_Y(uvd->videosize);
1290 if (VALID_CALLBACK(uvd, getFPS))
1291 vw.flags = GET_CALLBACK(uvd, getFPS)(uvd);
1292 @@ -1529,8 +1559,8 @@
1296 - if ((vm.frame != 0) && (vm.frame != 1)) {
1297 - err("VIDIOCMCAPTURE: vm.frame=%d. !E [0,1]", vm.frame);
1298 + if ((vm.frame < 0) && (vm.frame >= USBVIDEO_NUMFRAMES)) {
1299 + err("VIDIOCMCAPTURE: vm.frame=%d. !E [0-%d]", vm.frame, USBVIDEO_NUMFRAMES-1);
1302 if (uvd->frame[vm.frame].frameState == FrameState_Grabbing) {
1303 @@ -1629,7 +1659,7 @@
1305 static const char proc[] = "usbvideo_v4l_read";
1306 uvd_t *uvd = (uvd_t *) dev;
1309 usbvideo_frame_t *frame;
1311 if (!CAMERA_IS_OPERATIONAL(uvd) || (buf == NULL))
1312 @@ -1641,14 +1671,13 @@
1315 /* See if a frame is completed, then use it. */
1316 - if ((uvd->frame[0].frameState == FrameState_Done) ||
1317 - (uvd->frame[0].frameState == FrameState_Done_Hold) ||
1318 - (uvd->frame[0].frameState == FrameState_Error)) {
1320 - } else if ((uvd->frame[1].frameState >= FrameState_Done) ||
1321 - (uvd->frame[1].frameState == FrameState_Done_Hold) ||
1322 - (uvd->frame[1].frameState >= FrameState_Done)) {
1324 + for(i = 0; i < USBVIDEO_NUMFRAMES; i++) {
1325 + if ((uvd->frame[i].frameState == FrameState_Done) ||
1326 + (uvd->frame[i].frameState == FrameState_Done_Hold) ||
1327 + (uvd->frame[i].frameState == FrameState_Error)) {
1333 /* FIXME: If we don't start a frame here then who ever does? */
1334 @@ -1663,10 +1692,12 @@
1335 * We will need to wait until it becomes cooked, of course.
1338 - if (uvd->frame[0].frameState == FrameState_Grabbing)
1340 - else if (uvd->frame[1].frameState == FrameState_Grabbing)
1342 + for(i = 0; i < USBVIDEO_NUMFRAMES; i++) {
1343 + if (uvd->frame[i].frameState == FrameState_Grabbing) {
1351 @@ -1764,7 +1795,7 @@
1353 /* Mark it as available to be used again. */
1354 uvd->frame[frmx].frameState = FrameState_Unused;
1355 - if (usbvideo_NewFrame(uvd, frmx ? 0 : 1)) {
1356 + if (usbvideo_NewFrame(uvd, (frmx + 1) % USBVIDEO_NUMFRAMES)) {
1357 err("%s: usbvideo_NewFrame failed.", proc);
1360 @@ -2001,7 +2032,7 @@
1361 uvd->settingsAdjusted = 1;
1364 - n = (framenum - 1 + USBVIDEO_NUMFRAMES) % USBVIDEO_NUMFRAMES;
1365 + n = (framenum + 1) % USBVIDEO_NUMFRAMES;
1366 if (uvd->frame[n].frameState == FrameState_Ready)
1369 @@ -2033,7 +2064,8 @@
1371 if (!(uvd->flags & FLAGS_SEPARATE_FRAMES)) {
1372 /* This copies previous frame into this one to mask losses */
1373 - memmove(frame->data, uvd->frame[1-framenum].data, uvd->max_frame_size);
1374 + int prev = (framenum - 1 + USBVIDEO_NUMFRAMES) % USBVIDEO_NUMFRAMES;
1375 + memmove(frame->data, uvd->frame[prev].data, uvd->max_frame_size);
1377 if (uvd->flags & FLAGS_CLEAN_FRAMES) {
1378 /* This provides a "clean" frame but slows things down */
1379 diff -urN -X /home/spse/dontdiff linux-2.4.19-pre8/drivers/usb/usbvideo.h linux-2.4.19-pre8-konicawc/drivers/usb/usbvideo.h
1380 --- linux-2.4.19-pre8/drivers/usb/usbvideo.h Thu Oct 11 07:42:46 2001
1381 +++ linux-2.4.19-pre8-konicawc/drivers/usb/usbvideo.h Sat May 11 01:58:58 2002
1382 @@ -113,9 +113,10 @@
1383 mr = LIMIT_RGB(mm_r); \
1386 -#define RING_QUEUE_ADVANCE_INDEX(rq,ind,n) (rq)->ind = ((rq)->ind + (n)) % (rq)->length
1387 +#define RING_QUEUE_SIZE (128*1024) /* Must be a power of 2 */
1388 +#define RING_QUEUE_ADVANCE_INDEX(rq,ind,n) (rq)->ind = ((rq)->ind + (n)) & ((rq)->length-1)
1389 #define RING_QUEUE_DEQUEUE_BYTES(rq,n) RING_QUEUE_ADVANCE_INDEX(rq,ri,n)
1390 -#define RING_QUEUE_PEEK(rq,ofs) ((rq)->queue[((ofs) + (rq)->ri) % (rq)->length])
1391 +#define RING_QUEUE_PEEK(rq,ofs) ((rq)->queue[((ofs) + (rq)->ri) & ((rq)->length-1)])
1394 unsigned char *queue; /* Data from the Isoc data pump */
1396 int (*getFrame)(uvd_t *, int);
1397 int (*procfs_read)(char *page,char **start,off_t off,int count,int *eof,void *data);
1398 int (*procfs_write)(struct file *file,const char *buffer,unsigned long count,void *data);
1399 + int (*startDataPump)(uvd_t *uvd);
1400 + void (*stopDataPump)(uvd_t *uvd);
1401 + int (*setVideoMode)(uvd_t *, int, int, int);
1404 struct s_usbvideo_t {
1405 @@ -302,15 +306,20 @@
1406 #define VALID_CALLBACK(uvd,cbName) ((((uvd) != NULL) && \
1407 ((uvd)->handle != NULL)) ? GET_CALLBACK(uvd,cbName) : NULL)
1409 -void RingQueue_Initialize(RingQueue_t *rq);
1410 -void RingQueue_Allocate(RingQueue_t *rq, int rqLen);
1411 -int RingQueue_IsAllocated(const RingQueue_t *rq);
1412 -void RingQueue_Free(RingQueue_t *rq);
1413 int RingQueue_Dequeue(RingQueue_t *rq, unsigned char *dst, int len);
1414 int RingQueue_Enqueue(RingQueue_t *rq, const unsigned char *cdata, int n);
1415 -int RingQueue_GetLength(const RingQueue_t *rq);
1416 -void RingQueue_InterruptibleSleepOn(RingQueue_t *rq);
1417 void RingQueue_WakeUpInterruptible(RingQueue_t *rq);
1418 +void RingQueue_Flush(RingQueue_t *rq);
1420 +static inline int RingQueue_GetLength(const RingQueue_t *rq)
1422 + return (rq->wi - rq->ri + rq->length) & (rq->length-1);
1425 +static inline int RingQueue_GetFreeSpace(const RingQueue_t *rq)
1427 + return rq->length - RingQueue_GetLength(rq);
1430 void usbvideo_CollectRawData(uvd_t *uvd, usbvideo_frame_t *frame);
1431 void usbvideo_DrawLine(
1433 const int num_extra,
1434 const char *driverName,
1435 const usbvideo_cb_t *cbTable,
1436 - struct module *md);
1437 + struct module *md,
1438 + const struct usb_device_id *id_table);
1439 uvd_t *usbvideo_AllocateDevice(usbvideo_t *cams);
1440 int usbvideo_RegisterVideoDevice(uvd_t *uvd);
1441 void usbvideo_Deregister(usbvideo_t **uvt);