Linux 内核进程管理 ( 内核线程概念 | 内核线程普通进程用户线程 | 内核线程与普通进程区别 | 内核线程主要用途 | 内核线程创建函数 kernel_thread 源码 )

Posted 韩曙亮

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Linux 内核进程管理 ( 内核线程概念 | 内核线程普通进程用户线程 | 内核线程与普通进程区别 | 内核线程主要用途 | 内核线程创建函数 kernel_thread 源码 )相关的知识,希望对你有一定的参考价值。

文章目录





一、内核线程概念



直接 由 Linux 内核 启动的线程 , 被称为 " 内核线程 " ;


" 内核线程 " 是一种 特殊进程 , 独立运行在 " 内核空间 " , 其将 " 内核函数 " 委托给 独立进程 , 该 " 独立进程 "其它进程 ( 包括 普通进程 , 内核自身 , 用户级线程 ) 并行执行 ;


" 内核线程 " 也称为 " 守护进程 " ;





二、内核线程、普通进程、用户线程



【Linux 内核】进程管理 ( 进程特殊形式 | 内核线程 | 用户线程 | C 标准库与 Linux 内核中进程相关概念 | Linux 查看进程命令及输出字段解析 ) 一、进程特殊形式 ( 内核线程 | 用户线程 ) 博客章节 , 有介绍内核线程 ,

普通的进程 , 包含 内核虚拟地址空间用户虚拟地址空间 , 其中 内核虚拟地址空间 所有进程共享 , 用户虚拟地址空间 由进程独立拥有 ; 除了 普通进程 外 , 进程 还有 2 2 2 种 特殊形式 :

  • 内核级线程 : 只有 内核虚拟地址空间 , 没有 用户虚拟地址空间 的进程 , 称为 内核线程 ;
  • 用户级线程 : 共享 用户虚拟地址空间 的进程 , 称为 用户线程 ;




三、内核线程、普通进程区别



" 内核线程 "" 普通进程 " 区别是 , 内核进程 没有 " 独立的进程地址空间 " ;


task_struct 进程描述符 结构体中的 , mm 指针指向的空间就是 " 独立的进程地址空间 " ;


Linux 内核 中 , " 进程控制块 " 是通过 task_struct 结构体 进行描述的 ; Linux 内核中 , 所有 进程管理 相关算法逻辑 , 都是基于 task_struct 结构体的 ; task_struct " 进程描述符 " 结构体在 linux-5.6.18\\include\\linux\\sched.h 头文件中 第 629 629 629 ~ 1300 1300 1300 行定义 ;


mm 字段是一个指针 , 指针指向 mm_struct 结构体 , 这是 " 内存描述符 " , 与 tsak_struct 进程描述符性质相似 ;

对于 进程 来说 , active_mm 字段 与 mm 字段 指向同一个 " 内存描述符 " ;

但对于 " 内核线程 " 来说 , mm 字段 指向 空指针 , active_mm 字段 指向 从进程借用的 " 内存描述符 " ;

	struct mm_struct		*mm;
	struct mm_struct		*active_mm;

源码地址 : linux-5.6.18\\include\\linux\\sched.h





四、内核线程主要用途



内核线程主要用途 :

内存同步 : 周期性执行如下同步操作 , 同步 " 修改的内存页 "" 页来源块设备 " , 如 : mmap 文件映射 ;

写交换区 : 假如同步 " 修改的内存页 " 与 " 页来源块设备 " 时 , 内存页使用率很低 , 则 将同步内容写入 " 交换区 " ;

延时管理 : 管理 " 延时动作 " , Deferred Action ;

系统日志 : 管理控制 文件系统 事务日志 生成 ;





五、内核线程创建函数 kernel_thread 源码



linux-5.6.18\\kernel\\fork.c 源码中的

pid_t kernel_thread(int (*fn)(void *), void *arg, unsigned long flags)

方法 , 就是创建内核线程的函数 , 该函数中最终也是调用了 _do_fork() 函数 , 与 fork() , vfork() , clone() 等系统调用函数创建进程的方式类似 ;

/*
 * Create a kernel thread.
 */
pid_t kernel_thread(int (*fn)(void *), void *arg, unsigned long flags)

	struct kernel_clone_args args = 
		.flags		= ((lower_32_bits(flags) | CLONE_VM |
				    CLONE_UNTRACED) & ~CSIGNAL),
		.exit_signal	= (lower_32_bits(flags) & CSIGNAL),
		.stack		= (unsigned long)fn,
		.stack_size	= (unsigned long)arg,
	;

	return _do_fork(&args);

以上是关于Linux 内核进程管理 ( 内核线程概念 | 内核线程普通进程用户线程 | 内核线程与普通进程区别 | 内核线程主要用途 | 内核线程创建函数 kernel_thread 源码 )的主要内容,如果未能解决你的问题,请参考以下文章

Linux下的进程类别(内核线程轻量级进程和用户进程)以及其创建方式--Linux进程的管理与调度

Linux(内核剖析):06---进程之线程的实现

《内核设计与实现》读书笔记- 进程管理

内核空间进程和线程等概念

内核空间进程和线程等概念

进程和线程内核空间等概念