Android驱动入门-Led控制+app+ndk库+底层驱动

2024-06-22 09:18

本文主要是介绍Android驱动入门-Led控制+app+ndk库+底层驱动,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!



硬件平台: FriendlyARM Tiny4412 Cortex-A9

操作系统: UBUNTU 14.04 LTS

本次实验使用的是 安卓APP + NDK库 + Linux底层驱动。

一、 首先在 Android Studio 上编写APP。

对软件进行布局。

<?xml version="1.0" encoding="utf-8"?>
<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="com.example.zws.test_led.MainActivity"><LinearLayoutandroid:layout_width="match_parent"android:layout_height="wrap_content"android:orientation="vertical"><LinearLayoutandroid:layout_width="fill_parent"android:layout_height="fill_parent"android:orientation="horizontal"><CheckBoxandroid:id="@+id/checkbox_cmd_led1"android:layout_width="wrap_content"android:layout_height="wrap_content"android:text="Led1"/>  <!--id在判断复选框状态时使用--><CheckBoxandroid:id="@+id/checkbox_cmd_led2"android:layout_width="wrap_content"android:layout_height="wrap_content"android:text="Led2"/><CheckBoxandroid:id="@+id/checkbox_cmd_led3"android:layout_width="wrap_content"android:layout_height="wrap_content"android:text="Led3"/><CheckBoxandroid:id="@+id/checkbox_cmd_led4"android:layout_width="wrap_content"android:layout_height="wrap_content"android:text="Led4"/></LinearLayout><Buttonandroid:layout_width="fill_parent"android:layout_height="wrap_content"android:text="发送命令"android:onClick="on_click_cmd"/>    <!--按钮响应函数名称--></LinearLayout></RelativeLayout>

其中, android:id="@+id/checkbox_cmd_led1"在程序中识别复选框的状态时使用。

     android:onClick="on_click_cmd"   on_click_cmd为在按下按钮时,调用的函数名称。

 

编写app程序。

package com.example.zws.test_led;import android.app.Activity;        //可更改
import android.os.Bundle;
import android.view.View;
import android.widget.CheckBox;import java.util.zip.CheckedInputStream;public class MainActivity extends Activity {    //需和import android.app.Activity;一致private CheckBox[] cbCmdLeds  = new CheckBox[4];    //创建按钮数组@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);cbCmdLeds[0] = (CheckBox)findViewById(R.id.checkbox_cmd_led1);  //需要强制转化cbCmdLeds[1] = (CheckBox)findViewById(R.id.checkbox_cmd_led2);cbCmdLeds[2] = (CheckBox)findViewById(R.id.checkbox_cmd_led3);cbCmdLeds[3] = (CheckBox)findViewById(R.id.checkbox_cmd_led4);}public void on_click_cmd( View view ){for( int i=0; i<4; i++ ){if( cbCmdLeds[i].isChecked() )   //如复选框被选中CmdLeds( 1,i ); //此函数在下面有声明。elseCmdLeds( 0,i );}}public native void CmdLeds( int cmd, int arg ); //需要调用的外部函数可自己设定static{System.loadLibrary("ndk_test_myled");       //外部库的名称,可自己设定}
}

编写完成后需要在工程的app/src/main目录下创建目录jniLibs.这里将放ndk生成的库文件。

然后在Build->make project.

 

二、编写底层驱动。

#include <linux/kernel.h>
#include <linux/module.h>
#include <linux/fs.h>
#include <linux/slab.h>
#include <linux/device.h>
#include <asm/io.h>
#include <asm/uaccess.h>
#include <linux/cdev.h>MODULE_LICENSE("GPL");#define GPM4CON 0X110002E0    //io口控制寄存器硬件地址
#define GPM4DAT 0X110002E4    //io口数据寄存器硬件地址#define LED_ON _IOW('G',0,int)  //打开命令#define LED_OFF _IOW('G',1,int) //关闭命令static struct cdev dev;//1.1 分配cdev结构static dev_t dev_no;  //设备号
struct class *led_class;static unsigned int *led_con;
static unsigned int *led_dat;long led_ioctl( struct file *file, unsigned int cmd, unsigned long arg )
{switch( cmd ){case LED_ON:writel((readl(led_dat)&(~(0x1<<(arg-1)))),led_dat);break;case LED_OFF:writel( (readl(led_dat)|(0x1<<(arg-1))),led_dat);break;default: return -EINVAL;        break;}return 0;
}struct file_operations led_fops = {.owner = THIS_MODULE,.unlocked_ioctl = led_ioctl,
};static void hw_init()
{//初始化GPIO控制寄存器led_con = ioremap( GPM4CON, 4 );    //地址映射  将实际的硬件地址映射成可访问的虚拟地址led_dat = ioremap( GPM4DAT, 4 );writel((readl(led_con)&~0xffff)|0x1111,led_con);    writel(readl(led_dat)|0xf,led_dat);    }static int led_init()
{//1.2 初始化cdev结构alloc_chrdev_region( &dev_no, 0, 1, "my_led" );cdev_init( &dev, &led_fops );    dev.owner = THIS_MODULE;//1.3 注册cdev结构cdev_add( &dev, dev_no, 1 );//2.硬件初始化hw_init();//3.创建设备文件led_class = class_create(THIS_MODULE,"my_led");                //创建设备类device_create( led_class, NULL, dev_no,NULL,"%s","my_led");  // 创建设备文件 my_ledprintk("init led device ok!\n");return 0;
}void led_exit()
{device_destroy(led_class,dev_no);class_destroy(led_class);iounmap(led_con);iounmap(led_dat);cdev_del(&dev);unregister_chrdev_region(dev_no,1);
}module_init( led_init );
module_exit( led_exit );

编写Makefile。

obj-m := led.o
KDIR := /home/share/linux-3.0.86  //内核地址all:make -C $(KDIR) M=$(PWD) modules CROSS_COMPILE=arm-linux- ARCH=arm
clean:rm -f *.ko *.o

然后make即可。将生成led.ko文件。

通过adb将文件穿发送到安卓设备。这里将ko文件发送到安卓设备的/data/local目录下。

adb push led.ko  /data/local/  

 

三、 NDK程序库编写。

首先生成之前app中写的接口头文件。

javah -d jni -classpath /opt/android-sdk-linux/platforms/android-23/android.jar:/home/my_Android/led\
/NDK/NDK_APP/app/build/intermediates/classes/debug/ com.android.jack.ndk.happy.MainActivity

其中/opt/android-sdk-linux/platforms/android-23/android.jar是安卓sdk中的地址。

/home/my_Android/led/NDK/NDK_APP/app/build/intermediates/classes/debug/ 是相应安卓app源文件工程中的地址。

com.android.jack.ndk.happy.MainActivity为安卓的项目名称。

运行命令后,会在目录中生成jni文件夹。其中com_android_jack_ndk_happy_MainActivity.h为我们需要的头文件

 

创建ndk_led.c文件,编写相应的接口。

#include "com_example_zws_test_led_MainActivity.h"  //生成的头文件#include <jni.h>
#include <fcntl.h>
#include <stdio.h>
#include <sys/stat.h>
#include <sys/ioctl.h>
#include <unistd.h>
#include <stdlib.h>#define LED_ON _IOW('G',0,int)
#define LED_OFF _IOW('G',1,int)JNIEXPORT void JNICALL Java_com_example_zws_test_1led_MainActivity_CmdLeds(JNIEnv *env, jobject this, jint cmd, jint arg)    //头文件中的接口函数  此函数名称为系统自动生成的  勿修改。
{int fd;int tmp_cmd;fd = open("/dev/my_led",O_WRONLY);        //此为在写驱动程序时,生成的设备文件名称if( cmd == 1 )tmp_cmd = LED_ON;else if(cmd==0)tmp_cmd = LED_OFF;ioctl(fd,tmp_cmd,arg+1);close(fd);
}

创建编写 Android.mk 文件

LOCAL_PATH := $(call my-dir)
include $(CLEAR_VARS)
LOCAL_MODULE := ndk_test_myled    //这个是库的名称 可自己设定
LOCAL_SRC_FILES := ndk_led.c    //之前写的c文件名称
include $(BUILD_SHARED_LIBRARY)

编写完成后,回到本层目录的上一层。

执行命令生成ndk库。

ndk-build

即会生成 libs/armeabi/libndk_test_myled.so

将armeabi/libndk_test_myled.so复制到安卓app目录下的 app/src/mian/jinLibs中。

重新编译工程。

 

连接开发板的串口进入控制台。

执行一下命令获取root权限并安装内核模块。

$ su
$ insmod /data/local/led.ko
$ chmod 777 /dev/my_led

led.ko为在编写NDK时,生成的KO模块名称。

my_led为在设备安装后生成的设备文件名称,将其权限改为任何人均可访问。

 

在studio上点击run app ,让app在tiny4412上运行即可。

 

这篇关于Android驱动入门-Led控制+app+ndk库+底层驱动的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1083877

相关文章

Android协程高级用法大全

《Android协程高级用法大全》这篇文章给大家介绍Android协程高级用法大全,本文结合实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友跟随小编一起学习吧... 目录1️⃣ 协程作用域(CoroutineScope)与生命周期绑定Activity/Fragment 中手

Java+AI驱动实现PDF文件数据提取与解析

《Java+AI驱动实现PDF文件数据提取与解析》本文将和大家分享一套基于AI的体检报告智能评估方案,详细介绍从PDF上传、内容提取到AI分析、数据存储的全流程自动化实现方法,感兴趣的可以了解下... 目录一、核心流程:从上传到评估的完整链路二、第一步:解析 PDF,提取体检报告内容1. 引入依赖2. 封装

C++ STL-string类底层实现过程

《C++STL-string类底层实现过程》本文实现了一个简易的string类,涵盖动态数组存储、深拷贝机制、迭代器支持、容量调整、字符串修改、运算符重载等功能,模拟标准string核心特性,重点强... 目录实现框架一、默认成员函数1.默认构造函数2.构造函数3.拷贝构造函数(重点)4.赋值运算符重载函数

Java List 使用举例(从入门到精通)

《JavaList使用举例(从入门到精通)》本文系统讲解JavaList,涵盖基础概念、核心特性、常用实现(如ArrayList、LinkedList)及性能对比,介绍创建、操作、遍历方法,结合实... 目录一、List 基础概念1.1 什么是 List?1.2 List 的核心特性1.3 List 家族成

Redis分布式锁中Redission底层实现方式

《Redis分布式锁中Redission底层实现方式》Redission基于Redis原子操作和Lua脚本实现分布式锁,通过SETNX命令、看门狗续期、可重入机制及异常处理,确保锁的可靠性和一致性,是... 目录Redis分布式锁中Redission底层实现一、Redission分布式锁的基本使用二、Red

c++日志库log4cplus快速入门小结

《c++日志库log4cplus快速入门小结》文章浏览阅读1.1w次,点赞9次,收藏44次。本文介绍Log4cplus,一种适用于C++的线程安全日志记录API,提供灵活的日志管理和配置控制。文章涵盖... 目录简介日志等级配置文件使用关于初始化使用示例总结参考资料简介log4j 用于Java,log4c

史上最全MybatisPlus从入门到精通

《史上最全MybatisPlus从入门到精通》MyBatis-Plus是MyBatis增强工具,简化开发并提升效率,支持自动映射表名/字段与实体类,提供条件构造器、多种查询方式(等值/范围/模糊/分页... 目录1.简介2.基础篇2.1.通用mapper接口操作2.2.通用service接口操作3.进阶篇3

Android 缓存日志Logcat导出与分析最佳实践

《Android缓存日志Logcat导出与分析最佳实践》本文全面介绍AndroidLogcat缓存日志的导出与分析方法,涵盖按进程、缓冲区类型及日志级别过滤,自动化工具使用,常见问题解决方案和最佳实... 目录android 缓存日志(Logcat)导出与分析全攻略为什么要导出缓存日志?按需过滤导出1. 按

Python自定义异常的全面指南(入门到实践)

《Python自定义异常的全面指南(入门到实践)》想象你正在开发一个银行系统,用户转账时余额不足,如果直接抛出ValueError,调用方很难区分是金额格式错误还是余额不足,这正是Python自定义异... 目录引言:为什么需要自定义异常一、异常基础:先搞懂python的异常体系1.1 异常是什么?1.2

Python实现Word转PDF全攻略(从入门到实战)

《Python实现Word转PDF全攻略(从入门到实战)》在数字化办公场景中,Word文档的跨平台兼容性始终是个难题,而PDF格式凭借所见即所得的特性,已成为文档分发和归档的标准格式,下面小编就来和大... 目录一、为什么需要python处理Word转PDF?二、主流转换方案对比三、五套实战方案详解方案1: