aboutsummaryrefslogtreecommitdiffstats
path: root/tools/perf/builtin-top.c
diff options
context:
space:
mode:
Diffstat (limited to 'tools/perf/builtin-top.c')
-rw-r--r--tools/perf/builtin-top.c56
1 files changed, 3 insertions, 53 deletions
diff --git a/tools/perf/builtin-top.c b/tools/perf/builtin-top.c
index df85c1f9417b..58352ad807c7 100644
--- a/tools/perf/builtin-top.c
+++ b/tools/perf/builtin-top.c
@@ -1100,67 +1100,17 @@ static void event__process_sample(const event_t *self,
1100 1100
1101static void perf_session__mmap_read_cpu(struct perf_session *self, int cpu) 1101static void perf_session__mmap_read_cpu(struct perf_session *self, int cpu)
1102{ 1102{
1103 struct perf_mmap *md = &evsel_list->mmap[cpu];
1104 unsigned int head = perf_mmap__read_head(md);
1105 unsigned int old = md->prev;
1106 unsigned char *data = md->base + page_size;
1107 struct sample_data sample; 1103 struct sample_data sample;
1108 int diff; 1104 event_t *event;
1109
1110 /*
1111 * If we're further behind than half the buffer, there's a chance
1112 * the writer will bite our tail and mess up the samples under us.
1113 *
1114 * If we somehow ended up ahead of the head, we got messed up.
1115 *
1116 * In either case, truncate and restart at head.
1117 */
1118 diff = head - old;
1119 if (diff > md->mask / 2 || diff < 0) {
1120 fprintf(stderr, "WARNING: failed to keep up with mmap data.\n");
1121
1122 /*
1123 * head points to a known good entry, start there.
1124 */
1125 old = head;
1126 }
1127
1128 for (; old != head;) {
1129 event_t *event = (event_t *)&data[old & md->mask];
1130
1131 event_t event_copy;
1132
1133 size_t size = event->header.size;
1134
1135 /*
1136 * Event straddles the mmap boundary -- header should always
1137 * be inside due to u64 alignment of output.
1138 */
1139 if ((old & md->mask) + size != ((old + size) & md->mask)) {
1140 unsigned int offset = old;
1141 unsigned int len = min(sizeof(*event), size), cpy;
1142 void *dst = &event_copy;
1143
1144 do {
1145 cpy = min(md->mask + 1 - (offset & md->mask), len);
1146 memcpy(dst, &data[offset & md->mask], cpy);
1147 offset += cpy;
1148 dst += cpy;
1149 len -= cpy;
1150 } while (len);
1151
1152 event = &event_copy;
1153 }
1154 1105
1106 while ((event = perf_evlist__read_on_cpu(evsel_list, cpu)) != NULL) {
1155 event__parse_sample(event, self, &sample); 1107 event__parse_sample(event, self, &sample);
1108
1156 if (event->header.type == PERF_RECORD_SAMPLE) 1109 if (event->header.type == PERF_RECORD_SAMPLE)
1157 event__process_sample(event, &sample, self); 1110 event__process_sample(event, &sample, self);
1158 else 1111 else
1159 event__process(event, &sample, self); 1112 event__process(event, &sample, self);
1160 old += size;
1161 } 1113 }
1162
1163 md->prev = old;
1164} 1114}
1165 1115
1166static void perf_session__mmap_read(struct perf_session *self) 1116static void perf_session__mmap_read(struct perf_session *self)