Android驅動入門-Led控制+app+ndk庫+底層驅動


硬件平台: FriendlyARM Tiny4412 Cortex-A9

操作系統: UBUNTU 14.04 LTS

本次實驗使用的是 安卓APP + NDK庫 + Linux底層驅動。

一、 首先在 Android Studio 上編寫APP。

對軟件進行布局。

  1. <?xml version="1.0" encoding="utf-8"?>  
  2. <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"  
  3.     xmlns:tools="http://schemas.android.com/tools"  
  4.     android:layout_width="match_parent"  
  5.     android:layout_height="match_parent"  
  6.     android:paddingBottom="@dimen/activity_vertical_margin"  
  7.     android:paddingLeft="@dimen/activity_horizontal_margin"  
  8.     android:paddingRight="@dimen/activity_horizontal_margin"  
  9.     android:paddingTop="@dimen/activity_vertical_margin"  
  10.     tools:context="com.example.zws.test_led.MainActivity">  
  11.     <LinearLayout  
  12.         android:layout_width="match_parent"  
  13.         android:layout_height="wrap_content"  
  14.         android:orientation="vertical">  
  15.   
  16.         <LinearLayout  
  17.             android:layout_width="fill_parent"  
  18.             android:layout_height="fill_parent"  
  19.             android:orientation="horizontal">  
  20.             <CheckBox  
  21.                 android:id="@+id/checkbox_cmd_led1"  
  22.                 android:layout_width="wrap_content"  
  23.                 android:layout_height="wrap_content"  
  24.                 android:text="Led1"/>  <!--id在判斷復選框狀態時使用-->  
  25.   
  26.             <CheckBox  
  27.                 android:id="@+id/checkbox_cmd_led2"  
  28.                 android:layout_width="wrap_content"  
  29.                 android:layout_height="wrap_content"  
  30.                 android:text="Led2"/>  
  31.   
  32.             <CheckBox  
  33.                 android:id="@+id/checkbox_cmd_led3"  
  34.                 android:layout_width="wrap_content"  
  35.                 android:layout_height="wrap_content"  
  36.                 android:text="Led3"/>  
  37.   
  38.             <CheckBox  
  39.                 android:id="@+id/checkbox_cmd_led4"  
  40.                 android:layout_width="wrap_content"  
  41.                 android:layout_height="wrap_content"  
  42.                 android:text="Led4"/>  
  43.         </LinearLayout>  
  44.         <Button  
  45.             android:layout_width="fill_parent"  
  46.             android:layout_height="wrap_content"  
  47.             android:text="發送命令"  
  48.             android:onClick="on_click_cmd"/>    <!--按鈕響應函數名稱-->  
  49.     </LinearLayout>  
  50.   
  51. </RelativeLayout>  
<?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">
    <LinearLayout
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:orientation="vertical">

        <LinearLayout
            android:layout_width="fill_parent"
            android:layout_height="fill_parent"
            android:orientation="horizontal">
            <CheckBox
                android:id="@+id/checkbox_cmd_led1"
                android:layout_width="wrap_content"
                android:layout_height="wrap_content"
                android:text="Led1"/>  <!--id在判斷復選框狀態時使用-->

            <CheckBox
                android:id="@+id/checkbox_cmd_led2"
                android:layout_width="wrap_content"
                android:layout_height="wrap_content"
                android:text="Led2"/>

            <CheckBox
                android:id="@+id/checkbox_cmd_led3"
                android:layout_width="wrap_content"
                android:layout_height="wrap_content"
                android:text="Led3"/>

            <CheckBox
                android:id="@+id/checkbox_cmd_led4"
                android:layout_width="wrap_content"
                android:layout_height="wrap_content"
                android:text="Led4"/>
        </LinearLayout>
        <Button
            android: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程序。

  1. package com.example.zws.test_led;  
  2.   
  3. import android.app.Activity;        //可更改  
  4. import android.os.Bundle;  
  5. import android.view.View;  
  6. import android.widget.CheckBox;  
  7.   
  8. import java.util.zip.CheckedInputStream;  
  9.   
  10.   
  11. public class MainActivity extends Activity {    //需和import android.app.Activity;一致  
  12.   
  13.     private CheckBox[] cbCmdLeds  = new CheckBox[4];    //創建按鈕數組  
  14.     @Override  
  15.     protected void onCreate(Bundle savedInstanceState) {  
  16.         super.onCreate(savedInstanceState);  
  17.         setContentView(R.layout.activity_main);  
  18.   
  19.         cbCmdLeds[0] = (CheckBox)findViewById(R.id.checkbox_cmd_led1);  //需要強制轉化  
  20.         cbCmdLeds[1] = (CheckBox)findViewById(R.id.checkbox_cmd_led2);  
  21.         cbCmdLeds[2] = (CheckBox)findViewById(R.id.checkbox_cmd_led3);  
  22.         cbCmdLeds[3] = (CheckBox)findViewById(R.id.checkbox_cmd_led4);  
  23.     }  
  24.   
  25.     public void on_click_cmd( View view )  
  26.     {  
  27.         for( int i=0; i<4; i++ )  
  28.         {  
  29.             if( cbCmdLeds[i].isChecked() )   //如復選框被選中  
  30.                 CmdLeds( 1,i ); //此函數在下面有聲明。  
  31.             else  
  32.                 CmdLeds( 0,i );  
  33.         }  
  34.     }  
  35.   
  36.     public native void CmdLeds( int cmd, int arg ); //需要調用的外部函數可自己設定  
  37.   
  38.     static  
  39.     {  
  40.         System.loadLibrary("ndk_test_myled");       //外部庫的名稱,可自己設定  
  41.     }  
  42. }  
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];    //創建按鈕數組
    @Override
    protected 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 ); //此函數在下面有聲明。
            else
                CmdLeds( 0,i );
        }
    }

    public native void CmdLeds( int cmd, int arg ); //需要調用的外部函數可自己設定

    static
    {
        System.loadLibrary("ndk_test_myled");       //外部庫的名稱,可自己設定
    }
}

編寫完成后需要在工程的app/src/main目錄下創建目錄jniLibs.這里將放ndk生成的庫文件。

然后在Build->make project.

 

二、編寫底層驅動。

  1. #include <linux/kernel.h>  
  2. #include <linux/module.h>  
  3. #include <linux/fs.h>  
  4. #include <linux/slab.h>  
  5. #include <linux/device.h>  
  6. #include <asm/io.h>  
  7. #include <asm/uaccess.h>  
  8. #include <linux/cdev.h>  
  9.   
  10.   
  11. MODULE_LICENSE("GPL");  
  12.   
  13. #define GPM4CON 0X110002E0    //io口控制寄存器硬件地址  
  14. #define GPM4DAT 0X110002E4    //io口數據寄存器硬件地址  
  15.   
  16. #define LED_ON _IOW('G',0,int)  //打開命令  
  17.   
  18. #define LED_OFF _IOW('G',1,int) //關閉命令  
  19.   
  20. static struct cdev dev;//1.1 分配cdev結構  
  21.   
  22. static dev_t dev_no;  //設備號  
  23. struct class *led_class;  
  24.   
  25. static unsigned int *led_con;  
  26. static unsigned int *led_dat;  
  27.   
  28. long led_ioctl( struct file *file, unsigned int cmd, unsigned long arg )  
  29. {  
  30.     switch( cmd )  
  31.     {  
  32.         case LED_ON:  
  33.             writel((readl(led_dat)&(~(0x1<<(arg-1)))),led_dat);  
  34.             break;  
  35.   
  36.         case LED_OFF:  
  37.             writel( (readl(led_dat)|(0x1<<(arg-1))),led_dat);  
  38.             break;  
  39.   
  40.         default:   
  41.             return -EINVAL;          
  42.             break;  
  43.     }  
  44.     return 0;  
  45. }  
  46.   
  47. struct file_operations led_fops = {  
  48.     .owner = THIS_MODULE,  
  49.     .unlocked_ioctl = led_ioctl,  
  50. };  
  51.   
  52.   
  53. static void hw_init()  
  54. {  
  55.       
  56.     //初始化GPIO控制寄存器  
  57.     led_con = ioremap( GPM4CON, 4 );    //地址映射  將實際的硬件地址映射成可訪問的虛擬地址  
  58.     led_dat = ioremap( GPM4DAT, 4 );  
  59.     writel((readl(led_con)&~0xffff)|0x1111,led_con);      
  60.     writel(readl(led_dat)|0xf,led_dat);      
  61.       
  62. }  
  63.   
  64. static int led_init()  
  65. {  
  66.       
  67.     //1.2 初始化cdev結構  
  68.     alloc_chrdev_region( &dev_no, 0, 1, "my_led" );  
  69.     cdev_init( &dev, &led_fops );      
  70.     dev.owner = THIS_MODULE;  
  71.     //1.3 注冊cdev結構  
  72.     cdev_add( &dev, dev_no, 1 );  
  73.     //2.硬件初始化  
  74.     hw_init();  
  75.     //3.創建設備文件  
  76.     led_class = class_create(THIS_MODULE,"my_led");                //創建設備類  
  77.     device_create( led_class, NULL, dev_no,NULL,"%s","my_led");  // 創建設備文件 my_led  
  78.   
  79.     printk("init led device ok!\n");  
  80.   
  81.     return 0;  
  82. }  
  83.   
  84. void led_exit()  
  85. {  
  86.     device_destroy(led_class,dev_no);  
  87.     class_destroy(led_class);  
  88.   
  89.     iounmap(led_con);  
  90.     iounmap(led_dat);  
  91.   
  92.     cdev_del(&dev);  
  93.     unregister_chrdev_region(dev_no,1);  
  94. }  
  95.   
  96. module_init( led_init );  
  97. module_exit( led_exit );  
#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_led

    printk("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。

  1. obj-m := led.o  
  2. KDIR := /home/share/linux-3.0.86  //內核地址  
  3.   
  4. all:  
  5.     make -C $(KDIR) M=$(PWD) modules CROSS_COMPILE=arm-linux- ARCH=arm  
  6. clean:  
  7.     rm -f *.ko *.o  
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文件,編寫相應的接口。

  1. #include "com_example_zws_test_led_MainActivity.h"  //生成的頭文件  
  2.   
  3. #include <jni.h>  
  4. #include <fcntl.h>  
  5. #include <stdio.h>  
  6. #include <sys/stat.h>  
  7. #include <sys/ioctl.h>  
  8. #include <unistd.h>  
  9. #include <stdlib.h>  
  10.   
  11. #define LED_ON _IOW('G',0,int)  
  12. #define LED_OFF _IOW('G',1,int)  
  13.   
  14. JNIEXPORT void JNICALL Java_com_example_zws_test_1led_MainActivity_CmdLeds  
  15.   (JNIEnv *env, jobject this, jint cmd, jint arg)    //頭文件中的接口函數  此函數名稱為系統自動生成的  勿修改。  
  16. {  
  17.     int fd;  
  18.     int tmp_cmd;  
  19.       
  20.     fd = open("/dev/my_led",O_WRONLY);        //此為在寫驅動程序時,生成的設備文件名稱  
  21.   
  22.     if( cmd == 1 )  
  23.         tmp_cmd = LED_ON;  
  24.     else if(cmd==0)  
  25.         tmp_cmd = LED_OFF;  
  26.   
  27.     ioctl(fd,tmp_cmd,arg+1);  
  28.     close(fd);  
  29. }  
#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 文件

  1. LOCAL_PATH := $(call my-dir)  
  2. include $(CLEAR_VARS)  
  3. LOCAL_MODULE := ndk_test_myled    //這個是庫的名稱 可自己設定  
  4. LOCAL_SRC_FILES := ndk_led.c    //之前寫的c文件名稱  
  5. include $(BUILD_SHARED_LIBRARY)  
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上運行即可。


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM