Linux中国 Linux中国门户站!
设为主页 设为主页
收藏本站 收藏本站
 
当前位置 :首页 ->Linux技术 ->内核研究 ->正文

使用 Linux 系统调用的内核命令

来源:linux.chinaunix.net 作者:M. Tim Jones  时间:2007-04-22 点击: [收藏] [投稿]

内核 jiffies

Linux 内核具有一个名为 jiffies 的全局变量,它代表从机器启动时算起的时间滴答数。这个变量最初被初始化为 0,每次时钟中断时都会加 1。您可以使用 get_jiffies_64 函数来读取 jiffies 的值,然后使用 jiffies_to_msecs 将其换算成毫秒或使用 jiffies_to_usecs 将其换算成微秒。jiffies 的全局定义和相关函数是在 ./linux/include/linux/jiffies.h 中提供的。

清单 2 给出了第三个函数。这个函数使用了两个参数:一个 long 类型,以及一个指向被定义为 __userlong 的指针。__user 宏简单告诉编译器(通过 noderef)不应该解除这个指针的引用(因为在当前地址空间中它是没有意义的)。这个函数会计算这两个 jiffies 值之间的差值,然后通过一个用户空间指针将结果提供给用户。put_user 函数将结果值放入 presult 所指定的用户空间位置。如果在这个操作过程中出现错误,将立即返回,您也可以通知用户空间调用者。

对于步骤 2 来说,我对头文件进行了更新:在系统调用表中为这几个新函数安排空间。对于本例来说,我使用新系统调用号更新了 linux/include/asm/unistd.h 头文件。更新如清单 3 中的黑体所示。


清单 3. 更新 unistd.h 文件为新系统调用安排空间

                
#define __NR_getcpu		318
#define __NR_epoll_pwait	319
#define __NR_getjiffies		320
                #define __NR_diffjiffies	321
                #define __NR_pdiffjiffies	322
                #define NR_syscalls	323
            

现在已经有了自己的内核系统调用,以及表示这些系统调用的编号。接下来需要做的是要在这些编号(表索引)和函数本身之间建立一种对等关系。这就是第 3 个步骤,更新系统调用表。如清单 4 所示,我将为这个新函数更新 linux/arch/i386/kernel/syscall_table.S 文件,它会填充清单 3 显示的特定索引。


清单 4. 使用新函数更新系统调用表

                
.long sys_getcpu
.long sys_epoll_pwait
.long sys_getjiffies		/* 320 */
.long sys_diffjiffies
                .long sys_pdiffjiffies
            

注意: 这个表的大小是由符号常量 NR_syscalls 定义的。

现在,我们已经完成了对内核的更新。接下来必须对内核重新进行编译,并在测试用户空间应用程序之前使引导使用的新映像变为可用。

对用户内存进行读写

Linux 内核提供了几个函数,可以用来将系统调用参数移动到用户空间中,或从中移出。方法包括一些基本类型的简单函数(例如 get_userput_user)。要移动一块儿数据(如结构或数组),您可以使用另外一组函数: copy_from_usercopy_to_user。可以使用专门的调用移动以 null 结尾的字符串: strncpy_from_userstrlen_from_user。您也可以通过调用 access_ok 来测试用户空间指针是否有效。这些函数都是在 linux/include/asm/uaccess.h 中定义的。

您可以使用 access_ok 宏来验证给定操作的用户空间指针。这个函数有 3 个参数,分别是访问类型(VERIFY_READVERIFY_WRITE),指向用户空间内存块的指针,以及块的大小(单位为字节)。如果成功,这个函数就返回 0:

int access_ok( type, address, size );

要在内核和用户空间移动一些简单类型(例如 int 或 long 类型),可以使用 get_userput_user 轻松地实现。这两个宏都包含一个值以及一个指向变量的指针。get_user 函数将用户空间地址(ptr)指定的值移动到所指定的内核变量(var)中。 put_user 函数则将内核变量(var)指定的值移动到用户空间地址(ptr)。 如果成功,这两个函数都返回 0:

int get_user( var, ptr );
int put_user( var, ptr );

要移动更大的对象,例如结构或数组,您可以使用 copy_from_usercopy_to_user 函数。这些函数将在用户空间和内核之间移动完整的数据块。 copy_from_user 函数会将一块数据从用户空间移动到内核空间,copy_to_user 则会将一块数据从内核空间移动到用户空间:



 如果您对本文有任何疑问或者建议,请到讨论区发表您的意见: >> 论坛入口 <<

上一页 1 23 4 下一页


上一篇:Linux 和对称多处理   下一篇:没有了

文章评论】 【收藏本文】 【推荐好友】 【打印本文】 【我要投稿】 【论坛讨论
更多相关文章
·使用 Linux 系统调用的内核命令
·Linux 和对称多处理
·Relay:一种高效的数据传输技术
·使用 SystemTap 调试内核
·配置FreeBSD内核
·在FreeBSD4.x中制作启动菜单
·制作软盘上运行的FreeBSD系统
·FreeBSD光盘运行版的制作过程
·制作RAMDISK in KERNEL的NetBSD
·制作RAMDISK in KERNEL的OpenBSD
推荐文章
·制作RAMDISK in KERNEL的OpenBSD
·Linux系统应用程序和内核模块的区别
·使用GDB与QEMU调试内核时的问题分析
·Linus谈调试器和内核怎么样发展
·Linux 2.4 到 Linux 2.6内核升级指
·系统技术开发新篇章:编译linux内核
·2.4.22-1内核架设openmosix集群
·剖析Linux 2.6内核移植-内核设置篇
精彩文章
·奇妙的 sys 请求
·Linux创始人:内核臭虫太多,需喘口
·Linux操作系统的内核编译内幕详解
·Linus谈调试器和内核怎么样发展
·利用异常表处理Linux内核态缺页异常
·使用异步 I/O 大大提高应用程序的性
·Linux内核开发:使用Git管理源代码
·开源技术评论:Unix内核的“新生命
·Linux内核开发:使用 Git 管理源代
·剖析Linux 2.6内核移植—系统移植篇
·基于Linux2.6内核ACL功能体验之旅
·Liunx用户和内核空间之间的通信实现
·2.6.x版Linux内核发现安全漏洞
·REDIce-Linux--灵活的实时Linux内核
·FreeBSD光盘运行版的制作过程
·配置FreeBSD内核
·Linux内核分析--系统调用实现代码分
·借助异常表解决Linux内核态缺页异常
·关于Linux系统内核抢占补丁的原理说
·走向Linux系统高手之路 编译内核解
·浅谈关于Linux系统内核的源代码分析
·Linux操作系统内核抢占补丁的基本原
·揭秘Linux内核调试器之内幕
·简单编译升级内核激活FC1上的USB硬
·编译内核以使其支持PS并口改装手柄
·Linux系统可卸载内核模块完全指南(
Power by linux-cn.com 粤ICP备05006655号