Amdroid直播开发之旅(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-r14b
jiangdongguo@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目录。
Amdroid直播开发之旅(5):详解ffmpeg编译与在Android平台上的移植_第1张图片
     其中,--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 clean
root@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共享库。
Amdroid直播开发之旅(5):详解ffmpeg编译与在Android平台上的移植_第2张图片

 

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

1. 创建Android NDK工程
Amdroid直播开发之旅(5):详解ffmpeg编译与在Android平台上的移植_第3张图片
讲解一下:
    有关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 
#include 
#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 ; inb_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 lib
include $(CLEAR_VARS)
LOCAL_MODULE    := avcodec_prebuilt
LOCAL_SRC_FILES := libavcodec-57.so
include $(PREBUILT_SHARED_LIBRARY)


include $(CLEAR_VARS)
LOCAL_MODULE    := avdevice_prebuilt
LOCAL_SRC_FILES := libavdevice-57.so
include $(PREBUILT_SHARED_LIBRARY)


include $(CLEAR_VARS)
LOCAL_MODULE    := avfilter_prebuilt
LOCAL_SRC_FILES := libavfilter-6.so
include $(PREBUILT_SHARED_LIBRARY)


include $(CLEAR_VARS)
LOCAL_MODULE    := avformat_prebuilt
LOCAL_SRC_FILES := libavformat-57.so
include $(PREBUILT_SHARED_LIBRARY)


include $(CLEAR_VARS)
LOCAL_MODULE    := avutil_prebuilt
LOCAL_SRC_FILES := libavutil-55.so
include $(PREBUILT_SHARED_LIBRARY)


include $(CLEAR_VARS)
LOCAL_MODULE    := swresample_prebuilt
LOCAL_SRC_FILES := libswresample-2.so
include $(PREBUILT_SHARED_LIBRARY)


include $(CLEAR_VARS)
LOCAL_MODULE    := swscale_prebuilt
LOCAL_SRC_FILES := libswscale-4.so
include $(PREBUILT_SHARED_LIBRARY)


#myapp lib
include $(CLEAR_VARS)
LOCAL_MODULE    := FFMPEG4Android
LOCAL_SRC_FILES := FFMPEG4Android.c
LOCAL_C_INCLUDES +=$(LOCAL_PATH)/include
LOCAL_LDLIBS := -llog
LOCAL_SHARED_LIBRARIES := avcodec_prebuilt avdevice_prebuilt avfilter_prebuilt avformat_prebuilt avutil_prebuilt swresample_prebuilt swscale_prebuilt
include $(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共享库
Amdroid直播开发之旅(5):详解ffmpeg编译与在Android平台上的移植_第4张图片
7. 调用native方法,查看运行结果

 

public class MainActivity extends Activity {
	private TextView mTvDegreeInfo;
	@Override
	protected 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软件查看视频信息,对比旋转角度:

 

Amdroid直播开发之旅(5):详解ffmpeg编译与在Android平台上的移植_第5张图片          Amdroid直播开发之旅(5):详解ffmpeg编译与在Android平台上的移植_第6张图片

 

 

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

 

你可能感兴趣的:(【ffmpeg,开发艺术】,Android视频直播)