Android 4.4.2 动态加入JNI库方法记录 (一 JNI库层)

欢迎转载,务必注明出处。http://blog.csdn.net/wang_shuai_ww/article/details/44456755

本篇是继《s5p4418 Android 4.4.2 驱动层 HAL层 服务层
应用层 开发流程记录
》之后的第二种加入JNI和服务的方法。

前面的方法是直接把HAL和服务层加入到了。Android的api中。这种方式优点是操作系统已开发完毕,剩下做APP的开发,那么我们仅仅须要一个classes.jar文件就可以使用我们自己Android系统的被隐藏的PI了(在Android官方的sdk中没有的API和用户自己加入的API),在执行自己Android系统的平台可直接方便的測试。当然也能够做出自己的android.jar文件放到sdk的platforms相应的android版本号中。替换之前的android.jar。那么以后选择该目标Android版本号时,载入的就是我们自己的Android系统的API了。能够像android.os.xxx这样调用我们的接口。

本篇介绍的是动态载入JNI的方式。顾名思义,就是在apk执行的时候去载入我们写好的.so JNI库。本方法灵活,开发移植使用也非常方便。也不用在源代码树中进入framework等文件夹进行各种繁琐的操作,仅仅须要写一个满足要求的.c文件,编译生成相应的.so文件即可了,并且对于存放的位置没有什么特别要求,我一般放在相应的板级文件夹下(/device/nexell/realarm)。

看起来比直接加入到Android的api中要方便多了。最起码简洁多了,没那么繁琐,仅仅是该方法在eclipse中须要多建立一个类文件,只是也没什么大不了的,O(∩_∩)O。对于想高速学习。加入自己的led操作的朋友来说。本篇的方法很合适。

对于动态JNI。首先引用 三篇博文,地址是http://blog.sina.com.cn/s/blog_4c451e0e0101339i.html、http://www.cnblogs.com/simonshi/archive/2011/01/25/1944910.html、http://blog.csdn.net/happy08god/article/details/11405607。

朋友们能够多读读,看看他们介绍的同样和不同之处。

本篇文章是在參考多篇博文之后自己编写代码在开发板上測试通过后记录写下的,大家能够參考代码写出自己的程序。然后对照与其它博文的不同之处。

我的源代码文件夹是/device/nexell/realarm/led2,在该文件夹下有两个文件led2.c和Android.mk。它们的源代码例如以下:

led2.c:

#include <stdio.h>  

#include "jni.h"
#include "JNIHelp.h"
#include <assert.h> 

// 引入log头文件
#include <android/log.h>
// log标签
#define  TAG    "Led_Load_JNI"
// 定义info信息
#define LOGI(...) __android_log_print(ANDROID_LOG_INFO, TAG, __VA_ARGS__)
// 定义debug信息
#define LOGD(...) __android_log_print(ANDROID_LOG_DEBUG, TAG, __VA_ARGS__)
// 定义error信息
#define LOGE(...) __android_log_print(ANDROID_LOG_ERROR, TAG, __VA_ARGS__)

<span style="color:#ff0000;">#define DEVICE_NAME		"/dev/real_led"

#define NO_REGISTER</span>

<span style="color:#ff0000;">#ifndef NO_REGISTER</span>
static jint Java_realarm_hardware_HardwareControl_LedSetState
  	(JNIEnv *env, jobject thiz,jint ledNum,jint ledState)
#else
JNIEXPORT jint JNICALL Java_realarm_hardware_HardwareControl_LedSetState
	(JNIEnv *env, jobject thiz,jint ledNum,jint ledState)
#endif
{
	int fd = open(DEVICE_NAME, 0);

	if (fd == -1)
	{
		LOGE("led open error");
		return 1;
	}

	if(ledState == 0)
		LOGD("Led close success");
	else if(ledState == 1)
		LOGD("Led open success");
	else {
		LOGD("Led ledState parameters ERROR.Only 0 or 1.");
		return 1;
	}
	ledState &= 0x01;
	ioctl(fd, ledState, 0);

	close(fd);
	return 0;
}
<span style="color:#ff0000;">#ifndef NO_REGISTER</span>
static JNINativeMethod gMethods[] = {
	{"LedSetState", "(II)I", (void *)Java_realarm_hardware_HardwareControl_LedSetState},
}; 

static int register_android_test_led(JNIEnv *env)
{
   	jclass clazz;
    static const char* const kClassName =  "realarm/hardware/HardwareControl";

    /* look up the class */
    clazz = (*env)->FindClass(env, kClassName);
    //clazz = env->FindClass(env,kClassBoa);
    if (clazz == NULL) {
        LOGE("Can't find class %s\n", kClassName);
        return -1;
    }

    /* register all the methods */
    if ((*env)->RegisterNatives(env,clazz, gMethods, sizeof(gMethods) / sizeof(gMethods[0])) != JNI_OK)
    //if (env->RegisterNatives(env,clazz, gMethods, sizeof(gMethods) / sizeof(gMethods[0])) != JNI_OK)
    {
        LOGE("Failed registering methods for %s\n", kClassName);
        return -1;
    }

    /* fill out the rest of the ID cache */
    return 0;
}
#endif
jint JNI_OnLoad(JavaVM* vm, void* reserved) {

<span style="color:#ff0000;">#ifndef NO_REGISTER</span>
	JNIEnv *env = NULL;
	if ((*vm)->GetEnv(vm,(void**) &env, JNI_VERSION_1_4) != JNI_OK) {
	//if (vm->GetEnv((void **)&env, JNI_VERSION_1_4) != JNI_OK) {
		LOGI("Error GetEnv\n");
		return -1;
	}
	assert(env != NULL);
	if (register_android_test_led(env) < 0) {
		printf("register_android_test_led error.\n");
		return -1;
	}
#endif
    /* success -- return valid version number */
	LOGI("/*****************realarm**********************/");

    return JNI_VERSION_1_4;
}

本代码在了解其原理后做了一些处理。本代码能够使用两个方式来终于引用LedSetState的C/C++实现,详细的控制,就是上面代码的红色部分的宏。

1.假设定义了上面的宏。那么JNI_OnLoad函数里面实用的就是最后一句话。return JNI_VERSION_1_4;返回JNI版本号。并没有对LedSetState的C/C++实现函数进行注冊。那么Android又怎么识别呢。

假设是这样的情况的话,Android会依据java层对JNI引用的类里面的native定义自己主动搜索相应的JNI方法。那么这就对函数定义的名字有要求了,格式为java_包名_类名_函数名字,Android的app在调用HardwareControl类里面的LedSetState方法时。就会自己主动匹配到相应的JNI方法。

这样的方法有个缺点,就是须要消耗CPU资源去匹配函数,导致执行效率不高。当程序大时就麻烦了。

2.因此推荐使用不定义上面红色的宏的方式。这种话,在调用这个库时,就会把LedSetState函数通过(*env)->RegisterNatives这个注冊,把它与Java_realarm_hardware_HardwareControl_LedSetState绑定在一起。而apk在调用LedSetState方法时运行效率就高多了,不用去自己主动匹配了,由于已经明确告诉操作系统该调用哪个函数了。

Android.mk:

LOCAL_PATH:= $(call my-dir)
include $(CLEAR_VARS)
LOCAL_MODULE_TAGS := optional
LOCAL_SRC_FILES := led2.c
LOCAL_SHARED_LIBRARIES := liblog
LOCAL_C_INCLUDES += $(JNI_H_INCLUDE)
LOCAL_LDLIBS:=-L$(SYSROOT)/usr/lib -llog
LOCAL_PRELINK_MODULE := false
LOCAL_MODULE_PATH := $(TARGET_OUT_SHARED_LIBRARIES)
LOCAL_MODULE := libLedJni

include $(BUILD_SHARED_LIBRARY)

从上面代码可知道,我编译生成的库名为libLedJni.so,注意LOCAL_MODULE_PATH要使用上图所看到的的路径,也就是/system/lib文件夹,在动态载入时。也是去这个文件夹下寻找。

好,到这里JNI库就完毕了,我们使用adb push命令把它发到开发板的/system/lib文件夹下。在下一篇将介绍怎么使用它。

时间: 2024-10-25 10:35:15

Android 4.4.2 动态加入JNI库方法记录 (一 JNI库层)的相关文章

Android 4.4.2 动态添加JNI库方法记录 (一 JNI库层)

欢迎转载,但必须注明出处.http://blog.csdn.net/wang_shuai_ww/article/details/44456755 本篇是继<s5p4418 Android 4.4.2 驱动层 HAL层 服务层 应用层 开发流程记录>之后的另外一种添加JNI和服务的方法. 前面的方法是直接把HAL和服务层添加到了,Android的api中,这样的方式好处是操作系统已开发完成,剩下做APP的开发,那么我们只需要一个classes.jar文件即可使用我们自己Android系统的被隐藏

Android 4.4.2 动态加入JNI库方法记录 (二 app应用层)

欢迎转载,务必注明出处:http://blog.csdn.net/wang_shuai_ww/article/details/44458553 源代码下载地址:http://download.csdn.net/detail/u010406724/8515377 本篇介绍怎么使用前面建立好的库文件. 要使用JNI库文件,那么首先我们是须要把它载入到系统中.并对其定义接口,供给应用来调用. 建立一个project,我的project名为RealArmTest.过程就省略了,完毕后再在src下建立一个

Android 4.4.2 动态添加JNI库方法记录 (二 app应用层)

欢迎转载,务必注明出处:http://blog.csdn.net/wang_shuai_ww/article/details/44458553 源码下载地址:http://download.csdn.net/detail/u010406724/8515377 本篇介绍怎么使用前面建立好的库文件. 要使用JNI库文件,那么首先我们是需要把它加载到系统中,并对其定义接口,供给应用来调用. 建立一个工程,我的工程名为RealArmTest,过程就省略了,完成后再在src下建立一个类,不继承其他类,包名

Android逆向之旅---动态方式破解apk进阶篇(IDA调试so源码)

一.前言 今天我们继续来看破解apk的相关知识,在前一篇:Eclipse动态调试smali源码破解apk 我们今天主要来看如何使用IDA来调试Android中的native源码,因为现在一些app,为了安全或者效率问题,会把一些重要的功能放到native层,那么这样一来,我们前篇说到的Eclipse调试smali源码就显得很无力了,因为核心的都在native层,Android中一般native层使用的是so库文件,所以我们这篇就来介绍如何调试so文件的内容,从而让我们破解成功率达到更高的一层.

【转载】cocos2dx 中 Android NDK 加载动态库的问题

原文地址:http://blog.csdn.net/sozell/article/details/10551309 cocos2dx 中 Android NDK 加载动态库的问题 闲聊 最近在接入各个平台的SDK,遇到了不少问题,也从中了解了不少知识,之前一直觉得没啥好写的,毕竟做了4个月的游戏开发,也没有碰上什么真正的大问题,cocos2dx的引擎包得也很好,能让人把大部分时间都关注在游戏逻辑.效果的处理上,当然,之前的libevent还是小坑一下,但是和后来遇到的相比,也算不上什么了. 我最

Android jni/ndk编程二:jni数据类型转换(primitive,String,array)

一.数据类型映射概述 从我们开始jni编程起,就不可能避开函数的参数与返回值的问题.java语言的数据类型和c/c++有很多不同的地方,所以我们必须考虑当在java层调用c/c++函数时,怎么正确的把java的参数传给c/c++函数,怎么正确的从c/c++函数获取正确的函数返回值:反之,当我们在c/c++中使用java的方法或属性时,如何确保数据类型能正确的在java和c/c++之间转换. 回顾我们上一篇文章中的那个c函数: #include <stdio.h> #include <jn

Android 插件化之动态加载jar

有时候会看到一些应用对应的SDcard里的文件夹里有 ***.jar 等文件,现在明白这些文件大概是用来做应用内自动更新用的. 打比方说,类似eclipse 可以通过预留接口,安装各种插件一样. Android 也可以通过动态加载jar 来实现类似的业务代码更新:(这里所说的jar要通过dx工具来转化成Dalvik byte code,下文会讲到) 注意:首先需要了解一点:在Android中可以动态加载,但无法像Java中那样方便动态加载jar 原因:Dalvik虚拟机如同其他Java虚拟机一样

Android JNI使用方法

经过几天的努力终于搞定了android JNI部分,下面将我的这个小程序和大家分享一下.android JNI是连接android Java部分和C/C++部分的纽带,完整使用JNI需要Java代码和C/C++代码.其中C/C++代码用于生成库文件,Java代码用于引用C /C++库文件以及调用C/C++方法. android Java部分代码: jnitest.java package com.hello.jnitest; import android.app.Activity; import

android 在布局中动态添加控件

第一步 Java代码 final LayoutInflater inflater = LayoutInflater.from(this); 第二步:获取需要被添加控件的布局 Java代码 final LinearLayout lin = (LinearLayout) findViewById(R.id.LinearLayout01); 第三步:获取需要添加的布局(控件) Java代码 LinearLayout layout = (LinearLayout) inflater.inflate( R