Linux信号集、信号的阻塞&信号驱动

一、信号集

信号的三种方式,即使是忽略,也仍然打断了进程的进行(相当于是捕捉了信号,执行的了空函数什么都没做),可如果在访问重要资源时不希望被打断呢?

可以用阻塞 ,即产生了信号却没及时送达到进程 ,它处于未决状态 ,并不是不响应该信号,而是推迟响应

信号的状态

  • 递达delivery:信号产生并递达到进程

  • 未决pending:信号产生但未递达到进程,主要是阻塞屏蔽导致的,SIGKILL、SIGSTOP不可阻塞

Linux内核的进程控制块 PCB 是一个结构体task_struct, 除了包含进程 id,状态,工作目录,用户 id,组 id,文件描述符表,还包含了信号相关的信息,主要指阻塞信号集和未决信号集,信号集中每个bit位对应一个信号

阻塞信号集mask:信号对应的比特位置1,即被阻塞屏蔽,信号将处于未决状态,屏蔽解除后才能递达

未决信号集pending:信号产生时,未决信号集中描述该信号的位置1表示未决(相当于别人给你发送了一个消息未读),当信号被处理后,该位置0(相当于消息已回)

因此可以通过设置阻塞信号集的这些bit位来决定是否阻塞

信号集有关的操作函数

cpp 复制代码
sigset_t set;               //信号集结构体变量  相当于设置了一个信号集
sigemptyset(sigset_t *set)  //清空信号集
sigfillset(sigset_t *set)   //信号全添加进信号集
sigaddset(sigset_t *set, int signum) //指定信号添加到信号集
sigdelset(sigset_t *set, int signum) //指定信号从信号集合中删除
sigismember(sigset_t *set, int signum) //判断一个信号是否在集合中

int sigprocmask(int how, sigset_t *set, sigset_t *oset)
功能:将进程中的信号阻塞集替换成定义好的信号集set,并根据how决定是否阻塞信号
@param: how SIG_BLOCK   置1
    		SIG_UNBOLCK 置0
    		SIG_SETMASK 信号集设置为参数set中的信号
    	set 设置的信号集  oset 一般给NULL即可
@return: 成功返回0失败返回-1

示例:

cpp 复制代码
#include <stdio.h>
#include <signal.h>
#include <unistd.h>


void handler(int signum)
{
	if(signum == SIGINT)
	{
		printf("i caught SIGINT,i won't stop\n");
	}


}


int main()
{

        struct sigaction act;
        act.sa_handler = handler;
        sigemptyset(&act.sa_mask);
        act.sa_flags = 0;
	sigaction(SIGINT,&act,NULL);

	//block signal for a while and unblock
	sigset_t set;
	sigemptyset(&set);
	sigaddset(&set,SIGINT);
	sigprocmask(SIG_BLOCK,&set,NULL);
	sleep(5);
	sigprocmask(SIG_UNBLOCK,&set,NULL);

	while(1)
	{
		sleep(1);
	}

	return 0;
}

二、信号驱动

cpp 复制代码
int pause(void)
功能:暂停当前进程,进入休眠状态,直至收到任意信号处理完才会返回,所以可以用作信号驱动
  • 信号被忽略时,进程继续挂起暂停
  • 信号被默认终止处理时,则进程直接终止退出
  • 信号被捕捉处理,则处理完后返回-1,继续往下运行
  • 信号从屏蔽->解除屏蔽,进程仍挂起暂停,pause感受不到这个信号的产生和处理,因为在解除屏蔽的瞬间就信号抵达就处理完了,这种情况下要用sigsuspend
cpp 复制代码
int sigsuspend(sigset_t *sigmask)
功能:将进程的阻塞信号集替换成sigmask信号集,然后暂停进程,相当于sigprocmask + pause的结合体,适合用于解除信号屏蔽的同时驱动

示例:

cpp 复制代码
#include <stdio.h>
#include <signal.h>
#include <unistd.h>


void handler(int signum){ }

void task(void)
{
	puts("task start");
	sleep(5);
	puts("task end");
}

int main()
{

        struct sigaction act;
        act.sa_handler = handler;
        sigemptyset(&act.sa_mask);
        act.sa_flags = 0;
	sigaction(SIGINT,&act,NULL);
	
	sigset_t set,set1;
	sigemptyset(&set);
	sigemptyset(&set1);
	sigaddset(&set,SIGINT);

	pause();

	while(1)
	{
		sigprocmask(SIG_BLOCK,&set,NULL);
		task();
		sigsuspend(&set1);
	}

	return 0;
}
相关推荐
Industio_触觉智能5 分钟前
瑞芯微方案主板Linux修改系统串口波特率教程,触觉智能RK3562开发板演示
linux·开发板·串口调试·rk3562·波特率
橘子真甜~33 分钟前
Linux操作系统3-文件与IO操作1(从C语言IO操作到系统调用)
linux·运维·服务器·c语言·io·文件操作·文件fd
A charmer44 分钟前
Linux 权限管理:用户分类、权限解读与常见问题剖析
linux·运维·服务器
羊村懒哥1 小时前
linux-安全-iptables防火墙基础笔记
linux·网络·安全
孙同学_1 小时前
【Linux篇】权限管理 - 用户与组权限详解
java·linux·服务器
skywalk81631 小时前
完全按照手册win10里装Ubuntu 虚拟机然后编译ESP32(主要是想针对ESP32C3和S3)开发板的鸿蒙系统(失败)
ubuntu·华为·harmonyos·liteos
猫猫的小茶馆1 小时前
【Linux系统】Linux内核框架(详细版本)
linux·运维·服务器·开发语言·嵌入式软件
风行無痕1 小时前
Ubuntu 22.04 解决EasyExcel导出报错和读写问题 缺失字体
linux·运维·ubuntu
boss-dog1 小时前
Ubuntu22.04系统源码编译OpenCV 4.10.0(包含opencv_contrib)
opencv·ubuntu
码农老张Zy1 小时前
【PHP小课堂】学习PHP中的变量处理相关操作
android·开发语言·学习·php