xref: /minix3/external/bsd/bind/dist/unit/atf-src/atf-run/timer.cpp (revision 00b67f09dd46474d133c95011a48590a8e8f94c7)
1*00b67f09SDavid van Moolenbroek //
2*00b67f09SDavid van Moolenbroek // Automated Testing Framework (atf)
3*00b67f09SDavid van Moolenbroek //
4*00b67f09SDavid van Moolenbroek // Copyright (c) 2010 The NetBSD Foundation, Inc.
5*00b67f09SDavid van Moolenbroek // All rights reserved.
6*00b67f09SDavid van Moolenbroek //
7*00b67f09SDavid van Moolenbroek // Redistribution and use in source and binary forms, with or without
8*00b67f09SDavid van Moolenbroek // modification, are permitted provided that the following conditions
9*00b67f09SDavid van Moolenbroek // are met:
10*00b67f09SDavid van Moolenbroek // 1. Redistributions of source code must retain the above copyright
11*00b67f09SDavid van Moolenbroek //    notice, this list of conditions and the following disclaimer.
12*00b67f09SDavid van Moolenbroek // 2. Redistributions in binary form must reproduce the above copyright
13*00b67f09SDavid van Moolenbroek //    notice, this list of conditions and the following disclaimer in the
14*00b67f09SDavid van Moolenbroek //    documentation and/or other materials provided with the distribution.
15*00b67f09SDavid van Moolenbroek //
16*00b67f09SDavid van Moolenbroek // THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND
17*00b67f09SDavid van Moolenbroek // CONTRIBUTORS ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES,
18*00b67f09SDavid van Moolenbroek // INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
19*00b67f09SDavid van Moolenbroek // MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
20*00b67f09SDavid van Moolenbroek // IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS BE LIABLE FOR ANY
21*00b67f09SDavid van Moolenbroek // DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
22*00b67f09SDavid van Moolenbroek // DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE
23*00b67f09SDavid van Moolenbroek // GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
24*00b67f09SDavid van Moolenbroek // INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER
25*00b67f09SDavid van Moolenbroek // IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
26*00b67f09SDavid van Moolenbroek // OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN
27*00b67f09SDavid van Moolenbroek // IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
28*00b67f09SDavid van Moolenbroek //
29*00b67f09SDavid van Moolenbroek 
30*00b67f09SDavid van Moolenbroek #if defined(HAVE_CONFIG_H)
31*00b67f09SDavid van Moolenbroek #   include <bconfig.h>
32*00b67f09SDavid van Moolenbroek #endif
33*00b67f09SDavid van Moolenbroek 
34*00b67f09SDavid van Moolenbroek extern "C" {
35*00b67f09SDavid van Moolenbroek #include <sys/time.h>
36*00b67f09SDavid van Moolenbroek }
37*00b67f09SDavid van Moolenbroek 
38*00b67f09SDavid van Moolenbroek #include <cerrno>
39*00b67f09SDavid van Moolenbroek #include <csignal>
40*00b67f09SDavid van Moolenbroek #include <ctime>
41*00b67f09SDavid van Moolenbroek 
42*00b67f09SDavid van Moolenbroek extern "C" {
43*00b67f09SDavid van Moolenbroek #include "atf-c/defs.h"
44*00b67f09SDavid van Moolenbroek }
45*00b67f09SDavid van Moolenbroek 
46*00b67f09SDavid van Moolenbroek #include "atf-c++/detail/exceptions.hpp"
47*00b67f09SDavid van Moolenbroek #include "atf-c++/detail/sanity.hpp"
48*00b67f09SDavid van Moolenbroek 
49*00b67f09SDavid van Moolenbroek #include "signals.hpp"
50*00b67f09SDavid van Moolenbroek #include "timer.hpp"
51*00b67f09SDavid van Moolenbroek 
52*00b67f09SDavid van Moolenbroek namespace impl = atf::atf_run;
53*00b67f09SDavid van Moolenbroek #define IMPL_NAME "atf::atf_run"
54*00b67f09SDavid van Moolenbroek 
55*00b67f09SDavid van Moolenbroek #if !defined(HAVE_TIMER_T)
56*00b67f09SDavid van Moolenbroek static impl::timer* compat_handle;
57*00b67f09SDavid van Moolenbroek #endif
58*00b67f09SDavid van Moolenbroek 
59*00b67f09SDavid van Moolenbroek // ------------------------------------------------------------------------
60*00b67f09SDavid van Moolenbroek // Auxiliary functions.
61*00b67f09SDavid van Moolenbroek // ------------------------------------------------------------------------
62*00b67f09SDavid van Moolenbroek 
63*00b67f09SDavid van Moolenbroek #if defined(HAVE_TIMER_T)
64*00b67f09SDavid van Moolenbroek static
65*00b67f09SDavid van Moolenbroek void
handler(const int signo ATF_DEFS_ATTRIBUTE_UNUSED,siginfo_t * si,void * uc ATF_DEFS_ATTRIBUTE_UNUSED)66*00b67f09SDavid van Moolenbroek handler(const int signo ATF_DEFS_ATTRIBUTE_UNUSED, siginfo_t* si,
67*00b67f09SDavid van Moolenbroek         void* uc ATF_DEFS_ATTRIBUTE_UNUSED)
68*00b67f09SDavid van Moolenbroek {
69*00b67f09SDavid van Moolenbroek     impl::timer* timer = static_cast< impl::timer* >(si->si_value.sival_ptr);
70*00b67f09SDavid van Moolenbroek     timer->set_fired();
71*00b67f09SDavid van Moolenbroek     timer->timeout_callback();
72*00b67f09SDavid van Moolenbroek }
73*00b67f09SDavid van Moolenbroek #else
74*00b67f09SDavid van Moolenbroek static
75*00b67f09SDavid van Moolenbroek void
handler(const int signo ATF_DEFS_ATTRIBUTE_UNUSED,siginfo_t * si ATF_DEFS_ATTRIBUTE_UNUSED,void * uc ATF_DEFS_ATTRIBUTE_UNUSED)76*00b67f09SDavid van Moolenbroek handler(const int signo ATF_DEFS_ATTRIBUTE_UNUSED,
77*00b67f09SDavid van Moolenbroek         siginfo_t* si ATF_DEFS_ATTRIBUTE_UNUSED,
78*00b67f09SDavid van Moolenbroek         void* uc ATF_DEFS_ATTRIBUTE_UNUSED)
79*00b67f09SDavid van Moolenbroek {
80*00b67f09SDavid van Moolenbroek     compat_handle->set_fired();
81*00b67f09SDavid van Moolenbroek     compat_handle->timeout_callback();
82*00b67f09SDavid van Moolenbroek }
83*00b67f09SDavid van Moolenbroek #endif
84*00b67f09SDavid van Moolenbroek 
85*00b67f09SDavid van Moolenbroek // ------------------------------------------------------------------------
86*00b67f09SDavid van Moolenbroek // The "timer" class.
87*00b67f09SDavid van Moolenbroek // ------------------------------------------------------------------------
88*00b67f09SDavid van Moolenbroek 
89*00b67f09SDavid van Moolenbroek struct impl::timer::impl {
90*00b67f09SDavid van Moolenbroek #if defined(HAVE_TIMER_T)
91*00b67f09SDavid van Moolenbroek     ::timer_t m_timer;
92*00b67f09SDavid van Moolenbroek     ::itimerspec m_old_it;
93*00b67f09SDavid van Moolenbroek #else
94*00b67f09SDavid van Moolenbroek     ::itimerval m_old_it;
95*00b67f09SDavid van Moolenbroek #endif
96*00b67f09SDavid van Moolenbroek 
97*00b67f09SDavid van Moolenbroek     struct ::sigaction m_old_sa;
98*00b67f09SDavid van Moolenbroek     volatile bool m_fired;
99*00b67f09SDavid van Moolenbroek 
implimpl::timer::impl100*00b67f09SDavid van Moolenbroek     impl(void) : m_fired(false)
101*00b67f09SDavid van Moolenbroek     {
102*00b67f09SDavid van Moolenbroek     }
103*00b67f09SDavid van Moolenbroek };
104*00b67f09SDavid van Moolenbroek 
timer(const unsigned int seconds)105*00b67f09SDavid van Moolenbroek impl::timer::timer(const unsigned int seconds) :
106*00b67f09SDavid van Moolenbroek     m_pimpl(new impl())
107*00b67f09SDavid van Moolenbroek {
108*00b67f09SDavid van Moolenbroek     struct ::sigaction sa;
109*00b67f09SDavid van Moolenbroek     sigemptyset(&sa.sa_mask);
110*00b67f09SDavid van Moolenbroek     sa.sa_flags = SA_SIGINFO;
111*00b67f09SDavid van Moolenbroek     sa.sa_sigaction = ::handler;
112*00b67f09SDavid van Moolenbroek     if (::sigaction(SIGALRM, &sa, &m_pimpl->m_old_sa) == -1)
113*00b67f09SDavid van Moolenbroek         throw system_error(IMPL_NAME "::timer::timer",
114*00b67f09SDavid van Moolenbroek                            "Failed to set signal handler", errno);
115*00b67f09SDavid van Moolenbroek 
116*00b67f09SDavid van Moolenbroek #if defined(HAVE_TIMER_T)
117*00b67f09SDavid van Moolenbroek     struct ::sigevent se;
118*00b67f09SDavid van Moolenbroek     se.sigev_notify = SIGEV_SIGNAL;
119*00b67f09SDavid van Moolenbroek     se.sigev_signo = SIGALRM;
120*00b67f09SDavid van Moolenbroek     se.sigev_value.sival_ptr = static_cast< void* >(this);
121*00b67f09SDavid van Moolenbroek     se.sigev_notify_function = NULL;
122*00b67f09SDavid van Moolenbroek     se.sigev_notify_attributes = NULL;
123*00b67f09SDavid van Moolenbroek     if (::timer_create(CLOCK_MONOTONIC, &se, &m_pimpl->m_timer) == -1) {
124*00b67f09SDavid van Moolenbroek         ::sigaction(SIGALRM, &m_pimpl->m_old_sa, NULL);
125*00b67f09SDavid van Moolenbroek         throw system_error(IMPL_NAME "::timer::timer",
126*00b67f09SDavid van Moolenbroek                            "Failed to create timer", errno);
127*00b67f09SDavid van Moolenbroek     }
128*00b67f09SDavid van Moolenbroek 
129*00b67f09SDavid van Moolenbroek     struct ::itimerspec it;
130*00b67f09SDavid van Moolenbroek     it.it_interval.tv_sec = 0;
131*00b67f09SDavid van Moolenbroek     it.it_interval.tv_nsec = 0;
132*00b67f09SDavid van Moolenbroek     it.it_value.tv_sec = seconds;
133*00b67f09SDavid van Moolenbroek     it.it_value.tv_nsec = 0;
134*00b67f09SDavid van Moolenbroek     if (::timer_settime(m_pimpl->m_timer, 0, &it, &m_pimpl->m_old_it) == -1) {
135*00b67f09SDavid van Moolenbroek        ::sigaction(SIGALRM, &m_pimpl->m_old_sa, NULL);
136*00b67f09SDavid van Moolenbroek        ::timer_delete(m_pimpl->m_timer);
137*00b67f09SDavid van Moolenbroek         throw system_error(IMPL_NAME "::timer::timer",
138*00b67f09SDavid van Moolenbroek                            "Failed to program timer", errno);
139*00b67f09SDavid van Moolenbroek     }
140*00b67f09SDavid van Moolenbroek #else
141*00b67f09SDavid van Moolenbroek     ::itimerval it;
142*00b67f09SDavid van Moolenbroek     it.it_interval.tv_sec = 0;
143*00b67f09SDavid van Moolenbroek     it.it_interval.tv_usec = 0;
144*00b67f09SDavid van Moolenbroek     it.it_value.tv_sec = seconds;
145*00b67f09SDavid van Moolenbroek     it.it_value.tv_usec = 0;
146*00b67f09SDavid van Moolenbroek     if (::setitimer(ITIMER_REAL, &it, &m_pimpl->m_old_it) == -1) {
147*00b67f09SDavid van Moolenbroek         ::sigaction(SIGALRM, &m_pimpl->m_old_sa, NULL);
148*00b67f09SDavid van Moolenbroek         throw system_error(IMPL_NAME "::timer::timer",
149*00b67f09SDavid van Moolenbroek                            "Failed to program timer", errno);
150*00b67f09SDavid van Moolenbroek     }
151*00b67f09SDavid van Moolenbroek     INV(compat_handle == NULL);
152*00b67f09SDavid van Moolenbroek     compat_handle = this;
153*00b67f09SDavid van Moolenbroek #endif
154*00b67f09SDavid van Moolenbroek }
155*00b67f09SDavid van Moolenbroek 
~timer(void)156*00b67f09SDavid van Moolenbroek impl::timer::~timer(void)
157*00b67f09SDavid van Moolenbroek {
158*00b67f09SDavid van Moolenbroek #if defined(HAVE_TIMER_T)
159*00b67f09SDavid van Moolenbroek     {
160*00b67f09SDavid van Moolenbroek         const int ret = ::timer_delete(m_pimpl->m_timer);
161*00b67f09SDavid van Moolenbroek         INV(ret != -1);
162*00b67f09SDavid van Moolenbroek     }
163*00b67f09SDavid van Moolenbroek #else
164*00b67f09SDavid van Moolenbroek     {
165*00b67f09SDavid van Moolenbroek         const int ret = ::setitimer(ITIMER_REAL, &m_pimpl->m_old_it, NULL);
166*00b67f09SDavid van Moolenbroek         INV(ret != -1);
167*00b67f09SDavid van Moolenbroek     }
168*00b67f09SDavid van Moolenbroek #endif
169*00b67f09SDavid van Moolenbroek     const int ret = ::sigaction(SIGALRM, &m_pimpl->m_old_sa, NULL);
170*00b67f09SDavid van Moolenbroek     INV(ret != -1);
171*00b67f09SDavid van Moolenbroek 
172*00b67f09SDavid van Moolenbroek #if !defined(HAVE_TIMER_T)
173*00b67f09SDavid van Moolenbroek     compat_handle = NULL;
174*00b67f09SDavid van Moolenbroek #endif
175*00b67f09SDavid van Moolenbroek }
176*00b67f09SDavid van Moolenbroek 
177*00b67f09SDavid van Moolenbroek bool
fired(void) const178*00b67f09SDavid van Moolenbroek impl::timer::fired(void)
179*00b67f09SDavid van Moolenbroek     const
180*00b67f09SDavid van Moolenbroek {
181*00b67f09SDavid van Moolenbroek     return m_pimpl->m_fired;
182*00b67f09SDavid van Moolenbroek }
183*00b67f09SDavid van Moolenbroek 
184*00b67f09SDavid van Moolenbroek void
set_fired(void)185*00b67f09SDavid van Moolenbroek impl::timer::set_fired(void)
186*00b67f09SDavid van Moolenbroek {
187*00b67f09SDavid van Moolenbroek     m_pimpl->m_fired = true;
188*00b67f09SDavid van Moolenbroek }
189*00b67f09SDavid van Moolenbroek 
190*00b67f09SDavid van Moolenbroek // ------------------------------------------------------------------------
191*00b67f09SDavid van Moolenbroek // The "child_timer" class.
192*00b67f09SDavid van Moolenbroek // ------------------------------------------------------------------------
193*00b67f09SDavid van Moolenbroek 
child_timer(const unsigned int seconds,const pid_t pid,volatile bool & terminate)194*00b67f09SDavid van Moolenbroek impl::child_timer::child_timer(const unsigned int seconds, const pid_t pid,
195*00b67f09SDavid van Moolenbroek                                volatile bool& terminate) :
196*00b67f09SDavid van Moolenbroek     timer(seconds),
197*00b67f09SDavid van Moolenbroek     m_pid(pid),
198*00b67f09SDavid van Moolenbroek     m_terminate(terminate)
199*00b67f09SDavid van Moolenbroek {
200*00b67f09SDavid van Moolenbroek }
201*00b67f09SDavid van Moolenbroek 
~child_timer(void)202*00b67f09SDavid van Moolenbroek impl::child_timer::~child_timer(void)
203*00b67f09SDavid van Moolenbroek {
204*00b67f09SDavid van Moolenbroek }
205*00b67f09SDavid van Moolenbroek 
206*00b67f09SDavid van Moolenbroek void
timeout_callback(void)207*00b67f09SDavid van Moolenbroek impl::child_timer::timeout_callback(void)
208*00b67f09SDavid van Moolenbroek {
209*00b67f09SDavid van Moolenbroek     static const timespec ts = { 1, 0 };
210*00b67f09SDavid van Moolenbroek     m_terminate = true;
211*00b67f09SDavid van Moolenbroek     ::kill(-m_pid, SIGTERM);
212*00b67f09SDavid van Moolenbroek     ::nanosleep(&ts, NULL);
213*00b67f09SDavid van Moolenbroek     if (::kill(-m_pid, 0) != -1)
214*00b67f09SDavid van Moolenbroek        ::kill(-m_pid, SIGKILL);
215*00b67f09SDavid van Moolenbroek }
216