Запуск собственного кода Android вызывает java.lang.UnsatisfiedLinkError

У меня есть приложение для Android с собственным кодом, в котором я использую библиотеку openCV. При запуске приложения на телефоне все работает хорошо, но когда я пытаюсь запустить его на планшете (Lenovo Yoga 2 pro), я сталкиваюсь с этой ошибкой:

01-03 11:53:13.007: E/AndroidRuntime(25632): FATAL EXCEPTION: main
01-03 11:53:13.007: E/AndroidRuntime(25632): Process: <appname>, PID: 25632
01-03 11:53:13.007: E/AndroidRuntime(25632): java.lang.UnsatisfiedLinkError: dlopen failed: "/data/app-lib/<appname>/libnative_module.so" has unexpected e_machine: 40
01-03 11:53:13.007: E/AndroidRuntime(25632):    at java.lang.Runtime.loadLibrary(Runtime.java:364)
01-03 11:53:13.007: E/AndroidRuntime(25632):    at java.lang.System.loadLibrary(System.java:526)
01-03 11:53:13.007: E/AndroidRuntime(25632):    at <appname>.SplashActivity$1.onManagerConnected(SplashActivity.java:35)
01-03 11:53:13.007: E/AndroidRuntime(25632):    at org.opencv.android.AsyncServiceHelper$1.onServiceConnected(AsyncServiceHelper.java:318)
01-03 11:53:13.007: E/AndroidRuntime(25632):    at android.app.LoadedApk$ServiceDispatcher.doConnected(LoadedApk.java:1127)
01-03 11:53:13.007: E/AndroidRuntime(25632):    at android.app.LoadedApk$ServiceDispatcher$RunConnection.run(LoadedApk.java:1144)
01-03 11:53:13.007: E/AndroidRuntime(25632):    at android.os.Handler.handleCallback(Handler.java:733)
01-03 11:53:13.007: E/AndroidRuntime(25632):    at android.os.Handler.dispatchMessage(Handler.java:95)
01-03 11:53:13.007: E/AndroidRuntime(25632):    at android.os.Looper.loop(Looper.java:149)
01-03 11:53:13.007: E/AndroidRuntime(25632):    at android.app.ActivityThread.main(ActivityThread.java:5283)
01-03 11:53:13.007: E/AndroidRuntime(25632):    at java.lang.reflect.Method.invokeNative(Native Method)
01-03 11:53:13.007: E/AndroidRuntime(25632):    at java.lang.reflect.Method.invoke(Method.java:515)
01-03 11:53:13.007: E/AndroidRuntime(25632):    at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:825)
01-03 11:53:13.007: E/AndroidRuntime(25632):    at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:641)
01-03 11:53:13.007: E/AndroidRuntime(25632):    at dalvik.system.NativeStart.main(Native Method)

это в строке 45 в SplashActivity:

System.loadLibrary("native_module");

Может ли кто-нибудь помочь мне решить эту ошибку? или что я делаю не так?


person mark    schedule 03.01.2015    source источник
comment
См. sco.com/developers/gabi/2000-07. -17/ch4.eheader.html: e_machine=40 означает EM_ARM.   -  person Alex Cohn    schedule 03.01.2015


Ответы (2)


Ваш планшет основан на процессоре Intel Atom. Вы должны создать свой native_module для APP_ABI=x86. Обратите внимание, что вам также нужна совместимая версия OpenCV.

Обновление: в данном конкретном случае приложение было создано как для x86, так и для ARM. Но установщик приложения выбрал неправильную подпапку, возможно, потому, что имена файлов, перечисленные в libs/x86 и libs/armeabi, не совсем соответствует. Надежный и эффективный подход для приложений с нетривиальным нативным компонентом — загружать отдельные APK в PlayStore.

person Alex Cohn    schedule 03.01.2015
comment
Я подумал то же самое и проверил APP_ABI, но он настроен так: APP_ABI := armeabi armeabi-v7a x86 mips, так что все должно быть в порядке. Как вы имеете в виду совместимую версию OpenCV? потому что, когда я загружал openCV для Android, не было возможности выбрать тип процессора - person mark; 03.01.2015
comment
Вы используете общую или статическую версию OpenCV? - person Alex Cohn; 03.01.2015
comment
В моем проекте у меня есть каталоги libs, где находятся подпапки с библиотеками .so и .a для разных типов процессоров (эти подпапки взяты из загруженного проекта opencv для Android в структуре папок sdk/native/libs). И затем я связал проект android opencv в eclipse (sdk/java). - person mark; 03.01.2015
comment
Вы можете проверить содержимое папки libs/ вашего проекта Eclipse. В перспективе C/C++ вы можете увидеть вкладку двоичных файлов, и на ней будет список so файлов, которые у вас есть, помеченных для архитектуры (x86/arm/независимо). Но вы также должны проверить, какие файлы на самом деле установлены на устройстве. - person Alex Cohn; 04.01.2015
comment
Как я могу проверить, какие файлы установлены на устройстве? Но для запуска приложения мне пришлось загрузить приложение opencv manager, и это приложение работает (не падает), так что это немного странно. - person mark; 05.01.2015
comment
Даже если ваше устройство не рутировано, вы можете adb pull /data/app-lib/<appname>/libnative_module.so. Вы даже можете adb shell ls -l /data/app-lib/<appname>/ узнать, какие родные библиотеки установлены на устройстве. Сравните результаты на телефоне (на руке) и на планшете (x86). - person Alex Cohn; 05.01.2015
comment
извините за поздний ответ, но я пробовал это с adb, но это не сработало, я не мог получить к нему доступ. Когда я попытался перечислить его с помощью ls-l, мне было отказано в доступе. Возможно, мне нужно рутировать его, но я попробовал его на смартфоне Intel, и он работал хорошо. Вы знаете, почему это было так? или Возможно ли, что некоторые библиотеки .so не устанавливаются/перемещаются на планшет во время развертывания приложения? - person mark; 14.01.2015
comment
Если вы не смогли вытащить файл, скорее всего его просто нет. Но вы также можете попробовать adb shell ls -l /data/app-lib/<appname>/libnative_module.so или adb shell ls -l /data/data/<appname>/lib/libnative_module.so. Убедитесь, что вы не ошиблись в написании ‹appname›, потому что /data/app-lib/ не является общедоступным. Чтобы проверить ‹appname›, вы можете использовать adb shell pm list packages. - person Alex Cohn; 14.01.2015
comment
Я действительно извиняюсь за поздний ответ, но у меня не было много времени для этого. Я перечислил libnative_module.so, и он был там, затем я посмотрел, какие еще файлы .so находятся в папке lib. Там было все, кроме файлов .a, но дело в том, что имена файлов .so соответствовали подпапке armeabi в папке libs в проекте Eclipse, а не x86. Есть ли способ получить эти файлы x86 на устройстве, когда устройство Intel? - person mark; 24.02.2015
comment
например: в папке armeabi находится libnative_camera_r2.2.0.so, а в x86 — libnative_camera_r2.3.3.so, и когда я использую adb для отображения папки lib, я вижу libnative_camera_r2.2.0.so и другие файлы из armeabi, но не из папки x86. - person mark; 24.02.2015
comment
Я бы посоветовал попробовать только пакетные библиотеки x86. Будет ли приложение установлено и запущено на вашем целевом устройстве? Была проведена плодотворная дискуссия о том, как можно разделить APK на отдельные меньшие APK целевой архитектуры. - person Alex Cohn; 24.02.2015
comment
Спасибо, я попробовал это, но по-другому, и я использовал только APP_ABI=x86 вместо указания других типов, и это сработало. Я действительно не понимаю, почему, если я указываю там и другие типы архитектур, он выбирает руку вместо x86, несмотря на то, что это планшет Intel. Кстати, я нашел действительно хорошее приложение, в котором перечислены все приложения и показано, какие библиотеки ndk и какого типа имеет каждое установленное приложение ( play.google.com/store/apps/) Таким образом, ваш первоначальный ответ правильный, но вы можете изменить его, что иногда, несмотря на то, что он указан также с помощью другое у него есть - person mark; 03.03.2015
comment
Установщик приложения ожидает, что содержимое libs/<ABI> папок будет иметь одинаковые имена файлов, поэтому вы попадаете в неизведанные воды с _r2.2.0 против _r2.3.3. Проблема с Atom заключается в том, что слишком часто приложения не имеют его встроенной поддержки; поэтому у поставщика нет другого выбора, кроме как добавить урезанную эмуляцию ARM. В вашем конкретном случае это выстрелило в ответ. Я считаю, что если вы строите только для APP_ABI=armeabi-v7a, ваше приложение будет работать (но намного медленнее). - person Alex Cohn; 03.03.2015

У меня есть Asus Zenfone 5 (который использует x86 ABI), и у меня была та же проблема. По-видимому, проблема была с версиями OpenCV 2.x. Начните использовать версию OpenCV 3.x, и проблема будет решена.

Версия OpenCV 3.x имеет больше параметров ABI, например:

arm64v8a
arm
armv7a
mips
mips64
x86
x86-64

Вы можете найти версию 3.x здесь .

Надеюсь, это помогло!

person Geraldo Neto    schedule 08.01.2016