本文共 1021 字,大约阅读时间需要 3 分钟。
在前面讲过,驱动层是应用层是分离,驱动层的代码不能使用再应用层,应用层也不能直接操作驱动代码,那么应用层和驱动层之间是如何来实现数据间的交换的能,方法就是通过相应的接口函数。
(1)copy_from_user
1 2 3 4 5 6 7 8 9 | unsigned long copy_from_user( void *to, const void __user *from, unsigned long n) { might_sleep(); if (access_ok(VERIFY_READ, from, n)) n = __copy_from_user(to, from, n); else memset (to, 0, n); return n; } |
这个函数是从用户空间拷贝数据到内核空间,失败返回没有被拷贝的字节数,成功返回0,注意用户空间的数据不能直接通过memcpy复制到内核空间,原因是内核空间和用户空间的地址不在同一个映射区域里面。内核空间和用户空间的内存是不能直接访问的。
(2)copy_to_user
1 2 3 4 5 6 7 8 | unsigned long copy_to_user( void __user *to, const void *from, unsigned long n) { might_sleep(); BUG_ON(( long ) n < 0); if (access_ok(VERIFY_WRITE, to, n)) n = __copy_to_user(to, from, n); return n; } |
返回值和copy_from_user一样,成功返回0,失败返回没有拷贝成功的字节数。
参数to有个__user限定,这个在~/include/linux/compiler.h中有如下定义:
# define __user __attribute__((noderef, address_space(1)))
表示这是一个用户空间的地址,即其指向的为用户空间的内存。
__attribute__是gnu c编译器的一个功能,它用来让开发者使用此功能给所声明的函数或者变量附加一个属性,以方便编译器进行错误检查,其实就是一个内核检查器。
以上两个函数参考
本文转自 菜鸟养成记 51CTO博客,原文链接:http://blog.51cto.com/11674570/1872418
转载地址:http://jrylx.baihongyu.com/