Android多媒体之二:jni调用ffmpeg命令

FFmpeg除了提供了强大的编解码库之外,也提供了一些命令行工具ffmpeg、ffplay、ffprobe、ffserver。如果对lib不太熟悉,而要实现的功能也比较简单,可以直接调用ffmpeg的命令实现。

准备

以ffmpeg2.5为例,准备ffmpeg2.5源码,以及编译好的lib和include。编译方法见:

Android多媒体之一:编译ffmpeg

准备ndk,并配好环境变量;准备Android studio。

ffmpeg简要分析

首先看下ffmpeg各个库的作用,这样简单的函数根据函数名就能猜到作用是什么:

libavcodec encoding/decoding library
libavfilter graph-based frame editing library
libavformat I/O and muxing/demuxing library
libavdevice special devices muxing/demuxing library
libavutil common utility library
libswresample audio resampling, format conversion and mixing
libpostproc post processing library
libswscale color conversion and scaling library

接下来分析ffmpeg.c的main函数源码,因为我们调用命令就是执行main函数:


main函数主要调用了以下函数:

register_exit(ffmpeg_cleanup):注册退出时清理函数;

av_log_set_flags和parse_loglevel:跟log相关的,不管;

avcodec_register_all :注册编码解码器;

avdevice_register_all:注册特殊设备的封装库;

avfilter_register_all:注册帧编辑库;

av_register_all:注册所有封装和分离器;

avformat_network_init:初始化网络组建,ffmpeg是支持拉取远程视频流的;

show_banner:打印输出FFmpeg版本信息(编译时间,编译选项,类库信息等);

term_init:初始化对终端命令的响应;

ffmpeg_parse_options:解析输入的参数;

transcode:转码过程;

exit_progam:退出和清理。

编写代码

编写jni代码

既然执行命令的入口是在main函数,那我们在java代码中定义一个native方法,并在相应的c代码中调用main函数不就可以了吗。
新建一个android工程,我这里用包名org.mqstack.ffmpegjni,新建FFmpegJni.java,写入:

public class FFmpegJni {

    static {
        System.loadLibrary("avutil");
        System.loadLibrary("swresample");
        System.loadLibrary("avcodec");
        System.loadLibrary("avformat");
        System.loadLibrary("swscale");
        System.loadLibrary("avfilter");
        System.loadLibrary("avdevice");
        System.loadLibrary("ffmpegjni");
    }
    
    public int ffmpegRunCommand(String command) {
        if (command.isEmpty()) {
            return 1;
        }
        String[] args = command.split(" ");
        for (int i = 0; i < args.length; i++) {
            Log.d("ffmpeg-jni", args[i]);
        }
        return run(args.length, args);
    }
    
    public native int run(int argc, String[] args);
}

使用javah生成头文件:

javah -classpath path_to_FFmpegJni -jni class_name

生成的代码类似如下:

/* DO NOT EDIT THIS FILE - it is machine generated */
#include <jni.h>
/* Header for class org_mqstack_ffmpegjni_FFmpegJni */

#ifndef _Included_org_mqstack_ffmpegjni_FFmpegJni
#define _Included_org_mqstack_ffmpegjni_FFmpegJni
#ifdef __cplusplus
extern "C" {
#endif
/*
 * Class:     org_mqstack_ffmpegjni_FFmpegJni
 * Method:    run
 * Signature: (I[Ljava/lang/String;)I
 */
JNIEXPORT jint JNICALL Java_org_mqstack_ffmpegjni_FFmpegJni_run
        (JNIEnv *, jobject, jint, jobjectArray);

#ifdef __cplusplus
}
#endif
#endif

编写相应的c文件FFmpegJni.c,代码很好理解:
#include "logjni.h"
#include "FFmpegJni.h"

#include <stdlib.h>
#include <stdbool.h>

int main(int argc, char **argv);

JNIEXPORT jint JNICALL Java_org_mqstack_ffmpegjni_FFmpegJni_run(JNIEnv *env, jobject obj, jint argc, jobjectArray args) {
    int i = 0;
    char **argv = NULL;
    jstring *strr = NULL;

    if (args != NULL) {
        argv = (char **) malloc(sizeof(char *) * argc);
        strr = (jstring *) malloc(sizeof(jstring) * argc);

        for (i = 0; i < argc; ++i) {
            strr[i] = (jstring)(*env)->GetObjectArrayElement(env, args, i);
            argv[i] = (char *)(*env)->GetStringUTFChars(env, strr[i], 0);
            LOGD("args: %s", argv[i]);
        }
    }

    LOGD("Run ffmpeg");
    int result = main(argc, argv);
    LOGD("ffmpeg result %d", result);

    for (i = 0; i < argc; ++i) {
        (*env)->ReleaseStringUTFChars(env, strr[i], argv[i]);
    }
    free(argv);
    free(strr);

    return result;
}

其中logjni.h是我写的一个android打印方法的封装:
#ifndef LOGJAM_H
#define LOGJAM_H

#include <android/log.h>

#define LOGTAG "FFmpegJni"

#define LOGV(...) __android_log_print(ANDROID_LOG_VERBOSE, LOGTAG, __VA_ARGS__) 
#define LOGD(...) __android_log_print(ANDROID_LOG_DEBUG  , LOGTAG, __VA_ARGS__) 
#define LOGI(...) __android_log_print(ANDROID_LOG_INFO   , LOGTAG, __VA_ARGS__) 
#define LOGW(...) __android_log_print(ANDROID_LOG_WARN   , LOGTAG, __VA_ARGS__) 
#define LOGE(...) __android_log_print(ANDROID_LOG_ERROR  , LOGTAG, __VA_ARGS__) 

#endif

将上述文件放到工程的jni目录下,既然要执行ffmpeg.c中的函数,显然ffmpeg.c和ffmpeg.h要拷贝到jni目录下。
再看ffmpeg.c代码,里面include了一堆lib,
#include "libavformat/avformat.h"
#include "libavdevice/avdevice.h"
#include "libswresample/swresample.h"
#include "libavutil/opt.h"
#include "libavutil/channel_layout.h"
#include "libavutil/parseutils.h"
#include "libavutil/samplefmt.h"
#include "libavutil/fifo.h"
#include "libavutil/intreadwrite.h"
#include "libavutil/dict.h"
#include "libavutil/mathematics.h"
#include "libavutil/pixdesc.h"
#include "libavutil/avstring.h"
#include "libavutil/libm.h"
#include "libavutil/imgutils.h"
#include "libavutil/timestamp.h"
#include "libavutil/bprint.h"
#include "libavutil/time.h"
#include "libavutil/threadmessage.h"
#include "libavformat/os_support.h"

那显然要将之前编译成的include文件夹放在jni目录下,同时在jni目录下新建prebuild/armeabi文件夹,将之前编译的so拷贝进去。

ffmpeg.c中还用到了不是lib中的c

cmdutils.c ffmpeg_opt.c ffmpeg_filter.c 

还有这些代码中include到的h

cmdutils.h cmdutils_common_opts.h config.h

将这些文件全部拷贝到jni文件夹。

修改main函数

由于在终端中输入命令,ffmpeg可以新建进程、执行命令然后在销毁进程。但在java程序中,我们不想让命令执行完就直接退出。于是main函数中要做些改动:

将所有的异常的exit_progam,都改为return;

将打印lig的方法都改为android的打印;

将正常的exit_program改为ffmpeg_cleanup(0)。

/* parse options and open all input/output files */
ret = ffmpeg_parse_options(argc, argv);
    if (ret < 0) {
        LOGD("ffmpeg_parse_options err");
        return 1;
}

if (nb_output_files <= 0 && nb_input_files == 0) {
        show_usage();
LOGD("Use -h to get full help or, even better, run 'man %s'\n", program_name);
        return 1;
}

/* file converter / grab */
if (nb_output_files <= 0) {
        LOGD("At least one output file must be specified\n");
        return 1;
}

//     if (nb_input_files == 0) {
//         av_log(NULL, AV_LOG_FATAL, "At least one input file must be specified\n");
//         exit_program(1);
//     }

current_time = ti = getutime();
    if (transcode() < 0) {
        LOGD("failed~~");
        return 1;
}
    ti = getutime() - ti;
    if (do_benchmark) {
        LOGD("bench: utime=%0.3fs\n", ti / 1000000.0);
}
    LOGD("%"PRIu64" frames successfully decoded, %"PRIu64" decoding errors\n",
decode_error_stat[0], decode_error_stat[1]);
    if ((decode_error_stat[0] + decode_error_stat[1]) * max_error_rate < decode_error_stat[1])
        LOGD("log error");

ffmpeg_cleanup(0);
    return main_return_code;

还要在ffmpeg_cleanup函数的最后加上几行,在进程不销毁的情况下,将一些变量初始化。

nb_filtergraphs = 0;
nb_output_files = 0;
nb_output_streams = 0;
nb_input_files = 0;
nb_input_streams = 0;

编写mk文件

不多说了,参考官方说明

LOCAL_PATH:= $(call my-dir)

include $(CLEAR_VARS)
LOCAL_MODULE:= avcodec-prebuilt-armeabi
LOCAL_SRC_FILES:= prebuilt/armeabi/libavcodec.so
include $(PREBUILT_SHARED_LIBRARY)

include $(CLEAR_VARS)
LOCAL_MODULE:= avdevice-prebuilt-armeabi
LOCAL_SRC_FILES:= prebuilt/armeabi/libavdevice.so
include $(PREBUILT_SHARED_LIBRARY)

include $(CLEAR_VARS)
LOCAL_MODULE:= avfilter-prebuilt-armeabi
LOCAL_SRC_FILES:= prebuilt/armeabi/libavfilter.so
include $(PREBUILT_SHARED_LIBRARY)

include $(CLEAR_VARS)
LOCAL_MODULE:= avformat-prebuilt-armeabi
LOCAL_SRC_FILES:= prebuilt/armeabi/libavformat.so
include $(PREBUILT_SHARED_LIBRARY)

include $(CLEAR_VARS)
LOCAL_MODULE :=  avutil-prebuilt-armeabi
LOCAL_SRC_FILES := prebuilt/armeabi/libavutil.so
include $(PREBUILT_SHARED_LIBRARY)

include $(CLEAR_VARS)
LOCAL_MODULE := swresample-prebuilt-armeabi
LOCAL_SRC_FILES := prebuilt/armeabi/libswresample.so
include $(PREBUILT_SHARED_LIBRARY)

include $(CLEAR_VARS)
LOCAL_MODULE := swscale-prebuilt-armeabi
LOCAL_SRC_FILES := prebuilt/armeabi/libswscale.so
include $(PREBUILT_SHARED_LIBRARY)

include $(CLEAR_VARS)

LOCAL_MODULE := libffmpegjni

LOCAL_ARM_MODE := arm

LOCAL_SRC_FILES := FFmpegJni.c \
                   ffmpeg.c \
                   cmdutils.c \
                   ffmpeg_opt.c \
                   ffmpeg_filter.c

LOCAL_LDLIBS := -L$(SYSROOT)/usr/lib -llog -lz

LOCAL_SHARED_LIBRARIES:= avcodec-prebuilt-armeabi \
                         avdevice-prebuilt-armeabi \
                         avfilter-prebuilt-armeabi \
                         avformat-prebuilt-armeabi \
                         avutil-prebuilt-armeabi \
                         swresample-prebuilt-armeabi \
                         swscale-prebuilt-armeabi

LOCAL_C_INCLUDES += -L$(SYSROOT)/usr/include
LOCAL_C_INCLUDES += $(LOCAL_PATH)/include

LOCAL_CFLAGS := -DUSE_ARM_CONFIG

include $(BUILD_SHARED_LIBRARY)

ndk-build

接下来就cd到jni目录,然后ndk-build编译。编译过程中,应该会提示缺少一些头文件如下,从ffmpeg工程中拷贝过来就可以了。

libavutil/libm.h
libavformat/os_support.h
libavformat/ffm.h network.h url.h
compat/va_copy.h
libavresample/avresample.h version.h
libpostproc/postprocess.h version.h

编译结果在libs/armeabi目录下,如下所示

将这些so拷贝到android工程的jniLibs或其他依赖的目录中,跑一条命令试试看吧。

本文中代码可在我的github中看到。

https://github.com/mqstack/FFmpegJni

更多

Android多媒体之一:编译ffmpeg
Android多媒体之三:编译并使用x264库

参考

http://blog.csdn.net/leixiaohua1020/article/details/39760711

https://github.com/dxjia/ffmpeg-jni-sample

https://github.com/andynicholson/android-ffmpeg-x264

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,384评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,845评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,148评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,640评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,731评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,712评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,703评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,473评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,915评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,227评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,384评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,063评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,706评论 3 324
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,302评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,531评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,321评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,248评论 2 352

推荐阅读更多精彩内容