我正在尝试在Android上使用libsodium加密库(可移植版本的NaCl(nacl.cr.yp.to)),但我无法弄清楚如何正确编译/使用{ {3}}语言绑定。
到目前为止,我成功编写了libsodium网站上描述的libsodium。另外,我使用了3天前添加的新的android-arm编译脚本。作为测试,我编译了android-x86版本,以比较生成的库文件的文件大小。它们不同,所以这似乎有效。 下一步我使用新生成的android-arm libsodium文件安装KaliumJNI。这里没问题。
在我的小型android-maven测试应用程序中,它基本上是一个“Hello World”原型,我只是想生成一个新的密钥对:
import org.abstractj.kalium.keys.KeyPair;
KeyPair Alice = new KeyPair();
IntelliJ成功编译了我的小程序,但当我尝试在我的模拟器(或我的Nexus S)上运行它时,它会崩溃。首先它崩溃了,因为它无法找到kaliumjni库,所以我将libtestjni.so(这是由KaliumJNI生成的.so库)添加到我的项目属性中,修复了这个问题。我收到的下一条错误消息告诉我libtestjni.so不是32bit(我正在运行64位Ubuntu 12.04LTS)。我不明白为什么它必须是32位但无论如何我设法建立一个32位Ubuntu 12.04LTS VM并再次编译android-arm libsodium和KaliumJNI以生成一个32位libtestjni.so文件。但是当我在原始应用程序中使用这个32位文件时(在我的64位机器上),我收到以下错误:
java.lang.IllegalStateException: Could not execute method of the activity
at android.view.View$1.onClick(View.java:3823)
at android.view.View.performClick(View.java:4438)
at android.view.View$PerformClick.run(View.java:18422)
at android.os.Handler.handleCallback(Handler.java:733)
at android.os.Handler.dispatchMessage(Handler.java:95)
at android.os.Looper.loop(Looper.java:136)
at android.app.ActivityThread.main(ActivityThread.java:5017)
at java.lang.reflect.Method.invokeNative(Native Method)
at java.lang.reflect.Method.invoke(Method.java:515)
at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:779)
at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:595)
at dalvik.system.NativeStart.main(Native Method)
Caused by: java.lang.reflect.InvocationTargetException
at java.lang.reflect.Method.invokeNative(Native Method)
at java.lang.reflect.Method.invoke(Method.java:515)
at android.view.View$1.onClick(View.java:3818)
at android.view.View.performClick(View.java:4438)
at android.view.View$PerformClick.run(View.java:18422)
at android.os.Handler.handleCallback(Handler.java:733)
at android.os.Handler.dispatchMessage(Handler.java:95)
at android.os.Looper.loop(Looper.java:136)
at android.app.ActivityThread.main(ActivityThread.java:5017)
at java.lang.reflect.Method.invokeNative(Native Method)
at java.lang.reflect.Method.invoke(Method.java:515)
at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:779)
at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:595)
at dalvik.system.NativeStart.main(Native Method)
Caused by: java.lang.UnsatisfiedLinkError: dlopen failed: "/data/app-lib/IM.CRYPTO-1/libtestjni.so" has unexpected e_machine: 3
at java.lang.Runtime.loadLibrary(Runtime.java:364)
at java.lang.System.loadLibrary(System.java:526)
at org.abstractj.kalium.NaCl.<clinit>(NaCl.java:36)
at org.abstractj.kalium.keys.KeyPair.<init>(KeyPair.java:36)
at IM.CRYPTO.HelloAndroidActivity.buttonClick(HelloAndroidActivity.java:43)
at java.lang.reflect.Method.invokeNative(Native Method)
at java.lang.reflect.Method.invoke(Method.java:515)
at android.view.View$1.onClick(View.java:3818)
at android.view.View.performClick(View.java:4438)
at android.view.View$PerformClick.run(View.java:18422)
at android.os.Handler.handleCallback(Handler.java:733)
at android.os.Handler.dispatchMessage(Handler.java:95)
at android.os.Looper.loop(Looper.java:136)
at android.app.ActivityThread.main(ActivityThread.java:5017)
at java.lang.reflect.Method.invokeNative(Native Method)
at java.lang.reflect.Method.invoke(Method.java:515)
at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:779)
at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:595)
at dalvik.system.NativeStart.main(Native Method)
我花了好几个小时才到达这一点,但现在我无法从这里开始。似乎e_machine:3代表某种架构问题,但我无法弄清楚我哪里出错了。该应用程序是针对Android 4.0(Android API 14)编译的,并且仿真器配置为使用arm和相同的API 14(Nexus手机运行CM与Android 4.3.1)。如前所述,libsodium也是为手臂编制的。
更多信息:
<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd">
<modelVersion>4.0.0</modelVersion>
<groupId>IM.CRYPTO</groupId>
<artifactId>cryptolibAndroid</artifactId>
<version>1.0-SNAPSHOT</version>
<packaging>apk</packaging>
<name>cryptolibAndroid</name>
<properties>
<project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
<platform.version> 4.0.1.2
</platform.version>
<android.plugin.version>3.6.0</android.plugin.version>
</properties>
<dependencies>
<dependency>
<groupId>com.google.android</groupId>
<artifactId>android</artifactId>
<version>${platform.version}</version>
<scope>provided</scope>
</dependency>
<dependency>
<groupId>org.abstractj.kalium</groupId>
<artifactId>kalium-jni</artifactId>
<version>1.0.0-SNAPSHOT</version>
<scope>compile</scope>
</dependency>
</dependencies>
<build>
<finalName>${project.artifactId}</finalName>
<pluginManagement>
<plugins>
<plugin>
<groupId>com.jayway.maven.plugins.android.generation2</groupId>
<artifactId>android-maven-plugin</artifactId>
<version>${android.plugin.version}</version>
<extensions>true</extensions>
<configuration>
<undeployBeforeDeploy>true</undeployBeforeDeploy>
<attachJar>true</attachJar>
</configuration>
</plugin>
</plugins>
</pluginManagement>
<plugins>
<plugin>
<groupId>com.jayway.maven.plugins.android.generation2</groupId>
<artifactId>android-maven-plugin</artifactId>
<configuration>
<sdk>
<platform>14</platform>
</sdk>
</configuration>
</plugin>
</plugins>
</build>
(此类被调用以加载testjni(KaliumJNI))
package org.abstractj.kalium;
public class NaCl {
public static Sodium sodium() {
return SingletonHolder.SODIUM_INSTANCE;
}
private static final String LIBRARY_NAME = "sodium";
private static final class SingletonHolder {
public static final Sodium SODIUM_INSTANCE = new Sodium();
}
private NaCl() {
}
static {
System.loadLibrary("testjni");
}
}
我很感激有任何暗示让我指向正确的方向。
请注意我有一个小的JAVA桌面应用程序,它使用libsodium + Kalium(不是KaliumJNI)并且运行没有任何问题。
答案 0 :(得分:1)
正如Alex Cohn暗示的那样:库(.so-file)是针对Intel x86架构编译的,而您尝试运行应用程序的设备(或仿真器)则使用其他架构(假设为ARM)。因此,您发布的输出中的相关行是:
引起:java.lang.UnsatisfiedLinkError:dlopen失败:&#34; /data/app-lib/IM.CRYPTO-1/libtestjni.so"有意想不到的e_machine:3
你是如何编译图书馆的?通常,您下载Android NDK并使用一些提供的Makefile(名为Android.mk)来编译库。不要使用Linux-Distribution附带的编译器(称为主机编译器)来生成被认为在Android设备上运行的库。