瑞芯微Rk3288硬件编码,硬件解码

main.c,实例demo。

[cpp]  view plain  copy
  1. #include "stdio.h"  
  2. #include "vpu_global.h"  
  3. #include "utils/Log.h"  
  4.   
  5. #include "vpu_api.h"  
  6.   
  7. #undef LOG_TAG  
  8. #define LOG_TAG "vpu_api_demo"  
  9.   
  10. static RK_U32 VPU_API_DEMO_DEBUG_DISABLE = 0;  
  11.   
  12. #define VPU_API_DEMO_DEBUG 1  
  13.   
  14. #ifdef  VPU_API_DEMO_DEBUG  
  15. #ifdef AVS40  
  16. #define VPU_DEMO_LOG(...)   do { if (VPU_API_DEMO_DEBUG_DISABLE==0) LOGI(__VA_ARGS__); } while (0)  
  17. #else  
  18. #define VPU_DEMO_LOG(...)   do { if (VPU_API_DEMO_DEBUG_DISABLE==0) ALOGI(__VA_ARGS__); } while (0)  
  19. #endif  
  20. #else  
  21. #define VPU_DEMO_LOG  
  22. #endif  
  23.   
  24. #define BSWAP32(x) \  
  25.     ((((x) & 0xff000000) >> 24) | (((x) & 0x00ff0000) >>  8) | \  
  26.       (((x) & 0x0000ff00) <<  8) | (((x) & 0x000000ff) << 24))  
  27.   
  28.   
  29. #define DEMO_ERR_RET(err) do { ret = err; goto DEMO_OUT; } while (0)  
  30. #define DECODE_ERR_RET(err) do { ret = err; goto DECODE_OUT; } while (0)  
  31. #define ENCODE_ERR_RET(err) do { ret = err; goto ENCODE_OUT; } while (0)  
  32.   
  33.   
  34. typedef enum VPU_API_DEMO_RET {  
  35.     VPU_DEMO_OK = 0,  
  36.     VPU_DEMO_PARSE_HELP_OK  = 1,  
  37.   
  38.     VPU_DEMO_ERROR_BASE     = -100,  
  39.     ERROR_INVALID_PARAM     = VPU_DEMO_ERROR_BASE - 1,  
  40.     ERROR_INVALID_STREAM    = VPU_DEMO_ERROR_BASE - 2,  
  41.     ERROR_IO                = VPU_DEMO_ERROR_BASE - 3,  
  42.     ERROR_MEMORY            = VPU_DEMO_ERROR_BASE - 4,  
  43.     ERROR_INIT_VPU          = VPU_DEMO_ERROR_BASE - 5,  
  44.   
  45.     ERROR_VPU_DECODE        = VPU_DEMO_ERROR_BASE - 90,  
  46. } VPU_API_DEMO_RET;  
  47.   
  48. typedef struct VpuApiCmd {  
  49.     RK_U8* name;  
  50.     RK_U8* argname;  
  51.     RK_U8* help;  
  52. }VpuApiCmd_t;  
  53.   
  54. typedef struct VpuApiDemoCmdContext {  
  55.     RK_U32  width;  
  56.     RK_U32  height;  
  57.     CODEC_TYPE  codec_type;  
  58.     OMX_ON2_VIDEO_CODINGTYPE coding;  
  59.     RK_U8   input_file[200];  
  60.     RK_U8   output_file[200];  
  61.     RK_U8   have_input;  
  62.     RK_U8   have_output;  
  63.     RK_U8   disable_debug;  
  64.     RK_U32  record_frames;  
  65.     RK_S64  record_start_ms;  
  66. }VpuApiDemoCmdContext_t;  
  67.   
  68. typedef struct VpuApiEncInput {  
  69.     EncInputStream_t stream;  
  70.     RK_U32 capability;  
  71. }VpuApiEncInput;  
  72.   
  73. static VpuApiCmd_t vpuApiCmd[] = {  
  74.     {"i",               "input_file",           "input bitstream file"},  
  75.     {"o",               "output_file",          "output bitstream file, "},  
  76.     {"w",               "width",                "the width of input bitstream"},  
  77.     {"h",               "height",               "the height of input bitstream"},  
  78.     {"t",               "codec_type",           "the codec type, dec: deoder, enc: encoder, default: decoder"},  
  79.     {"coding",          "coding_type",          "encoding type of the bitstream"},  
  80.     {"vframes",         "number",               "set the number of video frames to record"},  
  81.     {"ss",              "time_off",             "set the start time offset, use Ms as the unit."},  
  82.     {"d",               "disable",              "disable the debug output info."},  
  83. };  
  84.   
  85. static void show_usage()  
  86. {  
  87.     VPU_DEMO_LOG("usage: vpu_apiDemo [options] input_file, \n\n");  
  88.   
  89.     VPU_DEMO_LOG("Getting help:\n");  
  90.     VPU_DEMO_LOG("-help  --print options of vpu api demo\n");  
  91. }  
  92.   
  93. static RK_S32 show_help()  
  94. {  
  95.     VPU_DEMO_LOG("usage: vpu_apiDemo [options] input_file, \n\n");  
  96.   
  97.     RK_S32 i =0;  
  98.     RK_U32 n = sizeof(vpuApiCmd)/sizeof(VpuApiCmd_t);  
  99.     for (i =0; i 
  100.         VPU_DEMO_LOG("-%s  %s\t\t%s\n",  
  101.             vpuApiCmd[i].name, vpuApiCmd[i].argname, vpuApiCmd[i].help);  
  102.     }  
  103.   
  104.     return 0;  
  105. }  
  106.   
  107. static RK_S32 parse_options(int argc, char **argv, VpuApiDemoCmdContext_t* cmdCxt)  
  108. {  
  109.     RK_S8 *opt;  
  110.     RK_U32 optindex, handleoptions = 1, ret =0;  
  111.   
  112.     if ((argc <2) || (cmdCxt == NULL)) {  
  113.         VPU_DEMO_LOG("vpu api demo, input parameter invalid\n");  
  114.         show_usage();  
  115.         return ERROR_INVALID_PARAM;  
  116.     }  
  117.   
  118.     /* parse options */  
  119.     optindex = 1;  
  120.     while (optindex < argc) {  
  121.         opt = argv[optindex++];  
  122.   
  123.         if (handleoptions && opt[0] == '-' && opt[1] != '\0') {  
  124.             if (opt[1] == '-') {  
  125.                 if (opt[2] != '\0') {  
  126.                     opt++;  
  127.                 } else {  
  128.                      handleoptions = 0;  
  129.                     continue;  
  130.                 }  
  131.             }  
  132.   
  133.             opt++;  
  134.   
  135.             switch (*opt) {  
  136.                 case 'i':  
  137.                     if (argv[optindex]) {  
  138.                         memcpy(cmdCxt->input_file, argv[optindex], strlen(argv[optindex]));  
  139.                         cmdCxt->input_file[strlen(argv[optindex])] = '\0';  
  140.                         cmdCxt->have_input = 1;  
  141.                     } else {  
  142.                         VPU_DEMO_LOG("input file is invalid\n");  
  143.                         ret = -1;  
  144.                         goto PARSE_OPINIONS_OUT;  
  145.                     }  
  146.                     break;  
  147.                 case 'o':  
  148.                     if (argv[optindex]) {  
  149.                         memcpy(cmdCxt->output_file, argv[optindex], strlen(argv[optindex]));  
  150.                         cmdCxt->output_file[strlen(argv[optindex])] = '\0';  
  151.                         cmdCxt->have_output = 1;  
  152.                         break;  
  153.                     } else {  
  154.                         VPU_DEMO_LOG("out file is invalid\n");  
  155.                         ret = -1;  
  156.                         goto PARSE_OPINIONS_OUT;  
  157.                     }  
  158.                 case 'd':  
  159.                     cmdCxt->disable_debug = 1;  
  160.                     break;  
  161.                 case 'w':  
  162.                     if (argv[optindex]) {  
  163.                         cmdCxt->width = atoi(argv[optindex]);  
  164.                         break;  
  165.                     } else {  
  166.                         VPU_DEMO_LOG("input width is invalid\n");  
  167.                         ret = -1;  
  168.                         goto PARSE_OPINIONS_OUT;  
  169.                     }  
  170.                 case 'h':  
  171.                     if ((*(opt+1) != '\0') && !strncmp(opt, "help", 4)) {  
  172.                         show_help();  
  173.                         ret = VPU_DEMO_PARSE_HELP_OK;  
  174.                         goto PARSE_OPINIONS_OUT;  
  175.                     } else if (argv[optindex]) {  
  176.                         cmdCxt->height = atoi(argv[optindex]);  
  177.                     } else {  
  178.                         VPU_DEMO_LOG("input height is invalid\n");  
  179.                         ret = -1;  
  180.                         goto PARSE_OPINIONS_OUT;  
  181.                     }  
  182.                     break;  
  183.                 case 't':  
  184.                     if (argv[optindex]) {  
  185.                         cmdCxt->codec_type = atoi(argv[optindex]);  
  186.                         break;  
  187.                     } else {  
  188.                         VPU_DEMO_LOG("input codec_type is invalid\n");  
  189.                         ret = -1;  
  190.                         goto PARSE_OPINIONS_OUT;  
  191.                     }  
  192.   
  193.                 default:  
  194.                     if ((*(opt+1) != '\0') && argv[optindex]) {  
  195.                         if (!strncmp(opt, "coding", 6)) {  
  196.                             VPU_DEMO_LOG("coding, argv[optindex]: %s",  
  197.                                     argv[optindex]);  
  198.                             cmdCxt->coding = atoi(argv[optindex]);  
  199.                         } else if (!strncmp(opt, "vframes", 7)) {  
  200.                             cmdCxt->record_frames = atoi(argv[optindex]);  
  201.                         } else if (!strncmp(opt, "ss", 2)) {  
  202.                             cmdCxt->record_start_ms = atoi(argv[optindex]);  
  203.                         } else {  
  204.                             ret = -1;  
  205.                             goto PARSE_OPINIONS_OUT;  
  206.                         }  
  207.                     } else {  
  208.                         ret = -1;  
  209.                         goto PARSE_OPINIONS_OUT;  
  210.                     }  
  211.                     break;  
  212.             }  
  213.   
  214.             optindex += ret;  
  215.         }  
  216.     }  
  217.   
  218. PARSE_OPINIONS_OUT:  
  219.     if (ret <0) {  
  220.         VPU_DEMO_LOG("vpu api demo, input parameter invalid\n");  
  221.         show_usage();  
  222.         return ERROR_INVALID_PARAM;  
  223.     }  
  224.     return ret;  
  225. }  
  226.   
  227. static RK_S32 readBytesFromFile(RK_U8* buf, RK_S32 aBytes, FILE* file)  
  228. {  
  229.     if ((NULL ==buf) || (NULL ==file) || (0 ==aBytes)) {  
  230.         return -1;  
  231.     }  
  232.   
  233.     RK_S32 ret = fread(buf, 1, aBytes, file);  
  234.     if(ret != aBytes)  
  235.     {  
  236.         VPU_DEMO_LOG("read %d bytes from file fail", aBytes);  
  237.         return -1;  
  238.     }  
  239.   
  240.     return 0;  
  241. }  
  242.   
  243. static RK_S32 vpu_encode_demo(VpuApiDemoCmdContext_t *cmd)  
  244. {  
  245.     if (cmd == NULL) {  
  246.         return -1;  
  247.     }  
  248.   
  249.     FILE* pInFile = NULL;  
  250.     FILE* pOutFile = NULL;  
  251.     struct VpuCodecContext *ctx = NULL;  
  252.     RK_S32 nal = 0x00000001;  
  253.     RK_S32 fileSize, frame_count, ret, size;  
  254.     EncoderOut_t    enc_out_yuv;  
  255.     EncoderOut_t *enc_out = NULL;  
  256.     VpuApiEncInput enc_in_strm;  
  257.     VpuApiEncInput *api_enc_in = &enc_in_strm;  
  258.     EncInputStream_t *enc_in =NULL;  
  259.     EncParameter_t *enc_param = NULL;  
  260.     RK_S64 fakeTimeUs =0;  
  261.   
  262.     if ((cmd->have_input == NULL) || (cmd->width <=0) || (cmd->height <=0)  
  263.             || (cmd->coding <= OMX_ON2_VIDEO_CodingAutoDetect)) {  
  264.         VPU_DEMO_LOG("Warning: missing needed parameters for vpu api demo\n");  
  265.     }  
  266.   
  267.     if (cmd->have_input) {  
  268.         VPU_DEMO_LOG("input bitstream w: %d, h: %d, coding: %d(%s), path: %s\n",  
  269.             cmd->width, cmd->height, cmd->coding,  
  270.             cmd->codec_type == CODEC_DECODER ? "decode" : "encode",  
  271.             cmd->input_file);  
  272.   
  273.         pInFile = fopen(cmd->input_file, "rb");  
  274.         if (pInFile == NULL) {  
  275.             VPU_DEMO_LOG("input file not exsist\n");  
  276.             ENCODE_ERR_RET(ERROR_INVALID_PARAM);  
  277.         }  
  278.     } else {  
  279.         VPU_DEMO_LOG("please set input bitstream file\n");  
  280.         ENCODE_ERR_RET(ERROR_INVALID_PARAM);  
  281.     }  
  282.   
  283.     if (cmd->have_output) {  
  284.         VPU_DEMO_LOG("vpu api demo output file: %s\n",  
  285.             cmd->output_file);  
  286.         pOutFile = fopen(cmd->output_file, "wb");  
  287.         if (pOutFile == NULL) {  
  288.             VPU_DEMO_LOG("can not write output file\n");  
  289.             ENCODE_ERR_RET(ERROR_INVALID_PARAM);  
  290.         }  
  291.     }  
  292.   
  293.     fseek(pInFile, 0L, SEEK_END);  
  294.     fileSize = ftell(pInFile);  
  295.     fseek(pInFile, 0L, SEEK_SET);  
  296.   
  297.     memset(&enc_in_strm, 0, sizeof(VpuApiEncInput));  
  298.     enc_in = &enc_in_strm.stream;  
  299.     enc_in->buf = NULL;  
  300.   
  301.     memset(&enc_out_yuv, 0, sizeof(EncoderOut_t));  
  302.     enc_out = &enc_out_yuv;  
  303.     enc_out->data = (RK_U8*)malloc(cmd->width * cmd->height);  
  304.     if (enc_out->data == NULL) {  
  305.         ENCODE_ERR_RET(ERROR_MEMORY);  
  306.     }  
  307.   
  308.     ret = vpu_open_context(&ctx);  
  309.     if (ret || (ctx ==NULL)) {  
  310.         ENCODE_ERR_RET(ERROR_MEMORY);  
  311.     }  
  312.   
  313.     /* 
  314.      ** now init vpu api context. codecType, codingType, width ,height 
  315.      ** are all needed before init. 
  316.     */  
  317.     ctx->codecType = cmd->codec_type;  
  318.     ctx->videoCoding = cmd->coding;  
  319.     ctx->width = cmd->width;  
  320.     ctx->height = cmd->height;  
  321.     ctx->no_thread = 1;  
  322.   
  323.     ctx->private_data = malloc(sizeof(EncParameter_t));  
  324.     memset(ctx->private_data,0,sizeof(EncParameter_t));  
  325.   
  326.     enc_param = (EncParameter_t*)ctx->private_data;  
  327.     enc_param->width = cmd->width;  
  328.     enc_param->height = cmd->height;  
  329.     enc_param->bitRate = 100000;  
  330.     enc_param->framerate = 25;  
  331.     enc_param->enableCabac   = 0;  
  332.     enc_param->cabacInitIdc  = 0;  
  333.     enc_param->intraPicRate  = 30;  
  334.   
  335.     if ((ret = ctx->init(ctx, NULL, 0)) !=0) {  
  336.        VPU_DEMO_LOG("init vpu api context fail, ret: 0x%X", ret);  
  337.        ENCODE_ERR_RET(ERROR_INIT_VPU);  
  338.     }  
  339.   
  340.     /* 
  341.      ** init of VpuCodecContext while running encode, it returns 
  342.      ** sps and pps of encoder output, you need to save sps and pps 
  343.      ** after init. 
  344.     */  
  345.     VPU_DEMO_LOG("encode init ok, sps len: %d", ctx->extradata_size);  
  346.     if(pOutFile && (ctx->extradata_size >0)) {  
  347.         VPU_DEMO_LOG("dump %d bytes enc output stream to file",  
  348.             ctx->extradata_size);  
  349.   
  350.         /* save sps and pps */  
  351.         fwrite(ctx->extradata, 1, ctx->extradata_size, pOutFile);  
  352.         fflush(pOutFile);  
  353.     }  
  354.   
  355.     /* 
  356.      ** vpu api encode process. 
  357.     */  
  358.     VPU_DEMO_LOG("init vpu api context ok, input yuv stream file size: %d", fileSize);  
  359.     RK_U32 w_align = ((ctx->width + 15) & (~15));  
  360.     RK_U32 h_align = ((ctx->height + 15) & (~15));  
  361.     size = w_align * h_align * 3/2;  
  362.     nal = BSWAP32(nal);  
  363.   
  364.     do {  
  365.         if (ftell(pInFile) >=fileSize) {  
  366.            VPU_DEMO_LOG("read end of file, complete");  
  367.            break;  
  368.         }  
  369.   
  370.         if (enc_in && (enc_in->size ==0)) {  
  371.             if (enc_in->buf == NULL) {  
  372.                 enc_in->buf = (RK_U8*)(malloc)(size);  
  373.                 if (enc_in->buf == NULL) {  
  374.                     ENCODE_ERR_RET(ERROR_MEMORY);  
  375.                 }  
  376.                 api_enc_in->capability = size;  
  377.             }  
  378.   
  379.             if (api_enc_in->capability <((RK_U32)size)) {  
  380.                 enc_in->buf = (RK_U8*)(realloc)((void*)(enc_in->buf), size);  
  381.                 if (enc_in->buf == NULL) {  
  382.                     ENCODE_ERR_RET(ERROR_MEMORY);  
  383.                 }  
  384.                 api_enc_in->capability = size;  
  385.             }  
  386.   
  387.             if (readBytesFromFile(enc_in->buf, size, pInFile)) {  
  388.                 break;  
  389.             } else {  
  390.                 enc_in->size = size;  
  391.                 enc_in->timeUs = fakeTimeUs;  
  392.                 fakeTimeUs +=40000;  
  393.             }  
  394.   
  395.             VPU_DEMO_LOG("read one frame, size: %d, timeUs: %lld, filePos: %ld",  
  396.                 enc_in->size, enc_in->timeUs , ftell(pInFile));  
  397.         }  
  398.   
  399.         if ((ret = ctx->encode(ctx, enc_in, enc_out)) !=0) {  
  400.            ENCODE_ERR_RET(ERROR_VPU_DECODE);  
  401.         } else {  
  402.             VPU_DEMO_LOG("vpu encode one frame, out len: %d, left size: %d",  
  403.                 enc_out->size, enc_in->size);  
  404.   
  405.             /* 
  406.              ** encoder output stream is raw bitstream, you need to add nal 
  407.              ** head by yourself. 
  408.             */  
  409.             if ((enc_out->size) && (enc_out->data)) {  
  410.                 if(pOutFile) {  
  411.                     VPU_DEMO_LOG("dump %d bytes enc output stream to file",  
  412.                         enc_out->size);  
  413.                     fwrite((uint8_t*)&nal, 1, 4, pOutFile);  
  414.                     fwrite(enc_out->data, 1, enc_out->size, pOutFile);  
  415.                     fflush(pOutFile);  
  416.                 }  
  417.   
  418.                 enc_out->size = 0;  
  419.             }  
  420.         }  
  421.   
  422.         usleep(30);  
  423.     }while(1);  
  424.   
  425. ENCODE_OUT:  
  426.     if (enc_in && enc_in->buf) {  
  427.         free(enc_in->buf);  
  428.         enc_in->buf = NULL;  
  429.     }  
  430.     if (enc_out && (enc_out->data)) {  
  431.         free(enc_out->data);  
  432.         enc_out->data = NULL;  
  433.     }  
  434.     if (ctx) {  
  435.         if (ctx->private_data) {  
  436.             free(ctx->private_data);  
  437.             ctx->private_data = NULL;  
  438.         }  
  439.         vpu_close_context(&ctx);  
  440.         ctx = NULL;  
  441.     }  
  442.     if (pInFile) {  
  443.         fclose(pInFile);  
  444.         pInFile = NULL;  
  445.     }  
  446.     if (pOutFile) {  
  447.         fclose(pOutFile);  
  448.         pOutFile = NULL;  
  449.     }  
  450.   
  451.     if (ret) {  
  452.         VPU_DEMO_LOG("encode demo fail, err: %d", ret);  
  453.     } else {  
  454.         VPU_DEMO_LOG("encode demo complete OK.");  
  455.     }  
  456.     return ret;  
  457.   
  458. }  
  459.   
  460. static RK_S32 vpu_decode_demo(VpuApiDemoCmdContext_t *cmd)  
  461. {  
  462.     if (cmd == NULL) {  
  463.         return -1;  
  464.     }  
  465.   
  466.     FILE* pInFile = NULL;  
  467.     FILE* pOutFile = NULL;  
  468.     struct VpuCodecContext* ctx = NULL;  
  469.     RK_S32 fileSize =0, pkt_size =0;  
  470.     RK_S32 ret = 0, frame_count = 0;  
  471.     DecoderOut_t    decOut;  
  472.     VideoPacket_t demoPkt;  
  473.     VideoPacket_t* pkt =NULL;  
  474.     DecoderOut_t *pOut = NULL;  
  475.     VPU_FRAME *frame = NULL;  
  476.     RK_S64 fakeTimeUs =0;  
  477.     RK_U8* pExtra = NULL;  
  478.     RK_U32 extraSize = 0;  
  479.   
  480.     if ((cmd->have_input == NULL) || (cmd->width <=0) || (cmd->height <=0)  
  481.             || (cmd->coding <= OMX_ON2_VIDEO_CodingAutoDetect)) {  
  482.         VPU_DEMO_LOG("Warning: missing needed parameters for vpu api demo\n");  
  483.     }  
  484.   
  485.     if (cmd->have_input) {  
  486.         VPU_DEMO_LOG("input bitstream w: %d, h: %d, coding: %d(%s), path: %s\n",  
  487.             cmd->width, cmd->height, cmd->coding,  
  488.             cmd->codec_type == CODEC_DECODER ? "decode" : "encode",  
  489.             cmd->input_file);  
  490.   
  491.         pInFile = fopen(cmd->input_file, "rb");  
  492.         if (pInFile == NULL) {  
  493.             VPU_DEMO_LOG("input file not exsist\n");  
  494.             DECODE_ERR_RET(ERROR_INVALID_PARAM);  
  495.         }  
  496.     } else {  
  497.         VPU_DEMO_LOG("please set input bitstream file\n");  
  498.         DECODE_ERR_RET(ERROR_INVALID_PARAM);  
  499.     }  
  500.   
  501.     if (cmd->have_output) {  
  502.         VPU_DEMO_LOG("vpu api demo output file: %s\n",  
  503.             cmd->output_file);  
  504.         pOutFile = fopen(cmd->output_file, "wb");  
  505.         if (pOutFile == NULL) {  
  506.             VPU_DEMO_LOG("can not write output file\n");  
  507.             DECODE_ERR_RET(ERROR_INVALID_PARAM);  
  508.         }  
  509.         if (cmd->record_frames ==0)  
  510.             cmd->record_frames = 5;  
  511.     }  
  512.   
  513.     fseek(pInFile, 0L, SEEK_END);  
  514.     fileSize = ftell(pInFile);  
  515.     fseek(pInFile, 0L, SEEK_SET);  
  516.   
  517.     memset(&demoPkt, 0, sizeof(VideoPacket_t));  
  518.     pkt = &demoPkt;  
  519.     pkt->data = NULL;  
  520.     pkt->pts = VPU_API_NOPTS_VALUE;  
  521.     pkt->dts = VPU_API_NOPTS_VALUE;  
  522.   
  523.     memset(&decOut, 0, sizeof(DecoderOut_t));  
  524.     pOut = &decOut;  
  525.     pOut->data = (RK_U8*)(malloc)(sizeof(VPU_FRAME));  
  526.     if (pOut->data ==NULL) {  
  527.         DECODE_ERR_RET(ERROR_MEMORY);  
  528.     }  
  529.     memset(pOut->data, 0, sizeof(VPU_FRAME));  
  530.   
  531.     ret = vpu_open_context(&ctx);  
  532.     if (ret || (ctx ==NULL)) {  
  533.         DECODE_ERR_RET(ERROR_MEMORY);  
  534.     }  
  535.   
  536.     /* 
  537.      ** read codec extra data from input stream file. 
  538.     */  
  539.     if (readBytesFromFile((RK_U8*)(&extraSize), 4, pInFile)) {  
  540.         DECODE_ERR_RET(ERROR_IO);  
  541.     }  
  542.   
  543.     VPU_DEMO_LOG("codec extra data size: %d", extraSize);  
  544.   
  545.     pExtra = (RK_U8*)(malloc)(extraSize);  
  546.     if (pExtra ==NULL) {  
  547.         DECODE_ERR_RET(ERROR_MEMORY);  
  548.     }  
  549.     memset(pExtra, 0, extraSize);  
  550.   
  551.     if (readBytesFromFile(pExtra, extraSize, pInFile)) {  
  552.         DECODE_ERR_RET(ERROR_IO);  
  553.     }  
  554.   
  555.     /* 
  556.      ** now init vpu api context. codecType, codingType, width ,height 
  557.      ** are all needed before init. 
  558.     */  
  559.     ctx->codecType = cmd->codec_type;  
  560.     ctx->videoCoding = cmd->coding;  
  561.     ctx->width = cmd->width;  
  562.     ctx->height = cmd->height;  
  563.     ctx->no_thread = 1;  
  564.   
  565.     if ((ret = ctx->init(ctx, pExtra, extraSize)) !=0) {  
  566.        VPU_DEMO_LOG("init vpu api context fail, ret: 0x%X", ret);  
  567.        DECODE_ERR_RET(ERROR_INIT_VPU);  
  568.     }  
  569.   
  570.     /* 
  571.      ** vpu api decoder process. 
  572.     */  
  573.     VPU_DEMO_LOG("init vpu api context ok, fileSize: %d", fileSize);  
  574.   
  575.     do {  
  576.         if (ftell(pInFile) >=fileSize) {  
  577.            VPU_DEMO_LOG("read end of file, complete");  
  578.            break;  
  579.         }  
  580.   
  581.         if (pkt && (pkt->size ==0)) {  
  582.             if (readBytesFromFile((RK_U8*)(&pkt_size), 4, pInFile)) {  
  583.                 break;  
  584.             }  
  585.   
  586.             if (pkt->data ==NULL) {  
  587.                 pkt->data = (RK_U8*)(malloc)(pkt_size);  
  588.                 if (pkt->data ==NULL) {  
  589.                     DECODE_ERR_RET(ERROR_MEMORY);  
  590.                 }  
  591.                 pkt->capability = pkt_size;  
  592.             }  
  593.   
  594.             if (pkt->capability <((RK_U32)pkt_size)) {  
  595.                 pkt->data = (RK_U8*)(realloc)((void*)(pkt->data), pkt_size);  
  596.                 if (pkt->data ==NULL) {  
  597.                     DECODE_ERR_RET(ERROR_MEMORY);  
  598.                 }  
  599.                 pkt->capability = pkt_size;  
  600.             }  
  601.   
  602.             if (readBytesFromFile(pkt->data, pkt_size, pInFile)) {  
  603.                 break;  
  604.             } else {  
  605.                 pkt->size = pkt_size;  
  606.                 pkt->pts = fakeTimeUs;  
  607.                 fakeTimeUs +=40000;  
  608.             }  
  609.   
  610.             VPU_DEMO_LOG("read one packet, size: %d, pts: %lld, filePos: %ld",  
  611.                 pkt->size, pkt->pts, ftell(pInFile));  
  612.         }  
  613.   
  614.         /* note: must set out put size to 0 before do decoder. */  
  615.         pOut->size = 0;  
  616.   
  617.         if ((ret = ctx->decode(ctx, pkt, pOut)) !=0) {  
  618.            DECODE_ERR_RET(ERROR_VPU_DECODE);  
  619.         } else {  
  620.             VPU_DEMO_LOG("vpu decode one frame, out len: %d, left size: %d",  
  621.                 pOut->size, pkt->size);  
  622.   
  623.             /* 
  624.              ** both virtual and physical address of the decoded frame are contained 
  625.              ** in structure named VPU_FRAME, if you want to use virtual address, make 
  626.              ** sure you have done VPUMemLink before. 
  627.             */  
  628.             if ((pOut->size) && (pOut->data)) {  
  629.                 VPU_FRAME *frame = (VPU_FRAME *)(pOut->data);  
  630.                 VPUMemLink(&frame->vpumem);  
  631.                 RK_U32 wAlign16 = ((frame->DisplayWidth+ 15) & (~15));  
  632.                 RK_U32 hAlign16 = ((frame->DisplayHeight + 15) & (~15));  
  633.                 RK_U32 frameSize = wAlign16*hAlign16*3/2;  
  634.   
  635.                 if(pOutFile && (frame_count++ record_frames)) {  
  636.                     VPU_DEMO_LOG("write %d frame(yuv420sp) data, %d bytes to file",  
  637.                         frame_count, frameSize);  
  638.   
  639.                     fwrite((RK_U8*)(frame->vpumem.vir_addr), 1, frameSize, pOutFile);  
  640.                     fflush(pOutFile);  
  641.                 }  
  642.   
  643.                 /* 
  644.                  ** remember use VPUFreeLinear to free, other wise memory leak will 
  645.                  ** give you a surprise. 
  646.                 */  
  647.                 VPUFreeLinear(&frame->vpumem);  
  648.                 pOut->size = 0;  
  649.             }  
  650.         }  
  651.   
  652.         usleep(30);  
  653.     }while(!(ctx->decoder_err));  
  654.   
  655. DECODE_OUT:  
  656.     if (pkt && pkt->data) {  
  657.         free(pkt->data);  
  658.         pkt->data = NULL;  
  659.     }  
  660.     if (pOut && (pOut->data)) {  
  661.         free(pOut->data);  
  662.         pOut->data = NULL;  
  663.     }  
  664.     if (pExtra) {  
  665.         free(pExtra);  
  666.         pExtra = NULL;  
  667.     }  
  668.     if (ctx) {  
  669.         vpu_close_context(&ctx);  
  670.         ctx = NULL;  
  671.     }  
  672.     if (pInFile) {  
  673.         fclose(pInFile);  
  674.         pInFile = NULL;  
  675.     }  
  676.     if (pOutFile) {  
  677.         fclose(pOutFile);  
  678.         pOutFile = NULL;  
  679.     }  
  680.   
  681.     if (ret) {  
  682.         VPU_DEMO_LOG("decode demo fail, err: %d", ret);  
  683.     } else {  
  684.         VPU_DEMO_LOG("encode demo complete OK.");  
  685.     }  
  686.     return ret;  
  687. }  
  688.   
  689.   
  690. int main(int argc, char **argv)  
  691. {  
  692.     VPU_DEMO_LOG("/*******  vpu api demo in *******/");  
  693.   
  694.     VpuApiDemoCmdContext_t demoCmdCtx;  
  695.     RK_S32 ret =0;  
  696.     VPU_API_DEMO_DEBUG_DISABLE = 0;  
  697.   
  698.     if (argc == 1) {  
  699.         show_usage();  
  700.         VPU_DEMO_LOG("vpu api demo complete directly\n");  
  701.         return 0;  
  702.     }  
  703.   
  704.     VpuApiDemoCmdContext_t* cmd = &demoCmdCtx;  
  705.     memset (cmd, 0, sizeof(VpuApiDemoCmdContext_t));  
  706.     cmd->codec_type = CODEC_DECODER;  
  707.     if ((ret = parse_options(argc, argv, cmd)) !=0) {  
  708.         if (ret == VPU_DEMO_PARSE_HELP_OK) {  
  709.             return 0;  
  710.         }  
  711.   
  712.         VPU_DEMO_LOG("parse_options fail\n\n");  
  713.         show_usage();  
  714.         DEMO_ERR_RET(ERROR_INVALID_PARAM);  
  715.     }  
  716.   
  717.     if (cmd->disable_debug) {  
  718.         VPU_API_DEMO_DEBUG_DISABLE = 1;  
  719.     }  
  720.   
  721.     switch (cmd->codec_type) {  
  722.     case CODEC_DECODER:  
  723.         ret = vpu_decode_demo(cmd);  
  724.         break;  
  725.     case CODEC_ENCODER:  
  726.         ret = vpu_encode_demo(cmd);  
  727.         break;  
  728.   
  729.     default:  
  730.         ret = ERROR_INVALID_PARAM;  
  731.         break;  
  732.     }  
  733.   
  734. DEMO_OUT:  
  735.     if (ret) {  
  736.         VPU_DEMO_LOG("vpu api demo fail, err: %d", ret);  
  737.     } else {  
  738.         VPU_DEMO_LOG("vpu api demo complete OK.");  
  739.     }  
  740.     return ret;  
  741. }  

readme.txt文件内容:

[plain]  view plain  copy
  1. /* vpu api demo code on Rockchip Android platform */  
  2.   
  3. 1: please put demo source code to your project, here you can put it on  
  4. the root directory of the SDK project, also other directory will be OK.    
  5.   
  6. 2: build api demo source code just use: mm -B  
  7.   
  8. 3: after step 2, you will generate a test bin file named "vpu_apiDemo" in the out directory of Android SDK platfrom. you can just use adb to push it to /system/bin   
  9. of your device. Note if you can not execute it, chmod 777 to it.  
  10.   
  11. 4: relative head files are given in hardware/rk29/libon2/, which has been described   
  12. in Android.mk of demo source code.   
  13.   
  14. 5: do your test,  help info about how to use demo is included in the demo source code, also you can execute "vpu_apiDemo --help" in adb shell mode to get help info.  
  15.   
  16. 6: run your test in adb shell mode.   
  17.   
  18. (1) decode test example:  
  19. here is the example for test WMV3 decode.  
  20. The following input wmv3 bitstream is also given with this demo code, -coding 5 means the coding_type is OMX_ON2_VIDEO_CodingWMV, defined in vpu_api.h, you can find it under hardware/rk29/libon2/  
  21.   
  22. run wmv3 decode test command in adb shell mode:   
  23. vpu_apiDemo -i /mnt/sdcard/test_bitstream_320x240,259 Kbps,29.970fps,WMV3,32.0 Kbps-Journey.bin -coding 5 -w 320 -h 240  
  24.   
  25. (2) encode test example:  
  26. here is the example for test AVC encode.  
  27. The encode test input yuv bitstream(YUV420_SEMI_PLANAR) is also given with this demo code, -coding 7 means the coding_type is OMX_ON2_VIDEO_CodingAVC, -t 2 means codec_type is encoder, all defined in vpu_api.h, you can find it under hardware/rk29/libon2/  
  28.   
  29. run avc encode test command in adb shell mode:  
  30. vpu_apiDemo -i /mnt/sdcard/vpu_enc_input_yuv420sp_320x240.yuv -o /mnt/sdcard/enc_out.264 -coding 7 -t 2 -w 320 -h 240  
  31.   
  32. Note:   
  33. Current demo source just use one test bin file(with WMV3 coding) for decoding test,  
  34. because we have not parser or demuxer in this demo source code, so we just write video bitstream in the test bin file. note we have added some extra info such as every frame size. so if you want to use this demo with your own demuxer,  you need to add a little modify to the demo source code.   
剩下另外两个文件是bin文件和视频编码需要的。

你可能感兴趣的:(Android,拍照源码分析)