android驱动学习1-驱动开发流程(Android.mk)

QQ:971586331

软件环境:

操作系统:windows 10

IDE版本:Android Studio 3.4.2

JAVA版本:jdk-8u221-windows-x64

NDK版本:android-ndk-r20-windows-x86_64

Kernel版本:linux 3.0

开发板android版本:android 4.0.3

硬件环境:

开发板:itop-4412 精英版

本文内容:本文描述了如何使用android应用程序调用linux驱动控制LED灯的亮灭。要实现android应用程序控制LED,需要三个程序,LED的linux驱动,JNI库和android应用程序。android应用程序通过JNI库调用LED驱动程序,从而实现android应用程序控制LED。

android驱动学习1-驱动开发流程(Android.mk)_第1张图片

1.开发板环境搭建

开发环境搭建请参考《iTOP-4412开发板之精英版使用手册_V3.1.pdf》,本文使用的配置是

uboot:iTop4412_uboot_20180320.tar

kernel:iTop4412_Kernel_3.0_20180604.tar

android:iTop4412_ICS_git_20151120.tar

编译完成后将ramdisk-uboot.img,system.img,zImage,u-boot-iTOP-4412.bin文件通过OTG或SD烧写到开发板的EMMC中,如果在uboot下使用OTG,发现windows 10装不上光盘中的android_drv_90000_64.exe驱动,可以谷歌搜索安装android_11000010001_x64_718.exe。

2.LED的驱动程序

LED驱动在kernel的drivers/char/itop4412-leds.c中,从itop4412-leds.c中我们可以得知LED驱动的设备文件名叫“leds”。驱动程序实现了ioctl函数。

long leds_ioctl(struct file *filp,unsigned int cmd,unsigned long arg)
{
	printk("debug: leds_ioctl cmd is %d\n" , cmd);

	switch(cmd)
	{
		case 0:
		case 1:
			if (arg > LED_NUM) {
				return -EINVAL;
			}

			gpio_set_value(led_gpios[arg], cmd);
			break;

		default:
			return -EINVAL;
	}

	return 0;
}

leds_ioctl的cmd参数表示灯的亮灯,arg参数表示灯的编号,根据文件中的定义可以知,0表示GPL2_0,也就是LED2,1表示GPK1_1,也就是LED3。

static int led_gpios[] = {
	EXYNOS4_GPL2(0),
	EXYNOS4_GPK1(1),
};

接下来我们查看drivers/char/Makefile文件,宏CONFIG_LEDS_CTL控制LED驱动是否编译

obj-$(CONFIG_LEDS_CTL)		+= itop4412_leds.o

再查看drivers/char/Kconfig文件,默认就是y

config LEDS_CTL
        bool "Enable LEDS config"
        default y
        help
          Enable LEDS config

再查看.config,已经将LED驱动编入了内核

CONFIG_LEDS_CTL=y

看来板子的驱动已经做好了,完全不用我们动手,接下我们看怎么编写JNI接口调用linux驱动

3.JNI和NDK

因为android是使用java语言进行开发的,但linux驱动是用C语言进行开发的,所以面临java如果调用C语言接口的问题,JNI提供的API就是解决java和其他语言通信的问题。NDK 是一套工具集合,允许你使用C语言来实现应用程序的部分功能。我们写好JNI接口后使用NDK打包成库文件,就可以提供给android应用程序调用了。接下来我们新建工程编写JNI。

新建一个空activity

android驱动学习1-驱动开发流程(Android.mk)_第2张图片

填写工程名,选择工程路径,开发语言选择java,API选择15

android驱动学习1-驱动开发流程(Android.mk)_第3张图片

创建工程后得待编译完成,然后在包名下创建一个名叫jni_led的类

android驱动学习1-驱动开发流程(Android.mk)_第4张图片

文件内容如下:

package com.example.led_test;

public class jni_led {

    public native static String Leds_Operation(int ledNum, boolean status); //操作接口
}

打开 Android Studio 的 Terminal,使用javac命令将java文件编译成.class文件

F:\OneDrive\Linux\android_project\led_test>javac .\app\src\main\java\com\example\led_test\jni_led.java

使用javah命令创建头文件。-encoding UTF-8表示指定编码格式,防止出现“错误: 编码GBK的不可映射字符”,-d jni表示在当前目录下创建jni目录,将生成的文件放在jni目录中,-classpath表示指定类文件的路径。这里有一个地方要注意,类文件在写的时候是包名+类名,所有路径只用写到java目录,后面的com,example和led_test虽然都是文件夹,但这里表示包名(第一次写在这里纠结了好久,我想我路径明明写对了啊,为什么找不到\app\src\main\java\com\example\led_test文件夹下的类)

F:\OneDrive\Linux\android_project\led_test>javah -encoding UTF-8 -d jni -classpath ./app/src/main/java com.example.led_test.jni_led

指令执行完成后可以发现在jni目录下生成了包名加类名的头文件

/* DO NOT EDIT THIS FILE - it is machine generated */
#include 
/* Header for class com_example_led_test_jni_led */

#ifndef _Included_com_example_led_test_jni_led
#define _Included_com_example_led_test_jni_led
#ifdef __cplusplus
extern "C" {
#endif
/*
 * Class:     com_example_led_test_jni_led
 * Method:    Leds_Operation
 * Signature: (IZ)Ljava/lang/String;
 */
JNIEXPORT jstring JNICALL Java_com_example_led_1test_jni_1led_Leds_1Operation
  (JNIEnv *, jclass, jint, jboolean);

#ifdef __cplusplus
}
#endif
#endif

可以发现,头文件中根据jni_led.java中定义的java类接口生成了JNI接口函数,我们要实现这个接口函数。

然后在JNI下创建com_example_led_test_jni_led.c文件

android驱动学习1-驱动开发流程(Android.mk)_第5张图片

在com_example_led_test_jni_led.c中,我们将头文件中的接口函数据复制过来,然后使用linux API操作linux设备文件

//
// Created by shiyu on 2019/8/17.
//

#include
#include
#include 
#include 
//导入我们创建的头文件
#include "com_example_led_test_jni_led.h"

#define DEVICE_NAME		"/dev/leds"

JNIEXPORT jstring JNICALL Java_com_example_led_JNITest_Leds_1Operation
  (JNIEnv *env, jclass obj, jint ledsNum, jboolean status){

int leds_fd = 0;

	leds_fd = open(DEVICE_NAME, O_RDWR);  //打开设备节点
	if (leds_fd == -1) {
		return 1;
	}

	switch (ledsNum) {
	case 0:
		if (status)
			ioctl(leds_fd, 0, 0);
		else
			ioctl(leds_fd, 1, 0);
		break;
	case 1:
		if (status)
			ioctl(leds_fd, 0, 1);
		else
			ioctl(leds_fd, 1, 1);
		break;
	defautl :
		break;
	}

	close(leds_fd);

	return 0;  //操作成功返回0
}

在jni下创建一个Android.mk文件


LOCAL_PATH := $(call my-dir)
include $(CLEAR_VARS)

LOCAL_MODULE := jni_led
LOCAL_SRC_FILES := com_example_led_test_jni_led.c
include $(BUILD_SHARED_LIBRARY)

这时指定了生成库的名字和源文件,再新建一个Application.mk文件

APP_ABI := all

安装NDK工具集后,进入jni目录使用ndk-build命令将JNI接口程序编译成库文件

android驱动学习1-驱动开发流程(Android.mk)_第6张图片

在libs目录下生成了各种平台的库文件

android驱动学习1-驱动开发流程(Android.mk)_第7张图片

为了让项目能够找到我们的生成的库,在 build.gradle 文件夹的 android 下添加:

sourceSets {
        main() {
            jniLibs.srcDirs = ['../libs']
            jni.srcDirs = [] //屏蔽掉默认的jni编译生成过程
        }
    }

然后在jni_led.java中加载生成的库文件

package com.example.led_test;

public class jni_led {
    static {
        System.loadLibrary("jni_led");  //加载生成的.so文件
    }
    public native static String Leds_Operation(int ledNum, boolean status); //操作接口
}

接下来我们编写android应用程序利用Leds_Operation接口控制LED灯

4.编写android应用程序

打开工程目录下的activity_main.xml文件,添加4个button,并指写button的onClick回调函数




    

        

            

我们为4个按键指定了4个回调函数据,接下来我们在MainActivity.java中实现这4个回调函数

package com.example.led_test;

import androidx.appcompat.app.AppCompatActivity;

import android.os.Bundle;
import android.view.View;

public class MainActivity extends AppCompatActivity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
    }

    public void led2_on_click( View view )
    {
        jni_led.Leds_Operation(0, false);
    }

    public void led2_off_click( View view )
    {
        jni_led.Leds_Operation(0, true);
    }

    public void led3_on_click( View view )
    {
        jni_led.Leds_Operation(1, false);
    }

    public void led3_off_click( View view )
    {
        jni_led.Leds_Operation(1, true);
    }
}

如上,我们实现了这4个回调函数,调用jni_led库中的Leds_Operation函数,Leds_Operation会调用Java_com_example_led_JNITest_Leds_1Operation函数,这样就实现了android应用程序调用linux驱动接口。

连接开发板,编译运行。

android驱动学习1-驱动开发流程(Android.mk)_第8张图片

 

 

你可能感兴趣的:(android驱动)