/*
* V4L2 video stereo capture example with openCV
* htpp://lowcost-robot.org
* compile : g++ V4L2_cap.cpp -o V4L2_cap `pkg-config --libs --cflags opencv`
*/
#include "opencv/cv.h"
#include "opencv/highgui.h"
#include "opencv/cvaux.h"
#include "cxmisc.h"
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <assert.h>
#include <ctype.h>
#include <vector>
#include <string>
#include <algorithm>
#include <pthread.h>
#include <sys/resource.h>
#include <getopt.h> /* getopt_long() */
#include <fcntl.h> /* low-level i/o */
#include <unistd.h>
#include <errno.h>
#include <malloc.h>
#include <sys/stat.h>
#include <sys/types.h>
#include <sys/time.h>
#include <sys/mman.h>
#include <sys/ioctl.h>
#include <asm/types.h> /* for videodev2.h */
#include <linux/videodev2.h>
using namespace std;
#define CLEAR(x) memset (&(x), 0, sizeof (x))
/* This is the critical section object (statically allocated). */
static pthread_mutex_t cs_mutex = PTHREAD_MUTEX_INITIALIZER;
typedef enum {
IO_METHOD_READ,
IO_METHOD_MMAP,
IO_METHOD_USERPTR,
} io_method;
struct buffer {
void * start;
size_t length;
};
static char * dev_name = NULL;
static char * dev_name2 = NULL;
int width_cap=352;
int height_cap=288;
static io_method io = IO_METHOD_MMAP;
static int fd = -1;
static int fd2 = -1;
struct buffer * buffers = NULL;
static unsigned int n_buffers = 0;
struct buffer * buffers2 = NULL;
static unsigned int n_buffers2 = 0;
IplImage* frame=0;
IplImage* frame2=0;
IplImage* grayImg=0;
IplImage* cannyImg=0;
bool exit_app=false;
static void
errno_exit (const char * s)
{
fprintf (stderr, "%s error %d, %s\n",
s, errno, strerror (errno));
exit (EXIT_FAILURE);
}
static int
xioctl (int fd,
int request,
void * arg)
{
int r;
do r = ioctl (fd, request, arg);
while (-1 == r && (EINTR == errno ));
return r;
}
#define CLAMP(x) if (x < 0) {x = 0;} if (x > 255) {x = 255;}
static void convert_yuyv_to_rgb(char *raw_buf, char *proc_buf,char *raw_buf2, char *proc_buf2)
{
int x, y;
for(y = 0; y < height_cap; ++y) {
for(x = 0; x < width_cap; x+=2) {
int i = (width_cap * y + x);
int y1 = raw_buf[2*i];
int u = raw_buf[2*i+1];
int y2 = raw_buf[2*i+2];
int v = raw_buf[2*i+3];
int y1_ = raw_buf2[2*i];
int u_ = raw_buf2[2*i+1];
int y2_ = raw_buf2[2*i+2];
int v_ = raw_buf2[2*i+3];
int r1 = (y1-16)*1164/1000 + (v-128)*1596/1000;
int g1 = (y1-16)*1164/1000 - (v-128)*813/1000 - (u-128)*391/1000;
int b1 = (y1-16)*1164/1000 + (u-128)*2018/1000;
int r2 = (y2-16)*1164/1000 + (v-128)*1596/1000;
int g2 = (y2-16)*1164/1000 - (v-128)*813/1000 - (u-128)*391/1000;
int b2 = (y2-16)*1164/1000 + (u-128)*2018/1000;
int r1_ = (y1_-16)*1164/1000 + (v_-128)*1596/1000;
int g1_ = (y1_-16)*1164/1000 - (v_-128)*813/1000 - (u_-128)*391/1000;
int b1_ = (y1_-16)*1164/1000 + (u_-128)*2018/1000;
int r2_ = (y2_-16)*1164/1000 + (v_-128)*1596/1000;
int g2_ = (y2_-16)*1164/1000 - (v_-128)*813/1000 - (u_-128)*391/1000;
int b2_ = (y2_-16)*1164/1000 + (u_-128)*2018/1000;
CLAMP(r1)
CLAMP(g1)
CLAMP(b1)
CLAMP(r2)
CLAMP(g2)
CLAMP(b2)
CLAMP(r1_)
CLAMP(g1_)
CLAMP(b1_)
CLAMP(r2_)
CLAMP(g2_)
CLAMP(b2_)
proc_buf[3*i] =b1;
proc_buf[3*i+1]=g1;
proc_buf[3*i+2]=r1;
proc_buf[3*i+3]=b2;
proc_buf[3*i+4]=g2;
proc_buf[3*i+5]=r2;
proc_buf2[3*i] =b1_;
proc_buf2[3*i+1]=g1_;
proc_buf2[3*i+2]=r1_;
proc_buf2[3*i+3]=b2_;
proc_buf2[3*i+4]=g2_;
proc_buf2[3*i+5]=r2_;
}
}
}
#undef CLAMP
static void
process_image(const void * p,const void * p2)
{
/* Enter the critical section -- other threads are locked out */
pthread_mutex_lock( &cs_mutex );
convert_yuyv_to_rgb((char*)p,(char*)frame->imageData,(char*)p2,(char*)frame2->imageData);
cvCvtColor( frame, grayImg, CV_BGR2GRAY );
// canny edge detection
cvCanny(grayImg, cannyImg, 50, 150, 3);
cvShowImage( "Canny", cannyImg);
cvShowImage( "Cam1", frame);
cvShowImage( "Cam2", frame2);
if( (cvWaitKey(5) & 255) == 27 ) exit_app=true;
/*Leave the critical section -- other threads can now pthread_mutex_lock() */
pthread_mutex_unlock( &cs_mutex );
}
static int
read_frame(void)
{
struct v4l2_buffer buf;
struct v4l2_buffer buf2;
unsigned int i;
switch (io) {
case IO_METHOD_READ:
if (-1 == read (fd, buffers[0].start, buffers[0].length)) {
switch (errno) {
case EAGAIN:
return 0;
case EIO:
/* Could ignore EIO, see spec. */
/* fall through */
default:
errno_exit ("read");
}
}
if (-1 == read (fd2, buffers2[0].start, buffers2[0].length)) {
switch (errno) {
case EAGAIN:
return 0;
case EIO:
/* Could ignore EIO, see spec. */
/* fall through */
default:
errno_exit ("read2");
}
}
process_image (buffers[0].start,buffers2[0].start);
break;
case IO_METHOD_MMAP:
CLEAR (buf);
buf.type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
buf.memory = V4L2_MEMORY_MMAP;
if (-1 == xioctl (fd, VIDIOC_DQBUF, &buf)) {
switch (errno) {
case EAGAIN:
return 0;
case EIO:
/* Could ignore EIO, see spec. */
/* fall through */
default:
errno_exit ("VIDIOC_DQBUF");
}
}
CLEAR (buf2);
buf2.type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
buf2.memory = V4L2_MEMORY_MMAP;
if (-1 == xioctl (fd2, VIDIOC_DQBUF, &buf2)) {
switch (errno) {
case EAGAIN:
return 0;
case EIO:
/* Could ignore EIO, see spec. */
/* fall through */
default:
errno_exit ("VIDIOC_DQBUF2");
}
}
assert (buf.index < n_buffers);
assert (buf2.index < n_buffers);
process_image (buffers[buf.index].start,buffers2[buf.index].start);
if (-1 == xioctl (fd, VIDIOC_QBUF, &buf))
errno_exit ("VIDIOC_QBUF");
if (-1 == xioctl (fd2, VIDIOC_QBUF, &buf2))
errno_exit ("VIDIOC_QBUF2");
break;
case IO_METHOD_USERPTR:
CLEAR (buf);
buf.type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
buf.memory = V4L2_MEMORY_USERPTR;
if (-1 == xioctl (fd, VIDIOC_DQBUF, &buf)) {
switch (errno) {
case EAGAIN:
return 0;
case EIO:
/* Could ignore EIO, see spec. */
/* fall through */
default:
errno_exit ("VIDIOC_DQBUF");
}
}
CLEAR (buf2);
buf2.type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
buf2.memory = V4L2_MEMORY_USERPTR;
if (-1 == xioctl (fd2, VIDIOC_DQBUF, &buf2)) {
switch (errno) {
case EAGAIN:
return 0;
case EIO:
/* Could ignore EIO, see spec. */
/* fall through */
default:
errno_exit ("VIDIOC_DQBUF2");
}
}
for (i = 0; i < n_buffers; ++i)
if (buf.m.userptr == (unsigned long) buffers[i].start
&& buf.length == buffers[i].length)
break;
assert (i < n_buffers);
for (i = 0; i < n_buffers; ++i)
if (buf2.m.userptr == (unsigned long) buffers2[i].start
&& buf2.length == buffers2[i].length)
break;
assert (i < n_buffers);
process_image ((void *) buf.m.userptr,(void *) buf2.m.userptr);
if (-1 == xioctl (fd, VIDIOC_QBUF, &buf))
errno_exit ("VIDIOC_QBUF");
if (-1 == xioctl (fd2, VIDIOC_QBUF, &buf2))
errno_exit ("VIDIOC_QBUF");
break;
}
return 1;
}
static void
mainloop (void)
{
while (!exit_app) {
for (;;) {
fd_set fds;
struct timeval tv;
int r;
FD_ZERO (&fds);
FD_SET (fd, &fds);
/* Timeout. */
tv.tv_sec = 5;
tv.tv_usec = 1;
r = select (fd + 1, &fds, NULL, NULL, &tv);
if (-1 == r) {
if (EINTR == errno)
continue;
errno_exit ("select");
}
if (0 == r) {
fprintf (stderr, "select timeout\n");
exit (EXIT_FAILURE);
}
if (read_frame)
break;
/* EAGAIN - continue select loop. */
}
}
}
static void
stop_capturing (void)
{
enum v4l2_buf_type type;
switch (io) {
case IO_METHOD_READ:
/* Nothing to do. */
break;
case IO_METHOD_MMAP:
case IO_METHOD_USERPTR:
type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
if (-1 == xioctl (fd, VIDIOC_STREAMOFF, &type))
errno_exit ("VIDIOC_STREAMOFF");
if (-1 == xioctl (fd2, VIDIOC_STREAMOFF, &type))
errno_exit ("VIDIOC_STREAMOFF");
break;
}
}
static void
start_capturing (void)
{
unsigned int i;
enum v4l2_buf_type type;
switch (io) {
case IO_METHOD_READ:
/* Nothing to do. */
break;
case IO_METHOD_MMAP:
for (i = 0; i < n_buffers; ++i) {
struct v4l2_buffer buf;
CLEAR (buf);
buf.type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
buf.memory = V4L2_MEMORY_MMAP;
buf.index = i;
if (-1 == xioctl (fd, VIDIOC_QBUF, &buf))
errno_exit ("VIDIOC_QBUF");
if (-1 == xioctl (fd2, VIDIOC_QBUF, &buf))
errno_exit ("VIDIOC_QBUF");
}
type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
if (-1 == xioctl (fd, VIDIOC_STREAMON, &type))
errno_exit ("VIDIOC_STREAMON");
if (-1 == xioctl (fd2, VIDIOC_STREAMON, &type))
errno_exit ("VIDIOC_STREAMON");
break;
case IO_METHOD_USERPTR:
for (i = 0; i < n_buffers; ++i) {
struct v4l2_buffer buf;
struct v4l2_buffer buf2;
CLEAR (buf);
buf.type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
buf.memory = V4L2_MEMORY_USERPTR;
buf.index = i;
buf.m.userptr = (unsigned long) buffers[i].start;
buf.length = buffers[i].length;
CLEAR (buf2);
buf2.type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
buf2.memory = V4L2_MEMORY_USERPTR;
buf2.index = i;
buf2.m.userptr = (unsigned long) buffers2[i].start;
buf2.length = buffers2[i].length;
if (-1 == xioctl (fd, VIDIOC_QBUF, &buf))
errno_exit ("VIDIOC_QBUF");
if (-1 == xioctl (fd2, VIDIOC_QBUF, &buf2))
errno_exit ("VIDIOC_QBUF");
}
type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
if (-1 == xioctl (fd, VIDIOC_STREAMON, &type))
errno_exit ("VIDIOC_STREAMON");
if (-1 == xioctl (fd2, VIDIOC_STREAMON, &type))
errno_exit ("VIDIOC_STREAMON");
break;
}
}
static void
uninit_device (void)
{
unsigned int i;
switch (io) {
case IO_METHOD_READ:
free (buffers[0].start);
free (buffers2[0].start);
break;
case IO_METHOD_MMAP:
for (i = 0; i < n_buffers; ++i)
{
if (-1 == munmap (buffers[i].start, buffers[i].length))
errno_exit ("munmap");
if (-1 == munmap (buffers2[i].start, buffers2[i].length))
errno_exit ("munmap");
}
break;
case IO_METHOD_USERPTR:
for (i = 0; i < n_buffers; ++i)
{
free (buffers[i].start);
free (buffers2[i].start);
}
break;
}
free (buffers);
free (buffers2);
}
static void
init_read (unsigned int buffer_size)
{
char* buff;
buff = (char*) calloc (1, sizeof (*buffers));
buffers = (buffer*) buff;
if (!buffers) {
fprintf (stderr, "Out of memory\n");
exit (EXIT_FAILURE);
}
buffers[0].length = buffer_size;
buffers[0].start = malloc (buffer_size);
if (!buffers[0].start) {
fprintf (stderr, "Out of memory\n");
exit (EXIT_FAILURE);
}
//===========================================================
buff = (char*) calloc (1, sizeof (*buffers));
buffers2 = (buffer*) buff;
if (!buffers2) {
fprintf (stderr, "Out of memory\n");
exit (EXIT_FAILURE);
}
buffers2[0].length = buffer_size;
buffers2[0].start = malloc (buffer_size);
if (!buffers2[0].start) {
fprintf (stderr, "Out of memory\n");
exit (EXIT_FAILURE);
}
}
static void
init_mmap (void)
{
struct v4l2_requestbuffers req;
CLEAR (req);
req.count = 4;
req.type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
req.memory = V4L2_MEMORY_MMAP;
if (-1 == xioctl (fd, VIDIOC_REQBUFS, &req)) {
if (EINVAL == errno) {
fprintf (stderr, "%s does not support "
"memory mapping\n", dev_name);
exit (EXIT_FAILURE);
} else {
errno_exit ("VIDIOC_REQBUFS");
}
}
if (req.count < 2) {
fprintf (stderr, "Insufficient buffer memory on %s\n",
dev_name);
exit (EXIT_FAILURE);
}
char* buff;
buff = (char*) calloc (req.count, sizeof (*buffers));
buffers = (buffer*) buff;
if (!buffers) {
fprintf (stderr, "Out of memory\n");
exit (EXIT_FAILURE);
}
for (n_buffers = 0; n_buffers < req.count; ++n_buffers) {
struct v4l2_buffer buf;
CLEAR (buf);
buf.type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
buf.memory = V4L2_MEMORY_MMAP;
buf.index = n_buffers;
if (-1 == xioctl (fd, VIDIOC_QUERYBUF, &buf))
errno_exit ("VIDIOC_QUERYBUF");
buffers[n_buffers].length = buf.length;
buffers[n_buffers].start =
mmap (NULL /* start anywhere */,
buf.length,
PROT_READ | PROT_WRITE /* required */,
MAP_SHARED /* recommended */,
fd, buf.m.offset);
if (MAP_FAILED == buffers[n_buffers].start)
errno_exit ("mmap");
}
//========================================================================
struct v4l2_requestbuffers req2;
CLEAR (req2);
req2.count = 4;
req2.type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
req2.memory = V4L2_MEMORY_MMAP;
if (-1 == xioctl (fd2, VIDIOC_REQBUFS, &req2)) {
if (EINVAL == errno) {
fprintf (stderr, "%s does not support "
"memory mapping\n", dev_name);
exit (EXIT_FAILURE);
} else {
errno_exit ("VIDIOC_REQBUFS2");
}
}
if (req2.count < 2) {
fprintf (stderr, "Insufficient buffer memory on %s\n",
dev_name);
exit (EXIT_FAILURE);
}
buff = (char*) calloc (req2.count, sizeof (*buffers));
buffers2 = (buffer*) buff;
if (!buffers2) {
fprintf (stderr, "Out of memory2\n");
exit (EXIT_FAILURE);
}
for (n_buffers = 0; n_buffers < req2.count; ++n_buffers) {
struct v4l2_buffer buf2;
CLEAR (buf2);
buf2.type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
buf2.memory = V4L2_MEMORY_MMAP;
buf2.index = n_buffers;
if (-1 == xioctl (fd2, VIDIOC_QUERYBUF, &buf2))
errno_exit ("VIDIOC_QUERYBUF");
buffers2[n_buffers].length = buf2.length;
buffers2[n_buffers].start =
mmap (NULL /* start anywhere */,
buf2.length,
PROT_READ | PROT_WRITE /* required */,
MAP_SHARED /* recommended */,
fd2, buf2.m.offset);
if (MAP_FAILED == buffers2[n_buffers].start)
errno_exit ("mmap2");
}
}
static void
init_userp (unsigned int buffer_size)
{
struct v4l2_requestbuffers req;
unsigned int page_size;
page_size = getpagesize ();
buffer_size = (buffer_size + page_size - 1) & ~(page_size - 1);
CLEAR (req);
req.count = 4;
req.type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
req.memory = V4L2_MEMORY_USERPTR;
if (-1 == xioctl (fd, VIDIOC_REQBUFS, &req)) {
if (EINVAL == errno) {
fprintf (stderr, "%s does not support "
"user pointer i/o\n", dev_name);
exit (EXIT_FAILURE);
} else {
errno_exit ("VIDIOC_REQBUFS");
}
}
char* buff;
buff = (char*) calloc (4, sizeof (*buffers));
buffers = (buffer*) buff;
if (!buffers) {
fprintf (stderr, "Out of memory\n");
exit (EXIT_FAILURE);
}
for (n_buffers = 0; n_buffers < 4; ++n_buffers) {
buffers[n_buffers].length = buffer_size;
buffers[n_buffers].start = memalign (/* boundary */ page_size,
buffer_size);
if (!buffers[n_buffers].start) {
fprintf (stderr, "Out of memory\n");
exit (EXIT_FAILURE);
}
}
//===============================================================
struct v4l2_requestbuffers req2;
CLEAR (req2);
req2.count = 4;
req2.type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
req2.memory = V4L2_MEMORY_MMAP;
if (-1 == xioctl (fd2, VIDIOC_REQBUFS, &req2)) {
if (EINVAL == errno) {
fprintf (stderr, "%s does not support "
"memory mapping\n", dev_name2);
exit (EXIT_FAILURE);
} else {
errno_exit ("VIDIOC_REQBUFS2");
}
}
if (req.count < 2) {
fprintf (stderr, "Insufficient buffer memory on %s\n",
dev_name2);
exit (EXIT_FAILURE);
}
buff = (char*) calloc (req.count, sizeof (*buffers));
buffers2 = (buffer*) buff;
if (!buffers2) {
fprintf (stderr, "Out of memory\n");
exit (EXIT_FAILURE);
}
for (n_buffers = 0; n_buffers < req.count; ++n_buffers) {
struct v4l2_buffer buf;
CLEAR (buf);
buf.type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
buf.memory = V4L2_MEMORY_MMAP;
buf.index = n_buffers;
if (-1 == xioctl (fd2, VIDIOC_QUERYBUF, &buf))
errno_exit ("VIDIOC_QUERYBUF");
buffers2[n_buffers].length = buf.length;
buffers2[n_buffers].start =
mmap (NULL /* start anywhere */,
buf.length,
PROT_READ | PROT_WRITE /* required */,
MAP_SHARED /* recommended */,
fd2, buf.m.offset);
if (MAP_FAILED == buffers2[n_buffers].start)
errno_exit ("mmap");
}
}
static void
init_device (void)
{
struct v4l2_capability cap;
struct v4l2_cropcap cropcap;
struct v4l2_crop crop;
struct v4l2_format fmt;
unsigned int min;
if (-1 == xioctl (fd, VIDIOC_QUERYCAP, &cap)) {
if (EINVAL == errno) {
fprintf (stderr, "%s is no V4L2 device\n",
dev_name);
exit (EXIT_FAILURE);
} else {
errno_exit ("VIDIOC_QUERYCAP");
}
}
if (!(cap.capabilities & V4L2_CAP_VIDEO_CAPTURE)) {
fprintf (stderr, "%s is no video capture device\n",
dev_name);
exit (EXIT_FAILURE);
}
switch (io) {
case IO_METHOD_READ:
if (!(cap.capabilities & V4L2_CAP_READWRITE)) {
fprintf (stderr, "%s does not support read i/o\n",
dev_name);
exit (EXIT_FAILURE);
}
break;
case IO_METHOD_MMAP:
case IO_METHOD_USERPTR:
if (!(cap.capabilities & V4L2_CAP_STREAMING)) {
fprintf (stderr, "%s does not support streaming i/o\n",
dev_name);
exit (EXIT_FAILURE);
}
break;
}
/* Select video input, video standard and tune here. */
CLEAR (cropcap);
cropcap.type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
if (0 == xioctl (fd, VIDIOC_CROPCAP, &cropcap)) {
crop.type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
crop.c = cropcap.defrect; /* reset to default */
if (-1 == xioctl (fd, VIDIOC_S_CROP, &crop)) {
switch (errno) {
case EINVAL:
/* Cropping not supported. */
break;
default:
/* Errors ignored. */
break;
}
}
} else {
/* Errors ignored. */
}
CLEAR (fmt);
fmt.type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
fmt.fmt.pix.width = width_cap;
fmt.fmt.pix.height = height_cap;
fmt.fmt.pix.pixelformat = V4L2_PIX_FMT_YUYV;
fmt.fmt.pix.field = V4L2_FIELD_NONE;
if (-1 == xioctl (fd, VIDIOC_S_FMT, &fmt))
errno_exit ("VIDIOC_S_FMT");
//=====================================================================
struct v4l2_capability cap2;
struct v4l2_cropcap cropcap2;
struct v4l2_crop crop2;
struct v4l2_format fmt2;
if (-1 == xioctl (fd2, VIDIOC_QUERYCAP, &cap2)) {
if (EINVAL == errno) {
fprintf (stderr, "%s is no V4L2 device\n",
dev_name2);
exit (EXIT_FAILURE);
} else {
errno_exit ("VIDIOC_QUERYCAP");
}
}
if (!(cap2.capabilities & V4L2_CAP_VIDEO_CAPTURE)) {
fprintf (stderr, "%s is no video capture device\n",
dev_name2);
exit (EXIT_FAILURE);
}
switch (io) {
case IO_METHOD_READ:
if (!(cap2.capabilities & V4L2_CAP_READWRITE)) {
fprintf (stderr, "%s does not support read i/o\n",
dev_name2);
exit (EXIT_FAILURE);
}
break;
case IO_METHOD_MMAP:
case IO_METHOD_USERPTR:
if (!(cap2.capabilities & V4L2_CAP_STREAMING)) {
fprintf (stderr, "%s does not support streaming i/o\n",
dev_name2);
exit (EXIT_FAILURE);
}
break;
}
/* Select video input, video standard and tune here. */
CLEAR (cropcap2);
cropcap2.type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
if (0 == xioctl (fd2, VIDIOC_CROPCAP, &cropcap2)) {
crop2.type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
crop2.c = cropcap2.defrect; /* reset to default */
if (-1 == xioctl (fd2, VIDIOC_S_CROP, &crop2)) {
switch (errno) {
case EINVAL:
/* Cropping not supported. */
break;
default:
/* Errors ignored. */
break;
}
}
} else {
/* Errors ignored. */
}
CLEAR (fmt2);
fmt2.type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
fmt2.fmt.pix.width = width_cap;
fmt2.fmt.pix.height = height_cap;
fmt2.fmt.pix.pixelformat = V4L2_PIX_FMT_YUYV;
fmt2.fmt.pix.field = V4L2_FIELD_NONE;
if (-1 == xioctl (fd2, VIDIOC_S_FMT, &fmt2))
errno_exit ("VIDIOC_S_FMT");
/* Note VIDIOC_S_FMT may change width and height. */
/* Buggy driver paranoia. */
min = fmt.fmt.pix.width * 2;
if (fmt.fmt.pix.bytesperline < min)
fmt.fmt.pix.bytesperline = min;
min = fmt.fmt.pix.bytesperline * fmt.fmt.pix.height;
if (fmt.fmt.pix.sizeimage < min)
fmt.fmt.pix.sizeimage = min;
switch (io) {
case IO_METHOD_READ:
init_read (fmt.fmt.pix.sizeimage);
break;
case IO_METHOD_MMAP:
init_mmap ();
break;
case IO_METHOD_USERPTR:
init_userp (fmt.fmt.pix.sizeimage);
break;
}
}
static void
close_device (void)
{
if (-1 == close (fd))
errno_exit ("close");
fd = -1;
if (-1 == close (fd2))
errno_exit ("close2");
fd2 = -1;
}
static void
open_device (void)
{
struct stat st;
if (-1 == stat (dev_name, &st)) {
fprintf (stderr, "Cannot identify '%s': %d, %s\n",
dev_name, errno, strerror (errno));
exit (EXIT_FAILURE);
}
if (!S_ISCHR (st.st_mode)) {
fprintf (stderr, "%s is no device\n", dev_name);
exit (EXIT_FAILURE);
}
fd = open (dev_name, O_RDWR /* required */ | O_NONBLOCK, 0);
if (-1 == fd) {
fprintf (stderr, "Cannot open '%s': %d, %s\n",
dev_name, errno, strerror (errno));
exit (EXIT_FAILURE);
}
//======================================================================
struct stat st2;
if (-1 == stat (dev_name2, &st2)) {
fprintf (stderr, "Cannot identify '%s': %d, %s\n",
dev_name2, errno, strerror (errno));
exit (EXIT_FAILURE);
}
if (!S_ISCHR (st2.st_mode)) {
fprintf (stderr, "%s is no device\n", dev_name2);
exit (EXIT_FAILURE);
}
fd2 = open (dev_name2, O_RDWR /* required */ | O_NONBLOCK, 0);
if (-1 == fd2) {
fprintf (stderr, "Cannot open '%s': %d, %s\n",
dev_name2, errno, strerror (errno));
exit (EXIT_FAILURE);
}
}
static void
usage (FILE * fp,
int argc,
char ** argv)
{
fprintf (fp,
"Usage: %s [options]\n\n"
"Options:\n"
"-x | --width width capture resolution \n"
"-y | --height height capture resolution \n"
"-h | --help Print this message\n"
"-m | --mmap Use memory mapped buffers\n"
"-r | --read Use read() calls\n"
"-u | --userp Use application allocated buffers\n"
"",
argv[0]);
}
static const char short_options [] = "x:y:hmru";
static const struct option
long_options [] = {
{ "width", required_argument, NULL, 'x' },
{ "height", required_argument, NULL, 'y' },
{ "help", no_argument, NULL, 'h' },
{ "mmap", no_argument, NULL, 'm' },
{ "read", no_argument, NULL, 'r' },
{ "userp", no_argument, NULL, 'u' },
{ 0, 0, 0, 0 }
};
int
main (int argc,char ** argv)
{
dev_name = "/dev/video0";
dev_name2 = "/dev/video1";
for (;;) {
int index;
int c;
c = getopt_long (argc, argv,
short_options, long_options,
&index);
if (-1 == c)
break;
switch (c) {
case 0: /* getopt_long() flag */
break;
case 'x':
width_cap = atoi(optarg);
break;
case 'y':
height_cap = atoi(optarg);
break;
case 'h':
usage (stdout, argc, argv);
exit (EXIT_SUCCESS);
case 'm':
io = IO_METHOD_MMAP;
break;
case 'r':
io = IO_METHOD_READ;
break;
case 'u':
io = IO_METHOD_USERPTR;
break;
default:
usage (stderr, argc, argv);
exit (EXIT_FAILURE);
}
}
int which = PRIO_PROCESS;
id_t pid;
int priority = 19;
int ret;
pid = getpid();
ret = setpriority(which, pid, priority);
fprintf (stdout, "begin ...\r\n");
cvNamedWindow( "Cam1", CV_WINDOW_AUTOSIZE );
frame = cvCreateImage(cvSize(width_cap,height_cap), IPL_DEPTH_8U, 3);
cvNamedWindow( "Cam2", CV_WINDOW_AUTOSIZE );
frame2 = cvCreateImage(cvSize(width_cap,height_cap), IPL_DEPTH_8U, 3);
cvNamedWindow( "Canny", CV_WINDOW_AUTOSIZE );
grayImg = cvCreateImage( cvGetSize(frame), IPL_DEPTH_8U, 1 );
cannyImg = cvCreateImage(cvGetSize(frame), IPL_DEPTH_8U, 1);
fprintf (stdout, "capture resolution %dx%d \r\n",width_cap,height_cap);
fprintf (stdout, "open_device\r\n");
open_device();
fprintf (stdout, "init_device\r\n");
init_device ();
fprintf (stdout, "start_capturing\r\n");
start_capturing ();
fprintf (stdout, "mainloop\r\n");
mainloop ();
stop_capturing ();
fprintf (stdout, "stop_capturing\r\n");
uninit_device ();
fprintf (stdout, "uninit_device \r\n");
close_device ();
fprintf (stdout, " close_device\r\n");
cvReleaseImage( &grayImg );
cvReleaseImage( &cannyImg );
cvDestroyWindow( "Canny" );
cvReleaseImage( &frame );
cvDestroyWindow( "Cam1" );
cvReleaseImage( &frame2 );
cvDestroyWindow( "Cam2" );
fprintf (stdout, "...end \r\n");
exit (EXIT_SUCCESS);
return 0;
}