首先是
//四字符代码,为了标示视频数据流格式 //在次文件的下,比如大家熟悉的mjpeg //#define V4L2_PIX_FMT_MJPEG v4l2_fourcc(’M’, ’J’, ’P’, ’G’) /* Motion-JPEG */ /* Four-character-code (FOURCC) */ #define v4l2_fourcc(a, b, c, d) ((__u32)(a) | ((__u32)(b) << 8) | ((__u32)(c) << 16) | ((__u32)(d) << 24))
/* * F O R M A T E N U M E R A T I O N */ struct v4l2_fmtdesc { __u32 index; /* Format number */ //格式索引 enum v4l2_buf_type type; /* buffer type */ //缓冲类型 /* ioctl会根据这个type选择对应的函数,例如VIDIOC_ENUM_FMT时type为V4L2_BUF_TYPE_VIDEO_CAPTURE, 那么就会调用 int (*vidioc_enum_fmt_vid_cap)(struct file *file, void *fh, struct v4l2_fmtdesc *f); enum v4l2_buf_type { V4L2_BUF_TYPE_VIDEO_CAPTURE = 1, V4L2_BUF_TYPE_VIDEO_OUTPUT = 2, V4L2_BUF_TYPE_VIDEO_OVERLAY = 3, V4L2_BUF_TYPE_VBI_CAPTURE = 4, V4L2_BUF_TYPE_VBI_OUTPUT = 5, V4L2_BUF_TYPE_SLICED_VBI_CAPTURE = 6, V4L2_BUF_TYPE_SLICED_VBI_OUTPUT = 7, V4L2_BUF_TYPE_VIDEO_OUTPUT_OVERLAY = 8, V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE = 9, V4L2_BUF_TYPE_VIDEO_OUTPUT_MPLANE = 10, V4L2_BUF_TYPE_PRIVATE = 0x80, }; */ __u32 flags;//标志 __u8 description[32]; /* Description string */ //描述 __u32 pixelformat; /* Format fourcc */ //用定义好的的宏v4l2_fourcc __u32 reserved[4]; }; //经常在应用层使用VIDIOC_ENUM_FMT去获取支持的格式,主要就是获取这个结构体。有的可能支持几种格式,可以通过 //设置v4l2_fmtdesc对应的index去获取。
/** * struct v4l2_format - stream data format * @type: type of the data stream * @pix: definition of an image format * @pix_mp: definition of a multiplanar image format * @win: definition of an overlaid image * @vbi: raw VBI capture or output parameters * @sliced: sliced VBI capture or output parameters * @raw_data: placeholder for future extensions and custom formats */ //这个结构体在设置格式和获取当然格式时会用到。 struct v4l2_format { enum v4l2_buf_type type;//类型,已说过 union { struct v4l2_pix_format pix; /* V4L2_BUF_TYPE_VIDEO_CAPTURE */ struct v4l2_pix_format_mplane pix_mp; /* V4L2_BUF_TYPE_VIDEO_CAPTURE_MPLANE */ struct v4l2_window win; /* V4L2_BUF_TYPE_VIDEO_OVERLAY */ struct v4l2_vbi_format vbi; /* V4L2_BUF_TYPE_VBI_CAPTURE */ struct v4l2_sliced_vbi_format sliced; /* V4L2_BUF_TYPE_SLICED_VBI_CAPTURE */ __u8 raw_data[200]; /* user-defined */ } fmt; };
下面我们重点说union中的东西,vbi就不说了
struct v4l2_pix_format { __u32 width; //宽 __u32 height; //高 __u32 pixelformat;//图素格式, rgb565 rgb888 yuv422 yuv420等等 enum v4l2_field field; //场格式,下面详细说 __u32 bytesperline; /* for padding, zero if unused */ //表明缓冲区中有多少字节用于表示图像中一行像素的所有像素值。 //由于一个像素可能有多个字节表示,所以 bytesPerLine 可能是字段 width 值的若干倍 __u32 sizeimage;//图像大小 enum v4l2_colorspace colorspace;//色彩空间 SMPTE170M等 http://linuxtv.org/downloads/v4l-dvb-apis/colorspaces.html介绍了转换方法 //http://vektor.theorem.ca/graphics/ycbcr/介绍 __u32 priv; /* private data, depends on pixelformat */ };
enum v4l2_field { V4L2_FIELD_ANY = 0, /* driver can choose from none, top, bottom, interlaced depending on whatever it thinks is approximate ... V4L2_FIELD_NONE = 1, /* this device has no fields ... V4L2_FIELD_TOP = 2, /* top field only V4L2_FIELD_BOTTOM = 3, /* bottom field only V4L2_FIELD_INTERLACED = 4, /* both fields interlaced V4L2_FIELD_SEQ_TB = 5, /* both fields sequential into one buffer, top-bottom order V4L2_FIELD_SEQ_BT = 6, /* same as above + bottom-top order V4L2_FIELD_ALTERNATE = 7, /* both fields alternating into separate buffers V4L2_FIELD_INTERLACED_TB = 8, /* both fields interlaced, top field first and the top field is transmitted first V4L2_FIELD_INTERLACED_BT = 9, /* both fields interlaced, top field first and the bottom field is transmitted first };
隔行和逐行:video的帧可区分为隔行和逐行:
逐行顺序的传输一帧所有的行,而隔行则把一帧划分成两个fields,分别保存帧的奇数行和偶数行,被称作奇场和偶场.
阴极射线电视机需要交替的显示两场来组成一个完整的帧,交替的时延需要我们交替的传输奇场和偶场。这个奇怪技术的引入是因为:在刷新率接近电影时,图片会
消退的过快。使用奇偶场可以避免使用双倍的buffer以及额外的带宽需求。
瞬间序:首先要明确模拟camera(数字摄像头不在这个讨论之列。)并不是在同一时间曝光一帧,camera通过场来传输这些帧的,这些场是在不
同瞬间拍摄的。屏幕上的一个运动对象因此会在两个field之间产生动画效果。这种情况下需要识别哪一帧更老一点,也称作“瞬间序”。
顶场和低场:当驱动通过场提供或者接收images,应用应该知道如何通过这些场组合成帧,通过划分为top bottom
场,顶场的第一行是帧的第一行,底场的第一行是帧的第二行。然而因为field是一个跟着一个拍的,争论帧是由top还是bottom开始的是没意义的,
任何两个相邻的top bottom场, 或者 bottom top场都可以组成一个有效的帧。与直觉相反,top 场不一定排在bottom前面,
top和bottom到底谁先谁后,是由video标准决定的。因此要区分瞬间序和空间序,下面的图会给出清晰的解释。
我只说一下:PAL + V4L2_FIELD_INTERLACED_BT
从上图看PAL下 top: 红绿红 bottom:蓝黄蓝
V4L2_FIELD_INTERLACED_BT是两场隔行扫描,top在先,bottom场先发送
那就会是黄红蓝绿,就出现了红黄和绿蓝彩条。其他的自己分析吧!
struct v4l2_plane_pix_format { __u32 sizeimage; //被使用层的最大字节数 __u16 bytesperline;//每行字节数 __u16 reserved[7]; } __attribute__ ((packed)); /** * struct v4l2_pix_format_mplane - multiplanar format definition * @width: image width in pixels * @height: image height in pixels * @pixelformat: little endian four character code (fourcc) * @field: field order (for interlaced video) * @colorspace: supplemental to pixelformat * @plane_fmt: per-plane information * @num_planes: number of planes for this format */ struct v4l2_pix_format_mplane { __u32 width; //宽 __u32 height; //高 __u32 pixelformat; //像素格式 fourcc定义 enum v4l2_field field; //见struct v4l2_pix_format enum v4l2_colorspace colorspace; //struct v4l2_pix_format struct v4l2_plane_pix_format plane_fmt[VIDEO_MAX_PLANES]; //每平面信息 __u8 num_planes; //此格式的平面数 __u8 reserved[11]; } __attribute__ ((packed));
多层的格式有,只是举几个例子
/* two non contiguous planes - one Y, one Cr + Cb interleaved */ //两层的 #define V4L2_PIX_FMT_NV12M v4l2_fourcc(’N’, ’M’, ’1’, ’2’) /* 12 Y/CbCr 4:2:0 */ #define V4L2_PIX_FMT_NV12MT v4l2_fourcc(’T’, ’M’, ’1’, ’2’) /* 12 Y/CbCr 4:2:0 64x32 macroblocks */ /* three non contiguous planes - Y, Cb, Cr */ //三层的 #define V4L2_PIX_FMT_YUV420M v4l2_fourcc(’Y’, ’M’, ’1’, ’2’) /* 12 YUV420 planar */
struct v4l2_clip { struct v4l2_rect c; struct v4l2_clip __user *next; }; //这是video overlay使用的结构体 struct v4l2_window { struct v4l2_rect w; //窗口位置 enum v4l2_field field; //也说过 __u32 chromakey; //色度:色调和饱和度 struct v4l2_clip __user *clips; //剪切 __u32 clipcount; //剪切计数 void __user *bitmap; //每个位对应覆盖图像的一个像素,此位置一时对应像素才显示。 __u8 global_alpha; //全局alpha值 };
vbi不说了