Android应用程序通过JNI调用驱动程序(友善Smart210)

实现目标:

     写一个简单的测试smart210上LED的应用程序,应用程序通过JNI调用Android系统下的Linux内核中的LED的驱动程序,实现在应用程序上控制开发板上4个LED的目的。

开发环境:

     win7 32位的系统;

      开发板:友善Smart210(s5pv210);

      Android版本:Android-4.0.3;

      Linux内核版本:Linux-3.0.8

环境搭建以及完成JNI部分:

     这里假设你的电脑上已经装好了开发应用程序的环境,其中Android sdk的下载地址为http://developer.android.com/sdk/index.html   根据自己电脑的系统来下载适合自己的sdk吧。

    ndk的简介:(参考网络博客)

    NDK全称:Native Development Kit。 

1、NDK是一系列工具的集合。 
        NDK提供了一系列的工具,帮助开发者快速开发C(或C++)的动态库,并能自动将so和java应用一起打包成apk。这些工具对开发者的帮助是巨大的。 
        NDK集成了交叉编译器,并提供了相应的mk文件隔离CPU、平台、ABI等差异,开发人员只需要简单修改mk文件(指出“哪些文件需要编译”、“编   译特性要求”等),就可以创建出so。 
        NDK可以自动地将so和Java应用一起打包,极大地减轻了开发人员的打包工作。 

 2、NDK提供了一份稳定、功能有限的API头文件声明。 
       Google明确声明该API是稳定的,在后续所有版本中都稳定支持当前发布的API。从该版本的NDK中看出,这些API支持的功能非常有限,包含有:C标准库(libc)、标准数学库(libm)、压缩库(libz)、Log库(liblog)。

     对于Windows环境下NDK的开发,如果使用的NDK是r7之前的版本,必须要安装Cygwin才能使用NDK,所以为Eclipse需要配置的builder,其实是执行Cygwin,然后传递ndk-build作为参数。在NDKr7开始,Google的Windows版的NDK提供了一个ndk-build.cmd的脚本,这样,就可以直接利用这个脚本编译,而不需要使用Cygwin了。只需要为Eclipse Android工程添加一个Builders,就能让Eclipse自动编译NDK。

    现在我们来讲解怎么搭建ndk的环境,ndk的下载地址为:https://developer.android.com/tools/sdk/ndk/index.html   下载后解压相应的ndk,我解压在D盘的android-ndk目录下,解压后的的文件为android-ndk-r9c,如下图:Android应用程序通过JNI调用驱动程序(友善Smart210)

1.解压完ndk后,接下来我们来在Eclipse中怎么把ndk的部分设置进去,首先打开Eclipse,创建一个新工程,我去的工程名字为:LEDAPP,包的名字为:com.ndk.led,如下图:

Android应用程序通过JNI调用驱动程序(友善Smart210)

至于具体怎么在Eclipse下建立Android的工程,网上很多资料,只是这里需要注意,因为后边要用到所以专门弄出来说说。

2.在新建立的工程里面建立一个jni文件夹,该文件存放ndk需要编译的文件,具体为:在所建工程的名字LEDAPP上右键->New->folder,然后写入jni的名字,完成后,如图下图:

Android应用程序通过JNI调用驱动程序(友善Smart210)

3.建立并配置Builder

(a)右键LEDAPP->Properties->Builders,如下图:

Android应用程序通过JNI调用驱动程序(友善Smart210)

(b)点击右边的New,如下图:

Android应用程序通过JNI调用驱动程序(友善Smart210)

(c)然后点击"Program",并点击OK按钮,出现下图:

Android应用程序通过JNI调用驱动程序(友善Smart210)

(d)在弹出的【EditConfiguration】对话框中,配置选项卡【Main】。
    在“Name“中输入新builders的名称(我取名为ndk_Builder)。
    在“Location”中输入nkd-build.cmd的路径。
   (我的是D:\android-ndk\android-ndk-r9c\ndk-build.cmd,根据各自的ndk路径设置,也可以点击“Browser File System…”来选取这个路径)。在“WorkingDiretcoty”中输入${workspace_loc:/LEDAPP}(也可以点击“Browse Workspace”来选取LEDAPP目录)。
具体设置完成后,如下图:Android应用程序通过JNI调用驱动程序(友善Smart210)

 (e)【EditConfiguration】对话框中,配置选项卡【Refresh】。
      勾选“Refresh resources upon completion”,
      勾选“The entire workspace”,
      勾选“Recuresively include sub-folders”。 具体入下图:

Android应用程序通过JNI调用驱动程序(友善Smart210)

(f)【Edit Configuration】对话框中,配置选项卡【Build options】。
      勾选“After a “Clean””,
      勾选“During manual builds”,
      勾选“During auto builds”,
      勾选“Specify working set of relevantresources”。
      点击“Specify Resources…”
      勾选LEDAPP工程的“jni“目录,点击”finish“。  具体设置完如下图:

Android应用程序通过JNI调用驱动程序(友善Smart210)

(g)点击上图中的OK按钮后,再点击刚开始出现的对话框的OK按钮,这样就设置完成了

4.在LEDAPP下新建一个JniLed.class,这就是存放Android应用程序所调用的类的地方,步骤为:右键LEDAPP->New->class,具体如下图:

Android应用程序通过JNI调用驱动程序(友善Smart210)

在“name”的输入框中输入:JniLed,在package输入框中,选择com.ndk.led,然后点击finish,完成JniLed.class的创建。接下来双击刚创建的JniLed.class,在里面输入如下的内容:

package com.ndk.led;

public class JniLed {
    static public native int LedInit();
    static public native int LedIOCTL(int cmd, int led_num);
}
其中JniLed中的那两个函数,就是我们对led驱动程序经过封装后的函数,你从名字当中也可以看出来,第一个函数就是初始化led,第二个就是通过命令和选择哪一个led进行控制

5.生成com_ndk_led_JniLed.h文件,具体为通过cmd命令进入windows的命令行控制界面,然后进入到工程所在的目录下,具体为进入D盘->test-android/LEDAPP/bin/classes,然后输入:javah com.ndk.led.JniLed,再回车。这样会在classes文件下面生成:com_ndk_led_JniLed.h文件,这就是生成的c/c++文件的头文件。

6.在工程中的jni文件夹下,建立Android.mk文件,名字一定是这个,不能是其他的名字。并且输入如下的内容:

LOCAL_PATH := $(call my-dir)
include $(CLEAR_VARS)
LOCAL_MODULE := LEDAPP
LOCAL_SRC_FILES := com_ndk_led_JniLed.c
include $(BUILD_SHARED_LIBRARY)
这就是Android下面的Makefile,注意第三行中的“LEDAPP”和第四行中的“com_ndk_led_JniLed.c”,一个是模块的名字,另外一个是要编译的c语言的源文件,要和上面的.h文件保存一致。

7.把在classes下面生成的com_ndk_led_JniLed.h文件拷贝到刚开始所建立的jni文件夹下,同时建立com_ndk_led_JniLed.c的文件,这个c文件就是通过调用Linux内核中的驱动函数,以一定的方式进行封装,使Android应用程序能调用驱动的关键。这个文件中的函数要看具体的驱动函数来完成,比如说,咱们现在就得看友善所提供的smart210的Linux-3.0.8内核中的led的驱动函数,具体位置为:友善的Linux-3.0.8/char/mini210_leds.c,我会把这个文件共享,方便大家查看,下面说说com_ndk_led_JniLed.c文件的内容,具体如下:

#include "com_ndk_led_JniLed.h"
#include <stdlib.h>
#include <stdio.h>
#include <string.h>
#include <jni.h>
#include <fcntl.h> /*包括文件操作,如open() read() close() write()等*/
//----for output the debug log message
#include <android/log.h>


int fd;   //文件标识符

#define DEVICE_NAME  "/dev/leds"   //这是linux下面的设备节点,Android应用程序就是通过这个来识别驱动程序的

#ifdef __cplusplus
extern "C"
{
#endif

JNIEXPORT jint JNICALL Java_com_ndk_led_JniLed_LedInit(JNIEnv *env, jclass arg)
{
	fd = open(DEVICE_NAME,O_RDWR);//打开设备

	ioctl(fd,1,2);
    if(fd == -1)
		return 0;
    else
	    return 1;
}

JNIEXPORT jint JNICALL Java_com_ndk_led_JniLed_LedIOCTL(JNIEnv *env, jclass arg, jint cmd, jint led_num)
{
   int CTLCODE = led_num;

   switch(CTLCODE)
   {
      case 0:
    	  if(cmd==0)
    		  ioctl(fd,0,0);  //调用led驱动函数中的ioctl,这里需要注意,要按驱动里的函数保持一致
    	  else if(cmd==1)
    		  ioctl(fd,1,0);
    	  break;
      case 1:
          if(cmd==0)
          	  ioctl(fd,0,1);
          else if(cmd==1)
          	  ioctl(fd,1,1);
          break;
      case 2:
          if(cmd==0)
          	  ioctl(fd,0,2);
          else if(cmd==1)
          	  ioctl(fd,1,2);
          break;
      case 3:
          if(cmd==0)
              ioctl(fd,0,3);
          else if(cmd==1)
          	  ioctl(fd,1,3);
          break;
      default: break;
   }

   return 1;
}

#ifdef __cplusplus
}
#endif

8.同时要更新com_ndk_led_JniLed.h文件中的内容,具体的更新代码如下:

/* DO NOT EDIT THIS FILE - it is machine generated */
#include <jni.h>
/* Header for class com_ndk_led_JniLed */

#ifndef _Included_com_ndk_led_JniLed
#define _Included_com_ndk_led_JniLed
#ifdef __cplusplus
extern "C" {
#endif
/*
 * Class:     com_ndk_led_JniLed
 * Method:    LedInit
 * Signature: ()I
 */
//这就是.c文件中实现函数的声明,注意要一致啊,负责会出现“unfortunately,xxx has stopped!”的错误
JNIEXPORT jint JNICALL Java_com_ndk_led_JniLed_LedInit
  (JNIEnv *, jclass);

/*
 * Class:     com_ndk_led_JniLed
 * Method:    LedIOCTL
 * Signature: (II)I
 */
JNIEXPORT jint JNICALL Java_com_ndk_led_JniLed_LedIOCTL
  (JNIEnv *, jclass, jint, jint);

#ifdef __cplusplus
}
#endif
#endif

以上就是JNI的部分,下面我将进行Android应用程序的开发

Android的应用开发

这一部分我们粗糙的讲一下,就是弄几个按键,然后调用刚才生成的类,然后下载到smart210上做一个测试。

1.更新LEDAPP下面的res/layout/activity_main.xml,这是Android应用程序的界面布局文件,具体的更新内容如下:

<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:paddingBottom="@dimen/activity_vertical_margin"
    android:paddingLeft="@dimen/activity_horizontal_margin"
    android:paddingRight="@dimen/activity_horizontal_margin"
    android:paddingTop="@dimen/activity_vertical_margin"
    tools:context=".MainActivity" >

    <Button
        android:id="@+id/led1_on"
        android:text="@string/myButton1"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_marginLeft="5dip"
    />
    
    <Button
        android:id="@+id/led1_off"
        android:text="@string/myButton2"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_marginLeft="115dip"
    />
    
    <Button
        android:id="@+id/led2_on"
        android:text="@string/myButton3"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_marginLeft="245dip"
    />
    
    <Button
        android:id="@+id/led2_off"
        android:text="@string/myButton4"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_marginLeft="355dip"
    />
    
     <Button
        android:id="@+id/led3_on"
        android:text="@string/myButton5"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_marginTop="70dip"
        android:layout_marginLeft="5dip"
    />
    
    <Button
        android:id="@+id/led3_off"
        android:text="@string/myButton6"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_marginTop="70dip"
        android:layout_marginLeft="115dip"
    />
    
    <Button
        android:id="@+id/led4_on"
        android:text="@string/myButton7"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_marginTop="70dip"
        android:layout_marginLeft="245dip"
    />
    
    <Button
        android:id="@+id/led4_off"
        android:text="@string/myButton8"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_marginTop="70dip"
        android:layout_marginLeft="355dip"
    />

</RelativeLayout>

2.更新LEDAPP下的res/values/strings.xml的内容,这个里面的内容在前面的activity_main.xml的布局文件中需要,具体内容如下:

<?xml version="1.0" encoding="utf-8"?>
<resources>

    <string name="app_name">LEDAPP</string>
    <string name="action_settings">Settings</string>
    <string name="hello_world">Hello world!</string>
    
    <string name="myButton1">LED1_ON</string>
    <string name="myButton2">LED1_OFF</string>
    <string name="myButton3">LED2_ON</string>
    <string name="myButton4">LED2_OFF</string>
    <string name="myButton5">LED3_ON</string>
    <string name="myButton6">LED3_OFF</string>
    <string name="myButton7">LED4_ON</string>
    <string name="myButton8">LED4_OFF</string>

</resources>
3.现在看看MainActivity.java文件,在他里面的具体调用刚生成的类,具体的代码如下:

package com.ndk.led;

import android.os.Bundle;
import android.app.Activity;
import android.view.View;
import android.view.View.OnClickListener;
import android.widget.Button;

public class MainActivity extends Activity {

	static {
        System.loadLibrary("LEDAPP");
    }//上面的System.loadLibrary("LEDAPP")这一句一定要加上,这是调用刚才生成的库的函数,负责会出现“unfortunately,xxx has stopped!”的错误
	Button Button1,Button2,Button3,Button4,Button5,Button6,Button7,Button8;
    /** Called when the activity is first created. */
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        
        JniLed.LedInit();
        
        Button1 = (Button)findViewById(R.id.led1_on);
        Button2 = (Button)findViewById(R.id.led1_off);
        Button3 = (Button)findViewById(R.id.led2_on);
        Button4 = (Button)findViewById(R.id.led2_off);
        Button5 = (Button)findViewById(R.id.led3_on);
        Button6 = (Button)findViewById(R.id.led3_off);
        Button7 = (Button)findViewById(R.id.led4_on);
        Button8 = (Button)findViewById(R.id.led4_off);
        
        class ButtonClick implements OnClickListener
        {

			@Override
			public void onClick(View v) {
				// TODO Auto-generated method stub
				switch(v.getId())
				{
					case R.id.led1_on:
						//Toast.makeText(MainActivity.this, "按钮1", Toast.LENGTH_LONG).show();
						JniLed.LedIOCTL(1,0);
						break;
					case R.id.led1_off:
						JniLed.LedIOCTL(0,0);
						break;
					case R.id.led2_on:
						JniLed.LedIOCTL(1,1);
						break;
					case R.id.led2_off:
						JniLed.LedIOCTL(0,1);
						break;
					case R.id.led3_on:
						JniLed.LedIOCTL(1,2);
						break;
					case R.id.led3_off:
						JniLed.LedIOCTL(0,2);
						break;
					case R.id.led4_on:
						JniLed.LedIOCTL(1,3);
						break;
					case R.id.led4_off:
						JniLed.LedIOCTL(0,3);
						break;
				}
			}
        }
        
        Button1.setOnClickListener(new ButtonClick());
        Button2.setOnClickListener(new ButtonClick());
        Button3.setOnClickListener(new ButtonClick());
        Button4.setOnClickListener(new ButtonClick());
        Button5.setOnClickListener(new ButtonClick());
        Button6.setOnClickListener(new ButtonClick());
        Button7.setOnClickListener(new ButtonClick());
        Button8.setOnClickListener(new ButtonClick());

    }  
}

这里再简单第说一下引起“unfortunately,xxxhas stopped!”的错误,一般是函数的名字不一致,没有加载创建的库所引起的。好吧,就这么多吧,希望对大家有用,下面是Android下的源代码以及smart210的led驱动文件,链接如下:

Android源代码链接:http://download.csdn.net/detail/xie0812/6947967

smart210的led驱动链接:http://download.csdn.net/detail/xie0812/6947979

下面是在smart210上截取效果图片:

Android应用程序通过JNI调用驱动程序(友善Smart210)

Android应用程序通过JNI调用驱动程序(友善Smart210)

上一篇:the code place where the binding is converted to final value displayed in ui


下一篇:Android 布局之LinearLayout 子控件weight权重的作用详析