V4L2采集YUYV数据—X264编码H264视频实例

  • Post author:
  • Post category:其他


转自: http://blog.csdn.net/li_wen01/article/details/53557949

前几天在网上买个罗技的C270摄像头,它支持YUYV(YUV422)和JPEG数据输出。它规格书上写的是支持HD720P(1280*720像素),在实际的调试过程中,我使用该分辨率会导致数据采集过慢。这里需要注意一下,罗技的摄像头C270在有些虚拟机上使用是有异常的,有些是不能映射到虚拟机上,有些是映射过去操作非常缓慢。因为之前在自己的开发板上调试过YUV420的摄像头,在此基础上改为YUYV数据格式,调试的时候还是遇到不少的问题。

x264库的编译可以见之前博客:

http://blog.csdn.net/li_wen01/article/details/53571929

在PC上编译X264,可以直接执行下面三条命令:


./configure –enable-shared




make




make install

下面贴出x264部分的代码:



  1. /*=============================================================================



  2. #     FileName: h264encoder.c



  3. #         Desc: this program aim to get image from USB camera,



  4. #               used the V4L2 interface.



  5. #       Author: licaibiao



  6. #      Version:



  7. #   LastChange: 2017-02-21



  8. =============================================================================*/





  9. #include <stdio.h>





  10. #include <stdlib.h>





  11. #include <string.h>





  12. #include “./include/h264encoder.h”






  13. int


    WIDTH =


    6


    4


    0


    ;



  14. int


    HEIGHT =


    4


    8


    0


    ;




  15. void


    compress_begin(


    Encoder


    *en,


    int


    width,


    int


    height) {


  16. en->param = (

    x264_param_t


    *) malloc(


    sizeof


    (x


    2


    6


    4


    _param_t));


  17. en->picture = (

    x264_picture_t


    *) malloc(


    sizeof


    (x


    2


    6


    4


    _picture_t));


  18. x

    2


    6


    4


    _param_default(en->param);


    //set default param





  19. //en->param->rc.i_rc_method = X264_RC_CQP;





  20. // en->param->i_log_level = X264_LOG_NONE;





  21. en->param->i_threads  = X

    2


    6


    4


    _SYNC_LOOKAHEAD_AUTO;


  22. en->param->i_width = width;

    //set frame width




  23. en->param->i_height = height;

    //set frame height




  24. WIDTH = width;

  25. HEIGHT = height;



  26. //en->param->i_frame_total = 0;






  27. //en->param->i_keyint_max = 10;




  28. en->param->rc

    .i_lookahead


    =


    0


    ;



  29. //en->param->i_bframe = 5;






  30. //en->param->b_open_gop = 0;





  31. //en->param->i_bframe_pyramid = 0;





  32. //en->param->i_bframe_adaptive = X264_B_ADAPT_TRELLIS;






  33. //en->param->rc.i_bitrate = 1024 * 10;//rate 10 kbps




  34. en->param->i_fps_num =

    3


    0


    ;


  35. en->param->i_fps_den =

    1


    ;


  36. en->param->i_csp = X

    2


    6


    4


    _CSP_I


    4


    2


    2


    ;



  37. x

    2


    6


    4


    _param_apply_profile(en->param, x


    2


    6


    4


    _profile_names[


    4


    ]);




  38. if


    ((en->handle = x


    2


    6


    4


    _encoder_open(en->param)) ==


    0


    ) {



  39. return


    ;


  40. }


  41. /* Create a new pic */




  42. x

    2


    6


    4


    _picture_alloc(en->picture, X


    2


    6


    4


    _CSP_I


    4


    2


    2


    , en->param->i_width,


  43. en->param->i_height);

  44. }



  45. int


    compress_frame(


    Encoder


    *en,


    int


    type,


    uint8_t


    *in,


    uint8_t


    *out) {


  46. x

    2


    6


    4


    _picture_t pic_out;



  47. int


    index_y, index_u, index_v;



  48. int


    num;



  49. int


    nNal = –


    1


    ;



  50. int


    result =


    0


    ;



  51. int


    i =


    0


    ;



  52. static




    long




    int


    pts =


    0


    ;



  53. uint8_t


    *p_out = out;



  54. char


    char


    *y = en->picture->img


    .plane


    [


    0


    ];



  55. char


    char


    *u = en->picture->img


    .plane


    [


    1


    ];



  56. char


    char


    *v = en->picture->img


    .plane


    [


    2


    ];



  57. char


    char


    * ptr;



  58. index_y =

    0


    ;


  59. index_u =

    0


    ;


  60. index_v =

    0


    ;



  61. num =

    WIDTH


    *


    HEIGHT


    *


    2





    4


    ;




  62. for


    (i=


    0


    ; i<num; i=i+


    4


    )


  63. {

  64. *(y + (index_y++)) = *(in + i);

  65. *(u + (index_u++)) = *(in + i +

    1


    );


  66. *(y + (index_y++)) = *(in + i +

    2


    );


  67. *(v + (index_v++)) = *(in + i +

    3


    );


  68. }



  69. switch


    (type) {



  70. case


    0


    :


  71. en->picture->i_type = X

    2


    6


    4


    _TYPE_P;



  72. break


    ;



  73. case


    1


    :


  74. en->picture->i_type = X

    2


    6


    4


    _TYPE_IDR;



  75. break


    ;



  76. case


    2


    :


  77. en->picture->i_type = X

    2


    6


    4


    _TYPE_I;



  78. break


    ;



  79. default


    :


  80. en->picture->i_type = X

    2


    6


    4


    _TYPE_AUTO;



  81. break


    ;


  82. }


  83. en->picture->i_pts = pts++;



  84. if


    (x


    2


    6


    4


    _encoder_encode(en->handle, &(en->nal), &nNal, en->picture,


  85. &pic_out) <

    0


    ) {



  86. return





    1


    ;


  87. }



  88. for


    (i =


    0


    ; i < nNal; i++) {


  89. memcpy(p_out, en->nal[i]

    .p_payload


    , en->nal[i]


    .i_payload


    );


  90. p_out += en->nal[i]

    .i_payload


    ;


  91. result += en->nal[i]

    .i_payload


    ;


  92. }



  93. return


    result;


  94. }



  95. void


    compress_end(


    Encoder


    *en) {



  96. if


    (en->handle) {


  97. x

    2


    6


    4


    _encoder_close(en->handle);


  98. }


  99. if


    (en->picture) {


  100. x

    2


    6


    4


    _picture_clean(en->picture);


  101. free(en->picture);

  102. en->picture =

    0


    ;


  103. }


  104. if


    (en->param) {


  105. free(en->param);

  106. en->param =

    0


    ;


  107. }

  108. }

上面的代码是配置x264编码器的,有下面几个地方需要特别注意:

(1)CSP参数的配置



  1. en->param->i_csp = X


    2


    6


    4


    _CSP_I


    4


    2


    2


    ;


在X264中默认的i_csp值是3,也就是X264_CSP_NV12 的值,如果采用YUYV(422)输入格式,这个值一定需要重新设置,不然会出现错误提示:x264 [error]: Invalid input colorspace 。这是因为在x264内核中他会把输入格式装换为下面三种中的一种:X264_CSP_NV12,X264_CSP_NV16,X264_CSP_I444.转换如下:



  1. static




    int


    x


    2


    6


    4


    _frame_internal_csp(


    int


    external_csp )


  2. {


  3. switch


    ( external_csp & X


    2


    6


    4


    _CSP_MASK )


  4. {


  5. case


    X264_CSP_NV12


    :



  6. case


    X264_CSP_NV21


    :



  7. case


    X264_CSP_I420


    :



  8. case


    X264_CSP_YV12


    :



  9. return


    X


    2


    6


    4


    _CSP_NV


    1


    2


    ;



  10. case


    X264_CSP_NV16


    :



  11. case


    X264_CSP_I422


    :



  12. case


    X264_CSP_YV16


    :



  13. case


    X264_CSP_V210


    :



  14. return


    X


    2


    6


    4


    _CSP_NV


    1


    6


    ;



  15. case


    X264_CSP_I444


    :



  16. case


    X264_CSP_YV24


    :



  17. case


    X264_CSP_BGR


    :



  18. case


    X264_CSP_BGRA


    :



  19. case


    X264_CSP_RGB


    :



  20. return


    X


    2


    6


    4


    _CSP_I


    4


    4


    4


    ;



  21. default


    :



  22. return


    X


    2


    6


    4


    _CSP_NONE;


  23. }

  24. }


(2)profile类型设置



  1. x


    2


    6


    4


    _param_apply_profile(en->param, x


    2


    6


    4


    _profile_names[


    4


    ]);


在YUV422中,它不支持baseline,默认设置会提示:x264 [error]: baseline profile doesn’t support 4:2:2 可以设置下面的其他参数:



  1. x


    2


    6


    4


    _profile_names[] = {


    “baseline”


    ,


    “main”


    ,


    “high”


    ,


    “high10”


    ,


    “high422”


    ,


    “high444”


    ,


    0


    };

(3)图片内存分配



  1. x


    2


    6


    4


    _picture_alloc(en->picture, X


    2


    6


    4


    _CSP_I


    4


    2


    2


    , en->param->i_width,


  2. en->param->i_height);


这里的第二个参数一定要与你的输入格式先对应,不然的话会出现内存溢出的错误。因为默认的分配图片内存大小是YUV420的。以640*480 分辨率来举例,YUV420 分配一帧图像的内存是450K,而我们YUV422的数据量是600K。





(4)Y,U,V 数据需要分离



  1. for


    (i=


    0


    ; i<num; i=i+


    4


    )


  2. {

  3. *(y + (index_y++)) = *(in + i);

  4. *(u + (index_u++)) = *(in + i +

    1


    );


  5. *(y + (index_y++)) = *(in + i +

    2


    );


  6. *(v + (index_v++)) = *(in + i +

    3


    );


  7. }


YUYV的数据是交错存储的,因此需要把他们分离出来单独存储,如果这里不做处理,图像就会出现异常。

(5)i_pts 参数需要递增



  1. en->picture->i_pts = pts++;


i_pts = pts的参数是需要递增的,不让回出现警告:x264 [warning]: non-strictly-monotonic PTS

完整的编译运行结果如下:



  1. [root


    @redhat


    test


    ]# ls


  2. h

    2


    6


    4


    encoder


    .c


    include  lib  main


    .c


    Makefile  out


  3. [root

    @redhat


    test


    ]# make


  4. gcc -g    -c -o main

    .o


    main


    .c




  5. gcc -g    -c -o h

    2


    6


    4


    encoder


    .o


    h


    2


    6


    4


    encoder


    .c




  6. gcc -g -o x

    2


    6


    4


    _test main


    .o


    h


    2


    6


    4


    encoder


    .o


    -lpthread -lx


    2


    6


    4


    -lm


  7. [root

    @redhat


    test


    ]# ls


  8. h

    2


    6


    4


    encoder


    .c


    h


    2


    6


    4


    encoder


    .o


    include  lib  main


    .c


    main


    .o


    Makefile  out  x


    2


    6


    4


    _test


  9. [root

    @redhat


    test


    ]# ./x


    2


    6


    4


    _test



  10. camera driver name is : uvcvideo

  11. camera device name is : UVC Camera (

    0


    4


    6


    d:


    0


    8


    2


    5


    )


  12. camera bus

    information


    : usb-


    0


    0


    0


    0


    :


    0


    0


    :


    1


    a


    .0





    1


    .1




  13. n_buffer =

    4




  14. x

    2


    6


    4


    [warning]: lookaheadless mb-tree requires intra refresh or infinite keyint


  15. x

    2


    6


    4


    [info]:


    using


    cpu


    capabilities


    : MMX


    2


    SSE


    2


    Fast SSSE


    3


    SSE


    4


    .2


    AVX


  16. x

    2


    6


    4


    [info]: profile High


    4


    :


    2


    :


    2


    , level


    3


    .0


    ,


    4


    :


    2


    :


    2




    8


    -bit


  17. spend time

    8


    5


    .082031


    s


  18. x

    2


    6


    4


    [info]: frame


    I


    :


    8


    Avg


    QP


    :


    2


    0


    .27




    size


    :


    2


    1


    5


    9


    2




  19. x

    2


    6


    4


    [info]: frame


    P


    :


    5


    0


    3


    Avg


    QP


    :


    2


    1


    .18




    size


    :


    3


    1


    1


    9




  20. x

    2


    6


    4


    [info]: frame


    B


    :


    1


    4


    8


    5


    Avg


    QP


    :


    2


    2


    .03




    size


    :


    9


    5


    2




  21. x

    2


    6


    4


    [info]: consecutive B-frames:


    0


    .8


    %


    0


    .0


    %


    0


    .0


    %


    9


    9


    .2


    %


  22. x

    2


    6


    4


    [info]: mb I  I


    1


    6


    .


    .4


    :


    1


    1


    .9


    %


    5


    5


    .2


    %


    3


    2


    .9


    %


  23. x

    2


    6


    4


    [info]: mb P  I


    1


    6


    .


    .4


    :


    0


    .4


    %


    0


    .2


    %


    0


    .1


    %  P


    1


    6


    .


    .4


    :


    4


    4


    .8


    %


    7


    .9


    %


    8


    .5


    %


    0


    .0


    %


    0


    .0


    %


    skip


    :


    3


    8


    .2


    %


  24. x

    2


    6


    4


    [info]: mb B  I


    1


    6


    .


    .4


    :


    0


    .0


    %


    0


    .0


    %


    0


    .0


    %  B


    1


    6


    .


    .8


    :


    2


    5


    .9


    %


    0


    .6


    %


    0


    .1


    %


    direct


    :


    1


    .7


    %


    skip


    :


    7


    1


    .7


    %


    L0


    :


    5


    1


    .6


    %


    L1


    :


    4


    7


    .0


    %


    BI


    :


    1


    .4


    %


  25. x

    2


    6


    4


    [info]:


    8


    x


    8


    transform


    intra


    :


    4


    6


    .7


    %


    inter


    :


    9


    5


    .7


    %


  26. x

    2


    6


    4


    [info]: coded y,uvDC,uvAC


    intra


    :


    6


    0


    .5


    %


    8


    7


    .6


    %


    5


    9


    .7


    %


    inter


    :


    5


    .7


    %


    2


    3


    .2


    %


    0


    .9


    %


  27. x

    2


    6


    4


    [info]: i


    1


    6


    v,h,dc,p:


    4


    %


    8


    %


    1


    %


    8


    7


    %


  28. x

    2


    6


    4


    [info]: i


    8


    v,h,dc,ddl,ddr,vr,hd,vl,hu:


    1


    6


    %


    3


    9


    %


    1


    2


    %


    3


    %


    5


    %


    4


    %


    1


    0


    %


    4


    %


    7


    %


  29. x

    2


    6


    4


    [info]: i


    4


    v,h,dc,ddl,ddr,vr,hd,vl,hu:


    2


    4


    %


    4


    1


    %


    8


    %


    3


    %


    4


    %


    3


    %


    8


    %


    3


    %


    4


    %


  30. x

    2


    6


    4


    [info]: i


    8


    c dc,h,v,p:


    5


    0


    %


    2


    2


    %


    2


    1


    %


    6


    %


  31. x

    2


    6


    4


    [info]: Weighted P-Frames:


    Y


    :


    0


    .2


    %


    UV


    :


    0


    .0


    %


  32. x

    2


    6


    4


    [info]: ref P


    L0


    :


    4


    0


    .2


    %


    4


    .8


    %


    3


    9


    .3


    %


    1


    5


    .7


    %


  33. x

    2


    6


    4


    [info]: ref B


    L0


    :


    6


    5


    .6


    %


    2


    0


    .4


    %


    1


    4


    .0


    %


  34. x

    2


    6


    4


    [info]: ref B


    L1


    :


    9


    1


    .2


    %


    8


    .8


    %


  35. x

    2


    6


    4


    [info]: kb/s:


    3


    7


    9


    .47




  36. [root

    @redhat


    test


    ]#


这里设置的分辨率是640*480 ,这样采集数据比较快。我编码2000帧数据需要的是大约85s,帧率大约在23fps 。也不知道具体是什么原因导致耗时这么长时间。

视频运行如下: