strace/delay.c

140 lines
4.1 KiB
C
Raw Normal View History

Implement delay injection Add -e inject=SET:delay_enter= and -e inject=SET:delay_exit= options. * configure.ac (AC_SEARCH_LIBS): Check for timer_create -lrt. * delay.c: New file. * Makefile.am (strace_SOURCES): Add it. (strace_LDADD): Add $(timer_LIBS). * defs.h (INJECT_F_DELAY_ENTER, INJECT_F_DELAY_EXIT, TCB_INJECT_DELAY_EXIT, TCB_DELAYED, inject_delay_exit, syscall_delayed): New macros. (alloc_delay_data, fill_delay_data, is_delay_timer_created, arm_delay_timer, delay_tcb): New prototypes. (struct inject_data): Replace reserved field with delay_idx. (struct tcb): Add delay_expiration_time field. * filter_qualify.c (parse_delay_token): New function. (parse_inject_token): Use it. (qualify_inject_common): Initialize struct inject_opts.data.delay_idx. * strace.c: Include <setjmp.h> (timer_jmp_buf, timer_set): New static variables. (timer_sighandler, restart_delayed_tcb, restart_delayed_tcbs): New functions. (init): Block SIGALRM, set SIGALRM handler. (dispatch_event): Do not restart delayed syscalls. (next_event): Unblock SIGALRM during wait4 invocation. * syscall.c (tamper_with_syscall_entering): Arm delay timer if INJECT_F_DELAY_ENTER injection flag is set, set TCB_INJECT_DELAY_EXIT flag if INJECT_F_DELAY_EXIT injection flag is set. tamper_with_syscall_exiting): Arm delay timer if inject_delay_exit. (syscall_exiting_trace): Call tamper_with_syscall_exiting in case of inject_delay_exit. (syscall_exiting_finish): Clear TCB_INJECT_DELAY_EXIT flag. * strace.1.in: Document delay injection. * NEWS: Mention this improvement. Co-Authored-by: Dmitry V. Levin <ldv@altlinux.org>
2018-02-18 23:21:48 +03:00
/*
* Copyright (c) 2018 The strace developers.
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
* 3. The name of the author may not be used to endorse or promote products
* derived from this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
* IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
* OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
* IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
* NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
* DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
* THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
* THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
#include "defs.h"
struct inject_delay_data {
struct timespec ts_enter;
struct timespec ts_exit;
};
static struct inject_delay_data *delay_data_vec;
static size_t delay_data_vec_capacity; /* size of the arena */
static size_t delay_data_vec_size; /* size of the used arena */
static timer_t delay_timer = (timer_t) -1;
static void
expand_delay_data_vec(void)
{
const size_t old_capacity = delay_data_vec_capacity;
delay_data_vec = xgrowarray(delay_data_vec, &delay_data_vec_capacity,
sizeof(*delay_data_vec));
memset(delay_data_vec + old_capacity, 0,
(delay_data_vec_capacity - old_capacity)
* sizeof(*delay_data_vec));
}
uint16_t
alloc_delay_data(void)
{
const uint16_t rval = delay_data_vec_size;
if (rval < delay_data_vec_size)
error_func_msg_and_die("delay index overflow");
if (delay_data_vec_size == delay_data_vec_capacity)
expand_delay_data_vec();
++delay_data_vec_size;
return rval;
}
void
fill_delay_data(uint16_t delay_idx, int intval, bool isenter)
{
if (delay_idx >= delay_data_vec_size)
error_func_msg_and_die("delay_idx >= delay_data_vec_size");
struct timespec *ts;
if (isenter)
ts = &(delay_data_vec[delay_idx].ts_enter);
else
ts = &(delay_data_vec[delay_idx].ts_exit);
ts->tv_sec = intval / 1000000;
ts->tv_nsec = intval % 1000000 * 1000;
}
bool
is_delay_timer_created(void)
{
return delay_timer != (timer_t) -1;
}
void
arm_delay_timer(const struct tcb *const tcp)
{
const struct itimerspec its = {
.it_value = tcp->delay_expiration_time
};
if (timer_settime(delay_timer, TIMER_ABSTIME, &its, NULL))
perror_msg_and_die("timer_settime");
debug_func_msg("timer set to %lld.%09ld for pid %d",
(long long) tcp->delay_expiration_time.tv_sec,
(long) tcp->delay_expiration_time.tv_nsec,
tcp->pid);
}
void
delay_tcb(struct tcb *tcp, uint16_t delay_idx, bool isenter)
{
if (delay_idx >= delay_data_vec_size)
error_func_msg_and_die("delay_idx >= delay_data_vec_size");
debug_func_msg("delaying pid %d on %s",
tcp->pid, isenter ? "enter" : "exit");
tcp->flags |= TCB_DELAYED;
struct timespec *ts_diff;
if (isenter)
ts_diff = &(delay_data_vec[delay_idx].ts_enter);
else
ts_diff = &(delay_data_vec[delay_idx].ts_exit);
struct timespec ts_now;
clock_gettime(CLOCK_MONOTONIC, &ts_now);
ts_add(&tcp->delay_expiration_time, &ts_now, ts_diff);
if (is_delay_timer_created()) {
struct itimerspec its;
if (timer_gettime(delay_timer, &its))
perror_msg_and_die("timer_gettime");
const struct timespec *const ts_old = &its.it_value;
if (ts_nz(ts_old) && ts_cmp(ts_diff, ts_old) > 0)
return;
} else {
if (timer_create(CLOCK_MONOTONIC, NULL, &delay_timer))
perror_msg_and_die("timer_create");
}
arm_delay_timer(tcp);
}