热门标签 | HotTags
当前位置:  开发笔记 > 运维 > 正文

Linux系统可卸载内核模块完全指南(上)

文章标题:Linux系统可卸载内核模块完全指南(上)。Linux是中国IT实验室的一个技术频道。包含桌面应用,Linux系统管理,内核研究,嵌入式系统和开源等一些基本分类

  简介
  
  将Linux操作系统用于服务器在现在是越来越普遍了。因此,入侵Linux在今天也变得越来越有趣.目前最好的攻击Linux的技术就是修改内核代码.由于一种叫做可卸载内核(Loadable
  
  KernelModules(LKMs))的机制,我们有可能编写在内核级别运行的代码,而这种代码可以允许我们接触到操作系统中非常敏感的部分.在过去有一些很好的关于LKM知识的文本或者文件,他们介绍一些新的想法,方法以及一名Hacker所梦寐以求的完整的LKMs.而且也有一些很有趣的公开的讨论(在新闻组,邮件列表).
  
  然而为什么我再重新写这些关于LKMs的东西呢?下面是我的一些理由:
  
  在过去的教材中常常没有为那些初学者提供很好的解释.而这个教材中有很大一部分的基础章节.这是为了帮助那些初学者理解概念的.我见过很多人使用系统的缺陷或者监听器然而却丝毫不了解他们是如何工作的.在这篇文章中我包含了很多带有注释的源代码,只是为了帮助那些认为入侵仅仅是一些工具游戏的初学者!
  
  每一个发布的教材不过把话题集中在某个特别的地方.没有一个完整的指导给那些关注LKMs的Hacker.这篇文章会覆盖几乎所有的关于LKMs的资料(甚至是病毒方面的).
  
  这篇文章是从Hacker或者病毒的角度进行讨论的,但是系统管理员或者内核的开发者也可以参考并从中学到很多东西.
  
  以前的文章介绍一些利用LKMs进行入侵的优点或者方法,但是总是还有一些东西是我们过去从来没有听说过的.这篇文章会介绍一些新的想法给大家.(不是所有的新的资料,只是一些对我们有帮助的)
  
  这篇文章会介绍一些简单的防止LKM攻击的方法,同时也会介绍如何通过使用一些像运行时内核补丁(Runtime Kernel Patching)这样的方法来对付这些防御措施.
  
  要记住这些新的想法仅仅是通过利用一些特殊的模块来实现的.要在现实中真正使用他们还需要对他们进行改进.这篇文章的主要目的是给大家在整个LKM上一个大方向上的指导.在附录A中,我会给大家一些实用的LKMs,并附上一些简短的注释(这是为那些新手的),以及如何使用他们.
  
  整篇文章(除了第五部分)是基于 Linux 2.0.x的80x86机器的.我测试了所有的程序和代码段.为了能够正常使用这里提供的绝大部分代码,你的Linux系统必须有LKM支持.只有在第四部分会给大家一些不需要LKM支持的源代码.本文的绝大多数想法一样可以在Linux2.2.x上实现(也许你会需要一些小小的改动).
  
  这篇文章会有一个特别的章节来帮助系统管理员进行系统安全防护.你(作为一名Hacker)也必须仔细阅读这些章节.你必须要知道所有系统管理员知道的,甚至更多.你也会从中发现很多优秀的想法.这也会对你开发高级的入侵系统的LKMs有所帮助.
  
  因此,通读这篇文章吧.
  
  第一部分. 基础知识
  
  1.1 什么是LKMs
  
  LKMs就是可卸载的内核模块(Loadable Kernel
  
  Modules)。这些模块本来是Linux系统用于扩展他的功能的。使用LKMs的优点有:他们可以被动态的加载,而且不需要重新编译内核。由于这些优点,他们常常被特殊的设备(或者文件系统),例如声卡等使用。
  
  每个LKM至少由两个基本的函数组成:
  
  int init_module(void) /*用于初始化所有的数据*/
  
  {
  
  ...
  
  }
  
  void cleanup_module(void) /*用于清除数据从而能有一个安全的退出*/
  
  {
  
  ...
  
  }
  
  加载一个模块(常常只限于root能够使用)的命令是:
  
  # insmod module.o
  
  这个命令让系统进行了如下工作:
  
  加载可执行的目标文件(在这儿是module.o)
  
  调用 create_module这个系统调用(至于什么叫系统调用,见1.2)来分配内存.
  
  不能解决的引用由系统调用get_kernel_syms进行查找引用.
  
  在此之后系统调用init_module将会被调用用来初始化LKM->执行 int inti_module(void) 等等
  
  (内核符号将会在1.3节中内核符号表中解释)
  
  OK,到目前为止,我想我们可以写出我们第一个小的LKM来演示一下这些基本的功能是如何工作的了.
  
  #define MODULE
  
  #include
  
  int init_module(void)
  
  {
  
  printk("<1>Hello World\n");
  
  return 0;
  
  }
  
  void cleanup_module(void)
  
  {
  
  printk("<1>Bye, Bye");
  
  }
  
  你可能会奇怪为什么在这里我用printk(....)而不是printf(.....).在这里你要明白内核编程是完全不同于普通的用户环境下的编程的.你只能使用很有限的一些函数(见1.6)仅使用这些函数你是干不了什么的.因此,你将会学会如何使用你在用户级别中用的那么多函数来帮助你入侵内核.耐心一些,在此之前我们必须做一点其他的.....
  
  上面的那个例子可以很容易的被编译:
  
  # gcc -c -O3 helloworld.c
  
  # insmod helloworld.o
  
  OK,现在我们的模块已经被加载了并且给我们打印出了那句很经典的话.现在你可以通过下面这个命令来确认你的LKM确实运行在内核级别中:
  
  # lsmod
  
  Module     Pages  Used by
  
  helloworld     1    0
  
  这个命令读取在 /proc/modules 的信息来告诉你当前那个模块正被加载.'Pages'
  
  显示的是内存的信息(这个模块占了多少内存页面).'Used by'显示了这个模块被系统
  
  使用的次数(引用计数).这个模块只有当这个计数为0时才可以被除去.在检查过这个以后,你可以用下面的命令卸载这个模块
  
  # rmmod helloworld
  
  OK,这不过是我们朝LKMs迈出的很小的一步.我常常把这些LKMs于老的DOS TSR程序做比较,(是的,我知道他们之间有很多地方不一样),那些TSR能够常驻在内存并且截获到我们想要的中断.Microsoft's Win9x有一些类似的东西叫做VxD.关于这些程序的最有意思的一点在于他们都能够挂在一些系统的功能上,在Linux中我们称这些功能为系统调用.
  
  1.2什么是系统调用
  
  我希望你能够懂,每个操作系统在内核中都有一些最为基本的函数给系统的其他操作调用.在Linux系统中这些函数就被称为系统调用(System Call).他们代表了一个从用户级别到内核级别的转换.在用户级别中打开一个文件在内核级别中是通过sys_open这个系统调用实现的.在/usr/include/sys/syscall.h中有一个完整的系统调用列表.下面的列表是我的syscall.h
  
  #ifndef _SYS_SYSCALL_H
  
  #define _SYS_SYSCALL_H
  
  #define SYS_setup 0
  
  /* 只被init使用,用来启动系统的*/
  
  #define SYS_exit 1
  
  #define SYS_fork 2
  
  #define SYS_read 3
  
  #define SYS_write 4
  
  #define SYS_open 5
  
  #define SYS_close 6
  
  #define SYS_waitpid 7
  
  #define SYS_creat 8
  
  #define SYS_link 9
  
  #define SYS_unlink 10
  
  #define SYS_execve 11
  
  #define SYS_chdir 12
  
  #define SYS_time 13
  
  #define SYS_prev_mknod 14
  
  #define SYS_chmod 15
  
  #define SYS_chown 16
  
  #define SYS_break 17
  
  #define SYS_oldstat 18
  
  #define SYS_lseek 19
  
  #define SYS_getpid 20
  
  #define SYS_mount 21
  
  #define SYS_umount 22
  
  #define SYS_setuid 23
  
  #define SYS_getuid 24
  
  #define SYS_stime 25
  
  #define SYS_ptrace 26
  
  #define SYS_alarm 27
  
  #define SYS_oldfstat 28
  
  #define SYS_pause 29
  
  #define SYS_utime 30
  
  #define SYS_stty 31
  
  #define SYS_gtty 32
  
  #define SYS_access 33
  
  #define SYS_nice 34
  
  #define SYS_ftime 35
  
  #define SYS_sync 36
  
  #define SYS_kill 37
  
  #define SYS_rename 38
  
  #define SYS_mkdir 39
  
  #define SYS_rmdir 40
  
  #define SYS_dup 41
  
  #define SYS_pipe 42
  
  #define SYS_times 43
  
  #define SYS_prof 44
  
  #define SYS_brk 45
  
  #define SYS_setgid 46
  
  #define SYS_getgid 47
  
  #define SYS_signal 48
  
  #define SYS_geteuid 49
  
  #define SYS_getegid 50
  
  #define SYS_acct 51
  
  #define SYS_phys 52
  
  #define SYS_lock 53
  
  #define SYS_ioctl 54
  
  #define SYS_fcntl 55
  
  #define SYS_mpx 56
  
  #define SYS_setpgid 57
  
  #define SYS_ulimit 58
  
  #define SYS_oldolduname 59
  
  #define SYS_umask 60
  
  #define SYS_chroot 61
  
  #define SYS_prev_ustat 62
  
  #define SYS_dup2 63
  
  #define SYS_getppid 64
推荐阅读
  • 本文目录一览:1、\mysybase.dump对数据库正常使用有影响吗 ... [详细]
  • 本文介绍了Python高级网络编程及TCP/IP协议簇的OSI七层模型。首先简单介绍了七层模型的各层及其封装解封装过程。然后讨论了程序开发中涉及到的网络通信内容,主要包括TCP协议、UDP协议和IPV4协议。最后还介绍了socket编程、聊天socket实现、远程执行命令、上传文件、socketserver及其源码分析等相关内容。 ... [详细]
  • Centos7.6安装Gitlab教程及注意事项
    本文介绍了在Centos7.6系统下安装Gitlab的详细教程,并提供了一些注意事项。教程包括查看系统版本、安装必要的软件包、配置防火墙等步骤。同时,还强调了使用阿里云服务器时的特殊配置需求,以及建议至少4GB的可用RAM来运行GitLab。 ... [详细]
  • 图解redis的持久化存储机制RDB和AOF的原理和优缺点
    本文通过图解的方式介绍了redis的持久化存储机制RDB和AOF的原理和优缺点。RDB是将redis内存中的数据保存为快照文件,恢复速度较快但不支持拉链式快照。AOF是将操作日志保存到磁盘,实时存储数据但恢复速度较慢。文章详细分析了两种机制的优缺点,帮助读者更好地理解redis的持久化存储策略。 ... [详细]
  • 如何在服务器主机上实现文件共享的方法和工具
    本文介绍了在服务器主机上实现文件共享的方法和工具,包括Linux主机和Windows主机的文件传输方式,Web运维和FTP/SFTP客户端运维两种方式,以及使用WinSCP工具将文件上传至Linux云服务器的操作方法。此外,还介绍了在迁移过程中需要安装迁移Agent并输入目的端服务器所在华为云的AK/SK,以及主机迁移服务会收集的源端服务器信息。 ... [详细]
  • Linux服务器密码过期策略、登录次数限制、私钥登录等配置方法
    本文介绍了在Linux服务器上进行密码过期策略、登录次数限制、私钥登录等配置的方法。通过修改配置文件中的参数,可以设置密码的有效期、最小间隔时间、最小长度,并在密码过期前进行提示。同时还介绍了如何进行公钥登录和修改默认账户用户名的操作。详细步骤和注意事项可参考本文内容。 ... [详细]
  • 这是原文链接:sendingformdata许多情况下,我们使用表单发送数据到服务器。服务器处理数据并返回响应给用户。这看起来很简单,但是 ... [详细]
  • 本文介绍了在Hibernate配置lazy=false时无法加载数据的问题,通过采用OpenSessionInView模式和修改数据库服务器版本解决了该问题。详细描述了问题的出现和解决过程,包括运行环境和数据库的配置信息。 ... [详细]
  • 禁止程序接收鼠标事件的工具_VNC Viewer for Mac(远程桌面工具)免费版
    VNCViewerforMac是一款运行在Mac平台上的远程桌面工具,vncviewermac版可以帮助您使用Mac的键盘和鼠标来控制远程计算机,操作简 ... [详细]
  • 本文详细介绍了云服务器API接口的概念和作用,以及如何使用API接口管理云上资源和开发应用程序。通过创建实例API、调整实例配置API、关闭实例API和退还实例API等功能,可以实现云服务器的创建、配置修改和销毁等操作。对于想要学习云服务器API接口的人来说,本文提供了详细的入门指南和使用方法。如果想进一步了解相关知识或阅读更多相关文章,请关注编程笔记行业资讯频道。 ... [详细]
  • 计算机存储系统的层次结构及其优势
    本文介绍了计算机存储系统的层次结构,包括高速缓存、主存储器和辅助存储器三个层次。通过分层存储数据可以提高程序的执行效率。计算机存储系统的层次结构将各种不同存储容量、存取速度和价格的存储器有机组合成整体,形成可寻址存储空间比主存储器空间大得多的存储整体。由于辅助存储器容量大、价格低,使得整体存储系统的平均价格降低。同时,高速缓存的存取速度可以和CPU的工作速度相匹配,进一步提高程序执行效率。 ... [详细]
  • 本文介绍了Web学习历程记录中关于Tomcat的基本概念和配置。首先解释了Web静态Web资源和动态Web资源的概念,以及C/S架构和B/S架构的区别。然后介绍了常见的Web服务器,包括Weblogic、WebSphere和Tomcat。接着详细讲解了Tomcat的虚拟主机、web应用和虚拟路径映射的概念和配置过程。最后简要介绍了http协议的作用。本文内容详实,适合初学者了解Tomcat的基础知识。 ... [详细]
  • Linux环境变量函数getenv、putenv、setenv和unsetenv详解
    本文详细解释了Linux中的环境变量函数getenv、putenv、setenv和unsetenv的用法和功能。通过使用这些函数,可以获取、设置和删除环境变量的值。同时给出了相应的函数原型、参数说明和返回值。通过示例代码演示了如何使用getenv函数获取环境变量的值,并打印出来。 ... [详细]
  • 1引言在多线程并发编程中Synchronized一直是元老级角色,很多人都会称呼它为重量级锁,但是随着JavaSE1.6对Synchronized进行 ... [详细]
  • xmake-ftmpxxx.luaxmake--filexxx.lua最简单:--添加名为demo的目标到工程target(demo)--设置目标程序类型为二进制,一般为 ... [详细]
author-avatar
lmaster
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有