linux内核分析之signal.c函数

2024-04-20 20:38

本文主要是介绍linux内核分析之signal.c函数,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

该文件分装了信号处理函数
#include <linux/sched.h>
#include <linux/kernel.h>
#include <asm/segment.h>

#include <signal.h>

用volatile修饰符修饰函数,表示该函数不会返回,并且
保证编译器不会给出告警
volatile void do_exit(int error_code);

取得当前进程被阻塞的信号掩码
int sys_sgetmask()
{
 return current->blocked;
}

设置当前进程的被阻塞的信号掩码
int sys_ssetmask(int newmask)
{
  保存旧的被阻塞的信号掩码,供返回
 int old=current->blocked;

  SIGKILL信号不能被阻塞
 current->blocked = newmask & ~(1<<(SIGKILL-1));
 return old;
}

将内核数据区的数据复制到用户数据区中,以字节为单位进行复制。
开始地址from,目标地址to
static inline void save_old(char * from,char * to)
{
 int i;
/
  关于verify_area函数的定义我摘操在下面了。
  验证给定地址是否越界
 void verify_area(void * addr,int size)
 {
  unsigned long start;
 
  start = (unsigned long) addr;
   验证区域大小加上该页起始地址到start的偏移
  size += start & 0xfff;
  将开始地址进行页边界对齐
  start &= 0xfffff000;
    从当前进程的局部描述符表中找出用户数据段描述符,然后从描述符中取得段基地址
    然后求出线性地址,存入start中
  start += get_base(current->ldt[2]);
  进行验证,以一页为单位进行
  while (size>0) {
   size -= 4096;
   判断该页的写标志位是否置位
   write_verify(start);
   start += 4096;
  }
 }
// 
  验证用户数据段中申请的to空间是否可写
 verify_area(to, sizeof(struct sigaction));
 进行复制
 for (i=0 ; i< sizeof(struct sigaction) ; i++) {
  put_fs_byte(*from,to);
  from++;
  to++;
 }
}

将用过户数据段中的数据复制到内核数据段中
static inline void get_new(char * from,char * to)
{
 int i;

  进行复制
 for (i=0 ; i< sizeof(struct sigaction) ; i++)
  *(to++) = get_fs_byte(from++);
}

信号处理的系统调用函数
int sys_signal(int signum, long handler, long restorer)
{
 struct sigaction tmp;

  进行验证:
  1. 信号是否越界
  2. 信号是不是不可捕获不可忽略的信号
 if (signum<1 || signum>32 || signum==SIGKILL)
  return -1;
 对sigaction数据结构进行付值
 信号处理函数
 tmp.sa_handler = (void (*)(int)) handler;
 阻塞的信号掩码
 tmp.sa_mask = 0;
 信号标志
 tmp.sa_flags = SA_ONESHOT | SA_NOMASK;
 信号的处理函数调用完毕,进行的清理操作回调函数,不对用户开放
 tmp.sa_restorer = (void (*)(void)) restorer;
 保存旧的信号处理函数以供返回
 handler = (long) current->sigaction[signum-1].sa_handler;
 重新设置新的sigaction结构
 current->sigaction[signum-1] = tmp;
 return handler;
}

sigaction系统调用函数
int sys_sigaction(int signum, const struct sigaction * action,
 struct sigaction * oldaction)
{
 struct sigaction tmp;

  进行验证:
  1. 信号是否越界
  2. 信号是不是不可捕获不可忽略的信号 
 if (signum<1 || signum>32 || signum==SIGKILL)
  return -1;
 取得旧的sigaction结构
 tmp = current->sigaction[signum-1];
 因为进程的sigaction结构数组是在内核数据段中,所以调用get_new对新的sigaction结构
 进行付值
 get_new((char *) action,
  (char *) (signum-1+current->sigaction));
 如果oldaction不是空,那么将内核数据点中sigaction结构复制到用户数据段中的
 结构中。
 if (oldaction)
  save_old((char *) &tmp,(char *) oldaction);
 如果该进程所对应的信号不阻塞自身,那么sa_mask设置为0
 否则将自身加入到被阻塞的信号位图中
 if (current->sigaction[signum-1].sa_flags & SA_NOMASK)
  current->sigaction[signum-1].sa_mask = 0;
 else
  current->sigaction[signum-1].sa_mask |= (1<<(signum-1));
 return 0;
}

该函数是在系统调用0x80最后调用的函数,用来对信号的处理
void do_signal(long signr,long eax, long ebx, long ecx, long edx,
 long fs, long es, long ds,
 long eip, long cs, long eflags,
 unsigned long * esp, long ss)
{
 unsigned long sa_handler;
 将系统调用返回地址保存起来
 long old_eip=eip;
 取得指定当前进程信号行为数组中的sigaction结构
 struct sigaction * sa = current->sigaction + signr - 1;
 int longs;
 unsigned long * tmp_esp;

  取得信号处理句柄
 sa_handler = (unsigned long) sa->sa_handler;
 如果为忽略该信号,直接返回
 if (sa_handler==1)
  return;
 如果为缺省处理方式
 if (!sa_handler) {
   如果不是SIGCHILD那么久直接退出进程,否则直接忽略
  if (signr==SIGCHLD)
   return;
  else
   do_exit(1<<(signr-1));
 }
 如果该信号标志位设置了SA_ONESHOT,那么将信号处理句柄置为0
 表示执行默认动作
 if (sa->sa_flags & SA_ONESHOT)
  sa->sa_handler = NULL;
 因为c语言是传值的,所以将原来eip的栈位置存入信号处理函数的地址,也就是说
 该函数返回之后,回去执行信号处理函数
 *(&eip) = sa_handler;
 看看是否需要保存被阻塞的信号处理位图在用户堆栈上。
 longs = (sa->sa_flags & SA_NOMASK)?7:8;
 申请堆栈,以供保存内核堆栈上的数据
 *(&esp) -= longs;
 验证用户堆栈是否越界
 verify_area(esp,longs*4);
 tmp_esp=esp;
 将信号处理清理函数地址复制到用户堆栈上保存
 put_fs_long((long) sa->sa_restorer,tmp_esp++);
 将信号号复制到用户堆栈上
 put_fs_long(signr,tmp_esp++);
 复制被阻塞的信号位图
 if (!(sa->sa_flags & SA_NOMASK))
  put_fs_long(current->blocked,tmp_esp++);
 复制各个寄存器的值
 put_fs_long(eax,tmp_esp++);
 put_fs_long(ecx,tmp_esp++);
 put_fs_long(edx,tmp_esp++);
 put_fs_long(eflags,tmp_esp++);
 将系统调用前的eip复制到用户堆栈上,以至于信号处理函数返回后可以接着执行
 系统调用前的代码段
 put_fs_long(old_eip,tmp_esp++);
 将该sigaction结构中的被阻塞的信号位图加到当前进程的被阻塞的信号位图中去
 current->blocked |= sa->sa_mask;
}

 

这篇关于linux内核分析之signal.c函数的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/921262

相关文章

linux生产者,消费者问题

pthread_cond_wait() :用于阻塞当前线程,等待别的线程使用pthread_cond_signal()或pthread_cond_broadcast来唤醒它。 pthread_cond_wait() 必须与pthread_mutex 配套使用。pthread_cond_wait()函数一进入wait状态就会自动release mutex。当其他线程通过pthread

Linux 安装、配置Tomcat 的HTTPS

Linux 安装 、配置Tomcat的HTTPS 安装Tomcat 这里选择的是 tomcat 10.X ,需要Java 11及更高版本 Binary Distributions ->Core->选择 tar.gz包 下载、上传到内网服务器 /opt 目录tar -xzf 解压将解压的根目录改名为 tomat-10 并移动到 /opt 下, 形成个人习惯的路径 /opt/tomcat-10

RedHat运维-Linux文本操作基础-AWK进阶

你不用整理,跟着敲一遍,有个印象,然后把它保存到本地,以后要用再去看,如果有了新东西,你自个再添加。这是我参考牛客上的shell编程专项题,只不过换成了问答的方式而已。不用背,就算是我自己亲自敲,我现在好多也记不住。 1. 输出nowcoder.txt文件第5行的内容 2. 输出nowcoder.txt文件第6行的内容 3. 输出nowcoder.txt文件第7行的内容 4. 输出nowcode

【Linux进阶】UNIX体系结构分解——操作系统,内核,shell

1.什么是操作系统? 从严格意义上说,可将操作系统定义为一种软件,它控制计算机硬件资源,提供程序运行环境。我们通常将这种软件称为内核(kerel),因为它相对较小,而且位于环境的核心。  从广义上说,操作系统包括了内核和一些其他软件,这些软件使得计算机能够发挥作用,并使计算机具有自己的特生。这里所说的其他软件包括系统实用程序(system utility)、应用程序、shell以及公用函数库等

[职场] 公务员的利弊分析 #知识分享#经验分享#其他

公务员的利弊分析     公务员作为一种稳定的职业选择,一直备受人们的关注。然而,就像任何其他职业一样,公务员职位也有其利与弊。本文将对公务员的利弊进行分析,帮助读者更好地了解这一职业的特点。 利: 1. 稳定的职业:公务员职位通常具有较高的稳定性,一旦进入公务员队伍,往往可以享受到稳定的工作环境和薪资待遇。这对于那些追求稳定的人来说,是一个很大的优势。 2. 薪资福利优厚:公务员的薪资和

【操作系统】信号Signal超详解|捕捉函数

🔥博客主页: 我要成为C++领域大神🎥系列专栏:【C++核心编程】 【计算机网络】 【Linux编程】 【操作系统】 ❤️感谢大家点赞👍收藏⭐评论✍️ 本博客致力于知识分享,与更多的人进行学习交流 ​ 如何触发信号 信号是Linux下的经典技术,一般操作系统利用信号杀死违规进程,典型进程干预手段,信号除了杀死进程外也可以挂起进程 kill -l 查看系统支持的信号

Windows/macOS/Linux 安装 Redis 和 Redis Desktop Manager 可视化工具

本文所有安装都在macOS High Sierra 10.13.4进行,Windows安装相对容易些,Linux安装与macOS类似,文中会做区分讲解 1. Redis安装 1.下载Redis https://redis.io/download 把下载的源码更名为redis-4.0.9-source,我喜欢跟maven、Tomcat放在一起,就放到/Users/zhan/Documents

java中查看函数运行时间和cpu运行时间

android开发调查性能问题中有一个现象,函数的运行时间远低于cpu执行时间,因为函数运行期间线程可能包含等待操作。native层可以查看实际的cpu执行时间和函数执行时间。在java中如何实现? 借助AI得到了答案 import java.lang.management.ManagementFactory;import java.lang.management.Threa

Linux系统稳定性的奥秘:探究其背后的机制与哲学

在计算机操作系统的世界里,Linux以其卓越的稳定性和可靠性著称,成为服务器、嵌入式系统乃至个人电脑用户的首选。那么,是什么造就了Linux如此之高的稳定性呢?本文将深入解析Linux系统稳定性的几个关键因素,揭示其背后的技术哲学与实践。 1. 开源协作的力量Linux是一个开源项目,意味着任何人都可以查看、修改和贡献其源代码。这种开放性吸引了全球成千上万的开发者参与到内核的维护与优化中,形成了

SQL Server中,isnull()函数以及null的用法

SQL Serve中的isnull()函数:          isnull(value1,value2)         1、value1与value2的数据类型必须一致。         2、如果value1的值不为null,结果返回value1。         3、如果value1为null,结果返回vaule2的值。vaule2是你设定的值。        如