From: "Steven Rostedt (Google)" <rostedt@xxxxxxxxxxx> The trace_stream_init() function creates a temp file to save the headers in, and then reads it for processing the information. It does this by using tmpfile() to create the temp file, grabs the file descriptor from it, and then calls tracecmd_output_create_fd() on it to save the information. After it is written, the trace_output is no longer needed, and the tracecmd_input descriptor is created by reading this temporary file. The problem arises with the movement of when the trace option are written. As some options need to happen after recording, they are finished up at the end where tracecmd_output_close() is called. But in this case, we only want the options written, and not to close the descriptor (as it will be used for reading too). Create a new tracecmd_output_flush() that writes out the options and use that. Bugzilla: https://bugzilla.kernel.org/show_bug.cgi?id=216577 Reported-by: Charlemagne Lasse <charlemagnelasse@xxxxxxxxx> Fixes: 5d4d7ec30255 ("trace-cmd: Move reading of trace.dat options to tracecmd_read_headers()") Signed-off-by: Steven Rostedt (Google) <rostedt@xxxxxxxxxxx> --- Sorry, for not submitting this sooner. I thought I had, but apparently, I never committed the patch. .../include/private/trace-cmd-private.h | 1 + lib/trace-cmd/trace-output.c | 10 +++++++- tracecmd/trace-stream.c | 25 ++++++++----------- 3 files changed, 21 insertions(+), 15 deletions(-) diff --git a/lib/trace-cmd/include/private/trace-cmd-private.h b/lib/trace-cmd/include/private/trace-cmd-private.h index 271f0541982f..2cb6f35cb250 100644 --- a/lib/trace-cmd/include/private/trace-cmd-private.h +++ b/lib/trace-cmd/include/private/trace-cmd-private.h @@ -341,6 +341,7 @@ int tracecmd_write_options(struct tracecmd_output *handle); int tracecmd_write_meta_strings(struct tracecmd_output *handle); int tracecmd_append_options(struct tracecmd_output *handle); void tracecmd_output_close(struct tracecmd_output *handle); +void tracecmd_output_flush(struct tracecmd_output *handle); void tracecmd_output_free(struct tracecmd_output *handle); struct tracecmd_output *tracecmd_copy(struct tracecmd_input *ihandle, const char *file, enum tracecmd_file_states state, int file_version, diff --git a/lib/trace-cmd/trace-output.c b/lib/trace-cmd/trace-output.c index bdec75d67bc4..1c0f7b773ddc 100644 --- a/lib/trace-cmd/trace-output.c +++ b/lib/trace-cmd/trace-output.c @@ -281,7 +281,7 @@ void tracecmd_output_free(struct tracecmd_output *handle) free(handle); } -void tracecmd_output_close(struct tracecmd_output *handle) +void tracecmd_output_flush(struct tracecmd_output *handle) { if (!handle) return; @@ -293,6 +293,14 @@ void tracecmd_output_close(struct tracecmd_output *handle) /* write strings section */ save_string_section(handle, true); } +} + +void tracecmd_output_close(struct tracecmd_output *handle) +{ + if (!handle) + return; + + tracecmd_output_flush(handle); if (handle->fd >= 0) { close(handle->fd); diff --git a/tracecmd/trace-stream.c b/tracecmd/trace-stream.c index ee310f3d0507..d83513f8aa89 100644 --- a/tracecmd/trace-stream.c +++ b/tracecmd/trace-stream.c @@ -24,11 +24,10 @@ trace_stream_init(struct buffer_instance *instance, int cpu, int fd, int cpus, struct hook_list *hooks, tracecmd_handle_init_func handle_init, int global) { - struct tracecmd_input *trace_input; struct tracecmd_output *trace_output; + struct tracecmd_input *trace_input; static FILE *fp = NULL; static int tfd; - static int ofd; long flags; if (instance->handle) { @@ -42,23 +41,21 @@ trace_stream_init(struct buffer_instance *instance, int cpu, int fd, int cpus, return NULL; tfd = fileno(fp); - ofd = dup(tfd); - trace_output = tracecmd_output_create_fd(ofd); - if (!trace_output) { - fclose(fp); - return NULL; - } + trace_output = tracecmd_output_create_fd(tfd); + if (!trace_output) + goto fail; + tracecmd_output_write_headers(trace_output, NULL); + tracecmd_output_flush(trace_output); + /* Don't close the descriptor, use it for reading */ tracecmd_output_free(trace_output); } - lseek(ofd, 0, SEEK_SET); + lseek(tfd, 0, SEEK_SET); - trace_input = tracecmd_alloc_fd(ofd, 0); - if (!trace_input) { - close(ofd); + trace_input = tracecmd_alloc_fd(tfd, 0); + if (!trace_input) goto fail; - } if (tracecmd_read_headers(trace_input, TRACECMD_FILE_PRINTK) < 0) goto fail_free_input; @@ -82,7 +79,7 @@ trace_stream_init(struct buffer_instance *instance, int cpu, int fd, int cpus, tracecmd_close(trace_input); fail: fclose(fp); - + fp = NULL; /* Try again later? */ return NULL; } -- 2.35.1