/* * Calibrate LTTng. * * Returns a CMD_* error. */ static int calibrate_lttng(void) { int ret = CMD_SUCCESS; struct lttng_domain dom; struct lttng_calibrate calibrate; memset(&dom, 0, sizeof(dom)); memset(&calibrate, 0, sizeof(calibrate)); /* Create lttng domain */ if (opt_kernel) { dom.type = LTTNG_DOMAIN_KERNEL; } else if (opt_userspace) { dom.type = LTTNG_DOMAIN_UST; } else { ERR("Please specify a tracer (-k/--kernel or -u/--userspace)"); ret = CMD_ERROR; goto error; } handle = lttng_create_handle(NULL, &dom); if (handle == NULL) { ret = CMD_ERROR; goto error; } switch (opt_event_type) { case LTTNG_EVENT_TRACEPOINT: DBG("Calibrating kernel tracepoints"); break; case LTTNG_EVENT_PROBE: DBG("Calibrating kernel probes"); break; case LTTNG_EVENT_FUNCTION: DBG("Calibrating kernel functions"); calibrate.type = LTTNG_CALIBRATE_FUNCTION; ret = lttng_calibrate(handle, &calibrate); if (ret < 0) { goto error; } MSG("%s calibration done", opt_kernel ? "Kernel" : "UST"); break; case LTTNG_EVENT_FUNCTION_ENTRY: DBG("Calibrating kernel function entry"); break; case LTTNG_EVENT_SYSCALL: DBG("Calibrating kernel syscall"); break; default: ret = CMD_UNDEFINED; goto error; } ret = CMD_SUCCESS; error: lttng_destroy_handle(handle); return ret; }
/* * Ask session daemon for all user space tracepoints available. */ static int list_ust_events(void) { int i, size; struct lttng_domain domain; struct lttng_handle *handle; struct lttng_event *event_list; pid_t cur_pid = 0; char *cmdline = NULL; memset(&domain, 0, sizeof(domain)); DBG("Getting UST tracing events"); domain.type = LTTNG_DOMAIN_UST; handle = lttng_create_handle(NULL, &domain); if (handle == NULL) { goto error; } size = lttng_list_tracepoints(handle, &event_list); if (size < 0) { ERR("Unable to list UST events: %s", lttng_strerror(size)); lttng_destroy_handle(handle); return size; } MSG("UST events:\n-------------"); if (size == 0) { MSG("None"); } for (i = 0; i < size; i++) { if (cur_pid != event_list[i].pid) { cur_pid = event_list[i].pid; cmdline = get_cmdline_by_pid(cur_pid); MSG("\nPID: %d - Name: %s", cur_pid, cmdline); free(cmdline); } print_events(&event_list[i]); } MSG(""); free(event_list); lttng_destroy_handle(handle); return CMD_SUCCESS; error: lttng_destroy_handle(handle); return -1; }
/* * Ask for all trace events in the kernel */ static int list_kernel_events(void) { int i, size, ret = CMD_SUCCESS; struct lttng_domain domain; struct lttng_handle *handle; struct lttng_event *event_list; memset(&domain, 0, sizeof(domain)); DBG("Getting kernel tracing events"); domain.type = LTTNG_DOMAIN_KERNEL; handle = lttng_create_handle(NULL, &domain); if (handle == NULL) { ret = CMD_ERROR; goto error; } size = lttng_list_tracepoints(handle, &event_list); if (size < 0) { ERR("Unable to list kernel events: %s", lttng_strerror(size)); lttng_destroy_handle(handle); return CMD_ERROR; } if (lttng_opt_mi) { /* Mi print */ ret = mi_list_kernel_events(event_list, size, &domain); if (ret) { ret = CMD_ERROR; goto end; } } else { MSG("Kernel events:\n-------------"); for (i = 0; i < size; i++) { print_events(&event_list[i]); } MSG(""); } end: free(event_list); lttng_destroy_handle(handle); return ret; error: lttng_destroy_handle(handle); return ret; }
static int setup_session(const char *session_name, const char *path) { int ret; struct lttng_domain dom; struct lttng_event ev; struct lttng_handle *chan_handle; printf("Creating session %s\n", session_name); ret = lttng_create_session(session_name, path); if (ret) { fprintf(stderr, "Failed to create session, ret = %d\n", ret); goto end; } dom.type = LTTNG_DOMAIN_KERNEL; dom.buf_type = LTTNG_BUFFER_GLOBAL; chan_handle = lttng_create_handle(session_name, &dom); if (chan_handle == NULL) { ret = -1; goto end; } memset(&ev, 0, sizeof(ev)); ev.type = LTTNG_EVENT_SYSCALL; strcpy(ev.name, "*"); ev.loglevel_type = LTTNG_EVENT_LOGLEVEL_ALL; ret = lttng_enable_event_with_exclusions(chan_handle, &ev, "mychan", NULL, 0, NULL); if (ret < 0) { fprintf(stderr, "Failed to enable events (ret = %i)\n", ret); goto end; } printf("Enabled all system call kernel events\n"); ret = lttng_start_tracing(session_name); if (ret < 0) { fprintf(stderr, "Failed to start tracing\n"); goto end; } lttng_destroy_handle(chan_handle); ret = 0; end: return ret; }
/* * Ask for all trace events in the kernel and pretty print them. */ static int list_kernel_events(void) { int i, size; struct lttng_domain domain; struct lttng_handle *handle; struct lttng_event *event_list; memset(&domain, 0, sizeof(domain)); DBG("Getting kernel tracing events"); domain.type = LTTNG_DOMAIN_KERNEL; handle = lttng_create_handle(NULL, &domain); if (handle == NULL) { goto error; } size = lttng_list_tracepoints(handle, &event_list); if (size < 0) { ERR("Unable to list kernel events"); lttng_destroy_handle(handle); return size; } MSG("Kernel events:\n-------------"); for (i = 0; i < size; i++) { print_events(&event_list[i]); } MSG(""); free(event_list); lttng_destroy_handle(handle); return CMD_SUCCESS; error: lttng_destroy_handle(handle); return -1; }
/* * disable_events * * Disabling event using the lttng API. */ static int disable_events(char *session_name) { int ret = CMD_SUCCESS, warn = 0, command_ret = CMD_SUCCESS; int enabled = 1, success = 1; char *event_name, *channel_name = NULL; struct lttng_domain dom; struct lttng_event event; memset(&dom, 0, sizeof(dom)); /* Create lttng domain */ if (opt_kernel) { dom.type = LTTNG_DOMAIN_KERNEL; } else if (opt_userspace) { dom.type = LTTNG_DOMAIN_UST; } else if (opt_jul) { dom.type = LTTNG_DOMAIN_JUL; } else if (opt_log4j) { dom.type = LTTNG_DOMAIN_LOG4J; } else if (opt_python) { dom.type = LTTNG_DOMAIN_PYTHON; } else { /* Checked by the caller. */ assert(0); } channel_name = opt_channel_name; handle = lttng_create_handle(session_name, &dom); if (handle == NULL) { ret = -1; goto error; } /* Mi print the channel and open the events element */ if (lttng_opt_mi) { ret = mi_lttng_writer_open_element(writer, config_element_channel); if (ret) { ret = CMD_ERROR; goto end; } ret = mi_lttng_writer_write_element_string(writer, config_element_name, print_channel_name(channel_name)); if (ret) { ret = CMD_ERROR; goto end; } /* Open events element */ ret = mi_lttng_writer_open_element(writer, config_element_events); if (ret) { ret = CMD_ERROR; goto end; } } memset(&event, 0, sizeof(event)); /* Set default loglevel to any/unknown */ event.loglevel = -1; /* opt_event_type contain the event type to disable at this point */ event.type = opt_event_type; if (opt_disable_all) { command_ret = lttng_disable_event_ext(handle, &event, channel_name, NULL); if (command_ret < 0) { ERR("%s", lttng_strerror(command_ret)); enabled = 1; success = 0; } else { enabled = 0; success = 1; MSG("All %s events of type %s are disabled in channel %s", get_domain_str(dom.type), print_event_type(opt_event_type), print_channel_name(channel_name)); } if (lttng_opt_mi) { ret = mi_print_event("*", enabled, success); if (ret) { ret = CMD_ERROR; goto error; } } } else { /* Strip event list */ event_name = strtok(opt_event_list, ","); while (event_name != NULL) { DBG("Disabling event %s", event_name); strncpy(event.name, event_name, sizeof(event.name)); event.name[sizeof(event.name) - 1] = '\0'; command_ret = lttng_disable_event_ext(handle, &event, channel_name, NULL); if (command_ret < 0) { ERR("%s of type %s : %s (channel %s, session %s)", event_name, print_event_type(opt_event_type), lttng_strerror(command_ret), command_ret == -LTTNG_ERR_NEED_CHANNEL_NAME ? print_raw_channel_name(channel_name) : print_channel_name(channel_name), session_name); warn = 1; success = 0; /* * If an error occurred we assume that the event is still * enabled. */ enabled = 1; } else { MSG("%s %s of type %s disabled in channel %s for session %s", get_domain_str(dom.type), event_name, print_event_type(opt_event_type), print_channel_name(channel_name), session_name); success = 1; enabled = 0; } if (lttng_opt_mi) { ret = mi_print_event(event_name, enabled, success); if (ret) { ret = CMD_ERROR; goto error; } } /* Next event */ event_name = strtok(NULL, ","); } } end: if (lttng_opt_mi) { /* Close events element and channel element */ ret = mi_lttng_close_multi_element(writer, 2); if (ret) { ret = CMD_ERROR; } } error: /* if there is already an error preserve it */ if (warn && !ret) { ret = CMD_WARNING; } /* Overwrite ret if an error occurred */ ret = command_ret ? command_ret : ret; lttng_destroy_handle(handle); return ret; }
/* * Enabling event using the lttng API. * Note: in case of error only the last error code will be return. */ static int enable_events(char *session_name) { int ret = CMD_SUCCESS, command_ret = CMD_SUCCESS; int error_holder = CMD_SUCCESS, warn = 0, error = 0, success = 1; char *event_name, *channel_name = NULL; struct lttng_event *ev; struct lttng_domain dom; char **exclusion_list = NULL; memset(&dom, 0, sizeof(dom)); ev = lttng_event_create(); if (!ev) { ret = CMD_ERROR; goto error; } if (opt_kernel) { if (opt_loglevel) { WARN("Kernel loglevels are not supported."); } } /* Create lttng domain */ if (opt_kernel) { dom.type = LTTNG_DOMAIN_KERNEL; dom.buf_type = LTTNG_BUFFER_GLOBAL; } else if (opt_userspace) { dom.type = LTTNG_DOMAIN_UST; /* Default. */ dom.buf_type = LTTNG_BUFFER_PER_UID; } else if (opt_jul) { dom.type = LTTNG_DOMAIN_JUL; /* Default. */ dom.buf_type = LTTNG_BUFFER_PER_UID; } else if (opt_log4j) { dom.type = LTTNG_DOMAIN_LOG4J; /* Default. */ dom.buf_type = LTTNG_BUFFER_PER_UID; } else if (opt_python) { dom.type = LTTNG_DOMAIN_PYTHON; /* Default. */ dom.buf_type = LTTNG_BUFFER_PER_UID; } else { /* Checked by the caller. */ assert(0); } if (opt_exclude) { switch (dom.type) { case LTTNG_DOMAIN_KERNEL: case LTTNG_DOMAIN_JUL: case LTTNG_DOMAIN_LOG4J: case LTTNG_DOMAIN_PYTHON: ERR("Event name exclusions are not yet implemented for %s events", get_domain_str(dom.type)); ret = CMD_ERROR; goto error; case LTTNG_DOMAIN_UST: /* Exclusions supported */ break; default: assert(0); } } /* * Adding a filter to a probe, function or userspace-probe would be * denied by the kernel tracer as it's not supported at the moment. We * do an early check here to warn the user. */ if (opt_filter && opt_kernel) { switch (opt_event_type) { case LTTNG_EVENT_ALL: case LTTNG_EVENT_TRACEPOINT: case LTTNG_EVENT_SYSCALL: break; case LTTNG_EVENT_PROBE: case LTTNG_EVENT_USERSPACE_PROBE: case LTTNG_EVENT_FUNCTION: ERR("Filter expressions are not supported for %s events", get_event_type_str(opt_event_type)); ret = CMD_ERROR; goto error; default: ret = CMD_UNDEFINED; goto error; } } channel_name = opt_channel_name; handle = lttng_create_handle(session_name, &dom); if (handle == NULL) { ret = -1; goto error; } /* Prepare Mi */ if (lttng_opt_mi) { /* Open a events element */ ret = mi_lttng_writer_open_element(writer, config_element_events); if (ret) { ret = CMD_ERROR; goto error; } } if (opt_enable_all) { /* Default setup for enable all */ if (opt_kernel) { ev->type = opt_event_type; strcpy(ev->name, "*"); /* kernel loglevels not implemented */ ev->loglevel_type = LTTNG_EVENT_LOGLEVEL_ALL; } else { ev->type = LTTNG_EVENT_TRACEPOINT; strcpy(ev->name, "*"); ev->loglevel_type = opt_loglevel_type; if (opt_loglevel) { assert(opt_userspace || opt_jul || opt_log4j || opt_python); if (opt_userspace) { ev->loglevel = loglevel_str_to_value(opt_loglevel); } else if (opt_jul) { ev->loglevel = loglevel_jul_str_to_value(opt_loglevel); } else if (opt_log4j) { ev->loglevel = loglevel_log4j_str_to_value(opt_loglevel); } else if (opt_python) { ev->loglevel = loglevel_python_str_to_value(opt_loglevel); } if (ev->loglevel == -1) { ERR("Unknown loglevel %s", opt_loglevel); ret = -LTTNG_ERR_INVALID; goto error; } } else { assert(opt_userspace || opt_jul || opt_log4j || opt_python); if (opt_userspace) { ev->loglevel = -1; } else if (opt_jul) { ev->loglevel = LTTNG_LOGLEVEL_JUL_ALL; } else if (opt_log4j) { ev->loglevel = LTTNG_LOGLEVEL_LOG4J_ALL; } else if (opt_python) { ev->loglevel = LTTNG_LOGLEVEL_PYTHON_DEBUG; } } } if (opt_exclude) { ret = create_exclusion_list_and_validate("*", opt_exclude, &exclusion_list); if (ret) { ret = CMD_ERROR; goto error; } ev->exclusion = 1; warn_on_truncated_exclusion_names(exclusion_list, &warn); } if (!opt_filter) { ret = lttng_enable_event_with_exclusions(handle, ev, channel_name, NULL, exclusion_list ? strutils_array_of_strings_len(exclusion_list) : 0, exclusion_list); if (ret < 0) { switch (-ret) { case LTTNG_ERR_KERN_EVENT_EXIST: WARN("Kernel events already enabled (channel %s, session %s)", print_channel_name(channel_name), session_name); warn = 1; break; case LTTNG_ERR_TRACE_ALREADY_STARTED: { const char *msg = "The command tried to enable an event in a new domain for a session that has already been started once."; ERR("Events: %s (channel %s, session %s)", msg, print_channel_name(channel_name), session_name); error = 1; break; } default: ERR("Events: %s (channel %s, session %s)", lttng_strerror(ret), ret == -LTTNG_ERR_NEED_CHANNEL_NAME ? print_raw_channel_name(channel_name) : print_channel_name(channel_name), session_name); error = 1; break; } goto end; } switch (opt_event_type) { case LTTNG_EVENT_TRACEPOINT: if (opt_loglevel && dom.type != LTTNG_DOMAIN_KERNEL) { char *exclusion_string = print_exclusions(exclusion_list); if (!exclusion_string) { PERROR("Cannot allocate exclusion_string"); error = 1; goto end; } MSG("All %s tracepoints%s are enabled in channel %s for loglevel %s", get_domain_str(dom.type), exclusion_string, print_channel_name(channel_name), opt_loglevel); free(exclusion_string); } else { char *exclusion_string = print_exclusions(exclusion_list); if (!exclusion_string) { PERROR("Cannot allocate exclusion_string"); error = 1; goto end; } MSG("All %s tracepoints%s are enabled in channel %s", get_domain_str(dom.type), exclusion_string, print_channel_name(channel_name)); free(exclusion_string); } break; case LTTNG_EVENT_SYSCALL: if (opt_kernel) { MSG("All %s system calls are enabled in channel %s", get_domain_str(dom.type), print_channel_name(channel_name)); } break; case LTTNG_EVENT_ALL: if (opt_loglevel && dom.type != LTTNG_DOMAIN_KERNEL) { char *exclusion_string = print_exclusions(exclusion_list); if (!exclusion_string) { PERROR("Cannot allocate exclusion_string"); error = 1; goto end; } MSG("All %s events%s are enabled in channel %s for loglevel %s", get_domain_str(dom.type), exclusion_string, print_channel_name(channel_name), opt_loglevel); free(exclusion_string); } else { char *exclusion_string = print_exclusions(exclusion_list); if (!exclusion_string) { PERROR("Cannot allocate exclusion_string"); error = 1; goto end; } MSG("All %s events%s are enabled in channel %s", get_domain_str(dom.type), exclusion_string, print_channel_name(channel_name)); free(exclusion_string); } break; default: /* * We should not be here since lttng_enable_event should have * failed on the event type. */ goto error; } } if (opt_filter) { command_ret = lttng_enable_event_with_exclusions(handle, ev, channel_name, opt_filter, exclusion_list ? strutils_array_of_strings_len(exclusion_list) : 0, exclusion_list); if (command_ret < 0) { switch (-command_ret) { case LTTNG_ERR_FILTER_EXIST: WARN("Filter on all events is already enabled" " (channel %s, session %s)", print_channel_name(channel_name), session_name); warn = 1; break; case LTTNG_ERR_TRACE_ALREADY_STARTED: { const char *msg = "The command tried to enable an event in a new domain for a session that has already been started once."; ERR("All events: %s (channel %s, session %s, filter \'%s\')", msg, print_channel_name(channel_name), session_name, opt_filter); error = 1; break; } default: ERR("All events: %s (channel %s, session %s, filter \'%s\')", lttng_strerror(command_ret), command_ret == -LTTNG_ERR_NEED_CHANNEL_NAME ? print_raw_channel_name(channel_name) : print_channel_name(channel_name), session_name, opt_filter); error = 1; break; } error_holder = command_ret; } else { ev->filter = 1; MSG("Filter '%s' successfully set", opt_filter); } } if (lttng_opt_mi) { /* The wildcard * is used for kernel and ust domain to * represent ALL. We copy * in event name to force the wildcard use * for kernel domain * * Note: this is strictly for semantic and printing while in * machine interface mode. */ strcpy(ev->name, "*"); /* If we reach here the events are enabled */ if (!error && !warn) { ev->enabled = 1; } else { ev->enabled = 0; success = 0; } ret = mi_lttng_event(writer, ev, 1, handle->domain.type); if (ret) { ret = CMD_ERROR; goto error; } /* print exclusion */ ret = mi_print_exclusion(exclusion_list); if (ret) { ret = CMD_ERROR; goto error; } /* Success ? */ ret = mi_lttng_writer_write_element_bool(writer, mi_lttng_element_command_success, success); if (ret) { ret = CMD_ERROR; goto error; } /* Close event element */ ret = mi_lttng_writer_close_element(writer); if (ret) { ret = CMD_ERROR; goto error; } } goto end; } /* Strip event list */ event_name = strtok(opt_event_list, ","); while (event_name != NULL) { /* Copy name and type of the event */ strncpy(ev->name, event_name, LTTNG_SYMBOL_NAME_LEN); ev->name[LTTNG_SYMBOL_NAME_LEN - 1] = '\0'; ev->type = opt_event_type; /* Kernel tracer action */ if (opt_kernel) { DBG("Enabling kernel event %s for channel %s", event_name, print_channel_name(channel_name)); switch (opt_event_type) { case LTTNG_EVENT_ALL: /* Enable tracepoints and syscalls */ /* If event name differs from *, select tracepoint. */ if (strcmp(ev->name, "*")) { ev->type = LTTNG_EVENT_TRACEPOINT; } break; case LTTNG_EVENT_TRACEPOINT: break; case LTTNG_EVENT_PROBE: ret = parse_probe_opts(ev, opt_probe); if (ret) { ERR("Unable to parse probe options"); ret = CMD_ERROR; goto error; } break; case LTTNG_EVENT_USERSPACE_PROBE: ret = parse_userspace_probe_opts(ev, opt_userspace_probe); if (ret) { switch (ret) { case CMD_UNSUPPORTED: /* * Error message describing * what is not supported was * printed in the function. */ break; case CMD_ERROR: default: ERR("Unable to parse userspace probe options"); break; } goto error; } break; case LTTNG_EVENT_FUNCTION: ret = parse_probe_opts(ev, opt_function); if (ret) { ERR("Unable to parse function probe options"); ret = CMD_ERROR; goto error; } break; case LTTNG_EVENT_SYSCALL: ev->type = LTTNG_EVENT_SYSCALL; break; default: ret = CMD_UNDEFINED; goto error; } /* kernel loglevels not implemented */ ev->loglevel_type = LTTNG_EVENT_LOGLEVEL_ALL; } else if (opt_userspace) { /* User-space tracer action */ DBG("Enabling UST event %s for channel %s, loglevel %s", event_name, print_channel_name(channel_name), opt_loglevel ? : "<all>"); switch (opt_event_type) { case LTTNG_EVENT_ALL: /* Default behavior is tracepoint */ /* Fall-through */ case LTTNG_EVENT_TRACEPOINT: /* Copy name and type of the event */ ev->type = LTTNG_EVENT_TRACEPOINT; strncpy(ev->name, event_name, LTTNG_SYMBOL_NAME_LEN); ev->name[LTTNG_SYMBOL_NAME_LEN - 1] = '\0'; break; case LTTNG_EVENT_PROBE: case LTTNG_EVENT_FUNCTION: case LTTNG_EVENT_SYSCALL: case LTTNG_EVENT_USERSPACE_PROBE: default: ERR("Event type not available for user-space tracing"); ret = CMD_UNSUPPORTED; goto error; } if (opt_exclude) { ev->exclusion = 1; if (opt_event_type != LTTNG_EVENT_ALL && opt_event_type != LTTNG_EVENT_TRACEPOINT) { ERR("Exclusion option can only be used with tracepoint events"); ret = CMD_ERROR; goto error; } /* Free previously allocated items */ strutils_free_null_terminated_array_of_strings( exclusion_list); exclusion_list = NULL; ret = create_exclusion_list_and_validate( event_name, opt_exclude, &exclusion_list); if (ret) { ret = CMD_ERROR; goto error; } warn_on_truncated_exclusion_names( exclusion_list, &warn); } ev->loglevel_type = opt_loglevel_type; if (opt_loglevel) { ev->loglevel = loglevel_str_to_value(opt_loglevel); if (ev->loglevel == -1) { ERR("Unknown loglevel %s", opt_loglevel); ret = -LTTNG_ERR_INVALID; goto error; } } else { ev->loglevel = -1; } } else if (opt_jul || opt_log4j || opt_python) { if (opt_event_type != LTTNG_EVENT_ALL && opt_event_type != LTTNG_EVENT_TRACEPOINT) { ERR("Event type not supported for domain."); ret = CMD_UNSUPPORTED; goto error; } ev->loglevel_type = opt_loglevel_type; if (opt_loglevel) { if (opt_jul) { ev->loglevel = loglevel_jul_str_to_value(opt_loglevel); } else if (opt_log4j) { ev->loglevel = loglevel_log4j_str_to_value(opt_loglevel); } else if (opt_python) { ev->loglevel = loglevel_python_str_to_value(opt_loglevel); } if (ev->loglevel == -1) { ERR("Unknown loglevel %s", opt_loglevel); ret = -LTTNG_ERR_INVALID; goto error; } } else { if (opt_jul) { ev->loglevel = LTTNG_LOGLEVEL_JUL_ALL; } else if (opt_log4j) { ev->loglevel = LTTNG_LOGLEVEL_LOG4J_ALL; } else if (opt_python) { ev->loglevel = LTTNG_LOGLEVEL_PYTHON_DEBUG; } } ev->type = LTTNG_EVENT_TRACEPOINT; strncpy(ev->name, event_name, LTTNG_SYMBOL_NAME_LEN); ev->name[LTTNG_SYMBOL_NAME_LEN - 1] = '\0'; } else { assert(0); } if (!opt_filter) { char *exclusion_string; command_ret = lttng_enable_event_with_exclusions(handle, ev, channel_name, NULL, exclusion_list ? strutils_array_of_strings_len(exclusion_list) : 0, exclusion_list); exclusion_string = print_exclusions(exclusion_list); if (!exclusion_string) { PERROR("Cannot allocate exclusion_string"); error = 1; goto end; } if (command_ret < 0) { /* Turn ret to positive value to handle the positive error code */ switch (-command_ret) { case LTTNG_ERR_KERN_EVENT_EXIST: WARN("Kernel event %s%s already enabled (channel %s, session %s)", event_name, exclusion_string, print_channel_name(channel_name), session_name); warn = 1; break; case LTTNG_ERR_TRACE_ALREADY_STARTED: { const char *msg = "The command tried to enable an event in a new domain for a session that has already been started once."; ERR("Event %s%s: %s (channel %s, session %s)", event_name, exclusion_string, msg, print_channel_name(channel_name), session_name); error = 1; break; } case LTTNG_ERR_SDT_PROBE_SEMAPHORE: ERR("SDT probes %s guarded by semaphores are not supported (channel %s, session %s)", event_name, print_channel_name(channel_name), session_name); error = 1; break; default: ERR("Event %s%s: %s (channel %s, session %s)", event_name, exclusion_string, lttng_strerror(command_ret), command_ret == -LTTNG_ERR_NEED_CHANNEL_NAME ? print_raw_channel_name(channel_name) : print_channel_name(channel_name), session_name); error = 1; break; } error_holder = command_ret; } else { switch (dom.type) { case LTTNG_DOMAIN_KERNEL: case LTTNG_DOMAIN_UST: MSG("%s event %s%s created in channel %s", get_domain_str(dom.type), event_name, exclusion_string, print_channel_name(channel_name)); break; case LTTNG_DOMAIN_JUL: case LTTNG_DOMAIN_LOG4J: case LTTNG_DOMAIN_PYTHON: /* * Don't print the default channel * name for agent domains. */ MSG("%s event %s%s enabled", get_domain_str(dom.type), event_name, exclusion_string); break; default: assert(0); } } free(exclusion_string); } if (opt_filter) { char *exclusion_string; /* Filter present */ ev->filter = 1; command_ret = lttng_enable_event_with_exclusions(handle, ev, channel_name, opt_filter, exclusion_list ? strutils_array_of_strings_len(exclusion_list) : 0, exclusion_list); exclusion_string = print_exclusions(exclusion_list); if (!exclusion_string) { PERROR("Cannot allocate exclusion_string"); error = 1; goto end; } if (command_ret < 0) { switch (-command_ret) { case LTTNG_ERR_FILTER_EXIST: WARN("Filter on event %s%s is already enabled" " (channel %s, session %s)", event_name, exclusion_string, print_channel_name(channel_name), session_name); warn = 1; break; case LTTNG_ERR_TRACE_ALREADY_STARTED: { const char *msg = "The command tried to enable an event in a new domain for a session that has already been started once."; ERR("Event %s%s: %s (channel %s, session %s, filter \'%s\')", ev->name, exclusion_string, msg, print_channel_name(channel_name), session_name, opt_filter); error = 1; break; } default: ERR("Event %s%s: %s (channel %s, session %s, filter \'%s\')", ev->name, exclusion_string, lttng_strerror(command_ret), command_ret == -LTTNG_ERR_NEED_CHANNEL_NAME ? print_raw_channel_name(channel_name) : print_channel_name(channel_name), session_name, opt_filter); error = 1; break; } error_holder = command_ret; } else { MSG("Event %s%s: Filter '%s' successfully set", event_name, exclusion_string, opt_filter); } free(exclusion_string); } if (lttng_opt_mi) { if (command_ret) { success = 0; ev->enabled = 0; } else { ev->enabled = 1; } ret = mi_lttng_event(writer, ev, 1, handle->domain.type); if (ret) { ret = CMD_ERROR; goto error; } /* print exclusion */ ret = mi_print_exclusion(exclusion_list); if (ret) { ret = CMD_ERROR; goto error; } /* Success ? */ ret = mi_lttng_writer_write_element_bool(writer, mi_lttng_element_command_success, success); if (ret) { ret = CMD_ERROR; goto end; } /* Close event element */ ret = mi_lttng_writer_close_element(writer); if (ret) { ret = CMD_ERROR; goto end; } } /* Next event */ event_name = strtok(NULL, ","); /* Reset warn, error and success */ success = 1; }
/* * Enabling event using the lttng API. * Note: in case of error only the last error code will be return. */ static int enable_events(char *session_name) { int ret = CMD_SUCCESS, command_ret = CMD_SUCCESS; int error_holder = CMD_SUCCESS, warn = 0, error = 0, success = 1; char *event_name, *channel_name = NULL; struct lttng_event ev; struct lttng_domain dom; int exclusion_count = 0; char **exclusion_list = NULL; memset(&ev, 0, sizeof(ev)); memset(&dom, 0, sizeof(dom)); if (opt_kernel) { if (opt_filter) { ERR("Filter not implement for kernel tracing yet"); ret = CMD_ERROR; goto error; } if (opt_loglevel) { WARN("Kernel loglevels are not supported."); } } /* Create lttng domain */ if (opt_kernel) { dom.type = LTTNG_DOMAIN_KERNEL; dom.buf_type = LTTNG_BUFFER_GLOBAL; } else if (opt_userspace) { dom.type = LTTNG_DOMAIN_UST; /* Default. */ dom.buf_type = LTTNG_BUFFER_PER_UID; } else if (opt_jul) { dom.type = LTTNG_DOMAIN_JUL; /* Default. */ dom.buf_type = LTTNG_BUFFER_PER_UID; } else if (opt_log4j) { dom.type = LTTNG_DOMAIN_LOG4J; /* Default. */ dom.buf_type = LTTNG_BUFFER_PER_UID; } else if (opt_python) { dom.type = LTTNG_DOMAIN_PYTHON; /* Default. */ dom.buf_type = LTTNG_BUFFER_PER_UID; } else { print_missing_domain(); ret = CMD_ERROR; goto error; } if (opt_kernel && opt_exclude) { ERR("Event name exclusions are not yet implemented for kernel events"); ret = CMD_ERROR; goto error; } channel_name = opt_channel_name; handle = lttng_create_handle(session_name, &dom); if (handle == NULL) { ret = -1; goto error; } /* Prepare Mi */ if (lttng_opt_mi) { /* Open a events element */ ret = mi_lttng_writer_open_element(writer, config_element_events); if (ret) { ret = CMD_ERROR; goto error; } } if (opt_enable_all) { /* Default setup for enable all */ if (opt_kernel) { ev.type = opt_event_type; ev.name[0] = '\0'; /* kernel loglevels not implemented */ ev.loglevel_type = LTTNG_EVENT_LOGLEVEL_ALL; } else { ev.type = LTTNG_EVENT_TRACEPOINT; strcpy(ev.name, "*"); ev.loglevel_type = opt_loglevel_type; if (opt_loglevel) { assert(opt_userspace || opt_jul || opt_log4j || opt_python); if (opt_userspace) { ev.loglevel = loglevel_str_to_value(opt_loglevel); } else if (opt_jul) { ev.loglevel = loglevel_jul_str_to_value(opt_loglevel); } else if (opt_log4j) { ev.loglevel = loglevel_log4j_str_to_value(opt_loglevel); } else if (opt_python) { ev.loglevel = loglevel_python_str_to_value(opt_loglevel); } if (ev.loglevel == -1) { ERR("Unknown loglevel %s", opt_loglevel); ret = -LTTNG_ERR_INVALID; goto error; } } else { assert(opt_userspace || opt_jul || opt_log4j || opt_python); if (opt_userspace) { ev.loglevel = -1; } else if (opt_jul || opt_log4j) { ev.loglevel = LTTNG_LOGLEVEL_JUL_ALL; } else if (opt_python) { ev.loglevel = LTTNG_LOGLEVEL_PYTHON_DEBUG; } } } if (opt_exclude) { ret = check_exclusion_subsets("*", opt_exclude, &exclusion_count, &exclusion_list); if (ret == CMD_ERROR) { goto error; } ev.exclusion = 1; } if (!opt_filter) { ret = lttng_enable_event_with_exclusions(handle, &ev, channel_name, NULL, exclusion_count, exclusion_list); if (ret < 0) { switch (-ret) { case LTTNG_ERR_KERN_EVENT_EXIST: WARN("Kernel events already enabled (channel %s, session %s)", print_channel_name(channel_name), session_name); warn = 1; break; case LTTNG_ERR_TRACE_ALREADY_STARTED: { const char *msg = "The command tried to enable an event in a new domain for a session that has already been started once."; ERR("Events: %s (channel %s, session %s)", msg, print_channel_name(channel_name), session_name); error = 1; break; } default: ERR("Events: %s (channel %s, session %s)", lttng_strerror(ret), ret == -LTTNG_ERR_NEED_CHANNEL_NAME ? print_raw_channel_name(channel_name) : print_channel_name(channel_name), session_name); error = 1; break; } goto end; } switch (opt_event_type) { case LTTNG_EVENT_TRACEPOINT: if (opt_loglevel && dom.type != LTTNG_DOMAIN_KERNEL) { char *exclusion_string = print_exclusions(exclusion_count, exclusion_list); if (!exclusion_string) { PERROR("Cannot allocate exclusion_string"); error = 1; goto end; } MSG("All %s tracepoints%s are enabled in channel %s for loglevel %s", get_domain_str(dom.type), exclusion_string, print_channel_name(channel_name), opt_loglevel); free(exclusion_string); } else { char *exclusion_string = print_exclusions(exclusion_count, exclusion_list); if (!exclusion_string) { PERROR("Cannot allocate exclusion_string"); error = 1; goto end; } MSG("All %s tracepoints%s are enabled in channel %s", get_domain_str(dom.type), exclusion_string, print_channel_name(channel_name)); free(exclusion_string); } break; case LTTNG_EVENT_SYSCALL: if (opt_kernel) { MSG("All %s system calls are enabled in channel %s", get_domain_str(dom.type), print_channel_name(channel_name)); } break; case LTTNG_EVENT_ALL: if (opt_loglevel && dom.type != LTTNG_DOMAIN_KERNEL) { char *exclusion_string = print_exclusions(exclusion_count, exclusion_list); if (!exclusion_string) { PERROR("Cannot allocate exclusion_string"); error = 1; goto end; } MSG("All %s events%s are enabled in channel %s for loglevel %s", get_domain_str(dom.type), exclusion_string, print_channel_name(channel_name), opt_loglevel); free(exclusion_string); } else { char *exclusion_string = print_exclusions(exclusion_count, exclusion_list); if (!exclusion_string) { PERROR("Cannot allocate exclusion_string"); error = 1; goto end; } MSG("All %s events%s are enabled in channel %s", get_domain_str(dom.type), exclusion_string, print_channel_name(channel_name)); free(exclusion_string); } break; default: /* * We should not be here since lttng_enable_event should have * failed on the event type. */ goto error; } } if (opt_filter) { command_ret = lttng_enable_event_with_exclusions(handle, &ev, channel_name, opt_filter, exclusion_count, exclusion_list); if (command_ret < 0) { switch (-command_ret) { case LTTNG_ERR_FILTER_EXIST: WARN("Filter on all events is already enabled" " (channel %s, session %s)", print_channel_name(channel_name), session_name); warn = 1; break; case LTTNG_ERR_TRACE_ALREADY_STARTED: { const char *msg = "The command tried to enable an event in a new domain for a session that has already been started once."; ERR("All events: %s (channel %s, session %s, filter \'%s\')", msg, print_channel_name(channel_name), session_name, opt_filter); error = 1; break; } default: ERR("All events: %s (channel %s, session %s, filter \'%s\')", lttng_strerror(command_ret), command_ret == -LTTNG_ERR_NEED_CHANNEL_NAME ? print_raw_channel_name(channel_name) : print_channel_name(channel_name), session_name, opt_filter); error = 1; break; } error_holder = command_ret; } else { ev.filter = 1; MSG("Filter '%s' successfully set", opt_filter); } } if (lttng_opt_mi) { /* The wildcard * is used for kernel and ust domain to * represent ALL. We copy * in event name to force the wildcard use * for kernel domain * * Note: this is strictly for semantic and printing while in * machine interface mode. */ strcpy(ev.name, "*"); /* If we reach here the events are enabled */ if (!error && !warn) { ev.enabled = 1; } else { ev.enabled = 0; success = 0; } ret = mi_lttng_event(writer, &ev, 1, handle->domain.type); if (ret) { ret = CMD_ERROR; goto error; } /* print exclusion */ ret = mi_print_exclusion(exclusion_count, exclusion_list); if (ret) { ret = CMD_ERROR; goto error; } /* Success ? */ ret = mi_lttng_writer_write_element_bool(writer, mi_lttng_element_command_success, success); if (ret) { ret = CMD_ERROR; goto error; } /* Close event element */ ret = mi_lttng_writer_close_element(writer); if (ret) { ret = CMD_ERROR; goto error; } } goto end; } /* Strip event list */ event_name = strtok(opt_event_list, ","); while (event_name != NULL) { /* Copy name and type of the event */ strncpy(ev.name, event_name, LTTNG_SYMBOL_NAME_LEN); ev.name[LTTNG_SYMBOL_NAME_LEN - 1] = '\0'; ev.type = opt_event_type; /* Kernel tracer action */ if (opt_kernel) { DBG("Enabling kernel event %s for channel %s", event_name, print_channel_name(channel_name)); switch (opt_event_type) { case LTTNG_EVENT_ALL: /* Default behavior is tracepoint */ ev.type = LTTNG_EVENT_TRACEPOINT; /* Fall-through */ case LTTNG_EVENT_TRACEPOINT: break; case LTTNG_EVENT_PROBE: ret = parse_probe_opts(&ev, opt_probe); if (ret) { ERR("Unable to parse probe options"); ret = 0; goto error; } break; case LTTNG_EVENT_FUNCTION: ret = parse_probe_opts(&ev, opt_function); if (ret) { ERR("Unable to parse function probe options"); ret = 0; goto error; } break; case LTTNG_EVENT_FUNCTION_ENTRY: strncpy(ev.attr.ftrace.symbol_name, opt_function_entry_symbol, LTTNG_SYMBOL_NAME_LEN); ev.attr.ftrace.symbol_name[LTTNG_SYMBOL_NAME_LEN - 1] = '\0'; break; case LTTNG_EVENT_SYSCALL: ev.type = LTTNG_EVENT_SYSCALL; break; default: ret = CMD_UNDEFINED; goto error; } /* kernel loglevels not implemented */ ev.loglevel_type = LTTNG_EVENT_LOGLEVEL_ALL; } else if (opt_userspace) { /* User-space tracer action */ #if 0 if (opt_cmd_name != NULL || opt_pid) { MSG("Only supporting tracing all UST processes (-u) for now."); ret = CMD_UNDEFINED; goto error; } #endif DBG("Enabling UST event %s for channel %s, loglevel %s", event_name, print_channel_name(channel_name), opt_loglevel ? : "<all>"); switch (opt_event_type) { case LTTNG_EVENT_ALL: /* Default behavior is tracepoint */ /* Fall-through */ case LTTNG_EVENT_TRACEPOINT: /* Copy name and type of the event */ ev.type = LTTNG_EVENT_TRACEPOINT; strncpy(ev.name, event_name, LTTNG_SYMBOL_NAME_LEN); ev.name[LTTNG_SYMBOL_NAME_LEN - 1] = '\0'; break; case LTTNG_EVENT_PROBE: case LTTNG_EVENT_FUNCTION: case LTTNG_EVENT_FUNCTION_ENTRY: case LTTNG_EVENT_SYSCALL: default: ERR("Event type not available for user-space tracing"); ret = CMD_UNSUPPORTED; goto error; } if (opt_exclude) { ev.exclusion = 1; if (opt_event_type != LTTNG_EVENT_ALL && opt_event_type != LTTNG_EVENT_TRACEPOINT) { ERR("Exclusion option can only be used with tracepoint events"); ret = CMD_ERROR; goto error; } /* Free previously allocated items */ if (exclusion_list != NULL) { while (exclusion_count--) { free(exclusion_list[exclusion_count]); } free(exclusion_list); exclusion_list = NULL; } /* Check for proper subsets */ ret = check_exclusion_subsets(event_name, opt_exclude, &exclusion_count, &exclusion_list); if (ret == CMD_ERROR) { goto error; } } ev.loglevel_type = opt_loglevel_type; if (opt_loglevel) { ev.loglevel = loglevel_str_to_value(opt_loglevel); if (ev.loglevel == -1) { ERR("Unknown loglevel %s", opt_loglevel); ret = -LTTNG_ERR_INVALID; goto error; } } else { ev.loglevel = -1; } } else if (opt_jul || opt_log4j || opt_python) { if (opt_event_type != LTTNG_EVENT_ALL && opt_event_type != LTTNG_EVENT_TRACEPOINT) { ERR("Event type not supported for domain."); ret = CMD_UNSUPPORTED; goto error; } ev.loglevel_type = opt_loglevel_type; if (opt_loglevel) { if (opt_jul) { ev.loglevel = loglevel_jul_str_to_value(opt_loglevel); } else if (opt_log4j) { ev.loglevel = loglevel_log4j_str_to_value(opt_loglevel); } else if (opt_python) { ev.loglevel = loglevel_python_str_to_value(opt_loglevel); } if (ev.loglevel == -1) { ERR("Unknown loglevel %s", opt_loglevel); ret = -LTTNG_ERR_INVALID; goto error; } } else { if (opt_jul) { ev.loglevel = LTTNG_LOGLEVEL_JUL_ALL; } else if (opt_log4j) { ev.loglevel = LTTNG_LOGLEVEL_LOG4J_ALL; } else if (opt_python) { ev.loglevel = LTTNG_LOGLEVEL_PYTHON_DEBUG; } } ev.type = LTTNG_EVENT_TRACEPOINT; strncpy(ev.name, event_name, LTTNG_SYMBOL_NAME_LEN); ev.name[LTTNG_SYMBOL_NAME_LEN - 1] = '\0'; } else { print_missing_domain(); ret = CMD_ERROR; goto error; } if (!opt_filter) { char *exclusion_string; command_ret = lttng_enable_event_with_exclusions(handle, &ev, channel_name, NULL, exclusion_count, exclusion_list); exclusion_string = print_exclusions(exclusion_count, exclusion_list); if (!exclusion_string) { PERROR("Cannot allocate exclusion_string"); error = 1; goto end; } if (command_ret < 0) { /* Turn ret to positive value to handle the positive error code */ switch (-command_ret) { case LTTNG_ERR_KERN_EVENT_EXIST: WARN("Kernel event %s%s already enabled (channel %s, session %s)", event_name, exclusion_string, print_channel_name(channel_name), session_name); warn = 1; break; case LTTNG_ERR_TRACE_ALREADY_STARTED: { const char *msg = "The command tried to enable an event in a new domain for a session that has already been started once."; ERR("Event %s%s: %s (channel %s, session %s)", event_name, exclusion_string, msg, print_channel_name(channel_name), session_name); error = 1; break; } default: ERR("Event %s%s: %s (channel %s, session %s)", event_name, exclusion_string, lttng_strerror(command_ret), command_ret == -LTTNG_ERR_NEED_CHANNEL_NAME ? print_raw_channel_name(channel_name) : print_channel_name(channel_name), session_name); error = 1; break; } error_holder = command_ret; } else { /* So we don't print the default channel name for agent domain. */ if (dom.type == LTTNG_DOMAIN_JUL || dom.type == LTTNG_DOMAIN_LOG4J) { MSG("%s event %s%s enabled.", get_domain_str(dom.type), event_name, exclusion_string); } else { MSG("%s event %s%s created in channel %s", get_domain_str(dom.type), event_name, exclusion_string, print_channel_name(channel_name)); } } free(exclusion_string); } if (opt_filter) { char *exclusion_string; /* Filter present */ ev.filter = 1; command_ret = lttng_enable_event_with_exclusions(handle, &ev, channel_name, opt_filter, exclusion_count, exclusion_list); exclusion_string = print_exclusions(exclusion_count, exclusion_list); if (!exclusion_string) { PERROR("Cannot allocate exclusion_string"); error = 1; goto end; } if (command_ret < 0) { switch (-command_ret) { case LTTNG_ERR_FILTER_EXIST: WARN("Filter on event %s%s is already enabled" " (channel %s, session %s)", event_name, exclusion_string, print_channel_name(channel_name), session_name); warn = 1; break; case LTTNG_ERR_TRACE_ALREADY_STARTED: { const char *msg = "The command tried to enable an event in a new domain for a session that has already been started once."; ERR("Event %s%s: %s (channel %s, session %s, filter \'%s\')", ev.name, exclusion_string, msg, print_channel_name(channel_name), session_name, opt_filter); error = 1; break; } default: ERR("Event %s%s: %s (channel %s, session %s, filter \'%s\')", ev.name, exclusion_string, lttng_strerror(command_ret), command_ret == -LTTNG_ERR_NEED_CHANNEL_NAME ? print_raw_channel_name(channel_name) : print_channel_name(channel_name), session_name, opt_filter); error = 1; break; } error_holder = command_ret; } else { MSG("Event %s%s: Filter '%s' successfully set", event_name, exclusion_string, opt_filter); } free(exclusion_string); } if (lttng_opt_mi) { if (command_ret) { success = 0; ev.enabled = 0; } else { ev.enabled = 1; } ret = mi_lttng_event(writer, &ev, 1, handle->domain.type); if (ret) { ret = CMD_ERROR; goto error; } /* print exclusion */ ret = mi_print_exclusion(exclusion_count, exclusion_list); if (ret) { ret = CMD_ERROR; goto error; } /* Success ? */ ret = mi_lttng_writer_write_element_bool(writer, mi_lttng_element_command_success, success); if (ret) { ret = CMD_ERROR; goto end; } /* Close event element */ ret = mi_lttng_writer_close_element(writer); if (ret) { ret = CMD_ERROR; goto end; } } /* Next event */ event_name = strtok(NULL, ","); /* Reset warn, error and success */ success = 1; }
/* * Ask session daemon for all user space tracepoint fields available. */ static int list_ust_event_fields(void) { int i, size, ret = CMD_SUCCESS; struct lttng_domain domain; struct lttng_handle *handle; struct lttng_event_field *event_field_list; pid_t cur_pid = 0; char *cmdline = NULL; struct lttng_event cur_event; memset(&domain, 0, sizeof(domain)); memset(&cur_event, 0, sizeof(cur_event)); DBG("Getting UST tracing event fields"); domain.type = LTTNG_DOMAIN_UST; handle = lttng_create_handle(NULL, &domain); if (handle == NULL) { ret = CMD_ERROR; goto end; } size = lttng_list_tracepoint_fields(handle, &event_field_list); if (size < 0) { ERR("Unable to list UST event fields: %s", lttng_strerror(size)); ret = CMD_ERROR; goto end; } if (lttng_opt_mi) { /* Mi print */ ret = mi_list_ust_event_fields(event_field_list, size, &domain); if (ret) { ret = CMD_ERROR; goto error; } } else { /* Pretty print */ MSG("UST events:\n-------------"); if (size == 0) { MSG("None"); } for (i = 0; i < size; i++) { if (cur_pid != event_field_list[i].event.pid) { cur_pid = event_field_list[i].event.pid; cmdline = get_cmdline_by_pid(cur_pid); if (cmdline == NULL) { ret = CMD_ERROR; goto error; } MSG("\nPID: %d - Name: %s", cur_pid, cmdline); free(cmdline); /* Wipe current event since we are about to print a new PID. */ memset(&cur_event, 0, sizeof(cur_event)); } if (strcmp(cur_event.name, event_field_list[i].event.name) != 0) { print_events(&event_field_list[i].event); memcpy(&cur_event, &event_field_list[i].event, sizeof(cur_event)); } print_event_field(&event_field_list[i]); } MSG(""); } error: free(event_field_list); end: lttng_destroy_handle(handle); return ret; }
/* * disable_events * * Disabling event using the lttng API. */ static int disable_events(char *session_name) { int err, ret = CMD_SUCCESS, warn = 0; char *event_name, *channel_name = NULL; struct lttng_domain dom; memset(&dom, 0, sizeof(dom)); /* Create lttng domain */ if (opt_kernel) { dom.type = LTTNG_DOMAIN_KERNEL; } else if (opt_userspace) { dom.type = LTTNG_DOMAIN_UST; } else { ERR("Please specify a tracer (-k/--kernel or -u/--userspace)"); ret = CMD_ERROR; goto error; } /* Get channel name */ if (opt_channel_name == NULL) { err = asprintf(&channel_name, DEFAULT_CHANNEL_NAME); if (err < 0) { ret = CMD_FATAL; goto error; } } else { channel_name = opt_channel_name; } handle = lttng_create_handle(session_name, &dom); if (handle == NULL) { ret = -1; goto error; } if (opt_disable_all) { ret = lttng_disable_event(handle, NULL, channel_name); if (ret < 0) { /* Don't set ret so lttng can interpret the sessiond error. */ goto error; } MSG("All %s events are disabled in channel %s", opt_kernel ? "kernel" : "UST", channel_name); goto end; } /* Strip event list */ event_name = strtok(opt_event_list, ","); while (event_name != NULL) { DBG("Disabling event %s", event_name); ret = lttng_disable_event(handle, event_name, channel_name); if (ret < 0) { ERR("Event %s: %s (channel %s, session %s)", event_name, lttng_strerror(ret), channel_name, session_name); warn = 1; } else { MSG("%s event %s disabled in channel %s for session %s", opt_kernel ? "kernel" : "UST", event_name, channel_name, session_name); } /* Next event */ event_name = strtok(NULL, ","); } ret = CMD_SUCCESS; end: error: if (warn) { ret = CMD_WARNING; } if (opt_channel_name == NULL) { free(channel_name); } lttng_destroy_handle(handle); return ret; }
/* * The 'list <options>' first level command */ int cmd_list(int argc, const char **argv) { int opt, ret = CMD_SUCCESS; const char *session_name; static poptContext pc; struct lttng_domain domain; struct lttng_domain *domains = NULL; memset(&domain, 0, sizeof(domain)); if (argc < 1) { usage(stderr); ret = CMD_ERROR; goto end; } pc = poptGetContext(NULL, argc, argv, long_options, 0); poptReadDefaultConfig(pc, 0); while ((opt = poptGetNextOpt(pc)) != -1) { switch (opt) { case OPT_HELP: usage(stdout); goto end; case OPT_USERSPACE: opt_userspace = 1; break; case OPT_LIST_OPTIONS: list_cmd_options(stdout, long_options); goto end; default: usage(stderr); ret = CMD_UNDEFINED; goto end; } } /* Mi check */ if (lttng_opt_mi) { writer = mi_lttng_writer_create(fileno(stdout), lttng_opt_mi); if (!writer) { ret = CMD_ERROR; goto end; } /* Open command element */ ret = mi_lttng_writer_command_open(writer, mi_lttng_element_command_list); if (ret) { ret = CMD_ERROR; goto end; } /* Open output element */ ret = mi_lttng_writer_open_element(writer, mi_lttng_element_command_output); if (ret) { ret = CMD_ERROR; goto end; } } /* Get session name (trailing argument) */ session_name = poptGetArg(pc); DBG2("Session name: %s", session_name); if (opt_kernel) { domain.type = LTTNG_DOMAIN_KERNEL; } else if (opt_userspace) { DBG2("Listing userspace global domain"); domain.type = LTTNG_DOMAIN_UST; } else if (opt_jul) { DBG2("Listing JUL domain"); domain.type = LTTNG_DOMAIN_JUL; } else if (opt_log4j) { domain.type = LTTNG_DOMAIN_LOG4J; } else if (opt_python) { domain.type = LTTNG_DOMAIN_PYTHON; } if (!opt_kernel && opt_syscall) { WARN("--syscall will only work with the Kernel domain (-k)"); ret = CMD_ERROR; goto end; } if (opt_kernel || opt_userspace || opt_jul || opt_log4j || opt_python) { handle = lttng_create_handle(session_name, &domain); if (handle == NULL) { ret = CMD_FATAL; goto end; } } if (session_name == NULL) { if (!opt_kernel && !opt_userspace && !opt_jul && !opt_log4j && !opt_python) { ret = list_sessions(NULL); if (ret) { goto end; } } if (opt_kernel) { if (opt_syscall) { ret = list_syscalls(); if (ret) { goto end; } } else { ret = list_kernel_events(); if (ret) { goto end; } } } if (opt_userspace) { if (opt_fields) { ret = list_ust_event_fields(); } else { ret = list_ust_events(); } if (ret) { goto end; } } if (opt_jul || opt_log4j || opt_python) { ret = list_agent_events(); if (ret) { goto end; } } } else { /* List session attributes */ if (lttng_opt_mi) { /* Open element sessions * Present for xml consistency */ ret = mi_lttng_sessions_open(writer); if (ret) { goto end; } } /* MI: the ouptut of list_sessions is an unclosed session element */ ret = list_sessions(session_name); if (ret) { goto end; } /* Domain listing */ if (opt_domain) { ret = list_domains(session_name); goto end; } /* Channel listing */ if (opt_kernel || opt_userspace) { if (lttng_opt_mi) { /* Add of domains and domain element for xml * consistency and validation */ ret = mi_lttng_domains_open(writer); if (ret) { goto end; } /* Open domain and leave it open for * nested channels printing */ ret = mi_lttng_domain(writer, &domain, 1); if (ret) { goto end; } } ret = list_tracker_pids(); if (ret) { goto end; } ret = list_channels(opt_channel); if (ret) { goto end; } if (lttng_opt_mi) { /* Close domain and domain element */ ret = mi_lttng_close_multi_element(writer, 2); } if (ret) { goto end; } } else { int i, nb_domain; /* We want all domain(s) */ nb_domain = lttng_list_domains(session_name, &domains); if (nb_domain < 0) { ret = CMD_ERROR; ERR("%s", lttng_strerror(nb_domain)); goto end; } if (lttng_opt_mi) { ret = mi_lttng_domains_open(writer); if (ret) { ret = CMD_ERROR; goto end; } } for (i = 0; i < nb_domain; i++) { switch (domains[i].type) { case LTTNG_DOMAIN_KERNEL: MSG("=== Domain: Kernel ===\n"); break; case LTTNG_DOMAIN_UST: MSG("=== Domain: UST global ===\n"); MSG("Buffer type: %s\n", domains[i].buf_type == LTTNG_BUFFER_PER_PID ? "per PID" : "per UID"); break; case LTTNG_DOMAIN_JUL: MSG("=== Domain: JUL (Java Util Logging) ===\n"); break; case LTTNG_DOMAIN_LOG4J: MSG("=== Domain: LOG4j (Logging for Java) ===\n"); break; case LTTNG_DOMAIN_PYTHON: MSG("=== Domain: Python (logging) ===\n"); break; default: MSG("=== Domain: Unimplemented ===\n"); break; } if (lttng_opt_mi) { ret = mi_lttng_domain(writer, &domains[i], 1); if (ret) { ret = CMD_ERROR; goto end; } } /* Clean handle before creating a new one */ if (handle) { lttng_destroy_handle(handle); } handle = lttng_create_handle(session_name, &domains[i]); if (handle == NULL) { ret = CMD_FATAL; goto end; } if (domains[i].type == LTTNG_DOMAIN_JUL || domains[i].type == LTTNG_DOMAIN_LOG4J || domains[i].type == LTTNG_DOMAIN_PYTHON) { ret = list_session_agent_events(); if (ret) { goto end; } continue; } switch (domains[i].type) { case LTTNG_DOMAIN_KERNEL: case LTTNG_DOMAIN_UST: ret = list_tracker_pids(); if (ret) { goto end; } break; default: break; } ret = list_channels(opt_channel); if (ret) { goto end; } if (lttng_opt_mi) { /* Close domain element */ ret = mi_lttng_writer_close_element(writer); if (ret) { ret = CMD_ERROR; goto end; } } } if (lttng_opt_mi) { /* Close the domains, session and sessions element */ ret = mi_lttng_close_multi_element(writer, 3); if (ret) { ret = CMD_ERROR; goto end; } } } } /* Mi closing */ if (lttng_opt_mi) { /* Close output element */ ret = mi_lttng_writer_close_element(writer); if (ret) { ret = CMD_ERROR; goto end; } /* Command element close */ ret = mi_lttng_writer_command_close(writer); if (ret) { ret = CMD_ERROR; goto end; } } end: /* Mi clean-up */ if (writer && mi_lttng_writer_destroy(writer)) { /* Preserve original error code */ ret = ret ? ret : -LTTNG_ERR_MI_IO_FAIL; } free(domains); if (handle) { lttng_destroy_handle(handle); } poptFreeContext(pc); return ret; }
static enum cmd_error_code track_untrack_pid(enum cmd_type cmd_type, const char *cmd_str, const char *session_name, const char *pid_string, int all, struct mi_writer *writer) { int ret, success = 1 , i; enum cmd_error_code retval = CMD_SUCCESS; int *pid_list = NULL; int nr_pids; struct lttng_domain dom; struct lttng_handle *handle = NULL; int (*cmd_func)(struct lttng_handle *handle, int pid); switch (cmd_type) { case CMD_TRACK: cmd_func = lttng_track_pid; break; case CMD_UNTRACK: cmd_func = lttng_untrack_pid; break; default: ERR("Unknown command"); retval = CMD_ERROR; goto end; } memset(&dom, 0, sizeof(dom)); if (opt_kernel) { dom.type = LTTNG_DOMAIN_KERNEL; } else if (opt_userspace) { dom.type = LTTNG_DOMAIN_UST; } else { /* Checked by the caller. */ assert(0); } ret = parse_pid_string(pid_string, all, &pid_list, &nr_pids); if (ret != CMD_SUCCESS) { ERR("Error parsing PID string"); retval = CMD_ERROR; goto end; } handle = lttng_create_handle(session_name, &dom); if (handle == NULL) { retval = CMD_ERROR; goto end; } if (writer) { /* Open process element */ ret = mi_lttng_targets_open(writer); if (ret) { retval = CMD_ERROR; goto end; } } for (i = 0; i < nr_pids; i++) { DBG("%s PID %d", cmd_str, pid_list[i]); ret = cmd_func(handle, pid_list[i]); if (ret) { switch (-ret) { case LTTNG_ERR_PID_TRACKED: WARN("PID %i already tracked in session %s", pid_list[i], session_name); success = 1; retval = CMD_SUCCESS; break; case LTTNG_ERR_PID_NOT_TRACKED: WARN("PID %i not tracked in session %s", pid_list[i], session_name); success = 1; retval = CMD_SUCCESS; break; default: ERR("%s", lttng_strerror(ret)); success = 0; retval = CMD_ERROR; break; } } else { MSG("PID %i %sed in session %s", pid_list[i], cmd_str, session_name); success = 1; } /* Mi */ if (writer) { ret = mi_lttng_pid_target(writer, pid_list[i], 1); if (ret) { retval = CMD_ERROR; goto end; } ret = mi_lttng_writer_write_element_bool(writer, mi_lttng_element_success, success); if (ret) { retval = CMD_ERROR; goto end; } ret = mi_lttng_writer_close_element(writer); if (ret) { retval = CMD_ERROR; goto end; } } } if (writer) { /* Close targets element */ ret = mi_lttng_writer_close_element(writer); if (ret) { retval = CMD_ERROR; goto end; } } end: if (handle) { lttng_destroy_handle(handle); } free(pid_list); return retval; }
/* * The 'list <options>' first level command */ int cmd_list(int argc, const char **argv) { int opt, ret = CMD_SUCCESS; const char *session_name; static poptContext pc; struct lttng_domain domain; struct lttng_domain *domains = NULL; memset(&domain, 0, sizeof(domain)); if (argc < 1) { usage(stderr); ret = CMD_ERROR; goto end; } pc = poptGetContext(NULL, argc, argv, long_options, 0); poptReadDefaultConfig(pc, 0); while ((opt = poptGetNextOpt(pc)) != -1) { switch (opt) { case OPT_HELP: usage(stdout); goto end; case OPT_USERSPACE: opt_userspace = 1; break; case OPT_LIST_OPTIONS: list_cmd_options(stdout, long_options); goto end; default: usage(stderr); ret = CMD_UNDEFINED; goto end; } } /* Get session name (trailing argument) */ session_name = poptGetArg(pc); DBG2("Session name: %s", session_name); if (opt_kernel) { domain.type = LTTNG_DOMAIN_KERNEL; } else if (opt_userspace) { DBG2("Listing userspace global domain"); domain.type = LTTNG_DOMAIN_UST; } else if (opt_jul) { DBG2("Listing JUL domain"); domain.type = LTTNG_DOMAIN_JUL; } if (opt_kernel || opt_userspace || opt_jul) { handle = lttng_create_handle(session_name, &domain); if (handle == NULL) { ret = CMD_FATAL; goto end; } } if (session_name == NULL) { if (!opt_kernel && !opt_userspace && !opt_jul) { ret = list_sessions(NULL); if (ret != 0) { goto end; } } if (opt_kernel) { ret = list_kernel_events(); if (ret < 0) { ret = CMD_ERROR; goto end; } } if (opt_userspace) { if (opt_fields) { ret = list_ust_event_fields(); } else { ret = list_ust_events(); } if (ret < 0) { ret = CMD_ERROR; goto end; } } if (opt_jul) { ret = list_jul_events(); if (ret < 0) { ret = CMD_ERROR; goto end; } } } else { /* List session attributes */ ret = list_sessions(session_name); if (ret != 0) { goto end; } /* Domain listing */ if (opt_domain) { ret = list_domains(session_name); goto end; } if (opt_kernel || opt_userspace) { /* Channel listing */ ret = list_channels(opt_channel); if (ret < 0) { goto end; } } else { int i, nb_domain; /* We want all domain(s) */ nb_domain = lttng_list_domains(session_name, &domains); if (nb_domain < 0) { ret = nb_domain; ERR("%s", lttng_strerror(ret)); goto end; } for (i = 0; i < nb_domain; i++) { switch (domains[i].type) { case LTTNG_DOMAIN_KERNEL: MSG("=== Domain: Kernel ===\n"); break; case LTTNG_DOMAIN_UST: MSG("=== Domain: UST global ===\n"); MSG("Buffer type: %s\n", domains[i].buf_type == LTTNG_BUFFER_PER_PID ? "per PID" : "per UID"); break; case LTTNG_DOMAIN_JUL: MSG("=== Domain: JUL (Java Util Logging) ===\n"); break; default: MSG("=== Domain: Unimplemented ===\n"); break; } /* Clean handle before creating a new one */ if (handle) { lttng_destroy_handle(handle); } handle = lttng_create_handle(session_name, &domains[i]); if (handle == NULL) { ret = CMD_FATAL; goto end; } if (domains[i].type == LTTNG_DOMAIN_JUL) { ret = list_session_jul_events(); if (ret < 0) { goto end; } continue; } ret = list_channels(opt_channel); if (ret < 0) { goto end; } } } } end: free(domains); if (handle) { lttng_destroy_handle(handle); } poptFreeContext(pc); return ret; }
/* * Adding channel using the lttng API. */ static int enable_channel(char *session_name) { struct lttng_channel *channel = NULL; int ret = CMD_SUCCESS, warn = 0, error = 0, success = 0; char *channel_name; struct lttng_domain dom; memset(&dom, 0, sizeof(dom)); /* Validate options. */ if (opt_kernel) { if (opt_blocking_timeout.set) { ERR("Retry timeout option not supported for kernel domain (-k)"); ret = CMD_ERROR; goto error; } } /* Create lttng domain */ if (opt_kernel) { dom.type = LTTNG_DOMAIN_KERNEL; dom.buf_type = LTTNG_BUFFER_GLOBAL; if (opt_buffer_uid || opt_buffer_pid) { ERR("Buffer type not supported for domain -k"); ret = CMD_ERROR; goto error; } } else if (opt_userspace) { dom.type = LTTNG_DOMAIN_UST; if (opt_buffer_pid) { dom.buf_type = LTTNG_BUFFER_PER_PID; } else { if (opt_buffer_global) { ERR("Buffer type not supported for domain -u"); ret = CMD_ERROR; goto error; } dom.buf_type = LTTNG_BUFFER_PER_UID; } } else { /* Checked by the caller. */ assert(0); } set_default_attr(&dom); if (chan_opts.attr.tracefile_size == 0 && chan_opts.attr.tracefile_count) { ERR("Missing option --tracefile-size. " "A file count without a size won't do anything."); ret = CMD_ERROR; goto error; } if ((chan_opts.attr.tracefile_size > 0) && (chan_opts.attr.tracefile_size < chan_opts.attr.subbuf_size)) { WARN("Tracefile size rounded up from (%" PRIu64 ") to subbuffer size (%" PRIu64 ")", chan_opts.attr.tracefile_size, chan_opts.attr.subbuf_size); chan_opts.attr.tracefile_size = chan_opts.attr.subbuf_size; } /* Setting channel output */ if (opt_output) { if (!strncmp(output_mmap, opt_output, strlen(output_mmap))) { chan_opts.attr.output = LTTNG_EVENT_MMAP; } else if (!strncmp(output_splice, opt_output, strlen(output_splice))) { chan_opts.attr.output = LTTNG_EVENT_SPLICE; } else { ERR("Unknown output type %s. Possible values are: %s, %s\n", opt_output, output_mmap, output_splice); ret = CMD_ERROR; goto error; } } handle = lttng_create_handle(session_name, &dom); if (handle == NULL) { ret = -1; goto error; } /* Mi open channels element */ if (lttng_opt_mi) { assert(writer); ret = mi_lttng_channels_open(writer); if (ret) { ret = CMD_ERROR; goto error; } } /* Strip channel list (format: chan1,chan2,...) */ channel_name = strtok(opt_channels, ","); while (channel_name != NULL) { void *extended_ptr; /* Validate channel name's length */ if (strlen(channel_name) >= sizeof(chan_opts.name)) { ERR("Channel name is too long (max. %zu characters)", sizeof(chan_opts.name) - 1); error = 1; goto skip_enable; } /* * A dynamically-allocated channel is used in order to allow * the configuration of extended attributes (post-2.9). */ channel = lttng_channel_create(&dom); if (!channel) { ERR("Unable to create channel object"); error = 1; goto error; } /* Copy channel name */ strcpy(channel->name, channel_name); channel->enabled = 1; extended_ptr = channel->attr.extended.ptr; memcpy(&channel->attr, &chan_opts.attr, sizeof(chan_opts.attr)); channel->attr.extended.ptr = extended_ptr; if (opt_monitor_timer.set) { ret = lttng_channel_set_monitor_timer_interval(channel, opt_monitor_timer.interval); if (ret) { ERR("Failed to set the channel's monitor timer interval"); error = 1; goto error; } } if (opt_blocking_timeout.set) { ret = lttng_channel_set_blocking_timeout(channel, opt_blocking_timeout.value); if (ret) { ERR("Failed to set the channel's blocking timeout"); error = 1; goto error; } } DBG("Enabling channel %s", channel_name); ret = lttng_enable_channel(handle, channel); if (ret < 0) { success = 0; switch (-ret) { case LTTNG_ERR_KERN_CHAN_EXIST: case LTTNG_ERR_UST_CHAN_EXIST: case LTTNG_ERR_CHAN_EXIST: WARN("Channel %s: %s (session %s)", channel_name, lttng_strerror(ret), session_name); warn = 1; break; case LTTNG_ERR_INVALID_CHANNEL_NAME: ERR("Invalid channel name: \"%s\". " "Channel names may not start with '.', and " "may not contain '/'.", channel_name); error = 1; break; default: ERR("Channel %s: %s (session %s)", channel_name, lttng_strerror(ret), session_name); error = 1; break; } } else { MSG("%s channel %s enabled for session %s", get_domain_str(dom.type), channel_name, session_name); success = 1; } skip_enable: if (lttng_opt_mi) { /* Mi print the channel element and leave it open */ ret = mi_lttng_channel(writer, channel, 1); if (ret) { ret = CMD_ERROR; goto error; } /* Individual Success ? */ ret = mi_lttng_writer_write_element_bool(writer, mi_lttng_element_command_success, success); if (ret) { ret = CMD_ERROR; goto error; } /* Close channel element */ ret = mi_lttng_writer_close_element(writer); if (ret) { ret = CMD_ERROR; goto error; } } /* Next channel */ channel_name = strtok(NULL, ","); lttng_channel_destroy(channel); channel = NULL; } if (lttng_opt_mi) { /* Close channels element */ ret = mi_lttng_writer_close_element(writer); if (ret) { ret = CMD_ERROR; goto error; } } ret = CMD_SUCCESS; error: if (channel) { lttng_channel_destroy(channel); } /* If more important error happen bypass the warning */ if (!ret && warn) { ret = CMD_WARNING; } /* If more important error happen bypass the warning */ if (!ret && error) { ret = CMD_ERROR; } lttng_destroy_handle(handle); return ret; }
/* * Adding channel using the lttng API. */ static int enable_channel(char *session_name) { int ret = CMD_SUCCESS, warn = 0, error = 0, success = 0; char *channel_name; struct lttng_domain dom; memset(&dom, 0, sizeof(dom)); /* Create lttng domain */ if (opt_kernel) { dom.type = LTTNG_DOMAIN_KERNEL; dom.buf_type = LTTNG_BUFFER_GLOBAL; if (opt_buffer_uid || opt_buffer_pid) { ERR("Buffer type not supported for domain -k"); ret = CMD_ERROR; goto error; } } else if (opt_userspace) { dom.type = LTTNG_DOMAIN_UST; if (opt_buffer_pid) { dom.buf_type = LTTNG_BUFFER_PER_PID; } else { if (opt_buffer_global) { ERR("Buffer type not supported for domain -u"); ret = CMD_ERROR; goto error; } dom.buf_type = LTTNG_BUFFER_PER_UID; } } else { print_missing_domain(); ret = CMD_ERROR; goto error; } set_default_attr(&dom); if (chan.attr.tracefile_size == 0 && chan.attr.tracefile_count) { ERR("Missing option --tracefile-size. " "A file count without a size won't do anything."); ret = CMD_ERROR; goto error; } if ((chan.attr.tracefile_size > 0) && (chan.attr.tracefile_size < chan.attr.subbuf_size)) { WARN("Tracefile size rounded up from (%" PRIu64 ") to subbuffer size (%" PRIu64 ")", chan.attr.tracefile_size, chan.attr.subbuf_size); chan.attr.tracefile_size = chan.attr.subbuf_size; } /* Setting channel output */ if (opt_output) { if (!strncmp(output_mmap, opt_output, strlen(output_mmap))) { chan.attr.output = LTTNG_EVENT_MMAP; } else if (!strncmp(output_splice, opt_output, strlen(output_splice))) { chan.attr.output = LTTNG_EVENT_SPLICE; } else { ERR("Unknown output type %s. Possible values are: %s, %s\n", opt_output, output_mmap, output_splice); usage(stderr); ret = CMD_ERROR; goto error; } } handle = lttng_create_handle(session_name, &dom); if (handle == NULL) { ret = -1; goto error; } /* Mi open channels element */ if (lttng_opt_mi) { assert(writer); ret = mi_lttng_channels_open(writer); if (ret) { ret = CMD_ERROR; goto error; } } /* Strip channel list (format: chan1,chan2,...) */ channel_name = strtok(opt_channels, ","); while (channel_name != NULL) { /* Copy channel name and normalize it */ strncpy(chan.name, channel_name, NAME_MAX); chan.name[NAME_MAX - 1] = '\0'; DBG("Enabling channel %s", channel_name); ret = lttng_enable_channel(handle, &chan); if (ret < 0) { success = 0; switch (-ret) { case LTTNG_ERR_KERN_CHAN_EXIST: case LTTNG_ERR_UST_CHAN_EXIST: case LTTNG_ERR_CHAN_EXIST: WARN("Channel %s: %s (session %s)", channel_name, lttng_strerror(ret), session_name); warn = 1; break; default: ERR("Channel %s: %s (session %s)", channel_name, lttng_strerror(ret), session_name); error = 1; break; } } else { MSG("%s channel %s enabled for session %s", get_domain_str(dom.type), channel_name, session_name); success = 1; } if (lttng_opt_mi) { /* Mi print the channel element and leave it open */ ret = mi_lttng_channel(writer, &chan, 1); if (ret) { ret = CMD_ERROR; goto error; } /* Individual Success ? */ ret = mi_lttng_writer_write_element_bool(writer, mi_lttng_element_command_success, success); if (ret) { ret = CMD_ERROR; goto error; } /* Close channel element */ ret = mi_lttng_writer_close_element(writer); if (ret) { ret = CMD_ERROR; goto error; } } /* Next channel */ channel_name = strtok(NULL, ","); } if (lttng_opt_mi) { /* Close channels element */ ret = mi_lttng_writer_close_element(writer); if (ret) { ret = CMD_ERROR; goto error; } } ret = CMD_SUCCESS; error: /* If more important error happen bypass the warning */ if (!ret && warn) { ret = CMD_WARNING; } /* If more important error happen bypass the warning */ if (!ret && error) { ret = CMD_ERROR; } lttng_destroy_handle(handle); return ret; }
/* * Adding channel using the lttng API. */ static int enable_channel(char *session_name) { int ret = CMD_SUCCESS, warn = 0; char *channel_name; struct lttng_domain dom; memset(&dom, 0, sizeof(dom)); /* Create lttng domain */ if (opt_kernel) { dom.type = LTTNG_DOMAIN_KERNEL; dom.buf_type = LTTNG_BUFFER_GLOBAL; if (opt_buffer_uid || opt_buffer_pid) { ERR("Buffer type not supported for domain -k"); ret = CMD_ERROR; goto error; } } else if (opt_userspace) { dom.type = LTTNG_DOMAIN_UST; if (opt_buffer_uid) { dom.buf_type = LTTNG_BUFFER_PER_UID; } else { if (opt_buffer_global) { ERR("Buffer type not supported for domain -u"); ret = CMD_ERROR; goto error; } dom.buf_type = LTTNG_BUFFER_PER_PID; } } else { ERR("Please specify a tracer (-k/--kernel or -u/--userspace)"); ret = CMD_ERROR; goto error; } set_default_attr(&dom); if (chan.attr.tracefile_size == 0 && chan.attr.tracefile_count) { ERR("Missing option --tracefile-size. " "A file count without a size won't do anything."); ret = CMD_ERROR; goto error; } if ((chan.attr.tracefile_size > 0) && (chan.attr.tracefile_size < chan.attr.subbuf_size)) { WARN("Tracefile size rounded up from (%" PRIu64 ") to subbuffer size (%" PRIu64 ")", chan.attr.tracefile_size, chan.attr.subbuf_size); chan.attr.tracefile_size = chan.attr.subbuf_size; } /* Setting channel output */ if (opt_output) { if (!strncmp(output_mmap, opt_output, strlen(output_mmap))) { chan.attr.output = LTTNG_EVENT_MMAP; } else if (!strncmp(output_splice, opt_output, strlen(output_splice))) { chan.attr.output = LTTNG_EVENT_SPLICE; } else { ERR("Unknown output type %s. Possible values are: %s, %s\n", opt_output, output_mmap, output_splice); usage(stderr); ret = CMD_ERROR; goto error; } } handle = lttng_create_handle(session_name, &dom); if (handle == NULL) { ret = -1; goto error; } /* Strip channel list (format: chan1,chan2,...) */ channel_name = strtok(opt_channels, ","); while (channel_name != NULL) { /* Copy channel name and normalize it */ strncpy(chan.name, channel_name, NAME_MAX); chan.name[NAME_MAX - 1] = '\0'; DBG("Enabling channel %s", channel_name); ret = lttng_enable_channel(handle, &chan); if (ret < 0) { switch (-ret) { case LTTNG_ERR_KERN_CHAN_EXIST: case LTTNG_ERR_UST_CHAN_EXIST: WARN("Channel %s: %s (session %s)", channel_name, lttng_strerror(ret), session_name); goto error; default: ERR("Channel %s: %s (session %s)", channel_name, lttng_strerror(ret), session_name); break; } warn = 1; } else { MSG("%s channel %s enabled for session %s", opt_kernel ? "Kernel" : "UST", channel_name, session_name); } /* Next event */ channel_name = strtok(NULL, ","); } ret = CMD_SUCCESS; error: if (warn) { ret = CMD_WARNING; } lttng_destroy_handle(handle); return ret; }
static int list_agent_events(void) { int i, size, ret = CMD_SUCCESS; struct lttng_domain domain; struct lttng_handle *handle = NULL; struct lttng_event *event_list = NULL; pid_t cur_pid = 0; char *cmdline = NULL; const char *agent_domain_str; memset(&domain, 0, sizeof(domain)); if (opt_jul) { domain.type = LTTNG_DOMAIN_JUL; } else if (opt_log4j) { domain.type = LTTNG_DOMAIN_LOG4J; } else if (opt_python) { domain.type = LTTNG_DOMAIN_PYTHON; } else { ERR("Invalid agent domain selected."); ret = CMD_ERROR; goto error; } agent_domain_str = get_domain_str(domain.type); DBG("Getting %s tracing events", agent_domain_str); handle = lttng_create_handle(NULL, &domain); if (handle == NULL) { ret = CMD_ERROR; goto end; } size = lttng_list_tracepoints(handle, &event_list); if (size < 0) { ERR("Unable to list %s events: %s", agent_domain_str, lttng_strerror(size)); ret = CMD_ERROR; goto end; } if (lttng_opt_mi) { /* Mi print */ ret = mi_list_agent_ust_events(event_list, size, &domain); if (ret) { ret = CMD_ERROR; goto error; } } else { /* Pretty print */ MSG("%s events (Logger name):\n-------------------------", agent_domain_str); if (size == 0) { MSG("None"); } for (i = 0; i < size; i++) { if (cur_pid != event_list[i].pid) { cur_pid = event_list[i].pid; cmdline = get_cmdline_by_pid(cur_pid); if (cmdline == NULL) { ret = CMD_ERROR; goto error; } MSG("\nPID: %d - Name: %s", cur_pid, cmdline); free(cmdline); } MSG("%s- %s", indent6, event_list[i].name); } MSG(""); } error: free(event_list); end: lttng_destroy_handle(handle); return ret; }