Posted on 2010-09-16 15:46 zhangweia 閱讀(2797) 評論(1) 編輯 收藏 html
NDK動態庫的調用分兩種狀況。第一種是:so庫和調用程序在同一個目錄和源程序 經過同一個mk文件來編譯;另一種狀況是so是外部已經編譯好了的,調用程序加載調用。下面咱們就來分析下面2種so的調用狀況:java
第一部分: 內部so調用node
A. 先看下工程的目錄:linux
test01.java :Java文件android
test01.c 主so,生成libtest.so,在libtest.so中調用libtutorial.so中的方法express
tutorial01.c tutorial02.c tutorial02.h :生成libtutorial.soapache
B. 而後看下具體的mk文件的寫法以及java中的文件內容windows
test01.java/* * Copyright (C) 2009 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ package com.example.test;import android.R.integer;import android.app.Activity;import android.widget.TextView;import android.os.Bundle;public class test01 extends Activity {/** Called when the activity is first created. */ @Overridepublic void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);/* Create a TextView and set its content. * the text is retrieved by calling a native * function. */ TextView tv = new TextView(this); int a = getinformation(); String lls = a +""; tv.setText(lls); setContentView(tv); }/* A native method that is implemented by the * 'hello-jni' native library, which is packaged * with this application. */ public native int getinformation(); /* this is used to load the 'hello-jni' library on application * startup. The library has already been unpacked into * /data/data/com.example.HelloJni/lib/libhello-jni.so at * installation time by the package manager. */ static { System.loadLibrary("test"); } }
Android.mk LOCAL_PATH := $(call my-dir) #獲取當前目錄 include $(CLEAR_VARS) #清除一些變量 LOCAL_MODULE := tutorial #要生成的庫名 LOCAL_SRC_FILES := tutorial01.c tutorial02.c #庫對應的源文件 include $(BUILD_SHARED_LIBRARY) #生成動態庫libtutorial.so include $(CLEAR_VARS) #清除一些變量 LOCAL_MODULE := test #定義另一個庫的名 LOCAL_SRC_FILES := test01.c #定義庫對應的源文件 LOCAL_LDLIBS := -ldl -llog #libtest.so須要引用的庫libdl.so:加載動態函數須要,liblog.so 日誌打印須要,默認是system/lib目錄下 include $(BUILD_SHARED_LIBRARY) #生成共享庫
test01.c #include <string.h> #include <jni.h> #include <dlfcn.h> #include <android/log.h> #include <stdlib.h> #define LOG_TAG "libgl2jni" #define LOGI(...) __android_log_print(ANDROID_LOG_INFO,LOG_TAG,__VA_ARGS__) #define LOGE(...) __android_log_print(ANDROID_LOG_ERROR,LOG_TAG,__VA_ARGS__) //extern int getinformation(); jint Java_com_example_test_test01_getinformation(JNIEnv* env,jobject thiz) { //getinformation(); ??thr .so file will load to the sdcard with the folder data/data/com.example.test/lib/ void* filehandle = dlopen("/data/data/com.example.test/lib/libtutorial.so", RTLD_LAZY ); int ll = -1; if(filehandle) { LOGI("open so success!"); int( * getinformation ) (); getinformation = dlsym(filehandle, "getinformation"); if( getinformation ) { LOGI("call function getinformation OK!"); ll = getinformation(); } else { ll = -3; LOGE("call function getinformation! ERROR!"); } LOGI("return value=%d",ll); dlclose(filehandle); filehandle=0; } else { ll = -2; LOGE("open so ERROR!"); }return ll; }
tutorial01.c #include <stdio.h> #include "tutorial02.h"int getinformation() { int c = getinformation2(191,81); return c; }
tutorial02.c #include <stdio.h> #include "tutorial02.h"int getinformation2(int i,int j) { return (i+j); }
tutorial02.h int getinformation2(int,int);
C: 編譯:app
進入工程目錄,執行 $NDK/ndk-buildless
輸出結果:
$NDK/ndk-build /cygdrive/e/cygwin/android-ndk-r4 Compile thumb : test <= /cygdrive/e/cygwin/android-ndk-r4/samples/testappso/jni/test01.c SharedLibrary : libtest.so Install : libtest.so => /cygdrive/e/cygwin/android-ndk-r4/samples/testappso/libs/armeabi Compile thumb : tutorial <= /cygdrive/e/cygwin/android-ndk-r4/samples/testappso/jni/tutorial01.c Compile thumb : tutorial <= /cygdrive/e/cygwin/android-ndk-r4/samples/testappso/jni/tutorial02.c SharedLibrary : libtutorial.so Install : libtutorial.so => /cygdrive/e/cygwin/android-ndk-r4/samples/testappso/libs/armeabi
D:在從新刷新工程,你就能夠在lib下面看到生成的so了,點擊加載運行
第二部分:調用經過本身編寫make方式生成的so
咱們仍是引用上面的工程,不過把libtutorial.so 拿出來,經過本身寫makefile來編譯成so,這裏請你們注意了,調用外部的so時候,不可以直接用在linux下生產的so來調用,必須經過ndk提供的gcc來編譯,不然會由於平臺的不一樣沒法調用。
A. 廢話很少說,看makefile怎麼寫!個人ndk是1.6版本的.
新建一個文件夾,把 tutorial01.c tutorial02.c tutorial02.h 文件拷貝過去,而後編寫makefile文件
libtutorial.so Makefile CC = /cygdrive/e/cygwin/android-ndk-r4/build/prebuilt/windows/arm-eabi-4.4.0/bin/arm-eabi-gcc-4.4.0 CFLAGS = -g -O2 -fPIC -I/cygdrive/e/cygwin/android-ndk-r4/build/platforms/android-5/arch-arm/usr/include SDFLAGS = -nostdlib -Wl,-T,armelf.xsc -Wl,-shared,-Bsymbolic -Wl,-soname,$@ -lc CRT_OBJS= -L/cygdrive/e/cygwin/android-ndk-r4/build/platforms/android-5/arch-arm/usr/lib -lz -lm -ldl # source files: SRCS= tutorial01.c tutorial02.c tutorial02.h all: libtutorial.so libtutorial.so: tutorial01.o tutorial02.o $(CC) $(SDFLAGS) -o $@ tutorial01.o tutorial02.o $(CRT_OBJS) tutorial01.o: tutorial02.h tutorial02.o: tutorial02.h clean: rm -f libtutorial.so *.o
在執行make的時候須要注意一下幾點:
1. 指定程序須要的頭文件目錄 經過-I指定/cygdrive/e/cygwin/android-ndk-r4/build/platforms/android-5/arch-arm/usr/include NDK提供的一些基本c語言庫的頭文件
2. 指定庫路徑-L/cygdrive/e/cygwin/android-ndk-r4/build/platforms/android-5/arch-arm/usr/lib 裏面有libz.so,libm.so,libdl.so等庫的路徑
3. 拷貝交叉編譯文件armelf.xsc到android-ndk-r4/build/prebuilt/windows/arm-eabi-4.4.0/lib/gcc/arm-eabi/4.4.0下,armelf.xsc在目錄./build/prebuilt/linux-x86/arm-eabi-4.4.0/arm-eabi/lib/ldscripts下
4. 拷貝libc.so到./build/prebuilt/linux-x86/arm-eabi-4.4.0/lib/gcc/arm-eabi/4.4.0目錄下,libc.so位於./build/platforms/android-5/arch-arm/usr/lib庫
完成上面的工做你就能夠放心make了,成功後生成libtutorial.so文件
libtest.so Makefile CC = /cygdrive/e/cygwin/android-ndk-r4/build/prebuilt/windows/arm-eabi-4.4.0/bin/arm-eabi-gcc-4.4.0 CFLAGS = -g -O2 -fPIC -I/cygdrive/e/cygwin/android-ndk-r4/build/platforms/android-5/arch-arm/usr/include SDFLAGS = -nostdlib -Wl,-T,armelf.xsc -Wl,-shared,-Bsymbolic -Wl,-soname,$@ -lc CRT_OBJS= -L/cygdrive/e/cygwin/android-ndk-r4/build/platforms/android-5/arch-arm/usr/lib -lz -lm -ldl # source files: SRCS= test01.c all: libtest.so libtest.so: test01.o $(CC) $(SDFLAGS) -o $@ test01.o $(CRT_OBJS) test01.o: test01.c clean: rm -f libtest.so *.o
B. 生成so文件後,在工程目錄下創建目錄libs/armeabi,而後把so拷貝進去,在打開工程,刷新加載,一切搞定
C. 一些小技巧:
a. 查看so是否是編譯成ARM模式下的so
$ file libtest.so
libtest.so: ELF 32-bit LSB shared object, ARM, version 1 (SYSV), dynamically linked, not stripped
b. 若是別人提供了你一個so,查看提供了那些方法,更詳細的用法,查看nm命令
$ nm libtutorial.so |grep T
00001344 a _GLOBAL_OFFSET_TABLE_
000002a8 T getinformation
000002b4 T getinformation2
D. 下面提供一個大工程的編譯的makefile,吧opencore --omx下的testapp獨立編譯成so,編譯經過,未加載庫,給你們參考。
testapp Makefile cc=/home/zhangweia/android/android-ndk-r4b/build/prebuilt/linux-x86/arm-eabi-4.4.0/bin/arm-eabi-gcc-4.4.0 CFLAGS = -g -O2 -fPIC -DANDROID -I./include -I/home/zhangweia/android/android-ndk-r4b/build/platforms/android-5/arch-arm/usr/include CFLAGS += -I$(SDK)/out/target/product/generic/obj/include/libpv \ -I$(opencore)/codecs_v2/common/include \ -I$(opencore)/pvmi/pvmf/include \ -I$(opencore)/nodes/common/include \ -I$(opencore)/extern_libs_v2/khronos/openmax/include \ -I$(opencore)/build_config/opencore_dynamic \ -I$(opencore)/android/drm/oma1/src \ -I$(opencore)/engines/author/include \ -I$(opencore)/oscl/oscl/config/shared \ -I$(opencore)/oscl/oscl/config/android \ -I$(opencore)/engines/common/include \ -I$(opencore)/extern_libs_v2/khronos/openmax/include \ -I$(opencore)/android \ #SDFLAGS = -nostdlib -Wl,-T,armelf.xsc -Wl,-shared,-Bsymbolic -Wl,-soname,$@ -lc SDFLAGS = -nostdlib -Wl,-T,armelf.xsc -Wl,-Bsymbolic -Wl,-soname,$@ -lc CRT_OBJS=-L/home/zhangweia/android/android-ndk-r4b/build/platforms/android-5/arch-arm/usr/lib -lz -lm -ldl #all : libomxde.so OBJECT= omx_threadsafe_callbacks.o \ omxdectest.o \ omxdectestbase.o \ omxtest_buffer_busy.o \ omxtest_corrupt_nal.o \ omxtest_dynamic_reconfig.o \ omxtest_eos_missing.o \ omxtest_extra_partialframes.o \ omxtest_flush_eos.o \ omxtest_flush_port.o \ omxtest_get_role.o \ omxtest_incomplete_nal.o \ omxtest_missing_nal.o \ omxtest_multiple_instance.o \ omxtest_param_negotiation.o \ omxtest_partialframes.o \ omxtest_pause_resume.o \ omxtest_portreconfig_transit_1.o \ omxtest_portreconfig_transit_2.o \ omxtest_portreconfig_transit_3.o \ omxtest_reposition.o \ omxtest_usebuffer.o \ omxtest_without_marker.o #libomxde.so : $(OBJECT) omxde : $(OBJECT) $(cc) $(SDFLAGS) -o $@ $(CRT_OBJS) omx_threadsafe_callbacks.o:src/omx_threadsafe_callbacks.cpp $(cc) $(CFLAGS) $(SDFLAGS) -c src/omx_threadsafe_callbacks.cpp omxdectest.o:src/omxdectest.cpp $(cc) $(CFLAGS) $(SDFLAGS) -c src/omxdectest.cpp omxdectestbase.o:src/omxdectestbase.cpp $(cc) $(CFLAGS) $(SDFLAGS) -c src/omxdectestbase.cpp omxtest_corrupt_nal.o:src/omxtest_corrupt_nal.cpp $(cc) $(CFLAGS) $(SDFLAGS) -c src/omxtest_corrupt_nal.cpp omxtest_buffer_busy.o:src/omxtest_buffer_busy.cpp $(cc) $(CFLAGS) $(SDFLAGS) -c src/omxtest_buffer_busy.cpp omxtest_dynamic_reconfig.o:src/omxtest_dynamic_reconfig.cpp $(cc) $(CFLAGS) $(SDFLAGS) -c src/omxtest_dynamic_reconfig.cpp omxtest_eos_missing.o:src/omxtest_eos_missing.cpp $(cc) $(CFLAGS) $(SDFLAGS) -c src/omxtest_eos_missing.cpp omxtest_extra_partialframes.o:src/omxtest_extra_partialframes.cpp $(cc) $(CFLAGS) $(SDFLAGS) -c src/omxtest_extra_partialframes.cpp omxtest_flush_eos.o:src/omxtest_flush_eos.cpp $(cc) $(CFLAGS) $(SDFLAGS) -c src/omxtest_flush_eos.cpp omxtest_flush_port.o:src/omxtest_flush_port.cpp $(cc) $(CFLAGS) $(SDFLAGS) -c src/omxtest_flush_port.cpp omxtest_get_role.o:src/omxtest_get_role.cpp $(cc) $(CFLAGS) $(SDFLAGS) -c src/omxtest_get_role.cpp omxtest_incomplete_nal.o:src/omxtest_incomplete_nal.cpp $(cc) $(CFLAGS) $(SDFLAGS) -c src/omxtest_incomplete_nal.cpp omxtest_missing_nal.o:src/omxtest_missing_nal.cpp $(cc) $(CFLAGS) $(SDFLAGS) -c src/omxtest_missing_nal.cpp omxtest_multiple_instance.o:src/omxtest_multiple_instance.cpp $(cc) $(CFLAGS) $(SDFLAGS) -c src/omxtest_multiple_instance.cpp omxtest_param_negotiation.o:src/omxtest_param_negotiation.cpp $(cc) $(CFLAGS) $(SDFLAGS) -c src/omxtest_param_negotiation.cpp omxtest_partialframes.o:src/omxtest_partialframes.cpp $(cc) $(CFLAGS) $(SDFLAGS) -c src/omxtest_partialframes.cpp omxtest_pause_resume.o:src/omxtest_pause_resume.cpp $(cc) $(CFLAGS) $(SDFLAGS) -c src/omxtest_pause_resume.cpp omxtest_portreconfig_transit_1.o:src/omxtest_portreconfig_transit_1.cpp $(cc) $(CFLAGS) $(SDFLAGS) -c src/omxtest_portreconfig_transit_1.cpp omxtest_portreconfig_transit_2.o:src/omxtest_portreconfig_transit_2.cpp $(cc) $(CFLAGS) $(SDFLAGS) -c src/omxtest_portreconfig_transit_1.cpp omxtest_portreconfig_transit_3.o:src/omxtest_portreconfig_transit_3.cpp $(cc) $(CFLAGS) $(SDFLAGS) -c src/omxtest_portreconfig_transit_1.cpp omxtest_reposition.o:src/omxtest_reposition.cpp $(cc) $(CFLAGS) $(SDFLAGS) -c src/omxtest_reposition.cpp omxtest_usebuffer.o:src/omxtest_usebuffer.cpp $(cc) $(CFLAGS) $(SDFLAGS) -c src/omxtest_usebuffer.cpp omxtest_without_marker.o:src/omxtest_without_marker.cpp $(cc) $(CFLAGS) $(SDFLAGS) -c src/omxtest_without_marker.cpp clean: rm -rf *.o