理论上来说,信号处理函数里面调用“可重入”函数是安全的,比如下面的代码:
大部分情况下,上面的代码执行正常,因为权限不足,rmdir("/home")会返回错误,并且得到以下输出:
signal.c:23 rmdir error: Permission denied
但是如果把第9行的注释打开,程序的输出结果就不太对了:
signal.c:23 rmdir error:No such file or directory
为什么呢,根据输出的错误信息我们知道,两种情况下strerror返回的字符串不一样,也就是errno不一样。原因在于第8行调用rmdir后,errno被设置成了“Permission denied”但是在第9行发出了信号,于是程序跳到io_handler中执行chdir,因为chdir执行出错(目标目录不存在)所以errno被改成了“No such file or directory”此时io_handler函数返回,第12行的时候errno已经是chdir设置的了,而rmdir设置的errno已经丢了。
根本原因在于errno是一个全局变量,在io_handler中可以对它进行修改,从而导致第12行得不到准确的出错提示。可以这样修改io_handler函数:
把errno保存起来,在io_handler时再恢复。虽然可以解决问题,但是还是不建议在信号处理函数里面做太复杂的逻辑,一般来说,设置一个flag就行了。