projects
/
lttng-tools.git
/ blobdiff
commit
grep
author
committer
pickaxe
?
search:
re
summary
|
shortlog
|
log
|
commit
|
commitdiff
|
tree
raw
|
inline
| side by side
Fix: handle new streams in live mode in relayd
[lttng-tools.git]
/
src
/
bin
/
lttng-relayd
/
main.c
diff --git
a/src/bin/lttng-relayd/main.c
b/src/bin/lttng-relayd/main.c
index 7e5733ec8c3253432bd61dbac17b0968fefef21a..730eaf203102886b206aa5d267afc2f8bc9e98b6 100644
(file)
--- a/
src/bin/lttng-relayd/main.c
+++ b/
src/bin/lttng-relayd/main.c
@@
-45,6
+45,7
@@
#include <common/compat/poll.h>
#include <common/compat/socket.h>
#include <common/defaults.h>
#include <common/compat/poll.h>
#include <common/compat/socket.h>
#include <common/defaults.h>
+#include <common/daemonize.h>
#include <common/futex.h>
#include <common/sessiond-comm/sessiond-comm.h>
#include <common/sessiond-comm/inet.h>
#include <common/futex.h>
#include <common/sessiond-comm/sessiond-comm.h>
#include <common/sessiond-comm/inet.h>
@@
-60,10
+61,21
@@
#include "lttng-relayd.h"
#include "live.h"
#include "health-relayd.h"
#include "lttng-relayd.h"
#include "live.h"
#include "health-relayd.h"
+#include "testpoint.h"
/* command line options */
char *opt_output_path;
/* command line options */
char *opt_output_path;
-static int opt_daemon;
+static int opt_daemon, opt_background;
+
+/*
+ * We need to wait for listener and live listener threads, as well as
+ * health check thread, before being ready to signal readiness.
+ */
+#define NR_LTTNG_RELAY_READY 3
+static int lttng_relay_ready = NR_LTTNG_RELAY_READY;
+static int recv_child_signal; /* Set to 1 when a SIGUSR1 signal is received. */
+static pid_t child_ppid; /* Internal parent PID use with daemonize. */
+
static struct lttng_uri *control_uri;
static struct lttng_uri *data_uri;
static struct lttng_uri *live_uri;
static struct lttng_uri *control_uri;
static struct lttng_uri *data_uri;
static struct lttng_uri *live_uri;
@@
-79,7
+91,7
@@
const char * const config_section_name = "relayd";
* Quit pipe for all threads. This permits a single cancellation point
* for all threads when receiving an event on the pipe.
*/
* Quit pipe for all threads. This permits a single cancellation point
* for all threads when receiving an event on the pipe.
*/
-
static
int thread_quit_pipe[2] = { -1, -1 };
+int thread_quit_pipe[2] = { -1, -1 };
/*
* This pipe is used to inform the worker thread that a command is queued and
/*
* This pipe is used to inform the worker thread that a command is queued and
@@
-131,6
+143,7
@@
static struct option long_options[] = {
{ "data-port", 1, 0, 'D', },
{ "live-port", 1, 0, 'L', },
{ "daemonize", 0, 0, 'd', },
{ "data-port", 1, 0, 'D', },
{ "live-port", 1, 0, 'L', },
{ "daemonize", 0, 0, 'd', },
+ { "background", 0, 0, 'b', },
{ "group", 1, 0, 'g', },
{ "help", 0, 0, 'h', },
{ "output", 1, 0, 'o', },
{ "group", 1, 0, 'g', },
{ "help", 0, 0, 'h', },
{ "output", 1, 0, 'o', },
@@
-150,6
+163,7
@@
void usage(void)
fprintf(stderr, "Usage: %s OPTIONS\n\nOptions:\n", progname);
fprintf(stderr, " -h, --help Display this usage.\n");
fprintf(stderr, " -d, --daemonize Start as a daemon.\n");
fprintf(stderr, "Usage: %s OPTIONS\n\nOptions:\n", progname);
fprintf(stderr, " -h, --help Display this usage.\n");
fprintf(stderr, " -d, --daemonize Start as a daemon.\n");
+ fprintf(stderr, " -b, --background Start as a daemon, keeping console open.\n");
fprintf(stderr, " -C, --control-port URL Control port listening.\n");
fprintf(stderr, " -D, --data-port URL Data port listening.\n");
fprintf(stderr, " -L, --live-port URL Live view port listening.\n");
fprintf(stderr, " -C, --control-port URL Control port listening.\n");
fprintf(stderr, " -D, --data-port URL Data port listening.\n");
fprintf(stderr, " -L, --live-port URL Live view port listening.\n");
@@
-210,6
+224,9
@@
int set_option(int opt, const char *arg, const char *optname)
case 'd':
opt_daemon = 1;
break;
case 'd':
opt_daemon = 1;
break;
+ case 'b':
+ opt_background = 1;
+ break;
case 'g':
tracing_group_name = strdup(arg);
tracing_group_name_override = 1;
case 'g':
tracing_group_name = strdup(arg);
tracing_group_name_override = 1;
@@
-504,6
+521,9
@@
void sighandler(int sig)
DBG("SIGTERM caught");
stop_threads();
break;
DBG("SIGTERM caught");
stop_threads();
break;
+ case SIGUSR1:
+ CMM_STORE_SHARED(recv_child_signal, 1);
+ break;
default:
break;
}
default:
break;
}
@@
-543,11
+563,26
@@
int set_signal_handler(void)
return ret;
}
return ret;
}
- DBG("Signal handler set for SIGTERM, SIGPIPE and SIGINT");
+ if ((ret = sigaction(SIGUSR1, &sa, NULL)) < 0) {
+ PERROR("sigaction");
+ return ret;
+ }
+
+ DBG("Signal handler set for SIGTERM, SIGUSR1, SIGPIPE and SIGINT");
return ret;
}
return ret;
}
+void lttng_relay_notify_ready(void)
+{
+ /* Notify the parent of the fork() process that we are ready. */
+ if (opt_daemon || opt_background) {
+ if (uatomic_sub_return(<tng_relay_ready, 1) == 0) {
+ kill(child_ppid, SIGUSR1);
+ }
+ }
+}
+
/*
* Init thread quit pipe.
*
/*
* Init thread quit pipe.
*
@@
-582,7
+617,7
@@
int create_thread_poll_set(struct lttng_poll_event *events, int size)
}
/* Add quit pipe */
}
/* Add quit pipe */
- ret = lttng_poll_add(events, thread_quit_pipe[0], LPOLLIN);
+ ret = lttng_poll_add(events, thread_quit_pipe[0], LPOLLIN
| LPOLLERR
);
if (ret < 0) {
goto error;
}
if (ret < 0) {
goto error;
}
@@
-717,6
+752,12
@@
void *relay_thread_listener(void *data)
goto error_poll_add;
}
goto error_poll_add;
}
+ lttng_relay_notify_ready();
+
+ if (testpoint(relayd_thread_listener)) {
+ goto error_testpoint;
+ }
+
while (1) {
health_code_update();
while (1) {
health_code_update();
@@
-817,6
+858,7
@@
restart:
exit:
error:
error_poll_add:
exit:
error:
error_poll_add:
+error_testpoint:
lttng_poll_clean(&events);
error_create_poll:
if (data_sock->fd >= 0) {
lttng_poll_clean(&events);
error_create_poll:
if (data_sock->fd >= 0) {
@@
-860,6
+902,10
@@
void *relay_thread_dispatcher(void *data)
health_register(health_relayd, HEALTH_RELAYD_TYPE_DISPATCHER);
health_register(health_relayd, HEALTH_RELAYD_TYPE_DISPATCHER);
+ if (testpoint(relayd_thread_dispatcher)) {
+ goto error_testpoint;
+ }
+
health_code_update();
while (!CMM_LOAD_SHARED(dispatch_thread_exit)) {
health_code_update();
while (!CMM_LOAD_SHARED(dispatch_thread_exit)) {
@@
-906,6
+952,7
@@
void *relay_thread_dispatcher(void *data)
err = 0;
error:
err = 0;
error:
+error_testpoint:
if (err) {
health_error();
ERR("Health error occurred in %s", __func__);
if (err) {
health_error();
ERR("Health error occurred in %s", __func__);
@@
-1113,11
+1160,16
@@
int relay_create_session(struct lttcomm_relayd_hdr *recv_hdr,
session->sock = cmd->sock;
session->minor = cmd->minor;
session->major = cmd->major;
session->sock = cmd->sock;
session->minor = cmd->minor;
session->major = cmd->major;
+ pthread_mutex_init(&session->viewer_ready_lock, NULL);
cmd->session = session;
reply.session_id = htobe64(session->id);
switch (cmd->minor) {
cmd->session = session;
reply.session_id = htobe64(session->id);
switch (cmd->minor) {
+ case 1:
+ case 2:
+ case 3:
+ break;
case 4: /* LTTng sessiond 2.4 */
default:
ret = cmd_create_session_2_4(cmd, session);
case 4: /* LTTng sessiond 2.4 */
default:
ret = cmd_create_session_2_4(cmd, session);
@@
-1156,6
+1208,8
@@
void set_viewer_ready_flag(struct relay_command *cmd)
{
struct relay_stream_recv_handle *node, *tmp_node;
{
struct relay_stream_recv_handle *node, *tmp_node;
+ pthread_mutex_lock(&cmd->session->viewer_ready_lock);
+
cds_list_for_each_entry_safe(node, tmp_node, &cmd->recv_head, node) {
struct relay_stream *stream;
cds_list_for_each_entry_safe(node, tmp_node, &cmd->recv_head, node) {
struct relay_stream *stream;
@@
-1166,18
+1220,10
@@
void set_viewer_ready_flag(struct relay_command *cmd)
* Stream is most probably being cleaned up by the data thread thus
* simply continue to the next one.
*/
* Stream is most probably being cleaned up by the data thread thus
* simply continue to the next one.
*/
+ rcu_read_unlock();
continue;
}
continue;
}
- /*
- * If any of the streams in the list doesn't have a ctf_trace assigned,
- * it means that we never received the metadata stream, so we have to
- * wait until it arrives to make the streams available to the viewer.
- */
- if (!stream->ctf_trace) {
- goto end;
- }
-
stream->viewer_ready = 1;
rcu_read_unlock();
stream->viewer_ready = 1;
rcu_read_unlock();
@@
-1186,7
+1232,7
@@
void set_viewer_ready_flag(struct relay_command *cmd)
free(node);
}
free(node);
}
-end:
+ pthread_mutex_unlock(&cmd->session->viewer_ready_lock);
return;
}
return;
}
@@
-2086,6
+2132,11
@@
int relay_streams_sent(struct lttcomm_relayd_hdr *recv_hdr,
*/
set_viewer_ready_flag(cmd);
*/
set_viewer_ready_flag(cmd);
+ /*
+ * Inform the viewer that there are new streams in the session.
+ */
+ uatomic_set(&cmd->session->new_streams, 1);
+
reply.ret_code = htobe32(LTTNG_OK);
send_ret = cmd->sock->ops->sendmsg(cmd->sock, &reply, sizeof(reply), 0);
if (send_ret < 0) {
reply.ret_code = htobe32(LTTNG_OK);
send_ret = cmd->sock->ops->sendmsg(cmd->sock, &reply, sizeof(reply), 0);
if (send_ret < 0) {
@@
-2529,6
+2580,10
@@
void *relay_thread_worker(void *data)
health_register(health_relayd, HEALTH_RELAYD_TYPE_WORKER);
health_register(health_relayd, HEALTH_RELAYD_TYPE_WORKER);
+ if (testpoint(relayd_thread_worker)) {
+ goto error_testpoint;
+ }
+
health_code_update();
/* table of connections indexed on socket */
health_code_update();
/* table of connections indexed on socket */
@@
-2790,6
+2845,7
@@
relay_connections_ht_error:
}
DBG("Worker thread cleanup complete");
free(data_buffer);
}
DBG("Worker thread cleanup complete");
free(data_buffer);
+error_testpoint:
if (err) {
health_error();
ERR("Health error occurred in %s", __func__);
if (err) {
health_error();
ERR("Health error occurred in %s", __func__);
@@
-2822,11
+2878,6
@@
int main(int argc, char **argv)
void *status;
struct relay_local_data *relay_ctx;
void *status;
struct relay_local_data *relay_ctx;
- /* Create thread quit pipe */
- if ((ret = init_thread_quit_pipe()) < 0) {
- goto error;
- }
-
/* Parse arguments */
progname = argv[0];
if ((ret = set_options(argc, argv)) < 0) {
/* Parse arguments */
progname = argv[0];
if ((ret = set_options(argc, argv)) < 0) {
@@
-2852,12
+2903,28
@@
int main(int argc, char **argv)
}
/* Daemonize */
}
/* Daemonize */
- if (opt_daemon) {
- ret = daemon(0, 0);
+ if (opt_daemon || opt_background) {
+ int i;
+
+ ret = lttng_daemonize(&child_ppid, &recv_child_signal,
+ !opt_background);
if (ret < 0) {
if (ret < 0) {
- PERROR("daemon");
goto exit;
}
goto exit;
}
+
+ /*
+ * We are in the child. Make sure all other file
+ * descriptors are closed, in case we are called with
+ * more opened file descriptors than the standard ones.
+ */
+ for (i = 3; i < sysconf(_SC_OPEN_MAX); i++) {
+ (void) close(i);
+ }
+ }
+
+ /* Create thread quit pipe */
+ if ((ret = init_thread_quit_pipe()) < 0) {
+ goto error;
}
/* We need those values for the file/dir creation. */
}
/* We need those values for the file/dir creation. */
@@
-2886,6
+2953,7
@@
int main(int argc, char **argv)
/* Initialize communication library */
lttcomm_init();
/* Initialize communication library */
lttcomm_init();
+ lttcomm_inet_init();
relay_ctx = zmalloc(sizeof(struct relay_local_data));
if (!relay_ctx) {
relay_ctx = zmalloc(sizeof(struct relay_local_data));
if (!relay_ctx) {
@@
-2955,7
+3023,7
@@
int main(int argc, char **argv)
goto exit_listener;
}
goto exit_listener;
}
- ret = live_start_threads(live_uri, relay_ctx
, thread_quit_pipe
);
+ ret = live_start_threads(live_uri, relay_ctx);
if (ret != 0) {
ERR("Starting live viewer threads");
goto exit_live;
if (ret != 0) {
ERR("Starting live viewer threads");
goto exit_live;
This page took
0.028566 seconds
and
4
git commands to generate.