2011-05-05 66 views
5

我一直在关注如何移植LAME库和Android NDK的指南。 http://blog.libertadtech.com/2011/02/porting-lame-encoder-to-android-arm.html调用移植库的本地函数

每一个工作正常我有libmp3lame.so文件。

现在对于第一个测试,我想叫简单的函数:

void get_lame_version(char *strbuf, size_t buflen, const char *prefix); 

我写了一个包装了LAME库:

package de.goddchen.android.youtubeconverter; 

public class LameWrapper { 

public static native void get_1lame_1version(char[] buf, int len, 
     char[] prefix); 
} 

我这样称呼它是这样的:

public class MainActivity extends Activity { 

static { 
    System.loadLibrary("mp3lame"); 
} 

    /** Called when the activity is first created. */ 
    @Override 
    public void onCreate(Bundle savedInstanceState) { 
     super.onCreate(savedInstanceState); 
     setContentView(R.layout.main); 
     char[] buffer = new char[100]; 
     LameWrapper.get_1lame_1version(buffer, 100, "".toCharArray()); 
     ((TextView) findViewById(R.id.text)).setText(buffer.toString()); 
    } 

} 

但我得到以下错误:

05-05 10:36:11.973: DEBUG/dalvikvm(199): Trying to load lib /data/data/de.goddchen.android.youtubeconverter/lib/libmp3lame.so 0x43757f78 
05-05 10:36:12.053: DEBUG/dalvikvm(199): Added shared lib /data/data/de.goddchen.android.youtubeconverter/lib/libmp3lame.so 0x43757f78 
05-05 10:36:12.123: DEBUG/dalvikvm(199): No JNI_OnLoad found in /data/data/de.goddchen.android.youtubeconverter/lib/libmp3lame.so 0x43757f78 
05-05 10:36:12.573: DEBUG/dalvikvm(199): +++ not scanning '/system/lib/libwebcore.so' for 'get_1lame_1version' (wrong CL) 
05-05 10:36:12.693: DEBUG/dalvikvm(199): +++ not scanning '/system/lib/libexif.so' for 'get_1lame_1version' (wrong CL) 
05-05 10:36:12.693: DEBUG/dalvikvm(199): +++ not scanning '/system/lib/libFFTEm.so' for 'get_1lame_1version' (wrong CL) 
05-05 10:36:12.703: DEBUG/dalvikvm(199): +++ not scanning '/system/lib/libmedia_jni.so' for 'get_1lame_1version' (wrong CL) 
05-05 10:36:12.703: DEBUG/dalvikvm(199): +++ not scanning '/system/lib/libsrec_jni.so' for 'get_1lame_1version' (wrong CL) 
05-05 10:36:12.703: WARN/dalvikvm(199): No implementation found for native Lde/goddchen/android/youtubeconverter/LameWrapper;.get_1lame_1version ([CI[C)V 
05-05 10:36:12.703: DEBUG/AndroidRuntime(199): Shutting down VM 
05-05 10:36:12.753: INFO/ActivityManager(52): Start proc com.android.alarmclock for broadcast com.android.alarmclock/.AlarmInitReceiver: pid=213 uid=10017 gids={} 
05-05 10:36:12.773: WARN/dalvikvm(199): threadid=3: thread exiting with uncaught exception (group=0x4001aa28) 
05-05 10:36:12.783: ERROR/AndroidRuntime(199): Uncaught handler: thread main exiting due to uncaught exception 
05-05 10:36:12.883: ERROR/AndroidRuntime(199): java.lang.UnsatisfiedLinkError: get_1lame_1version 
05-05 10:36:12.883: ERROR/AndroidRuntime(199):  at de.goddchen.android.youtubeconverter.LameWrapper.get_1lame_1version(Native Method) 
05-05 10:36:12.883: ERROR/AndroidRuntime(199):  at de.goddchen.android.youtubeconverter.MainActivity.onCreate(MainActivity.java:19) 
05-05 10:36:12.883: ERROR/AndroidRuntime(199):  at android.app.Instrumentation.callActivityOnCreate(Instrumentation.java:1123) 
05-05 10:36:12.883: ERROR/AndroidRuntime(199):  at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2364) 
05-05 10:36:12.883: ERROR/AndroidRuntime(199):  at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:2417) 
05-05 10:36:12.883: ERROR/AndroidRuntime(199):  at android.app.ActivityThread.access$2100(ActivityThread.java:116) 
05-05 10:36:12.883: ERROR/AndroidRuntime(199):  at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1794) 
05-05 10:36:12.883: ERROR/AndroidRuntime(199):  at android.os.Handler.dispatchMessage(Handler.java:99) 
05-05 10:36:12.883: ERROR/AndroidRuntime(199):  at android.os.Looper.loop(Looper.java:123) 
05-05 10:36:12.883: ERROR/AndroidRuntime(199):  at android.app.ActivityThread.main(ActivityThread.java:4203) 
05-05 10:36:12.883: ERROR/AndroidRuntime(199):  at java.lang.reflect.Method.invokeNative(Native Method) 
05-05 10:36:12.883: ERROR/AndroidRuntime(199):  at java.lang.reflect.Method.invoke(Method.java:521) 
05-05 10:36:12.883: ERROR/AndroidRuntime(199):  at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:791) 
05-05 10:36:12.883: ERROR/AndroidRuntime(199):  at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:549) 
05-05 10:36:12.883: ERROR/AndroidRuntime(199):  at dalvik.system.NativeStart.main(Native Method) 

有人可以请我指出正确的方式来调用libmp3lame.so的功能吗?

+0

也许需要加个字最后? public static native get_1lame_1version(char [] buf,int len,final char [] prefix); – Pasha 2011-05-05 10:54:16

+0

nope,不起作用:( – Goddchen 2011-05-05 10:58:45

+0

看看这个:http://groups.google.com/group/android-ndk/browse_thread/thread/186678942625f46a/557bfe20de389d24 – Pasha 2011-05-05 11:05:30

回答

5

不幸的是,你不能从Java调用任何旧的C函数。您必须使用具有“损坏”名称的函数或使用JNI_OnLoad机制。无论哪种方式,Dalvik虚拟机都会为函数调用添加额外的参数,这是常规的蹩脚的get_version_number函数不期望的。

这个example of using JNI解释了这个名字mangling。对于您的Java类,您需要具有此原型的函数:

void Java_de_goddchen_android_youtubeconverter_LameWrapper_get_11lame_11version 
    (JNIEnv *, jclass, jcharArray, jint, jcharArray); 

JNIEnv和jclass参数是调用接口的一部分。如果您使用的是C++,请记得指定extern "C"。 javah工具将为您生成正确的头文件。

如果我这样做,我可能会编写JNI C包装函数为我需要的跛脚电话。在这种特殊情况下,我没有使用带有长度参数的char *,而是将空char *作为最后一个参数,我创建了一个更简单的Java to C接口,它只返回了Java String。所以在Java中:

package de.goddchen.android.youtubeconverter; 

public class LameWrapper { 
    public static native String getVersion(); 
} 

然后在C,其中调用实际跛脚API:

JNIEXPORT jstring JNICALL Java_de_goddchen_android_youtubeconverter_LameWrapper_getVersion 
    (JNIEnv * env, jclass c) 
{ 
    char buffer[80]; 
    get_lame_version(buffer, sizeof(buffer) - 1, ""); 
    return (*env)->NewStringUTF(env, buffer)); 
} 

在这里创建,然后在Java/Dalvik的侧控制的新String的生命周期。