#ifndef __COMMON_H
#define __COMMON_H
//该头文件定义的是摄像头在屏幕上显示的宽度和高度
#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#include<errno.h>
#include <stdint.h>

typedef unsigned char U8;
typedef unsigned int U32;
typedef unsigned short U16;

#define WIDTH  800
#define HEIGHT 600

#endif	//__COMMON_H
#ifndef __VIDEOCAPTURE_H
#define __VIDEOCAPTURE_H

#include"common.h"
#include<sys/mman.h>
#include<fcntl.h>
#include<linux/videodev2.h>

#define VIDEODEVNAME ("/dev/video0")
#define COUNT  4
//需要调用的摄像头相关的接口,包括初始化,开始,结束,采取一帧的数据并保存
int initCamera();
void unInitCamera();
int startCamera();
int stopCamera();
int getOneFrame(U8* pcYuvBuffer);

#endif	//__VIDEOCAPTURE_H
#include"videoCapture.h"

static U8* pcYuv[COUNT]={NULL};
static int vidDevFd = -1;
static int nLength = 0;
static struct v4l2_buffer dequeBuffer;
static struct v4l2_buffer queueBuffer;

//初始化摄像头程序
int initCamera()
{
	//open /dev/video*
	//open video device
	vidDevFd = open(VIDEODEVNAME, O_RDONLY);
	if (vidDevFd < 0)
	{
		fprintf(stderr, "open %s failed %s\n",VIDEODEVNAME, strerror(errno));
		return -1;
	}

	//printf("vidDevFd = %d!\n", vidDevFd);			

	//设置视频格式
	//set video device format
	struct v4l2_format vidDevFmt;
	vidDevFmt.type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
	vidDevFmt.fmt.pix.pixelformat = V4L2_PIX_FMT_YUYV;
	vidDevFmt.fmt.pix.width = WIDTH;
	vidDevFmt.fmt.pix.height = HEIGHT;

	if (0 != ioctl(vidDevFd, VIDIOC_S_FMT, &vidDevFmt))
	{
		fprintf(stderr, "VIDIOC_S_FMT failed %s\n", strerror(errno));
		unInitCamera();
		return -1;
	}

	//request buffers
	//max count need to be considered
	struct v4l2_requestbuffers reqBuffer;
	reqBuffer.count = COUNT;
	reqBuffer.type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
	reqBuffer.memory = V4L2_MEMORY_MMAP;

	if (0 != ioctl(vidDevFd, VIDIOC_REQBUFS, &reqBuffer))
	{
		fprintf(stderr, "VIDIOC_REQBUFS failed %s\n", strerror(errno));
		unInitCamera();
		return -1;
	}

	//query buffers
	int i = 0;
	struct v4l2_buffer qryBuffer;
	qryBuffer.type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
	qryBuffer.memory = V4L2_MEMORY_MMAP;

	for (i = 0; i < COUNT; i++)
	{
		qryBuffer.index = i;

		if (0 != ioctl(vidDevFd, VIDIOC_QUERYBUF, &qryBuffer))
		{
			fprintf(stderr, "VIDIOC_QUERYBUF failed %s\n", strerror(errno));
			unInitCamera();
			return -1;
		}

		printf("qryBuffer.m.offset=%d, qryBuffer.lenght=%d\n", qryBuffer.m.offset, qryBuffer.length);
		//mmap video device memory allocated by VIDIOC_REQBUFS
		pcYuv[i] = mmap(NULL, qryBuffer.length, PROT_READ, MAP_SHARED, vidDevFd, qryBuffer.m.offset);
		nLength = qryBuffer.length;

		//queue buffers
		struct v4l2_buffer queBuffer;
		queBuffer.type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
		queBuffer.memory = V4L2_MEMORY_MMAP;
		queBuffer.index = qryBuffer.index;
		if (0 != ioctl(vidDevFd, VIDIOC_QBUF, &queBuffer))
		{
			fprintf(stderr, "init VIDIOC_QBUF failed %s\n", strerror(errno));
			unInitCamera();
			return -1;
		}

	}

	dequeBuffer.type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
	dequeBuffer.memory = V4L2_MEMORY_MMAP;
	queueBuffer.type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
	queueBuffer.memory = V4L2_MEMORY_MMAP;

	return 0;
}

//取消摄像头
void unInitCamera()
{
	int i = 0;

	for (i = 0; i < COUNT; i++)
	{
		if (NULL != pcYuv[i])
		{
			if (-1 == munmap(pcYuv[i], nLength));
			{
				//fprintf(stderr,"munmap error:%s!\n",strerror(errno));
			}
			pcYuv[i] = NULL;
		}
	}

	if (vidDevFd >= 0)
		close(vidDevFd);
}
//摄像头开始
int startCamera()
{
	//start camera		

	int nArg = 1;
	if (0 != ioctl(vidDevFd, VIDIOC_STREAMON, &nArg))
	{
		fprintf(stderr, "VIDIOC_STREAMON failed %s\n", strerror(errno));
		unInitCamera();
		return -1;
	}

	return 0;
}

//停止摄像头
int stopCamera()
{
	//stop camera		

	int nArg = 1;
	if (0 != ioctl(vidDevFd, VIDIOC_STREAMOFF, &nArg))
	{
		fprintf(stderr, "VIDIOC_STREAMOFF failed %s\n", strerror(errno));
		unInitCamera();
		return -1;
	}

	return 0;
}

//获取Frame
int getOneFrame(unsigned char* pcYuyvBuffer)
{
	if (0 != ioctl(vidDevFd, VIDIOC_DQBUF, &dequeBuffer))
	{
		fprintf(stderr, "VIDIOC_DQBUF failed %s\n", strerror(errno));
		unInitCamera();
		return -1;
	}

	//get frame
	memcpy(pcYuyvBuffer, pcYuv[dequeBuffer.index], dequeBuffer.length);
	printf("len=%d\n",dequeBuffer.length);

	//queue
	queueBuffer.index = dequeBuffer.index;
	if (0 != ioctl(vidDevFd, VIDIOC_QBUF, &queueBuffer))
	{
		fprintf(stderr, "VIDIOC_QBUF failed %s\n", strerror(errno));
		unInitCamera();
		return -1;
	}
	return 0;
}
#include"videoCapture.h"

int main(void )
{
	int nRet = 0;
	//char yuyv[WIDTH*HEIGHT*2];
	char* pcYuvBuffer = NULL; 

	if(-1 == initCamera())
	{
		printf("initCamera failed!\n");
		goto ERR;
	}

	if (-1 == startCamera())
	{
		printf("startCamera failed!\n");
		goto ERR;
	}

	pcYuvBuffer = (char*)malloc((WIDTH*HEIGHT*2)*sizeof(char));
	if (NULL == pcYuvBuffer)
	{
		printf("malloc YUYV buffer failed!\n");
		goto ERR;
	}

	int nDstFd = open("mytest.yuv", O_CREAT|O_EXCL|O_RDWR|O_TRUNC, 0777);
	if (nDstFd == -1)
	{
		fprintf(stderr, "open mytest.yuv failed:%s\n", strerror(errno));
	}

	int nFrameNum = 0;

	while(1)
	{
		if (20 == nFrameNum)
			break;

		memset(pcYuvBuffer, 0, WIDTH*HEIGHT*2);

		//if (-1 == getOneFrame(yuyv))
		if (-1 == getOneFrame(pcYuvBuffer))
		{
			printf("getOneFrame error!\n");
			goto ERR;
		}

		//if (-1 == write(nDstFd, yuyv, WIDTH*HEIGHT*2))
		if (-1 == write(nDstFd, pcYuvBuffer, WIDTH*HEIGHT*2))
		{
			fprintf(stderr, "write error:%s\n", strerror(errno));
			goto ERR;
		}

		nFrameNum++;
	}

	if (-1 == stopCamera())
	{
		printf("stopCamera error!\n");
	}

ERR:
	if (NULL != pcYuvBuffer)
	{
		free(pcYuvBuffer);
		pcYuvBuffer = NULL;
	}

	unInitCamera();

	close(nDstFd);
	nDstFd = -1;

	return 0;
}

基于V4L2摄像头采集图片程序设计的更多相关文章

  1. Linux 下V4l2摄像头采集图片,实现yuyv转RGB,RGB转BMP,RGB伸缩,jpeglib 库实现压缩RGB到内存中,JPEG经UDP发送功(转)

    ./configure CC=arm-linux-gnueabihf-gcc LD=arm-linux-gnueabihf-ld --host=arm-linux --prefix=/usr/loca ...

  2. 基于ZedBoard的Webcam设计(一):USB摄像头(V4L2接口)的图片采集【转】

    转自:http://www.cnblogs.com/surpassal/archive/2012/12/19/zed_webcam_lab1.html 一直想把USB摄像头接到Zedboard上,搭建 ...

  3. 14、USB摄像头(V4L2接口)的图片采集

    参考网站http://www.cnblogs.com/surpassal/archive/2012/12/19/zed_webcam_lab1.html 一.一些知识 1.V4L和V4L2. V4L是 ...

  4. 基于opencv和mfc的摄像头采集代码(GOMFCTemplate2)

            编写带界面的图像处理程序,选择opencv+mfc是一种很好的选择:在读取摄像头数据方面,网上的方法很多,其中shiqiyu的camerads的方法是较好的.       基于现有资料 ...

  5. 基于opencv和mfc的摄像头采集代码(GOMFCTemplate2)持续更新

            编写带界面的图像处理程序,选择opencv+mfc是一种很好的选择:在读取摄像头数据方面,网上的方法很多,其中shiqiyu的camerads的方法是较好的.       基于现有资料 ...

  6. 基于视频压缩的实时监控系统-sprint1基于epoll架构的采集端程序设计

    part1:产品功能 part2:epoll机制   select与epoll区别 1.select与epoll没有太大的区别.除了select有文件描述符限制(1024个),select每次调用都需 ...

  7. V4L2视频采集原理

    一.简介 Video for Linuxtwo(Video4Linux2)简称V4L2,是V4L的改进版.V4L2是linux操作系统下用于采集图片.视频和音频数据的API接口,配合适当的视频采集设备 ...

  8. linux之V4L2摄像头应用流程【转】

    本文转载自:http://blog.csdn.net/tommy_wxie/article/details/11486907 对于v4l2,上次是在调试收音机驱动的时候用过,其他也就只是用i2c配置一 ...

  9. Linux之V4L2视频采集编程详解

     V4L2(Video For Linux Two) 是内核提供给应用程序访问音.视频驱动的统一接口. Linux系统中,视频设备被当作一个设备文件来看待,设备文件存放在 /dev目录下,完整路径的设 ...

随机推荐

  1. hdu 4514 并查集+树形dp

    湫湫系列故事——设计风景线 Time Limit: 6000/3000 MS (Java/Others)    Memory Limit: 65535/32768 K (Java/Others)Tot ...

  2. 智能优化算法对TSP问题的求解研究

    要求: TSP 算法(Traveling Salesman Problem)是指给定 n 个城市和各个城市之间的距离,要 求确定一条经过各个城市当且仅当一次的最短路径,它是一种典型的优化组合问题,其最 ...

  3. P2P技术详解(二):P2P中的NAT穿越(打洞)方案详解

    1.内容概述 P2P即点对点通信,或称为对等联网,与传统的服务器客户端模式(如下图"P2P结构模型"所示)有着明显的区别,在即时通讯方案中应用广泛(比如IM应用中的实时音视频通信. ...

  4. 安装win7提示“我们无法创建新的分区,也找不到现有分区”

    用U盘安装操作系统,但是遇到了这种问题. 来来回删掉主分区,重新建立主分区,都不能搞定.最后还是用古老的方法安装了.安装的方法如下: 大家首先要进入到winpe,这里我用的是大白菜winpe. 将C盘 ...

  5. Windows Server 2008 R2服务器系统安全设置参考指南

    Server 2008 R2服务器系统安全设置参考指南  重点比较重要的几部 1.更改默认administrator用户名,复杂密码 2.开启防火墙 3.安装杀毒软件 1)新做系统一定要先打上补丁(升 ...

  6. Linux下修改主机IP地址、DNS、主机名的三种方法

    使用root用户登录进入linux,打开进去终端 在终端中输入:vi /etc/sysconfig/network-scripts/ifcfg-eth0 (最后的eth0是网卡名,我的是Auto_et ...

  7. BI之SSIS入门最新版Visual Studio调试技巧

    简介 最近公司业务需要用到BI SSIS,SSIS是什么?"SSIS是Microsoft SQL Server Integration Services的简称,是生成高性能数据集成解决方案( ...

  8. Goland 提示 :configuration is still incorrect 的解决

    安装好 Goland 后,调试编译的时候提示 goland configuration is still incorrect,百度 和 Google 都没有明确答案 Google 上有一些提示,但是也 ...

  9. centos 6安装 H3C iNode 上网客户端

    我的安装目录是/usr/iNode 直接把客户端安装包拷到这个文件夹下然后解压: #rar x iNode2.-R0162.rar 然后进入文件夹,里边有一个install.sh文件,这是一个安装文件 ...

  10. EXISTS的使用详解

    .exists的使用场合: exists 用于只能用于子查询,可以替代in,若匹配到结果,则退出内部 查询,并将条件标志为true,传回全部结果资料,in 不管匹配到匹配不到都 全部匹配完毕,使用ex ...