您好,登錄后才能下訂單哦!
這篇文章主要介紹了Android怎么開發第一個驅動程序的相關知識,內容詳細易懂,操作簡單快捷,具有一定借鑒價值,相信大家閱讀完這篇Android怎么開發第一個驅動程序文章都會有所收獲,下面我們一起來看看吧。
frist_driver.c
#include <linux/types.h>#include <linux/kernel.h>#include <linux/delay.h>#include <linux/ide.h>#include <linux/init.h>#include <linux/module.h>#include <linux/errno.h>#include <linux/cdev.h>#include <linux/device.h>#include <linux/of.h>#include <asm/uaccess.h>#include <asm/io.h>#define HELLOON 1#define HELLOOFF 0 #define HELLO_CNT 1#define HELLO_NAME "hello driver"static char readbuf[100];static char writebuf[100];static char kerneldata[] = {"hello my first driver"};static char mybuf[100] ="1234";struct hello_driver{dev_t devid; /*設備號*/struct cdev cdev; /*cdev*/struct class *class; /*類*/struct device *device; /*設備*/int major; /*主設備號*/int minor; /*次設備號*/atomic_t atomic_lock; /*原子變量*/ };struct hello_driver hello_driver_dev;static ssize_t show_my_device(struct device *dev,struct device_attribute *attr, char *buf){return sprintf(buf, "%s\n", mybuf);}static ssize_t set_my_device(struct device *dev,struct device_attribute *attr, const char *buf, size_t len)//echo命令時,將會調用該函數{sprintf(mybuf, "%s", buf);return len;}static DEVICE_ATTR(my_device_test, S_IWUSR|S_IRUSR, show_my_device, set_my_device);//定義一個名字為my_device_test的設備屬性文件static int hello_driver_open(struct inode *inode, struct file *filp){ if(!atomic_dec_and_test(&hello_driver_dev.atomic_lock)) { atomic_inc(&hello_driver_dev.atomic_lock); return -EBUSY; }filp->private_data = &hello_driver_dev;return 0;}static ssize_t hello_driver_read(struct file *filp, char __user *buf, size_t cnt, loff_t *offt){int ret = 0;memcpy(readbuf,kerneldata,sizeof(kerneldata));ret = copy_to_user(buf,readbuf,cnt);if(ret == 0){printk("kernel senddata ok!\n");}else{printk("kernel senddata failed\n");}return 0;}static ssize_t hello_driver_write(struct file *filp, const char __user *buf, size_t cnt, loff_t *offt){int ret = 0;ret = copy_from_user(writebuf,buf,cnt);if(ret == 0){printk("kernel recvdata %s\n",writebuf);}else{printk("kernel recvdata failed\n");}return 0;}static long hello_driver_unlocked_ioctl(struct file *filp, unsigned int cmd, unsigned long arg){switch(cmd){case HELLOON: printk("Open hello driver\n");break;case HELLOOFF: printk("Close hello driver\n");break;default:break;}return 0;}static int hello_driver_release(struct inode *inode, struct file *filp){struct hello_driver *dev = filp->private_data;atomic_inc(&dev->atomic_lock);return 0;}static struct file_operations hello_driver_fops= {.owner = THIS_MODULE,.open = hello_driver_open, .read = hello_driver_read,.write = hello_driver_write,.unlocked_ioctl = hello_driver_unlocked_ioctl,.release = hello_driver_release,};static int __init hello_driver_init(void){int ret;/*1.設置原子變量,保證同一時刻只能有一個應用程序使用該驅動*/atomic_set(&hello_driver_dev.atomic_lock,1);/*2.分配設備號*//*2.1 之前分配了主設備號*/if(hello_driver_dev.major){/*注冊設備號*/hello_driver_dev.devid = MKDEV(hello_driver_dev.major,0);ret = register_chrdev_region(hello_driver_dev.devid, HELLO_CNT, HELLO_NAME);if(ret < 0){printk("can't register major\n"); return ret; }}else/*2.2 之前未分配設備號,向內核申請設備號*/{ alloc_chrdev_region(&hello_driver_dev.devid, 0, HELLO_CNT, HELLO_NAME); }printk(KERN_ERR"hello_driver_dev major = %d, minor = %d\n", hello_driver_dev.major,hello_driver_dev.minor);hello_driver_dev.cdev.owner = THIS_MODULE;cdev_init(&hello_driver_dev.cdev,&hello_driver_fops);ret = cdev_add(&hello_driver_dev.cdev, hello_driver_dev.devid,HELLO_CNT);if(ret){printk("Error cdev_add\n"); goto fail_to_cdev_add;}hello_driver_dev.class = class_create(THIS_MODULE,HELLO_NAME);if(IS_ERR(hello_driver_dev.class)){goto fail_to_class_create;}hello_driver_dev.device = device_create(hello_driver_dev.class , NULL, hello_driver_dev.devid, NULL, HELLO_NAME);if(IS_ERR(hello_driver_dev.device)){goto fail_to_device_create;} if(sysfs_create_file(&hello_driver_dev.device ->kobj), &dev_attr_my_device_test.attr)){ //在mytest_device設備目錄下創建一個my_device_test屬性文件 return -1;}return 0;fail_to_cdev_add:unregister_chrdev_region(hello_driver_dev.devid, HELLO_CNT); return -1;fail_to_class_create:cdev_del(&hello_driver_dev.cdev);unregister_chrdev_region(hello_driver_dev.devid, HELLO_CNT);return -1; fail_to_device_create:cdev_del(&hello_driver_dev.cdev);unregister_chrdev_region(hello_driver_dev.devid, HELLO_CNT); class_destroy(hello_driver_dev.class);return -1;}static void __exit hello_driver_exit(void){cdev_del(&hello_driver_dev.cdev);unregister_chrdev_region(hello_driver_dev.devid, HELLO_CNT); device_destroy(hello_driver_dev.class, hello_driver_dev.devid); class_destroy(hello_driver_dev.class);}module_init(hello_driver_init);module_exit(hello_driver_exit);MODULE_DESCRIPTION("my hello driver");MODULE_AUTHOR("Kong Lingze");MODULE_LICENSE("GPL");
在kernel/driver目錄下添加新的文件夾frist_driver,在frist_driver中添加文件Kconfig,Makefile,frist_driver.c
frist_driver/Kconfig中的內容:
frist_driver/Makefile中的內容:
在driver/Kconfig添加內容:
在driver/Makefile添加內容:
在kernel/driver下執行make menuconfig,選擇First Android Driver
在安卓源碼目錄目錄下source build/envsetup.sh
lunch xx xxxx (可選擇)
make bootimage -j32
執行fastboot flash boot boot.img將內核燒寫到開發板,在開發板目錄找到sys/class/hello driver/hello driver或者下圖目錄,執行如圖的命令
測試成功
首先在android源碼的packages目錄下新建一個helloapp目錄,該目錄下新建hello_app.c和Android.mk兩個文件。
#include <stdio.h>#include <unistd.h> #include <sys/types.h> #include <sys/stat.h>#include <fcntl.h>#include <stdlib.h> #include <string.h>#include <sys/ioctl.h>#define READ 3#define WRITE 4#define HELLO_ON 1#define HELLO_OFF 0// ./app /dev/hello\driver <cmd>static unsigned char cUserData[] = {"User data"};int main(int argc, char **argv){int iFd;int iRet;char *cFilename;unsigned int arg = 0;unsigned char cReadBuf[100];unsigned char cWriteBuf[100];if(argc!=3){Printf(“Usage:%s <file inode> <cmd>”,argv[0]);}//打開設備結點cFilename = argv[1];iFd = open(cFilename, O_RDWR);if(iFd < 0){printf(" open %s failed\n", cFilename);return -1;}/*讀取驅動中的數據*/if(atoi(argv[2]) == READ){memset(cReadBuf,sizeof(cReadBuf),sizeof(cReadBuf));iRet = read(iFd,cReadBuf,sizeof(cReadBuf));if(iRet < 0){printf("read %s data failed\n",cFilename);return -1;}else{printf("read data is:%s\n",cReadBuf);}}else if(atoi(argv[2]) == WRITE)//向驅動中寫數據{memset(cWriteBuf,sizeof(cWriteBuf),sizeof(cWriteBuf));memcpy(cWriteBuf,cUserData,sizeof(cUserData));iRet = write(iFd,cWriteBuf,sizeof(cWriteBuf));if(iRet < 0){printf("write %s data failed\n",cFilename);return -1;}}else if(atoi(argv[2]) == HELLO_ON)//給驅動發送HELLO_ON命令{ioctl(iFd,HELLO_ON,arg);}else if(atoi(argv[2]) == HELLO_OFF)//給驅動發送HELLO_OFF命令{ioctl(iFd,HELLO_OFF,arg);} iRet = close(iFd);if(iFd < 0){printf("close %s failed",cFilename);return -1;}return 0;}
LOCAL_PATH := $(call my-dir) include $(CLEAR_VARS) LOCAL_MODULE_TAGS := optional LOCAL_MODULE := hellotest LOCAL_SRC_FILES := $(call all-subdir-c-files) include $(BUILD_EXECUTABLE)
完成之后在安卓源碼目錄,輸入source build\envser.up.sh,再進入packages/helloapp目錄執行mm命令,生成可執行文件,編譯后的文件在out/target/product/xxx/obj/EXECUTABLES/hellotapp_intermediates/中
adb shell adb push hello_app /data把hello_app推送到android設備的data目錄下, chmod +x helloapp添加可執行權限。 執行./helloapp /dev/hello driver 1
打印如下:
Open hello driver
可見測試成功。
關于“Android怎么開發第一個驅動程序”這篇文章的內容就介紹到這里,感謝各位的閱讀!相信大家對“Android怎么開發第一個驅動程序”知識都有一定的了解,大家如果還想學習更多知識,歡迎關注億速云行業資訊頻道。
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。