基于Amlogic 安卓9.0, 驱动简说:字符设备驱动,手动创建设备

Posted 阿迷创客

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了基于Amlogic 安卓9.0, 驱动简说:字符设备驱动,手动创建设备相关的知识,希望对你有一定的参考价值。

文章目录

一、前言

  • 本文提取书籍中,较为经典的例子。
  • 让驱动的学习,变得更加简单。
  • 目前市面上,很难找到一本讲得特别好的嵌入式驱动开发教程,尤其是适配android 平台的。LDD3是一本翻译较好,写得较好的书,但其内容基于Linux kernel 2.6版本,接口和架构都较老了。国产书籍普遍较为片面,翻译的书籍则表述较差(机器翻译),于是乎,看完之后,想总结一下!

本文基于Amlogic T972 , Android 9.0, 内核版本 4.9.113

二、系列文章

第1篇:基于Amlogic 安卓9.0, 驱动简说(一):字符设备驱动,手动创建设备
第2篇:基于Amlogic 安卓9.0, 驱动简说(二):字符设备驱动,自动创建设备
第3篇:基于Amlogic 安卓9.0, 驱动简说(三):使用misc框架,让驱动更简单
第4篇:基于Amlogic 安卓9.0, 驱动简说(四):Platform平台驱动,驱动与设备的分离


三、解析:完整源码

提要:
(1)主设备号用于对设备进行分类
(2)此设备号代表具体的设备,第N个

1. helloworld_amlogic_char_driver.c

  • 缺点1:手动分配主设备号
  • 缺点2:需要手动创建设备文件
  • 文件,参考位置:android9.0\\common\\drivers\\amlogic\\input\\helloworld_amlogic_char_driver.c
/* 模块初始化、卸载的接口头文件  */
#include <linux/module.h>

/* 字符设备头文件 */
#include <linux/cdev.h>
#include <linux/fs.h>

/* 定义主设备号 */
#define MY_MAJOR_NUM	202

/* 定义设备的私有结构体,因为简单,直接使用struct cdev,
如果要支持同个相同的设备,则一般需要为每个设备分配一个结构体,
代表对应的设备 */
static struct cdev aml_cdev;

/* 应用层系统调用:open()、fopen 打开设备节点文件时,将回调此函数  */
static int aml_cdev_open(struct inode *inode, struct file *file)

	pr_info("aml_cdev_open() is called.\\n");
	return 0;


/* 应用层系统调用:close()、fclose 关闭设备节点文件时,将回调此函数  */
static int aml_cdev_close(struct inode *inode, struct file *file)

	pr_info("aml_cdev_close() is called.\\n");
	return 0;


/*  应用层系统调用:ioctrl() 操作设备节点文件时,将回调此函数  */
static long aml_cdev_ioctl(struct file *file, unsigned int cmd, unsigned long arg)

	pr_info("aml_cdev_ioctl() is called. cmd = %d, arg = %ld\\n", cmd, arg);
	return 0;


/* 注册回调函数:只需要实现需要的接口即可 */
static const struct file_operations aml_cdev_fops = 
	.owner = THIS_MODULE,
	.open = aml_cdev_open,
	.release = aml_cdev_close,
	.unlocked_ioctl = aml_cdev_ioctl,
;

/* 模块初始化函数  */
static int __init aml_cdev_init(void)

	int ret;
	/* 手动分配一个主设备号、次设备号,需要避开系统已使用的号码*/
	dev_t dev = MKDEV(MY_MAJOR_NUM, 0);
	pr_info("aml_cdev_init \\n");

	/* 注册并获取此设备号 */
	ret = register_chrdev_region(dev, 1, "aml_char_device");
	if (ret < 0)
		pr_info("Unable to allocate mayor number %d\\n", MY_MAJOR_NUM);
		return ret;
	

	/* 初始化aml_cdev,并将其添加内核中*/
	cdev_init(&aml_cdev, &aml_cdev_fops);
	ret= cdev_add(&aml_cdev, dev, 1);
	if (ret < 0)
		unregister_chrdev_region(dev, 1);
		pr_info("Unable to add cdev\\n");
		return ret;
	

	return 0;


/* 模块退出时执行卸载操作  */
static void __exit aml_cdev_exit(void)

	pr_info("aml_cdev_exit\\n");
	cdev_del(&aml_cdev);//删除设备
	unregister_chrdev_region(MKDEV(MY_MAJOR_NUM, 0), 1); //注销设备号



/* 固定的模板部分,将导出模块的初始化和卸载接口符号 */
module_init(aml_cdev_init);
module_exit(aml_cdev_exit);

MODULE_LICENSE("Dual BSD/GPL");
MODULE_AUTHOR("szhou <66176468@qq.com>");
MODULE_DESCRIPTION("This is a amlogic debug tool.");

2. Makefile

  • android9.0\\common\\drivers\\amlogic\\input\\Makefile
#
# Makefile for the input core drivers.
#

# Each configuration option enables a list of files.

obj-$(CONFIG_AMLOGIC_AVIN_DETECT) += avin_detect/

obj-$(CONFIG_AMLOGIC_INPUT_KEYBOARD) += keyboard/

obj-$(CONFIG_AMLOGIC_REMOTE) += remote/

obj-$(CONFIG_AMLOGIC_TOUCHSCREEN) += touchscreen/

obj-$(CONFIG_AMLOGIC_SENSOR) += sensor/

# 添加下面语句,并将helloworld_amlogic_char_driver.c放在同一目录下
obj-m +=  helloworld_amlogic_char_driver.o


四、编译执行

4.1 编译

  • (1)执行安卓编译命令,全部或局部编译kernel
  • (2)模块生成位置:out/target/product/x301/obj/KERNEL_OBJ/drivers/amlogic/input/helloworld_amlogic_char_driver.ko
make: Leaving directory `/home/builder/android_x301/source/t962x3-t972-android9.0/out/target/product/x301/obj/KERNEL_OBJ'
[ 75% 3/4] Kernel installed
 Prebuilt:  (out/target/product/x301/kernel)
[100% 4/4] Target boot image: out/target/product/x301/boot.img

#### build completed successfully (05:59 (mm:ss)) ####

Build kernel ok!
root@d185403d1e6f:/home/builder/android_x301/source/t962x3-t972-android9.0# ./build.sh kernel

4.2 执行

(1)部署

  • 启动一个windows 命令行串口:ctrl + r , 输入 cmd
>adb usb
restarting in USB mode

>adb root
restarting adbd as root

>adb push helloworld_amlogic_char_driver.ko /data/
helloworld_amlogic_char_driver.ko: 1 file pushed, 0 skipped. 5.1 MB/s (101656 bytes in 0.019s)

(2)加载ko文件

>adb shell
x301:/ #
x301:/ # insmod /data/helloworld_amlogic_char_driver.ko

(3)查看结果

  • cat /proc/devicecs 可以看到已分配的设备节点,及其节点设备名称
  • 如下所示,已按楼上驱动,分配了主设备号:202
x301:/ # cat /proc/devices  | grep 202
202 aml_char_device
x301:/ #

(4)是否有设备文件了呢?

  • 创建的设备节点文件,会被放置在设备文件系统下,即/dev目录下
  • 通过命令查找,无论是文件名,主设备号,可以发现/dev目录下没有对应驱动的设备存在
  • 原因:我们只是注册了驱动,尚未创建设备
x301:/dev # ls -al  | grep 202
x301:/dev #
x301:/dev # ls -l aml_*
crw------- 1 root root 243,   0 2018-01-01 08:00 aml_demod
crw------- 1 root root 242,   0 2018-01-01 08:00 aml_demod_ui
x301:/dev #

4.3 手动创建设备

  • 在板子上,执行mknod命令创建设备节点
  • 路径:/dev
  • 文件名:aml_char ,这个可以随意
  • 主设备号:202, 必须和驱动声明的主设备号一致,否则驱动不知道你是谁
  • 次设备号:0, 一般从0开始编号
1|x301:/dev # mknod /dev/aml_char c 202 0
x301:/dev #

如此,我们就创建了和驱动能匹配的设备文件,应用层的程序通过系统调用,open(“/dev/aml_char”, O_RDWR),既可以以可读写方式打开设备。

五、应用层调用

5.1 源码

  • hello_aml.c
#include <stdio.h>

int main()



  int buf[4]=0;
  int fd=0;

  //(1) 测试 open
  fd = open("/dev/aml_char", O_RDWR);
  if(fd == -1)
	printf("Failed: open /dev/aml_char \\n");
	return 0;
  

  //(2) 测试 ioctl
  ioctl(fd, 0x12, buf); 

  //(3) 测试 clsoe	
  close(fd);
  return 0;



5.2 测试结果

  • 打印默认是输出到串口上,所以此处是在串口上执行命令
  • 使用adb shell的同学,可以执行后,输入dmesg查看打印
:/ # ./data/hello_aml 
[11843.579654@2]- aml_cdev_open() is called.
[11843.579734@2]- aml_cdev_ioctl() is called. cmd = 18, arg = -1148368336
[11843.579744@2]- aml_cdev_close() is called.
:/ # 

效果如下图:

六、源码下载

需要尝试的同学,可从下面地址获取

  • 地址:https://gitee.com/amizhou/amlogic_t972_android9_driver/tree/master/index_01_simpleSay_drivers/lesson_01
git clone git@gitee.com:amizhou/amlogic_t972_android9_driver.git

七、篇尾

保持持续学习, 欢迎私信交流。

以上是关于基于Amlogic 安卓9.0, 驱动简说:字符设备驱动,手动创建设备的主要内容,如果未能解决你的问题,请参考以下文章

基于Amlogic 安卓9.0, 驱动简说:字符设备驱动,自动创建设备

基于Amlogic 安卓9.0, 驱动简说:Platform平台驱动,驱动与设备的分离

基于Amlogic 安卓9.0, 驱动简说:Platform平台驱动,驱动与设备的分离

基于Amlogic 安卓9.0, 驱动简说:使用misc框架,让驱动更简单

基于Amlogic 安卓9.0, 驱动简说:使用misc框架,让驱动更简单

基于Amlogic 安卓9.0, 驱动简说:基于GPIOLED子系统的LED驱动