*
*/
+#define inline inline __attribute__((always_inline))
+
#define _GNU_SOURCE
+#define LTT_TRACE
#include <sys/types.h>
#include <sys/wait.h>
#include <unistd.h>
#include <stdlib.h>
#include <sys/param.h>
#include <sys/time.h>
+#include <errno.h>
#include <asm/atomic.h>
#include <asm/timex.h> //for get_cycles()
-#include "ltt-usertrace-fast.h"
+_syscall0(pid_t,gettid)
+
+#include <ltt/ltt-usertrace-fast.h>
+
+#ifdef LTT_SHOW_DEBUG
+#define dbg_printf(...) dbg_printf(__VA_ARGS__)
+#else
+#define dbg_printf(...)
+#endif //LTT_SHOW_DEBUG
+
enum force_switch_mode { FORCE_ACTIVE, FORCE_FLUSH };
/* Writer (the traced application) */
/* signal handling */
static void handler_sigusr1(int signo)
{
- printf("LTT Signal %d received : parent buffer switch.\n", signo);
+ dbg_printf("LTT Signal %d received : parent buffer switch.\n", signo);
}
static void handler_sigusr2(int signo)
{
- printf("LTT Signal %d received : parent exited.\n", signo);
+ dbg_printf("LTT Signal %d received : parent exited.\n", signo);
parent_exited = 1;
}
static void handler_sigalarm(int signo)
{
- printf("LTT Signal %d received\n", signo);
+ dbg_printf("LTT Signal %d received\n", signo);
if(getppid() != traced_pid) {
/* Parent died */
- printf("LTT Parent %lu died, cleaning up\n", traced_pid);
+ dbg_printf("LTT Parent %lu died, cleaning up\n", traced_pid);
traced_pid = 0;
}
alarm(3);
* It can also happen if this is a buffer we never got. */
return -EIO;
} else {
- if(atomic_read(<t_buf->full) == 1) {
+ if(atomic_inc_return(<t_buf->writer_futex) <= 0) {
+ atomic_set(<t_buf->writer_futex, 1);
/* tell the client that buffer is now unfull */
- ret = futex((unsigned long)<t_buf->full,
+ ret = futex((unsigned long)<t_buf->writer_futex,
FUTEX_WAKE, 1, 0, 0, 0);
if(ret != 1) {
- printf("LTT warning : race condition : writer not waiting or too many writers\n");
+ dbg_printf("LTT warning : race condition : writer not waiting or too many writers\n");
}
- atomic_set(<t_buf->full, 0);
}
}
}
{
unsigned int consumed_old;
int err;
- printf("LTT read buffer\n");
+ dbg_printf("LTT read buffer\n");
err = ltt_buffer_get(ltt_buf, &consumed_old);
if(err != 0) {
- if(err != -EAGAIN) printf("LTT Reserving sub buffer failed\n");
+ if(err != -EAGAIN) dbg_printf("LTT Reserving sub buffer failed\n");
goto get_error;
}
if(err != 0) {
if(err == -EIO) {
- printf("Reader has been pushed by the writer, last subbuffer corrupted.\n");
+ dbg_printf("Reader has been pushed by the writer, last subbuffer corrupted.\n");
/* FIXME : we may delete the last written buffer if we wish. */
}
goto get_error;
{
struct sigaction act;
int ret;
- int fd_fac;
- int fd_cpu;
+ int fd_process;
char outfile_name[PATH_MAX];
char identifier_name[PATH_MAX];
traced_pid = l_traced_pid;
traced_tid = l_traced_tid;
- printf("LTT ltt_usertrace_fast_daemon : init is %d, pid is %lu, traced_pid is %lu, traced_tid is %lu\n",
+ dbg_printf("LTT ltt_usertrace_fast_daemon : init is %d, pid is %lu, traced_pid is %lu, traced_tid is %lu\n",
shared_trace_info->init, getpid(), traced_pid, traced_tid);
act.sa_handler = handler_sigusr1;
/* Enable signals */
ret = pthread_sigmask(SIG_SETMASK, &oldset, NULL);
if(ret) {
- printf("LTT Error in pthread_sigmask\n");
+ dbg_printf("LTT Error in pthread_sigmask\n");
}
alarm(3);
}
snprintf(identifier_name, PATH_MAX-1, "%lu.%lu.%llu",
traced_tid, traced_pid, get_cycles());
- snprintf(outfile_name, PATH_MAX-1, "facilities-%s", identifier_name);
- fd_fac = creat(outfile_name, 0644);
-
- snprintf(outfile_name, PATH_MAX-1, "cpu-%s", identifier_name);
- fd_cpu = creat(outfile_name, 0644);
-
+ snprintf(outfile_name, PATH_MAX-1, "process-%s", identifier_name);
+#ifndef LTT_NULL_OUTPUT_TEST
+ fd_process = creat(outfile_name, 0644);
+#else
+ /* NULL test */
+ ret = symlink("/dev/null", outfile_name);
+ if(ret < 0) {
+ perror("error in symlink");
+ }
+ fd_process = open(outfile_name, O_WRONLY);
+ if(fd_process < 0) {
+ perror("Error in open");
+ }
+#endif //LTT_NULL_OUTPUT_TEST
while(1) {
pause();
if(traced_pid == 0) break; /* parent died */
if(parent_exited) break;
- printf("LTT Doing a buffer switch read. pid is : %lu\n", getpid());
+ dbg_printf("LTT Doing a buffer switch read. pid is : %lu\n", getpid());
do {
- ret = read_subbuffer(&shared_trace_info->channel.cpu, fd_cpu);
- } while(ret == 0);
-
- do {
- ret = read_subbuffer(&shared_trace_info->channel.facilities, fd_fac);
+ ret = read_subbuffer(&shared_trace_info->channel.process, fd_process);
} while(ret == 0);
}
/* Buffer force switch (flush). Using FLUSH instead of ACTIVE because we know
* there is no writer. */
- flush_buffer(&shared_trace_info->channel.cpu, FORCE_FLUSH);
+ flush_buffer(&shared_trace_info->channel.process, FORCE_FLUSH);
do {
- ret = read_subbuffer(&shared_trace_info->channel.cpu, fd_cpu);
+ ret = read_subbuffer(&shared_trace_info->channel.process, fd_process);
} while(ret == 0);
- flush_buffer(&shared_trace_info->channel.facilities, FORCE_FLUSH);
- do {
- ret = read_subbuffer(&shared_trace_info->channel.facilities, fd_fac);
- } while(ret == 0);
-
- close(fd_fac);
- close(fd_cpu);
+ close(fd_process);
munmap(shared_trace_info, sizeof(*shared_trace_info));
/* parent : create the shared memory map */
shared_trace_info = mmap(0, sizeof(*thread_trace_info),
PROT_READ|PROT_WRITE, MAP_SHARED|MAP_ANONYMOUS, 0, 0);
- memset(shared_trace_info, 0, sizeof(*shared_trace_info));
+ shared_trace_info->init=0;
+ shared_trace_info->filter=0;
+ shared_trace_info->daemon_id=0;
+ shared_trace_info->nesting=0;
+ memset(&shared_trace_info->channel.process, 0,
+ sizeof(shared_trace_info->channel.process));
/* Tricky semaphore : is in a shared memory space, so it's ok for a fast
* mutex (futex). */
- atomic_set(&shared_trace_info->channel.facilities.full, 0);
- shared_trace_info->channel.facilities.alloc_size = LTT_BUF_SIZE_FACILITIES;
- shared_trace_info->channel.facilities.subbuf_size = LTT_SUBBUF_SIZE_FACILITIES;
- shared_trace_info->channel.facilities.start =
- shared_trace_info->channel.facilities_buf;
- ltt_buffer_begin_callback(&shared_trace_info->channel.facilities,
- ltt_get_timestamp(), 0);
-
- atomic_set(&shared_trace_info->channel.cpu.full, 0);
- shared_trace_info->channel.cpu.alloc_size = LTT_BUF_SIZE_CPU;
- shared_trace_info->channel.cpu.subbuf_size = LTT_SUBBUF_SIZE_CPU;
- shared_trace_info->channel.cpu.start = shared_trace_info->channel.cpu_buf;
- ltt_buffer_begin_callback(&shared_trace_info->channel.cpu,
+ atomic_set(&shared_trace_info->channel.process.writer_futex, LTT_N_SUBBUFS);
+ shared_trace_info->channel.process.alloc_size = LTT_BUF_SIZE_PROCESS;
+ shared_trace_info->channel.process.subbuf_size = LTT_SUBBUF_SIZE_PROCESS;
+ shared_trace_info->channel.process.start =
+ shared_trace_info->channel.process_buf;
+ ltt_buffer_begin_callback(&shared_trace_info->channel.process,
ltt_get_timestamp(), 0);
shared_trace_info->init = 1;
/* Disable signals */
ret = sigfillset(&set);
if(ret) {
- printf("LTT Error in sigfillset\n");
+ dbg_printf("LTT Error in sigfillset\n");
}
ret = pthread_sigmask(SIG_BLOCK, &set, &oldset);
if(ret) {
- printf("LTT Error in pthread_sigmask\n");
+ dbg_printf("LTT Error in pthread_sigmask\n");
}
pid = fork();
/* Enable signals */
ret = pthread_sigmask(SIG_SETMASK, &oldset, NULL);
if(ret) {
- printf("LTT Error in pthread_sigmask\n");
+ dbg_printf("LTT Error in pthread_sigmask\n");
}
} else if(pid == 0) {
+ pid_t sid;
/* Child */
role = LTT_ROLE_READER;
+ sid = setsid();
+ //Not a good idea to renice, unless futex wait eventually implement
+ //priority inheritence.
+ //ret = nice(1);
+ //if(ret < 0) {
+ // perror("Error in nice");
+ //}
+ if(sid < 0) {
+ perror("Error setting sid");
+ }
ltt_usertrace_fast_daemon(shared_trace_info, oldset, l_traced_pid,
l_traced_tid);
/* Should never return */
void __attribute__((constructor)) __ltt_usertrace_fast_init(void)
{
- printf("LTT usertrace-fast init\n");
+ dbg_printf("LTT usertrace-fast init\n");
ltt_rw_init();
}
void __attribute__((destructor)) __ltt_usertrace_fast_fini(void)
{
if(role == LTT_ROLE_WRITER) {
- printf("LTT usertrace-fast fini\n");
+ dbg_printf("LTT usertrace-fast fini\n");
ltt_usertrace_fast_cleanup(NULL);
}
}