Tiny4412 led之build JNI实现

Posted 冷冻的彩虹

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Tiny4412 led之build JNI实现相关的知识,希望对你有一定的参考价值。

PC机平台:ubuntu 12.04.5

硬件平台:Tiny4412标准版+android5.0


上一篇:Tiny4412 led之NDK JNI实现,介绍的是通过NDK工具进行JNI开发,*.so文件是通过NDK工具进行编译生成的,这次就介绍如何通过arm-linux-gcc编译生成对应的*.so文件,深入了解JNI底层的开发流程;


驱动代码、JNI接口、android应用程序都是Tiny4412 led之NDK JNI实现的代码;需要重新编写的只有*.c文件;


新建tiny4412-leds.c

#include <jni.h>  
#include <stdio.h>
#include <android/log.h> 

#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <linux/ioctl.h>

#define DEVICE_NAME		"/dev/tiny4412-leds"

#define LED_0		1
#define LED_1		2
#define LED_2		3
#define LED_3		4

//幻数
#define LEDS_MAGIC	'a'
#define LEDS_PGM4_0_ON		_IO(LEDS_MAGIC, 1)
#define LEDS_PGM4_0_OFF	_IO(LEDS_MAGIC, 2)
#define LEDS_PGM4_1_ON		_IO(LEDS_MAGIC, 3)
#define LEDS_PGM4_1_OFF	_IO(LEDS_MAGIC, 4)
#define LEDS_PGM4_2_ON		_IO(LEDS_MAGIC, 5)
#define LEDS_PGM4_2_OFF	_IO(LEDS_MAGIC, 6)
#define LEDS_PGM4_3_ON		_IO(LEDS_MAGIC, 7)
#define LEDS_PGM4_3_OFF	_IO(LEDS_MAGIC, 8)

jint JNICALL leds_operation(JNIEnv *env, jobject cls, jint ledsNum, jboolean status)

	int leds_fd = 0;

	leds_fd = open(DEVICE_NAME, O_RDWR);
	if (leds_fd == -1) 
		__android_log_print(ANDROID_LOG_DEBUG, "tiny4412-leds", "leds_operation fail!");
		return 1;
	
	
	__android_log_print(ANDROID_LOG_DEBUG, "tiny4412-leds", "leds_operation ledNum:%d,status:%d", ledsNum, status);

	switch (ledsNum) 
	case LED_0:
		if (status)
			ioctl(leds_fd, LEDS_PGM4_0_ON);
		else
			ioctl(leds_fd, LEDS_PGM4_0_OFF);
		break;
	case LED_1:
		if (status)
			ioctl(leds_fd, LEDS_PGM4_1_ON);
		else
			ioctl(leds_fd, LEDS_PGM4_1_OFF);
		break;
	case LED_2:
		if (status)
			ioctl(leds_fd, LEDS_PGM4_2_ON);
		else
			ioctl(leds_fd, LEDS_PGM4_2_OFF);
		break;
	case LED_3:
		if (status)
			ioctl(leds_fd, LEDS_PGM4_3_ON);
		else
			ioctl(leds_fd, LEDS_PGM4_3_OFF);
		break;
	defautl :
		break;
	

	close(leds_fd);

	return 0;


static const JNINativeMethod methods[] =    //本地方法列表
	"ledsOperation", "(IZ)I", (void *)leds_operation,  //"ledsOperation"为tiny4412Leds.java中的jni接口
;

/* System.loadLibrary */
JNIEXPORT jint JNICALL
JNI_OnLoad(JavaVM *jvm, void *reserved)   

	JNIEnv *env;
	jclass cls;

	if ((*jvm)->GetEnv(jvm, (void **)&env, JNI_VERSION_1_4)) 
		return JNI_ERR; /* JNI version not supported */
	
	cls = (*env)->FindClass(env, "com/example/Tiny4412_leds/tiny4412Leds");  //包名+类名,将‘.’换成‘/’
	if (cls == NULL) 
		return JNI_ERR;
	

	if ((*env)->RegisterNatives(env, cls, methods, 1) < 0)  //注册本地方法
		return JNI_ERR;

	return JNI_VERSION_1_4;

JNI的C代码不算特别复杂,就是简单的将本地方法注册java虚拟机中;当java调用System.loadLibrary("tiny4412-leds");时,就会调用JNI_OnLoad方法,通过JNI_OnLoad方法注册tiny4412Leds类的本地方法到java虚拟机中,就会虚拟机中为tiny4412Leds类中添加本地接口,之后应用程序才能正常使用tiny4412Leds类的本地方法;


编译libtiny4412-leds.so

在ubuntu下将tiny4412-leds.c编译成libtiny4412-leds.so

root@pc:/home/workplace/Tiny4412# arm-linux-gcc -I /usr/lib/jvm/java-1.7.0-openjdk-amd64/include/-I ./android-5.0.2/prebuilts/ndk/8/platforms/android-14/arch-arm/usr/include/ -fPIC -shared -o libtiny4412-leds.so tiny4412-leds.c -nostdlib android-5.0.2/prebuilts/ndk/8/platforms/android-14/arch-arm/usr/lib/libc.so   android-5.0.2/prebuilts/ndk/8/platforms/android-14/arch-arm/usr/lib/liblog.so

选项:-fPIC

-fPIC 作用于编译阶段,告诉编译器产生与位置无关代码(Position-Independent Code), 则产生的代码中,没有绝对地址,全部使用相对地址,故而代码可以被加载器加载到内存的任意位置,都可以正确的执行。这正是共享库所要求的,共享库被加载时,在内存的位置不是固定的。

选项:-nostdlib

-nostdlib作用于编译阶段,告诉编译器不使用标准的lib库;


    如果不加上 -I /usr/lib/jvm/java-1.7.0-openjdk-amd64/include/  就会出现找不到jni.h的错误提示;

    如果不加上 -I ./android-5.0.2/prebuilts/ndk/8/platforms/android-14/arch-arm/usr/include/ 就会出现找不到log.h的错误提示;

    如果不加上 android-5.0.2/prebuilts/ndk/8/platforms/android-14/arch-arm/usr/lib/libc.so 加载libtiny4412-leds.so 就会出现;dlopen("/data/app/com.example.Tiny4412_leds-2/lib/arm/libtiny4412-leds.so", RTLD_LAZY) failed: dlopen failed: could not load library "libc.so.6" needed by "libtiny4412-leds.so"; caused by library "libc.so.6" not found 的错误提示;系统中默认没有libc.so.6库,所以可以直接指定使用libc.so进行代替;

    如果不加上 android-5.0.2/prebuilts/ndk/8/platforms/android-14/arch-arm/usr/lib/liblog.so 就会出现java.lang.UnsatisfiedLinkError: dlopen failed: cannot locate symbol "__android_log_print" referenced by "libtiny4412-leds.so"... 的错误提示;使用打印接口需要使用到liblog.so否则就会出错,该liblog.so可以在android源码中查找得到;


编译完成后在libs目录下创建armeabi目录,并将libtiny4412-leds.so拷贝到armeabi目录下,然后就可以运行程序




以上是关于Tiny4412 led之build JNI实现的主要内容,如果未能解决你的问题,请参考以下文章

Tiny4412 led之JNI实现

Tiny4412 led之NDK JNI实现

Tiny4412 led之NDK JNI实现

基于TINY4412的Andorid开发-------简单的LED灯控制

(exynos4412)Tiny4412裸机开发-点亮LED灯

(exynos4412)Tiny4412裸机开发-按键检测