Skip to content

Commit

Permalink
perf tools: Fix ommitted mmap data update on remap
Browse files Browse the repository at this point in the history
Commit eac9eac "perf tools: Check we are able to read the event
size on mmap" brought a check to ensure we can read the size of the
event before dereferencing it, and do a remap otherwise to move the
buffer forward.

However that remap was ommitting all the necessary work to
update the new page offset, head, and to unmap previous pages,
etc...

To fix this, gather all the code that fetches the event in a
seperate helper which does all the necessary checks about the
header/event size and tells us anytime a remap is needed.

Signed-off-by: Frederic Weisbecker <[email protected]>
Cc: Peter Zijlstra <[email protected]>
Cc: Arnaldo Carvalho de Melo <[email protected]>
Cc: Stephane Eranian <[email protected]>
Link: http://lkml.kernel.org/r/[email protected]
Signed-off-by: Ingo Molnar <[email protected]>
  • Loading branch information
fweisbec authored and Ingo Molnar committed May 23, 2011
1 parent 4eec42f commit 998bedc
Showing 1 changed file with 26 additions and 13 deletions.
39 changes: 26 additions & 13 deletions tools/perf/util/session.c
Original file line number Diff line number Diff line change
Expand Up @@ -960,6 +960,30 @@ static int __perf_session__process_pipe_events(struct perf_session *self,
return err;
}

static union perf_event *
fetch_mmaped_event(struct perf_session *session,
u64 head, size_t mmap_size, char *buf)
{
union perf_event *event;

/*
* Ensure we have enough space remaining to read
* the size of the event in the headers.
*/
if (head + sizeof(event->header) > mmap_size)
return NULL;

event = (union perf_event *)(buf + head);

if (session->header.needs_swap)
perf_event_header__bswap(&event->header);

if (head + event->header.size > mmap_size)
return NULL;

return event;
}

int __perf_session__process_events(struct perf_session *session,
u64 data_offset, u64 data_size,
u64 file_size, struct perf_event_ops *ops)
Expand Down Expand Up @@ -1014,19 +1038,8 @@ int __perf_session__process_events(struct perf_session *session,
file_pos = file_offset + head;

more:
/*
* Ensure we have enough space remaining to read
* the size of the event in the headers.
*/
if (head + sizeof(event->header) > mmap_size)
goto remap;

event = (union perf_event *)(buf + head);

if (session->header.needs_swap)
perf_event_header__bswap(&event->header);

if (head + event->header.size > mmap_size) {
event = fetch_mmaped_event(session, head, mmap_size, buf);
if (!event) {
if (mmaps[map_idx]) {
munmap(mmaps[map_idx], mmap_size);
mmaps[map_idx] = NULL;
Expand Down

0 comments on commit 998bedc

Please sign in to comment.