1 Star 0 Fork 10

Srefan/v4l2-app

forked from 一口Linux/v4l2-app 
加入 Gitee
与超过 1200万 开发者一起发现、参与优秀开源项目,私有仓库也完全免费 :)
免费加入
文件
克隆/下载
capture.c 10.71 KB
一键复制 编辑 原始数据 按行查看 历史
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <fcntl.h>
#include <unistd.h>
#include <errno.h>
#include <sys/types.h>
#include <sys/time.h>
#include <sys/mman.h>
#include <sys/ioctl.h>
#include <asm/types.h>
#include <linux/videodev2.h>
#include <linux/v4l2-subdev.h>
#define FMT_NUM_PLANES 1
struct buffer
{
void *start;
size_t length;
};
struct v4l2_dev
{
int fd;
int sub_fd;
const char *path;
const char *name;
const char *subdev_path;
const char *out_type;
enum v4l2_buf_type buf_type;
int format;
int width;
int height;
unsigned int req_count;
enum v4l2_memory memory_type;
struct buffer *buffers;
unsigned long int timestamp;
int data_len;
unsigned char *out_data;
};
struct v4l2_dev ov13850 = {
.fd = -1,
.sub_fd = -1,
.path = "/dev/video0",
.name = "ov13850",
.subdev_path = "/dev/v4l-subdev3",
.out_type = "nv12",
.buf_type = V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE,
.format = V4L2_PIX_FMT_NV12,
.width = 800,
.height = 600,
.req_count = 4,
.memory_type = V4L2_MEMORY_MMAP,
.buffers = NULL,
.timestamp = 0,
.data_len = 0,
.out_data = NULL,
};
void close_device(struct v4l2_dev *dev)
{
if (dev->buffers) {
for (unsigned int i = 0; i < dev->req_count; ++i) {
if (dev->buffers[i].start) {
munmap(dev->buffers[i].start, dev->buffers[i].length);
}
}
free(dev->buffers);
}
if (-1 != dev->fd) {
close(dev->fd);
}
if (-1 != dev->sub_fd) {
close(dev->sub_fd);
}
return;
}
void exit_failure(struct v4l2_dev *dev)
{
close_device(dev);
exit(EXIT_FAILURE);
}
void open_device(struct v4l2_dev *dev)
{
dev->fd = open(dev->path, O_RDWR | O_CLOEXEC, 0);
if (dev->fd < 0) {
printf("Cannot open %s\n\n", dev->path);
exit_failure(dev);
}
printf("Open %s succeed - %d\n\n", dev->path, dev->fd);
dev->sub_fd = open(dev->subdev_path, O_RDWR|O_CLOEXEC, 0);
if (dev->sub_fd < 0) {
printf("Cannot open %s\n\n", dev->subdev_path);
exit_failure(dev);
}
printf("Open %s succeed\n\n", dev->subdev_path);
return;
}
void get_capabilities(struct v4l2_dev *dev)
{
struct v4l2_capability cap;
if (ioctl(dev->fd, VIDIOC_QUERYCAP, &cap) < 0) {
printf("VIDIOC_QUERYCAP failed\n");
return;
}
printf("------- VIDIOC_QUERYCAP ----\n");
printf(" driver: %s\n", cap.driver);
printf(" card: %s\n", cap.card);
printf(" bus_info: %s\n", cap.bus_info);
printf(" version: %d.%d.%d\n",
(cap.version >> 16) & 0xff,
(cap.version >> 8) & 0xff,
(cap.version & 0xff));
printf(" capabilities: %08X\n", cap.capabilities);
if (cap.capabilities & V4L2_CAP_VIDEO_CAPTURE)
printf(" Video Capture\n");
if (cap.capabilities & V4L2_CAP_VIDEO_OUTPUT)
printf(" Video Output\n");
if (cap.capabilities & V4L2_CAP_VIDEO_OVERLAY)
printf(" Video Overly\n");
if (cap.capabilities & V4L2_CAP_VIDEO_CAPTURE_MPLANE)
printf(" Video Capture Mplane\n");
if (cap.capabilities & V4L2_CAP_VIDEO_OUTPUT_MPLANE)
printf(" Video Output Mplane\n");
if (cap.capabilities & V4L2_CAP_READWRITE)
printf(" Read / Write\n");
if (cap.capabilities & V4L2_CAP_STREAMING)
printf(" Streaming\n");
printf("\n");
return;
}
void set_fmt(struct v4l2_dev *dev)
{
struct v4l2_format fmt;
memset(&fmt, 0, sizeof(fmt));
fmt.type = dev->buf_type;
fmt.fmt.pix.pixelformat = dev->format;
fmt.fmt.pix.width = dev->width;
fmt.fmt.pix.height = dev->height;
if (ioctl(dev->fd, VIDIOC_S_FMT, &fmt) < 0) {
printf("VIDIOC_S_FMT failed - [%d]!\n", errno);
exit_failure(dev);
}
printf("VIDIOC_S_FMT succeed!\n");
dev->data_len = fmt.fmt.pix.sizeimage;
printf("width %d, height %d, size %d, bytesperline %d, format %c%c%c%c\n\n",
fmt.fmt.pix.width, fmt.fmt.pix.height, dev->data_len,
fmt.fmt.pix.bytesperline,
fmt.fmt.pix.pixelformat & 0xFF,
(fmt.fmt.pix.pixelformat >> 8) & 0xFF,
(fmt.fmt.pix.pixelformat >> 16) & 0xFF,
(fmt.fmt.pix.pixelformat >> 24) & 0xFF);
return;
}
void require_buf(struct v4l2_dev *dev)
{
// 申请缓冲区
struct v4l2_requestbuffers req;
memset(&req, 0, sizeof(req));
req.count = dev->req_count;
req.type = dev->buf_type;
req.memory = dev->memory_type;
if (ioctl(dev->fd, VIDIOC_REQBUFS, &req) == -1) {
printf("VIDIOC_REQBUFS failed!\n\n");
exit_failure(dev);
}
if (dev->req_count != req.count) {
printf("!!! req count = %d\n", req.count);
dev->req_count = req.count;
}
printf("VIDIOC_REQBUFS succeed!\n\n");
return;
}
void alloc_buf(struct v4l2_dev *dev)
{
dev->buffers = (struct buffer *)calloc(dev->req_count, sizeof(*(dev->buffers)));
for (unsigned int i = 0; i < dev->req_count; ++i) {
struct v4l2_buffer buf;
struct v4l2_plane planes[FMT_NUM_PLANES];
memset(&buf, 0, sizeof(buf));
memset(&planes, 0, sizeof(planes));
buf.type = dev->buf_type;
buf.memory = dev->memory_type;
buf.index = i;
if (V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE == dev->buf_type) {
buf.m.planes = planes;
buf.length = FMT_NUM_PLANES;
}
if (ioctl(dev->fd, VIDIOC_QUERYBUF, &buf) == -1) {
printf("VIDIOC_QUERYBUF failed!\n\n");
exit_failure(dev);
}
if (V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE == dev->buf_type) {
dev->buffers[i].length = buf.m.planes[0].length;
dev->buffers[i].start =
mmap(NULL /* start anywhere */,
buf.m.planes[0].length,
PROT_READ | PROT_WRITE /* required */,
MAP_SHARED /* recommended */,
dev->fd, buf.m.planes[0].m.mem_offset);
} else {
dev->buffers[i].length = buf.length;
dev->buffers[i].start = mmap(NULL,
buf.length,
PROT_READ | PROT_WRITE,
MAP_SHARED,
dev->fd,
buf.m.offset);
}
if (dev->buffers[i].start == MAP_FAILED) {
printf("Memory map failed!\n\n");
exit_failure(dev);
}
}
printf("Memory map succeed!\n\n");
return;
}
void queue_buf(struct v4l2_dev *dev)
{
for (unsigned int i = 0; i < dev->req_count; ++i) {
struct v4l2_buffer buf;
struct v4l2_plane planes[FMT_NUM_PLANES];
memset(&buf, 0, sizeof(buf));
memset(&planes, 0, sizeof(planes));
buf.type = dev->buf_type;
buf.memory = dev->memory_type;
buf.index = i;
if (V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE == dev->buf_type) {
buf.m.planes = planes;
buf.length = FMT_NUM_PLANES;
}
if (ioctl(dev->fd, VIDIOC_QBUF, &buf) < 0) {
printf("VIDIOC_QBUF failed!\n\n");
exit_failure(dev);
}
}
printf("VIDIOC_QBUF succeed!\n\n");
return;
}
void stream_on(struct v4l2_dev *dev)
{
enum v4l2_buf_type type = dev->buf_type;
if (ioctl(dev->fd, VIDIOC_STREAMON, &type) == -1) {
printf("VIDIOC_STREAMON failed!\n\n");
exit_failure(dev);
}
printf("VIDIOC_STREAMON succeed!\n\n");
return;
}
void get_frame(struct v4l2_dev *dev, int skip_frame)
{
struct v4l2_buffer buf;
struct v4l2_plane planes[FMT_NUM_PLANES];
for (int i = 0; i <= skip_frame; ++i) {
memset(&buf, 0, sizeof(buf));
buf.type = dev->buf_type;
buf.memory = dev->memory_type;
if (V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE == dev->buf_type) {
buf.m.planes = planes;
buf.length = FMT_NUM_PLANES;
}
if (ioctl(dev->fd, VIDIOC_DQBUF, &buf) == -1) {
printf("VIDIOC_DQBUF failed!\n\n");
exit_failure(dev);
}
dev->out_data = (unsigned char *)dev->buffers[buf.index].start;
dev->timestamp = buf.timestamp.tv_sec * 1000000 + buf.timestamp.tv_usec;
printf("image: sequence = %d, timestamp = %lu\n", buf.sequence, dev->timestamp);
if (ioctl(dev->fd, VIDIOC_QBUF, &buf) == -1) {
printf("VIDIOC_QBUF failed!\n");
exit_failure(dev);
}
}
return;
}
void save_picture(const char *filename, unsigned char *file_data, unsigned int len, int is_overwrite)
{
FILE *fp;
if (is_overwrite)
fp = fopen(filename, "wb");
else
fp = fopen(filename, "ab");
if (fp < 0) {
printf("Open frame data file failed\n\n");
return;
}
if (fwrite(file_data, 1, len, fp) < len) {
printf("Out of memory!\n");
is_running = 0;
}
fflush(fp);
fclose(fp);
printf("Save one frame to %s succeed!\n\n", filename);
return;
}
void stream_off(struct v4l2_dev *dev)
{
enum v4l2_buf_type type;
type = dev->buf_type;
if (ioctl(dev->fd, VIDIOC_STREAMOFF, &type) == -1) {
printf("VIDIOC_STREAMOFF failed!\n\n");
exit_failure(dev);
}
printf("VIDIOC_STREAMOFF succeed!\n\n");
return;
}
void set_fps(struct v4l2_dev *dev, unsigned int fps)
{
int ret;
struct v4l2_subdev_frame_interval frame_int;
if (fps == 0) return;
memset(&frame_int, 0x00, sizeof(frame_int));
frame_int.interval.numerator = 10000;
frame_int.interval.denominator = fps * 10000;
ret = ioctl(dev->sub_fd, VIDIOC_SUBDEV_S_FRAME_INTERVAL, &frame_int);
if (ret < 0) {
printf("VIDIOC_SUBDEV_S_FRAME_INTERVAL error\n");
goto set_fps_err;
}
printf("VIDIOC_SUBDEV_S_FRAME_INTERVAL [%u fps] OK\n", fps);
set_fps_err:
return;
}
int main(int argc, char *argv[])
{
struct v4l2_dev *camdev = &ov13850;
unsigned int fps = 0;
unsigned int mode = 4;
char name[40] = {0};
if (argc > 1)
fps = atoi(argv[1]);
open_device(camdev); // 1 打开摄像头设备
get_capabilities(camdev);
set_fmt(camdev); // 2 设置出图格式
require_buf(camdev); // 3 申请缓冲区
alloc_buf(camdev); // 4 内存映射
queue_buf(camdev); // 5 将缓存帧加入队列
set_fps(camdev, fps);
stream_on(camdev); // 6 开启视频流
get_frame(camdev, 9); // 7 取一帧数据
snprintf(name, sizeof(name), "/data/%s.%s", camdev->name, camdev->out_type);
save_picture(name, camdev->out_data, camdev->data_len, 1);
stream_off(camdev); // 8 关闭视频流
close_device(camdev); // 9 释放内存关闭文件
return 0;
}
Loading...
马建仓 AI 助手
尝试更多
代码解读
代码找茬
代码优化
C
1
https://gitee.com/srefan/v4l2-app.git
git@gitee.com:srefan/v4l2-app.git
srefan
v4l2-app
v4l2-app
master

搜索帮助