On Wed, 23 Jun 2021 14:40:49 -0400 Steven Rostedt <rostedt@xxxxxxxxxxx> wrote: > CPU DATA > -------- > > The CPU data is located in the part of the file that is specified > in the end of the header. Padding is placed between the header and > the CPU data, placing the CPU data at a page aligned (target page) position > in the file. > > This data is copied directly from the Ftrace ring buffer and is of the > same format as the ring buffer specified by the event header files > loaded in the header format file. > > The trace-cmd tool will try to \fBmmap(2)\fR the data page by page with the > target's page size if possible. If it fails to mmap, it will just read the > data instead. A couple of things for the CPU date. I think we can update how TRACECMD_OPTION_BUFFER is parsed: So instead of: 3 TRACECMD_OPTION_BUFFER String containing the name of the buffer instance, "\0" if it is the top level buffer. 8 bytes that point to where an instance buffer exists We turn it into: 3 TRACECMD_OPTION_BUFFER String containing the name of the buffer instance, "\0" if it is the top level buffer. 4 bytes - number of CPUs Array of the above number of CPUS - 4 bytes - CPU identifier (allow for CPUs to have something other than 0 - nr_cpus), like just (CPU 2 and CPU 5) - 8 bytes - offset into the file where the CPU data section exists - 8 bytes - size of the CPU data section Then for the CPU data that is pointed to, they would have the TRACECMD_OPTION_BUFFER as their "type". 2 bytes - TRACECMD_OPTION_BUFFER 2 bytes - flags 8 bytes - size of section in file 8 bytes - size of uncompressed data (or size if not compressed) Page aligned (if not compressed) raw data. Now if we want to add a way to have a per page compression, then create a new option: TRACECMD_OPTION_BUFFER_COMPRESSED nul terminated string, for the instance name 4 bytes - number of CPUs 4 bytes - size of the buffer pages when uncompressed (they should all be the same) Array of the above number of CPUS - 4 bytes - CPU identifier (allow for CPUs to have something other than 0 - nr_cpus), like just (CPU 2 and CPU 5) - 8 bytes - offset into the file where the CPU data section exists - 8 bytes - size of the CPU data section The per page compressed buffers would have the TRACECMD_OPTION_BUFFER_COMPRESSED as their type. Since I believe all sections should still start with that special header, the 2 bytes - Type of section (this will be the same as the option type) 2 bytes - Flags - currently bit zero will define if the section is compressed or not. 8 bytes - The size of the section in the file. 8 bytes - The size of the section when uncompressed. If it is not compressed, then it will be equal to the size of the section in the file. There would be no sense in compressing the section if the per pages are going to be compressed, which would lead to this: 2 bytes - TRACECMD_OPTION_BUFFER_COMPRESSED 2 bytes - zero (no compression of the section itself) 8 bytes - size of the section in the file 8 bytes - size of the section in the files (uncompressed so the above two numbers are the same) 4 bytes - size of compressed page in file [ compressed page data ] 4 bytes - size of next compressed page [ ... ] Now to map the above if compressed by tracecmd_read_at(), it would require two different methods. For the TRACECMD_OPTION_BUFFER_COMPRESSED, it would be easy. We would need to create a "uncompressed start" virtual address that we can use for the record offsets (as they all need to be unique in the file). TRACECMD_OPTION_BUFFER_COMPRESSED_MAP 4 bytes - the size of each uncompressed page. 4 bytes - the CPU number of the mapping 8 bytes - The "uncompressed start" 8 bytes - The "uncompressed end" 8 bytes - offset into file where the compressed map is // the below is more pseudo code tracecmd_read_at(struct tracecmd_input *handle, unsigned long long offset, int *pcpu) { if (handle->compressed) return read_at_compressed(handle, offset, pcpu); [..] } read_at_compressed(struct tracecmd_input *handle, unsigned long long offset, int *pcpu) { for (compressed = handle->compressed; *compressed; compressed = compressed->next) { if (offset >= compressed->start && offset < compressed->end) beak; } if (!compressed) return NULL; *pcpu = compressed->cpu; offset -= compressed->start; index = offset / handle->page_size; page = uncompress_page(compress->offset, index); return read_record(page + (offset - index * handle->page_size), *pcpu); } It's important to note that the "uncompressed start" and "uncompressed end" must not overlap with any other buffer. They just need to be unique. Now, if we compress the data as one chunk, the above would not work, and we would need to come up with another plan. For now, we could just avoid compressing the data as one chunk (not by individual pages). -- Steve