ffmpeg开发之旅(5):详解ffmpeg编译与在Android平台上的移植

来源:互联网 发布:linux fifo 编辑:程序博客网 时间:2024/05/23 01:13

ffmpeg开发之旅(5):详解ffmpeg编译与在Android平台上的移植

(码字不易,转载请声明出处:http://blog.csdn.net/andrexpert/article/details/73823740)

一、ffmpeg在linux环境下的编译

1. 编译环境

(1) VirtualBox:VirtualBox_5.1.22.15126.exe
(2) Ubuntu:ubuntu-14.04.5-desktop-amd64.iso
(3) NDK:android-ndk-r14b-linux-x86_64.zip
(4) ffmpeg:ffmpeg-3.3.2.tar.bz2
      为了提高ffmpeg编译速度,这里选择在Linux环境下对其进行编译。VirtualBox安装Ubuntu比较简单,可自行查找相关资料,只是在为虚拟系统分配磁盘空间时建议大于20GB,因为NDK体积还是比较大的,默认的8GB根本不够用。其次,NDK的版本一定要与Ubuntu版本一致,我这里选择的是64位的,为什么这里要强调下,因为就是这个版本不一致问题,让我在configure ffmpeg时整整花了两天时间去找bug,只怪太相信自己的记忆力了。最后,解压NDK和ffmpeg到同一目录下即可,我的解压路径是/home/jiangdongguo/ffmpeg。
2. 配置ffmpeg
(1) 设置NDK路径
      为了方便配置configure命令的相关参数,这里我们使用export命令将NDK存储路径设置为全局的,另外,我们还需要设置一个临时目录,以便存储ffmpeg编译时产生的临时数据,当然,这里需要保证该目录已经存在且可读写。注:也可以将这两行命令放到脚本文件中。
jiangdongguo@jiangdg:~$ export NDK=/home/jiangdongguo/ffmpeg/android-ndk-r14bjiangdongguo@jiangdg:~$ export TMPDIR=/home/jiangdongguo/ffmpeg/ffmpegTmpDir
(2) 配置ffmpeg
a) 创建执行configure命令脚本文件
root@jiangdg:/home/jiangdongguo/ffmpeg/ffmpeg-3.3.2# vim configure_arm.sh
     在创建配置ffmpeg的脚本文件时,有三个地方需要根据自身情况更改:SYSTEMROOT,用于指定ndk platform的路径,一定要选择比你的目标机器使用的版本低,比如你的手机是Android 6.0,那么需要选择android-23以下;TOOLCHAIN_PREFIX,指定编译ffmpeg编译工具链所在路径;PREFIX,用于指定编译完成后so文件输出目录,会自动在改路径目录下生成Android使用所需的include和lib目录。

     其中,--target-os选项指定目标系统类型、--arch选项指定目标系统架构、--enbale-shared、-enable-static指定只生成so共享库,--enable-cross-compile开启使用指定交叉编译工具等等。至于其他配置选项,可自行在ffmpeg源码目录下执行"configure --help"命令查看。
// 修改ffmpeg源码下configure文件root@jiangdg:/home/jiangdongguo/ffmpeg/ffmpeg-3.3.2# vim configure
    对ffmpeg源码下的configure文件进行编辑,找到如下代码进行修改。修改如下几行代码的目的是为了得到Android平台能够识别的so库,比如libavcodec-57.so、libavdevice-57.so等,而不是libavcodec.so.57、libavcodec.so.57.89.100等。具体修改情况如下:
SLIBNAME_WITH_VERSION='$(SLIBNAME).$(LIBVERSION)'SLIBNAME_WITH_MAJOR='$(SLIBNAME).$(LIBMAJOR)'LIB_INSTALL_EXTRA_CMD='$$(RANLIB) "$(LIBDIR)/$(LIBNAME)"'SLIB_INSTALL_NAME='$(SLIBNAME_WITH_VERSION)'SLIB_INSTALL_LINKS='$(SLIBNAME_WITH_MAJOR) $(SLIBNAME)'将上面的内容修改如下:SLIBNAME_WITH_VERSION='$(SLIBNAME).$(LIBVERSION)'SLIBNAME_WITH_MAJOR='$(SLIBPREF)$(FULLNAME)-$(LIBMAJOR)$(SLIBSUF)'LIB_INSTALL_EXTRA_CMD='$$(RANLIB)"$(LIBDIR)/$(LIBNAME)"'SLIB_INSTALL_NAME='$(SLIBNAME_WITH_MAJOR)'SLIB_INSTALL_LINKS='$(SLIBNAME)'

// 赋予configure_arm.sh执行权限root@jiangdg:/home/jiangdongguo/ffmpeg/ffmpeg-3.3.2# chmod u+x configure_arm.sh// 执行脚本文件root@jiangdg:/home/jiangdongguo/ffmpeg/ffmpeg-3.3.2# ./configure_arm.sh
注:如果提示.../arm-linux-androideabi-pkg-config not found, library detection may fail.警告,忽视即可,编译时目前没有发现有什么影响。
3. 编译ffmpeg
root@jiangdg:/home/jiangdongguo/ffmpeg/ffmpeg-3.3.2#make cleanroot@jiangdg:/home/jiangdongguo/ffmpeg/ffmpeg-3.3.2#make root@jiangdg:/home/jiangdongguo/ffmpeg/ffmpeg-3.3.2#make install
    编译前需要"make clean"清理下,然后"make"大概需要5-10分钟,待编译完毕后执行后再执行"make install",就会在之前创建的android/arm目录下自动生成include和lib目录,其分别存放了Android使用ffmpeg所需的头文件和so共享库。

二、ffmpeg移植与在Android平台上的使用

1. 创建Android NDK工程

讲解一下:
    有关eclipse中开发NDK/JNI,我这篇文章已经讲解得比较清楚了,可自行前往按步骤搭建即可。这里提一下与ffmpeg有关的相关文件,通过上面讲解可以知道,linux环境下编译好ffmpeg后,会自动在../android/arm目录下生成include和lib目录,我们将分别整个include目录、lib目录so库文件(链接文件和pkgconfig除外)拷贝到Android工程的jni目录下,另外,还需要将ffmpeg源码根目录下的ffmpeg.h、config.h和cmdutils.h拷贝到jni目录,否则调用ffmpeg相关函数会报错。
2. VideoFixUtils.class:Java层创建native方法
/** 处理视频native方法工具类 *  * @author Created by jianddongguo on 2017年6月26日下午11:14:27 * @blogs http://blog.csdn.net/andrexpert */public class VideoFixUtils {/** 获得指定视频的角度 * @param videoPath 视频路径 * @return 拍摄角度值 */public native static int getVideoAngle(String videoPath);static {// 加载自定义动态库System.loadLibrary("FFMPEG4Android");// 加载ffmpeg相关动态库System.loadLibrary("avcodec-57");System.loadLibrary("avdevice-57");System.loadLibrary("avfilter-6");System.loadLibrary("avformat-57");System.loadLibrary("avutil-55");System.loadLibrary("swscale-4");System.loadLibrary("swresample-2");}}
讲解一下:
    通过上面代码可知,处理在static静态代码块中加载自定义的FFMPEG4Android动态库,还需加载与ffmpeg相关的所有动态库,至于需要加载哪些,可以到Android工程中的libs/armeabi目录下查看,动态库的名称通过去掉lib前缀可得。
3. FFMPEG4Android.c:C/C++层实现native函数原型
/** 处理视频c实现 *  * @author Created by jianddongguo on 2017年6月26日下午11:14:27 * @blogs http://blog.csdn.net/andrexpert */#include <jni.h>#include <stdlib.h>#include "com_jiangdg_ffmepg4android_VideoFixUtils.h"#include "ffmpeg.h"JNIEXPORT jint JNICALL Java_com_jiangdg_ffmepg4androidk_VideoFixUtils_getVideoAngle  (JNIEnv *env, jclass jcls, jstring j_videoPath){const char *c_videoPath = (*env)->GetStringUTFChars(env,j_videoPath,NULL);//1. 注册所有组件av_register_all();//2. 打开视频、获取视频信息,// 其中,fmtCtx为封装格式上下文AVFormatContext *fmtCtx = avformat_alloc_context();avformat_open_input(&fmtCtx,c_videoPath,NULL,NULL);//3. 获取视频流的索引位置int i;int v_stream_idx = -1;for(i=0 ; i<fmtCtx->nb_streams ; i++){if(fmtCtx->streams[i]->codec->codec_type == AVMEDIA_TYPE_VIDEO){v_stream_idx = i;break;}}// 4. 获取旋转角度,元数据AVDictionaryEntry *tag = NULL;tag = av_dict_get(fmtCtx->streams[v_stream_idx]->metadata,"rotate",tag,NULL);int angle = -1;if(tag != NULL){// 将char *强制转换为into类型angle = atoi(tag->value);}// 5.释放封装格式上下文avformat_free_context(fmtCtx);(*env)->ReleaseStringUTFChars(env,j_videoPath,c_videoPath);return angle;}
讲解一下:
    由于本篇文章重点在于讲解如何在linux系统环境下编译so共享库,并将其移植到Android平台上使用,这里就不详细讲解ffmpeg实现代码,稍微讲下这里面的相关原理:我们知道mp4是一种视频封装格式,它可能包含三个轨,即音频、视频、字幕,每个轨对应一个AVStream,如果要想知道mp4文件的拍摄角度,就需要对mp4格式进行解封装,然后抽离出视频轨,进而得到所需角度值。
4. 配置Android.mk
LOCAL_PATH := $(call my-dir)#ffmpeg prebuilt libinclude $(CLEAR_VARS)LOCAL_MODULE    := avcodec_prebuiltLOCAL_SRC_FILES := libavcodec-57.soinclude $(PREBUILT_SHARED_LIBRARY)include $(CLEAR_VARS)LOCAL_MODULE    := avdevice_prebuiltLOCAL_SRC_FILES := libavdevice-57.soinclude $(PREBUILT_SHARED_LIBRARY)include $(CLEAR_VARS)LOCAL_MODULE    := avfilter_prebuiltLOCAL_SRC_FILES := libavfilter-6.soinclude $(PREBUILT_SHARED_LIBRARY)include $(CLEAR_VARS)LOCAL_MODULE    := avformat_prebuiltLOCAL_SRC_FILES := libavformat-57.soinclude $(PREBUILT_SHARED_LIBRARY)include $(CLEAR_VARS)LOCAL_MODULE    := avutil_prebuiltLOCAL_SRC_FILES := libavutil-55.soinclude $(PREBUILT_SHARED_LIBRARY)include $(CLEAR_VARS)LOCAL_MODULE    := swresample_prebuiltLOCAL_SRC_FILES := libswresample-2.soinclude $(PREBUILT_SHARED_LIBRARY)include $(CLEAR_VARS)LOCAL_MODULE    := swscale_prebuiltLOCAL_SRC_FILES := libswscale-4.soinclude $(PREBUILT_SHARED_LIBRARY)#myapp libinclude $(CLEAR_VARS)LOCAL_MODULE    := FFMPEG4AndroidLOCAL_SRC_FILES := FFMPEG4Android.cLOCAL_C_INCLUDES +=$(LOCAL_PATH)/includeLOCAL_LDLIBS := -llogLOCAL_SHARED_LIBRARIES := avcodec_prebuilt avdevice_prebuilt avfilter_prebuilt avformat_prebuilt avutil_prebuilt swresample_prebuilt swscale_prebuiltinclude $(BUILD_SHARED_LIBRARY)
讲解一下:
     Android.mk是用来描述要编译某个具体的模块,所需要的一些资源,包括要编译的源码、要链接的库等等。对于ffmpeg相关库来说,LOCAL_MODULE变量用于指定该预编译库名称,可以任意,但是下面的LOCAL_SHARED_LIBRARIES要与指定的一致;LOCAL_SRC_FILES指定ffmpeg相关预编译so库所在路径,我这里存放在jni目录下。另外,还需要使用LOCAL_C_INCLUDES 变量指定ffmpeg相关头文件所在目录,$(LOCAL_PATH)为当前jni目录路径。
5. 配置Application.mk
#指定so支持的平台APP_ABI := armeabi
讲解一下:
    相对于Android.mk,Application.mk是用来描述你的应用程序需要哪些模块,以及这些模块所要具有的一些特性。由于我们在编译ffmpeg时,只编译了arm架构的so,因此,在Android工程的Application.mk文件中需要使用APP_ABI变量指定只生成armeiabi架构机器码,如果这里不处理,在ndk-build时会报错。
6.  执行ndk-build命令,生成so共享库

7. 调用native方法,查看运行结果
public class MainActivity extends Activity {private TextView mTvDegreeInfo;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);mTvDegreeInfo = (TextView) findViewById(R.id.tv_video_degree);}public void onRotateClick(View v) {String rootPath = Environment.getExternalStorageDirectory().getAbsolutePath();File file = new File(rootPath + File.separator + "20170627_145524.mp4");if (!file.exists()) {return;}mTvDegreeInfo.setText("读取到20170627_145524.mp4的旋转角度:\n rotate = " + VideoFixUtils.getVideoAngle(file.getAbsolutePath()) + "度");}
使用mediaInfo软件查看视频信息,对比旋转角度:

          


Demo下载:详解ffmpeg编译与在Android平台上的移植