IJKPlayer編譯FFmpeg命令行

編譯環境

Win10 + Oracle VM + Ubuntu 19

NDK version: r10e

編譯CMD進IJK中,使用正常的IJK編譯方法即可。

編譯HTTPS進入IJK中

  • 拉取IJK後,進入文件夾目錄進行命令編譯
  • ./init-android-openssl.sh
  • ./init-android.sh

完成後進入下一步

編譯SSL和FFMPEG

在config文件夾中選擇自己的編譯選項,一般來說用默認就ok(我是默認)
進入android/contrib執行如下命令

  • ./compile-openssl.sh clean
  • ./compile-openssl.sh all
  • ./compile-ffmpeg.sh clean
  • ./compile-ffmpeg.sh all

結束後開始編譯IJK,回到android目錄(cd …

  • ./compile-ijk.sh all

當編譯結束後,可以嘗試運行一下ijk的example代碼了,嘗試看看https是否能成功運行。

編譯FFmpeg命令行

上述操作執行完成後,能夠在對應的CPU框架下找到so包,那麼這就是之前ffmpeg編譯好的。直接拿來使用就好了。

編寫C,MK,Java

cmd.c
//cpp的寫法不同,請注意

#include <jni.h>
#include "fftools/ffmpeg.h"

JNIEXPORT jint JNICALL
Java_tv_danmaku_ijk_media_player_1armv7a_Pragma_FFmpegExec(JNIEnv *env,jclass type,jobjectArray cmd){
    int len = (*env)->GetArrayLength(env,cmd);
    char *argv[len];
    int i;
    for(i = 0;i < len;++i){
        argv[i] = (char *) (*env)->GetStringUTFChars(env,(jstring) (*env)->GetObjectArrayElement(env,cmd,i),0);
    }
    return ffmpeg_exec(len,argv);
}

下面這些各個屬性啥意思請自行理解,也不是很難(雖然現在是CMakeList.txt方式去編譯)

Android.mk
LOCAL_PATH := $(call my-dir)
FFMPEG_FILES := $(realpath $(MY_APP_ANDROID_ROOT)/contrib/ffmpeg-armv7a/fftools)
INCLUDE_FILES := $(realpath $(FFMPEG_FILES)/..)

include $(CLEAR_VARS)
LOCAL_MODULE := ffmpegcmd
LOCAL_SRC_FILES := cmd.c $(FFMPEG_FILES)/cmdutils.c $(FFMPEG_FILES)/ffmpeg.c $(FFMPEG_FILES)/ffmpeg_filter.c $(FFMPEG_FILES)/ffmpeg_hw.c $(FFMPEG_FILES)/ffmpeg_opt.c
LOCAL_C_INCLUDES := $(INCLUDE_FILES)
LOCAL_SHARED_LIBRARIES := ijkffmpeg
include $(BUILD_SHARED_LIBRARY)
# 我這裏沒有導入android的log,請自行查閱

能看到上面導入了ffmpeg.c之內的文件,這些能夠在\android\contrib\中ffmpeg的各個CPU架構中找到

package tv.danmaku.ijk.media.player_armv7a;
//native方法使用
//Pragma
public class Pragma {
    public static native int FFmpegExec(String[] cmd);
}

修改ffmpeg文件

找到ffmpeg.c文件,將最底下的main方法修改爲ffmpeg_exec,並在ffmpeg.h中增加這一方法
int ffmpeg_exec(int argc, char **argv);

並在C的最後的末尾添加如下內容:

	.........
	if ((decode_error_stat[0] + decode_error_stat[1]) * max_error_rate < decode_error_stat[1])	
		exit_program(69);
	//重置
    nb_filtergraphs = 0;
    nb_output_files = 0;
    nb_output_streams = 0;
    nb_input_files = 0;
    nb_input_streams = 0;
//    exit_program(received_nb_signals ? 255 : main_return_code);防止系統退出報錯
	......

接着找到cmdutil.c,找到exit_program方法,將內容置空即可

void exit_program(int ret)
{
//    if (program_exit)
//        program_exit(ret);

//    exit(ret);
}

上述源文件是用的鏈接的方式使用的,也可以將這些文件放入工程中使用,譬如這樣:
在這裏插入圖片描述
如上編寫完了後,回到/android目錄下,使用腳本:

  • ./compile-ijk.sh armv7a(我這裏只編譯armv7,其他的太麻煩就沒有編了)

如果想在其他平臺上也支持也可以自己嘗試編譯其他版本so包

使用

可直接調用方法傳入參數:

Pragma.FFmpegExec(new String[]{
                    "ffmpeg","-i", Environment.getExternalStorageDirectory()+"/ysgs.mp4",
                    "-ss","10","-t","10",
                    "-c", "copy",
                    Environment.getExternalStorageDirectory()+"/tem.mp4"
            });

ffmpeg的命令請自行查閱。

END

如此整個編譯就結束了,可以使用IJK的同時對視頻進行編輯(裁剪音視頻軌道替換等。。。。)

PS:如果Ubuntu報告找不到NDK的位置,請自行配置(或者在相應腳本中增加變量ANDROID_NDK)

直接取用,fork from ijk,基於IJK編譯:
IJK_CMD

不依賴IJK的話,如下方式:
NDK R2編譯ffmpeg命令行

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章