普通信号的复现

1)SIGHUP

  该信号在终端挂起或控制进程终止时发出,那可以通过在终端中运行进程,然后关闭终端来实现。

  代码如下:

 1 /**
 2  * filename: signal_1.c
 3  * author: Suzkfly
 4  * date: 2021-02-15
 5  * platform: Ubuntu
 6  *     操作步骤:
 7  *     1. 编译程序,生成可执行文件a.out;
 8  *    2. 执行a.out;
 9  *    3. 另开一个终端,输入ps -aux命令,可以看到有一个通过a.out命令创建的进程,
10  *        能看到进程ID,假设进程ID为3991(最后要通过这个ID来杀死进程);
11  *     4. 用鼠标关掉开启进程的终端,点击窗口左上角的“X”,然后Close Terminal;
12  *     5. 在刚刚使用ps -aux命令的终端上再次执行ps -aux命令,可以看到该进程还在;
13  *    6. 进到程序指定的路径中,用ls命令可以看到里面多了一个test.txt文件,用vi
14  *        查看可以发现里面的内容正是程序写入的内容;
15  *     7. 杀死进程,通过命令kill -9 3991 (这个3991是进程ID号,根据实际情况而定)
16  */
17 #include <stdio.h>
18 #include <fcntl.h>
19 #include <sys/types.h>
20 #include <sys/stat.h>
21 #include <signal.h>
22 #include <string.h>
23 
24 #define PATH "/home/linux/zkf/test/signal/1/test.txt"
25 
26 /* 自定义信号处理函数 */
27 void func(int sig)
28 {
29     FILE *fp = NULL;
30     char buf[128] = { 0 };
31     
32     switch (sig) {
33         case SIGHUP :   /* 将数据写入文件中 */
34             sprintf(buf, "SIGHUP signal captured. value = %d\n", sig);
35             fp = fopen(PATH, "w");
36             fwrite(buf, 1, strlen(buf), fp);
37             fclose(fp);
38             break;
39     }
40 }
41 
42 int main(int argc, const char *argv[])
43 {
44     int fd = 0;
45     void (*p)(int); /* 定义一个函数指针 */
46 
47     p = signal(SIGHUP, func);
48     printf("p = %p\n", p);
49     while (1) {
50         sleep(5);
51     }
52     
53     return 0;
54 }

操作步骤:

  1. 编译程序,生成可执行文件a.out;

  2. 执行a.out;

  3. 另开一个终端,输入ps -aux命令,可以看到有一个通过a.out命令创建的进程,进程ID为3991

  

   4. 用鼠标关掉开启进程的终端,点击窗口左上角的“X”,然后Close Terminal,如下图:

  

  5. 在刚刚使用ps -aux命令的终端上再次执行ps -aux命令,可以看到该进程还在;

  6. 进到程序指定的路径中,用ls命令可以看到里面多了一个test.txt文件,用vi查看可以发现里面的内容正是程序写入的内容

  

   7. 杀死进程,通过命令kill -9 3991 (这个3991是进程ID号,根据实际情况而定)

2) SIGINT

  该信号在上篇博客中验证了,在此不再赘述。

3) SIGQUIT

  该信号与SIGINT类似,但是由 Ctrl+\ 产生的,将SIGINT的代码稍微改一下,如下:

 1 /**
 2  * filename: signal_3.c
 3  * author: Suzkfly
 4  * date: 2021-02-15
 5  * platform: Ubuntu
 6  *     操作步骤:
 7  *     修改MODE的值为1,2,3,执行程序时按下Ctrl+\,观察结果。
 8  */
 9 #include <stdio.h>
10 #include <fcntl.h>
11 #include <sys/types.h>
12 #include <sys/stat.h>
13 #include <signal.h>
14 
15 #define MODE 3
16 
17 /* 自定义信号处理函数 */
18 void func(int sig)
19 {
20     if (sig == SIGQUIT) {
21         printf("SIGQUIT signal captured\n");
22     }
23 }
24 
25 int main(int argc, const char *argv[])
26 {
27     int fd = 0;
28     void (*p)(int); /* 定义一个函数指针 */
29 
30     printf("MODE = %d\n", MODE);
31 #if (MODE == 1)     /* 忽略 */
32     printf("SIG_IGN = %p\n", SIG_IGN);
33     p = signal(SIGQUIT, SIG_IGN);
34     printf("p = %p\n", p);
35     p = signal(SIGQUIT, SIG_IGN);
36     printf("p = %p\n", p);
37 #elif (MODE == 2)   /* 默认 */
38     printf("SIG_DFL = %p\n", SIG_DFL);
39     p = signal(SIGQUIT, SIG_DFL);
40     printf("p = %p\n", p);
41     p = signal(SIGQUIT, SIG_DFL);
42     printf("p = %p\n", p);
43 #elif (MODE == 3)   /* 自定义 */
44     printf("func = %p\n", func);
45     p = signal(SIGQUIT, func);
46     printf("p = %p\n", p);
47     p = signal(SIGQUIT, func);
48     printf("p = %p\n", p);
49 #endif
50     
51     while (1) {
52         sleep(5);
53         printf("running...\n");
54     }
55     
56     return 0;
57 }

运行结果:

 

 结果分析:

  当MODE为1时,按下 Ctrl+\ 程序不会终止,需要按 Ctrl+C 才会终止,当MODE为2时按下 Ctrl+\ 程序就会终止,当MODE为3时,按下 Ctrl+\ 会执行自己定义的函数,同时能看到SIGQUIT信号也能唤醒进程。那么看上去SIGQUIT与SIGINT的区别就是SIGQUIT会进行Core dump,并且在MODE为2时按下 Ctrl+\ 也能看到提示了(core dumped),然而我在该路径下没有看到core文件,这就不知道为啥了,也许涉及到了我的知识盲区,暂时先搁一边。(已经配置过了ulimit,并且用操作空指针的方法造成段错误,这样在该路径下能够生成core文件,这应该能说明ulimit配置是没问题的)。

4) SIGILL

  导致该信号的原因为非法指令,也就是说可执行文件的指令有问题,这个复现起来有点难,如果编译器没问题的话,编译出来的可执行文件就不会有问题。然鹅程序运行时的指令不一定就是它这个编译器编译出来的啊,这不是还有库文件吗,而且必须用动态库。那么复现SIGILL信号的思路就很清晰了,让一个程序链接到某个动态库,然后在运行程序的时候换掉那个动态库,这不就能导致非法指令了吗。我开始的思路是我在Ubuntu下运行程序,在运行过程中将库文件换成用arm-linux-gcc编译出来的库,但是并没有收到SIGILL信号,而是收到了SIGSEGV信号(无效的内存参考),那看来程序在运行时加载到了arm-linux-gcc的库,而那条指令刚好就是Ubuntu的内存操作指令,真是瞎猫碰上死耗子。为了确保收到SIGILL信号,我制作了一个6K的文件,该文件里面全是0xFF,全是0xFF的指令应该不存在吧。

  生成该文件的代码如下:

 1 #include <sys/types.h>
 2 #include <sys/stat.h>
 3 #include <signal.h>
 4 #include <stdlib.h>
 5 #include <string.h>
 6 
 7 int main(int argc, const char *argv[])
 8 {
 9     int fd = 0;
10     char buf[1024] = { 0 };
11     int i = 0;
12     
13     fd = open("libhello.so", O_RDWR | O_CREAT, 0777);
14     if (fd < 0) {
15         printf("open failed\n");
16         return 0;
17     }
18     
19     memset(buf, 0xff, sizeof(buf));
20     for (i = 0; i < 6; i++) {
21         write(fd, buf, sizeof(buf));
22     }
23     
24     return 0;
25 }

  执行完程序之后能生成libhello.so文件,但它里面的内容为6K的0XFF,将这个文件保存到别的文件夹中。然后编写真正的库文件,代码如下:

  hello.c

1 #include <stdio.h>
2 
3 void hello(void)
4 {
5     printf("Hello World\n");
6 }

  代码非常简单,就是打印Hello World,但是要将这个文件编译成动态库,使用如下指令:

  gcc -fPIC -Wall -c hello.c

  gcc -shared hello.o -o libhello.so

  这样就生成了libhello.so文件,将改文件复制到/lib目录下,当前目录下的libhello.so文件不要删掉,编写程序:

  signal_4.c

 1 /**
 2  * filename: signal_3.c
 3  * author: Suzkfly
 4  * date: 2021-02-15
 5  * platform: Ubuntu
 6  *     操作步骤:
 7  *     1. 将可执行的libhello.so文件拷贝至/lib目录
 8  *     2. 运行程序,每隔5秒打印Hello World
 9  *     3. 在程序运行过程中将全是0xFF的libhello.so文件拷贝至/lib目录
10  *     4. 程序将收到SIGILL信号,并打印"SIGILL signal captured",然后退出。
11  */
12 #include <stdio.h>
13 #include <fcntl.h>
14 #include <sys/types.h>
15 #include <sys/stat.h>
16 #include <signal.h>
17 #include <stdlib.h>
18 
19 
20 /* 自定义信号处理函数 */
21 void func(int sig)
22 {
23     if (sig == SIGILL) {
24         printf("SIGILL signal captured\n");
25         exit(0);        /* 这里一定要退出,因为不知道程序运行会带来什么后果 */
26     }
27 }
28 
29 extern void hello(void);
30 
31 typedef void (*func_t)(int);
32 
33 int main(int argc, const char *argv[])
34 {
35     func_t p;   /* 定义一个函数指针 */
36     
37     p = signal(SIGILL, func);
38     if (p == (func_t)-1) {
39         printf("signal failed\n");
40         return 0;
41     }
42     printf("p = %p\n", p);
43     
44     while (1) {
45         sleep(5);
46         hello();
47     }
48     
49     return 0;
50 }

按文件开始部分的注释操作,运行结果如下:

 5) SIGTRAP

  该信号在调试时使用。我用了各种方法去复现它都没有办法实现,由于我对gdb调试机制并不熟悉,暂时先放一放。