1 /*******************************************************************************
2 * Copyright (c) 2015, 2016 EfficiOS Inc., Ericsson
4 * All rights reserved. This program and the accompanying materials
5 * are made available under the terms of the Eclipse Public License v1.0
6 * which accompanies this distribution, and is available at
7 * http://www.eclipse.org/legal/epl-v10.html
8 *******************************************************************************/
10 package org
.eclipse
.tracecompass
.internal
.analysis
.os
.linux
.core
.latency
;
12 import static org
.eclipse
.tracecompass
.common
.core
.NonNullUtils
.checkNotNull
;
14 import java
.io
.IOException
;
15 import java
.io
.ObjectInputStream
;
16 import java
.util
.Collection
;
17 import java
.util
.Collections
;
18 import java
.util
.Comparator
;
19 import java
.util
.HashMap
;
22 import org
.eclipse
.core
.runtime
.IProgressMonitor
;
23 import org
.eclipse
.core
.runtime
.NullProgressMonitor
;
24 import org
.eclipse
.jdt
.annotation
.NonNull
;
25 import org
.eclipse
.jdt
.annotation
.Nullable
;
26 import org
.eclipse
.tracecompass
.analysis
.os
.linux
.core
.kernel
.KernelTidAspect
;
27 import org
.eclipse
.tracecompass
.analysis
.os
.linux
.core
.tid
.TidAnalysisModule
;
28 import org
.eclipse
.tracecompass
.analysis
.os
.linux
.core
.trace
.IKernelAnalysisEventLayout
;
29 import org
.eclipse
.tracecompass
.analysis
.os
.linux
.core
.trace
.IKernelTrace
;
30 import org
.eclipse
.tracecompass
.analysis
.timing
.core
.segmentstore
.AbstractSegmentStoreAnalysisEventBasedModule
;
31 import org
.eclipse
.tracecompass
.segmentstore
.core
.ISegment
;
32 import org
.eclipse
.tracecompass
.segmentstore
.core
.ISegmentStore
;
33 import org
.eclipse
.tracecompass
.tmf
.core
.analysis
.IAnalysisModule
;
34 import org
.eclipse
.tracecompass
.tmf
.core
.event
.ITmfEvent
;
35 import org
.eclipse
.tracecompass
.tmf
.core
.segment
.ISegmentAspect
;
36 import org
.eclipse
.tracecompass
.tmf
.core
.trace
.ITmfTrace
;
38 import com
.google
.common
.collect
.ImmutableList
;
39 import com
.google
.common
.collect
.ImmutableSet
;
42 * @author Alexandre Montplaisir
45 public class SystemCallLatencyAnalysis
extends AbstractSegmentStoreAnalysisEventBasedModule
{
48 * The ID of this analysis
50 public static final String ID
= "org.eclipse.tracecompass.analysis.os.linux.latency.syscall"; //$NON-NLS-1$
52 private static final String DATA_FILENAME
= "latency-analysis.dat"; //$NON-NLS-1$
54 private static final Collection
<ISegmentAspect
> BASE_ASPECTS
=
55 ImmutableList
.of(SyscallNameAspect
.INSTANCE
);
58 public String
getId() {
63 protected Iterable
<IAnalysisModule
> getDependentAnalyses() {
64 ITmfTrace trace
= getTrace();
66 throw new IllegalStateException();
68 IAnalysisModule module
= trace
.getAnalysisModule(TidAnalysisModule
.ID
);
70 return Collections
.EMPTY_SET
;
72 return ImmutableSet
.of(module
);
76 public Iterable
<ISegmentAspect
> getSegmentAspects() {
81 public @NonNull String
getDataFileName() {
86 public AbstractSegmentStoreAnalysisRequest
createAnalysisRequest(ISegmentStore
<ISegment
> syscalls
) {
87 return new SyscallLatencyAnalysisRequest(syscalls
);
91 protected Object
[] readObject(ObjectInputStream ois
) throws ClassNotFoundException
, IOException
{
92 return checkNotNull((Object
[]) ois
.readObject());
95 private class SyscallLatencyAnalysisRequest
extends AbstractSegmentStoreAnalysisRequest
{
97 private final Map
<Integer
, SystemCall
.InitialInfo
> fOngoingSystemCalls
= new HashMap
<>();
98 private @Nullable IKernelAnalysisEventLayout fLayout
;
99 private final IProgressMonitor fMonitor
= new NullProgressMonitor();
101 public SyscallLatencyAnalysisRequest(ISegmentStore
<ISegment
> syscalls
) {
106 public void handleData(final ITmfEvent event
) {
107 super.handleData(event
);
108 IKernelAnalysisEventLayout layout
= fLayout
;
109 if (layout
== null) {
110 IKernelTrace trace
= (IKernelTrace
) event
.getTrace();
111 layout
= trace
.getKernelEventLayout();
114 final String eventName
= event
.getType().getName();
116 if (eventName
.startsWith(layout
.eventSyscallEntryPrefix()) ||
117 eventName
.startsWith(layout
.eventCompatSyscallEntryPrefix())) {
118 /* This is a system call entry event */
122 tid
= KernelTidAspect
.INSTANCE
.resolve(event
, true, fMonitor
);
123 } catch (InterruptedException e
) {
127 // no information on this event/trace ?
131 /* Record the event's data into the intial system call info */
132 // String syscallName = fLayout.getSyscallNameFromEvent(event);
133 long startTime
= event
.getTimestamp().getValue();
134 String syscallName
= eventName
.substring(layout
.eventSyscallEntryPrefix().length());
136 SystemCall
.InitialInfo newSysCall
= new SystemCall
.InitialInfo(startTime
, syscallName
.intern());
137 fOngoingSystemCalls
.put(tid
, newSysCall
);
139 } else if (eventName
.startsWith(layout
.eventSyscallExitPrefix())) {
140 /* This is a system call exit event */
144 tid
= KernelTidAspect
.INSTANCE
.resolve(event
, true, fMonitor
);
145 } catch (InterruptedException e
) {
152 SystemCall
.InitialInfo info
= fOngoingSystemCalls
.remove(tid
);
155 * We have not seen the entry event corresponding to this
156 * exit (lost event, or before start of trace).
161 long endTime
= event
.getTimestamp().getValue();
162 ISegment syscall
= new SystemCall(info
, endTime
);
163 getSegmentStore().add(syscall
);
168 public void handleCompleted() {
169 fOngoingSystemCalls
.clear();
170 super.handleCompleted();
174 public void handleCancel() {
175 fMonitor
.setCanceled(true);
176 super.handleCancel();
180 private static class SyscallNameAspect
implements ISegmentAspect
{
181 public static final ISegmentAspect INSTANCE
= new SyscallNameAspect();
183 private SyscallNameAspect() { }
186 public String
getHelpText() {
187 return checkNotNull(Messages
.SegmentAspectHelpText_SystemCall
);
190 public String
getName() {
191 return checkNotNull(Messages
.SegmentAspectName_SystemCall
);
194 public @Nullable Comparator
<?
> getComparator() {
198 public @Nullable String
resolve(ISegment segment
) {
199 if (segment
instanceof SystemCall
) {
200 return ((SystemCall
) segment
).getName();