GNU/Linux - copy_{to,from}_user: 用户和内核空间的内存互拷贝

copy_{to,from}_user 函数是 Linux 内核编程的基本组成部分。它用于将数据从用户空间复制到内核空间。在编写内核模块或使用设备驱动程序时,安全地处理用户空间和内核空间之间的数据传输对防止安全漏洞和确保系统稳定至关重要。

The copy_{to,from}_user function is a fundamental part of Linux kernel programming. It is used to copy data from user space to kernel space. When writing kernel modules or working with device drivers, it's crucial to safely handle the data transfer between user space and kernel space to prevent security vulnerabilities and ensure system stability.

在内核编程中,正确使用 copy_{to,from}user(复制{to,from}_用户)对于维护系统稳定性和安全性至关重要。请始终遵循最佳实践并彻底测试您的代码。

Using copy_{to,from}_user correctly is critical in kernel programming to maintain system stability and security. Always follow best practices and thoroughly test your code.

这些函数是一系列 API 的一部分,旨在促进用户空间与内核空间之间安全可靠的数据传输。

These functions are part of a set of APIs designed to facilitate safe and secure data transfers between user space and kernel space.

在修改内核代码或编写内核模块时,要使用这两个函数的话,要包含头文件。" #include <linux/uaccess.h>",这个是high-level头文件,提供在内核中访问用户内存空间的APIs。

"#include <asm/uaccess.h> ", 这样使用头文件的话,是low-level的,架构依赖(architecture-specific)的头文件,提供了相关功能的具体实现, 这个一般是通过其他头文件来包含使用,开发者一般不直接包含此头文件。而且这个头文件的实际路径,是在arch目录里的,比如ARM架构: arch\arm\include\asm\uaccess.h。

static __always_inline unsigned long __must_check

copy_to_user(void __user *to, const void *from, unsigned long n)

static __always_inline unsigned long __must_check

copy_from_user(void *to, const void __user *from, unsigned long n)

这两个函数的底层由相应架构提供的接口来实现:raw_copy_{to,from}_user()。

这两个函数返回0表示成功,返回非0表示剩余的未copy成功的字节长度。

copy_{to,from}_user 函数定义在 Linux 内核源代码的特定架构代码中。不同体系结构的实现细节可能略有不同,但一般都是在体系结构的内存访问例程中定义的。

The copy_{to,from}_user function is defined in the Linux kernel source within the architecture-specific code. The implementation details can vary slightly between different architectures, but generally, it is defined in the architecture's memory access routines.

例如,在 x86 架构中,raw_copy_{to,from}_user()函数定义在以下文件中:

For example, in the x86 architecture, the copy_to_user function is defined in the following file:

arch/x86/include/asm/uaccess_32.h

arch/x86/include/asm/uaccess_64.h

还有一个struct拷贝函数。

static __always_inline __must_check int

copy_struct_from_user(void *dst, size_t ksize, const void __user *src,

size_t usize)

有三种情况需要考虑:

如果 @usize == @ksize,则是逐字复制。

如果 @usize < @ksize,则表示用户空间将旧结构传递给了新内核。@dst 的尾部字节(@ksize - @usize)将被清零。

如果 @usize > @ksize,则表示用户空间向旧内核传递了一个新结构体。内核未知的尾部字节(@usize - @ksize)将被检查以确保清零,否则将返回 -E2BIG。

There are three cases to consider:

If @usize == @ksize, then it's copied verbatim.

If @usize < @ksize, then the userspace has passed an old struct to a newer kernel. The rest of the trailing bytes in @dst (@ksize - @usize) are to be zero-filled.

If @usize > @ksize, then the userspace has passed a new struct to an older kernel. The trailing bytes unknown to the kernel (@usize - @ksize) are checked to ensure they are zeroed, otherwise -E2BIG is returned.

相关推荐
能喵烧香6 小时前
深度解析:Linux 与 Windows 超级权限账户的本质差异
linux·windows
Moshow郑锴8 小时前
Ubuntu 26.04 中文输入法 : fcitx5+Rime中州韵引擎
linux·运维·ubuntu
qq_163135759 小时前
Linux 【04-more命令超详细教程】
linux
sevencheng79810 小时前
【ADB】adb命令行常用按键模拟代码
linux·adb·模拟按键,返回键,音量键
暗影天帝10 小时前
BPI-R3 Mini 刷 Yuzhii DHCPD U-Boot 教程
linux
小赖同学啊11 小时前
智能连接器集群化高可用生产方案
linux·运维·人工智能
Cinema KI11 小时前
Linux第一个系统程序-进度条
linux·服务器
Moshow郑锴11 小时前
Ubuntu 26.04 更换阿里云源镜像
linux·运维·ubuntu
Jason_chen12 小时前
Linux 6.2 串口机制深度解析:AI驱动的自适应通信与零信任串口安全架构
linux
ShineWinsu12 小时前
对于Linux:线程概念与分页存储管理的解析
linux·运维·服务器·面试·线程·进程·虚拟空间地址