linux内核进程管理模块,[原创]开源一个Linux内核里进程内存管理模块源码

Linux它是一款开源的内核系统。本人也非常喜欢嵌入式Linux系统,特别是它的内核源码,书写的风格,都非常讨我心欢。这个驱动是之前业余的时候写的对于新手来说,还是有学习价值的。

fced3dfaae257839ca114d8ad0b9cdae.png

5df4896fca11e451697d1b0d6f198b30.png

下面将对源码进行简单的讲解。

首先是隐藏内核驱动模块。

list_del_init是将自身驱动模块从驱动列表(lsmod)中抹掉

kobject_del是将自己从/sys/class/xxxxxx中抹掉

接下来是打开进程接口。

在Linux内核里,不区分进程与线程。统一按照线程来看待。那么每个线程都有一个对应的pid_t、pid、task_struct。

他们之间的关系是这样的:

pid_t struct pid

nr为进程pid数值

看完以上的逻辑。大家是不是柳暗花明又一村,心里开朗了许多,他们之间是可以相互转换的。通过进程pid_t可以拿到pid,通过pid可以拿到task_struct。又可以反过来通过task_struct拿到进程pid。

然后是关闭进程接口

驱动源码是使用put_pid将进程pid*的使用次数减去1

在Linux内核源码/kernel/pid.c下可以看到

读进程内存、写进程内存接口

这里采用读、写物理内存的思路,因为这样简洁明了,不需要理会其他反调试干扰。

首先根据pid*用get_pid_task取出task_struct。再用get_task_mm取出mm_struct结构。因为这个结构包含了进程的内存信息。首先检查内存是否可读if (vma->vm_flags & VM_READ)

如果可读。那么开始计算物理内存地址位置。由于Linux内核默认开启MMU机制,所以只能以页为单位计算物理内存地址。计算物理内存地址的方法有很多,这里提供三种思路:

第一种是使用get_user_pages,

第二种是直接使用pagemap文件,

第三种是纯手写算法(将pgd、pud、pmd和pte拼接在一起)

我个人推荐使用第三种方法,也是我正在使用的方法,速度极快而且不触碰任何的进程文件,被调试进程无任何感知。

不推荐使用第一种get_user_pages方法,因为此方法会触发反调试机制,导致调试失败。

知道了物理内存地址后,读、写物理内存地址,其实Linux内核里面也有演示,即drivers/char/mem.c。写的非常详细。最后还要注意MMU机制的离散内存,即buffer不连续问题,通俗的说就是不要一下子读太多,读到另一页去了,要分开页来读

获取进程内存块列表

这个接口就很简单了,通过task_struct取出mm_struct,接下来在mm_struct中遍历取出vma。详情可以参考代码fs\proc\task_mmu.c

获取进程命令行

mm_struct结构体里面有个arg_start变量,储存的地址值即是进程命令行。

但这里有个要注意的地方,经过我多台设备测试发现,并不是每个Linux内核系统的arg_start变量偏移值是一样的,这样子就会非常危险,一旦读错就会死机,而且原因还不好查找。

这里我使用了一些玄学的技巧,驱动可以自适应地在不同的Linux内核中自行修正arg_start变量的偏移值,从而读取到正确的值,不会死机。

获取进程PID列表

驱动里写入了两种方法,第一种是遍历/proc/pid目录,第二种是遍历task_struct结构体。这里有个要注意的地方,经过我多台设备测试发现,并不是每个Linux内核系统的task_struct结构体里的tasks变量偏移值是一样的,但具体玄学修正方法我还没时间进行编写,待有空再补充。

获取进程物理内存大小

读取task_struct结构体里的mm_struct,再读取rss_stat就会有进程的物理内存占用大小,这个来源与/proc/pid/status里的源码编写。这里同样需要玄学技巧修正变量的偏移值,具体方法我已编写在内。

获取进程权限、设置进程权限为ROOT

在取得task_struct进程结构后,观察头文件可以发现里面有两个变量值,一个是real_cred,另一个是cred,其实很简单,将两个cred里面的uid、gid、euid、egid、fsuid、fsgid修改成0即可

real_cred指向主体和真实客体证书,cred指向有效客体证书。通常情况下,cred和real_cred指向相同的证书,但是cred可以被临时改变

同样需要注意,每个Linux内核的cred结构变量偏移值并不是一样的,读错会死机,同理,我也使用了玄学的技巧,驱动能智能修正Linux内核变量的偏移值,能准确的识别出每个Linux内核版本里real_cred与cred的正确偏移位置

892833_JF2VGK6DEF2SPKZ.png

892833_U7RAEH3G4RK62D9.png

开源地址(含使用demo)

总结:

首先,编译此源码需要一定的技巧,再者,手机厂商本身已设置多重障碍用来阻止第三方驱动的加载,如果您需要加载此驱动,则需要将内核中的一些限制给去除。(其实这些验证都可以用IDA暴力Patch之~)。

提醒:

本源码不针对任何程序,仅供交流、学习、调试Linux内核程序的用途,禁止用于任何非法用途,调试器是一把双刃剑,希望大家能营造一个良好的Linux内核Rootkit技术的交流环境。

后续:

后面即将开源:

“不需要源码,强制暴力修改手机Linux内核、去除加载内核驱动的所有验证”

“不需要源码,强制加载启动ko驱动文件、跨Linux内核版本、跨设备启动ko驱动模块文件”

“不需要源码,Linux内核进程万能调试器-可过所有的反调试机制-含硬件断点:Linux天下调,让天下没有难调试的Linux进程!”

“不需要源码,突破Linux内核Elf结构限制、将ollvm混淆加入到ko驱动文件中,增加驱动逆向难度”

最后于 2021-4-3 01:42

被abcz316编辑

,原因: 补充图片

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Linux 内核内存管理优化是一个持续不断的过程,需要不断地改进和优化。这个过程中涉及到很多的内核开发者和贡献者,下面简单介绍一些内存管理优化的进程: 1. Linux 内核社区:Linux 内核社区是 Linux 内核开发的主要场所,它包括了很多的内核开发者和贡献者。在 Linux 内核社区中,内存管理优化是一个重要的议题,社区中的开发者和贡献者会就内存管理的问题进行讨论和交流,并提出改进和优化的建议。 2. 内存管理子系统维护者:在 Linux 内核中,内存管理一个独立的子系统。这个子系统有专门的维护者,他们负责内存管理的开发、维护和优化工作。内存管理子系统维护者会根据社区的反馈和需求,不断地改进和优化内存管理的代码,提高内核的性能和稳定性。 3. Linux 内核开发者:除了内存管理子系统维护者之外,Linux 内核开发者也会参与到内存管理优化的工作中来。他们会在内核的不同模块中,对内存管理进行改进和优化,例如在文件系统中对内存进行缓存、在网络协议栈中对内存进行管理等。 4. 厂商和社区用户:Linux 内核不仅是开源软件,也是很多商业厂商采用的操作系统内核。因此,厂商和社区用户也会参与到内存管理优化的工作中来。他们会根据自己的需求和场景,提出内存管理的改进方案并提交到社区中,从而促进内存管理的优化。 总之,Linux 内核内存管理优化是一个社区化的进程,需要内核开发者、维护者、厂商和社区用户共同参与和推动。只有这样,才能不断地提高 Linux 内核内存管理性能和稳定性。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值