#include #include #include #include #include void print_data(struct qcam *data) { fprintf(stderr, "version=%d, (%d,%d) at (%d,%d) @%dbpp " "zoom=%d, exp=%d, b/w/c=%d/%d/%d\n", data->qc_version, data->qc_xsize, data->qc_ysize, data->qc_xorigin, data->qc_yorigin, data->qc_bpp, data->qc_zoom, data->qc_exposure, data->qc_brightness, data->qc_whitebalance, data->qc_contrast); } usage(void) { fprintf(stderr, "usage: qcamcontrol [-p port] [-x xsize] [-y ysize] " "[-z zoom] [-d depth]\n" " [-b brightness] [-w whitebal] " "[-c contrast] [-e exposure]\n"); exit(2); } main(int argc, char **argv) { struct qcam info; int fd, len; size_t bytes; char opt; static char buffer[QC_MAX_XSIZE*QC_MAX_YSIZE]; char *port = "/dev/qcam0"; int x_size, y_size, zoom, depth, brightness, whitebalance, contrast, exposure; /* * Default everything to unset. */ x_size = y_size = zoom = depth = brightness = whitebalance = contrast = exposure = -1; while ((opt = getopt(argc, argv, "p:x:y:z:d:b;w:c:e:")) != EOF) { switch (opt) { case 'p': port = optarg; break; case 'x': x_size = atoi(optarg); if (x_size > QC_MAX_XSIZE) { fprintf(stderr, "x size too large (max %d)\n", QC_MAX_XSIZE); exit(2); } break; case 'y': y_size = atoi(optarg); if (y_size > QC_MAX_YSIZE) { fprintf(stderr, "x size too large (max %d)\n", QC_MAX_YSIZE); exit(2); } break; case 'z': zoom = atoi(optarg); if (zoom > QC_ZOOM_200) { fprintf(stderr, "zoom too large (max %d)\n", QC_ZOOM_200); exit(2); } break; case 'd': depth = atoi(optarg); if (depth != 4 && depth != 6) { fprintf(stderr, "invalid depth (4 or 6)\n"); exit(2); } break; case 'b': brightness = atoi(optarg); if (brightness > 255) { fprintf(stderr, "bad brightness (max 255)\n"); exit(2); } break; case 'w': whitebalance = atoi(optarg); if (whitebalance > 255) { fprintf(stderr, "bad white balance (max 255)\n"); exit(2); } break; case 'c': contrast = atoi(optarg); if (contrast > 255) { fprintf(stderr, "bad contrast (max 255)\n"); exit(2); } break; case 'e': exposure = atoi(optarg); if (exposure < 100) { fprintf(stderr, "bad exposure (min 100)\n"); exit(2); } break; default: usage(); } argc--; argv++; } /* open device */ if ((fd = open(port, O_RDONLY)) < 0) { perror("open"); exit(1); } if (ioctl(fd, QC_GET, &info) < 0) { /* read in default info */ perror("ioctl(QC_GET)"); exit(1); } if (x_size > -1) info.qc_xsize = x_size; if (y_size > -1) info.qc_ysize = y_size; if (depth > -1) info.qc_bpp = depth; if (zoom > -1) info.qc_zoom = zoom; if (brightness > -1) info.qc_brightness = brightness; if (whitebalance > -1) info.qc_whitebalance = whitebalance; if (contrast > -1) info.qc_contrast = contrast; if (exposure > -1) info.qc_exposure = exposure; /* * make sure we're in sync with the kernel version of the driver * ioctl structure */ info.qc_version = QC_IOCTL_VERSION; if (ioctl(fd, QC_SET, &info) < 0) { perror("ioctl(QC_SET)"); exit(1); } /* * Tell us what the kernel thinks we're asking for */ if (ioctl(fd, QC_GET, &info) < 0) { perror("ioctl(QC_SET)"); exit(1); } print_data(&info); /* * Grab a frame -- a single read will always work, but give a * particularly paranoid example. */ len = info.qc_xsize * info.qc_ysize; while (len) { bytes = read(fd, buffer, len); if (bytes < 0) { perror("read"); exit(1); } len -= bytes; if (bytes == 0) exit(0); } /* * Write the frame to stdout as a PGM image. */ fprintf(stdout, "P5\n%d %d\n%d\n", info.qc_xsize, info.qc_ysize, (1<