从零开始写linux字符设备驱动程序(一)(基于友善之臂tiny4412开发

简介: 从零开始写linux字符设备驱动程序(一)(基于友善之臂tiny4412开发

从这篇博文开始,我将开始手把手教会大家写Linux设备驱动程序

这是开篇,如何来写第一个字符设备驱动程序。


首先,写一个最简单的字符设备驱动程序需要什么?或者说我们需要了解什么?

1、每一个字符设备至少需要有一个设备号

2、设备号 = 主设备号 + 次设备号

3、同一类设备的主设备号一般是相同的,但不是绝对的。


那么,写一个简单的字符设备驱动程序,我们需要内核里的这几个头文件,因为我们需要调用一个基本的宏和一些基本的函数来给我们使用。

#include <linux/cdev.h>
#include <linux/kdev_t.h>
#include <linux/fs.h>

打开linux内核源代码,进入include/linux/,找到cdev.h,打开,我们会看到这个结构体:

struct cdev {
  //设备模型相关的
  struct kobject kobj;
  //所属于哪个模块--->THIS MODULE
  struct module *owner;
  //利用file_operations跟用户态进行操作--->有open , read , write 等方法
  const struct file_operations *ops;
  //链表,将设备插入到一条链表里去
  struct list_head list;
  //通过设备号匹配对应的驱动
  dev_t dev;
  //要注册字符设备的个数
  unsigned int count;
};

还会看到以下的函数:

void cdev_init(struct cdev *, const struct file_operations *);
struct cdev *cdev_alloc(void);
void cdev_put(struct cdev *p);
int cdev_add(struct cdev *, dev_t, unsigned);
void cdev_del(struct cdev *);
void cd_forget(struct inode *);

这里我们需要的就是以上的这个结构体,还有cdev_init,cdev_add,cdev_del这三个函数,其余的暂时用不着。本节暂时不会用到以上的函数,下节将会使用。


然后看到#include 这个头文件,这里面有我们需要的东西:

#define MINORBITS 20
#define MINORMASK ((1U << MINORBITS) - 1)
//从设备号中取出主设备号
#define MAJOR(dev)  ((unsigned int) ((dev) >> MINORBITS))
//从设备号中取出次设备号
#define MINOR(dev)  ((unsigned int) ((dev) & MINORMASK))
//创建一个设备号
#define MKDEV(ma,mi)  (((ma) << MINORBITS) | (mi))

我们在接下来写的这个字符设备就需要创建一个设备号,所以我们需要MKDEV这个宏,第一个参数表示主设备号,第二个参数表示次设备号。


我们知道如何去创建一个设备号,那么创建了设备号,还没有对这个设备进行注册,这时候就需要#include 这个头文件里的一个函数:

extern int register_chrdev_region(dev_t, unsigned, const char *);

既然有注册,当然就有释放,所以还需要:

extern void unregister_chrdev_region(dev_t, unsigned);

好了,有了这些基本知识,可以开始我们的第一个字符设备驱动程序的编写。


编写这个简单的字符设备需要以下步骤:

1、创建设备号

2、注册设备号

3、如何驱动模块退出的时候,我们需要注销设备的操作。


好了,开始写代码:

#include <linux/init.h>
#include <linux/module.h>
#include <linux/sched.h>
#include <linux/kernel.h>
#include <linux/cdev.h>
#include <linux/kdev_t.h>
#include <linux/fs.h>
//定义一个结构体变量,用来表示设备号--->cdev.h--->dev_t
dev_t dev_no ;
static int __init  cdev_test_init(void)
{
  int ret ;
  printk("HELLO KERNEL FOR CDEV!\n");
  //1、创建设备号-->第一个是主设备号,第二个是次设备号
  //主设备号可以通过cat /proc/devices查看,如果设备号已经被占用,则需要使用没有使用过的设备号
  dev_no  = MKDEV(222,2);
  //2、注册设备号
  //count表示要分配多少个设备号
  ret = register_chrdev_region(dev_no,1,"my_dev");
  if(ret < 0){
    //如果注册失败,跳转到对应的位置。
    goto register_error ;
  }
  return 0 ;
  register_error: 
  return ret ;
}
static int __exit cdev_test_exit(void)
{
  //注销驱动-->后面写1表示从dev_no开始连续一个设备
  unregister_chrdev_region(dev_no,1);
  return 0 ;
}
module_init(cdev_test_init);
module_exit(cdev_test_exit);
MODULE_LICENSE("GPL");
再和以前一样,写一个Kconfig和Makefile
Kconfig
menu "4412_CDEV_DRV"
     config CDEV_TEST
          bool "cdev_test"
          default n
      help
    if you select , you can use it
endmenu
Makefile
obj-y += cdev_test.o

再到上层的driver目录下Kconfig和Makefile中添加相应的语句,跟以往一样这里是在driver目录下创建了一个4char_dev的目录。

接下来在内核根目录下make menuconfig配置相应的驱动:

640.jpg

640.jpg

将编译生成的zImage下载至开发板,打开串口调试,会看到以下log,说明驱动已经开始运行了:

640.jpg

接下来通过adb shell进入安卓系统的根目录下:

cat /proc/devices

我们成功的看到主设备号222的字符设备驱动my_dev已经成功装载了。

640.jpg

目录
相关文章
|
4月前
|
安全 Linux Shell
Linux上执行内存中的脚本和程序
【9月更文挑战第3天】在 Linux 系统中,可以通过多种方式执行内存中的脚本和程序:一是使用 `eval` 命令直接执行内存中的脚本内容;二是利用管道将脚本内容传递给 `bash` 解释器执行;三是将编译好的程序复制到 `/dev/shm` 并执行。这些方法虽便捷,但也需谨慎操作以避免安全风险。
239 6
|
5月前
|
网络协议 Linux
Linux查看端口监听情况,以及Linux查看某个端口对应的进程号和程序
Linux查看端口监听情况,以及Linux查看某个端口对应的进程号和程序
740 2
|
5月前
|
Linux Python
linux上根据运行程序的进程号,查看程序所在的绝对路径。linux查看进程启动的时间
linux上根据运行程序的进程号,查看程序所在的绝对路径。linux查看进程启动的时间
79 2
|
2月前
|
Linux 开发工具 Perl
在Linux中,有一个文件,如何删除包含“www“字样的字符?
在Linux中,如果你想删除一个文件中包含特定字样(如“www”)的所有字符或行,你可以使用多种文本处理工具来实现。以下是一些常见的方法:
44 5
|
3月前
|
Linux 开发工具 Perl
Linux命令替换目录下所有文件里有"\n"的字符为""如何操作?
【10月更文挑战第20天】Linux命令替换目录下所有文件里有"\n"的字符为""如何操作?
53 4
|
3月前
|
Linux API 开发工具
FFmpeg开发笔记(五十九)Linux编译ijkplayer的Android平台so库
ijkplayer是由B站研发的移动端播放器,基于FFmpeg 3.4,支持Android和iOS。其源码托管于GitHub,截至2024年9月15日,获得了3.24万星标和0.81万分支,尽管已停止更新6年。本文档介绍了如何在Linux环境下编译ijkplayer的so库,以便在较新的开发环境中使用。首先需安装编译工具并调整/tmp分区大小,接着下载并安装Android SDK和NDK,最后下载ijkplayer源码并编译。详细步骤包括环境准备、工具安装及库编译等。更多FFmpeg开发知识可参考相关书籍。
119 0
FFmpeg开发笔记(五十九)Linux编译ijkplayer的Android平台so库
|
3月前
|
运维 Java Linux
【运维基础知识】Linux服务器下手写启停Java程序脚本start.sh stop.sh及详细说明
### 启动Java程序脚本 `start.sh` 此脚本用于启动一个Java程序,设置JVM字符集为GBK,最大堆内存为3000M,并将程序的日志输出到`output.log`文件中,同时在后台运行。 ### 停止Java程序脚本 `stop.sh` 此脚本用于停止指定名称的服务(如`QuoteServer`),通过查找并终止该服务的Java进程,输出操作结果以确认是否成功。
88 1
|
4月前
|
消息中间件 分布式计算 Java
Linux环境下 java程序提交spark任务到Yarn报错
Linux环境下 java程序提交spark任务到Yarn报错
54 5
|
4月前
|
Linux 程序员 编译器
Linux内核驱动程序接口 【ChatGPT】
Linux内核驱动程序接口 【ChatGPT】
|
4月前
|
存储 Linux 开发工具
如何进行Linux内核开发【ChatGPT】
如何进行Linux内核开发【ChatGPT】

热门文章

最新文章