Re: [PATCH v2 51/87] trace-cmd library: Extend the input handler with trace data decompression context

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



On Thu, 29 Jul 2021 08:09:23 +0300
"Tzvetomir Stoyanov (VMware)" <tz.stoyanov@xxxxxxxxx> wrote:

> The CPU tarce data is compressed in chunks, as chunk's size is multiple

typos.

> trace pages. The input handler is extended with the necessary
> structures, to control the data decompression. There are two approaches
> for data decompression, both are supported and can be used in different
> use cases:
>  - in-memory decompression, page by page.
>  - using a temporary file
> 
> Signed-off-by: Tzvetomir Stoyanov (VMware) <tz.stoyanov@xxxxxxxxx>
> ---
>  lib/trace-cmd/trace-input.c | 66 ++++++++++++++++++++++++++++++-------
>  1 file changed, 54 insertions(+), 12 deletions(-)
> 
> diff --git a/lib/trace-cmd/trace-input.c b/lib/trace-cmd/trace-input.c
> index 520d611f..6fb63c0f 100644
> --- a/lib/trace-cmd/trace-input.c
> +++ b/lib/trace-cmd/trace-input.c
> @@ -29,6 +29,9 @@
>  
>  #define COMMIT_MASK ((1 << 27) - 1)
>  
> +/* force uncompressing in memory */
> +#define INMEMORY_DECOMPRESS

I wonder if we should just make this a variable in the handle, and
default it to in memory. Then it will be easy to change it from the
command line if we want to.

-- Steve

> +
>  /* for debugging read instead of mmap */
>  static int force_read = 0;
>  
> @@ -54,6 +57,24 @@ struct page {
>  #endif
>  };
>  
> +struct zchunk_cache {
> +	struct list_head		list;
> +	struct tracecmd_compress_chunk *chunk;
> +	void				*map;
> +	int				ref;
> +};
> +
> +struct cpu_zdata {
> +	/* uncompressed cpu data */
> +	int			fd;
> +	char			file[26]; /*
> strlen(COMPR_TEMP_FILE) */
> +	unsigned int		count;
> +	unsigned int		last_chunk;
> +	struct list_head	cache;
> +	struct tracecmd_compress_chunk	*chunks;
> +};
> +
> +#define COMPR_TEMP_FILE "/tmp/trace_cpu_dataXXXXXX"
>  struct cpu_data {
>  	/* the first two never change */
>  	unsigned long long	file_offset;
> @@ -72,6 +93,7 @@ struct cpu_data {
>  	int			page_cnt;
>  	int			cpu;
>  	int			pipe_fd;
> +	struct cpu_zdata	compress;
>  };
>  
>  struct cpu_file_data {
> @@ -150,6 +172,8 @@ struct tracecmd_input {
>  	bool			use_trace_clock;
>  	bool			read_page;
>  	bool			use_pipe;
> +	bool			read_zpage; /* uncompress pages
> in memory, do not use tmp files */
> +	bool			cpu_compressed;
>  	int			file_version;
>  	struct cpu_data 	*cpu_data;
>  	long long		ts_offset;
> @@ -3284,7 +3308,7 @@ static int read_cpu_data(struct tracecmd_input
> *handle) unsigned long long offset;
>  
>  		handle->cpu_data[cpu].cpu = cpu;
> -
> +		handle->cpu_data[cpu].compress.fd = -1;
>  		handle->cpu_data[cpu].kbuf =
> kbuffer_alloc(long_size, endian); if (!handle->cpu_data[cpu].kbuf)
>  			goto out_free;
> @@ -3701,6 +3725,9 @@ struct tracecmd_input *tracecmd_alloc_fd(int
> fd, int flags) /* By default, use usecs, unless told otherwise */
>  	handle->flags |= TRACECMD_FL_IN_USECS;
>  
> +#ifdef INMEMORY_DECOMPRESS
> +	handle->read_zpage = 1;
> +#endif
>  	if (do_read_check(handle, buf, 3))
>  		goto failed_read;
>  
> @@ -3915,6 +3942,7 @@ void tracecmd_ref(struct tracecmd_input *handle)
>   */
>  void tracecmd_close(struct tracecmd_input *handle)
>  {
> +	struct zchunk_cache *cache;
>  	struct file_section *del_sec;
>  	int i;
>  
> @@ -3933,17 +3961,31 @@ void tracecmd_close(struct tracecmd_input
> *handle) /* The tracecmd_peek_data may have cached a record */
>  		free_next(handle, i);
>  		free_page(handle, i);
> -		if (handle->cpu_data && handle->cpu_data[i].kbuf) {
> -			kbuffer_free(handle->cpu_data[i].kbuf);
> -			if (handle->cpu_data[i].page_map)
> -
> free_page_map(handle->cpu_data[i].page_map); -
> -			if (handle->cpu_data[i].page_cnt)
> -				tracecmd_warning("%d pages still
> allocated on cpu %d%s",
> -
> handle->cpu_data[i].page_cnt, i,
> -
> show_records(handle->cpu_data[i].pages,
> -
> handle->cpu_data[i].nr_pages));
> -			free(handle->cpu_data[i].pages);
> +		if (handle->cpu_data) {
> +			if (handle->cpu_data[i].kbuf) {
> +
> kbuffer_free(handle->cpu_data[i].kbuf);
> +				if (handle->cpu_data[i].page_map)
> +
> free_page_map(handle->cpu_data[i].page_map); +
> +				if (handle->cpu_data[i].page_cnt)
> +					tracecmd_warning("%d pages
> still allocated on cpu %d%s",
> +
> handle->cpu_data[i].page_cnt, i,
> +
> show_records(handle->cpu_data[i].pages,
> +
>  handle->cpu_data[i].nr_pages));
> +				free(handle->cpu_data[i].pages);
> +			}
> +			if (handle->cpu_data[i].compress.fd >= 0) {
> +
> close(handle->cpu_data[i].compress.fd);
> +
> unlink(handle->cpu_data[i].compress.file);
> +			}
> +			while
> (!list_empty(&handle->cpu_data[i].compress.cache)) {
> +				cache =
> container_of(handle->cpu_data[i].compress.cache.next,
> +						     struct
> zchunk_cache, list);
> +				list_del(&cache->list);
> +				free(cache->map);
> +				free(cache);
> +			}
> +			free(handle->cpu_data[i].compress.chunks);
>  		}
>  	}
>  




[Index of Archives]     [Linux USB Development]     [Linux USB Development]     [Linux Audio Users]     [Yosemite Hiking]     [Linux Kernel]     [Linux SCSI]

  Powered by Linux