編譯環境
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命令行