iOS ijkPlayer 改造——增加直播錄像功能

參考:http://www.reibang.com/p/a346f93ddaff
本人懶 從OC方法倒著復制的 建議倒著看

IJKMediaPlayback.h

- (void)stopRecord;
- (void)startRecordWithFileName:(NSString *)fileName;

IJKFFMoviePlayerController.m

- (void)stopRecord {
    ijkmp_stop_recording(_mediaPlayer);
}

- (void)startRecordWithFileName:(NSString *)fileName {
    // 視頻存儲的路徑
    const char *path = [fileName cStringUsingEncoding:NSUTF8StringEncoding];
    ijkmp_start_recording (_mediaPlayer, path);
}

- (BOOL)isRecording {
    return ijkmp_isRecording(_mediaPlayer);
}

ff_ffplay_def.h

/* ffplayer */
struct IjkMediaMeta;
struct IJKFF_Pipeline;
typedef struct FFPlayer {
    ...
    
    AVFormatContext *m_ofmt_ctx;        // 用于輸出的AVFormatContext結構體
    AVOutputFormat *m_ofmt;
    pthread_mutex_t record_mutex;       // 鎖
    int is_record;                      // 是否在錄制
    int record_error;
    
    int is_first;                       // 第一幀數(shù)據(jù)
    int64_t start_v_pts;                // 開始錄制時pts 視頻
    int64_t start_v_dts;                // 開始錄制時dts 視頻
    int64_t start_a_pts;                // 開始錄制時pts 音頻
    int64_t start_a_dts;                // 開始錄制時dts 音頻
    
} FFPlayer;

ijkplayer.c

static int ijkmp_start_recording_l(IjkMediaPlayer *mp,const char *filePath)
{
//  av_log(mp->ffplayer,AV_LOG_WARING,"cjz ijkmp_start_recording_l filePath %s",filePath);
  return ffp_start_recording_l(mp->ffplayer,filePath);
}

int ijkmp_start_recording(IjkMediaPlayer *mp,const char *filePath)
{
  assert(mp);
  pthread_mutex_lock(&mp->mutex);
  av_log(mp->ffplayer,AV_LOG_WARNING,"cjz ijkmp_start_recording");
  int retval = ijkmp_start_recording_l(mp,filePath);
  pthread_mutex_unlock(&mp->mutex);
  return retval;
}

static int ijkmp_stop_recording_l(IjkMediaPlayer *mp)
{
  return ffp_stop_recording_l(mp->ffplayer);
} 

int ijkmp_stop_recording(IjkMediaPlayer *mp)
{
  assert(mp);
  pthread_mutex_lock(&mp->mutex);
  av_log(mp->ffplayer,AV_LOG_WARNING,"cjz ijkmp_stop_recording");
  int retval = ijkmp_stop_recording_l(mp);
  pthread_mutex_unlock(&mp->mutex);
  return retval;
}

static int ijkmp_isRecordFinished_l(IjkMediaPlayer *mp)
{
 return ffp_record_isfinished_l(mp->ffplayer);
}

int ijkmp_isRecordFinished(IjkMediaPlayer *mp)
{
  assert(mp);
 pthread_mutex_lock(&mp->mutex);
 av_log(mp->ffplayer,AV_LOG_WARNING,"cjz ijkmp_isRecordFinished ");
 int retval = ijkmp_isRecordFinished_l(mp);
 pthread_mutex_unlock(&mp->mutex);
 return retval;
}

int ijkmp_isRecording(IjkMediaPlayer *mp) {
    return mp->ffplayer->is_record;
}

ff_ffplay.h

int ffp_start_recording_l(FFPlayer *ffp,const char *file_name);
int ffp_record_isfinished_l(FFPlayer *ffp);
int ffp_stop_recording_l(FFPlayer *ffp);

ff_ffplay.c

//插入錄制方法
int ffp_start_recording_l(FFPlayer *ffp,const char *file_name)
{
    assert(ffp);
    VideoState *is = ffp->is;
    
    ffp->m_ofmt_ctx = NULL;
    ffp->m_ofmt = NULL;
    ffp->is_record = 0;
    ffp->record_error = 0;
    
    if (!file_name || !strlen(file_name)) { // 沒有路徑
        av_log(ffp, AV_LOG_ERROR, "filename is invalid");
        goto end;
    }
    
    if (!is || !is->ic|| is->paused || is->abort_request) { // 沒有上下文,或者上下文已經停止
        av_log(ffp, AV_LOG_ERROR, "is,is->ic,is->paused is invalid");
        goto end;
    }
    
    if (ffp->is_record) { // 已經在錄制
        av_log(ffp, AV_LOG_ERROR, "recording has started");
        goto end;
    }
    
    // 初始化一個用于輸出的AVFormatContext結構體
    avformat_alloc_output_context2(&ffp->m_ofmt_ctx, NULL, NULL, file_name);
    if (!ffp->m_ofmt_ctx) {
        av_log(ffp, AV_LOG_ERROR, "Could not create output context filename is %s\n", file_name);
        goto end;
    }
    ffp->m_ofmt = ffp->m_ofmt_ctx->oformat;
    
    for (int i = 0; i < is->ic->nb_streams; i++) {
        // 對照輸入流創(chuàng)建輸出流通道
        AVStream *in_stream = is->ic->streams[i];
        AVStream *out_stream = avformat_new_stream(ffp->m_ofmt_ctx, in_stream->codec->codec);
        if (!out_stream) {
            av_log(ffp, AV_LOG_ERROR, "Failed allocating output stream\n");
            goto end;
        }
        
        // 將輸入視頻/音頻的參數(shù)拷貝至輸出視頻/音頻的AVCodecContext結構體
        av_log(ffp, AV_LOG_DEBUG, "in_stream->codec壹将;%p\n", in_stream->codec);
        if (avcodec_copy_context(out_stream->codec, in_stream->codec) < 0) {
            av_log(ffp, AV_LOG_ERROR, "Failed to copy context from input to output stream codec context\n");
            goto end;
        }
        
        out_stream->codec->codec_tag = 0;
        if (ffp->m_ofmt_ctx->oformat->flags & AVFMT_GLOBALHEADER) {
            out_stream->codec->flags |= CODEC_FLAG_GLOBAL_HEADER;
        }
    }
    
    av_dump_format(ffp->m_ofmt_ctx, 0, file_name, 1);
    
    // 打開輸出文件
    if (!(ffp->m_ofmt->flags & AVFMT_NOFILE)) {
        if (avio_open(&ffp->m_ofmt_ctx->pb, file_name, AVIO_FLAG_WRITE) < 0) {
            av_log(ffp, AV_LOG_ERROR, "Could not open output file '%s'", file_name);
            goto end;
        }
    }
    
    // 寫視頻文件頭
    if (avformat_write_header(ffp->m_ofmt_ctx, NULL) < 0) {
        av_log(ffp, AV_LOG_ERROR, "Error occurred when opening output file\n");
        goto end;
    }
    
    ffp->is_record = 1;
    ffp->record_error = 0;
    pthread_mutex_init(&ffp->record_mutex, NULL);
    
    return 0;
end:
    ffp->record_error = 1;
    return -1;
}

int ffp_record_isfinished_l(FFPlayer *ffp)
{
    return 0;
}

int ffp_record_file(FFPlayer *ffp, AVPacket *packet)
{
    assert(ffp);
    VideoState *is = ffp->is;
    int ret = 0;
    AVStream *in_stream;
    AVStream *out_stream;
    
    if (ffp->is_record) {
        if (packet == NULL) {
            ffp->record_error = 1;
            av_log(ffp, AV_LOG_ERROR, "packet == NULL");
            return -1;
        }
        
        AVPacket *pkt = (AVPacket *)av_malloc(sizeof(AVPacket)); // 與看直播的 AVPacket分開诽俯,不然卡屏
        av_new_packet(pkt, 0);
        if (0 == av_packet_ref(pkt, packet)) {
            pthread_mutex_lock(&ffp->record_mutex);
            if (!ffp->is_first) { // 錄制的第一幀承粤,時間從0開始
                ffp->is_first = 1;
                pkt->pts = 0;
                pkt->dts = 0;
            } else { // 之后的每一幀都要減去闯团,點擊開始錄制時的值仙粱,這樣的時間才是正確的
                if (pkt->stream_index == AVMEDIA_TYPE_AUDIO) {
                    pkt->pts = llabs(pkt->pts - ffp->start_a_pts);
                    pkt->dts = llabs(pkt->dts - ffp->start_a_dts);
                }
                else if (pkt->stream_index == AVMEDIA_TYPE_VIDEO) {
                    pkt->pts = pkt->dts = llabs(pkt->dts - ffp->start_v_dts);
                }
            }
            
            in_stream  = is->ic->streams[pkt->stream_index];
            out_stream = ffp->m_ofmt_ctx->streams[pkt->stream_index];
            
            // 轉換PTS/DTS
            pkt->pts = av_rescale_q_rnd(pkt->pts, in_stream->time_base, out_stream->time_base, (AV_ROUND_NEAR_INF|AV_ROUND_PASS_MINMAX));
            pkt->dts = av_rescale_q_rnd(pkt->dts, in_stream->time_base, out_stream->time_base, (AV_ROUND_NEAR_INF|AV_ROUND_PASS_MINMAX));
            pkt->duration = av_rescale_q(pkt->duration, in_stream->time_base, out_stream->time_base);
            pkt->pos = -1;
            
            // 寫入一個AVPacket到輸出文件
            if ((ret = av_interleaved_write_frame(ffp->m_ofmt_ctx, pkt)) < 0) {
                av_log(ffp, AV_LOG_ERROR, "Error muxing packet\n");
            }
            
            av_packet_unref(pkt);
            pthread_mutex_unlock(&ffp->record_mutex);
        } else {
            av_log(ffp, AV_LOG_ERROR, "av_packet_ref == NULL");
        }
    }
    return ret;
}


int ffp_stop_recording_l(FFPlayer *ffp){
    assert(ffp);
    if (ffp->is_record) {
        ffp->is_record = 0;
        pthread_mutex_lock(&ffp->record_mutex);
        if (ffp->m_ofmt_ctx != NULL) {
            av_write_trailer(ffp->m_ofmt_ctx);
            if (ffp->m_ofmt_ctx && !(ffp->m_ofmt->flags & AVFMT_NOFILE)) {
                avio_close(ffp->m_ofmt_ctx->pb);
            }
            avformat_free_context(ffp->m_ofmt_ctx);
            ffp->m_ofmt_ctx = NULL;
            ffp->is_first = 0;
        }
        pthread_mutex_unlock(&ffp->record_mutex);
        pthread_mutex_destroy(&ffp->record_mutex);
        av_log(ffp, AV_LOG_DEBUG, "stopRecord ok\n");
    } else {
        av_log(ffp, AV_LOG_ERROR, "don't need stopRecord\n");
    }
    return 0;
}

// ff_ffplay.c 自帶方法
static int read_thread(void *arg) {
    ...
    for (;;) {
        //插入
        if (!ffp->is_first && pkt->pts == pkt->dts) { // 獲取開始錄制前dts等于pts最后的值伐割,用于
            if (pkt->stream_index == AVMEDIA_TYPE_AUDIO) {
                ffp->start_a_pts = pkt->pts;
                ffp->start_a_dts = pkt->dts;
            }
        }
        if (pkt->stream_index == AVMEDIA_TYPE_VIDEO) {
            if (!ffp->is_first) {
                ffp->start_v_pts = pkt->pts;
                ffp->start_v_dts = pkt->dts;
                //                        printf("set start video start_pts: %lld +++++ start_dts: %lld\n", ffp->start_v_pts, ffp->start_v_dts);
            }
        }
        if (pkt->stream_index == AVMEDIA_TYPE_VIDEO) {
            //                    printf("video pts: %lld +++++ dts: %lld\n", pkt->pts, pkt->dts);
        }
#pragma  mark - 錄制插入
        if (ffp->is_record) { // 可以錄制時,寫入文件
            if (0 != ffp_record_file(ffp, pkt)) {
                ffp->record_error = 1;
                ffp_stop_recording_l(ffp);
            }
        }
        ...
      }
    ...
}
最后編輯于
?著作權歸作者所有,轉載或內容合作請聯(lián)系作者
  • 序言:七十年代末负溪,一起剝皮案震驚了整個濱河市济炎,隨后出現(xiàn)的幾起案子川抡,更是在濱河造成了極大的恐慌须尚,老刑警劉巖,帶你破解...
    沈念sama閱讀 217,657評論 6 505
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件密幔,死亡現(xiàn)場離奇詭異撩轰,居然都是意外死亡,警方通過查閱死者的電腦和手機堪嫂,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,889評論 3 394
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來淹办,“玉大人恶复,你說我怎么就攤上這事怜森“担” “怎么了?”我有些...
    開封第一講書人閱讀 164,057評論 0 354
  • 文/不壞的土叔 我叫張陵想许,是天一觀的道長。 經常有香客問我流纹,道長糜烹,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,509評論 1 293
  • 正文 為了忘掉前任漱凝,我火速辦了婚禮疮蹦,結果婚禮上,老公的妹妹穿的比我還像新娘茸炒。我一直安慰自己愕乎,他們只是感情好,可當我...
    茶點故事閱讀 67,562評論 6 392
  • 文/花漫 我一把揭開白布壁公。 她就那樣靜靜地躺著感论,像睡著了一般。 火紅的嫁衣襯著肌膚如雪紊册。 梳的紋絲不亂的頭發(fā)上比肄,一...
    開封第一講書人閱讀 51,443評論 1 302
  • 那天囊陡,我揣著相機與錄音芳绩,去河邊找鬼。 笑死撞反,一個胖子當著我的面吹牛妥色,可吹牛的內容都是我干的。 我是一名探鬼主播遏片,決...
    沈念sama閱讀 40,251評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼嘹害,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了吮便?” 一聲冷哼從身側響起吼拥,我...
    開封第一講書人閱讀 39,129評論 0 276
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎线衫,沒想到半個月后,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體惑折,經...
    沈念sama閱讀 45,561評論 1 314
  • 正文 獨居荒郊野嶺守林人離奇死亡授账,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 37,779評論 3 335
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了惨驶。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片白热。...
    茶點故事閱讀 39,902評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖粗卜,靈堂內的尸體忽然破棺而出屋确,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 35,621評論 5 345
  • 正文 年R本政府宣布攻臀,位于F島的核電站焕数,受9級特大地震影響,放射性物質發(fā)生泄漏刨啸。R本人自食惡果不足惜堡赔,卻給世界環(huán)境...
    茶點故事閱讀 41,220評論 3 328
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望设联。 院中可真熱鬧善已,春花似錦、人聲如沸离例。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,838評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽宫蛆。三九已至艘包,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間洒扎,已是汗流浹背辑甜。 一陣腳步聲響...
    開封第一講書人閱讀 32,971評論 1 269
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留袍冷,地道東北人磷醋。 一個月前我還...
    沈念sama閱讀 48,025評論 2 370
  • 正文 我出身青樓,卻偏偏與公主長得像胡诗,于是被迫代替她去往敵國和親邓线。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 44,843評論 2 354

推薦閱讀更多精彩內容