当前位置: 首页 > news >正文

linux 内核: 访问当前进程的 task_struct

一:概述        

        Linux 属于单内核架构,这意味着所有驱动、调度器、内存管理、文件系统模块都运行在内核态,驱动模块也运行在与内核相同的上下文中,可以访问当前正在执行的进程,本文介绍下通过 current 宏指向的 task_struct 结构体,来获取当前正在执行进程的相关信息。        

二: 本例子所用内核头文件介绍        

     1. <linux/init.h>

     作用:包含 module_init()module_exit() 宏的定义。用于标记模块的初始化与退出函数。属于模块加载的生命周期管理。

    2. <linux/module.h>

     作用: 支持编写内核模块的核心头文件。定义了 MODULE_AUTHORMODULE_LICENSEMODULE_DESCRIPTION 等元信息宏。包含模块注册、注销、符号导出等功能的声明。

   3. <linux/sched.h> (注:从 Linux 5.4 起常用 <linux/sched/task.h> 替代)

    作用: 定义了 task_structcurrent 宏。current 是一个宏,用于获取当前运行线程的 task_struct 指针。这是访问进程上下文的关键数据结构。

   4. <linux/preempt.h>

   作用: 提供了 in_task() 等宏。in_task() 判断当前是否在进程上下文中运行(非中断上下文)。与抢占模型和调度相关的定义在此。

   5. <linux/cred.h>

 作用:提供了获取当前进程的用户权限信息的接口,如:current_uid(),current_euid()。

   6. <linux/uidgid.h>

    作用:定义了 UID/GID 的处理函数,from_kuid():将内核内部的 kuid_t 类型转换为普通用户空间可读的整数 UID。make_kuid():反之,将整数 UID 转为 kuid_t。与用户命名空间(User Namespace)有关,确保多租户场景下权限隔离。

三:例子中的关键函数

        从当前进程的内核内部凭据(credentials)结构中提取用户标识(UID 和 EUID),其中 UID 表示真实用户 ID,即当前进程是谁启动的。EUID表示有效用户 ID,即当前的权限。

	unsigned int uid = from_kuid(&init_user_ns, current_uid());unsigned int euid = from_kuid(&init_user_ns, current_euid());

        判断当前是否是进程上下文,in_task() 是一个宏,判断当前是否是进程上下文(不是中断、软中断等)。

if (likely(in_task())) {}

         打印当前任务结构信息:

pr_info(" name        : %s\n", current->comm);
pr_info(" PID         : %6d\n", task_pid_nr(current));
pr_info(" TGID        : %6d\n", task_tgid_nr(current));
pr_info(" UID         : %6u\n", uid);
pr_info(" EUID        : %6u (%s root)\n", euid, (euid == 0 ? "have" : "don't have"));
pr_info(" state       : %c\n", task_state_to_char(current));/*
current->comm: 进程名(exec() 时被设置)
task_pid_nr(current): 进程 ID
task_tgid_nr(current): 线程组 ID
task_state_to_char(current): 返回 R、S 等状态字符
UID/EUID: 标识用户身份
*/

        打印 task_struct 和 stack 地址

pr_info("current : 0x%pK (0x%px)\n", current, current);
pr_info("stack start : 0x%pK (0x%px)\n", current->stack, current->stack);/*
%pK:如果开启了地址限制(kptr_restrict),非 root 用户会看到 0。
%px:不受 kptr_restrict 影响,调试时常用。
current->stack: 栈底地址(向下增长)
*/

        判断当前是否跑在虚拟机上

//当前是否跑在虚拟 CPU 上
if (task_state_to_char(current) == 'R')pr_info("on virtual CPU? %s\n", (current->flags & PF_VCPU)?"yes":"no");

 三:完整例子

#define pr_fmt(fmt) "%s:%s(): " fmt, KBUILD_MODNAME, __func__#include <linux/init.h>
#include <linux/module.h>
#include <linux/sched.h>      // current
#include <linux/preempt.h>    // in_task
#include <linux/cred.h>       // current_uid, current_euid
#include <linux/uidgid.h>     // from_kuidMODULE_AUTHOR("Anonymous");
MODULE_DESCRIPTION("Display current process task_struct info");
MODULE_LICENSE("Dual MIT/GPL");
MODULE_VERSION("0.1");static inline void show_ctx(void)
{unsigned int uid = from_kuid(&init_user_ns, current_uid());unsigned int euid = from_kuid(&init_user_ns, current_euid());pr_info("\n");if (likely(in_task())) {pr_info("Running in process context:\n"" Name        : %s\n"" PID         : %6d\n"" TGID        : %6d\n"" UID         : %6u\n"" EUID        : %6u (%s root)\n"" State       : %c\n"" task_struct : 0x%pK (0x%px)\n"" Stack start : 0x%pK (0x%px)\n",current->comm,task_pid_nr(current), task_tgid_nr(current),uid, euid,(euid == 0 ? "have" : "don't have"),task_state_to_char(current),current, current, current->stack, current->stack);} else {pr_alert("Running in interrupt context! (unexpected here)\n");}
}static int __init current_ctx_init(void)
{pr_info("Module inserted.\n");pr_info("sizeof(struct task_struct) = %zd\n", sizeof(struct task_struct));show_ctx();return 0;
}static void __exit current_ctx_exit(void)
{show_ctx();pr_info("Module removed.\n");
}module_init(current_ctx_init);
module_exit(current_ctx_exit);

  四:参考       

        《Linux Kernel Programming 2nd Edition》

 

        

        

http://www.lryc.cn/news/588258.html

相关文章:

  • [论文阅读] 人工智能 + 软件工程 | 用大语言模型+排名机制,让代码评论自动更新更靠谱
  • android Perfetto cpu分析教程及案例
  • 迁移学习之图像预训练理解
  • ICML 2025 | 从语言到视觉,自回归模型VARSR开启图像超分新范式
  • C# TCP粘包与拆包深度了解
  • CSP-S 模拟赛 17
  • 哈希扩展 --- 海量数据处理
  • 一文明白AI、AIGC、LLM、GPT、Agent、workFlow、MCP、RAG概念与关系
  • Linux操作系统从入门到实战(七)详细讲解编辑器Vim
  • 螺旋模型:风险分析驱动的渐进式开发
  • C++卸载了会影响电脑正常使用吗?解析C++运行库的作用与卸载后果
  • 什么是实时数仓?实时数仓又有哪些应用场景?
  • 疯狂星期四 - 第7天运营日报
  • 多线程/协程环境时间获取的“时间片陷阱“:深度解析与工程级解决方案
  • 16.避免使用裸 except
  • Sharding-Sphere学习专题(一)基础认识
  • sshpass原理详解及自动化运维实践
  • xss-lab靶场通关
  • GD32/STM32嵌入CMSIS-DSP的库(基于Keil)
  • 系统思考:跨境跨界团队学习
  • 后端接口通用返回格式与异常处理实现
  • Flask服务器公外网访问,IPv6(亲测有效!!!!)
  • 8.数据库索引
  • vmware使用说明
  • XML vs JSON:核心区别与最佳选择
  • 如何基于FFMPEG 实现视频推拉流
  • win10安装Elasticsearch
  • 分享三个python爬虫案例
  • nginx:SSL_CTX_use_PrivateKey failed
  • CentOS7 OpenSSL升级1.1.1t;OpenSSH 升级 9.8p1 保姆级教程