2017-07-04 76 views
0

我有一個在手機模式下從手機拍攝的視頻。這裏是關於視頻的傾銷信息:如何禁用libav自動旋轉顯示矩陣

Input #0, mov,mp4,m4a,3gp,3g2,mj2, from 'video.MOV': 
Metadata: 
major_brand  : qt 
minor_version : 0 
compatible_brands: qt 
creation_time : 2017-05-04 02:21:37 
Duration: 00:00:06.91, start: 0.000023, bitrate: 4700 kb/s 
Stream #0:0(und): Audio: aac (LC) (mp4a/0x6134706D), 44100 Hz, mono, fltp, 90 kb/s (default) 
Metadata: 
    creation_time : 2017-05-04 02:21:37 
    handler_name : Core Media Data Handler 
Stream #0:1(und): Video: h264 (High) (avc1/0x31637661), yuv420p(tv, bt709), 1280x720, 4602 kb/s, 29.98 fps, 29.97 tbr, 600 tbn, 1200 tbc (default) 
Metadata: 
    rotate   : 90 
    creation_time : 2017-05-04 02:21:37 
    handler_name : Core Media Data Handler 
    encoder   : H.264 
Side data: 
    displaymatrix: rotation of -90.00 degrees 
    5.78 A-V: -0.028 fd= 1 aq= 14KB vq= 351KB sq= 0B f=0/0 

我正在使用libav api解碼/編碼我的視頻。編碼後,我得到旋轉版本的視頻-90度。

如何停止解碼器以防止自動旋轉?

+0

查看與cli工具的'-noautorotate'選項相關的源代碼。 – LordNeckbeard

+0

@LordNeckbeard,我這樣做了,但是我仍然遇到了禁用libav api中autorotae的問題。你能提供關於它的細節嗎? –

回答

0

經過很長的時間來處理所造成的顯示矩陣端數據的問題,我遇到了以下解決方案:

複製元數據和端數據與視頻流一起。 此代碼的伎倆

AVStream *in_stream; //input stream 
AVStream *out_stream; //output stream 
....   
if(in_stream->side_data!=NULL){ 

    av_log(NULL, AV_LOG_ERROR, "side data size: %d , size: %d\n", in_stream->side_data->size, sizeof(uint8_t*)); 

     if(av_stream_get_side_data(in_stream , AV_PKT_DATA_DISPLAYMATRIX, sd_size)!=NULL){ 

      uint8_t* resp=(uint8_t*) av_mallocz(in_stream->side_data->size+ 
               AV_INPUT_BUFFER_MIN_SIZE); 


      resp = av_stream_get_side_data(in_stream , AV_PKT_DATA_DISPLAYMATRIX, sd_size); 

       av_log(NULL, AV_LOG_DEBUG,"side data detected, size :%d vs %d, " 
             "nb_side_data %d, sizeof data %d," 
             " sizeof resp %d\n", 
        *sd_size, in_stream->side_data->size ,in_stream->nb_side_data, 
        in_stream->side_data->size*sizeof(uint8_t*),sizeof(resp)); 
       in_stream->side_data->data = (uint8_t*) av_mallocz(
          in_stream->side_data->size*sizeof(uint8_t*) 
         ); 
       hasRotation = true; /* it will be used for 
       something like ffmpeg does with noautorotate flag */ 
       av_stream_add_side_data(out_stream, AV_PKT_DATA_DISPLAYMATRIX, resp, *sd_size); 

     } 
    } 

生成的視頻將有一個顯示矩陣,一些球員會告訴他們的權利。

使用簡單的過濾器 如上所示,可以檢測到視頻旋轉。在我的情況下,旋轉是-90,所以我使用轉置=時鐘過濾器來啓動我的過濾器圖。考慮ffmpeg 3.4.1中的轉碼示例。我改變代碼如下,禁用-90度的自動旋轉。

static int init_filters(void) 
{ 
    const char *filter_spec; 
    unsigned int i; 
    int ret; 
    filter_ctx = (FilteringContext*) av_malloc_array(ifmt_ctx->nb_streams, sizeof(*filter_ctx)); 
if (!filter_ctx) 
    return AVERROR(ENOMEM); 
    for (i = 0; i < ifmt_ctx->nb_streams; i++) { 
     filter_ctx[i].buffersrc_ctx = NULL; 
     filter_ctx[i].buffersink_ctx = NULL; 
     filter_ctx[i].filter_graph = NULL; 
     if (!(ifmt_ctx->streams[i]->codecpar->codec_type == AVMEDIA_TYPE_AUDIO 
      || ifmt_ctx->streams[i]->codecpar->codec_type == AVMEDIA_TYPE_VIDEO)) 
     continue; 
     if (ifmt_ctx->streams[i]->codecpar->codec_type == AVMEDIA_TYPE_VIDEO) 
     filter_spec = isRotated ?"transpose=clock":"null"; /* passthrough (dummy) filter for video */ 
    else 
     filter_spec = "anull"; /* passthrough (dummy) filter for audio */ 
    ret = init_filter(&filter_ctx[i], stream_ctx[i].dec_ctx, 
      stream_ctx[i].enc_ctx, filter_spec); 
    if (ret) 
     return ret; 
} 
return 0; 
} 

注意,編碼器的尺寸也應該被交換。這意味着:

enc_ctx->width = dec_ctx->height; 
enc_ctx->height = enc_ctx->width; 

我希望它能幫助別人。