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