aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/media/video
diff options
context:
space:
mode:
Diffstat (limited to 'drivers/media/video')
-rw-r--r--drivers/media/video/c-qcam.c483
1 files changed, 237 insertions, 246 deletions
diff --git a/drivers/media/video/c-qcam.c b/drivers/media/video/c-qcam.c
index e2cbebab959b..8f1dd88b32a6 100644
--- a/drivers/media/video/c-qcam.c
+++ b/drivers/media/video/c-qcam.c
@@ -79,17 +79,17 @@ static inline void qcam_set_ack(struct qcam_device *qcam, unsigned int i)
79{ 79{
80 /* note: the QC specs refer to the PCAck pin by voltage, not 80 /* note: the QC specs refer to the PCAck pin by voltage, not
81 software level. PC ports have builtin inverters. */ 81 software level. PC ports have builtin inverters. */
82 parport_frob_control(qcam->pport, 8, i?8:0); 82 parport_frob_control(qcam->pport, 8, i ? 8 : 0);
83} 83}
84 84
85static inline unsigned int qcam_ready1(struct qcam_device *qcam) 85static inline unsigned int qcam_ready1(struct qcam_device *qcam)
86{ 86{
87 return (parport_read_status(qcam->pport) & 0x8)?1:0; 87 return (parport_read_status(qcam->pport) & 0x8) ? 1 : 0;
88} 88}
89 89
90static inline unsigned int qcam_ready2(struct qcam_device *qcam) 90static inline unsigned int qcam_ready2(struct qcam_device *qcam)
91{ 91{
92 return (parport_read_data(qcam->pport) & 0x1)?1:0; 92 return (parport_read_data(qcam->pport) & 0x1) ? 1 : 0;
93} 93}
94 94
95static unsigned int qcam_await_ready1(struct qcam_device *qcam, 95static unsigned int qcam_await_ready1(struct qcam_device *qcam,
@@ -99,14 +99,13 @@ static unsigned int qcam_await_ready1(struct qcam_device *qcam,
99 unsigned int i; 99 unsigned int i;
100 100
101 for (oldjiffies = jiffies; 101 for (oldjiffies = jiffies;
102 time_before(jiffies, oldjiffies + msecs_to_jiffies(40)); ) 102 time_before(jiffies, oldjiffies + msecs_to_jiffies(40));)
103 if (qcam_ready1(qcam) == value) 103 if (qcam_ready1(qcam) == value)
104 return 0; 104 return 0;
105 105
106 /* If the camera didn't respond within 1/25 second, poll slowly 106 /* If the camera didn't respond within 1/25 second, poll slowly
107 for a while. */ 107 for a while. */
108 for (i = 0; i < 50; i++) 108 for (i = 0; i < 50; i++) {
109 {
110 if (qcam_ready1(qcam) == value) 109 if (qcam_ready1(qcam) == value)
111 return 0; 110 return 0;
112 msleep_interruptible(100); 111 msleep_interruptible(100);
@@ -125,14 +124,13 @@ static unsigned int qcam_await_ready2(struct qcam_device *qcam, int value)
125 unsigned int i; 124 unsigned int i;
126 125
127 for (oldjiffies = jiffies; 126 for (oldjiffies = jiffies;
128 time_before(jiffies, oldjiffies + msecs_to_jiffies(40)); ) 127 time_before(jiffies, oldjiffies + msecs_to_jiffies(40));)
129 if (qcam_ready2(qcam) == value) 128 if (qcam_ready2(qcam) == value)
130 return 0; 129 return 0;
131 130
132 /* If the camera didn't respond within 1/25 second, poll slowly 131 /* If the camera didn't respond within 1/25 second, poll slowly
133 for a while. */ 132 for a while. */
134 for (i = 0; i < 50; i++) 133 for (i = 0; i < 50; i++) {
135 {
136 if (qcam_ready2(qcam) == value) 134 if (qcam_ready2(qcam) == value)
137 return 0; 135 return 0;
138 msleep_interruptible(100); 136 msleep_interruptible(100);
@@ -149,22 +147,25 @@ static unsigned int qcam_await_ready2(struct qcam_device *qcam, int value)
149static int qcam_read_data(struct qcam_device *qcam) 147static int qcam_read_data(struct qcam_device *qcam)
150{ 148{
151 unsigned int idata; 149 unsigned int idata;
150
152 qcam_set_ack(qcam, 0); 151 qcam_set_ack(qcam, 0);
153 if (qcam_await_ready1(qcam, 1)) return -1; 152 if (qcam_await_ready1(qcam, 1))
153 return -1;
154 idata = parport_read_status(qcam->pport) & 0xf0; 154 idata = parport_read_status(qcam->pport) & 0xf0;
155 qcam_set_ack(qcam, 1); 155 qcam_set_ack(qcam, 1);
156 if (qcam_await_ready1(qcam, 0)) return -1; 156 if (qcam_await_ready1(qcam, 0))
157 idata |= (parport_read_status(qcam->pport) >> 4); 157 return -1;
158 idata |= parport_read_status(qcam->pport) >> 4;
158 return idata; 159 return idata;
159} 160}
160 161
161static int qcam_write_data(struct qcam_device *qcam, unsigned int data) 162static int qcam_write_data(struct qcam_device *qcam, unsigned int data)
162{ 163{
163 unsigned int idata; 164 unsigned int idata;
165
164 parport_write_data(qcam->pport, data); 166 parport_write_data(qcam->pport, data);
165 idata = qcam_read_data(qcam); 167 idata = qcam_read_data(qcam);
166 if (data != idata) 168 if (data != idata) {
167 {
168 printk(KERN_WARNING "cqcam: sent %x but received %x\n", data, 169 printk(KERN_WARNING "cqcam: sent %x but received %x\n", data,
169 idata); 170 idata);
170 return 1; 171 return 1;
@@ -212,13 +213,12 @@ static int qc_detect(struct qcam_device *qcam)
212 213
213 /* look for a heartbeat */ 214 /* look for a heartbeat */
214 ostat = stat = parport_read_status(qcam->pport); 215 ostat = stat = parport_read_status(qcam->pport);
215 for (i=0; i<250; i++) 216 for (i = 0; i < 250; i++) {
216 {
217 mdelay(1); 217 mdelay(1);
218 stat = parport_read_status(qcam->pport); 218 stat = parport_read_status(qcam->pport);
219 if (ostat != stat) 219 if (ostat != stat) {
220 { 220 if (++count >= 3)
221 if (++count >= 3) return 1; 221 return 1;
222 ostat = stat; 222 ostat = stat;
223 } 223 }
224 } 224 }
@@ -232,13 +232,12 @@ static int qc_detect(struct qcam_device *qcam)
232 count = 0; 232 count = 0;
233 233
234 ostat = stat = parport_read_status(qcam->pport); 234 ostat = stat = parport_read_status(qcam->pport);
235 for (i=0; i<250; i++) 235 for (i = 0; i < 250; i++) {
236 {
237 mdelay(1); 236 mdelay(1);
238 stat = parport_read_status(qcam->pport); 237 stat = parport_read_status(qcam->pport);
239 if (ostat != stat) 238 if (ostat != stat) {
240 { 239 if (++count >= 3)
241 if (++count >= 3) return 1; 240 return 1;
242 ostat = stat; 241 ostat = stat;
243 } 242 }
244 } 243 }
@@ -263,7 +262,7 @@ static void qc_setup(struct qcam_device *q)
263{ 262{
264 qc_reset(q); 263 qc_reset(q);
265 264
266 /* Set the brightness. */ 265 /* Set the brightness. */
267 qcam_set(q, 11, q->brightness); 266 qcam_set(q, 11, q->brightness);
268 267
269 /* Set the height and width. These refer to the actual 268 /* Set the height and width. These refer to the actual
@@ -292,25 +291,25 @@ static unsigned int qcam_read_bytes(struct qcam_device *q, unsigned char *buf, u
292 unsigned int bytes = 0; 291 unsigned int bytes = 0;
293 292
294 qcam_set_ack(q, 0); 293 qcam_set_ack(q, 0);
295 if (q->bidirectional) 294 if (q->bidirectional) {
296 {
297 /* It's a bidirectional port */ 295 /* It's a bidirectional port */
298 while (bytes < nbytes) 296 while (bytes < nbytes) {
299 {
300 unsigned int lo1, hi1, lo2, hi2; 297 unsigned int lo1, hi1, lo2, hi2;
301 unsigned char r, g, b; 298 unsigned char r, g, b;
302 299
303 if (qcam_await_ready2(q, 1)) return bytes; 300 if (qcam_await_ready2(q, 1))
301 return bytes;
304 lo1 = parport_read_data(q->pport) >> 1; 302 lo1 = parport_read_data(q->pport) >> 1;
305 hi1 = ((parport_read_status(q->pport) >> 3) & 0x1f) ^ 0x10; 303 hi1 = ((parport_read_status(q->pport) >> 3) & 0x1f) ^ 0x10;
306 qcam_set_ack(q, 1); 304 qcam_set_ack(q, 1);
307 if (qcam_await_ready2(q, 0)) return bytes; 305 if (qcam_await_ready2(q, 0))
306 return bytes;
308 lo2 = parport_read_data(q->pport) >> 1; 307 lo2 = parport_read_data(q->pport) >> 1;
309 hi2 = ((parport_read_status(q->pport) >> 3) & 0x1f) ^ 0x10; 308 hi2 = ((parport_read_status(q->pport) >> 3) & 0x1f) ^ 0x10;
310 qcam_set_ack(q, 0); 309 qcam_set_ack(q, 0);
311 r = (lo1 | ((hi1 & 1)<<7)); 310 r = lo1 | ((hi1 & 1) << 7);
312 g = ((hi1 & 0x1e)<<3) | ((hi2 & 0x1e)>>1); 311 g = ((hi1 & 0x1e) << 3) | ((hi2 & 0x1e) >> 1);
313 b = (lo2 | ((hi2 & 1)<<7)); 312 b = lo2 | ((hi2 & 1) << 7);
314 if (force_rgb) { 313 if (force_rgb) {
315 buf[bytes++] = r; 314 buf[bytes++] = r;
316 buf[bytes++] = g; 315 buf[bytes++] = g;
@@ -321,21 +320,20 @@ static unsigned int qcam_read_bytes(struct qcam_device *q, unsigned char *buf, u
321 buf[bytes++] = r; 320 buf[bytes++] = r;
322 } 321 }
323 } 322 }
324 } 323 } else {
325 else
326 {
327 /* It's a unidirectional port */ 324 /* It's a unidirectional port */
328 int i = 0, n = bytes; 325 int i = 0, n = bytes;
329 unsigned char rgb[3]; 326 unsigned char rgb[3];
330 327
331 while (bytes < nbytes) 328 while (bytes < nbytes) {
332 {
333 unsigned int hi, lo; 329 unsigned int hi, lo;
334 330
335 if (qcam_await_ready1(q, 1)) return bytes; 331 if (qcam_await_ready1(q, 1))
332 return bytes;
336 hi = (parport_read_status(q->pport) & 0xf0); 333 hi = (parport_read_status(q->pport) & 0xf0);
337 qcam_set_ack(q, 1); 334 qcam_set_ack(q, 1);
338 if (qcam_await_ready1(q, 0)) return bytes; 335 if (qcam_await_ready1(q, 0))
336 return bytes;
339 lo = (parport_read_status(q->pport) & 0xf0); 337 lo = (parport_read_status(q->pport) & 0xf0);
340 qcam_set_ack(q, 0); 338 qcam_set_ack(q, 0);
341 /* flip some bits */ 339 /* flip some bits */
@@ -374,28 +372,26 @@ static long qc_capture(struct qcam_device *q, char __user *buf, unsigned long le
374 return -EFAULT; 372 return -EFAULT;
375 373
376 /* Wait for camera to become ready */ 374 /* Wait for camera to become ready */
377 for (;;) 375 for (;;) {
378 {
379 int i = qcam_get(q, 41); 376 int i = qcam_get(q, 41);
377
380 if (i == -1) { 378 if (i == -1) {
381 qc_setup(q); 379 qc_setup(q);
382 return -EIO; 380 return -EIO;
383 } 381 }
384 if ((i & 0x80) == 0) 382 if ((i & 0x80) == 0)
385 break; 383 break;
386 else 384 schedule();
387 schedule();
388 } 385 }
389 386
390 if (qcam_set(q, 7, (q->mode | (is_bi_dir?1:0)) + 1)) 387 if (qcam_set(q, 7, (q->mode | (is_bi_dir ? 1 : 0)) + 1))
391 return -EIO; 388 return -EIO;
392 389
393 lines = q->height; 390 lines = q->height;
394 pixelsperline = q->width; 391 pixelsperline = q->width;
395 bitsperxfer = (is_bi_dir) ? 24 : 8; 392 bitsperxfer = (is_bi_dir) ? 24 : 8;
396 393
397 if (is_bi_dir) 394 if (is_bi_dir) {
398 {
399 /* Turn the port around */ 395 /* Turn the port around */
400 parport_data_reverse(q->pport); 396 parport_data_reverse(q->pport);
401 mdelay(3); 397 mdelay(3);
@@ -413,16 +409,17 @@ static long qc_capture(struct qcam_device *q, char __user *buf, unsigned long le
413 409
414 wantlen = lines * pixelsperline * 24 / 8; 410 wantlen = lines * pixelsperline * 24 / 8;
415 411
416 while (wantlen) 412 while (wantlen) {
417 {
418 size_t t, s; 413 size_t t, s;
419 s = (wantlen > BUFSZ)?BUFSZ:wantlen; 414
415 s = (wantlen > BUFSZ) ? BUFSZ : wantlen;
420 t = qcam_read_bytes(q, tmpbuf, s); 416 t = qcam_read_bytes(q, tmpbuf, s);
421 if (outptr < len) 417 if (outptr < len) {
422 {
423 size_t sz = len - outptr; 418 size_t sz = len - outptr;
424 if (sz > t) sz = t; 419
425 if (__copy_to_user(buf+outptr, tmpbuf, sz)) 420 if (sz > t)
421 sz = t;
422 if (__copy_to_user(buf + outptr, tmpbuf, sz))
426 break; 423 break;
427 outptr += sz; 424 outptr += sz;
428 } 425 }
@@ -434,33 +431,31 @@ static long qc_capture(struct qcam_device *q, char __user *buf, unsigned long le
434 431
435 len = outptr; 432 len = outptr;
436 433
437 if (wantlen) 434 if (wantlen) {
438 { 435 printk(KERN_ERR "qcam: short read.\n");
439 printk("qcam: short read.\n");
440 if (is_bi_dir) 436 if (is_bi_dir)
441 parport_data_forward(q->pport); 437 parport_data_forward(q->pport);
442 qc_setup(q); 438 qc_setup(q);
443 return len; 439 return len;
444 } 440 }
445 441
446 if (is_bi_dir) 442 if (is_bi_dir) {
447 {
448 int l; 443 int l;
444
449 do { 445 do {
450 l = qcam_read_bytes(q, tmpbuf, 3); 446 l = qcam_read_bytes(q, tmpbuf, 3);
451 cond_resched(); 447 cond_resched();
452 } while (l && (tmpbuf[0] == 0x7e || tmpbuf[1] == 0x7e || tmpbuf[2] == 0x7e)); 448 } while (l && (tmpbuf[0] == 0x7e || tmpbuf[1] == 0x7e || tmpbuf[2] == 0x7e));
453 if (force_rgb) { 449 if (force_rgb) {
454 if (tmpbuf[0] != 0xe || tmpbuf[1] != 0x0 || tmpbuf[2] != 0xf) 450 if (tmpbuf[0] != 0xe || tmpbuf[1] != 0x0 || tmpbuf[2] != 0xf)
455 printk("qcam: bad EOF\n"); 451 printk(KERN_ERR "qcam: bad EOF\n");
456 } else { 452 } else {
457 if (tmpbuf[0] != 0xf || tmpbuf[1] != 0x0 || tmpbuf[2] != 0xe) 453 if (tmpbuf[0] != 0xf || tmpbuf[1] != 0x0 || tmpbuf[2] != 0xe)
458 printk("qcam: bad EOF\n"); 454 printk(KERN_ERR "qcam: bad EOF\n");
459 } 455 }
460 qcam_set_ack(q, 0); 456 qcam_set_ack(q, 0);
461 if (qcam_await_ready1(q, 1)) 457 if (qcam_await_ready1(q, 1)) {
462 { 458 printk(KERN_ERR "qcam: no ack after EOF\n");
463 printk("qcam: no ack after EOF\n");
464 parport_data_forward(q->pport); 459 parport_data_forward(q->pport);
465 qc_setup(q); 460 qc_setup(q);
466 return len; 461 return len;
@@ -468,27 +463,25 @@ static long qc_capture(struct qcam_device *q, char __user *buf, unsigned long le
468 parport_data_forward(q->pport); 463 parport_data_forward(q->pport);
469 mdelay(3); 464 mdelay(3);
470 qcam_set_ack(q, 1); 465 qcam_set_ack(q, 1);
471 if (qcam_await_ready1(q, 0)) 466 if (qcam_await_ready1(q, 0)) {
472 { 467 printk(KERN_ERR "qcam: no ack to port turnaround\n");
473 printk("qcam: no ack to port turnaround\n");
474 qc_setup(q); 468 qc_setup(q);
475 return len; 469 return len;
476 } 470 }
477 } 471 } else {
478 else
479 {
480 int l; 472 int l;
473
481 do { 474 do {
482 l = qcam_read_bytes(q, tmpbuf, 1); 475 l = qcam_read_bytes(q, tmpbuf, 1);
483 cond_resched(); 476 cond_resched();
484 } while (l && tmpbuf[0] == 0x7e); 477 } while (l && tmpbuf[0] == 0x7e);
485 l = qcam_read_bytes(q, tmpbuf+1, 2); 478 l = qcam_read_bytes(q, tmpbuf + 1, 2);
486 if (force_rgb) { 479 if (force_rgb) {
487 if (tmpbuf[0] != 0xe || tmpbuf[1] != 0x0 || tmpbuf[2] != 0xf) 480 if (tmpbuf[0] != 0xe || tmpbuf[1] != 0x0 || tmpbuf[2] != 0xf)
488 printk("qcam: bad EOF\n"); 481 printk(KERN_ERR "qcam: bad EOF\n");
489 } else { 482 } else {
490 if (tmpbuf[0] != 0xf || tmpbuf[1] != 0x0 || tmpbuf[2] != 0xe) 483 if (tmpbuf[0] != 0xf || tmpbuf[1] != 0x0 || tmpbuf[2] != 0xe)
491 printk("qcam: bad EOF\n"); 484 printk(KERN_ERR "qcam: bad EOF\n");
492 } 485 }
493 } 486 }
494 487
@@ -503,164 +496,166 @@ static long qc_capture(struct qcam_device *q, char __user *buf, unsigned long le
503static long qcam_do_ioctl(struct file *file, unsigned int cmd, void *arg) 496static long qcam_do_ioctl(struct file *file, unsigned int cmd, void *arg)
504{ 497{
505 struct video_device *dev = video_devdata(file); 498 struct video_device *dev = video_devdata(file);
506 struct qcam_device *qcam=(struct qcam_device *)dev; 499 struct qcam_device *qcam = (struct qcam_device *)dev;
507 500
508 switch(cmd) 501 switch (cmd) {
502 case VIDIOCGCAP:
509 { 503 {
510 case VIDIOCGCAP: 504 struct video_capability *b = arg;
511 { 505
512 struct video_capability *b = arg; 506 strcpy(b->name, "Quickcam");
513 strcpy(b->name, "Quickcam"); 507 b->type = VID_TYPE_CAPTURE | VID_TYPE_SCALES;
514 b->type = VID_TYPE_CAPTURE|VID_TYPE_SCALES; 508 b->channels = 1;
515 b->channels = 1; 509 b->audios = 0;
516 b->audios = 0; 510 b->maxwidth = 320;
517 b->maxwidth = 320; 511 b->maxheight = 240;
518 b->maxheight = 240; 512 b->minwidth = 80;
519 b->minwidth = 80; 513 b->minheight = 60;
520 b->minheight = 60; 514 return 0;
521 return 0; 515 }
522 } 516 case VIDIOCGCHAN:
523 case VIDIOCGCHAN: 517 {
524 { 518 struct video_channel *v = arg;
525 struct video_channel *v = arg; 519
526 if(v->channel!=0) 520 if (v->channel != 0)
527 return -EINVAL; 521 return -EINVAL;
528 v->flags=0; 522 v->flags = 0;
529 v->tuners=0; 523 v->tuners = 0;
530 /* Good question.. its composite or SVHS so.. */ 524 /* Good question.. its composite or SVHS so.. */
531 v->type = VIDEO_TYPE_CAMERA; 525 v->type = VIDEO_TYPE_CAMERA;
532 strcpy(v->name, "Camera"); 526 strcpy(v->name, "Camera");
533 return 0; 527 return 0;
534 } 528 }
535 case VIDIOCSCHAN: 529 case VIDIOCSCHAN:
536 { 530 {
537 struct video_channel *v = arg; 531 struct video_channel *v = arg;
538 if(v->channel!=0) 532
539 return -EINVAL; 533 if (v->channel != 0)
540 return 0; 534 return -EINVAL;
541 } 535 return 0;
542 case VIDIOCGTUNER: 536 }
543 { 537 case VIDIOCGTUNER:
544 struct video_tuner *v = arg; 538 {
545 if(v->tuner) 539 struct video_tuner *v = arg;
546 return -EINVAL; 540
547 memset(v,0,sizeof(*v)); 541 if (v->tuner)
548 strcpy(v->name, "Format"); 542 return -EINVAL;
549 v->mode = VIDEO_MODE_AUTO; 543 memset(v, 0, sizeof(*v));
550 return 0; 544 strcpy(v->name, "Format");
551 } 545 v->mode = VIDEO_MODE_AUTO;
552 case VIDIOCSTUNER: 546 return 0;
553 { 547 }
554 struct video_tuner *v = arg; 548 case VIDIOCSTUNER:
555 if(v->tuner) 549 {
556 return -EINVAL; 550 struct video_tuner *v = arg;
557 if(v->mode!=VIDEO_MODE_AUTO) 551
558 return -EINVAL; 552 if (v->tuner)
559 return 0; 553 return -EINVAL;
560 } 554 if (v->mode != VIDEO_MODE_AUTO)
561 case VIDIOCGPICT: 555 return -EINVAL;
562 { 556 return 0;
563 struct video_picture *p = arg; 557 }
564 p->colour=0x8000; 558 case VIDIOCGPICT:
565 p->hue=0x8000; 559 {
566 p->brightness=qcam->brightness<<8; 560 struct video_picture *p = arg;
567 p->contrast=qcam->contrast<<8; 561
568 p->whiteness=qcam->whitebal<<8; 562 p->colour = 0x8000;
569 p->depth=24; 563 p->hue = 0x8000;
570 p->palette=VIDEO_PALETTE_RGB24; 564 p->brightness = qcam->brightness << 8;
571 return 0; 565 p->contrast = qcam->contrast << 8;
566 p->whiteness = qcam->whitebal << 8;
567 p->depth = 24;
568 p->palette = VIDEO_PALETTE_RGB24;
569 return 0;
570 }
571 case VIDIOCSPICT:
572 {
573 struct video_picture *p = arg;
574
575 /*
576 * Sanity check args
577 */
578 if (p->depth != 24 || p->palette != VIDEO_PALETTE_RGB24)
579 return -EINVAL;
580
581 /*
582 * Now load the camera.
583 */
584 qcam->brightness = p->brightness >> 8;
585 qcam->contrast = p->contrast >> 8;
586 qcam->whitebal = p->whiteness >> 8;
587
588 mutex_lock(&qcam->lock);
589 parport_claim_or_block(qcam->pdev);
590 qc_setup(qcam);
591 parport_release(qcam->pdev);
592 mutex_unlock(&qcam->lock);
593 return 0;
594 }
595 case VIDIOCSWIN:
596 {
597 struct video_window *vw = arg;
598
599 if (vw->flags)
600 return -EINVAL;
601 if (vw->clipcount)
602 return -EINVAL;
603 if (vw->height < 60 || vw->height > 240)
604 return -EINVAL;
605 if (vw->width < 80 || vw->width > 320)
606 return -EINVAL;
607
608 qcam->width = 80;
609 qcam->height = 60;
610 qcam->mode = QC_DECIMATION_4;
611
612 if (vw->width >= 160 && vw->height >= 120) {
613 qcam->width = 160;
614 qcam->height = 120;
615 qcam->mode = QC_DECIMATION_2;
572 } 616 }
573 case VIDIOCSPICT: 617 if (vw->width >= 320 && vw->height >= 240) {
574 { 618 qcam->width = 320;
575 struct video_picture *p = arg; 619 qcam->height = 240;
576 620 qcam->mode = QC_DECIMATION_1;
577 /*
578 * Sanity check args
579 */
580 if (p->depth != 24 || p->palette != VIDEO_PALETTE_RGB24)
581 return -EINVAL;
582
583 /*
584 * Now load the camera.
585 */
586 qcam->brightness = p->brightness>>8;
587 qcam->contrast = p->contrast>>8;
588 qcam->whitebal = p->whiteness>>8;
589
590 mutex_lock(&qcam->lock);
591 parport_claim_or_block(qcam->pdev);
592 qc_setup(qcam);
593 parport_release(qcam->pdev);
594 mutex_unlock(&qcam->lock);
595 return 0;
596 } 621 }
597 case VIDIOCSWIN: 622 qcam->mode |= QC_MILLIONS;
598 {
599 struct video_window *vw = arg;
600
601 if(vw->flags)
602 return -EINVAL;
603 if(vw->clipcount)
604 return -EINVAL;
605 if(vw->height<60||vw->height>240)
606 return -EINVAL;
607 if(vw->width<80||vw->width>320)
608 return -EINVAL;
609
610 qcam->width = 80;
611 qcam->height = 60;
612 qcam->mode = QC_DECIMATION_4;
613
614 if(vw->width>=160 && vw->height>=120)
615 {
616 qcam->width = 160;
617 qcam->height = 120;
618 qcam->mode = QC_DECIMATION_2;
619 }
620 if(vw->width>=320 && vw->height>=240)
621 {
622 qcam->width = 320;
623 qcam->height = 240;
624 qcam->mode = QC_DECIMATION_1;
625 }
626 qcam->mode |= QC_MILLIONS;
627#if 0 623#if 0
628 if(vw->width>=640 && vw->height>=480) 624 if (vw->width >= 640 && vw->height >= 480) {
629 { 625 qcam->width = 640;
630 qcam->width = 640; 626 qcam->height = 480;
631 qcam->height = 480; 627 qcam->mode = QC_BILLIONS | QC_DECIMATION_1;
632 qcam->mode = QC_BILLIONS | QC_DECIMATION_1;
633 }
634#endif
635 /* Ok we figured out what to use from our
636 wide choice */
637 mutex_lock(&qcam->lock);
638 parport_claim_or_block(qcam->pdev);
639 qc_setup(qcam);
640 parport_release(qcam->pdev);
641 mutex_unlock(&qcam->lock);
642 return 0;
643 }
644 case VIDIOCGWIN:
645 {
646 struct video_window *vw = arg;
647 memset(vw, 0, sizeof(*vw));
648 vw->width=qcam->width;
649 vw->height=qcam->height;
650 return 0;
651 } 628 }
652 case VIDIOCKEY: 629#endif
653 return 0; 630 /* Ok we figured out what to use from our
654 case VIDIOCCAPTURE: 631 wide choice */
655 case VIDIOCGFBUF: 632 mutex_lock(&qcam->lock);
656 case VIDIOCSFBUF: 633 parport_claim_or_block(qcam->pdev);
657 case VIDIOCGFREQ: 634 qc_setup(qcam);
658 case VIDIOCSFREQ: 635 parport_release(qcam->pdev);
659 case VIDIOCGAUDIO: 636 mutex_unlock(&qcam->lock);
660 case VIDIOCSAUDIO: 637 return 0;
661 return -EINVAL; 638 }
662 default: 639 case VIDIOCGWIN:
663 return -ENOIOCTLCMD; 640 {
641 struct video_window *vw = arg;
642 memset(vw, 0, sizeof(*vw));
643 vw->width = qcam->width;
644 vw->height = qcam->height;
645 return 0;
646 }
647 case VIDIOCKEY:
648 return 0;
649 case VIDIOCCAPTURE:
650 case VIDIOCGFBUF:
651 case VIDIOCSFBUF:
652 case VIDIOCGFREQ:
653 case VIDIOCSFREQ:
654 case VIDIOCGAUDIO:
655 case VIDIOCSAUDIO:
656 return -EINVAL;
657 default:
658 return -ENOIOCTLCMD;
664 } 659 }
665 return 0; 660 return 0;
666} 661}
@@ -675,13 +670,13 @@ static ssize_t qcam_read(struct file *file, char __user *buf,
675 size_t count, loff_t *ppos) 670 size_t count, loff_t *ppos)
676{ 671{
677 struct video_device *v = video_devdata(file); 672 struct video_device *v = video_devdata(file);
678 struct qcam_device *qcam=(struct qcam_device *)v; 673 struct qcam_device *qcam = (struct qcam_device *)v;
679 int len; 674 int len;
680 675
681 mutex_lock(&qcam->lock); 676 mutex_lock(&qcam->lock);
682 parport_claim_or_block(qcam->pdev); 677 parport_claim_or_block(qcam->pdev);
683 /* Probably should have a semaphore against multiple users */ 678 /* Probably should have a semaphore against multiple users */
684 len = qc_capture(qcam, buf,count); 679 len = qc_capture(qcam, buf, count);
685 parport_release(qcam->pdev); 680 parport_release(qcam->pdev);
686 mutex_unlock(&qcam->lock); 681 mutex_unlock(&qcam->lock);
687 return len; 682 return len;
@@ -713,8 +708,7 @@ static const struct v4l2_file_operations qcam_fops = {
713 .read = qcam_read, 708 .read = qcam_read,
714}; 709};
715 710
716static struct video_device qcam_template= 711static struct video_device qcam_template = {
717{
718 .name = "Colour QuickCam", 712 .name = "Colour QuickCam",
719 .fops = &qcam_fops, 713 .fops = &qcam_fops,
720 .release = video_device_release_empty, 714 .release = video_device_release_empty,
@@ -727,17 +721,16 @@ static struct qcam_device *qcam_init(struct parport *port)
727 struct qcam_device *q; 721 struct qcam_device *q;
728 722
729 q = kmalloc(sizeof(struct qcam_device), GFP_KERNEL); 723 q = kmalloc(sizeof(struct qcam_device), GFP_KERNEL);
730 if(q==NULL) 724 if (q == NULL)
731 return NULL; 725 return NULL;
732 726
733 q->pport = port; 727 q->pport = port;
734 q->pdev = parport_register_device(port, "c-qcam", NULL, NULL, 728 q->pdev = parport_register_device(port, "c-qcam", NULL, NULL,
735 NULL, 0, NULL); 729 NULL, 0, NULL);
736 730
737 q->bidirectional = (q->pport->modes & PARPORT_MODE_TRISTATE)?1:0; 731 q->bidirectional = (q->pport->modes & PARPORT_MODE_TRISTATE) ? 1 : 0;
738 732
739 if (q->pdev == NULL) 733 if (q->pdev == NULL) {
740 {
741 printk(KERN_ERR "c-qcam: couldn't register for %s.\n", 734 printk(KERN_ERR "c-qcam: couldn't register for %s.\n",
742 port->name); 735 port->name);
743 kfree(q); 736 kfree(q);
@@ -765,12 +758,11 @@ static int init_cqcam(struct parport *port)
765{ 758{
766 struct qcam_device *qcam; 759 struct qcam_device *qcam;
767 760
768 if (parport[0] != -1) 761 if (parport[0] != -1) {
769 {
770 /* The user gave specific instructions */ 762 /* The user gave specific instructions */
771 int i, found = 0; 763 int i, found = 0;
772 for (i = 0; i < MAX_CAMS && parport[i] != -1; i++) 764
773 { 765 for (i = 0; i < MAX_CAMS && parport[i] != -1; i++) {
774 if (parport[0] == port->number) 766 if (parport[0] == port->number)
775 found = 1; 767 found = 1;
776 } 768 }
@@ -782,15 +774,14 @@ static int init_cqcam(struct parport *port)
782 return -ENOSPC; 774 return -ENOSPC;
783 775
784 qcam = qcam_init(port); 776 qcam = qcam_init(port);
785 if (qcam==NULL) 777 if (qcam == NULL)
786 return -ENODEV; 778 return -ENODEV;
787 779
788 parport_claim_or_block(qcam->pdev); 780 parport_claim_or_block(qcam->pdev);
789 781
790 qc_reset(qcam); 782 qc_reset(qcam);
791 783
792 if (probe && qc_detect(qcam)==0) 784 if (probe && qc_detect(qcam) == 0) {
793 {
794 parport_release(qcam->pdev); 785 parport_release(qcam->pdev);
795 parport_unregister_device(qcam->pdev); 786 parport_unregister_device(qcam->pdev);
796 kfree(qcam); 787 kfree(qcam);
@@ -840,14 +831,14 @@ static struct parport_driver cqcam_driver = {
840 .detach = cq_detach, 831 .detach = cq_detach,
841}; 832};
842 833
843static int __init cqcam_init (void) 834static int __init cqcam_init(void)
844{ 835{
845 printk(BANNER "\n"); 836 printk(BANNER "\n");
846 837
847 return parport_register_driver(&cqcam_driver); 838 return parport_register_driver(&cqcam_driver);
848} 839}
849 840
850static void __exit cqcam_cleanup (void) 841static void __exit cqcam_cleanup(void)
851{ 842{
852 unsigned int i; 843 unsigned int i;
853 844
@@ -862,9 +853,9 @@ MODULE_DESCRIPTION(BANNER);
862MODULE_LICENSE("GPL"); 853MODULE_LICENSE("GPL");
863 854
864/* FIXME: parport=auto would never have worked, surely? --RR */ 855/* FIXME: parport=auto would never have worked, surely? --RR */
865MODULE_PARM_DESC(parport ,"parport=<auto|n[,n]...> for port detection method\n\ 856MODULE_PARM_DESC(parport, "parport=<auto|n[,n]...> for port detection method\n"
866probe=<0|1|2> for camera detection method\n\ 857 "probe=<0|1|2> for camera detection method\n"
867force_rgb=<0|1> for RGB data format (default BGR)"); 858 "force_rgb=<0|1> for RGB data format (default BGR)");
868module_param_array(parport, int, NULL, 0); 859module_param_array(parport, int, NULL, 0);
869module_param(probe, int, 0); 860module_param(probe, int, 0);
870module_param(force_rgb, bool, 0); 861module_param(force_rgb, bool, 0);