Openssh使用alarm接口实现登陆超时警告的功能;这个接口的功能非常单一,有一个全局的变量来记录这个ITIMER_REAL;
https://www.man7.org/linux/man-pages/man2/alarm.2.html
https://www.man7.org/linux/man-pages/man2/setitimer.2.html
c
/*
* We don't want to listen forever unless the other side
* successfully authenticates itself. So we set up an alarm which is
* cleared after successful authentication. A limit of zero
* indicates no limit. Note that we don't set the alarm in debugging
* mode; it is just annoying to have the server exit just when you
* are about to discover the bug.
*/
ssh_signal(SIGALRM, grace_alarm_handler);
if (!debug_flag)
alarm(options.login_grace_time);
如果登陆失败,就清理alarm
c
/*
* Cancel the alarm we set to limit the time taken for
* authentication.
*/
alarm(0);
glibc的实现
c
/* Schedule an alarm. In SECONDS seconds, the process will get a SIGALRM.
If SECONDS is zero, any currently scheduled alarm will be cancelled.
The function returns the number of seconds remaining until the last
alarm scheduled would have signaled, or zero if there wasn't one.
There is no return value to indicate an error, but you can set `errno'
to 0 and check its value after calling `alarm', and this might tell you.
The signal may come late due to processor scheduling. */
unsigned int
alarm (unsigned int seconds)
{
struct itimerval old, new;
unsigned int retval;
new.it_interval.tv_usec = 0;
new.it_interval.tv_sec = 0;
new.it_value.tv_usec = 0;
new.it_value.tv_sec = (long int) seconds;
if (__setitimer (ITIMER_REAL, &new, &old) < 0)
详细的解释:ITIMER_REAL
https://www.man7.org/linux/man-pages/man2/setitimer.2.html
- alarm() 和 setitimer()
级别: 进程级别 (Process-level)。
它们设置的定时器是针对整个进程的。当定时器到期时,内核会向整个进程发送相应的信号(SIGALRM, SIGVTALRM, SIGPROF)。
如果进程是多线程的,这些信号会发送给进程中的任意一个线程(通常是主线程,除非有特定的信号处理配置)。
数量: 每个进程只有一个。
alarm() 只能设置一个实时定时器,新的 alarm() 调用会覆盖旧的。
setitimer() 有三种类型 (ITIMER_REAL, ITIMER_VIRTUAL, ITIMER_PROF),每种类型在每个进程中也只能有一个。设置同类型的定时器会覆盖之前的设置。