环境:
操作系统:Ubuntu 10.04
ffmpeg源码版本:ffmpeg-0.6.1
android ndk版本:android-ndk-r5b-linux-x86
目的:
用来了解ffmpeg源码的编译场景和编译过程
准备工作:
将android-ndk-r5b解压后放在/home/mfcai目录下
在/home/mfcai/android-ndk-r5b/samples/目录下创建一个FFMPEG文件夹,在FFMPEG目录下再新建一个jni文件夹,然后把ffmpeg-0.6源码放在jni里面,所以最后ffmpeg源码的路径是:
/home/mfcai/android-ndk-r5b/samples/FFMPEG/jni/ffmpeg-0.6
一、设置NDK环境变量
配置NDK环境,增加/home/ndk目录到PATH环境变量:
$sudo gedit ~/.bashrc
文件底部添加以下两行:
NDK_ROOT=/home/mfcai/android-ndk-r5b
export NDK_ROOT
二、配置编译参数
在ffmpeg-0.6源文件夹下创建一个config.sh
config.sh是一个脚本,执行这个脚本的时候又调用了另外一个脚本configure,
configure主要是根据编译选项,生成相应的编译配置,即自己定制编译选项的内容。
1)新建一个config.sh
#!/bin/bash
PREBUILT=${NDK_ROOT}/toolchains/arm-eabi-4.4.0/prebuilt/linux-x86
PLATFORM=${NDK_ROOT}/platforms/android-8/arch-arm
./configure --target-os=linux \
--arch=arm \
--enable-version3 \
--enable-gpl \
--enable-nonfree \
--disable-stripping \
--disable-ffmpeg \
--disable-ffplay \
--disable-ffserver \
--disable-ffprobe \
--disable-encoders \
--disable-muxers \
--disable-devices \
--disable-protocols \
--enable-protocol=file \
--enable-avfilter \
--disable-network \
--disable-mpegaudio-hp \
--disable-avdevice \
--enable-cross-compile \
--cc=$PREBUILT/bin/arm-eabi-gcc \
--cross-prefix=$PREBUILT/bin/arm-eabi- \
--nm=$PREBUILT/bin/arm-eabi-nm \
--extra-cflags="-fPIC -DANDROID" \
--disable-asm \
--enable-neon \
--enable-armv5te \
--extra-ldflags="-Wl,-T,$PREBUILT/arm-eabi/lib/ldscripts/armelf.x -Wl,-rpath-link=$PLATFORM/usr/lib -L$PLATFORM/usr/lib -nostdlib $PREBUILT/lib/gcc/arm-eabi/4.4.0/crtbegin.o $PREBUILT/lib/gcc/arm-eabi/4.4.0/crtend.o -lc -lm -ldl"
2)执行config.sh脚本文件
$ chmod +x config.sh
$ ./config.sh
如果配置正确的话显示出来的最后两行是这样的:
License: nonfree and unredistributable
Creating config.mak and config.h...
3)编辑config.h文件
编辑 /root/ffmpeg/jni/ffmpeg-0.6.1 目录中的 config.h 文件,
将#define restrict restrict修改为#define restrict
将
#define HAVE_LLRINT 0
#define HAVE_LLRINTF 0
#define HAVE_LRINT 0
#define HAVE_LRINTF 0
#define HAVE_ROUND 0
#define HAVE_ROUNDF 0
#define HAVE_TRUNCF 0
修改为:
#define HAVE_LLRINT 1
#define HAVE_LLRINTF 1
#define HAVE_LRINT 1
#define HAVE_LRINTF 1
#define HAVE_ROUND 1
#define HAVE_ROUNDF 1
#define HAVE_TRUNCF 1
三、准备编译ffmpeg
1、编辑ffmpegffmpeg-0.6.1下各个模块的Makefile文件
分别把libavutil、libavcodec、libavformat、libavfilter、libpostproct和libswscale下的Makefile文件中下面两行删除掉:
include $(SUBDIR)../subdir.mak
include $(SUBDIR)../config.mak
2、在ffmpegffmpeg-0.6.1根目录下新建一个av.mk文件
# LOCAL_PATH is one of libavutil, libavcodec, libavformat, or libswscale
#include $(LOCAL_PATH)/../config-$(TARGET_ARCH).mak
include $(LOCAL_PATH)/../config.mak
OBJS :=
OBJS-yes :=
MMX-OBJS-yes :=
include $(LOCAL_PATH)/Makefile
# collect objects
OBJS-$(HAVE_MMX) += $(MMX-OBJS-yes)
OBJS += $(OBJS-yes)
FFNAME := lib$(NAME)
FFLIBS := $(foreach,NAME,$(FFLIBS),lib$(NAME))
FFCFLAGS = -DHAVE_AV_CONFIG_H -Wno-sign-compare -Wno-switch -Wno-pointer-sign
FFCFLAGS += -DTARGET_CONFIG=/"config-$(TARGET_ARCH).h/"
ALL_S_FILES := $(wildcard $(LOCAL_PATH)/$(TARGET_ARCH)/*.S)
ALL_S_FILES := $(addprefix $(TARGET_ARCH)/, $(notdir $(ALL_S_FILES)))
ifneq ($(ALL_S_FILES),)
ALL_S_OBJS := $(patsubst %.S,%.o,$(ALL_S_FILES))
C_OBJS := $(filter-out $(ALL_S_OBJS),$(OBJS))
S_OBJS := $(filter $(ALL_S_OBJS),$(OBJS))
else
C_OBJS := $(OBJS)
S_OBJS :=
endif
C_FILES := $(patsubst %.o,%.c,$(C_OBJS))
S_FILES := $(patsubst %.o,%.S,$(S_OBJS))
FFFILES := $(sort $(S_FILES)) $(sort $(C_FILES))
3、新建Android.mk文件
在jni目录建立Android.mk文件.
在ffmpegffmpeg-0.6.1下新建Android.mk文件
在libavformat,libavcodec,libavfilter、libavutil、libpostproc和libswscale目录建立同样的文件
1)在jni文件夹新建Android.mk文件
include $(all-subdir-makefiles)
2)在ffmpeg下新建Android.mk文件
LOCAL_PATH := $(call my-dir)
include $(CLEAR_VARS)
LOCAL_WHOLE_STATIC_LIBRARIES := libavformat libavcodec libavutil libpostproc libswscale libavfilter
LOCAL_MODULE := ffmpeg
include $(BUILD_SHARED_LIBRARY)
include $(call all-makefiles-under,$(LOCAL_PATH))
3)在ffmpeg/libavformat下新建Android.mk
LOCAL_PATH := $(call my-dir)
include $(CLEAR_VARS)
include $(LOCAL_PATH)/../av.mk
LOCAL_SRC_FILES := $(FFFILES)
LOCAL_C_INCLUDES := \
$(LOCAL_PATH) \
$(LOCAL_PATH)/..
LOCAL_CFLAGS += $(FFCFLAGS)
LOCAL_CFLAGS += -include "string.h" -Dipv6mr_interface=ipv6mr_ifindex
LOCAL_LDLIBS := -lz
LOCAL_SHARED_LIBRARIES := libavutil libavcodec
LOCAL_MODULE := $(FFNAME)
include $(BUILD_SHARED_LIBRARY)
4)在ffmpeg/libavcodec下新建Android.mk
LOCAL_PATH := $(call my-dir)
include $(CLEAR_VARS)
include $(LOCAL_PATH)/../av.mk
LOCAL_SRC_FILES := $(FFFILES)
LOCAL_C_INCLUDES := \
$(LOCAL_PATH) \
$(LOCAL_PATH)/..
LOCAL_CFLAGS += $(FFCFLAGS)
LOCAL_LDLIBS := -lz
LOCAL_SHARED_LIBRARIES := libavutil
LOCAL_MODULE := $(FFNAME)
include $(BUILD_SHARED_LIBRARY)
5)在ffmpeg/libavfilter下新建Android.mk
LOCAL_PATH := $(call my-dir)
include $(CLEAR_VARS)
include $(LOCAL_PATH)/../av.mk
LOCAL_SRC_FILES := $(FFFILES)
LOCAL_C_INCLUDES := \
$(LOCAL_PATH) \
$(LOCAL_PATH)/..
LOCAL_CFLAGS += $(FFCFLAGS)
LOCAL_SHARED_LIBRARIES := libavutil libswscale libavcodec
LOCAL_MODULE := $(FFNAME)
include $(BUILD_SHARED_LIBRARY)
6)在ffmpeg/libavutil下新建Android.mk
LOCAL_PATH := $(call my-dir)
include $(CLEAR_VARS)
include $(LOCAL_PATH)/../av.mk
LOCAL_SRC_FILES := $(FFFILES)
LOCAL_C_INCLUDES := \
$(LOCAL_PATH) \
$(LOCAL_PATH)/..
LOCAL_CFLAGS += $(FFCFLAGS)
LOCAL_MODULE := $(FFNAME)
include $(BUILD_SHARED_LIBRARY)
7)在ffmpeg/libpostproc下新建Android.mk
LOCAL_PATH := $(call my-dir)
include $(CLEAR_VARS)
include $(LOCAL_PATH)/../av.mk
LOCAL_SRC_FILES := $(FFFILES)
LOCAL_C_INCLUDES := \
$(LOCAL_PATH) \
$(LOCAL_PATH)/..
LOCAL_CFLAGS += $(FFCFLAGS)
LOCAL_SHARED_LIBRARIES :=libavutil
LOCAL_MODULE := $(FFNAME)
include $(BUILD_SHARED_LIBRARY)
8)在ffmpeg/libswscale下的Android.mk文件
LOCAL_PATH := $(call my-dir)
include $(CLEAR_VARS)
include $(LOCAL_PATH)/../av.mk
LOCAL_SRC_FILES := $(FFFILES)
LOCAL_C_INCLUDES := \
$(LOCAL_PATH) \
$(LOCAL_PATH)/..
LOCAL_CFLAGS += $(FFCFLAGS)
LOCAL_SHARED_LIBRARIES := libavutil
LOCAL_MODULE := $(FFNAME)
include $(BUILD_SHARED_LIBRARY)
四、编译
在jni目录下运行ndk-build
如果编译成功,产生出libavcodec.so、libavfilter.so、libavformat.so、libavutil.so、 libffmpeg.so,libpostproc.so和libswscale.so。
五、问题
1、同样的配置,在cygwin下编译了两天通不过。所以建议还是在linux环境下编译
2、libffmpeg.so大小只有1k。
网上的做法修改ffmpeg下的android.mk
发现出来的libffmpeg.so只有1KB,按照网上搜索的资料,
方法一:修改ffmpeg下的android.mk:
LOCAL_STATIC_LIBRARIES := libavcodec libavformat libavutil libavfilter libpostproc libswscale
改后为
LOCAL_WHOLE_STATIC_LIBRARIES := libavcodec libavformat libavutil libavfilter libpostproc libswscale
方法二:指定使用 arm-eabi-4.4.0 来编译:
由于android-ndk-r5 默认是使用 arm-linux-androideabi-4.4.3 编译,而不是 arm-eabi-4.4.0。
但 android-ndk-r5/toolchains/arm-linux-androideabi-4.4.3 目录中的 setup.mk 文件里定义的cmd-build-shared-library 函数
并没有将静态库文件链接在一起生成动态库文件。所以解决的办法就是在执行 ndk-build 时加上 NDK_TOOLCHAIN 参数,指定使用 arm-eabi-4.4.0 来编译。
完整命令如下
ndk-build NDK_TOOLCHAIN=arm-eabi-4.4.0 NDK_PLATFORM=android-8
同样不行。
希望高手能给指条道了
需要编译通过的ffmpeg-0.61文件的,请自行下载
本文欢迎转载,转载请注明出处与作者
出处:http://blog.sina.com.cn/staratsky
作者:流星
分享到:
相关推荐
在FFmpeg源码目录下创建一个名为`build-android.sh`的脚本,配置编译选项,例如: ```bash #!/bin/bash export NDK_PATH=/path/to/your/android-ndk-r25c export TOOLCHAIN=$NDK_PATH/toolchains/llvm/prebuilt...
5. **Android可使用**:在Android平台上使用FFmpeg-jni-x264-x265,需要对FFmpeg源码进行特定的编译配置,以适应Android的不同架构(armeabi, armeabi-v7a, arm64-v8a, x86, x86_64等)。编译完成后,生成的`.so`...
4. **配置和编译FFmpeg**:运行`configure`脚本来配置FFmpeg,指定目标平台(Android),例如添加`--target-os=linux --arch=arm --cpu=cortex-a8 --cross-prefix=/path/to/ndk/toolchains/arm-linux-androideabi-...
在Android平台上,编译FFmpeg源码是一项技术性较强的任务,因为FFmpeg是一个跨平台的音频和视频处理库,主要用于多媒体文件的编码、解码、流处理等操作。本篇文章将详细阐述如何在Android环境下利用JNI接口编译...
在这个“android-ffmpeg-3.3.9”项目中,我们得到了针对Android平台编译的FFmpeg 3.3.9版本的动态链接库(SO文件)。这些SO文件是FFmpeg在Android环境下运行所必需的,允许开发者在Android应用中实现丰富的音视频...
在Android上使用FFmpeg,需要将FFmpeg源码编译为适合Android平台的静态或动态库(so文件)。这个过程涉及到Android NDK(Native Development Kit),它允许开发者使用C/C++编写原生代码,并将其集成到Android应用中...
1. **下载源码**:首先,从 FFmpeg 官方仓库获取源码,或者直接使用已编译的库文件 "android-ffmpeg-4.0.6.new.zip"。 2. **配置 NDK**:确保你的 Android 工程配置了 NDK 环境,并正确设置了 NDK 路径。 3. **...
通常使用Android Studio的CMakeLists.txt文件或者ndk-build脚本来编译FFmpeg源码。 4. **JNI接口**:在Android应用中调用FFmpeg库,需要通过Java Native Interface (JNI) 来创建C/C++与Java之间的桥梁。在JNI中定义...
ffmpeg6.1安卓交叉编译库 解压查看FFmpegLib/src/main/jniLibs/ 目录下的 arm64-v8a armeabi-v7a x86 x86_64 工程里有调用的示例,可以直接编译。
这涉及到配置不同的编译器(如arm-linux-androideabi-gcc),以及设置目标平台的架构、API级别和ABI(armeabi, armeabi-v7a, arm64-v8a, x86, x86_64等)。 3. **配置选项**: 移植过程中需要选择合适的FFmpeg配置...
FFmpeg 的源码编译是一个复杂的过程,需要对编译工具链、Android 平台以及 FFmpeg 内部结构有深入理解。这个经过修改的源码版本简化了这个过程,使得开发者可以直接编译而避免常见的问题。然而,由于每个项目的需求...
- 在Android环境中,可能需要使用`ndk-build`命令来编译FFmpeg,该命令会自动调用`make`并应用Android特定的构建规则。 5. **FFmpeg的集成**: - 编译完成后,将生成的静态库(`.a`文件)或动态库(`.so`文件)...
这个CMakeLists文件会告诉CMake如何编译FFmpeg源码,设置编译选项、目标架构和API级别。注意替换`your_target_name`为你的模块名。 在编译FFmpeg之前,可能还需要根据FFmpeg的配置文件进行一些预处理,例如移除不必...
接下来,我们需要编写`CMakeLists.txt`文件来编译FFmpeg。在这个文件中,我们将FFmpeg的源码目录添加到`include_directories()`,然后使用`add_library()`来定义FFmpeg库,并通过`target_sources()`指定源码文件。...
"build-scripts-of-ffmpeg-x264-for-android-ndk-master"这个压缩包文件很可能包含了一系列用于编译FFmpeg和x264(H264的开源实现)的脚本。这些脚本会指导编译过程,包括选择要编译的组件、指定目标架构、设置优化...
这个过程通常需要对FFmpeg源码结构和Android.mk文件有深入理解。 4. **x264编码库**:x264是高效的H.264视频编码库,广泛用于视频编码和转码任务。将x264集成到FFmpeg中,使得FFmpeg具备了处理H.264编码的能力,这...
--cross-prefix=$NDK/toolchains/llvm/prebuilt/linux-x86_64/bin/armv7a-linux-androideabi- \ --sysroot=$NDK/platforms/android-XX/arch-arm/ \ --enable-cross-compile \ --disable-shared \ --enable-pic \ --...
本文将详细介绍如何在Android Studio项目中编译FFmpeg以及如何在应用程序中对其进行普通调用。 1. **Android Studio 配置** - 首先,确保你的Android Studio已经安装了NDK,可以在`File` > `Project Structure` > ...
本资料包含的是针对 Android 平台编译的 FFmpeg so 库及其源码,采用了现代的 CMake 编译系统进行构建。 一、FFmpeg 框架介绍 FFmpeg 包含了多种音视频编解码器、容器格式解析器、滤波器和流媒体工具。它支持众多的...
编译FFmpeg 2.5.2在Android上的步骤通常涉及以下关键环节: 1. **环境配置**:安装Android NDK(Native Development Kit),设置NDK路径,配置交叉编译环境。 2. **源码获取**:从FFmpeg官方网站下载2.5.2版本的源...