FFmpeg: java.lang.UnsatisfiedLinkError при вызове класса Runnable

Мне нужно взять файл изображения и аудиофайл и создать видео. Я знаю, что это можно сделать с помощью

Runtime.getRuntime().exec("ffmpeg -i image.jpeg -i audio.mp3 out.avi") 

но только для рутованных устройств, поэтому я попытался создать оболочку JNI для main() из ffmpeg.c и вызвать ее из своей Activity, как здесь: http://demo860.blogspot.com/2010/07/android-ffmpeg-dynamic-module-jni.html

1.Этот код находится в ffmpeg.c:

int m_argc = 0;
char *m_pargv [30];

int dynamic_ffpmeg_main (int argc, char **argv);
jint JNICALL Java_com_ccmedia_codec_ffmpeg_mod_1run ( JNIEnv *, jclass, jstring,     jstring );

jint JNICALL Java_com_ccmedia_codec_ffmpeg_mod_1run ( JNIEnv *env, jclass class, jstring pj1, jstring pj2)
{
    // as in http://demo860.blogspot.com/2010/07/android-ffmpeg-dynamic-module-jni.html
}

int dynamic_ffpmeg_main(int argc, char **argv)
{
// as in http://demo860.blogspot.com/2010/07/android-ffmpeg-dynamic-module-jni.html
}

int main(int argc, char **argv)
{
    dynamic_ffpmeg_main ( argc, argv );
    return 0;
}

2. Этот код в моей.java:

public class FFmpegCreator implements Runnable {

    static boolean m_bret = false;
static String m_szconfig = " -i /sdcard/file.mpg -vcodec mpeg4 aaa.mpg";

//public native String unimplementedStringFromJNI();

static {
    try {

        System.out.println("[AdDBCache] Module load try ffmpeg : "
                + System.getProperty("java.library.path"));

        // System.load("/sdcard/arm_and/bin/libffmpeg.so");

        System.loadLibrary("ffmpeg");

        System.out.println("[AdDBCache] Module load success");

    }

    catch (Exception e) {

        System.out.println("[AdDBCache] Module load err : "
                + System.getProperty("java.library.path"));

    }

}

private static synchronized final native int Java_com_ccmedia_codec_ffmpeg_mod_1run(String name, String sztoken);

public void set_config(String sz_config) {

    m_szconfig = sz_config;

}

public void run_core(String sz_file, String sz_token) {

    int n_stat;
    m_bret = false;
    n_stat = Java_com_ccmedia_codec_ffmpeg_mod_1run(m_szconfig, sz_token);
    m_bret = true;

}

public void run() {

    run_core("", "");

}
}

3. И это в моей деятельности:

FFmpegCreator f = new FFmpegCreator ();
new Thread(f).start();

Но у меня есть

E/AndroidRuntime(25682): java.lang.UnsatisfiedLinkError: Java_com_ccmedia_codec_ffmpeg_mod_1run .

И я не могу понять, почему... Сборка FFmpeg была успешной... Может кто-нибудь помочь мне, пожалуйста? Я буду очень признателен, если вы поможете мне. Спасибо.

1 ответ

Проблема заключается в наименовании нативного метода Java_com_ccmedia_codec_ffmpeg_mod_1run на стороне Java. Вы должны просто дать ему нормальное имя метода, без всех частей Java_package.... Затем, чтобы сопоставить это с функцией C, вам нужно использовать пакет и класс, которому принадлежит метод. Самый надежный способ сделать это - сначала обновить сторону Java:

public class FFmpegCreator implements Runnable {
    // ...

    private static synchronized final native int mod_1run(String name, String sztoken);

    //...
}

а потом запусти джаву по классу:

$ javah -o FFmpegCreator.h -classpath bin/classes com.yourpackage.FFmpegCreator

(заменить bin/classes с каталогом, где ваши файлы.class компилируются, и com.yourpackage с пакетом, который FFmpegCreator внутри). Если вы посмотрите на FFmpegCreator.h он генерирует это будет включать правильную подпись для вашего собственного метода.

Другие вопросы по тегам