video4linux簡介

Video4linux(簡稱V4L),是linux中關于視頻設備的內核驅動,現在已有Video4linux2,還未加入linux內核,使用需自己下載補丁。在Linux中,視頻設備是設備文件,可以像訪問普通文件一樣對其進行讀寫,攝像頭在/dev/video0下。


2.Video4linux下視頻編程的流程
(1)打開視頻設備:
(2) 讀取設備信息
(3)更改設備當前設置(沒必要的話可以不做)
(4)進行視頻采集,兩種方法:
??????? a.內存映射
??????? b.直接從設備讀取
(5)對采集的視頻進行處理
(6)關閉視頻設備。

為程序定義的數據結構
typedef struct v4l_struct
{
?? int fd;
?? struct video_capability capability;
?? struct video_channel channel[4];
?? struct video_picture picture;
?? struct video_window window;
?? struct video_capture capture;
?? struct video_buffer buffer;
?? struct video_mmap mmap;
?? struct video_mbuf mbuf;???
?? unsigned char *map;
?? int frame;
?? int framestat[2];
}vd;


3.Video4linux支持的數據結構及其用途
(1) video_capability 包含設備的基本信息(設備名稱、支持的最大最小分辨率、信號源信息等),包含的分量:
?name[32]?? //設備名稱
?maxwidth ,maxheight,minwidth,minheight
?Channels //信號源個數
?type??? //是否能capture,彩色還是黑白,是否能裁剪等等。值如VID_TYPE_CAPTURE等
?
(2)video_picture 設備采集的圖象的各種屬性
?brightness 0~65535
?hue
?colour
?contrast
?whiteness
?depth // 24
?palette //VIDEO_PALETTE_RGB24

(3)video_channel???????? 關于各個信號源的屬性
??? Channel //信號源的編號
??? name
??? tuners
??? Type???? VIDEO_TYPE_TV | IDEO_TYPE_CAMERA
??? Norm制式

(4)video_window //包含關于capture area的信息
??? xx windows 中的坐標.
??? y???? x windows 中的坐標.
??? width??? The width of the image capture.
??? height?? The height of the image capture.
??? chromakey A host order RGB32 value for the chroma key.
??? flags????? Additional capture flags.
??? clips????? A list of clipping rectangles. (Set only)
??? clipcount??? The number of clipping rectangles. (Set only)
(5)video_mbuf?? //利用mmap進行映射的幀的信息
????? size //每幀大小
????? Frames //最多支持的幀數
????? Offsets //每幀相對基址的偏移
(6)video_buffer?? 最底層對buffer的描述
????? void *baseBase physical address of the buffer
????? int heightHeight of the frame buffer
????? int widthWidth of the frame buffer
????? int depthDepth of the frame buffer
????? int bytesperlineNumber of bytes of memory between the start of two adjacent lines
?? 實際顯示的部分一般比它描述的部分小
(7)video_mmap //用于mmap

4.關鍵步驟介紹
(1)打開視頻:
Open(”/dev/video0”,vdàfd);
關閉視頻設備用close(”/dev/video0”,vdàfd);
(2)讀video_capability 中信息
ioctl(vd->fd, VIDIOCGCAP, &(vd->capability))
成功后可讀取vd->capability各分量 eg.
(3)讀video_picture中信息
ioctl(vd->fd, VIDIOCGPICT, &(vd->picture));
(4)改變video_picture中分量的值 (可以不做的)
先為分量賦新值,再調用VIDIOCSPICT
Eg.
?vd->picture.colour = 65535;
?if(ioctl(vd->fd, VIDIOCSPICT, &(vd->picture)) < 0)
?{
?perror("VIDIOCSPICT");
?return -1;
?}
(5)初始化channel (可以不做的)
?必須先做得到vd->capability中的信息
?for (i = 0; i < vd->capability.channels; i++)
??? {
?????? vd->channel[i].channel = i;
?????? if (ioctl(vd->fd, VIDIOCGCHAN, &(vd->channel[i])) < 0)
? {
????????? perror("v4l_get_channel:");
????????? return -1;
????????? }
??? }

重點:截取圖象的兩種方法
1,用mmap(內存映射)方式截取視頻
?mmap( )系統調用使得進程之間通過映射同一個普通文件實現共享內存。普通文件被映射到進程地址空間后,進程可以向訪問普通內存一樣對文件進行訪問,不必再調用read(),write()等操作。
?兩個不同進程A、B共享內存的意思是,同一塊物理內存被映射到進程A、B各自的進程地址空間。進程A可以即時看到進程B對共享內存中數據的更新,反之亦然
?采用共享內存通信的一個顯而易見的好處是效率高,因為進程可以直接讀寫內存,而不需要任何數據的拷貝
(1)設置picture的屬性
(2) 初始化video_mbuf,以得到所映射的buffer的信息
ioctl(vd->fd, VIDIOCGMBUF, &(vd->mbuf))
(3)可以修改video_mmap和幀狀態的當前設置
???? Eg.???? vd->mmap.format = VIDEO_PALETTE_RGB24
?????????????? vd->framestat[0] = vd->framestat[1] = 0; vd->frame = 0;
(4)將mmap與video_mbuf綁定
?void* mmap ( void * addr , size_t len , int prot , int flags , int fd , off_t offset )
?len //映射到調用進程地址空間的字節數,它從被映射文件開頭offset個字節開始算起
?Prot //指定共享內存的訪問權限 PROT_READ(可讀), PROT_WRITE (可寫), PROT_EXEC (可執行)
?flags // MAP_SHARED?? MAP_PRIVATE中必選一個 // MAP_ FIXED不推薦使用addr //共內存享的起始地址,一般設0,表示由系統分配
?Mmap( ) 返回值是系統實際分配的起始地址
?if((vd->map = (unsigned char*)mmap(0, vd->mbuf.size, PROT_READ|PROT_WRITE, MAP_SHARED, vd->fd, 0)) < 0)
?{
?perror("v4l_mmap mmap:");
?return -1;
?}
(5)Mmap方式下真正做視頻截取的 VIDIOCMCAPTURE
ioctl(vd->fd, VIDIOCMCAPTURE, &(vd->mmap)) ;
?若調用成功,開始一幀的截取,是非阻塞的,
?是否截取完畢留給VIDIOCSYNC來判斷
(6)調用VIDIOCSYNC等待一幀截取結束
?if(ioctl(vd->fd, VIDIOCSYNC, &frame) < 0)
?{
?perror("v4l_sync:VIDIOCSYNC");
?return -1;
?}
若成功,表明一幀截取已完成。可以開始做下一次 VIDIOCMCAPTURE
?frame是當前截取的幀的序號。

****關于雙緩沖:
?video_bmuf bmuf.frames = 2;
?一幀被處理時可以采集另一幀
?int frame; //當前采集的是哪一幀
?int framestat[2]; //幀的狀態 沒開始采集|等待采集結束
?幀的地址由vd->map + vd->mbuf.offsets[vd->frame]得到
?采集工作結束后調用munmap取消綁定
?munmap(vd->map, vd->mbuf.size)

2,視頻截取的第二種方法:直接讀設備
關于緩沖大小,圖象等的屬性須由使用者事先設置
?調用read();
?int read (要訪問的文件描述符;指向要讀寫的信息的指針;應該讀寫的字符數);
?返回值為實際讀寫的字符數
?int len ;
?unsigned char *vd->map= (unsigned char *) malloc(vdàcapability.maxwidth*vdàcapability.maxheight );
?len = read(vdàfd,vdà vd->map,
??????????????????? vdàcapability.maxwidth*vdàcapability.maxheight*3 );

一.什么是video4linux
Video4linux2(簡稱V4L2),是linux中關于視頻設備的內核驅動。在Linux中,視頻設備是設備文件,可以像訪問普通文件一樣對其進行讀寫,攝像頭在/dev/video0下。

二、一般操作流程(視頻設備):
1. 打開設備文件。 int fd=open(”/dev/video0″,O_RDWR);
2. 取得設備的capability,看看設備具有什么功能,比如是否具有視頻輸入,或者音頻輸入輸出等。
VIDIOC_QUERYCAP,struct v4l2_capability
3. 選擇視頻輸入,一個視頻設備可以有多個視頻輸入。
VIDIOC_S_INPUT,struct v4l2_input
4. 設置視頻的制式和幀格式,制式包括PAL,NTSC,幀的格式個包括寬度和高度等。

VIDIOC_S_STD,VIDIOC_S_FMT,struct v4l2_std_id,struct v4l2_format
5.
向驅動申請幀緩沖,一般不超過5個。struct v4l2_requestbuffers
6. 將申請到的幀緩沖映射到用戶空間,這樣就可以直接操作采集到的幀了,而不必去復制。
mmap
7. 將申請到的幀緩沖全部入隊列,以便存放采集到的數據
.VIDIOC_QBUF,struct v4l2_buffer
8. 開始視頻的采集。
VIDIOC_STREAMON
9. 出隊列以取得已采集數據的幀緩沖,取得原始采集數據。
VIDIOC_DQBUF
10. 將緩沖重新入隊列尾,這樣可以循環采集。
VIDIOC_QBUF
11. 停止視頻的采集。
VIDIOC_STREAMOFF
12. 關閉視頻設備。
close(fd);
三、常用的結構體(參見/usr/include/linux/videodev2.h):

struct v4l2_requestbuffers reqbufs;//向驅動申請幀緩沖的請求,里面包含申請的個數
struct v4l2_capability cap;//這個設備的功能,比如是否是視頻輸入設備
struct v4l2_input input; //視頻輸入
struct v4l2_standard std;//視頻的制式,比如PAL,NTSC
struct v4l2_format fmt;//幀的格式,比如寬度,高度等

struct v4l2_buffer buf;//代表驅動中的一幀
v4l2_std_id stdid;//視頻制式,例如:V4L2_STD_PAL_B
struct v4l2_queryctrl query;//查詢的控制

struct v4l2_control control;//具體控制的值

下面具體說明開發流程(網上找的啦,也在學習么)

打開視頻設備

在V4L2中,視頻設備被看做一個文件。使用open函數打開這個設備:

//用非阻塞模式打開攝像頭設備

intcameraFd;

cameraFd= open(“/dev/video0″,O_RDWR | O_NONBLOCK, 0);

//如果用阻塞模式打開攝像頭設備,上述代碼變為:

//cameraFd = open(”/dev/video0″, O_RDWR, 0);

關于阻塞模式和非阻塞模式

應用程序能夠使用阻塞模式或非阻塞模式打開視頻設備,如果使用非阻塞模式調用視頻設備,即使尚未捕獲到信息,驅動依舊會把緩存(DQBUFF)里的東西返回給應用程序。

設定屬性及采集方式

打開視頻設備后,可以設置該視頻設備的屬性,例如裁剪、縮放等。這一步是可選的。在Linux編程中,一般使用ioctl函數來對設備的I/O通道進行管理:

extern intioctl (int __fd,unsigned long int __request, …)__THROW;

__fd:設備的ID,例如剛才用open函數打開視頻通道后返回的cameraFd;

__request:具體的命令標志符。

在進行V4L2開發中,一般會用到以下的命令標志符:

  1. VIDIOC_REQBUFS:分配內存
  2. VIDIOC_QUERYBUF:把VIDIOC_REQBUFS中分配的數據緩存轉換成物理地址
  3. VIDIOC_QUERYCAP:查詢驅動功能
  4. VIDIOC_ENUM_FMT:獲取當前驅動支持的視頻格式
  5. VIDIOC_S_FMT:設置當前驅動的頻捕獲格式
  6. VIDIOC_G_FMT:讀取當前驅動的頻捕獲格式
  7. VIDIOC_TRY_FMT:驗證當前驅動的顯示格式
  8. VIDIOC_CROPCAP:查詢驅動的修剪能力
  9. VIDIOC_S_CROP:設置視頻信號的邊框
  10. VIDIOC_G_CROP:讀取視頻信號的邊框
  11. VIDIOC_QBUF:把數據從緩存中讀取出來
  12. VIDIOC_DQBUF:把數據放回緩存隊列
  13. VIDIOC_STREAMON:開始視頻顯示函數
  14. VIDIOC_STREAMOFF:結束視頻顯示函數
  15. VIDIOC_QUERYSTD:檢查當前視頻設備支持的標準,例如PAL或NTSC。

這些IO調用,有些是必須的,有些是可選擇的。

檢查當前視頻設備支持的標準

在亞洲,一般使用PAL(720X576)制式的攝像頭,而歐洲一般使用NTSC(720X480),使用VIDIOC_QUERYSTD來檢測:

v4l2_std_id std;

do{

ret= ioctl(fd,VIDIOC_QUERYSTD, &std);

} while (ret == -1 && errno == EAGAIN);

switch(std) {

caseV4L2_STD_NTSC:

//……

caseV4L2_STD_PAL:

//……

}

設置視頻捕獲格式

當檢測完視頻設備支持的標準后,還需要設定視頻捕獲格式:

structv4l2_format??? fmt;

memset( &fmt, 0, sizeof(fmt) );

fmt.type= V4L2_BUF_TYPE_VIDEO_CAPTURE;

fmt.fmt.pix.width= 720;

fmt.fmt.pix.height= 576;

fmt.fmt.pix.pixelformat= V4L2_PIX_FMT_YUYV;

fmt.fmt.pix.field= V4L2_FIELD_INTERLACED;

if(ioctl(fd,VIDIOC_S_FMT, &fmt) == -1) {

return-1;

}

v4l2_format結構體定義如下:

structv4l2_format

{

enumv4l2_buf_type type;??? // 數據流類型,必須永遠是//V4L2_BUF_TYPE_VIDEO_CAPTURE

union

{

structv4l2_pix_format??? pix;

structv4l2_window??????? win;

structv4l2_vbi_format??? vbi;

__u8??? raw_data[200];

} fmt;

};

structv4l2_pix_format

{

__u32?????????????????? width;????????// 寬,必須是16的倍數

__u32??????????????? ??? height;???????// 高,必須是16的倍數

__u32?????????????????? pixelformat;??// 視頻數據存儲類型,例如是//YUV4:2:2還是RGB

enumv4l2_field???????? field;

__u32?????????????????? bytesperline;

__u32?????????????????? sizeimage;

enumv4l2_colorspace??? colorspace;

__u32?????????????????? priv;

};

?

前言:目前正在忙于ARM平臺的Linux應用程序的開發(其實是剛剛起步學習啦)。底層的東西不用考慮了,開發板子提供了NAND Bootloader,和Linux 2.6的源碼,而且都編譯好了。自己編譯的bootloader可以用,但是Linux編譯后,文件很大,暫且就用人家編譯的系統,先專心寫應用程序 吧。。

正文:要做的任務是,把一塊板子上的攝像頭采集的圖像和聲卡采集的聲音(貌似很啰嗦哈)通過TCP/IP協議傳輸到另一塊板子上。第一步,先把視頻獲取并且在本地LCD上顯示。看了板子提供的文檔,視頻傳輸需要用V4L2的API。

一.什么是video4linux
Video4linux2(簡稱V4L2),是linux中關于視頻設備的內核驅動。在Linux中,視頻設備是設備文件,可以像訪問普通文件一樣對其進行讀寫,攝像頭在/dev/video0下。

二、一般操作流程(視頻設備):
1. 打開設備文件。 int fd=open(”/dev/video0″,O_RDWR);
2. 取得設備的capability,看看設備具有什么功能,比如是否具有視頻輸入,或者音頻輸入輸出等。
VIDIOC_QUERYCAP,struct v4l2_capability
3. 選擇視頻輸入,一個視頻設備可以有多個視頻輸入。
VIDIOC_S_INPUT,struct v4l2_input
4. 設置視頻的制式和幀格式,制式包括PAL,NTSC,幀的格式個包括寬度和高度等。

VIDIOC_S_STD,VIDIOC_S_FMT,struct v4l2_std_id,struct v4l2_format
5.
向驅動申請幀緩沖,一般不超過5個。struct v4l2_requestbuffers
6. 將申請到的幀緩沖映射到用戶空間,這樣就可以直接操作采集到的幀了,而不必去復制。
mmap
7. 將申請到的幀緩沖全部入隊列,以便存放采集到的數據
.VIDIOC_QBUF,struct v4l2_buffer
8. 開始視頻的采集。
VIDIOC_STREAMON
9. 出隊列以取得已采集數據的幀緩沖,取得原始采集數據。
VIDIOC_DQBUF
10. 將緩沖重新入隊列尾,這樣可以循環采集。
VIDIOC_QBUF
11. 停止視頻的采集。
VIDIOC_STREAMOFF
12. 關閉視頻設備。
close(fd);

三、常用的結構體(參見/usr/include/linux/videodev2.h):

struct v4l2_requestbuffers reqbufs;? //向驅動申請幀緩沖的請求,里面包含申請的個數
struct v4l2_capability cap;???????????????? //這個設備的功能,比如是否是視頻輸入設備
struct v4l2_input input;????????????????????? //視頻輸入
struct v4l2_standard std;?????????????????? //視頻的制式,比如PAL,NTSC
struct v4l2_format fmt;????????????????????? //幀的格式,比如寬度,高度等

struct v4l2_buffer buf;?????????????????????? //代表驅動中的一幀
v4l2_std_id stdid;?????????????????????????????? //視頻制式,例如:V4L2_STD_PAL_B
struct v4l2_queryctrl query;??????????? //查詢的控制

struct v4l2_control control;???????????? //具體控制的值

下面具體說明開發流程(網上找的啦,也在學習么)

?

?

打開視頻設備

在V4L2中,視頻設備被看做一個文件。使用open函數打開這個設備:

// 用非阻塞模式打開攝像頭設備

int cameraFd;

cameraFd = open(“/dev/video0″, O_RDWR | O_NONBLOCK, 0);

// 如果用阻塞模式打開攝像頭設備,上述代碼變為:

//cameraFd = open(”/dev/video0″, O_RDWR, 0);

關于阻塞模式和非阻塞模式

應用程序能夠使用阻塞模式或非阻塞模式打開視頻設備,如果使用非阻塞模式調用視頻設備,即使尚未捕獲到信息,驅動依舊會把緩存(DQBUFF)里的東西返回給應用程序。

設定屬性及采集方式

打開視頻設備后,可以設置該視頻設備的屬性,例如裁剪、縮放等。這一步是可選的。在Linux編程中,一般使用ioctl函數來對設備的I/O通道進行管理:

extern int ioctl (int __fd, unsigned long int __request, …) __THROW;

__fd:設備的ID,例如剛才用open函數打開視頻通道后返回的cameraFd;

__request:具體的命令標志符。

在進行V4L2開發中,一般會用到以下的命令標志符:

  1. VIDIOC_REQBUFS分配內存
  2. VIDIOC_QUERYBUF把VIDIOC_REQBUFS中分配的數據緩存轉換成物理地址
  3. VIDIOC_QUERYCAP查詢驅動功能
  4. VIDIOC_ENUM_FMT獲取當前驅動支持的視頻格式
  5. VIDIOC_S_FMT設置當前驅動的頻捕獲格式
  6. VIDIOC_G_FMT讀取當前驅動的頻捕獲格式
  7. VIDIOC_TRY_FMT驗證當前驅動的顯示格式
  8. VIDIOC_CROPCAP查詢驅動的修剪能力
  9. VIDIOC_S_CROP設置視頻信號的邊框
  10. VIDIOC_G_CROP讀取視頻信號的邊框
  11. VIDIOC_QBUF把數據從緩存中讀取出來
  12. VIDIOC_DQBUF把數據放回緩存隊列
  13. VIDIOC_STREAMON開始視頻顯示函數
  14. VIDIOC_STREAMOFF結束視頻顯示函數
  15. VIDIOC_QUERYSTD檢查當前視頻設備支持的標準,例如PAL或NTSC

這些IO調用,有些是必須的,有些是可選擇的。

檢查當前視頻設備支持的標準

在亞洲,一般使用PAL(720X576)制式的攝像頭,而歐洲一般使用NTSC(720X480)使用VIDIOC_QUERYSTD來檢測

v4l2_std_id std;

do {

ret = ioctl(fd, VIDIOC_QUERYSTD, &std);

} while (ret == -1 && errno == EAGAIN);

switch (std) {

case V4L2_STD_NTSC:

//……

case V4L2_STD_PAL:

//……

}

設置視頻捕獲格式

當檢測完視頻設備支持的標準后,還需要設定視頻捕獲格式:

struct v4l2_format??? fmt;

memset ( &fmt, 0, sizeof(fmt) );

fmt.type = V4L2_BUF_TYPE_VIDEO_CAPTURE;

fmt.fmt.pix.width = 720;

fmt.fmt.pix.height = 576;

fmt.fmt.pix.pixelformat = V4L2_PIX_FMT_YUYV;

fmt.fmt.pix.field = V4L2_FIELD_INTERLACED;

if (ioctl(fd, VIDIOC_S_FMT, &fmt) == -1) {

return -1;

}

v4l2_format結構體定義如下:

struct v4l2_format

{

enum v4l2_buf_type type;??? // 數據流類型,必須永遠是//V4L2_BUF_TYPE_VIDEO_CAPTURE

union

{

struct v4l2_pix_format??? pix;

struct v4l2_window??????? win;

struct v4l2_vbi_format??? vbi;

__u8??? raw_data[200];

} fmt;

};

struct v4l2_pix_format

{

__u32?????????????????? width;???????? // 寬,必須是16的倍數

__u32??????????????? ??? height;??????? // 高,必須是16的倍數

__u32?????????????????? pixelformat;?? // 視頻數據存儲類型,例如是//YUV4:2:2還是RGB

enum v4l2_field???????? field;

__u32?????????????????? bytesperline;

__u32?????????????????? sizeimage;

enum v4l2_colorspace??? colorspace;

__u32?????????????????? priv;

};

分配內存

接下來可以為視頻捕獲分配內存:

struct v4l2_requestbuffers req;

if (ioctl(fd, VIDIOC_REQBUFS, &req) == -1) {

return -1;

}

v4l2_requestbuffers定義如下:

struct v4l2_requestbuffers

{

__u32?????????????? count; // 緩存數量,也就是說在緩存隊列里保持多少張照片

enum v4l2_buf_type type;?? // 數據流類型,必須永遠是V4L2_BUF_TYPE_VIDEO_CAPTURE

enum v4l2_memory??? memory; // V4L2_MEMORY_MMAP 或 V4L2_MEMORY_USERPTR

__u32?????????????? reserved[2];

};

獲取并記錄緩存的物理空間

使用VIDIOC_REQBUFS,我們獲取了req.count個緩存,下一步通過調用VIDIOC_QUERYBUF命令來獲取這些緩存的地址,然后使用mmap函數轉換成應用程序中的絕對地址,最后把這段緩存放入緩存隊列:

?

?

typedef struct VideoBuffer {

void *start;

size_t length;

} VideoBuffer;


VideoBuffer*????????? buffers = calloc( req.count, sizeof(*buffers) );

//這里的 buffers可以理解為一個結構體數組,用來存放視頻幀,一共req.count個


struct v4l2_buffer??? buf;

//這里的buf,只有一個,它相當于內核與用戶空間傳遞數據的一個中介,會被循環利用:取出、放入、取出、放入……


for (numBufs = 0; numBufs < req.count; numBufs++) {

memset( &buf, 0, sizeof(buf) );

buf.type = V4L2_BUF_TYPE_VIDEO_CAPTURE;

buf.memory = V4L2_MEMORY_MMAP;

buf.index = numBufs;? //這里要標記索引

// 讀取緩存,buf取出(循環利用)

if (ioctl(fd, VIDIOC_QUERYBUF, &buf) == -1) {

return -1;

}

buffers[numBufs].length = buf.length;

// 轉換成相對地址

buffers[numBufs].start = mmap(NULL, buf.length,

PROT_READ | PROT_WRITE,

MAP_SHARED,

fd,

buf.m.offset);


if (buffers[numBufs].start == MAP_FAILED) {

return -1;

}


// 放入緩存隊列,buf放入(循環利用)

if (ioctl(fd, VIDIOC_QBUF, &buf) == -1) {

return -1;

}

}

關于視頻采集方式

操作系統一般把系統使用的內存劃分成用戶空間和內核空間,分別由應用程序管理和操作系統管理。應用程序可以直接訪問內存的地址,而內核空間存放的是 供內核訪問的代碼和數據,用戶不能直接訪問。v4l2捕獲的數據,最初是存放在內核空間的,這意味著用戶不能直接訪問該段內存,必須通過某些手段來轉換地 址。

一共有三種視頻采集方式:使用read、write方式;內存映射方式和用戶指針模式。

read、write方式:在用戶空間和內核空間不斷拷貝數據,占用了大量用戶內存空間,效率不高。

內存映射方式:把設備里的內存映射到應用程序中的內存控件,直接處理設備內存,這是一種有效的方式。上面的mmap函數就是使用這種方式。

用戶指針模式:內存片段由應用程序自己分配。這點需要在v4l2_requestbuffers里將memory字段設置成V4L2_MEMORY_USERPTR。

處理采集數據

V4L2有一個數據緩存,存放req.count數量的緩存數據數據緩存采用FIFO的方式,當應用程序調用緩存數據時,緩存隊列將最先采集到的 視頻數據緩存送出,并重新采集一張視頻數據。這個過程需要用到兩個ioctl命令,VIDIOC_DQBUF和VIDIOC_QBUF:

struct v4l2_buffer buf;

memset(&buf,0,sizeof(buf));

buf.type=V4L2_BUF_TYPE_VIDEO_CAPTURE;

buf.memory=V4L2_MEMORY_MMAP;

buf.index=0;

//讀取緩存

if (ioctl(cameraFd, VIDIOC_DQBUF, &buf) == -1)

{

return -1;

}

//…………視頻處理算法

//重新放入緩存隊列

if (ioctl(cameraFd, VIDIOC_QBUF, &buf) == -1) {

return -1;

}

關閉視頻設備

使用close函數關閉一個視頻設備

close(cameraFd)

還需要使用munmap方法。

?

附錄:標準的V4l2的API

http://v4l.videotechnology.com/dwg/v4l2.pdf

?

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/253735.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/253735.shtml
英文地址,請注明出處:http://en.pswp.cn/news/253735.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

動態DPC算法學習

造成壞點的原因 感光元件芯片自身工藝技術瑕疵造成;光線采集存在缺陷;制造商產品差異;壞點分類 hot pixel: 固定保持較高的像素值,一般呈現為畫面高亮的點;dead pixel: 固定保持較低的像素值,一般在畫面中呈現為暗點;noise pixel:信號強度隨光照呈現的變化規律不符合正…

windows 郵槽mailslot 在服務程序內建立后客戶端無權限訪問(GetLastError() == 5)的問題...

郵槽創建在服務程序內&#xff0c;可以創建成功&#xff0c; 但外部客戶端連接時 m_hMailslot CreateFile("\\\\.\\mailslot\\zdpMailslot",GENERIC_WRITE,FILE_SHARE_READ,NULL,OPEN_EXISTING,FILE_ATTRIBUTE_NORMAL,NULL);GetLastError返回錯誤 5 &#xff0c;無權…

遞歸下降分析

對于給定的文法G[E] : E→ET|E-T|TT→T*F| T/F|FF→(E)|i 消除左遞歸后的文法是&#xff1a;E→TE E→TE|-TE|∑ T→FT T→*FT|/FT|∑ F→(E)|i 是否是LL(1)文法&#xff1f; select(E→TE)first(TE){(,i}select(E→TE)first(TE){}select(E→-TE)first(-TE){-}select(E→∑)fol…

SYS簡介

"sysfs is a ram-based filesystem initially based on ramfs. It provides a means to export kernel data structures, their attributes, and the linkages between them to userspace.” --- documentation/filesystems/sysfs.txt 可以先把documentation/filesystems/…

數字后端——布圖規劃

布圖規劃&#xff08;floorplan&#xff09;與布局&#xff08;place&#xff09;在芯片設計中占據著重要的地位&#xff0c;它的合理與否直接關系到芯片的時序收斂、布線通暢、電源穩定以及良品率。所以在整個芯片設計中&#xff0c;從布圖規劃到完成布局一般需要占據整個物理…

利用SSH傳輸文件

在linux下一般用scp這個命令來通過ssh傳輸文件。 1、從服務器上下載文件scp usernameservername:/path/filename /var/www/local_dir&#xff08;本地目錄&#xff09; 2、上傳本地文件到服務器scp /path/filename usernameservername:/path 例如scp /var/www/test.php root19…

App WebView實例化

a&#xff0c;高級設置里的環境變量 jdk的配置 b&#xff0c;下載Google的sdk&#xff0c;里面直接包含eclipse 1&#xff0c;新建一個項目 2&#xff0c;起個名字 3&#xff0c;設么走不做&#xff0c;next 4&#xff0c;只操作選擇顯示的三種方式 5&#xff0c;next什么都不做…

[動態代理三部曲:下] - 從動態代理,看Retrofit的源碼實現

前言 關于動態代理的系列文章&#xff0c;到此便進入了最后的“一出好戲”。前倆篇內容分別展開了&#xff1a;從源碼上&#xff0c;了解JDK實現動態代理的原理&#xff1b;以及從動態代理切入&#xff0c;學會看class文件結構的含義。 如果還沒有看過這倆篇文章的小伙伴&#…

Ti的DM368系列芯片的所有PDF資料匯總

http://www.ti.com/sc/docs/psheets/man_dsp.htm

劉浩(專業打劫三十年)20155307的預備作業02:

我的技能&#xff1f;比大多數人好&#xff1f;經驗是什么&#xff1f;與老師的經驗的共同之處&#xff1f; 我的技能之一就是單詞翻譯王——其實看了婁老師的學習經驗之后便有些自慚形穢了&#xff0c;我目前的單詞量是7300,扇貝上測的&#xff0c;而且測試時是嚴格的“不會就…

數字后端——電源規劃

電源規劃是給整個芯片的供電設計出一個均勻的網絡&#xff0c;它是芯片物理設計中非常關鍵的一部分。電源規劃在芯片布圖規劃后或在布圖規劃過程中交叉完成,它貫穿于整個設計中&#xff0c;需要在芯片設計的不同階段對電源的供電網絡進行分析并根據要求進行修改。&#xff0c;主…

逆向project實戰--Acid burn

0x00 序言 這是第二次破解 crackme 小程序&#xff0c;感覺明顯比第一次熟練。破解過程非常順利&#xff0c;差點兒是分分鐘就能夠找到正確的 serial&#xff0c;可是我們的目標是破解計算過程。以下將具體介紹。 0x01 初次執行 剛開始拿到 crackme 先執行程序。看看有哪些明顯…

PyCharm使用技巧(六):Regullar Expressions的使用

2019獨角獸企業重金招聘Python工程師標準>>> PyCharm v2018.2最新版本下載 使用正則表達式查找和替換文件中的文本 示例代碼 使用正則表達式查找和替換字符串 假設您想用擴展標記<title> </title>替換元素&#xff08;title&#xff09;中的屬性&#x…

內核中_init,_exit中的作用

__init&#xff0c; __initdata等屬性標志&#xff0c;是要把這種屬性的代碼放入目標文件的.init.text節&#xff0c;數據放入.init.data節──這一過程是通過編譯內核時為相關目標平臺提供了xxx.lds鏈接腳本來指導ld完成的。 對編譯成module的代碼和數據來說&#xff0c;當模…

jQuery筆記總結

來源于&#xff1a;http://blog.poetries.top/2016/10/20/review-jQuery/ http://www.jianshu.com/p/f8e3936b34c9 首先&#xff0c;來了解一下jQuery學習的整體思路 第一節 jQuery初步認知 jQuery概述 JQuery概念 javascript概念 基于Js語言的API和語法組織邏輯&#xff0c;通…

芯片生產流程

每個半導體產品的制造都需要數百個工藝&#xff0c;泛林集團將整個制造過程分為八個步驟&#xff1a;晶圓加工-氧化-光刻-刻蝕-薄膜沉積-互連-測試-封裝。 一、晶圓加工 所有半導體工藝都始于一粒沙子&#xff01;因為沙子所含的硅是生產晶圓所需要的原材料。晶圓是將硅(Si)或砷…

GRE Sub math 報名

Step1 注冊ETS帳號 Step2 登錄帳號&#xff0c;點擊Register/Find Test Centers, Dates Step3 按照提示查詢考場 如果沒有結果而是出現了如下提示&#xff0c;意味著這個地方沒有考位了&#xff0c;需要選擇其他地方的考位 Step 4 接下來就和GRE general test的過程一樣了&…

????platform_device_系列函數及其設備注冊的作用

platform_device_系列函數&#xff0c;實際上是注冊了一個叫platform的虛擬總線。使用約定是如果一個不屬于任何總線的設備&#xff0c;例如藍牙&#xff0c;串口等設備&#xff0c;都需要掛在這個虛擬總線上。 driver/base/platform.c //platform設備聲明 struct device pla…

示例解讀 Python 2 和 Python 3 之間的主要差異

開發四年只會寫業務代碼&#xff0c;分布式高并發都不會還做程序員&#xff1f; 每門編程語言在發布更新之后&#xff0c;主要版本之間都會發生很大的變化。 在本文中&#xff0c;Vinodh Kumar 通過示例解釋了 Python 2 和 Python 3 之間的一些重大差異&#xff0c;以幫助說明…

數字后端——布局

由于I / O單元和模塊的布放已經在布圖規劃時完成&#xff0c;因此布局的剩余任務主要是對標準單元的布局。布局方案在布圖規劃時就已經做了決定&#xff0c;要么選擇展平式布局&#xff0c;要么就是層次化布局。 一、布局目標 布局的目標也即布局內容實施之后所要達到的預期值…