内核模块:
int a , b;
安装内核模块时:insmod demo.ko a = 100 b =10;
在安装内核模块时给内核模块中的变量进行数值传递,这样可以让我们的内核模块向上兼容更为复杂的应用程序,向下适配多种硬件
1.函数原型:module_param(name, type, perm)
功能: 声明可以进行内核模块传参的变量
参数:name : 变量名
type:要传参的数值类型---》byte(传递char), hexint(16进制的int)、short 、int、uint、long、ulong、charp(传递字符指针)、bool(0/1 y/n Y/N)
perm:文件权限,通过module_param声明了要传参的变量,那么在sys/module/当前模块/parameters/下会生成一个以当前变量为名的文件,文件的权限为perm指定的权限,文件内容为变量的值
注:通过modinfo查看您当前内核模块可以进行命令行传参的变量有哪些
2.函数原型:MODULE_PARM_DESC(_parm, desc)
功能:添加要传参的变量的描述,这个描述也可以通过modinfo查看
参数:_parm :传参的变量名
desc:添加的描述
注:给char类型的变量传参时,需要传递对应的ASCII十进制形式
给字符指针传递字符串时,字符串不可以有空格,若有空格,空格前的被当作参数传递给变量,后面的会被认为时一个不认识的变量
示例代码


导出符号表可以让不同模块之间实现资源的相互访问 ,比如模块2想要访问模块1的资源,只需将模块1资源的符号表导出给模块2,此时,模块2就可以访问模块1的资源了
EXPORT_SYMBOL(变量名|函数名)
或者
EXPORT_SYMBOL_CPL(变量名|函数名)
定义demo1.c,完成函数的定义
- #include
- #include
- int add(int i,int j)
- {
- return i+j;
- }
- //生成add的符号表文件
- EXPORT_SYMBOL(add);
- static int __init mycdev_init(void)
- {
-
- return 0;
- }
- static void __exit mycdev_exit(void)
- {
-
-
- }
- module_init(mycdev_init);
- module_exit(mycdev_exit);
- MODULE_LICENSE("GPL");
定义dem2.c,完成调用demo1.c中的函数
- #include
- #include
-
- extern int add(int i,int j);
- static int __init mycdev_init(void)
- {
- printk("调用模块1函数执行结果为:%d",add(3,5));
- return 0;
- }
- static void __exit mycdev_exit(void)
- {
-
-
- }
- module_init(mycdev_init);
- module_exit(mycdev_exit);
- MODULE_LICENSE("GPL");
先编译demo1.c将生成的符号表文件Module.symvers复制到demo2的路径下再编译demo2.c

此时通过modinifo查看demo2.ko,显示demo2依赖于demo1

注意 :在高版本的内核中不支持符号表文件直接复制,不然编译报未定义错误
解决办法:在demo2的Makefile中加上demo1符号表的文件路径
3.3安装流程先安装demo1,再安装demo2

3.4卸载先卸载demo2,再卸载demo1
字符设备是以字节流的形式进行顺序访问的设备,针对字节设备设计的驱动框架叫做字符设备驱动。当前市面上绝大多数的设备都属于字符设备,比如键盘、鼠标、摄像头...

1.当在内核中注册一个字符设备驱动后会得到驱动对应的设备号,设备号是设备的唯一标识。设备号分为主设备号和次设备号,主设备号(高12位):用来标识一类设备 ,此设备号(低20位):用来标识这一类中的某一个设备。设备号=主设备号<<20 | 次设备号
2.根据设备号可以通过某种方式在文件系统中创建设备文件,相当于通过设备号将设备文件和驱动绑定
3.当设备文件被创建后,应用程序中调用 open() | read() | write() | close()
4.调用以上函数时,驱动中对应的操作应用方法会被回调
5.在驱动的操作方法中完成硬件的控制
头文件:#include
注册字符设备驱动
int register_chrdev(unsigned int major, const char *name, const struct file_operations *fops)
功能:实现字符设备驱动的注册,一次申请256个设备的资源(0-255个此设备号)
参数:major:主设备号
>0 静态指定主设备号
=0 动态申请主设备号
name:注册得到的驱动名字
fops:操作方法结构体指针,指向操作方法结构体变量
返回值:
失败返回错误码
成功:若major>0,则返回申请得到主设备号
若major=0,则返回0
操作方法结构体,用于保存和管理驱动的各自操作方法
struct file_operations {
int (*open) (struct inode *, struct file *);
ssize_t (*read) (struct file *, char __user *, size_t, loff_t *);
ssize_t (*write) (struct file *, const char __user *, size_t, loff_t *);
int (*release) (struct inode *, struct file *);
};
注销字符设备驱动
void unregister_chrdev(unsigned int major, const char *name)
功能:实现字符设备驱动的注销
参数:major:驱动对应的主设备号
name:注册时填写的驱动号
返回值:无
1.编写代码
- #include
- #include
- #include
- unsigned int major;
- // 封装操作方法,这些操作方法在应用层进行系统调用时被回调
- int mycdev_open(struct inode *inode, struct file *file)
- {
- printk("%s:%s:%d\n",__FILE__,__func__,__LINE__);
- return 0;
- }
- ssize_t mycdev_read(struct file *file, char *ubuf, size_t size, loff_t *lof)
- {
- printk("%s:%s:%d\n",__FILE__,__func__,__LINE__);
- return 0;
- }
- ssize_t mycdev_write(struct file *file, const char *ubuf, size_t size, loff_t *lof)
- {
- printk("%s:%s:%d\n",__FILE__,__func__,__LINE__);
- return 0;
- }
- int mycdev_close(struct inode *inode, struct file *file)
- {
- printk("%s:%s:%d\n",__FILE__,__func__,__LINE__);
- return 0;
- }
- // 定义操作方法结构体对象,保存封装的操作方法
- struct file_operations fops = {
- .open=mycdev_open,
- .read=mycdev_read,
- .write=mycdev_write,
- .release=mycdev_close,
- };
- //入口函数
- static int __init mycdev_init(void)
- {
- // 注册字符设备驱动
- major = register_chrdev(0, "mychrdev", &fops);
- if (major < 0)
- {
- printk("字符设备驱动注册失败\n");
- return major;
- }
- printk("注册字符设备驱动成功major=%d\n", major);
- return 0;
- }
- //出口函数
- static void __exit mycdev_exit(void)
- {
- //注销字符设备驱动
- unregister_chrdev(major,"mychrdev");
- }
- module_init(mycdev_init);
- module_exit(mycdev_exit);
- MODULE_LICENSE("GPL");
2.编译驱动
3.安装驱动内核模块
可以查看/proc/devices文件,确定驱动是否被注册
4.创建设备文件
创建设备文件的命令:
mknod 设备文件的名字和路径 设备文件类型 驱动的主设备号 次设备号
例:mknod /dev/mychrdev c 240 0
解析:mknod:创捷设备文件的命令码
/dev/mychrdev:创建的设备文件的路径和名字
c 设备文件类型(字符设备文件) d(块设备文件)
240:主设备号
0:次设备号 0-255都可以
5.编写应用程序代码测试是否可以关联驱动
- #include
- #include
- #include
- #include
- #include
- int main(int argc, char const *argv[])
- {
- char buf[128] = {0};
- int fd = open("/dev/mychrdev", O_RDWR);
- if (fd < 0)
- {
- printf("打开设备文件失败\n");
- return -1;
- }
- printf("打开设备文件成功\n");
- //调用read
- read(fd, buf, sizeof(buf));
- //调用write
- write(fd, buf, sizeof(buf));
- //调用close
- close(fd);
- return 0;
- }
6.现象
执行应用程序,驱动中的操作方法被回调
4.用户和内核的数据传递用户空间和内核空间之间无法直接相互访问内存,二者进行数据交互需要使用数据传递函数
头文件 #include
函数原型unsigned long copy_to_user(void __user *to, const void *from, unsigned long n)
功能:传递内核空间的数据到用户空间
参数:to:用户空间保存数据的buf首地址
from:内核空间保存数据的buf首地址
n:传递的数据长度,以字节为单位
返回值:成功返回0,失败返回未拷贝的字节数
函数原型unsigned log copy_from_user(void *to, const void __user *from, unsigned long n)
功能:传递用户空间的数据到内核空间
参数:to:内核空间保存数据的buf首地址
from:用户空间保存数据的buf首地址
n:传递的数据长度,以字节为单位
返回值:成功返回0,失败返回未拷贝的字节数
应用程序代码
- #include
- #include
- #include
- #include
- #include
- #include
- int main(int argc, char const *argv[])
- {
- char buf[128] = {0};
- int fd = open("/dev/mychrdev", O_RDWR);
- if (fd < 0)
- {
- printf("打开设备文件失败\n");
- return -1;
- }
- printf("打开设备文件成功\n");
- fgets(buf,sizeof(buf),stdin);//在终端读一个字符串
- buf[strlen(buf)-1]='\0';
-
- write(fd, buf, sizeof(buf));//将数据传递给内核
- memset(buf,0,sizeof(buf));//清空数组
- read(fd, buf, sizeof(buf));//将内核空间数据传递到用户
- printf("buf:%s\n",buf);
- close(fd);
- return 0;
- }
驱动程序代码
- #include
- #include
- #include
- #include
- unsigned int major;
- char kbuf[128]={};
- // 封装操作方法
- int mycdev_open(struct inode *inode, struct file *file)
- {
- printk("%s:%s:%d\n",__FILE__,__func__,__LINE__);
- return 0;
- }
- ssize_t mycdev_read(struct file *file, char *ubuf, size_t size, loff_t *lof)
- {
- printk("%s:%s:%d\n",__FILE__,__func__,__LINE__);
- int ret;
- //拷贝数据到用户空间
- ret=copy_to_user(ubuf,kbuf,size);
- if(ret)
- {
- printk("copy_to_user filed\n");
- return -EIO;
- }
- return 0;
- }
- ssize_t mycdev_write(struct file *file, const char *ubuf, size_t size, loff_t *lof)
- {
- printk("%s:%s:%d\n",__FILE__,__func__,__LINE__);
- int ret;
- //从用户空间拷贝数据到内核空间
- ret=copy_from_user(kbuf,ubuf,size);
- if(ret)
- {
- printk("copy_from_user filed\n");
- return -EIO;
- }
- return 0;
- }
- int mycdev_close(struct inode *inode, struct file *file)
- {
- printk("%s:%s:%d\n",__FILE__,__func__,__LINE__);
- return 0;
- }
- // 定义操作方法结构体对象
- struct file_operations fops = {
- .open=mycdev_open,
- .read=mycdev_read,
- .write=mycdev_write,
- .release=mycdev_close,
- };
- static int __init mycdev_init(void)
- {
- // 注册字符设备驱动
- major = register_chrdev(0, "mychrdev", &fops);
- if (major < 0)
- {
- printk("字符设备驱动注册失败\n");
- return major;
- }
- printk("注册字符设备驱动成功major=%d\n", major);
- return 0;
- }
- static void __exit mycdev_exit(void)
- {
- //注销字符设备驱动
- unregister_chrdev(major,"mychrdev");
- }
- module_init(mycdev_init);
- module_exit(mycdev_exit);
- MODULE_LICENSE("GPL");
现象:在终端输入的字符串可以正常的打印出来,说明用户空间和内核空间数据传递成功
驱动控制硬件需要操作硬件的特殊功能寄存器,但是特殊功能寄存器内存物属于理内存,驱动加载到虚拟内存,想要在驱动中操作硬件寄存器,需要将硬件寄存器内存映射为虚拟内存
#include
函数原型:void *ioremap(phys_addr_t paddr, unsigned long size)
功能:映射指定大小的物理内存为虚拟内存
参数:paddr:要映射的物理内存首地址
size:要映射的物理内存大小
返回值:成功,返回映射成功的虚拟内存首地址,失败,返回NULL
函数原型:void iounmap(const void __iomem *addr)
功能:取消物理内存的映射
参数:addr:要取消的虚拟内存首地址
返回值:无
注:有关字符设备驱动的示例代码,目前可查看上一篇文章"编写驱动代码控制LED灯亮灭"http://t.csdnimg.cn/ZBsgG