1. 程式人生 > 其它 >字元裝置驅動-Linux驅動學習(5)

字元裝置驅動-Linux驅動學習(5)

【學習筆記】

一、申請字元類裝置號

1、字元裝置和雜項裝置的區別

(1)裝置號的不同:雜項裝置的主裝置號是固定的,固定為10,而字元類裝置需要我們自己或者系統來給我們分配。

(2)裝置節點的生成方式不同:雜項裝置可以自動生成裝置節點,而字元裝置需要我們自己生成裝置節點。

2、兩種方法註冊字元類裝置號

(1)靜態分配裝置號

需要明確知道系統裡面哪些裝置號沒有被使用,然後手動分配。

函式定義在linux-4.9.268/include/linux/fs.h
extern int register_chrdev_region(dev_t, unsigned, const char *);
引數:
	第一個:裝置的起始值,型別是dev_t型別
	第二個:次裝置號的個數
	第三個:裝置的名稱

dev_t型別: dev_t是用來儲存裝置號的,是一個32位數
其中高12為用來儲存裝置號,低12為用來儲存次裝置號

dev_t定義在linux-4.9.268/include/linux/types.h裡邊
typedef __u32 __kernel_dev_t;
typedef __kernel_dev_t		dev_t;

Linux 提供了幾個巨集定義來操作裝置號

定義在linux-4.9.268/include/linux/kdev_t.h裡邊

#define MINORBITS	20	//提供了次裝置的位數,一共20位
#define MINORMASK	((1U << MINORBITS) - 1)//次裝置的掩碼

#define MAJOR(dev)	((unsigned int) ((dev) >> MINORBITS))//在dev_t裡面獲取主裝置號
#define MINOR(dev)	((unsigned int) ((dev) & MINORMASK))//在dev_t裡面獲取主次裝置號
#define MKDEV(ma,mi)	(((ma) << MINORBITS) | (mi))//將主裝置號和次裝置號組成dev_t型別(上面提到,dev_t型別是用來儲存裝置號的)
其中MKDEV(ma,mi)引數:
	ma:主裝置號
	mi:次裝置號
返回值:
	成功:返回0
	失敗:返回非零

(2)動態分配裝置號

這個函式同樣也定義在linux-4.9.268/include/linux/fs.h中
extern int alloc_chrdev_region(dev_t *, unsigned, unsigned, const char *);
引數:
	第一個:儲存生成的裝置號
	第二個:我們請求的第一個裝置號,通常是0
	第三個:連續申請的裝置號的個數
	第四個:裝置名稱
	
返回值:
	成功:返回0
	失敗:返回負數
使用動態分配會優先使用255~234裝置號

3、登出裝置號

這個函式同樣也定義在linux-4.9.268/include/linux/fs.h中
extern void unregister_chrdev_region(dev_t, unsigned);
引數:
	第一個:分配裝置號的起始地址
	第二個:申請的連續裝置號個數

例項操作:

chrdev.c

#include <linux/init.h>
#include <linux/module.h>
//註冊裝置號函式所在標頭檔案
#include <linux/fs.h>
//處理裝置號巨集定義所在標頭檔案
#include <linux/kdev_t.h>

//定義次裝置號個數
#define DEVICE_NUMBER 1
//次裝置號起始地址,通常為0
#define DEVICE_MINOR_NUMBER 0

//定義裝置名稱
#define DEVICE_SNAME "schrdev"	//靜態註冊
#define DEVICE_ANAME "achrdev"	//動態註冊


static int major_num,minor_num;

module_param(major_num,int,S_IRUSR);
module_param(minor_num,int,S_IRUSR);


static int hello_init(void){
	
	dev_t dev_num;

	int ret;//定義儲存函式返回值變數
	
	//靜態申請裝置號
	if(major_num){//判斷主裝置號有沒有傳遞進來,如果傳了引數,則使用靜態註冊方式,否則,使用動態註冊方式。

		//列印主次裝置號
		printk("major_num= %d\n",major_num);
		printk("minor_num= %d\n",minor_num);
		
		//組合主次裝置號
		dev_num = MKDEV(major_num, minor_num);
	
		//註冊裝置號函式,並儲存返回值
		ret = register_chrdev_region(dev_num, DEVICE_NUMBER, DEVICE_SNAME);

		if(ret < 0){//返回值<0,註冊失敗
			printk("register_chrdev_region error\n");
		}
		printk("register_chrdev_region successful\n");//否則說明註冊成功
	}
	else{//動態申請裝置號

		ret = alloc_chrdev_region(dev_num, DEVICE_MINOR_NUMBER, DEVICE_NUMBER, DEVICE_ANAME);

		if(ret < 0){//返回值<0,註冊失敗
			printk("register_chrdev_region error\n");
		}
		printk("register_chrdev_region successful\n");//否則說明註冊成功

		//使用巨集定義獲取裝置號
		major_num = MAJOR(dev_num);
		minor_num = MINOR(dev_num);

		//列印主次裝置號
		printk("major_num= %d\n",major_num);
		printk("minor_num= %d\n",minor_num);
	}



	return 0;
}

static void hello_exit(void){

	//登出裝置號
	unregister_chrdev_region(MKDEV(major_num, minor_num), DEVICE_NUMBER);
	printk("bye bye\n");
	
}

//入口和出口
module_init(hello_init);
module_exit(hello_exit);

//宣告許可證
MODULE_LICENSE("GPL");


在實際開發中,建議使用動態申請裝置號的方式,多人開發時,使用靜態申請很容易造成裝置號衝突。

二、註冊字元裝置

1、重要結構說明

cdev結構體:描述字元裝置的結構體

//它定義在linux-4.9.268/include/linux/cdev.h中
struct cdev {
	struct kobject kobj;
	struct module *owner;//說明模組所屬
	const struct file_operations *ops;//檔案操作集
	struct list_head list;//連結串列節點
	dev_t dev;//裝置號
	unsigned int count;//次裝置號的數量
};

2、操作步驟

(1)定義一個cdev結構體

(2)使用cdev_init函式初始化cdev結構體成員變數

void cdev_init(struct cdev *, const struct file_operations *){
	memset(cdev, 0, sizeof *cdev);
	INIT_LIST_HEAD(&cdev->list);
	kobject_init(&cdev->kobj, &ktype_cdev_default);
	
	cdev->ops = fops;//把檔案操作集寫給cdev的成員變數ops
}
引數:
	第一個:要初始化的cdev結構體指標
	第二個:檔案操作集
	

(3)使用cdev_add函式註冊字元裝置到核心

int cdev_add(struct cdev *, dev_t, unsigned);
引數:
	第一個:cdev的結構體指標
	第二個:裝置號
	第三個:次裝置號的數量

(4)登出字元裝置

void cdev_del(struct cdev *);

3、實際操作展示

直接在上面申請裝置號的程式碼修改,前面已有的程式碼註釋,下面不在寫,便於區別修改位置

chrdev.c

#include <linux/init.h>
#include <linux/module.h>
#include <linux/fs.h>
#include <linux/kdev_t.h>
//註冊字元裝置所在
#include <linux/cedv.h>


#define DEVICE_NUMBER 1
#define DEVICE_MINOR_NUMBER 0

#define DEVICE_SNAME "schrdev"
#define DEVICE_ANAME "achrdev"

static int major_num,minor_num;

//定義cdev結構體
struct cdev cdev;

module_param(major_num,int,S_IRUSR);
module_param(minor_num,int,S_IRUSR);

//應用層呼叫裝置節點,觸發的open函式
int chrdev_open(struct inode *inode, struct file *file){//(*open)函式實現
	printk("hello chrdev_open\n");
	return 0;
}

//定義檔案操作集
struct file_operations chrdev_ops = {
	.owner = THIS_MODULE,
	.open = chrdev_open
};

static int hello_init(void){
	
	dev_t dev_num;
	int ret;
	
	if(major_num){

		printk("major_num= %d\n",major_num);
		printk("minor_num= %d\n",minor_num);
		
		dev_num = MKDEV(major_num, minor_num);
	
		ret = register_chrdev_region(dev_num, DEVICE_NUMBER, DEVICE_SNAME);

		if(ret < 0){
			printk("register_chrdev_region error\n");
		}
		printk("register_chrdev_region successful\n");
	}
	else{
		ret = alloc_chrdev_region(dev_num, DEVICE_MINOR_NUMBER, DEVICE_NUMBER, DEVICE_ANAME);

		if(ret < 0){
			printk("register_chrdev_region error\n");
		}
		printk("register_chrdev_region successful\n");

		major_num = MAJOR(dev_num);
		minor_num = MINOR(dev_num);

		printk("major_num= %d\n",major_num);
		printk("minor_num= %d\n",minor_num);
	}


	cdev.owner = THIS_MODULE;//宣告所屬模組
	//初始化cdev結構體成員變數,第二個引數,要提前定義檔案操作集
	cdev_init(&cdev, chrdev_ops);
	//將字元設備註冊到核心
	cdev_add(&cdev, dev_num, DEVICE_NUMBER);

	return 0;
}

static void hello_exit(void){

	//登出裝置號
	unregister_chrdev_region(MKDEV(major_num, minor_num), DEVICE_NUMBER);
	
	//登出字元裝置(注意把它寫到註冊裝置號的下面,一個簡單的邏輯問題)
	void cdev_del(&cdev);
	printk("bye bye\n");
	
}

module_init(hello_init);
module_exit(hello_exit);

MODULE_LICENSE("GPL");



app.c(只保留開啟節點功能)

#include <stdio.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>

int main(int argc, char *argv[]){//如果開啟裝置節點成功,這會呼叫驅動裡邊的misc_open()函式

	int fd;
	
	char buf[64] = {0};

	fd = open("/dev/test",O_RDWR);//open the device node

	if(fd < 0){		//determine whether the opening is successful
	
		perror("open error\n");
		
		return fd;
	}

	//close(fd);//關閉節點
	
	return 0;
}


【注意】字元設備註冊完後並不會自動生成裝置節點,需要是哦那個mknod命令建立裝置節點

命令格式:

mknod [名稱] [型別] [主裝置號] [次裝置號]

例如:

mknod /dev/test c 247 0		//"dev/test"為app.c中定義的裝置節點名稱

三、自動建立裝置節點

當載入模組時,在/dev目錄下自動建立相應的裝置檔案

1、怎麼自動建立一個裝置節點

在嵌入式Linux中使用mdev來實現裝置節點檔案的自動建立和刪除

2、什麼是mdev

mdev是udev的簡化版本,是busybox中所帶的程式,最適合用在嵌入式系統

3、什麼是udev

udev是一種工具,它跟狗根據系統中的硬體裝置的狀態動態更新裝置檔案,包括裝置檔案的建立,刪除等。裝置檔案通常放在/dev目錄下。使用udev後,在/dev目錄下就只包含系統中真正存在的裝置,udev一般用在PC上的linux中,相對於mdev來說複製些。

4、怎麼建立裝置節點

自動建立裝置節點分為兩個步驟:

(1)使用class_create函式建立一個class的類

(2)使用device_create函式在我們建立的類下面建立一個裝置

5、建立和刪除類函式

在Linux驅動程式中一般通過兩個函式來完成裝置節點的建立和刪除。首先要建立一個class類結構體。

calss結構體定義在include/linux/device.h中。class_create是類建立函式,class_create是一個巨集定義,內容如下

#define class_create(owner, name)		\
({						\
	static struct lock_class_key __key;	\
	__class_create(owner, name, &__key);	\
})

class_create一共有兩個引數,引數owner一般為THIS_MODULE,引數name是類的名字。返回值是個指向結構體class的指標,也就是建立的類。

解除安裝驅動程式的時候需要刪除掉類,類刪除函式為class_destory,函式原型如下:

void class_destroy(struct class *cls);
//引數cls就是要刪除的類。

6、建立裝置函式

當使用上節點的函式建立完成一個類後,使用device_create函式在這個類下建立一個裝置device_create

函式原型如下:

//同樣定義在include/linux/device.h中
struct device *device_create_vargs(struct class *cls, struct device *parent,
				   dev_t devt, void *drvdata,
				   const char *fmt, va_list vargs);
引數說明:
device_create 是個可變引數函式
class:裝置要建立在哪個類下面
parent:父裝置,一般為NULL,也就是沒有父裝置
devt:裝置號
drvdata:是裝置可能會使用的一些資料,一般為NULL
fmt:是裝置名字,如果設定fmt=xxx的話,就會生成/dev/xxx這個裝置檔案
返回值就是建立號的裝置

整理自嵌入式學習之Linux驅動篇