Linux0.11内核--内核空间、用户空间之间的数据传输
2015-06-02 21:10
295 查看
内核空间、用户空间之间的数据传输
内核空间数据段的选择符为0x10,用户空间数据段选择符为0x17。内核空间、用户空间之间的数据传输,是段间数据传输。C语言中的赋值语句编译成汇编后,“=”两边的变量默认段选择符都是DS,因此只能用于同一段内数据传输。
在segment.h中定义了一系列用于内核空间和用户空间传输数据的函数。从用户空间取得数据的函数中, mov指令的源操作数段寄存器都明确指出是fs,向用户空间写数据的函数中, mov指令的目的操作数段寄存器都是fs。当系统调用发生时,int
0x80处理函数会把fs设成用户数据段选择符(0x17),参见中断异常处理和系统调用一章。
下面分析一组对byte操作的函数,其他的对word和long操作的函数与之类似。
<!-- @page { margin: 0.79in } P { margin-bottom: 0.08in } -->
get_fs_byte()
[b][cpp] view
plaincopy[/b]
// 功能:从用户空间中addr地址处取出一个字节
// 参数:addr 用户空间中的逻辑地址
// 返回:fs:[addr]处的一个字节内容
extern inline unsigned char get_fs_byte(const char * addr)
{
unsigned register char _v;
// addr是逻辑地址,也就是用户数据段内的偏移。
// 而当前数据段为内核数据段,所以要写成fs:[addr],这是虚拟地址
__asm__ ("movb %%fs:%1,%0":"=r" (_v):"m" (*addr));
return _v;
}
[b]put_fs_byte()[/b]
[b][cpp] view
plaincopy[/b]
[b]// 功能:向用户空间中addr地址处写一个字节的内容 [/b]
[b]// 参数:val 要写入的数据 [/b]
[b]// addr 用户空间中的逻辑地址 [/b]
[b]// 返回:(无) [/b]
[b]extern inline void put_fs_byte(char val,char *addr) [/b]
[b]{ // addr是相对于用户数据段的偏移,而当前数据段为内核数据段 [/b]
[b] // 所以要写成fs:[addr]的形式 [/b]
[b]__asm__ ("movb %0,%%fs:%1"::"r" (val),"m" (*addr)); [/b]
[b]} [/b]
内核空间数据段的选择符为0x10,用户空间数据段选择符为0x17。内核空间、用户空间之间的数据传输,是段间数据传输。C语言中的赋值语句编译成汇编后,“=”两边的变量默认段选择符都是DS,因此只能用于同一段内数据传输。
在segment.h中定义了一系列用于内核空间和用户空间传输数据的函数。从用户空间取得数据的函数中, mov指令的源操作数段寄存器都明确指出是fs,向用户空间写数据的函数中, mov指令的目的操作数段寄存器都是fs。当系统调用发生时,int
0x80处理函数会把fs设成用户数据段选择符(0x17),参见中断异常处理和系统调用一章。
下面分析一组对byte操作的函数,其他的对word和long操作的函数与之类似。
<!-- @page { margin: 0.79in } P { margin-bottom: 0.08in } -->
get_fs_byte()
[b][cpp] view
plaincopy[/b]
// 功能:从用户空间中addr地址处取出一个字节
// 参数:addr 用户空间中的逻辑地址
// 返回:fs:[addr]处的一个字节内容
extern inline unsigned char get_fs_byte(const char * addr)
{
unsigned register char _v;
// addr是逻辑地址,也就是用户数据段内的偏移。
// 而当前数据段为内核数据段,所以要写成fs:[addr],这是虚拟地址
__asm__ ("movb %%fs:%1,%0":"=r" (_v):"m" (*addr));
return _v;
}
[b]put_fs_byte()[/b]
[b][cpp] view
plaincopy[/b]
[b]// 功能:向用户空间中addr地址处写一个字节的内容 [/b]
[b]// 参数:val 要写入的数据 [/b]
[b]// addr 用户空间中的逻辑地址 [/b]
[b]// 返回:(无) [/b]
[b]extern inline void put_fs_byte(char val,char *addr) [/b]
[b]{ // addr是相对于用户数据段的偏移,而当前数据段为内核数据段 [/b]
[b] // 所以要写成fs:[addr]的形式 [/b]
[b]__asm__ ("movb %0,%%fs:%1"::"r" (val),"m" (*addr)); [/b]
[b]} [/b]
相关文章推荐
- Linux内核编程接口函数
- LINUX-软件安装(三)
- Linux IPC 之管道通信
- Linux系统编程——进程间通信:消息队列
- Android底层开发(二)之linux内核层 第一步
- linux主机如何添加swap分区
- Linux文件的默认权限与查找命令详解
- Linux文件的默认权限与查找命令详解
- makefile中的patsubst
- Linux目录管理全面掌握
- Linux目录管理全面掌握
- linux_awk_内部正则过滤
- [葫芦娃linux记录贴]ubuntu如何安装flash插件
- 教你从redhat官网获取ISO镜像
- 教你从redhat官网获取ISO镜像
- Centos6安装Qt5版本
- 从第一个linux模块做起—字符驱动
- Linux下套接字详解(二)----套接字Socket
- Linux库函数(6.2)
- linux中lvm的缩减