【Linux 驱动基础】第一个驱动

# 前置知识

        APP 打开文件时,可以得到一个整数,这个整数被称为文件句柄。对于 APP的每一个文件句柄,在内核里面都有一个“ struct file”与之对应。

  • 使用open函数时,用户态调用 API 触发异常进入内核
  • 内核识别异常后,取出异常值,从而调用sys_call_table[__NR_open],sys_call_table就是一个数组,其值为typedef void (*sys_call_ptr_t)(void)类型的函数指针,内容大致如下:
__SYSCALL_COMMON(0, sys_read, sys_read)
__SYSCALL_COMMON(1, sys_write, sys_write)
__SYSCALL_COMMON(2, sys_open, sys_open)
__SYSCALL_COMMON(3, sys_close, sys_close)
__SYSCALL_COMMON(5, sys_newfstat, sys_newfstat)
...
...
...
  •  最后调用sys_open函数,sys_open函数定义如下:
SYSCALL_DEFINE3(open, const char __user *, filename, int, flags, int, mode)
{
	long ret;
 
	if (force_o_largefile())
		flags |= O_LARGEFILE;
 
	ret = do_sys_open(AT_FDCWD, filename, flags, mode);
	/* avoid REGPARM breakage on x86: */
	asmlinkage_protect(3, ret, filename, flags, mode);
	return ret;
}
  • 在sys_open函数里面,又调用了 de_sys_open 函数,定义如下:
 
long do_sys_open(int dfd, const char __user *filename, int flags, int mode)
{
	char *tmp = getname(filename);
	int fd = PTR_ERR(tmp);
 
	if (!IS_ERR(tmp)) {
		fd = get_unused_fd_flags(flags);        /* 获取未使用的函数句柄 */
		if (fd >= 0) {
			struct file *f = do_filp_open(dfd, tmp, flags, mode, 0);    /* 打开文件,得到file结构体 */
			if (IS_ERR(f)) {
				put_unused_fd(fd);
				fd = PTR_ERR(f);
			} else {
				fsnotify_open(f);
				fd_install(fd, f);    /* 在当前进程里面记录下来 */
			}
		}
		putname(tmp);
	}
	return fd;
}
  • 通过以上过程,就绑定了文件句柄fd与file结构体,后续操作文件句柄fd,实际上就是操作文件结构体file,文件结构体file如下,在结构体中,有f_op,里面包含了读写等对文件的操作;有f_flags,这就是使用open函数时的打开模式;有f_pos,这就是使用读写函数时,指向文件的位置,整体情况如下图:

 # 驱动编写流程

         从上述可以得出,当我们使用openwritereadclose等函数操作驱动时,最终调用的是最底层file_operations结构体中的openwritereadclose函数。那么将这些函数完善,并且上报到系统,就是编写驱动程序。

#include <linux/module.h>
#include <asm/uaccess.h>
#include <linux/fs.h>
#include <linux/errno.h>
#include <linux/miscdevice.h>
#include <linux/kernel.h>
#include <linux/major.h>
#include <linux/mutex.h>
#include <linux/proc_fs.h>
#include <linux/seq_file.h>
#include <linux/stat.h>
#include <linux/init.h>
#include <linux/device.h>
#include <linux/tty.h>
#include <linux/kmod.h>
#include <linux/gfp.h>

#define __CHRDEV_REGION_

#define MIN(a, b) (a < b ? a : b)

#define DEVICE_DRIVER_NAME      "test_drv"      /* 设备驱动名称 */
#define DEVICE_DRIVER_CNT       1               /* 设备号个数 */

static char kernal_buffer[1024];

typedef struct {
    dev_t devid;			    /* 设备号 	 */
    int major;                  /* 主设备号 */
    int minor;				    /* 次设备号   */    

    struct cdev cdev;		    /* cdev 	*/
    char class_name[20];        /* 类名称 */
    struct class *class;		/* 类 */
    struct device *device;	    /* 设备 */
}test_drv_Typedef;

static test_drv_Typedef test_driver = {
    .devid = 0,
    .major = 0,
    .minor = 0,
    .class_name = "test_class",
    .class = NULL,
    .device = NULL
};

static int test_drv_open(struct inode *node, struct file *fle)
{
    printk("test_drv_open\r\n");

    return 0;
}

static int test_drv_release(struct inode *node, struct file *fle)
{
    printk("test_drv_open\r\n");

    return 0;
}

static ssize_t test_drv_write(struct file *fle, const char __user *buf, size_t size, loff_t *offest)
{
    ssize_t err;

    printk("test_drv_write\r\n");

    err = copy_from_user(kernal_buffer, buf, MIN(sizeof(kernal_buffer), size));

    return err;
}

static ssize_t test_drv_read(struct file *fle, char __user *buf, size_t size, loff_t *offest)
{
    ssize_t err;

    printk("test_drv_read\r\n");

    err = copy_to_user(buf, kernal_buffer, MIN(sizeof(kernal_buffer), size));

    return err;
}

/* 2. 编写 file_operations 结构体 */
static const struct file_operations test_drv_op = {
    .owner = THIS_MODULE,
    .open = test_drv_open,
    .release = test_drv_release,
    .write = test_drv_write,
    .read = test_drv_read,
};

/* 3. 编写字符设备驱动模块加载 */
/* 3.1 创建设备号、注册字符设备驱动 */
/* 3.2 创建一个class类型的对象 */
/* 3.3 生成设备节点并导出到用户空间 */
static int __init test_drv_init(void)
{
    int ret = 0;

    printk("test_drv_init\r\n");

    /* 创建设备号 */
#ifndef __CHRDEV_REGION_
    /* 静态和动态注册两种方法,主要是通过判断给定的主设备号是否为0来进行区别 */
    test_driver.major = register_chrdev(0, DEVICE_DRIVER_NAME, &test_drv_op);
    test_driver.devid = MKDEV(test_driver.major, test_driver.minor);
#else
    if(test_driver.major)   /* 静态注册 */
    {
        test_driver.devid = MKDEV(test_driver.major, test_driver.minor);
        ret = register_chrdev_region(test_driver.devid, DEVICE_DRIVER_CNT, DEVICE_DRIVER_NAME);
        if(ret < 0)
        {
            printk("register_chrdev_region fail\r\n");
            goto _err_out_regchrdev;
        }
    }
    else    /* 动态注册 */
    {
        ret = alloc_chrdev_region(&test_driver.devid, 0, DEVICE_DRIVER_CNT, DEVICE_DRIVER_NAME);
        if(ret < 0)
        {
            printk("alloc_chrdev_region fail\r\n");
            goto _err_out_regchrdev;
        }

        test_driver.major = MAJOR(test_driver.devid);
        test_driver.minor = MINOR(test_driver.devid);
    }

    /* 初始化cdev */
    test_driver.cdev.owner = THIS_MODULE;
    cdev_init(&test_driver.cdev, &test_drv_op);

    /* 添加一个cdev */
    cdev_add(&test_driver.cdev, test_driver.devid, DEVICE_DRIVER_CNT);
#endif

    test_driver.class = class_create(THIS_MODULE, test_driver.class_name);
    if (IS_ERR(test_driver.class)) 
    {
        printk("class_create fail\r\n");
        ret = PTR_ERR(test_driver.class);
        goto _err_out_class_crate;
	}

    test_driver.device = device_create(test_driver.class, NULL, test_driver.devid, NULL, DEVICE_DRIVER_NAME);
    if(IS_ERR(test_driver.device))
    {
        printk("device_create fail\r\n");
        ret = PTR_ERR(test_driver.device);
        goto _err_out_device_create;
    }

    return 0;

_err_out_device_create:
    class_destroy(test_driver.class);
_err_out_class_crate:
#ifndef __CHRDEV_REGION_
    unregister_chrdev(test_driver.devid, DEVICE_DRIVER_NAME);
#else
    unregister_chrdev_region(test_driver.devid, DEVICE_DRIVER_CNT);
    cdev_del(&test_driver.cdev);
_err_out_regchrdev:
#endif 
    return ret;
}

/* 4. 编写字符设备驱动模块卸载*/
/* 4.1 摧毁设备 */
/* 4.2 摧毁类 */
/* 4.3 卸载字符设备驱动 */
static void __exit test_drv_exit(void)
{
    printk("test_drv_exit\r\n");

    device_destroy(test_driver.class, test_driver.devid);
    class_destroy(test_driver.class);
#ifndef __CHRDEV_REGION_
    unregister_chrdev(test_driver.major, DEVICE_DRIVER_NAME);
#else
    unregister_chrdev_region(test_driver.devid, DEVICE_DRIVER_CNT);
    cdev_del(&test_driver.cdev);
#endif 
}

/* 5. 指定驱动出入口 */
module_init(test_drv_init);
module_exit(test_drv_exit);

/* 6. 完善驱动信息 */
MODULE_LICENSE("GPL");  //添加模块 LICENSE 信息
MODULE_AUTHOR("JYW");   //添加模块作者信息

步骤1:确定主设备号和次设备号

        设备号的组成:

        为了方便管理, Linux 中每个设备都有一个设备号,设备号由主设备号和次设备号两部分组成,主设备号表示某一个具体的驱动,次设备号表示使用这个驱动的各个设备,Linux 提供了一个名为 dev_t 的数据类型表示设备号, dev_t 定义在文件 include/linux/types.h 里面,定义如下:

typedef __u32 __kernel_dev_t;
typedef __kernel_dev_t		dev_t;

        dev_t 其实就是 unsigned int 类型,是一个 32 位的数据类型。这 32 位的数据构成了主设备号和次设备号两部分,其中高 12 位为主设备号, 低 20 位为次设备号

        设备号的操作函数: 

/* 从 dev_t 中获取主设备号 */
#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))

步骤2:编写 file_operations 结构体

        file_operations 的结构体就是 Linux 内核驱动操作函数集合,内容如下所示:

struct file_operations {
	struct module *owner;															/* 拥有该结构体的模块的指针,一般设置为 THIS_MODULE */
	loff_t (*llseek) (struct file *, loff_t, int);									/* 修改文件当前的读写位置 */
	ssize_t (*read) (struct file *, char __user *, size_t, loff_t *);				/* 读取设备文件 */
	ssize_t (*write) (struct file *, const char __user *, size_t, loff_t *);		/* 向设备文件写入(发送)数据 */
	ssize_t (*read_iter) (struct kiocb *, struct iov_iter *);
	ssize_t (*write_iter) (struct kiocb *, struct iov_iter *);
	int (*iterate) (struct file *, struct dir_context *);
	unsigned int (*poll) (struct file *, struct poll_table_struct *);				/* 轮询函数,用于查询设备是否可以进行非阻塞的读写 */
	long (*unlocked_ioctl) (struct file *, unsigned int, unsigned long);			/* 提供对于设备的控制功能,与应用程序中的 ioctl 函数对应 */
	long (*compat_ioctl) (struct file *, unsigned int, unsigned long);				/* 与unlocked_ioctl 函数功能一样 */
	int (*mmap) (struct file *, struct vm_area_struct *);							/* 将设备的内存映射到进程空间中(也就是用户空间) */
	int (*mremap)(struct file *, struct vm_area_struct *);
	int (*open) (struct inode *, struct file *);									/* 打开设备文件 */
	int (*flush) (struct file *, fl_owner_t id);
	int (*release) (struct inode *, struct file *);									/* 释放(关闭)设备文件 */
	int (*fsync) (struct file *, loff_t, loff_t, int datasync);						/* 刷新待处理的数据,用于将缓冲区中的数据刷新到磁盘中 */
	int (*aio_fsync) (struct kiocb *, int datasync);								/* 与 fasync 函数的功能类似,只是 aio_fsync 是异步刷新待处理的数据 */
	int (*fasync) (int, struct file *, int);
	int (*lock) (struct file *, int, struct file_lock *);
	ssize_t (*sendpage) (struct file *, struct page *, int, size_t, loff_t *, int);
	unsigned long (*get_unmapped_area)(struct file *, unsigned long, unsigned long, unsigned long, unsigned long);
	int (*check_flags)(int);
	int (*flock) (struct file *, int, struct file_lock *);
	ssize_t (*splice_write)(struct pipe_inode_info *, struct file *, loff_t *, size_t, unsigned int);
	ssize_t (*splice_read)(struct file *, loff_t *, struct pipe_inode_info *, size_t, unsigned int);
	int (*setlease)(struct file *, long, struct file_lock **, void **);
	long (*fallocate)(struct file *file, int mode, loff_t offset,
			  loff_t len);
	void (*show_fdinfo)(struct seq_file *m, struct file *f);
#ifndef CONFIG_MMU
	unsigned (*mmap_capabilities)(struct file *);
#endif
};

        常用函数介绍:


/*
* @description : 打开设备
* @param – : 传递给驱动的 inode
* @param - : 设备文件, file 结构体有个叫做 private_data 的成员变量
* 一般在 open 的时候将 private_data 指向设备结构体。
* @return : 0 成功;其他 失败
*/
int (*open) (struct inode *, struct file *);
/*
* @description : 关闭/释放设备
* @param - : 要关闭的设备文件(文件描述符)
* @return : 0 成功;其他 失败
*/
int (*release) (struct inode *, struct file *);	
/*
* @description : 向设备写数据
* @param - : 设备文件,表示打开的文件描述符
* @param - : 要写给设备写入的数据
* @param - : 要写入的数据长度
* @param - : 相对于文件首地址的偏移
* @return : 写入的字节数,如果为负值,表示写入失败
*/
ssize_t (*write) (struct file *, const char __user *, size_t, loff_t *);
/*
* @description : 从设备读取数据
* @param - : 要打开的设备文件(文件描述符)
* @param - : 返回给用户空间的数据缓冲区
* @param - : 要读取的数据长度
* @param - : 相对于文件首地址的偏移
* @return : 读取的字节数,如果为负值,表示读取失败
*/
ssize_t (*read) (struct file *, char __user *, size_t, loff_t *);
unsigned long copy_from_user(void *to, const void __user *from, unsigned long n)

unsigned long copy_to_user(void *to, const void __user *from, unsigned long n)

以下是对参数进行讲解:

  • struct inode
/*
* 内核使用inode结构体在内核内部表示一个文件。因此,它与表示一个已经打开的文件描述符的结构体(file
* 文件结构)是不同的,我们可以使用多个file 文件结构表示同一个文件的多个文件描述符,但此时,所有这
* 些file文件结构全部都必须只能指向一个inode结构体
*/
struct inode {
	umode_t			i_mode;				
	unsigned short		i_opflags;
	kuid_t			i_uid;				/* inode拥有者id   */
	kgid_t			i_gid;				/* inode所属群组id  */
	unsigned int		i_flags;

#ifdef CONFIG_FS_POSIX_ACL
	struct posix_acl	*i_acl;
	struct posix_acl	*i_default_acl;
#endif

	const struct inode_operations	*i_op;
	struct super_block	*i_sb;
	struct address_space	*i_mapping;

#ifdef CONFIG_SECURITY
	void			*i_security;
#endif

	/* Stat data, not accessed from path walking */
	unsigned long		i_ino;
	/*
	 * Filesystems may only read i_nlink directly.  They shall use the
	 * following functions for modification:
	 *
	 *    (set|clear|inc|drop)_nlink
	 *    inode_(inc|dec)_link_count
	 */
	union {
		const unsigned int i_nlink;
		unsigned int __i_nlink;
	};
	dev_t			i_rdev;			/* 若是设备文件,表示记录设备的设备号 */
	loff_t			i_size;			/* inode所代表大少  */
	struct timespec		i_atime;	/* inode最近一次的存取时间   */
	struct timespec		i_mtime;	/* inode最近一次修改时间  */
	struct timespec		i_ctime;	/* inode的生成时间 */
	spinlock_t		i_lock;	/* i_blocks, i_bytes, maybe i_size */
	unsigned short          i_bytes;
	unsigned int		i_blkbits;
	blkcnt_t		i_blocks;

#ifdef __NEED_I_SIZE_ORDERED
	seqcount_t		i_size_seqcount;
#endif

	/* Misc */
	unsigned long		i_state;
	struct mutex		i_mutex;

	unsigned long		dirtied_when;	/* jiffies of first dirtying */
	unsigned long		dirtied_time_when;

	struct hlist_node	i_hash;
	struct list_head	i_wb_list;	/* backing dev IO list */
	struct list_head	i_lru;		/* inode LRU list */
	struct list_head	i_sb_list;
	union {
		struct hlist_head	i_dentry;
		struct rcu_head		i_rcu;
	};
	u64			i_version;
	atomic_t		i_count;
	atomic_t		i_dio_count;
	atomic_t		i_writecount;
#ifdef CONFIG_IMA
	atomic_t		i_readcount; /* struct files open RO */
#endif
	const struct file_operations	*i_fop;	/* former ->i_op->default_file_ops */
	struct file_lock_context	*i_flctx;
	struct address_space	i_data;
	struct list_head	i_devices;
	union {
		struct pipe_inode_info	*i_pipe;
		struct block_device	*i_bdev;
		struct cdev		*i_cdev;		/* 若是字符设备,对应的为cdev结构  */
	};

	__u32			i_generation;

#ifdef CONFIG_FSNOTIFY
	__u32			i_fsnotify_mask; /* all events this inode cares about */
	struct hlist_head	i_fsnotify_marks;
#endif

	void			*i_private; /* fs or device private pointer */
};

/* 内核函数从inode中提取设备号 */

/* 提取主设备号 */
static inline unsigned imajor(const struct inode *inode){  return MAJOR(inode->i_rdev);
}
/* 提取次设备号 */
static inline unsigned iminor(const struct inode *inode)
{
  return MINOR(inode->i_rdev);
}
  • struct file
/*
* file结构体指示一个已经打开的文件(设备对应于设备文件),其实系统中的每个打开的文件在内核空间都有
* 一个相应的struct file结构体,它由内核在打开文件时创建,并传递给在文件上进行操作的任何函数,直至文
* 件被关闭。如果文件被关闭,内核就会释放相应的数据结构
*/
struct file {
	union {
		struct llist_node	fu_llist;
		struct rcu_head 	fu_rcuhead;
	} f_u;
	struct path		f_path;
	struct inode		*f_inode;	/* cached value */
	const struct file_operations	*f_op;	/*   与文件相关的各种操作 */

	/*
	 * Protects f_ep_links, f_flags.
	 * Must not be taken from IRQ context.
	 */
	spinlock_t		f_lock;
	atomic_long_t		f_count;
	unsigned int 		f_flags;		/* 文件标志,如 O_RDONLY ,  O_NONBLOCK 以及 O_SYNC  */
	fmode_t			f_mode;				/* 此文件模式通过 FMODE_READ ,  FMODE_WRITE 识别了文件为可读的,可写的,或者是二者 */
	struct mutex		f_pos_lock;
	loff_t			f_pos;				/*  当前读写文件的位置 */
	struct fown_struct	f_owner;
	const struct cred	*f_cred;
	struct file_ra_state	f_ra;

	u64			f_version;
#ifdef CONFIG_SECURITY
	void			*f_security;
#endif
	/* needed for tty driver, and maybe others */
	void			*private_data;		/*  在驱动调用open方法之前,open系统调用设置此指针为NULL值。你可以很自由的将其做为你自己需要的一些数据域或者不管它 */

#ifdef CONFIG_EPOLL
	/* Used by fs/eventpoll.c to link all the hooks to this file */
	struct list_head	f_ep_links;
	struct list_head	f_tfile_llink;
#endif /* #ifdef CONFIG_EPOLL */
	struct address_space	*f_mapping;
} __attribute__((aligned(4)));	/* lest something weird decides that 2 is OK */

步骤3: 编写字符设备驱动模块加载 

步骤3.1:创建并注册设备号

  • register_chrdev
/**
 * register_chrdev() - create and register a cdev occupying a range of minors
 * @major: major device number or 0 for dynamic allocation
 * @name: name of this range of devices
 * @fops: file operations associated with this devices
 *
 * If @major == 0 this functions will dynamically allocate a major and return
 * its number.
 *
 * If @major > 0 this function will attempt to reserve a device with the given
 * major number and will return zero on success.
 *
 * Returns a -ve errno on failure.
 *
 * The name of this device has nothing to do with the name of the device in
 * /dev. It only helps to keep track of the different owners of devices. If
 * your module name has only one type of devices it's ok to use e.g. the name
 * of the module here.
 */
static inline int register_chrdev(unsigned int major, const char *name, const struct file_operations *fops)

/**
 * unregister_chrdev - unregister and destroy a cdev
 * @major: major device number
 * @name: name of this range of devices
 *
 * Unregister and destroy the cdev occupying the region described by
 * @major, @baseminor and @count.  This function undoes what
 * register_chrdev() did.
 */
static inline void unregister_chrdev(unsigned int major, const char *name)
  • register_chrdev_region/alloc_chrdev_region + cdev
/**
 * register_chrdev_region() - register a range of device numbers
 * @from: the first in the desired range of device numbers; must include
 *        the major number.
 * @count: the number of consecutive device numbers required
 * @name: the name of the device or driver.
 *
 * Return value is zero on success, a negative error code on failure.
 */
int register_chrdev_region(dev_t from, unsigned count, const char *name)

/**
 * alloc_chrdev_region() - register a range of char device numbers
 * @dev: output parameter for first assigned number
 * @baseminor: first of the requested range of minor numbers
 * @count: the number of minor numbers required
 * @name: the name of the associated device or driver
 *
 * Allocates a range of char device numbers.  The major number will be
 * chosen dynamically, and returned (along with the first minor number)
 * in @dev.  Returns zero or a negative error code.
 */
int alloc_chrdev_region(dev_t *dev, unsigned baseminor, unsigned count, const char *name)

/**
 * unregister_chrdev_region() - return a range of device numbers
 * @from: the first in the range of numbers to unregister
 * @count: the number of device numbers to unregister
 *
 * This function will unregister a range of @count device numbers,
 * starting with @from.  The caller should normally be the one who
 * allocated those numbers in the first place...
 */
void unregister_chrdev_region(dev_t from, unsigned count)

/**
 * cdev_alloc() - allocate a cdev structure
 *
 * Allocates and returns a cdev structure, or NULL on failure.
 */
struct cdev *cdev_alloc(void)

/**
 * cdev_init() - initialize a cdev structure
 * @cdev: the structure to initialize
 * @fops: the file_operations for this device
 *
 * Initializes @cdev, remembering @fops, making it ready to add to the
 * system with cdev_add().
 */
void cdev_init(struct cdev *cdev, const struct file_operations *fops)

/**
 * cdev_add() - add a char device to the system
 * @p: the cdev structure for the device
 * @dev: the first device number for which this device is responsible
 * @count: the number of consecutive minor numbers corresponding to this
 *         device
 *
 * cdev_add() adds the device represented by @p to the system, making it
 * live immediately.  A negative error code is returned on failure.
 */
int cdev_add(struct cdev *p, dev_t dev, unsigned count)

/**
 * cdev_del() - remove a cdev from the system
 * @p: the cdev structure to be removed
 *
 * cdev_del() removes @p from the system, possibly freeing the structure
 * itself.
 */
void cdev_del(struct cdev *p)

步骤3.2:创建一个class类型的对象

/* This is a #define to keep the compiler from merging different
 * instances of the __key variable */
#define class_create(owner, name)		\
({						\
	static struct lock_class_key __key;	\
	__class_create(owner, name, &__key);	\
})

/**
 * class_create - create a struct class structure
 * @owner: pointer to the module that is to "own" this struct class
 * @name: pointer to a string for the name of this class.
 * @key: the lock_class_key for this class; used by mutex lock debugging
 *
 * This is used to create a struct class pointer that can then be used
 * in calls to device_create().
 *
 * Returns &struct class pointer on success, or ERR_PTR() on error.
 *
 * Note, the pointer created here is to be destroyed when finished by
 * making a call to class_destroy().
 */
struct class *__class_create(struct module *owner, const char *name, struct lock_class_key *key)

/**
 * class_destroy - destroys a struct class structure
 * @cls: pointer to the struct class that is to be destroyed
 *
 * Note, the pointer to be destroyed must have been created with a call
 * to class_create().
 */
void class_destroy(struct class *cls)

步骤3.3:生成设备节点并导出到用户空间

/**
 * device_create- 创建设备
 * @class: 该设备依附的类
 * @parent: 父设备
 * @devt: 设备号(此处的设备号为主次设备号)
 * @drvdata:私有数据
 * @fmt:设备名
 * return:创建的设备指针
*/
struct device *device_create(struct class *class, struct device *parent, dev_t devt, void *drvdata, const char *fmt, ...)


/**
 * device_create- 销毁设备
 * @class: 该设备依附的类
 * @devt: 设备号(此处的设备号为主次设备号)
*/
void device_destroy(struct class *class, dev_t devt)

步骤4: 编写字符设备驱动模块卸载

          查看步骤3

步骤5:指定驱动出入口

/**
 * module_init() - driver initialization entry point
 * @x: function to be run at kernel boot time or module insertion
 * 
 * module_init() will either be called during do_initcalls() (if
 * builtin) or at module insertion time (if a module).  There can only
 * be one per module.
 */
#define module_init(x)	__initcall(x);

/**
 * module_exit() - driver exit entry point
 * @x: function to be run when driver is removed
 * 
 * module_exit() will wrap the driver clean-up code
 * with cleanup_module() when used with rmmod when
 * the driver is a module.  If the driver is statically
 * compiled into the kernel, module_exit() has no effect.
 * There can only be one per module.
 */
#define module_exit(x)	__exitcall(x);

步骤6:完善驱动信息

/*
 * The following license idents are currently accepted as indicating free
 * software modules
 *
 *	"GPL"				[GNU Public License v2 or later]
 *	"GPL v2"			[GNU Public License v2]
 *	"GPL and additional rights"	[GNU Public License v2 rights and more]
 *	"Dual BSD/GPL"			[GNU Public License v2
 *					 or BSD license choice]
 *	"Dual MIT/GPL"			[GNU Public License v2
 *					 or MIT license choice]
 *	"Dual MPL/GPL"			[GNU Public License v2
 *					 or Mozilla license choice]
 *
 * The following other idents are available
 *
 *	"Proprietary"			[Non free products]
 *
 * There are dual licensed components, but when running with Linux it is the
 * GPL that is relevant so this is a non issue. Similarly LGPL linked with GPL
 * is a GPL combined work.
 *
 * This exists for several reasons
 * 1.	So modinfo can show license info for users wanting to vet their setup
 *	is free
 * 2.	So the community can ignore bug reports including proprietary modules
 * 3.	So vendors can do likewise based on their own policies
 */
#define MODULE_LICENSE(_license) MODULE_INFO(license, _license)

/*
 * Author(s), use "Name <email>" or just "Name", for multiple
 * authors use multiple MODULE_AUTHOR() statements/lines.
 */
#define MODULE_AUTHOR(_author) MODULE_INFO(author, _author)

/* What your module does. */
#define MODULE_DESCRIPTION(_description) MODULE_INFO(description, _description)

# 补充知识

1. 加载和卸载驱动模块

  • 加载模块

        驱动编译完成以后扩展名为.ko,有两种命令可以加载驱动模块: insmodmodprobe,其中insmod 命令不能解决模块的依赖关系,比如 drv.ko 依赖 first.ko 这个模块,就必须先使用insmod 命令加载 first.ko 这个模块,然后再加载 drv.ko 这个模块。但是 modprobe 就不会存在这个问题, modprobe 会分析模块的依赖关系,然后会将所有的依赖模块都加载到内核中,因此modprobe 命令相比 insmod 要智能一些。 modprobe 命令主要智能在提供了模块的依赖性分析、错误检查、错误报告等功能,modprobe 命令默认会去/lib/modules/<kernel-version>目录中查找模块,比如本书使用的 Linux kernel 的版本号为 4.1.15,因此 modprobe 命令默认会到/lib/modules/4.1.15 这个目录中查找相应的驱动模块,一般自己制作的根文件系统中是不会有这个目录的,所以需要自己手动创建

insmod drv.ko
modprobe drv.ko

  • 卸载模块

        驱动模块的卸载使用命令“rmmod”即可,也可以使用“modprobe -r”命令卸载驱动

rmmod drv.ko
modprobe -r drv.ko 

  • 查看模块

        输入“lsmod”命令即可查看当前系统中存在的模块 

2. 查看驱动设备

  • 使用 cat /proc/devices 可以查看当前已经被使用掉的设备号

  • 使用  ls /sys/class/ 查看类

  • 使用 ls /sys/class/test_class/ -l 查看设备

3. 查看日志信息 

  •  printk

        printk 运行在内核态。在内核中想要向控制台输出或显示一些内容,必须使用printk 这个函数。不同之处在于, printk 可以根据日志级别对消息进行分类,一共有 8 个消息级别,这 8 个消息级别定义在文件 include/linux/kern_levels.h 里面,定义如下:

#define KERN_SOH "\001"
#define KERN_EMERG KERN_SOH "0" /* 紧急事件,一般是内核崩溃 */
#define KERN_ALERT KERN_SOH "1" /* 必须立即采取行动 */
#define KERN_CRIT KERN_SOH "2" /* 临界条件,比如严重的软件或硬件错误*/
#define KERN_ERR KERN_SOH "3" /* 错误状态,一般设备驱动程序中使用
KERN_ERR 报告硬件错误 */
#define KERN_WARNING KERN_SOH "4" /* 警告信息,不会对系统造成严重影响 */
#define KERN_NOTICE KERN_SOH "5" /* 有必要进行提示的一些信息 */
#define KERN_INFO KERN_SOH "6" /* 提示性的信息 */
#define KERN_DEBUG KERN_SOH "7" /* 调试信息 */

一共定义了 8 个级别,其中 0 的优先级最高, 7 的优先级最低。如果要设置消息级别,参考如下示例:printk(KERN_EMERG "gsmi: Log Shutdown Reason\n");用 printk 的 时 候 不 显 式 的 设 置 消 息 级 别 , 那 么 printk 将 会 采 用 默 认 级 别MESSAGE_LOGLEVEL_DEFAULT, MESSAGE_LOGLEVEL_DEFAULT 默认为 4

  • printk等级

使用 # cat /proc/sys/kernel/printk  指令查看printk等级

        四个数字分表代表:DEFAULT_CONSOLE_LOGLEVEL(控制台输出等级)、DEFAULT_MESSAGE_LOGLEVEL(默认消息等级)、MINIMUM_CONSOLE_LOGLEVEL(最低输出等级)、DEFAULT_CONSOLE_LOGLEVEL(默认控制台输出等级) 

        当 printk() 中的消息日志级别小于当前控制台日志级别(CONSOLE_LOGLEVEL_DEFAULT)时,printk 的信息就会在控制台上显示,如果我们用 printk 的 时 候 不 显 式 的 设 置 消 息 级 别,即printk使用MESSAGE_LOGLEVEL_DEFAULT(级别为4),此时当前控制台日志级别为CONSOLE_LOGLEVEL_DEFAULT(级别为7),所示直接使用 printk 输出的信息是可以显示在控制台上的。

  • 查看日志

        第一种是使用dmesg命令打印。第二种是通过cat /proc/kmsg来打印

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/480168.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

【理解机器学习算法】之Clustering算法(Agglomerative Clustering)

聚合聚类(Agglomerative Clustering)是一种层次聚类算法&#xff0c;通过逐步合并或“聚集”它们来构建嵌套聚类。这种方法采用自底向上的方式构建聚类层次&#xff1a;它从将每个数据点作为单个聚类开始&#xff0c;然后迭代合并最接近的聚类对&#xff0c;直到所有数据点合并…

学习添加03(优惠卷)

1.优化卷模块的介绍 整体流程&#xff1a; 优惠卷表设计&#xff1a; 优惠卷范围表设计&#xff1a; 兑换码表设计&#xff1a;

【嵌入式——QT】Charts常见的图表的绘制

【嵌入式——QT】Charts常见的图表的绘制 柱状图QBarSetQBarSeriesQBarCategoryAxis图示 饼图堆叠柱状图百分比柱状图散点图和光滑曲线图代码示例 柱状图 QBarSet 用于创建柱状图的数据集。 主要函数 setLabel()&#xff1a;设置数据集标签 &#xff1b;setLabelBrush()&am…

解决arco-design下拉框回显id的问题

问题描述 下拉框回显选项中没有的选项&#xff0c;就会出现以下情况&#xff0c;只能把uid回显上去 解决方案 使用ui框架自带的属性fallback-option 用法 按以上操作&#xff0c;即可解决选择框回显uid问题

【数据结构】——排序之冒泡排序

&#x1f49e;&#x1f49e; 前言 hello hello~ &#xff0c;这里是大耳朵土土垚~&#x1f496;&#x1f496; &#xff0c;欢迎大家点赞&#x1f973;&#x1f973;关注&#x1f4a5;&#x1f4a5;收藏&#x1f339;&#x1f339;&#x1f339; &#x1f4a5;个人主页&#x…

使用CSS3画出一个叮当猫HTML源码

我们经常使用PS或者Flash制作动画&#xff0c;本文则介绍了如何用CSS3画出个叮当猫&#xff0c;实现过程很有趣&#xff0c;感兴趣的朋友可以参考一下 首先&#xff0c;先把HTML结构搭建好&#xff1a; <div class"wrapper"> <!--叮当猫整体--> <di…

【3GPP】【核心网】【4G】4G手机接入过程,手机附着过程(超详细)

1. 4G手机接入过程&#xff0c;手机附着过程 附着&#xff08;Attach&#xff09;&#xff1a; 终端在PLMN中注册&#xff0c;从而建立自己的档案&#xff0c;即终端上下文 进行附着的三种情况&#xff1a; ①终端开机后的附着&#xff0c;初始附着 ②终端从覆盖盲区返回到…

Day41:WEB攻防-ASP应用HTTP.SYS短文件文件解析Access注入数据库泄漏

目录 ASP-默认安装-MDB数据库泄漏下载 ASP-中间件-CVE&短文件&解析&写权限 HTTP.SYS&#xff08;CVE-2015-1635&#xff09;主要用作蓝屏破坏&#xff0c;跟权限不挂钩 IIS短文件(iis全版本都可能有这个问题) IIS文件解析 IIS写权限 ASP-SQL注入-SQLMAP使用…

基于python+vue的OA公文发文管理系统flask-django-php-nodejs

系统根据现有的管理模块进行开发和扩展&#xff0c;采用面向对象的开发的思想和结构化的开发方法对OA公文发文管理的现状进行系统调查。采用结构化的分析设计&#xff0c;该方法要求结合一定的图表&#xff0c;在模块化的基础上进行系统的开发工作。在设计中采用“自下而上”的…

数据挖掘与分析学习笔记

一、Numpy NumPy&#xff08;Numerical Python&#xff09;是一种开源的Python库&#xff0c;专注于数值计算和处理多维数组。它是Python数据科学和机器学习生态系统的基础工具包之一&#xff0c;因为它高效地实现了向量化计算&#xff0c;并提供了对大型多维数组和矩阵的支持…

鸿蒙预览报错 Only files in a module can be previewed

HarmonyOS第一课下载的源码无法运行&#xff0c;也无法预览&#xff0c;报错如题。 解决&#xff1a; 1、在预览页如“index.ets”文件下预览。 2、如果在通知栏看到如图提示&#xff0c;可看出是ohos/hvigor-ohos-plugin插件版本的问题&#xff0c;可点击蓝色解决方案同步并导…

01-Spark的Local模式与应用开发入门

1 Spark 的 local 模式 Spark 运行模式之一&#xff0c;用于在本地机器上单机模拟分布式计算的环境。在 local 模式下&#xff0c;Spark 会使用单个 JVM 进程来模拟分布式集群行为&#xff0c;所有 Spark 组件&#xff08;如 SparkContext、Executor 等&#xff09;都运行在同…

获取淘宝商品评论的爬虫技术分享(已封装API,可测试)

item_review-获得淘宝商品评论 公共参数 请求地址: taobao/item_review 名称类型必须描述keyString是调用key&#xff08;必须以GET方式拼接在URL中&#xff09;secretString是调用密钥api_nameString是API接口名称&#xff08;包括在请求地址中&#xff09;[item_search,it…

基于python+vue网络相册设计与实现flask-django-nodejs-php

网络相册设计与实现的目的是让使用者可以更方便的将人、设备和场景更立体的连接在一起。能让用户以更科幻的方式使用产品&#xff0c;体验高科技时代带给人们的方便&#xff0c;同时也能让用户体会到与以往常规产品不同的体验风格。 与安卓&#xff0c;iOS相比较起来&#xff0…

python 函数(解包**、互相调用、作用域、函数的封装、内置函数:eval()、zip()、文件处理open())

函数解包 """ 1、函数的注释&#xff1a;参数和返回值 在注释里可以自动添加显示&#xff0c;只需手动加说明。2、函数的解包【拆包】&#xff1a;函数的参数要传递数据有多个值的时候&#xff0c;中间步骤拿到数据 保存在元组或者列表 或者字典里。 - 传递参数…

机器学习-06-无监督算法-01-划分聚类Kmeans算法

总结 本系列是机器学习课程的系列课程&#xff0c;主要介绍机器学习中无监督算法&#xff0c;包括划分聚类等。 参考 数据分析实战 | K-means算法——蛋白质消费特征分析 欧洲48国英文名称的来龙去脉及其国旗动画 Kmeans在线动态演示 本门课程的目标 完成一个特定行业的…

【C#】使用C#窗体应用开启/停止Apache、MySQL服务

目录 一、前言 二、效果图 三、配置文件 四、代码 五、一键启动/停止所有服务 一、前言 使用C#窗体应用开启Apache、MySQL服务&#xff0c;不仅仅是Apache、MySQL&#xff0c;其他服务也可以使用同样的方法操作&#xff0c;包括开启自己写的脚本服务。 二、效果图 两种状…

短视频矩阵系统--技术实际开发打板3年真实开发分享

短视频矩阵系统--技术实际开发打板3年真实开发分享&#xff0c;短视频矩阵系统/矩阵获客系统是一种基于短视频平台的获客游戏。短视频矩阵系统可以通过多账号发布来替代传统的单账号游戏。可以一键发布所有账号&#xff0c;批量制作多个视频AI智能剪辑。过去很多人只能完成的工…

新版仿蓝奏网盘|城通网盘|百度网盘|闪客网盘|网盘源码系统,个人网盘系统

(购买本专栏可免费下载栏目内所有资源不受限制,持续发布中,需要注意的是,本专栏为批量下载专用,并无法保证某款源码或者插件绝对可用,介意不要购买!购买本专栏住如有什么源码需要,可向博主私信,第二天即可发布!博主有几万资源) 这是一款仿蓝奏网盘、城通网盘、百…

利用CSS3实现正在加载效果

一、代码区域 1.1css3代码 <style>* {padding: 0;margin: 0;list-style: none;}.loading {width: 300px;height: 100px;margin: 100px auto;}.loading ul {height: 100px;width: 65px;margin: 0 auto;display: flex;align-items: center;}.loading ul li {margin: 0 5p…