ubuntu 18.04编译FFmpeg并实现 Android NDK解码

1,008 阅读1分钟

1 下载官方FFmpeg Linux源码

例如: ffmpeg.org/releases/ff…

2 下载NDK并解压

例如:dl.google.com/android/rep…

Downloads# unzip android-ndk-r14b-linux-x86_64.zip

3 解压FFmpeg源码压缩包

Downloads/ffmpeg# tar -zxvf ffmpeg-2.6.9.tar.gz

4 修改ffmpeg源码配置

在源码根目录下找到configure文件

Downloads/ffmpeg/ffmpeg-2.6.9# subl configure

修改如下:

#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_MAJOR='$(SLIBPREF)$(FULLNAME)-$(LIBMAJOR)$(SLIBSUF)'
LIB_INSTALL_EXTRA_CMD='$$(RANLIB)"$(LIBDIR)/$(LIBNAME)"'
SLIB_INSTALL_NAME='$(SLIBNAME_WITH_MAJOR)'
SLIB_INSTALL_LINKS='$(SLIBNAME)'

5 编写FFmpeg编译armv7a脚本

在FFmpeg源码根目录下创建build_v7a.sh文件。例如需要编译armv7a动态库,脚本按如下编写。其他平台另外参考。

#!/bin/bash
make clean
export NDK=/home//Downloads/android-ndk-r14b
export SYSROOT=$NDK/platforms/android-19/arch-arm/
export TOOLCHAIN=$NDK/toolchains/arm-linux-androideabi-4.9/prebuilt/linux-x86_64
export CPU=armv7-a
export PREFIX=$(pwd)/android/$CPU
export ADDI_CFLAGS="-marm -march=armv7-a"
export ADDI_LDFLAGS="-marm -march=armv7-a"

./configure --target-os=linux \
--prefix=$PREFIX --arch=arm \
--disable-doc \
--enable-shared \
--disable-static \
--disable-yasm \
--disable-symver \
--enable-gpl \
--disable-ffmpeg \
--disable-ffplay \
--disable-ffprobe \
--disable-ffserver \
--disable-doc \
--disable-symver \
--cross-prefix=$TOOLCHAIN/bin/arm-linux-androideabi- \
--enable-cross-compile \
--sysroot=$SYSROOT \
--extra-cflags="-Os -fpic $ADDI_CFLAGS" \
--extra-ldflags="$ADDI_LDFLAGS"

make clean
make
make install

6 编译

chmod 777 -R build_v7a.sh
./build_v7a.sh

7 编译结果

8 测试Android平台下测试FFmpeg

8.1 在app/src/main下创建jni文件夹,将上图中的android/armv7-a/include文件夹拷贝到jni下;

8.2 在app/src/main/下创建jniLibs/armeabi-v7a文件夹,将android/armv7-a/lib/的不带软连接的so文件拷贝到jniLibs/armeabi-v7a中。

8.3 编写native方法

public class Jni {
    static {
        System.loadLibrary("_decode");
    }
    public static native void decode(String src, String dest);
}

生成jni头文件并将生成的com_jnitest_Jni.h剪切到app/src/main/jni中

jni/app/src/main/java/com/jnitest# javac -h . Jni.java 

8.4 在app/src/main/jni下编写decode.c,代码如下:

//
// Created by root on 20-8-15.
//

#include "com_jnitest_Jni.h"

#include <android/log.h>

#include "libavcodec/avcodec.h"
//封装格式
#include "libavformat/avformat.h"
//缩放
#include "libswscale/swscale.h"

#define LOGI(FORMAT, ...) __android_log_print(ANDROID_LOG_INFO, "jni", FORMAT, ##__VA_ARGS__);
#define LOGE(FORMAT, ...) __android_log_print(ANDROID_LOG_ERROR, "jni", FORMAT, ##__VA_ARGS__);


JNIEXPORT void JNICALL Java_com__jnitest_Jni_decode
(JNIEnv *env, jclass jclz, jstring input_jstr, jstring output_jstr){
    LOGE("%s","jni decode ffmpeg");

    const char* input_cstr = (*env) -> GetStringUTFChars(env, input_jstr, NULL);
    const char* output_cstr = (*env) -> GetStringUTFChars(env, output_jstr, NULL);

    //1. 注册所有组件
    av_register_all();

    //封装格式上下文
    AVFormatContext* pFormatCtx = avformat_alloc_context();
    //2. 打开输入视频文件,成功返回0,第三个参数为NULL,表示自动检测文件格式
    if (avformat_open_input(&pFormatCtx, input_cstr, NULL, NULL) != 0) {
        LOGE("%s", "打开输入视频文件失败");
        return;
    }

    //3. 获取视频文件信息
    if (avformat_find_stream_info(pFormatCtx, NULL) < 0) {
        LOGE("%s", "获取视频文件信息失败");
        return;
    }

    //查找视频流所在的位置
    //遍历所有类型的流(视频流、音频流可能还有字幕流),找到视频流的位置
    int video_stream_index = -1;
    int i = 0;
    for(; i < pFormatCtx -> nb_streams; i++) {
        if (pFormatCtx->streams[i]->codec-> codec_type == AVMEDIA_TYPE_VIDEO) {
            video_stream_index = i;
            break;
        }
    }

    //编解码上下文
    AVCodecContext* pCodecCtx = pFormatCtx->streams[video_stream_index]->codec;
    //4. 查找解码器 不能通过pCodecCtx->codec获得解码器
    AVCodec* pCodec = avcodec_find_decoder(pCodecCtx->codec_id);
    if (pCodec == NULL) {
        LOGE("%s", "查找解码器失败");
        return;
    }

    //5. 打开解码器
    if (avcodec_open2(pCodecCtx, pCodec, NULL) < 0) {
        LOGE("%s", "打开解码器失败");
        return;
    }

    //编码数据
    AVPacket* pPacket = (AVPacket*)av_malloc(sizeof(AVPacket));

    //像素数据(解码数据)
    AVFrame* pFrame = av_frame_alloc();
    AVFrame* pYuvFrame = av_frame_alloc();

    FILE* fp_yuv = fopen(output_cstr, "wb");

    //只有指定了AVFrame的像素格式、画面大小才能真正分配内存
    //缓冲区分配内存
    uint8_t* out_buffer = (uint8_t*)av_malloc(avpicture_get_size(AV_PIX_FMT_YUV420P, pCodecCtx->width, pCodecCtx->height));
    //初始化缓冲区
    avpicture_fill((AVPicture*)pYuvFrame, out_buffer, AV_PIX_FMT_YUV420P, pCodecCtx->width, pCodecCtx->height);

    //srcW:源图像的宽
    //srcH:源图像的高
    //srcFormat:源图像的像素格式
    //dstW:目标图像的宽
    //dstH:目标图像的高
    //dstFormat:目标图像的像素格式
    //flags:设定图像拉伸使用的算法
    struct SwsContext* pSwsCtx = sws_getContext(
            pCodecCtx->width, pCodecCtx->height, pCodecCtx->pix_fmt,
            pCodecCtx->width, pCodecCtx->height, AV_PIX_FMT_YUV420P,
            SWS_BILINEAR, NULL, NULL, NULL);

    int got_frame, len, frameCount = 0;
    //6. 从输入文件一帧一帧读取压缩的视频数据AVPacket
    while(av_read_frame(pFormatCtx, pPacket) >= 0) {
        if (pPacket->stream_index == video_stream_index) {
            //7. 解码一帧压缩数据AVPacket ---> AVFrame,第3个参数为0时表示解码完成
            len = avcodec_decode_video2(pCodecCtx, pFrame, &got_frame, pPacket);

            if (len < 0) {
                LOGE("%s", "解码失败");
                return;
            }
            //AVFrame ---> YUV420P
            //srcSlice[]、dst[]        输入、输出数据
            //srcStride[]、dstStride[] 输入、输出画面一行的数据的大小 AVFrame 转换是一行一行转换的
            //srcSliceY                输入数据第一列要转码的位置 从0开始
            //srcSliceH                输入画面的高度
            sws_scale(pSwsCtx,
                      pFrame->data, pFrame->linesize, 0, pFrame->height,
                      pYuvFrame->data, pYuvFrame->linesize);

            //非0表示正在解码
            if (got_frame) {
                //图像宽高的乘积就是视频的总像素,而一个像素包含一个y,u对应1/4个y,v对应1/4个y
                int yuv_size = pCodecCtx->width * pCodecCtx->height;
                //写入y的数据
                fwrite(pYuvFrame->data[0], 1, yuv_size, fp_yuv);
                //写入u的数据
                fwrite(pYuvFrame->data[1], 1, yuv_size/4, fp_yuv);
                //写入v的数据
                fwrite(pYuvFrame->data[2], 1, yuv_size/4, fp_yuv);

                LOGI("解码第%d帧", frameCount++);
            }
            av_free_packet(pPacket);
        }
    }

    fclose(fp_yuv);
    av_frame_free(&pFrame);
    av_frame_free(&pYuvFrame);
    avcodec_free_context(&pCodecCtx);
    avformat_free_context(pFormatCtx);

    (*env) -> ReleaseStringUTFChars(env, input_jstr, input_cstr);
    (*env) -> ReleaseStringUTFChars(env, output_jstr, output_cstr);
}

8.5 在app目录下创建CMakeLists.txt,

cmake_minimum_required(VERSION 3.4.1)

set(path_project /home//test/jni)

add_library(_decode
            SHARED
            src/main/jni/decode.c)

add_library(avcodec SHARED IMPORTED)
set_target_properties(avcodec PROPERTIES IMPORTED_LOCATION
                      ${path_project}/app/src/main/jniLibs/${ANDROID_ABI}/libavcodec-56.so
        IMPORTED_NO_SONAME 1)

add_library(avdevice SHARED IMPORTED)
set_target_properties(avdevice PROPERTIES IMPORTED_LOCATION
                      ${path_project}/app/src/main/jniLibs/${ANDROID_ABI}/libavdevice-56.so
        IMPORTED_NO_SONAME 1)

add_library(avfilter SHARED IMPORTED)
set_target_properties(avfilter PROPERTIES IMPORTED_LOCATION
                      ${path_project}/app/src/main/jniLibs/${ANDROID_ABI}/libavfilter-5.so
        IMPORTED_NO_SONAME 1)

add_library(avformat SHARED IMPORTED)
set_target_properties(avformat PROPERTIES IMPORTED_LOCATION
                      ${path_project}/app/src/main/jniLibs/${ANDROID_ABI}/libavformat-56.so
        IMPORTED_NO_SONAME 1)
add_library(avutil SHARED IMPORTED)
set_target_properties(avutil PROPERTIES IMPORTED_LOCATION
                      ${path_project}/app/src/main/jniLibs/${ANDROID_ABI}/libavutil-54.so
                      IMPORTED_NO_SONAME 1)

add_library(postproc SHARED IMPORTED)
set_target_properties(postproc PROPERTIES IMPORTED_LOCATION
                     ${path_project}/app/src/main/jniLibs/${ANDROID_ABI}/libpostproc-53.so)

add_library(swresample SHARED IMPORTED)
set_target_properties(swresample PROPERTIES IMPORTED_LOCATION
                      ${path_project}/app/src/main/jniLibs/${ANDROID_ABI}/libswresample-1.so
        IMPORTED_NO_SONAME 1)

add_library(swscale SHARED IMPORTED)
set_target_properties(swscale PROPERTIES IMPORTED_LOCATION
                      ${path_project}/app/src/main/jniLibs/${ANDROID_ABI}/libswscale-3.so
        IMPORTED_NO_SONAME 1)

include_directories(src/main/jni/include)

find_library(log-lib
             log )

target_link_libraries(_decode
                      avutil
                      avcodec
                      avdevice
                      swresample
                      swscale
                      avfilter
                      avformat
                      postproc
                      ${log-lib})

8.6 app/build.gradle配置如下

apply plugin: 'com.android.application'

android {
    compileSdkVersion 29
    buildToolsVersion "29.0.3"
    defaultConfig {
        applicationId "com..jnitest"
        minSdkVersion 15
        targetSdkVersion 29
        versionCode 1
        versionName "1.0"
        testInstrumentationRunner "androidx.test.runner.AndroidJUnitRunner"

        externalNativeBuild {
            cmake {
                cppFlags ""
            }
            ndk {
                abiFilters 'armeabi-v7a'
            }
        }

    }
    buildTypes {
        release {
            minifyEnabled false
            proguardFiles getDefaultProguardFile('proguard-android-optimize.txt'), 'proguard-rules.pro'
        }
    }
    sourceSets {
        main{
            jniLibs.srcDirs = ['src/main/jniLibs']
        }
    }
    externalNativeBuild {
        cmake {
            path file('CMakeLists.txt')
        }
    }
}

dependencies {
    implementation fileTree(dir: 'libs', include: ['*.jar'])
    implementation 'androidx.appcompat:appcompat:1.0.2'
    implementation 'androidx.constraintlayout:constraintlayout:1.1.3'
    testImplementation 'junit:junit:4.12'
    androidTestImplementation 'androidx.test.ext:junit:1.1.0'
    androidTestImplementation 'androidx.test.espresso:espresso-core:3.1.1'
}

添加权限

 <uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" />
    <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
    <uses-permission android:name="android.permission.MOUNT_UNMOUNT_FILESYSTEMS" />

测试解码

String input = new File(Environment.getExternalStorageDirectory(),"input.mp4").getAbsolutePath();
                String output = new File(Environment.getExternalStorageDirectory(),"output.avi").getAbsolutePath();
               Jni.decode(in

8.7 测试结果