- FFmpeg for Android with libvpx, x264 and lame options. The scripts in this repository are configured to build:
library | version | platform support | arch support |
ffmpeg | 4.1.1 | android | armeabi-v7a arm64-v8a x86 x86_64 |
libvpx | 1.10.0 | android | armeabi-v7a arm64-v8a x86 x86_64 |
x264 | 160 | android | armeabi-v7a arm64-v8a x86 x86_64 |
lame | 3.100 | android | armeabi-v7a arm64-v8a x86 x86_64 |
- armeabi*
- armeabi-v7a
- arm64-v8a
- x86
- x86_64
- mips*
- mips64*
Note: *-Deprecated in android ndk-r16. Will be removed in ndk-r17.
see https://developer.android.com/ndk/guides/abis.html#Supported ABIs
ffmpeg-andorid releases > v1.5.0 will only verify ABIS=("armeabi-v7a" "arm64-v8a" "x86" "x86_64")
- Set environment ANDROID_NDK variable; not all ndk releases work with all modules/ABIS combinations
- export ANDROID_NDK={Android NDK Base Path}
e.g. export ANDROID_NDK=/opt/android/android-ndk-r17c (recommended); - Note: android-ndk-r18b will have problem in some module builds e.g. lame
- export ANDROID_NDK={Android NDK Base Path}
- If necessary, fetch and update all libraries source (before build);
- review and edit ./init_update_libs.sh file for your desired modules' versions, then run
- ./init_update_libs.sh
- The default codec module extracted sub-directories are ffmpeg, libvpx, x264 and lame
- Optionally, go to the respective directory for each sub-module, and execute ./configure without option;
The configure process may list any missing sdk build tools, please install before continue - The actual configure with options for each submodule build is done in each submodule _<module>_build.sh build script
- Edit ./ffmpeg-android_build.sh [#1] and ./_settings.sh [#2];
a. remove any of the codec sub-modules or architectures you wish to be excluded from the build
b. add "ffmpeg" only to [#1] if you want to build standalone ffmpeg without inclusion of the sub-modules integration
c. the default values are defined as:- MODULES=("vpx" "x264" "lame") [#1]
- ABIS=("armeabi-v7a" "arm64-v8a" "x86" "x86_64") [#2]
- To support 64-bit libraries built, ensure _settings.sh#ANDROID_API=21 (min API for 64-bit library build).
Note: application.mk in Android Studio i.e. APP_PLATFORM := android-21 i.e. both must use the same API - Use the following commands to compile ffmpeg for all or custom defined modules/ABI's combinations
- ./ffmpeg-android_build.sh
: (combinaiton as specified in the [#1] and [#2]) - ./ffmpeg-android_build.sh armeabi-v7a
: (i.e. selected cpu with all predefined codec modules) - ./ffmpeg-android_build.sh armeabi-v7a x264
: (i.e. selected cpu with only x264 codec module)
- ./ffmpeg-android_build.sh
- All the generated static/shared libraries and includes files are installed in ./jni/<MODULE>/android/<ABI>.
- Note: Manual changes as outlined below are not further required.
the latest _vpx_build.sh has included the scripts to perform the following steps automatically. - Android has an issue with loading versioned .so shared libraries e.g. x264:
- Causing error during run: java.lang.UnsatisfiedLinkError: dlopen failed: library "libx264.so.147" not found
- Perform the following patches, if you want to link with shared .so libraries for x264.
- use GHex to change file content "libx264.so.147" to "libx264_147.so"
- change filename from libx264.so.147 to libx264_147.so
- Note: the .so filename must match with the file changed content
-
The scripts has been verified working with the following configurations:
- NDK version: ndk-r17c (build may failed with lower or higher versions - see below)
- ABIS: armeabi-v7a, arm64-v8a, x86, x86_64
- MODULES (with applied patches): ffmpeg-v4.1.1, libvpx-v1.10.0, x264-v160, lame-v3.100
- ANDROID_API: 21
-
x264 (v160, v157 and v152):
- x264 option: --disable-asm
Must exclude the option for arm64. Used by configure, config.mak and Makefile to define AS and to compile required *.S assembly files.
Otherwise will have undefined references e.g. x264_8_... x264_10...
However must include the option for x86 and x86_64; otherwise have relocate text, requires dynamic R_X86_64_PC32 etc when use in aTalk - ndk-r18b, ndk-r17c, ndk-16b, ndk-r15c (build all)
- x264 option: --disable-asm
-
libvpx (v1.10.0, v.1.8.2, v1.8.0):
- libvpx configure.sh needs patches to correctly build the arm64 with NDK standalone toolchains.
- Valid for libvpx v1.8.0 only, option removed for libvpx v1.8.2+:
When --sdk-path is specified, libvpx configure uses SDK toolchains compiler (gcc/g++);- ndk-r18b, ndk-r17c, ndk-r16b:
To avoid missing stdlib.h and other errors, need to include the following two options for SDK toolchains:
--extra-cflags="-isystem ${NDK}/sysroot/usr/include/${NDK_ABIARCH} -isystem ${NDK}/sysroot/usr/include"
--libc=${NDK_SYSROOT} => use standalone toolchains directory.
libvpx v1.8.0 configure.sh has a problem configure this with SDK properly - ndk-r18b: gcc option has been removed
build failed with: /home/cmeng/workspace/ndk/ffmpeg-android/toolchain-android/bin/aarch64-linux-android-ld: cannot find -lgcc
- ndk-r18b, ndk-r17c, ndk-r16b:
- libvpx v1.8.0: When using standalone toolchains, i.e. omit --sdk-path;
- ndk-r18b, ndk-r17c, ndk-r16b:
Build ok with ABIS=("arm64-v8a" "x86" "x86_64") but not "armeabi-v7a" and failed with:
/tmp/vpx-conf-4350-25363.o(.ARM.exidx.text.main+0x0): error: undefined reference to '__aeabi_unwind_cpp_pr0'
- ndk-r18b, ndk-r17c, ndk-r16b:
- For libvpx v1.8.2+
- ndk-r18b, ndk-r17c, ndk-r16b:
All built OK.
- ndk-r18b, ndk-r17c, ndk-r16b:
-
lame (v3.1000):
- ndk-r18b - failed as lame need C++ instead of clang compiler
- ndk-r17b - build all and can integrate with ffmpeg
- ndk-r16b, ndk-r15c (build all)
- PREFIX must use absolution path
-
ffmpeg (v4.1.1):
- Must include option --disable-asm for x86, otherwise
libavcodec/x86/cabac.h:193:9: error: inline assembly requires more registers than available
ffmpeg (v1.0.10) => must also include this option for arm/arm64 build, otherwise errors during compilation - ndk-r18b, ndk-r17c => give error on:
libavdevice/v4l2.c:135:9: error: assigning to 'int (*)(int, unsigned long, ...)' from incompatible type '' SET_WRAPPERS();
add to CFLAG: -DBIONIC_IOCTL_NO_SIGNEDNESS_OVERLOAD; see tanersener/mobile-ffmpeg#48 - ndk-r16b or lower => gives error on:
libavformat/udp.c:290:28: error: member reference base type '__be32' (aka 'unsigned int') is not a structure or union
mreqs.imr_multiaddr.s_addr = ((struct sockaddr_in *)addr)->sin_addr.s_addr;
- Must include option --disable-asm for x86, otherwise
-
Note:
- NDK verification status (build with clang/clang++ and API-21 unless otherwise specified):
- Recommendation: use ndk-r17c (work for all) and API-21 unless you have other considerations.
- clang support needs min ndk-r12c, but may not necessary works for all ABIS (64-bit built has problem)
- Change _settings.sh with clang=>gcc and clang++=>g++ if you need lower ndk version
- NDK >r18b has obsoleted the support for the gcc/g++
-
Set up Linux/Ubuntu development environment with the below build tools
- sudo apt-get --quiet --yes install build-essential git autoconf libtool pkg-config gperf gettext yasm python-lxml
-
Patches for Sub-module
- ./ffmpeg-android_build.sh includes the patches for the sub-modules
- ffmpeg-android_patch.sh applies patches to the relevant sub-module with patch files from ./pathes directory
- edit these files to include additional patches if required.
-
Configuration failed
- You may encounter this problem during the codec sub-module build, navigate to the respective sub-module directory:
- Issue the command line i.e. configure --help to check for the available options
- Refer to config.log of the sub-module for more info.
- Refer to the configure file for more information on CPU types supported
- Edit the script file to make the necessary modifications based on help and errors found in config.log
- You may encounter this problem during the codec sub-module build, navigate to the respective sub-module directory:
-
During sub-module built, you may encountered compilation or linker errors, changing ndk version
may help to resolve the issues. However it may create new problems in another areas. -
Utilize Modern Compiler Flags to Address Potential Security Issues
- Stack execution protection: LDFLAGS="-z noexecstack"
- Data relocation and protection (RELRO): LDLFAGS="-z relro -z now"
- Stack-based Buffer Overrun Detection: CFLAGS=”-fstack-protector-strong” if using GCC 4.9 or newer, otherwise CFLAGS="-fstack-protector"
- Position Independent Execution (PIE) CFLAGS="-fPIE -fPIC" LDFLAGS="-pie" (PIE for executables only)
- Fortify source: CFLAGS="-O2 -D_FORTIFY_SOURCE=2"
- Format string vulnerabilities: CFLAGS="-Wformat -Wformat-security"
-
All the information provided here are by trial and error; compiling and linkage are very sensitive to the
modules version, configure options, and the NDK version used. You are advice to build without any changes
to the scripts provided here to confirm working, before attempt to make any modifications.
-
The scripts in this folder are not compatible with Unified Headers:
See https://android.googlesource.com/platform/ndk/+/master/docs/UnifiedHeaders.md#supporting-unified-headers-in-your-build-system -
Both the android NDK, ffmpeg and its sub-modules are in continous update, it is likely that
some of the defined ABI's settings may need to be tweaked as submodule configure have new changes.
Please refer to the following sites which may offer solution for problems you may experience.
- https://ffmpeg.org/pipermail/ffmpeg-user/2016-January/030202.html
- https://www.mail-archive.com/ffmpeg-devel@ffmpeg.org/msg62644.html
- http://alientechlab.com/how-to-build-ffmpeg-for-android/
- https://en.wikipedia.org/wiki/List_of_ARM_microarchitectures
- google/ExoPlayer#3520 (VP9 builds failure with android-ndk-r16 #3520)
- android/ndk#190 (comment) (unknown type name __uint128_t on ndk-build #190)
- https://android.googlesource.com/platform/bionic/+/master/docs/32-bit-abi.md
- android/ndk#477 (mmap causes compile errors on r15c (unknown identifier) #477)
- android/ndk#503 (implicit declaration of function 'mmap' #503)
- nodejs/node#18671 (Utilize Modern Compiler Flags to Address Potential Security Issues #18671)
- https://android.googlesource.com/platform/external/libvpx/+/ca30a60d2d6fbab4ac07c63bfbf7bbbd1fe6a583 (Add visibility="protected" attribute for global variables referenced in asm files.)
ffmpeg, android static library for aTalk VoIP and Instant Messaging client
Copyright 2014 Eng Chong Meng
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.