aboutsummaryrefslogtreecommitdiffstats
path: root/traceview/src/com
diff options
context:
space:
mode:
Diffstat (limited to 'traceview/src/com')
-rw-r--r--traceview/src/com/android/traceview/Call.java141
-rw-r--r--traceview/src/com/android/traceview/ColorController.java113
-rw-r--r--traceview/src/com/android/traceview/DmTraceReader.java602
-rw-r--r--traceview/src/com/android/traceview/MainWindow.java192
-rw-r--r--traceview/src/com/android/traceview/MethodData.java458
-rw-r--r--traceview/src/com/android/traceview/ProfileData.java81
-rw-r--r--traceview/src/com/android/traceview/ProfileNode.java51
-rw-r--r--traceview/src/com/android/traceview/ProfileProvider.java361
-rw-r--r--traceview/src/com/android/traceview/ProfileSelf.java34
-rw-r--r--traceview/src/com/android/traceview/ProfileView.java308
-rw-r--r--traceview/src/com/android/traceview/QtraceReader.java45
-rw-r--r--traceview/src/com/android/traceview/Selection.java70
-rw-r--r--traceview/src/com/android/traceview/SelectionController.java35
-rw-r--r--traceview/src/com/android/traceview/ThreadData.java228
-rw-r--r--traceview/src/com/android/traceview/TickScaler.java148
-rw-r--r--traceview/src/com/android/traceview/TimeLineView.java1961
-rw-r--r--traceview/src/com/android/traceview/TraceReader.java55
-rw-r--r--traceview/src/com/android/traceview/TraceUnits.java93
18 files changed, 0 insertions, 4976 deletions
diff --git a/traceview/src/com/android/traceview/Call.java b/traceview/src/com/android/traceview/Call.java
deleted file mode 100644
index 40ac244..0000000
--- a/traceview/src/com/android/traceview/Call.java
+++ /dev/null
@@ -1,141 +0,0 @@
-/*
- * Copyright (C) 2006 The Android Open Source Project
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
-
-package com.android.traceview;
-
-import org.eclipse.swt.graphics.Color;
-
-class Call implements TimeLineView.Block {
-
- // Values for bits within the mFlags field.
- private static final int METHOD_ACTION_MASK = 0x3;
- private static final int IS_RECURSIVE = 0x10;
-
- private int mThreadId;
- private int mFlags;
- MethodData mMethodData;
-
- /** 0-based thread-local start time */
- long mThreadStartTime;
-
- /** global start time */
- long mGlobalStartTime;
-
- /** global end time */
- long mGlobalEndTime;
-
- private String mName;
-
- /**
- * This constructor is used for the root of a Call tree. The name is
- * the name of the corresponding thread.
- */
- Call(String name, MethodData methodData) {
- mName = name;
- mMethodData = methodData;
- }
-
- Call() {
- }
-
- Call(int threadId, MethodData methodData, long time, int methodAction) {
- mThreadId = threadId;
- mMethodData = methodData;
- mThreadStartTime = time;
- mFlags = methodAction & METHOD_ACTION_MASK;
- mName = methodData.getProfileName();
- }
-
- public void set(int threadId, MethodData methodData, long time, int methodAction) {
- mThreadId = threadId;
- mMethodData = methodData;
- mThreadStartTime = time;
- mFlags = methodAction & METHOD_ACTION_MASK;
- mName = methodData.getProfileName();
- }
-
- public void updateName() {
- mName = mMethodData.getProfileName();
- }
-
- public double addWeight(int x, int y, double weight) {
- return mMethodData.addWeight(x, y, weight);
- }
-
- public void clearWeight() {
- mMethodData.clearWeight();
- }
-
- public long getStartTime() {
- return mGlobalStartTime;
- }
-
- public long getEndTime() {
- return mGlobalEndTime;
- }
-
- public Color getColor() {
- return mMethodData.getColor();
- }
-
- public void addExclusiveTime(long elapsed) {
- mMethodData.addElapsedExclusive(elapsed);
- if ((mFlags & IS_RECURSIVE) == 0) {
- mMethodData.addTopExclusive(elapsed);
- }
- }
-
- public void addInclusiveTime(long elapsed, Call parent) {
- boolean isRecursive = (mFlags & IS_RECURSIVE) != 0;
- mMethodData.addElapsedInclusive(elapsed, isRecursive, parent);
- }
-
- public String getName() {
- return mName;
- }
-
- public void setName(String name) {
- mName = name;
- }
-
- int getThreadId() {
- return mThreadId;
- }
-
- public MethodData getMethodData() {
- return mMethodData;
- }
-
- int getMethodAction() {
- return mFlags & METHOD_ACTION_MASK;
- }
-
- public void dump() {
- System.out.printf("%s [%d, %d]\n", mName, mGlobalStartTime, mGlobalEndTime);
- }
-
- public void setRecursive(boolean isRecursive) {
- if (isRecursive) {
- mFlags |= IS_RECURSIVE;
- } else {
- mFlags &= ~IS_RECURSIVE;
- }
- }
-
- public boolean isRecursive() {
- return (mFlags & IS_RECURSIVE) != 0;
- }
-}
diff --git a/traceview/src/com/android/traceview/ColorController.java b/traceview/src/com/android/traceview/ColorController.java
deleted file mode 100644
index f5e4c0d..0000000
--- a/traceview/src/com/android/traceview/ColorController.java
+++ /dev/null
@@ -1,113 +0,0 @@
-/*
- * Copyright (C) 2006 The Android Open Source Project
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
-
-package com.android.traceview;
-
-import java.util.HashMap;
-
-import org.eclipse.swt.SWT;
-import org.eclipse.swt.graphics.Color;
-import org.eclipse.swt.graphics.GC;
-import org.eclipse.swt.graphics.Image;
-import org.eclipse.swt.graphics.RGB;
-import org.eclipse.swt.widgets.Display;
-
-public class ColorController {
- private static final int[] systemColors = { SWT.COLOR_BLUE, SWT.COLOR_RED,
- SWT.COLOR_GREEN, SWT.COLOR_CYAN, SWT.COLOR_MAGENTA, SWT.COLOR_DARK_BLUE,
- SWT.COLOR_DARK_RED, SWT.COLOR_DARK_GREEN, SWT.COLOR_DARK_YELLOW,
- SWT.COLOR_DARK_CYAN, SWT.COLOR_DARK_MAGENTA, SWT.COLOR_BLACK };
-
- private static RGB[] rgbColors = { new RGB(90, 90, 255), // blue
- new RGB(0, 240, 0), // green
- new RGB(255, 0, 0), // red
- new RGB(0, 255, 255), // cyan
- new RGB(255, 80, 255), // magenta
- new RGB(200, 200, 0), // yellow
- new RGB(40, 0, 200), // dark blue
- new RGB(150, 255, 150), // light green
- new RGB(150, 0, 0), // dark red
- new RGB(30, 150, 150), // dark cyan
- new RGB(200, 200, 255), // light blue
- new RGB(0, 120, 0), // dark green
- new RGB(255, 150, 150), // light red
- new RGB(140, 80, 140), // dark magenta
- new RGB(150, 100, 50), // brown
- new RGB(70, 70, 70), // dark grey
- };
-
- private static HashMap<Integer, Color> colorCache = new HashMap<Integer, Color>();
- private static HashMap<Integer, Image> imageCache = new HashMap<Integer, Image>();
-
- public ColorController() {
- }
-
- public static Color requestColor(Display display, RGB rgb) {
- return requestColor(display, rgb.red, rgb.green, rgb.blue);
- }
-
- public static Image requestColorSquare(Display display, RGB rgb) {
- return requestColorSquare(display, rgb.red, rgb.green, rgb.blue);
- }
-
- public static Color requestColor(Display display, int red, int green, int blue) {
- int key = (red << 16) | (green << 8) | blue;
- Color color = colorCache.get(key);
- if (color == null) {
- color = new Color(display, red, green, blue);
- colorCache.put(key, color);
- }
- return color;
- }
-
- public static Image requestColorSquare(Display display, int red, int green, int blue) {
- int key = (red << 16) | (green << 8) | blue;
- Image image = imageCache.get(key);
- if (image == null) {
- image = new Image(display, 8, 14);
- GC gc = new GC(image);
- Color color = requestColor(display, red, green, blue);
- gc.setBackground(color);
- gc.fillRectangle(image.getBounds());
- gc.dispose();
- imageCache.put(key, image);
- }
- return image;
- }
-
- public static void assignMethodColors(Display display, MethodData[] methods) {
- int nextColorIndex = 0;
- for (MethodData md : methods) {
- RGB rgb = rgbColors[nextColorIndex];
- if (++nextColorIndex == rgbColors.length)
- nextColorIndex = 0;
- Color color = requestColor(display, rgb);
- Image image = requestColorSquare(display, rgb);
- md.setColor(color);
- md.setImage(image);
-
- // Compute and set a faded color
- int fadedRed = 150 + rgb.red / 4;
- int fadedGreen = 150 + rgb.green / 4;
- int fadedBlue = 150 + rgb.blue / 4;
- RGB faded = new RGB(fadedRed, fadedGreen, fadedBlue);
- color = requestColor(display, faded);
- image = requestColorSquare(display, faded);
- md.setFadedColor(color);
- md.setFadedImage(image);
- }
- }
-}
diff --git a/traceview/src/com/android/traceview/DmTraceReader.java b/traceview/src/com/android/traceview/DmTraceReader.java
deleted file mode 100644
index 5a19c19..0000000
--- a/traceview/src/com/android/traceview/DmTraceReader.java
+++ /dev/null
@@ -1,602 +0,0 @@
-/*
- * Copyright (C) 2006 The Android Open Source Project
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
-
-package com.android.traceview;
-
-import java.io.BufferedReader;
-import java.io.File;
-import java.io.FileInputStream;
-import java.io.FileNotFoundException;
-import java.io.FileReader;
-import java.io.IOException;
-import java.nio.BufferUnderflowException;
-import java.nio.ByteOrder;
-import java.nio.MappedByteBuffer;
-import java.nio.channels.FileChannel;
-import java.util.ArrayList;
-import java.util.Arrays;
-import java.util.Collection;
-import java.util.Comparator;
-import java.util.HashMap;
-import java.util.regex.Matcher;
-import java.util.regex.Pattern;
-
-public class DmTraceReader extends TraceReader {
-
- private int mVersionNumber = 0;
- private boolean mDebug = false;
- private static final int TRACE_MAGIC = 0x574f4c53;
- private boolean mRegression;
- private ProfileProvider mProfileProvider;
- private String mTraceFileName;
- private MethodData mTopLevel;
- private ArrayList<Call> mCallList;
- private ArrayList<Call> mSwitchList;
- private HashMap<Integer, MethodData> mMethodMap;
- private HashMap<Integer, ThreadData> mThreadMap;
- private ThreadData[] mSortedThreads;
- private MethodData[] mSortedMethods;
- private long mGlobalEndTime;
- private MethodData mContextSwitch;
- private int mOffsetToData;
- private byte[] mBytes = new byte[8];
-
- // A regex for matching the thread "id name" lines in the .key file
- private static final Pattern mIdNamePattern = Pattern.compile("(\\d+)\t(.*)"); // $NON-NLS-1$
-
- DmTraceReader(String traceFileName, boolean regression) {
- mTraceFileName = traceFileName;
- mRegression = regression;
- mMethodMap = new HashMap<Integer, MethodData>();
- mThreadMap = new HashMap<Integer, ThreadData>();
-
- // Create a single top-level MethodData object to hold the profile data
- // for time spent in the unknown caller.
- mTopLevel = new MethodData(0, "(toplevel)");
- mContextSwitch = new MethodData(-1, "(context switch)");
- mMethodMap.put(0, mTopLevel);
- generateTrees();
- // dumpTrees();
- }
-
- void generateTrees() {
- try {
- long offset = parseKeys();
- parseData(offset);
- analyzeData();
- } catch (IOException e) {
- System.err.println(e.getMessage());
- System.exit(1);
- }
- }
-
- @Override
- public ProfileProvider getProfileProvider() {
- if (mProfileProvider == null)
- mProfileProvider = new ProfileProvider(this);
- return mProfileProvider;
- }
-
- Call readCall(MappedByteBuffer buffer, Call call) {
- int threadId;
- int methodId;
- long time;
-
- try {
- if (mVersionNumber == 1)
- threadId = buffer.get();
- else
- threadId = buffer.getShort();
- methodId = buffer.getInt();
- time = buffer.getInt();
- } catch (BufferUnderflowException ex) {
- return null;
- }
-
- int methodAction = methodId & 0x03;
- methodId = methodId & ~0x03;
- MethodData methodData = mMethodMap.get(methodId);
- if (methodData == null) {
- String name = String.format("(0x%1$x)", methodId); // $NON-NLS-1$
- methodData = new MethodData(methodId, name);
- }
-
- if (call != null) {
- call.set(threadId, methodData, time, methodAction);
- } else {
- call = new Call(threadId, methodData, time, methodAction);
- }
- return call;
- }
-
- private MappedByteBuffer mapFile(String filename, long offset) {
- MappedByteBuffer buffer = null;
- try {
- FileInputStream dataFile = new FileInputStream(filename);
- File file = new File(filename);
- FileChannel fc = dataFile.getChannel();
- buffer = fc.map(FileChannel.MapMode.READ_ONLY, offset, file.length() - offset);
- buffer.order(ByteOrder.LITTLE_ENDIAN);
- } catch (FileNotFoundException ex) {
- System.err.println(ex.getMessage());
- System.exit(1);
- } catch (IOException ex) {
- System.err.println(ex.getMessage());
- System.exit(1);
- }
-
- return buffer;
- }
-
- private void readDataFileHeader(MappedByteBuffer buffer) {
- int magic = buffer.getInt();
- if (magic != TRACE_MAGIC) {
- System.err.printf(
- "Error: magic number mismatch; got 0x%x, expected 0x%x\n",
- magic, TRACE_MAGIC);
- throw new RuntimeException();
- }
- // read version
- int version = buffer.getShort();
-
- // read offset
- mOffsetToData = buffer.getShort() - 16;
-
- // read startWhen
- buffer.getLong();
-
- // Skip over "mOffsetToData" bytes
- for (int ii = 0; ii < mOffsetToData; ii++) {
- buffer.get();
- }
-
- // Save this position so that we can re-read the data later
- buffer.mark();
- }
-
- private void parseData(long offset) {
- MappedByteBuffer buffer = mapFile(mTraceFileName, offset);
- readDataFileHeader(buffer);
- parseDataPass1(buffer);
-
- buffer.reset();
- parseDataPass2(buffer);
- }
-
- private void parseDataPass1(MappedByteBuffer buffer) {
- mSwitchList = new ArrayList<Call>();
-
- // Read the first call so that we can set "prevThreadData"
- Call call = new Call();
- call = readCall(buffer, call);
- if (call == null)
- return;
- long callTime = call.mThreadStartTime;
- long prevCallTime = 0;
- ThreadData threadData = mThreadMap.get(call.getThreadId());
- if (threadData == null) {
- String name = String.format("[%1$d]", call.getThreadId()); // $NON-NLS-1$
- threadData = new ThreadData(call.getThreadId(), name, mTopLevel);
- mThreadMap.put(call.getThreadId(), threadData);
- }
- ThreadData prevThreadData = threadData;
- while (true) {
- // If a context switch occurred, then insert a placeholder "call"
- // record so that we can do something reasonable with the global
- // timestamps.
- if (prevThreadData != threadData) {
- Call switchEnter = new Call(prevThreadData.getId(),
- mContextSwitch, prevCallTime, 0);
- prevThreadData.setLastContextSwitch(switchEnter);
- mSwitchList.add(switchEnter);
- Call contextSwitch = threadData.getLastContextSwitch();
- if (contextSwitch != null) {
- long prevStartTime = contextSwitch.mThreadStartTime;
- long elapsed = callTime - prevStartTime;
- long beforeSwitch = elapsed / 2;
- long afterSwitch = elapsed - beforeSwitch;
- long exitTime = callTime - afterSwitch;
- contextSwitch.mThreadStartTime = prevStartTime + beforeSwitch;
- Call switchExit = new Call(threadData.getId(),
- mContextSwitch, exitTime, 1);
-
- mSwitchList.add(switchExit);
- }
- prevThreadData = threadData;
- }
-
- // Read the next call
- call = readCall(buffer, call);
- if (call == null) {
- break;
- }
- prevCallTime = callTime;
- callTime = call.mThreadStartTime;
-
- threadData = mThreadMap.get(call.getThreadId());
- if (threadData == null) {
- String name = String.format("[%d]", call.getThreadId());
- threadData = new ThreadData(call.getThreadId(), name, mTopLevel);
- mThreadMap.put(call.getThreadId(), threadData);
- }
- }
- }
-
- void parseDataPass2(MappedByteBuffer buffer) {
- mCallList = new ArrayList<Call>();
-
- // Read the first call so that we can set "prevThreadData"
- Call call = readCall(buffer, null);
- long callTime = call.mThreadStartTime;
- long prevCallTime = callTime;
- ThreadData threadData = mThreadMap.get(call.getThreadId());
- ThreadData prevThreadData = threadData;
- threadData.setGlobalStartTime(0);
-
- int nthContextSwitch = 0;
-
- // Assign a global timestamp to each event.
- long globalTime = 0;
- while (true) {
- long elapsed = callTime - prevCallTime;
- if (threadData != prevThreadData) {
- // Get the next context switch. This one is entered
- // by the previous thread.
- Call contextSwitch = mSwitchList.get(nthContextSwitch++);
- mCallList.add(contextSwitch);
- elapsed = contextSwitch.mThreadStartTime - prevCallTime;
- globalTime += elapsed;
- elapsed = 0;
- contextSwitch.mGlobalStartTime = globalTime;
- prevThreadData.handleCall(contextSwitch, globalTime);
-
- if (!threadData.isEmpty()) {
- // This context switch is exited by the current thread.
- contextSwitch = mSwitchList.get(nthContextSwitch++);
- mCallList.add(contextSwitch);
- contextSwitch.mGlobalStartTime = globalTime;
- elapsed = callTime - contextSwitch.mThreadStartTime;
- threadData.handleCall(contextSwitch, globalTime);
- }
-
- // If the thread's global start time has not been set yet,
- // then set it.
- if (threadData.getGlobalStartTime() == -1)
- threadData.setGlobalStartTime(globalTime);
- prevThreadData = threadData;
- }
-
- globalTime += elapsed;
- call.mGlobalStartTime = globalTime;
-
- threadData.handleCall(call, globalTime);
- mCallList.add(call);
-
- // Read the next call
- call = readCall(buffer, null);
- if (call == null) {
- break;
- }
- prevCallTime = callTime;
- callTime = call.mThreadStartTime;
- threadData = mThreadMap.get(call.getThreadId());
- }
-
- // Allow each thread to do any cleanup of the call stack.
- // Also add the elapsed time for each thread to the toplevel
- // method's inclusive time.
- for (int id : mThreadMap.keySet()) {
- threadData = mThreadMap.get(id);
- long endTime = threadData.endTrace();
- if (endTime > 0)
- mTopLevel.addElapsedInclusive(endTime, false, null);
- }
-
- mGlobalEndTime = globalTime;
-
- if (mRegression) {
- dumpCallTimes();
- }
- }
-
- static final int PARSE_VERSION = 0;
- static final int PARSE_THREADS = 1;
- static final int PARSE_METHODS = 2;
- static final int PARSE_OPTIONS = 4;
-
- long parseKeys() throws IOException {
- BufferedReader in = null;
- try {
- in = new BufferedReader(new FileReader(mTraceFileName));
- } catch (FileNotFoundException ex) {
- System.err.println(ex.getMessage());
- }
-
- long offset = 0;
- int mode = PARSE_VERSION;
- String line = null;
- while (true) {
- line = in.readLine();
- if (line == null) {
- throw new IOException("Key section does not have an *end marker");
- }
-
- // Calculate how much we have read from the file so far. The
- // extra byte is for the line ending not included by readLine().
- offset += line.length() + 1;
- if (line.startsWith("*")) {
- if (line.equals("*version")) {
- mode = PARSE_VERSION;
- continue;
- }
- if (line.equals("*threads")) {
- mode = PARSE_THREADS;
- continue;
- }
- if (line.equals("*methods")) {
- mode = PARSE_METHODS;
- continue;
- }
- if (line.equals("*end")) {
- return offset;
- }
- }
- switch (mode) {
- case PARSE_VERSION:
- mVersionNumber = Integer.decode(line);
- mode = PARSE_OPTIONS;
- break;
- case PARSE_THREADS:
- parseThread(line);
- break;
- case PARSE_METHODS:
- parseMethod(line);
- break;
- case PARSE_OPTIONS:
- break;
- }
- }
- }
-
- void parseThread(String line) {
- String idStr = null;
- String name = null;
- Matcher matcher = mIdNamePattern.matcher(line);
- if (matcher.find()) {
- idStr = matcher.group(1);
- name = matcher.group(2);
- }
- if (idStr == null) return;
- if (name == null) name = "(unknown)";
-
- int id = Integer.decode(idStr);
- mThreadMap.put(id, new ThreadData(id, name, mTopLevel));
- }
-
- void parseMethod(String line) {
- String[] tokens = line.split("\t");
- int id = Long.decode(tokens[0]).intValue();
- String className = tokens[1];
- String methodName = null;
- String signature = null;
- String pathname = null;
- int lineNumber = -1;
- if (tokens.length == 6) {
- methodName = tokens[2];
- signature = tokens[3];
- pathname = tokens[4];
- lineNumber = Integer.decode(tokens[5]);
- pathname = constructPathname(className, pathname);
- } else if (tokens.length > 2) {
- if (tokens[3].startsWith("(")) {
- methodName = tokens[2];
- signature = tokens[3];
- } else {
- pathname = tokens[2];
- lineNumber = Integer.decode(tokens[3]);
- }
- }
-
- mMethodMap.put(id, new MethodData(id, className, methodName, signature,
- pathname, lineNumber));
- }
-
- private String constructPathname(String className, String pathname) {
- int index = className.lastIndexOf('/');
- if (index > 0 && index < className.length() - 1
- && pathname.endsWith(".java"))
- pathname = className.substring(0, index + 1) + pathname;
- return pathname;
- }
-
- private void analyzeData() {
- // Sort the threads into decreasing cpu time
- Collection<ThreadData> tv = mThreadMap.values();
- mSortedThreads = tv.toArray(new ThreadData[tv.size()]);
- Arrays.sort(mSortedThreads, new Comparator<ThreadData>() {
- public int compare(ThreadData td1, ThreadData td2) {
- if (td2.getCpuTime() > td1.getCpuTime())
- return 1;
- if (td2.getCpuTime() < td1.getCpuTime())
- return -1;
- return td2.getName().compareTo(td1.getName());
- }
- });
-
- // Analyze the call tree so that we can label the "worst" children.
- // Also set all the root pointers in each node in the call tree.
- long sum = 0;
- for (ThreadData t : mSortedThreads) {
- if (t.isEmpty() == false) {
- Call root = t.getCalltreeRoot();
- root.mGlobalStartTime = t.getGlobalStartTime();
- }
- }
-
- // Sort the methods into decreasing inclusive time
- Collection<MethodData> mv = mMethodMap.values();
- MethodData[] methods;
- methods = mv.toArray(new MethodData[mv.size()]);
- Arrays.sort(methods, new Comparator<MethodData>() {
- public int compare(MethodData md1, MethodData md2) {
- if (md2.getElapsedInclusive() > md1.getElapsedInclusive())
- return 1;
- if (md2.getElapsedInclusive() < md1.getElapsedInclusive())
- return -1;
- return md1.getName().compareTo(md2.getName());
- }
- });
-
- // Count the number of methods with non-zero inclusive time
- int nonZero = 0;
- for (MethodData md : methods) {
- if (md.getElapsedInclusive() == 0)
- break;
- nonZero += 1;
- }
-
- // Copy the methods with non-zero time
- mSortedMethods = new MethodData[nonZero];
- int ii = 0;
- for (MethodData md : methods) {
- if (md.getElapsedInclusive() == 0)
- break;
- md.setRank(ii);
- mSortedMethods[ii++] = md;
- }
-
- // Let each method analyze its profile data
- for (MethodData md : mSortedMethods) {
- md.analyzeData();
- }
-
- // Update all the calls to include the method rank in
- // their name.
- for (Call call : mCallList) {
- call.updateName();
- }
-
- if (mRegression) {
- dumpMethodStats();
- }
- }
-
- /*
- * This method computes a list of records that describe the the execution
- * timeline for each thread. Each record is a pair: (row, block) where: row:
- * is the ThreadData object block: is the call (containing the start and end
- * times)
- */
- @Override
- public ArrayList<TimeLineView.Record> getThreadTimeRecords() {
- TimeLineView.Record record;
- ArrayList<TimeLineView.Record> timeRecs;
- timeRecs = new ArrayList<TimeLineView.Record>();
-
- // For each thread, push a "toplevel" call that encompasses the
- // entire execution of the thread.
- for (ThreadData threadData : mSortedThreads) {
- if (!threadData.isEmpty() && threadData.getId() != 0) {
- Call call = new Call(threadData.getId(), mTopLevel,
- threadData.getGlobalStartTime(), 0);
- call.mGlobalStartTime = threadData.getGlobalStartTime();
- call.mGlobalEndTime = threadData.getGlobalEndTime();
- record = new TimeLineView.Record(threadData, call);
- timeRecs.add(record);
- }
- }
-
- for (Call call : mCallList) {
- if (call.getMethodAction() != 0 || call.getThreadId() == 0)
- continue;
- ThreadData threadData = mThreadMap.get(call.getThreadId());
- record = new TimeLineView.Record(threadData, call);
- timeRecs.add(record);
- }
-
- if (mRegression) {
- dumpTimeRecs(timeRecs);
- System.exit(0);
- }
- return timeRecs;
- }
-
- private void dumpCallTimes() {
- String action;
-
- System.out.format("id thread global start,end method\n");
- for (Call call : mCallList) {
- if (call.getMethodAction() == 0) {
- action = "+";
- } else {
- action = " ";
- }
- long callTime = call.mThreadStartTime;
- System.out.format("%2d %6d %8d %8d %s %s\n",
- call.getThreadId(), callTime, call.mGlobalStartTime,
- call.mGlobalEndTime, action, call.getMethodData().getName());
-// if (call.getMethodAction() == 0 && call.getGlobalEndTime() < call.getGlobalStartTime()) {
-// System.out.printf("endtime %d < startTime %d\n",
-// call.getGlobalEndTime(), call.getGlobalStartTime());
-// }
- }
- }
-
- private void dumpMethodStats() {
- System.out.format("\nExclusive Inclusive Calls Method\n");
- for (MethodData md : mSortedMethods) {
- System.out.format("%9d %9d %9s %s\n",
- md.getElapsedExclusive(), md.getElapsedInclusive(),
- md.getCalls(), md.getProfileName());
- }
- }
-
- private void dumpTimeRecs(ArrayList<TimeLineView.Record> timeRecs) {
- System.out.format("\nid thread global start,end method\n");
- for (TimeLineView.Record record : timeRecs) {
- Call call = (Call) record.block;
- long callTime = call.mThreadStartTime;
- System.out.format("%2d %6d %8d %8d %s\n",
- call.getThreadId(), callTime,
- call.mGlobalStartTime, call.mGlobalEndTime,
- call.getMethodData().getName());
- }
- }
-
- @Override
- public HashMap<Integer, String> getThreadLabels() {
- HashMap<Integer, String> labels = new HashMap<Integer, String>();
- for (ThreadData t : mThreadMap.values()) {
- labels.put(t.getId(), t.getName());
- }
- return labels;
- }
-
- @Override
- public MethodData[] getMethods() {
- return mSortedMethods;
- }
-
- @Override
- public ThreadData[] getThreads() {
- return mSortedThreads;
- }
-
- @Override
- public long getEndTime() {
- return mGlobalEndTime;
- }
-}
diff --git a/traceview/src/com/android/traceview/MainWindow.java b/traceview/src/com/android/traceview/MainWindow.java
deleted file mode 100644
index b0c24e9..0000000
--- a/traceview/src/com/android/traceview/MainWindow.java
+++ /dev/null
@@ -1,192 +0,0 @@
-/*
- * Copyright (C) 2006 The Android Open Source Project
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
-
-package com.android.traceview;
-
-import com.android.sdkstats.SdkStatsService;
-
-import org.eclipse.jface.window.ApplicationWindow;
-import org.eclipse.swt.SWT;
-import org.eclipse.swt.custom.SashForm;
-import org.eclipse.swt.graphics.Color;
-import org.eclipse.swt.layout.GridData;
-import org.eclipse.swt.layout.GridLayout;
-import org.eclipse.swt.widgets.Composite;
-import org.eclipse.swt.widgets.Control;
-import org.eclipse.swt.widgets.Display;
-import org.eclipse.swt.widgets.Shell;
-
-import java.io.File;
-import java.io.IOException;
-import java.io.FileInputStream;
-import java.io.FileOutputStream;
-import java.nio.channels.FileChannel;
-import java.util.HashMap;
-
-public class MainWindow extends ApplicationWindow {
-
- private final static String PING_NAME = "Traceview";
- private final static String PING_VERSION = "1.0";
-
- private TraceReader mReader;
- private String mTraceName;
-
- // A global cache of string names.
- public static HashMap<String, String> sStringCache = new HashMap<String, String>();
-
- public MainWindow(String traceName, TraceReader reader) {
- super(null);
- mReader = reader;
- mTraceName = traceName;
- }
-
- public void run() {
- setBlockOnOpen(true);
- open();
- Display.getCurrent().dispose();
- }
-
- @Override
- protected void configureShell(Shell shell) {
- super.configureShell(shell);
- shell.setText("Traceview: " + mTraceName);
- shell.setBounds(100, 10, 1282, 900);
- }
-
- @Override
- protected Control createContents(Composite parent) {
- ColorController.assignMethodColors(parent.getDisplay(), mReader.getMethods());
- SelectionController selectionController = new SelectionController();
-
- GridLayout gridLayout = new GridLayout(1, false);
- gridLayout.marginWidth = 0;
- gridLayout.marginHeight = 0;
- gridLayout.horizontalSpacing = 0;
- gridLayout.verticalSpacing = 0;
- parent.setLayout(gridLayout);
-
- Display display = parent.getDisplay();
- Color darkGray = display.getSystemColor(SWT.COLOR_DARK_GRAY);
-
- // Create a sash form to separate the timeline view (on top)
- // and the profile view (on bottom)
- SashForm sashForm1 = new SashForm(parent, SWT.VERTICAL);
- sashForm1.setBackground(darkGray);
- sashForm1.SASH_WIDTH = 3;
- GridData data = new GridData(GridData.FILL_BOTH);
- sashForm1.setLayoutData(data);
-
- // Create the timeline view
- new TimeLineView(sashForm1, mReader, selectionController);
-
- // Create the profile view
- new ProfileView(sashForm1, mReader, selectionController);
- return sashForm1;
- }
-
- /**
- * Convert the old two-file format into the current concatenated one.
- *
- * @param base Base path of the two files, i.e. base.key and base.data
- * @return Path to a temporary file that will be deleted on exit.
- * @throws IOException
- */
- private static String makeTempTraceFile(String base) throws IOException {
- // Make a temporary file that will go away on exit and prepare to
- // write into it.
- File temp = File.createTempFile(base, ".trace");
- temp.deleteOnExit();
- FileChannel dstChannel = new FileOutputStream(temp).getChannel();
-
- // First copy the contents of the key file into our temp file.
- FileChannel srcChannel = new FileInputStream(base + ".key").getChannel();
- long size = dstChannel.transferFrom(srcChannel, 0, srcChannel.size());
- srcChannel.close();
-
- // Then concatenate the data file.
- srcChannel = new FileInputStream(base + ".data").getChannel();
- dstChannel.transferFrom(srcChannel, size, srcChannel.size());
-
- // Clean up.
- srcChannel.close();
- dstChannel.close();
-
- // Return the path of the temp file.
- return temp.getPath();
- }
-
- public static void main(String[] args) {
- TraceReader reader = null;
- boolean regression = false;
-
- // ping the usage server
- SdkStatsService.ping(PING_NAME, PING_VERSION);
-
- // Process command line arguments
- int argc = 0;
- int len = args.length;
- while (argc < len) {
- String arg = args[argc];
- if (arg.charAt(0) != '-') {
- break;
- }
- if (arg.equals("-r")) {
- regression = true;
- } else {
- break;
- }
- argc++;
- }
- if (argc != len - 1) {
- System.out.printf("Usage: java %s [-r] trace%n", MainWindow.class.getName());
- System.out.printf(" -r regression only%n");
- return;
- }
-
- String traceName = args[len - 1];
- File file = new File(traceName);
- if (file.exists() && file.isDirectory()) {
- System.out.printf("Qemu trace files not supported yet.\n");
- System.exit(1);
- // reader = new QtraceReader(traceName);
- } else {
- // If the filename as given doesn't exist...
- if (!file.exists()) {
- // Try appending .trace.
- if (new File(traceName + ".trace").exists()) {
- traceName = traceName + ".trace";
- // Next, see if it is the old two-file trace.
- } else if (new File(traceName + ".data").exists()
- && new File(traceName + ".key").exists()) {
- try {
- traceName = makeTempTraceFile(traceName);
- } catch (IOException e) {
- System.err.printf("cannot convert old trace file '%s'\n", traceName);
- System.exit(1);
- }
- // Otherwise, give up.
- } else {
- System.err.printf("trace file '%s' not found\n", traceName);
- System.exit(1);
- }
- }
-
- reader = new DmTraceReader(traceName, regression);
- }
- reader.getTraceUnits().setTimeScale(TraceUnits.TimeScale.MilliSeconds);
- new MainWindow(traceName, reader).run();
- }
-}
diff --git a/traceview/src/com/android/traceview/MethodData.java b/traceview/src/com/android/traceview/MethodData.java
deleted file mode 100644
index 0bc9853..0000000
--- a/traceview/src/com/android/traceview/MethodData.java
+++ /dev/null
@@ -1,458 +0,0 @@
-/*
- * Copyright (C) 2006 The Android Open Source Project
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
-
-package com.android.traceview;
-
-import org.eclipse.swt.graphics.Color;
-import org.eclipse.swt.graphics.Image;
-
-import java.util.ArrayList;
-import java.util.Arrays;
-import java.util.Collection;
-import java.util.Comparator;
-import java.util.HashMap;
-
-public class MethodData {
-
- private int mId;
- private int mRank = -1;
- private String mClassName;
- private String mMethodName;
- private String mSignature;
- private String mName;
- private String mProfileName;
- private String mPathname;
- private int mLineNumber;
- private long mElapsedExclusive;
- private long mElapsedInclusive;
- private long mTopExclusive;
- private int[] mNumCalls = new int[2]; // index 0=normal, 1=recursive
- private Color mColor;
- private Color mFadedColor;
- private Image mImage;
- private Image mFadedImage;
- private HashMap<Integer, ProfileData> mParents;
- private HashMap<Integer, ProfileData> mChildren;
-
- // The parents of this method when this method was in a recursive call
- private HashMap<Integer, ProfileData> mRecursiveParents;
-
- // The children of this method when this method was in a recursive call
- private HashMap<Integer, ProfileData> mRecursiveChildren;
-
- private ProfileNode[] mProfileNodes;
- private int mX;
- private int mY;
- private double mWeight;
-
- public MethodData(int id, String className) {
- mId = id;
- mClassName = className;
- mMethodName = null;
- mSignature = null;
- mPathname = null;
- mLineNumber = -1;
- computeName();
- computeProfileName();
- }
-
- public MethodData(int id, String className, String methodName,
- String signature, String pathname, int lineNumber) {
- mId = id;
- mClassName = className;
- mMethodName = methodName;
- mSignature = signature;
- mPathname = pathname;
- mLineNumber = lineNumber;
- computeName();
- computeProfileName();
- }
-
- private Comparator<ProfileData> mByElapsedInclusive = new Comparator<ProfileData>() {
- public int compare(ProfileData pd1, ProfileData pd2) {
- if (pd2.getElapsedInclusive() > pd1.getElapsedInclusive())
- return 1;
- if (pd2.getElapsedInclusive() < pd1.getElapsedInclusive())
- return -1;
- return 0;
- }
- };
-
- public double addWeight(int x, int y, double weight) {
- if (mX == x && mY == y)
- mWeight += weight;
- else {
- mX = x;
- mY = y;
- mWeight = weight;
- }
- return mWeight;
- }
-
- public void clearWeight() {
- mWeight = 0;
- }
-
- public int getRank() {
- return mRank;
- }
-
- public void setRank(int rank) {
- mRank = rank;
- computeProfileName();
- }
-
- public void addElapsedExclusive(long time) {
- mElapsedExclusive += time;
- }
-
- public void addElapsedInclusive(long time, boolean isRecursive, Call parent) {
- if (isRecursive == false) {
- mElapsedInclusive += time;
- mNumCalls[0] += 1;
- } else {
- mNumCalls[1] += 1;
- }
-
- if (parent == null)
- return;
-
- // Find the child method in the parent
- MethodData parentMethod = parent.mMethodData;
- if (parent.isRecursive()) {
- parentMethod.mRecursiveChildren = updateInclusive(time,
- parentMethod, this, false,
- parentMethod.mRecursiveChildren);
- } else {
- parentMethod.mChildren = updateInclusive(time,
- parentMethod, this, false, parentMethod.mChildren);
- }
-
- // Find the parent method in the child
- if (isRecursive) {
- mRecursiveParents = updateInclusive(time, this, parentMethod, true,
- mRecursiveParents);
- } else {
- mParents = updateInclusive(time, this, parentMethod, true,
- mParents);
- }
- }
-
- private HashMap<Integer, ProfileData> updateInclusive(long time,
- MethodData contextMethod, MethodData elementMethod,
- boolean elementIsParent, HashMap<Integer, ProfileData> map) {
- if (map == null) {
- map = new HashMap<Integer, ProfileData>(4);
- } else {
- ProfileData profileData = map.get(elementMethod.mId);
- if (profileData != null) {
- profileData.addElapsedInclusive(time);
- return map;
- }
- }
-
- ProfileData elementData = new ProfileData(contextMethod,
- elementMethod, elementIsParent);
- elementData.setElapsedInclusive(time);
- elementData.setNumCalls(1);
- map.put(elementMethod.mId, elementData);
- return map;
- }
-
- public void analyzeData() {
- // Sort the parents and children into decreasing inclusive time
- ProfileData[] sortedParents;
- ProfileData[] sortedChildren;
- ProfileData[] sortedRecursiveParents;
- ProfileData[] sortedRecursiveChildren;
-
- sortedParents = sortProfileData(mParents);
- sortedChildren = sortProfileData(mChildren);
- sortedRecursiveParents = sortProfileData(mRecursiveParents);
- sortedRecursiveChildren = sortProfileData(mRecursiveChildren);
-
- // Add "self" time to the top of the sorted children
- sortedChildren = addSelf(sortedChildren);
-
- // Create the ProfileNode objects that we need
- ArrayList<ProfileNode> nodes = new ArrayList<ProfileNode>();
- ProfileNode profileNode;
- if (mParents != null) {
- profileNode = new ProfileNode("Parents", this, sortedParents,
- true, false);
- nodes.add(profileNode);
- }
- if (mChildren != null) {
- profileNode = new ProfileNode("Children", this, sortedChildren,
- false, false);
- nodes.add(profileNode);
- }
- if (mRecursiveParents!= null) {
- profileNode = new ProfileNode("Parents while recursive", this,
- sortedRecursiveParents, true, true);
- nodes.add(profileNode);
- }
- if (mRecursiveChildren != null) {
- profileNode = new ProfileNode("Children while recursive", this,
- sortedRecursiveChildren, false, true);
- nodes.add(profileNode);
- }
- mProfileNodes = nodes.toArray(new ProfileNode[nodes.size()]);
- }
-
- // Create and return a ProfileData[] array that is a sorted copy
- // of the given HashMap values.
- private ProfileData[] sortProfileData(HashMap<Integer, ProfileData> map) {
- if (map == null)
- return null;
-
- // Convert the hash values to an array of ProfileData
- Collection<ProfileData> values = map.values();
- ProfileData[] sorted = values.toArray(new ProfileData[values.size()]);
-
- // Sort the array by elapsed inclusive time
- Arrays.sort(sorted, mByElapsedInclusive);
- return sorted;
- }
-
- private ProfileData[] addSelf(ProfileData[] children) {
- ProfileData[] pdata;
- if (children == null) {
- pdata = new ProfileData[1];
- } else {
- pdata = new ProfileData[children.length + 1];
- System.arraycopy(children, 0, pdata, 1, children.length);
- }
- pdata[0] = new ProfileSelf(this);
- return pdata;
- }
-
- public void addTopExclusive(long time) {
- mTopExclusive += time;
- }
-
- public long getTopExclusive() {
- return mTopExclusive;
- }
-
- public int getId() {
- return mId;
- }
-
- private void computeName() {
- if (mMethodName == null) {
- mName = mClassName;
- return;
- }
-
- StringBuilder sb = new StringBuilder();
- sb.append(mClassName);
- sb.append("."); //$NON-NLS-1$
- sb.append(mMethodName);
- sb.append(" "); //$NON-NLS-1$
- sb.append(mSignature);
- mName = sb.toString();
- }
-
- public String getName() {
- return mName;
- }
-
- public String getClassName() {
- return mClassName;
- }
-
- public String getMethodName() {
- return mMethodName;
- }
-
- public String getProfileName() {
- return mProfileName;
- }
-
- public void computeProfileName() {
- if (mRank == -1) {
- mProfileName = mName;
- return;
- }
-
- StringBuilder sb = new StringBuilder();
- sb.append(mRank);
- sb.append(" "); //$NON-NLS-1$
- sb.append(getName());
- mProfileName = sb.toString();
- }
-
- public String getCalls() {
- return String.format("%d+%d", mNumCalls[0], mNumCalls[1]);
- }
-
- public int getTotalCalls() {
- return mNumCalls[0] + mNumCalls[1];
- }
-
- public Color getColor() {
- return mColor;
- }
-
- public void setColor(Color color) {
- mColor = color;
- }
-
- public void setImage(Image image) {
- mImage = image;
- }
-
- public Image getImage() {
- return mImage;
- }
-
- @Override
- public String toString() {
- return getName();
- }
-
- public long getElapsedExclusive() {
- return mElapsedExclusive;
- }
-
- public long getElapsedInclusive() {
- return mElapsedInclusive;
- }
-
- public void setFadedColor(Color fadedColor) {
- mFadedColor = fadedColor;
- }
-
- public Color getFadedColor() {
- return mFadedColor;
- }
-
- public void setFadedImage(Image fadedImage) {
- mFadedImage = fadedImage;
- }
-
- public Image getFadedImage() {
- return mFadedImage;
- }
-
- public void setPathname(String pathname) {
- mPathname = pathname;
- }
-
- public String getPathname() {
- return mPathname;
- }
-
- public void setLineNumber(int lineNumber) {
- mLineNumber = lineNumber;
- }
-
- public int getLineNumber() {
- return mLineNumber;
- }
-
- public ProfileNode[] getProfileNodes() {
- return mProfileNodes;
- }
-
- public static class Sorter implements Comparator<MethodData> {
- public int compare(MethodData md1, MethodData md2) {
- if (mColumn == Column.BY_NAME) {
- int result = md1.getName().compareTo(md2.getName());
- return (mDirection == Direction.INCREASING) ? result : -result;
- }
- if (mColumn == Column.BY_INCLUSIVE) {
- if (md2.getElapsedInclusive() > md1.getElapsedInclusive())
- return (mDirection == Direction.INCREASING) ? -1 : 1;
- if (md2.getElapsedInclusive() < md1.getElapsedInclusive())
- return (mDirection == Direction.INCREASING) ? 1 : -1;
- return md1.getName().compareTo(md2.getName());
- }
- if (mColumn == Column.BY_EXCLUSIVE) {
- if (md2.getElapsedExclusive() > md1.getElapsedExclusive())
- return (mDirection == Direction.INCREASING) ? -1 : 1;
- if (md2.getElapsedExclusive() < md1.getElapsedExclusive())
- return (mDirection == Direction.INCREASING) ? 1 : -1;
- return md1.getName().compareTo(md2.getName());
- }
- if (mColumn == Column.BY_CALLS) {
- int result = md1.getTotalCalls() - md2.getTotalCalls();
- if (result == 0)
- return md1.getName().compareTo(md2.getName());
- return (mDirection == Direction.INCREASING) ? result : -result;
- }
- if (mColumn == Column.BY_TIME_PER_CALL) {
- double time1 = md1.getElapsedInclusive();
- time1 = time1 / md1.getTotalCalls();
- double time2 = md2.getElapsedInclusive();
- time2 = time2 / md2.getTotalCalls();
- double diff = time1 - time2;
- int result = 0;
- if (diff < 0)
- result = -1;
- else if (diff > 0)
- result = 1;
- if (result == 0)
- return md1.getName().compareTo(md2.getName());
- return (mDirection == Direction.INCREASING) ? result : -result;
- }
- return 0;
- }
-
- public void setColumn(Column column) {
- // If the sort column specified is the same as last time,
- // then reverse the sort order.
- if (mColumn == column) {
- // Reverse the sort order
- if (mDirection == Direction.INCREASING)
- mDirection = Direction.DECREASING;
- else
- mDirection = Direction.INCREASING;
- } else {
- // Sort names into increasing order, data into decreasing order.
- if (column == Column.BY_NAME)
- mDirection = Direction.INCREASING;
- else
- mDirection = Direction.DECREASING;
- }
- mColumn = column;
- }
-
- public Column getColumn() {
- return mColumn;
- }
-
- public void setDirection(Direction direction) {
- mDirection = direction;
- }
-
- public Direction getDirection() {
- return mDirection;
- }
-
- public static enum Column {
- BY_NAME, BY_EXCLUSIVE, BY_INCLUSIVE, BY_CALLS, BY_TIME_PER_CALL
- };
-
- public static enum Direction {
- INCREASING, DECREASING
- };
-
- private Column mColumn;
- private Direction mDirection;
- }
-}
diff --git a/traceview/src/com/android/traceview/ProfileData.java b/traceview/src/com/android/traceview/ProfileData.java
deleted file mode 100644
index f0c1d61..0000000
--- a/traceview/src/com/android/traceview/ProfileData.java
+++ /dev/null
@@ -1,81 +0,0 @@
-/*
- * Copyright (C) 2006 The Android Open Source Project
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
-
-package com.android.traceview;
-
-
-public class ProfileData {
-
- protected MethodData mElement;
-
- /** mContext is either the parent or child of mElement */
- protected MethodData mContext;
- protected boolean mElementIsParent;
- protected long mElapsedInclusive;
- protected int mNumCalls;
-
- public ProfileData() {
- }
-
- public ProfileData(MethodData context, MethodData element,
- boolean elementIsParent) {
- mContext = context;
- mElement = element;
- mElementIsParent = elementIsParent;
- }
-
- public String getProfileName() {
- return mElement.getProfileName();
- }
-
- public MethodData getMethodData() {
- return mElement;
- }
-
- public void addElapsedInclusive(long elapsedInclusive) {
- mElapsedInclusive += elapsedInclusive;
- mNumCalls += 1;
- }
-
- public void setElapsedInclusive(long elapsedInclusive) {
- mElapsedInclusive = elapsedInclusive;
- }
-
- public long getElapsedInclusive() {
- return mElapsedInclusive;
- }
-
- public void setNumCalls(int numCalls) {
- mNumCalls = numCalls;
- }
-
- public String getNumCalls() {
- int totalCalls;
- if (mElementIsParent)
- totalCalls = mContext.getTotalCalls();
- else
- totalCalls = mElement.getTotalCalls();
- return String.format("%d/%d", mNumCalls, totalCalls);
- }
-
- public boolean isParent() {
- return mElementIsParent;
- }
-
- public MethodData getContext() {
- return mContext;
- }
-}
diff --git a/traceview/src/com/android/traceview/ProfileNode.java b/traceview/src/com/android/traceview/ProfileNode.java
deleted file mode 100644
index 7cb0b5d..0000000
--- a/traceview/src/com/android/traceview/ProfileNode.java
+++ /dev/null
@@ -1,51 +0,0 @@
-/*
- * Copyright (C) 2006 The Android Open Source Project
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
-
-package com.android.traceview;
-
-public class ProfileNode {
-
- private String mLabel;
- private MethodData mMethodData;
- private ProfileData[] mChildren;
- private boolean mIsParent;
- private boolean mIsRecursive;
-
- public ProfileNode(String label, MethodData methodData,
- ProfileData[] children, boolean isParent, boolean isRecursive) {
- mLabel = label;
- mMethodData = methodData;
- mChildren = children;
- mIsParent = isParent;
- mIsRecursive = isRecursive;
- }
-
- public String getLabel() {
- return mLabel;
- }
-
- public ProfileData[] getChildren() {
- return mChildren;
- }
-
- public boolean isParent() {
- return mIsParent;
- }
-
- public boolean isRecursive() {
- return mIsRecursive;
- }
-}
diff --git a/traceview/src/com/android/traceview/ProfileProvider.java b/traceview/src/com/android/traceview/ProfileProvider.java
deleted file mode 100644
index fe5c832..0000000
--- a/traceview/src/com/android/traceview/ProfileProvider.java
+++ /dev/null
@@ -1,361 +0,0 @@
-/*
- * Copyright (C) 2006 The Android Open Source Project
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
-
-package com.android.traceview;
-
-import java.io.InputStream;
-import java.util.Arrays;
-import java.util.regex.Pattern;
-
-import org.eclipse.jface.viewers.IColorProvider;
-import org.eclipse.jface.viewers.ITableLabelProvider;
-import org.eclipse.jface.viewers.ITreeContentProvider;
-import org.eclipse.jface.viewers.LabelProvider;
-import org.eclipse.jface.viewers.TreeViewer;
-import org.eclipse.jface.viewers.Viewer;
-import org.eclipse.swt.SWT;
-import org.eclipse.swt.events.SelectionAdapter;
-import org.eclipse.swt.events.SelectionEvent;
-import org.eclipse.swt.graphics.Color;
-import org.eclipse.swt.graphics.Image;
-import org.eclipse.swt.widgets.Display;
-import org.eclipse.swt.widgets.Tree;
-import org.eclipse.swt.widgets.TreeColumn;
-import org.eclipse.swt.widgets.TreeItem;
-
-class ProfileProvider implements ITreeContentProvider {
-
- private MethodData[] mRoots;
- private SelectionAdapter mListener;
- private TreeViewer mTreeViewer;
- private TraceReader mReader;
- private Image mSortUp;
- private Image mSortDown;
- private String mColumnNames[] = { "Name", "Incl %", "Inclusive", "Excl %",
- "Exclusive", "Calls+Recur\nCalls/Total", "Time/Call" };
- private int mColumnWidths[] = { 370, 70, 70, 70, 70, 90, 70 };
- private int mColumnAlignments[] = { SWT.LEFT, SWT.RIGHT, SWT.RIGHT,
- SWT.RIGHT, SWT.RIGHT, SWT.CENTER, SWT.RIGHT };
- private static final int COL_NAME = 0;
- private static final int COL_INCLUSIVE_PER = 1;
- private static final int COL_INCLUSIVE = 2;
- private static final int COL_EXCLUSIVE_PER = 3;
- private static final int COL_EXCLUSIVE = 4;
- private static final int COL_CALLS = 5;
- private static final int COL_TIME_PER_CALL = 6;
- private long mTotalTime;
- private Pattern mUppercase;
- private int mPrevMatchIndex = -1;
-
- public ProfileProvider(TraceReader reader) {
- mRoots = reader.getMethods();
- mReader = reader;
- mTotalTime = reader.getEndTime();
- Display display = Display.getCurrent();
- InputStream in = getClass().getClassLoader().getResourceAsStream(
- "icons/sort_up.png");
- mSortUp = new Image(display, in);
- in = getClass().getClassLoader().getResourceAsStream(
- "icons/sort_down.png");
- mSortDown = new Image(display, in);
- mUppercase = Pattern.compile("[A-Z]");
- }
-
- private MethodData doMatchName(String name, int startIndex) {
- // Check if the given "name" has any uppercase letters
- boolean hasUpper = mUppercase.matcher(name).matches();
- for (int ii = startIndex; ii < mRoots.length; ++ii) {
- MethodData md = mRoots[ii];
- String fullName = md.getName();
- // If there were no upper case letters in the given name,
- // then ignore case when matching.
- if (!hasUpper)
- fullName = fullName.toLowerCase();
- if (fullName.indexOf(name) != -1) {
- mPrevMatchIndex = ii;
- return md;
- }
- }
- mPrevMatchIndex = -1;
- return null;
- }
-
- public MethodData findMatchingName(String name) {
- return doMatchName(name, 0);
- }
-
- public MethodData findNextMatchingName(String name) {
- return doMatchName(name, mPrevMatchIndex + 1);
- }
-
- public MethodData findMatchingTreeItem(TreeItem item) {
- if (item == null)
- return null;
- String text = item.getText();
- if (Character.isDigit(text.charAt(0)) == false)
- return null;
- int spaceIndex = text.indexOf(' ');
- String numstr = text.substring(0, spaceIndex);
- int rank = Integer.valueOf(numstr);
- for (MethodData md : mRoots) {
- if (md.getRank() == rank)
- return md;
- }
- return null;
- }
-
- public void setTreeViewer(TreeViewer treeViewer) {
- mTreeViewer = treeViewer;
- }
-
- public String[] getColumnNames() {
- return mColumnNames;
- }
-
- public int[] getColumnWidths() {
- return mColumnWidths;
- }
-
- public int[] getColumnAlignments() {
- return mColumnAlignments;
- }
-
- public Object[] getChildren(Object element) {
- if (element instanceof MethodData) {
- MethodData md = (MethodData) element;
- return md.getProfileNodes();
- }
- if (element instanceof ProfileNode) {
- ProfileNode pn = (ProfileNode) element;
- return pn.getChildren();
- }
- return new Object[0];
- }
-
- public Object getParent(Object element) {
- return null;
- }
-
- public boolean hasChildren(Object element) {
- if (element instanceof MethodData)
- return true;
- if (element instanceof ProfileNode)
- return true;
- return false;
- }
-
- public Object[] getElements(Object element) {
- return mRoots;
- }
-
- public void dispose() {
- }
-
- public void inputChanged(Viewer arg0, Object arg1, Object arg2) {
- }
-
- public Object getRoot() {
- return "root";
- }
-
- public SelectionAdapter getColumnListener() {
- if (mListener == null)
- mListener = new ColumnListener();
- return mListener;
- }
-
- public LabelProvider getLabelProvider() {
- return new ProfileLabelProvider();
- }
-
- class ProfileLabelProvider extends LabelProvider implements
- ITableLabelProvider, IColorProvider {
- Color colorRed;
- Color colorParentsBack;
- Color colorChildrenBack;
- TraceUnits traceUnits;
-
- public ProfileLabelProvider() {
- Display display = Display.getCurrent();
- colorRed = display.getSystemColor(SWT.COLOR_RED);
- colorParentsBack = new Color(display, 230, 230, 255); // blue
- colorChildrenBack = new Color(display, 255, 255, 210); // yellow
- traceUnits = mReader.getTraceUnits();
- }
-
- public String getColumnText(Object element, int col) {
- if (element instanceof MethodData) {
- MethodData md = (MethodData) element;
- if (col == COL_NAME)
- return md.getProfileName();
- if (col == COL_EXCLUSIVE) {
- double val = md.getElapsedExclusive();
- val = traceUnits.getScaledValue(val);
- return String.format("%.3f", val);
- }
- if (col == COL_EXCLUSIVE_PER) {
- double val = md.getElapsedExclusive();
- double per = val * 100.0 / mTotalTime;
- return String.format("%.1f%%", per);
- }
- if (col == COL_INCLUSIVE) {
- double val = md.getElapsedInclusive();
- val = traceUnits.getScaledValue(val);
- return String.format("%.3f", val);
- }
- if (col == COL_INCLUSIVE_PER) {
- double val = md.getElapsedInclusive();
- double per = val * 100.0 / mTotalTime;
- return String.format("%.1f%%", per);
- }
- if (col == COL_CALLS)
- return md.getCalls();
- if (col == COL_TIME_PER_CALL) {
- int numCalls = md.getTotalCalls();
- double val = md.getElapsedInclusive();
- val = val / numCalls;
- val = traceUnits.getScaledValue(val);
- return String.format("%.3f", val);
- }
- } else if (element instanceof ProfileSelf) {
- ProfileSelf ps = (ProfileSelf) element;
- if (col == COL_NAME)
- return ps.getProfileName();
- if (col == COL_INCLUSIVE) {
- double val = ps.getElapsedInclusive();
- val = traceUnits.getScaledValue(val);
- return String.format("%.3f", val);
- }
- if (col == COL_INCLUSIVE_PER) {
- double total;
- double val = ps.getElapsedInclusive();
- MethodData context = ps.getContext();
- total = context.getElapsedInclusive();
- double per = val * 100.0 / total;
- return String.format("%.1f%%", per);
- }
- return "";
- } else if (element instanceof ProfileData) {
- ProfileData pd = (ProfileData) element;
- if (col == COL_NAME)
- return pd.getProfileName();
- if (col == COL_INCLUSIVE) {
- double val = pd.getElapsedInclusive();
- val = traceUnits.getScaledValue(val);
- return String.format("%.3f", val);
- }
- if (col == COL_INCLUSIVE_PER) {
- double total;
- double val = pd.getElapsedInclusive();
- MethodData context = pd.getContext();
- total = context.getElapsedInclusive();
- double per = val * 100.0 / total;
- return String.format("%.1f%%", per);
- }
- if (col == COL_CALLS)
- return pd.getNumCalls();
- return "";
- } else if (element instanceof ProfileNode) {
- ProfileNode pn = (ProfileNode) element;
- if (col == COL_NAME)
- return pn.getLabel();
- return "";
- }
- return "col" + col;
- }
-
- public Image getColumnImage(Object element, int col) {
- if (col != COL_NAME)
- return null;
- if (element instanceof MethodData) {
- MethodData md = (MethodData) element;
- return md.getImage();
- }
- if (element instanceof ProfileData) {
- ProfileData pd = (ProfileData) element;
- MethodData md = pd.getMethodData();
- return md.getImage();
- }
- return null;
- }
-
- public Color getForeground(Object element) {
- return null;
- }
-
- public Color getBackground(Object element) {
- if (element instanceof ProfileData) {
- ProfileData pd = (ProfileData) element;
- if (pd.isParent())
- return colorParentsBack;
- return colorChildrenBack;
- }
- if (element instanceof ProfileNode) {
- ProfileNode pn = (ProfileNode) element;
- if (pn.isParent())
- return colorParentsBack;
- return colorChildrenBack;
- }
- return null;
- }
- }
-
- class ColumnListener extends SelectionAdapter {
- MethodData.Sorter sorter = new MethodData.Sorter();
-
- @Override
- public void widgetSelected(SelectionEvent event) {
- TreeColumn column = (TreeColumn) event.widget;
- String name = column.getText();
- Tree tree = column.getParent();
- tree.setRedraw(false);
- TreeColumn[] columns = tree.getColumns();
- for (TreeColumn col : columns) {
- col.setImage(null);
- }
- if (name == mColumnNames[COL_NAME]) {
- // Sort names alphabetically
- sorter.setColumn(MethodData.Sorter.Column.BY_NAME);
- Arrays.sort(mRoots, sorter);
- } else if (name == mColumnNames[COL_EXCLUSIVE]) {
- sorter.setColumn(MethodData.Sorter.Column.BY_EXCLUSIVE);
- Arrays.sort(mRoots, sorter);
- } else if (name == mColumnNames[COL_EXCLUSIVE_PER]) {
- sorter.setColumn(MethodData.Sorter.Column.BY_EXCLUSIVE);
- Arrays.sort(mRoots, sorter);
- } else if (name == mColumnNames[COL_INCLUSIVE]) {
- sorter.setColumn(MethodData.Sorter.Column.BY_INCLUSIVE);
- Arrays.sort(mRoots, sorter);
- } else if (name == mColumnNames[COL_INCLUSIVE_PER]) {
- sorter.setColumn(MethodData.Sorter.Column.BY_INCLUSIVE);
- Arrays.sort(mRoots, sorter);
- } else if (name == mColumnNames[COL_CALLS]) {
- sorter.setColumn(MethodData.Sorter.Column.BY_CALLS);
- Arrays.sort(mRoots, sorter);
- } else if (name == mColumnNames[COL_TIME_PER_CALL]) {
- sorter.setColumn(MethodData.Sorter.Column.BY_TIME_PER_CALL);
- Arrays.sort(mRoots, sorter);
- }
- MethodData.Sorter.Direction direction = sorter.getDirection();
- if (direction == MethodData.Sorter.Direction.INCREASING)
- column.setImage(mSortDown);
- else
- column.setImage(mSortUp);
- tree.setRedraw(true);
- mTreeViewer.refresh();
- }
- }
-}
diff --git a/traceview/src/com/android/traceview/ProfileSelf.java b/traceview/src/com/android/traceview/ProfileSelf.java
deleted file mode 100644
index 3a4f3d9..0000000
--- a/traceview/src/com/android/traceview/ProfileSelf.java
+++ /dev/null
@@ -1,34 +0,0 @@
-/*
- * Copyright (C) 2006 The Android Open Source Project
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
-
-package com.android.traceview;
-
-public class ProfileSelf extends ProfileData {
- public ProfileSelf(MethodData methodData) {
- mElement = methodData;
- mContext = methodData;
- }
-
- @Override
- public String getProfileName() {
- return "self";
- }
-
- @Override
- public long getElapsedInclusive() {
- return mElement.getTopExclusive();
- }
-}
diff --git a/traceview/src/com/android/traceview/ProfileView.java b/traceview/src/com/android/traceview/ProfileView.java
deleted file mode 100644
index e48cb56..0000000
--- a/traceview/src/com/android/traceview/ProfileView.java
+++ /dev/null
@@ -1,308 +0,0 @@
-/*
- * Copyright (C) 2006 The Android Open Source Project
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
-
-package com.android.traceview;
-
-import java.util.ArrayList;
-import java.util.Observable;
-import java.util.Observer;
-
-import org.eclipse.jface.viewers.ISelection;
-import org.eclipse.jface.viewers.ISelectionChangedListener;
-import org.eclipse.jface.viewers.IStructuredSelection;
-import org.eclipse.jface.viewers.ITreeViewerListener;
-import org.eclipse.jface.viewers.SelectionChangedEvent;
-import org.eclipse.jface.viewers.StructuredSelection;
-import org.eclipse.jface.viewers.TreeExpansionEvent;
-import org.eclipse.jface.viewers.TreeViewer;
-import org.eclipse.swt.SWT;
-import org.eclipse.swt.events.KeyAdapter;
-import org.eclipse.swt.events.KeyEvent;
-import org.eclipse.swt.events.ModifyEvent;
-import org.eclipse.swt.events.ModifyListener;
-import org.eclipse.swt.events.SelectionAdapter;
-import org.eclipse.swt.graphics.Color;
-import org.eclipse.swt.graphics.Point;
-import org.eclipse.swt.layout.GridData;
-import org.eclipse.swt.layout.GridLayout;
-import org.eclipse.swt.widgets.Composite;
-import org.eclipse.swt.widgets.Display;
-import org.eclipse.swt.widgets.Event;
-import org.eclipse.swt.widgets.Label;
-import org.eclipse.swt.widgets.Listener;
-import org.eclipse.swt.widgets.Text;
-import org.eclipse.swt.widgets.Tree;
-import org.eclipse.swt.widgets.TreeColumn;
-import org.eclipse.swt.widgets.TreeItem;
-
-public class ProfileView extends Composite implements Observer {
-
- private TreeViewer mTreeViewer;
- private Text mSearchBox;
- private SelectionController mSelectionController;
- private ProfileProvider mProfileProvider;
- private Color mColorNoMatch;
- private Color mColorMatch;
- private MethodData mCurrentHighlightedMethod;
-
- public ProfileView(Composite parent, TraceReader reader,
- SelectionController selectController) {
- super(parent, SWT.NONE);
- setLayout(new GridLayout(1, false));
- this.mSelectionController = selectController;
- mSelectionController.addObserver(this);
-
- // Add a tree viewer at the top
- mTreeViewer = new TreeViewer(this, SWT.MULTI | SWT.NONE);
- mTreeViewer.setUseHashlookup(true);
- mProfileProvider = reader.getProfileProvider();
- mProfileProvider.setTreeViewer(mTreeViewer);
- SelectionAdapter listener = mProfileProvider.getColumnListener();
- final Tree tree = mTreeViewer.getTree();
- tree.setHeaderVisible(true);
- tree.setLayoutData(new GridData(GridData.FILL_BOTH));
-
- // Get the column names from the ProfileProvider
- String[] columnNames = mProfileProvider.getColumnNames();
- int[] columnWidths = mProfileProvider.getColumnWidths();
- int[] columnAlignments = mProfileProvider.getColumnAlignments();
- for (int ii = 0; ii < columnWidths.length; ++ii) {
- TreeColumn column = new TreeColumn(tree, SWT.LEFT);
- column.setText(columnNames[ii]);
- column.setWidth(columnWidths[ii]);
- column.setMoveable(true);
- column.addSelectionListener(listener);
- column.setAlignment(columnAlignments[ii]);
- }
-
- // Add a listener to the tree so that we can make the row
- // height smaller.
- tree.addListener(SWT.MeasureItem, new Listener() {
- public void handleEvent(Event event) {
- int fontHeight = event.gc.getFontMetrics().getHeight();
- event.height = fontHeight;
- }
- });
-
- mTreeViewer.setContentProvider(mProfileProvider);
- mTreeViewer.setLabelProvider(mProfileProvider.getLabelProvider());
- mTreeViewer.setInput(mProfileProvider.getRoot());
-
- // Create another composite to hold the label and text box
- Composite composite = new Composite(this, SWT.NONE);
- composite.setLayout(new GridLayout(2, false));
- composite.setLayoutData(new GridData(GridData.FILL_HORIZONTAL));
-
- // Add a label for the search box
- Label label = new Label(composite, SWT.NONE);
- label.setText("Find:");
-
- // Add a text box for searching for method names
- mSearchBox = new Text(composite, SWT.BORDER);
- mSearchBox.setLayoutData(new GridData(GridData.FILL_HORIZONTAL));
-
- Display display = getDisplay();
- mColorNoMatch = new Color(display, 255, 200, 200);
- mColorMatch = mSearchBox.getBackground();
-
- mSearchBox.addModifyListener(new ModifyListener() {
- public void modifyText(ModifyEvent ev) {
- String query = mSearchBox.getText();
- if (query.length() == 0)
- return;
- findName(query);
- }
- });
-
- // Add a key listener to the text box so that we can clear
- // the text box if the user presses <ESC>.
- mSearchBox.addKeyListener(new KeyAdapter() {
- @Override
- public void keyPressed(KeyEvent event) {
- if (event.keyCode == SWT.ESC) {
- mSearchBox.setText("");
- } else if (event.keyCode == SWT.CR) {
- String query = mSearchBox.getText();
- if (query.length() == 0)
- return;
- findNextName(query);
- }
- }
- });
-
- // Also add a key listener to the tree viewer so that the
- // user can just start typing anywhere in the tree view.
- tree.addKeyListener(new KeyAdapter() {
- @Override
- public void keyPressed(KeyEvent event) {
- if (event.keyCode == SWT.ESC) {
- mSearchBox.setText("");
- } else if (event.keyCode == SWT.BS) {
- // Erase the last character from the search box
- String text = mSearchBox.getText();
- int len = text.length();
- String chopped;
- if (len <= 1)
- chopped = "";
- else
- chopped = text.substring(0, len - 1);
- mSearchBox.setText(chopped);
- } else if (event.keyCode == SWT.CR) {
- String query = mSearchBox.getText();
- if (query.length() == 0)
- return;
- findNextName(query);
- } else {
- // Append the given character to the search box
- String str = String.valueOf(event.character);
- mSearchBox.append(str);
- }
- event.doit = false;
- }
- });
-
- // Add a selection listener to the tree so that the user can click
- // on a method that is a child or parent and jump to that method.
- mTreeViewer
- .addSelectionChangedListener(new ISelectionChangedListener() {
- public void selectionChanged(SelectionChangedEvent ev) {
- ISelection sel = ev.getSelection();
- if (sel.isEmpty())
- return;
- if (sel instanceof IStructuredSelection) {
- IStructuredSelection selection = (IStructuredSelection) sel;
- Object element = selection.getFirstElement();
- if (element == null)
- return;
- if (element instanceof MethodData) {
- MethodData md = (MethodData) element;
- highlightMethod(md, true);
- }
- if (element instanceof ProfileData) {
- MethodData md = ((ProfileData) element)
- .getMethodData();
- highlightMethod(md, true);
- }
- }
- }
- });
-
- // Add a tree listener so that we can expand the parents and children
- // of a method when a method is expanded.
- mTreeViewer.addTreeListener(new ITreeViewerListener() {
- public void treeExpanded(TreeExpansionEvent event) {
- Object element = event.getElement();
- if (element instanceof MethodData) {
- MethodData md = (MethodData) element;
- expandNode(md);
- }
- }
- public void treeCollapsed(TreeExpansionEvent event) {
- }
- });
-
- tree.addListener(SWT.MouseDown, new Listener() {
- public void handleEvent(Event event) {
- Point point = new Point(event.x, event.y);
- TreeItem treeItem = tree.getItem(point);
- MethodData md = mProfileProvider.findMatchingTreeItem(treeItem);
- if (md == null)
- return;
- ArrayList<Selection> selections = new ArrayList<Selection>();
- selections.add(Selection.highlight("MethodData", md));
- mSelectionController.change(selections, "ProfileView");
- }
- });
- }
-
- private void findName(String query) {
- MethodData md = mProfileProvider.findMatchingName(query);
- selectMethod(md);
- }
-
- private void findNextName(String query) {
- MethodData md = mProfileProvider.findNextMatchingName(query);
- selectMethod(md);
- }
-
- private void selectMethod(MethodData md) {
- if (md == null) {
- mSearchBox.setBackground(mColorNoMatch);
- return;
- }
- mSearchBox.setBackground(mColorMatch);
- highlightMethod(md, false);
- }
-
- public void update(Observable objservable, Object arg) {
- // Ignore updates from myself
- if (arg == "ProfileView")
- return;
- // System.out.printf("profileview update from %s\n", arg);
- ArrayList<Selection> selections;
- selections = mSelectionController.getSelections();
- for (Selection selection : selections) {
- Selection.Action action = selection.getAction();
- if (action != Selection.Action.Highlight)
- continue;
- String name = selection.getName();
- if (name == "MethodData") {
- MethodData md = (MethodData) selection.getValue();
- highlightMethod(md, true);
- return;
- }
- if (name == "Call") {
- Call call = (Call) selection.getValue();
- MethodData md = call.mMethodData;
- highlightMethod(md, true);
- return;
- }
- }
- }
-
- private void highlightMethod(MethodData md, boolean clearSearch) {
- if (md == null)
- return;
- // Avoid an infinite recursion
- if (md == mCurrentHighlightedMethod)
- return;
- if (clearSearch) {
- mSearchBox.setText("");
- mSearchBox.setBackground(mColorMatch);
- }
- mCurrentHighlightedMethod = md;
- mTreeViewer.collapseAll();
- // Expand this node and its children
- expandNode(md);
- StructuredSelection sel = new StructuredSelection(md);
- mTreeViewer.setSelection(sel, true);
- Tree tree = mTreeViewer.getTree();
- TreeItem[] items = tree.getSelection();
- tree.setTopItem(items[0]);
- // workaround a Mac bug by adding showItem().
- tree.showItem(items[0]);
- }
-
- private void expandNode(MethodData md) {
- ProfileNode[] nodes = md.getProfileNodes();
- mTreeViewer.setExpandedState(md, true);
- // Also expand the "Parents" and "Children" nodes.
- for (ProfileNode node : nodes) {
- if (node.isRecursive() == false)
- mTreeViewer.setExpandedState(node, true);
- }
- }
-}
diff --git a/traceview/src/com/android/traceview/QtraceReader.java b/traceview/src/com/android/traceview/QtraceReader.java
deleted file mode 100644
index c4db4a2..0000000
--- a/traceview/src/com/android/traceview/QtraceReader.java
+++ /dev/null
@@ -1,45 +0,0 @@
-/*
- * Copyright (C) 2006 The Android Open Source Project
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
-
-package com.android.traceview;
-
-import java.util.ArrayList;
-import java.util.HashMap;
-
-public class QtraceReader extends TraceReader {
- QtraceReader(String traceName) {
- }
-
- @Override
- public MethodData[] getMethods() {
- return null;
- }
-
- @Override
- public HashMap<Integer, String> getThreadLabels() {
- return null;
- }
-
- @Override
- public ArrayList<TimeLineView.Record> getThreadTimeRecords() {
- return null;
- }
-
- @Override
- public ProfileProvider getProfileProvider() {
- return null;
- }
-}
diff --git a/traceview/src/com/android/traceview/Selection.java b/traceview/src/com/android/traceview/Selection.java
deleted file mode 100644
index 3764619..0000000
--- a/traceview/src/com/android/traceview/Selection.java
+++ /dev/null
@@ -1,70 +0,0 @@
-/*
- * Copyright (C) 2006 The Android Open Source Project
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
-
-package com.android.traceview;
-
-public class Selection {
-
- private Action mAction;
- private String mName;
- private Object mValue;
-
- public Selection(Action action, String name, Object value) {
- mAction = action;
- mName = name;
- mValue = value;
- }
-
- public static Selection highlight(String name, Object value) {
- return new Selection(Action.Highlight, name, value);
- }
-
- public static Selection include(String name, Object value) {
- return new Selection(Action.Include, name, value);
- }
-
- public static Selection exclude(String name, Object value) {
- return new Selection(Action.Exclude, name, value);
- }
-
- public void setName(String name) {
- mName = name;
- }
-
- public String getName() {
- return mName;
- }
-
- public void setValue(Object value) {
- mValue = value;
- }
-
- public Object getValue() {
- return mValue;
- }
-
- public void setAction(Action action) {
- mAction = action;
- }
-
- public Action getAction() {
- return mAction;
- }
-
- public static enum Action {
- Highlight, Include, Exclude, Aggregate
- };
-}
diff --git a/traceview/src/com/android/traceview/SelectionController.java b/traceview/src/com/android/traceview/SelectionController.java
deleted file mode 100644
index 4c930ea..0000000
--- a/traceview/src/com/android/traceview/SelectionController.java
+++ /dev/null
@@ -1,35 +0,0 @@
-/*
- * Copyright (C) 2006 The Android Open Source Project
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
-
-package com.android.traceview;
-
-import java.util.ArrayList;
-import java.util.Observable;
-
-public class SelectionController extends Observable {
-
- private ArrayList<Selection> mSelections;
-
- public void change(ArrayList<Selection> selections, Object arg) {
- this.mSelections = selections;
- setChanged();
- notifyObservers(arg);
- }
-
- public ArrayList<Selection> getSelections() {
- return mSelections;
- }
-}
diff --git a/traceview/src/com/android/traceview/ThreadData.java b/traceview/src/com/android/traceview/ThreadData.java
deleted file mode 100644
index 54ea891..0000000
--- a/traceview/src/com/android/traceview/ThreadData.java
+++ /dev/null
@@ -1,228 +0,0 @@
-/*
- * Copyright (C) 2006 The Android Open Source Project
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
-
-package com.android.traceview;
-
-import java.util.ArrayList;
-import java.util.HashMap;
-
-class ThreadData implements TimeLineView.Row {
-
- private int mId;
- private String mName;
- private long mGlobalStartTime = -1;
- private long mGlobalEndTime = -1;
- private long mLastEventTime;
- private long mCpuTime;
- private Call mRoot;
- private Call mCurrent;
- private Call mLastContextSwitch;
- private ArrayList<Call> mStack = new ArrayList<Call>();
-
- // This is a hash of all the methods that are currently on the stack.
- private HashMap<MethodData, Integer> mStackMethods = new HashMap<MethodData, Integer>();
-
- // True if no calls have ever been added to this thread
- private boolean mIsEmpty;
-
- ThreadData(int id, String name, MethodData topLevel) {
- mId = id;
- mName = String.format("[%d] %s", id, name);
- mRoot = new Call(mName, topLevel);
- mCurrent = mRoot;
- mIsEmpty = true;
- }
-
- public boolean isEmpty() {
- return mIsEmpty;
- }
-
- public String getName() {
- return mName;
- }
-
- public Call getCalltreeRoot() {
- return mRoot;
- }
-
- void handleCall(Call call, long globalTime) {
- mIsEmpty = false;
- long currentTime = call.mThreadStartTime;
- if (currentTime < mLastEventTime) {
- System.err
- .printf(
- "ThreadData: '%1$s' call time (%2$d) is less than previous time (%3$d) for thread '%4$s'\n",
- call.getName(), currentTime, mLastEventTime, mName);
- System.exit(1);
- }
- long elapsed = currentTime - mLastEventTime;
- mCpuTime += elapsed;
- if (call.getMethodAction() == 0) {
- // This is a method entry.
- enter(call, elapsed);
- } else {
- // This is a method exit.
- exit(call, elapsed, globalTime);
- }
- mLastEventTime = currentTime;
- mGlobalEndTime = globalTime;
- }
-
- private void enter(Call c, long elapsed) {
- Call caller = mCurrent;
- push(c);
-
- // Check the stack for a matching method to determine if this call
- // is recursive.
- MethodData md = c.mMethodData;
- Integer num = mStackMethods.get(md);
- if (num == null) {
- num = 0;
- } else if (num > 0) {
- c.setRecursive(true);
- }
- num += 1;
- mStackMethods.put(md, num);
- mCurrent = c;
-
- // Add the elapsed time to the caller's exclusive time
- caller.addExclusiveTime(elapsed);
- }
-
- private void exit(Call c, long elapsed, long globalTime) {
- mCurrent.mGlobalEndTime = globalTime;
- Call top = pop();
- if (top == null) {
- return;
- }
-
- if (mCurrent.mMethodData != c.mMethodData) {
- String error = "Method exit (" + c.getName()
- + ") does not match current method (" + mCurrent.getName()
- + ")";
- throw new RuntimeException(error);
- } else {
- long duration = c.mThreadStartTime - mCurrent.mThreadStartTime;
- Call caller = top();
- mCurrent.addExclusiveTime(elapsed);
- mCurrent.addInclusiveTime(duration, caller);
- if (caller == null) {
- caller = mRoot;
- }
- mCurrent = caller;
- }
- }
-
- public void push(Call c) {
- mStack.add(c);
- }
-
- public Call pop() {
- ArrayList<Call> stack = mStack;
- if (stack.size() == 0)
- return null;
- Call top = stack.get(stack.size() - 1);
- stack.remove(stack.size() - 1);
-
- // Decrement the count on the method in the hash table and remove
- // the entry when it goes to zero.
- MethodData md = top.mMethodData;
- Integer num = mStackMethods.get(md);
- if (num != null) {
- num -= 1;
- if (num <= 0) {
- mStackMethods.remove(md);
- } else {
- mStackMethods.put(md, num);
- }
- }
- return top;
- }
-
- public Call top() {
- ArrayList<Call> stack = mStack;
- if (stack.size() == 0)
- return null;
- return stack.get(stack.size() - 1);
- }
-
- public long endTrace() {
- // If we have calls on the stack when the trace ends, then clean up
- // the stack and compute the inclusive time of the methods by pretending
- // that we are exiting from their methods now.
- while (mCurrent != mRoot) {
- long duration = mLastEventTime - mCurrent.mThreadStartTime;
- pop();
- Call caller = top();
- mCurrent.addInclusiveTime(duration, caller);
- mCurrent.mGlobalEndTime = mGlobalEndTime;
- if (caller == null) {
- caller = mRoot;
- }
- mCurrent = caller;
- }
- return mLastEventTime;
- }
-
- @Override
- public String toString() {
- return mName;
- }
-
- public int getId() {
- return mId;
- }
-
- public void setCpuTime(long cpuTime) {
- mCpuTime = cpuTime;
- }
-
- public long getCpuTime() {
- return mCpuTime;
- }
-
- public void setGlobalStartTime(long globalStartTime) {
- mGlobalStartTime = globalStartTime;
- }
-
- public long getGlobalStartTime() {
- return mGlobalStartTime;
- }
-
- public void setLastEventTime(long lastEventTime) {
- mLastEventTime = lastEventTime;
- }
-
- public long getLastEventTime() {
- return mLastEventTime;
- }
-
- public void setGlobalEndTime(long globalEndTime) {
- mGlobalEndTime = globalEndTime;
- }
-
- public long getGlobalEndTime() {
- return mGlobalEndTime;
- }
-
- public void setLastContextSwitch(Call lastContextSwitch) {
- mLastContextSwitch = lastContextSwitch;
- }
-
- public Call getLastContextSwitch() {
- return mLastContextSwitch;
- }
-}
diff --git a/traceview/src/com/android/traceview/TickScaler.java b/traceview/src/com/android/traceview/TickScaler.java
deleted file mode 100644
index 79fa160..0000000
--- a/traceview/src/com/android/traceview/TickScaler.java
+++ /dev/null
@@ -1,148 +0,0 @@
-/*
- * Copyright (C) 2006 The Android Open Source Project
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
-
-package com.android.traceview;
-
-class TickScaler {
-
- private double mMinVal; // required input
- private double mMaxVal; // required input
- private double mRangeVal;
- private int mNumPixels; // required input
- private int mPixelsPerTick; // required input
- private double mPixelsPerRange;
- private double mTickIncrement;
- private double mMinMajorTick;
-
- TickScaler(double minVal, double maxVal, int numPixels, int pixelsPerTick) {
- mMinVal = minVal;
- mMaxVal = maxVal;
- mNumPixels = numPixels;
- mPixelsPerTick = pixelsPerTick;
- }
-
- public void setMinVal(double minVal) {
- mMinVal = minVal;
- }
-
- public double getMinVal() {
- return mMinVal;
- }
-
- public void setMaxVal(double maxVal) {
- mMaxVal = maxVal;
- }
-
- public double getMaxVal() {
- return mMaxVal;
- }
-
- public void setNumPixels(int numPixels) {
- mNumPixels = numPixels;
- }
-
- public int getNumPixels() {
- return mNumPixels;
- }
-
- public void setPixelsPerTick(int pixelsPerTick) {
- mPixelsPerTick = pixelsPerTick;
- }
-
- public int getPixelsPerTick() {
- return mPixelsPerTick;
- }
-
- public void setPixelsPerRange(double pixelsPerRange) {
- mPixelsPerRange = pixelsPerRange;
- }
-
- public double getPixelsPerRange() {
- return mPixelsPerRange;
- }
-
- public void setTickIncrement(double tickIncrement) {
- mTickIncrement = tickIncrement;
- }
-
- public double getTickIncrement() {
- return mTickIncrement;
- }
-
- public void setMinMajorTick(double minMajorTick) {
- mMinMajorTick = minMajorTick;
- }
-
- public double getMinMajorTick() {
- return mMinMajorTick;
- }
-
- // Convert a time value to a 0-based pixel value
- public int valueToPixel(double value) {
- return (int) Math.ceil(mPixelsPerRange * (value - mMinVal) - 0.5);
- }
-
- // Convert a time value to a 0-based fractional pixel
- public double valueToPixelFraction(double value) {
- return mPixelsPerRange * (value - mMinVal);
- }
-
- // Convert a 0-based pixel value to a time value
- public double pixelToValue(int pixel) {
- return mMinVal + (pixel / mPixelsPerRange);
- }
-
- public void computeTicks(boolean useGivenEndPoints) {
- int numTicks = mNumPixels / mPixelsPerTick;
- mRangeVal = mMaxVal - mMinVal;
- mTickIncrement = mRangeVal / numTicks;
- double dlogTickIncrement = Math.log10(mTickIncrement);
- int logTickIncrement = (int) Math.floor(dlogTickIncrement);
- double scale = Math.pow(10, logTickIncrement);
- double scaledTickIncr = mTickIncrement / scale;
- if (scaledTickIncr > 5.0)
- scaledTickIncr = 10;
- else if (scaledTickIncr > 2)
- scaledTickIncr = 5;
- else if (scaledTickIncr > 1)
- scaledTickIncr = 2;
- else
- scaledTickIncr = 1;
- mTickIncrement = scaledTickIncr * scale;
-
- if (!useGivenEndPoints) {
- // Round up the max val to the next minor tick
- double minorTickIncrement = mTickIncrement / 5;
- double dval = mMaxVal / minorTickIncrement;
- int ival = (int) dval;
- if (ival != dval)
- mMaxVal = (ival + 1) * minorTickIncrement;
-
- // Round down the min val to a multiple of tickIncrement
- ival = (int) (mMinVal / mTickIncrement);
- mMinVal = ival * mTickIncrement;
- mMinMajorTick = mMinVal;
- } else {
- int ival = (int) (mMinVal / mTickIncrement);
- mMinMajorTick = ival * mTickIncrement;
- if (mMinMajorTick < mMinVal)
- mMinMajorTick = mMinMajorTick + mTickIncrement;
- }
-
- mRangeVal = mMaxVal - mMinVal;
- mPixelsPerRange = (double) mNumPixels / mRangeVal;
- }
-}
diff --git a/traceview/src/com/android/traceview/TimeLineView.java b/traceview/src/com/android/traceview/TimeLineView.java
deleted file mode 100644
index 67dc97b..0000000
--- a/traceview/src/com/android/traceview/TimeLineView.java
+++ /dev/null
@@ -1,1961 +0,0 @@
-/*
- * Copyright (C) 2006 The Android Open Source Project
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
-
-package com.android.traceview;
-
-import org.eclipse.jface.resource.FontRegistry;
-import org.eclipse.swt.SWT;
-import org.eclipse.swt.custom.SashForm;
-import org.eclipse.swt.events.MouseAdapter;
-import org.eclipse.swt.events.MouseEvent;
-import org.eclipse.swt.events.MouseMoveListener;
-import org.eclipse.swt.events.PaintEvent;
-import org.eclipse.swt.events.PaintListener;
-import org.eclipse.swt.graphics.Color;
-import org.eclipse.swt.graphics.Cursor;
-import org.eclipse.swt.graphics.FontData;
-import org.eclipse.swt.graphics.GC;
-import org.eclipse.swt.graphics.Image;
-import org.eclipse.swt.graphics.Point;
-import org.eclipse.swt.graphics.Rectangle;
-import org.eclipse.swt.layout.FillLayout;
-import org.eclipse.swt.layout.GridData;
-import org.eclipse.swt.layout.GridLayout;
-import org.eclipse.swt.widgets.Canvas;
-import org.eclipse.swt.widgets.Composite;
-import org.eclipse.swt.widgets.Display;
-import org.eclipse.swt.widgets.Event;
-import org.eclipse.swt.widgets.Listener;
-import org.eclipse.swt.widgets.ScrollBar;
-
-import java.util.ArrayList;
-import java.util.Arrays;
-import java.util.Collection;
-import java.util.Collections;
-import java.util.Comparator;
-import java.util.HashMap;
-import java.util.Observable;
-import java.util.Observer;
-
-public class TimeLineView extends Composite implements Observer {
-
- private HashMap<String, RowData> mRowByName;
- private double mTotalElapsed;
- private RowData[] mRows;
- private Segment[] mSegments;
- private ArrayList<Segment> mSegmentList = new ArrayList<Segment>();
- private HashMap<Integer, String> mThreadLabels;
- private Timescale mTimescale;
- private Surface mSurface;
- private RowLabels mLabels;
- private SashForm mSashForm;
- private int mScrollOffsetY;
-
- public static final int PixelsPerTick = 50;
- private TickScaler mScaleInfo = new TickScaler(0, 0, 0, PixelsPerTick);
- private static final int LeftMargin = 10; // blank space on left
- private static final int RightMargin = 60; // blank space on right
-
- private Color mColorBlack;
- private Color mColorGray;
- private Color mColorDarkGray;
- private Color mColorForeground;
- private Color mColorRowBack;
- private Color mColorZoomSelection;
- private FontRegistry mFontRegistry;
-
- /** vertical height of drawn blocks in each row */
- private static final int rowHeight = 20;
-
- /** the blank space between rows */
- private static final int rowYMargin = 12;
- private static final int rowYMarginHalf = rowYMargin / 2;
-
- /** total vertical space for row */
- private static final int rowYSpace = rowHeight + rowYMargin;
- private static final int majorTickLength = 8;
- private static final int minorTickLength = 4;
- private static final int timeLineOffsetY = 38;
- private static final int tickToFontSpacing = 2;
-
- /** start of first row */
- private static final int topMargin = 70;
- private int mMouseRow = -1;
- private int mNumRows;
- private int mStartRow;
- private int mEndRow;
- private TraceUnits mUnits;
- private int mSmallFontWidth;
- private int mSmallFontHeight;
- private int mMediumFontWidth;
- private SelectionController mSelectionController;
- private MethodData mHighlightMethodData;
- private Call mHighlightCall;
- private static final int MinInclusiveRange = 3;
-
- /** Setting the fonts looks good on Linux but bad on Macs */
- private boolean mSetFonts = false;
-
- public static interface Block {
- public String getName();
- public MethodData getMethodData();
- public long getStartTime();
- public long getEndTime();
- public Color getColor();
- public double addWeight(int x, int y, double weight);
- public void clearWeight();
- }
-
- public static interface Row {
- public int getId();
- public String getName();
- }
-
- public static class Record {
- Row row;
- Block block;
-
- public Record(Row row, Block block) {
- this.row = row;
- this.block = block;
- }
- }
-
- public TimeLineView(Composite parent, TraceReader reader,
- SelectionController selectionController) {
- super(parent, SWT.NONE);
- mRowByName = new HashMap<String, RowData>();
- this.mSelectionController = selectionController;
- selectionController.addObserver(this);
- mUnits = reader.getTraceUnits();
- mThreadLabels = reader.getThreadLabels();
-
- Display display = getDisplay();
- mColorGray = display.getSystemColor(SWT.COLOR_GRAY);
- mColorDarkGray = display.getSystemColor(SWT.COLOR_DARK_GRAY);
- mColorBlack = display.getSystemColor(SWT.COLOR_BLACK);
- // mColorBackground = display.getSystemColor(SWT.COLOR_WHITE);
- mColorForeground = display.getSystemColor(SWT.COLOR_BLACK);
- mColorRowBack = new Color(display, 240, 240, 255);
- mColorZoomSelection = new Color(display, 230, 230, 230);
-
- mFontRegistry = new FontRegistry(display);
- mFontRegistry.put("small", // $NON-NLS-1$
- new FontData[] { new FontData("Arial", 8, SWT.NORMAL) }); // $NON-NLS-1$
- mFontRegistry.put("courier8", // $NON-NLS-1$
- new FontData[] { new FontData("Courier New", 8, SWT.BOLD) }); // $NON-NLS-1$
- mFontRegistry.put("medium", // $NON-NLS-1$
- new FontData[] { new FontData("Courier New", 10, SWT.NORMAL) }); // $NON-NLS-1$
-
- Image image = new Image(display, new Rectangle(100, 100, 100, 100));
- GC gc = new GC(image);
- if (mSetFonts) {
- gc.setFont(mFontRegistry.get("small")); // $NON-NLS-1$
- }
- mSmallFontWidth = gc.getFontMetrics().getAverageCharWidth();
- mSmallFontHeight = gc.getFontMetrics().getHeight();
-
- if (mSetFonts) {
- gc.setFont(mFontRegistry.get("medium")); // $NON-NLS-1$
- }
- mMediumFontWidth = gc.getFontMetrics().getAverageCharWidth();
-
- image.dispose();
- gc.dispose();
-
- setLayout(new FillLayout());
-
- // Create a sash form for holding two canvas views, one for the
- // thread labels and one for the thread timeline.
- mSashForm = new SashForm(this, SWT.HORIZONTAL);
- mSashForm.setBackground(mColorGray);
- mSashForm.SASH_WIDTH = 3;
-
- // Create a composite for the left side of the sash
- Composite composite = new Composite(mSashForm, SWT.NONE);
- GridLayout layout = new GridLayout(1, true /* make columns equal width */);
- layout.marginHeight = 0;
- layout.marginWidth = 0;
- layout.verticalSpacing = 1;
- composite.setLayout(layout);
-
- // Create a blank corner space in the upper left corner
- BlankCorner corner = new BlankCorner(composite);
- GridData gridData = new GridData(GridData.FILL_HORIZONTAL);
- gridData.heightHint = topMargin;
- corner.setLayoutData(gridData);
-
- // Add the thread labels below the blank corner.
- mLabels = new RowLabels(composite);
- gridData = new GridData(GridData.FILL_BOTH);
- mLabels.setLayoutData(gridData);
-
- // Create another composite for the right side of the sash
- composite = new Composite(mSashForm, SWT.NONE);
- layout = new GridLayout(1, true /* make columns equal width */);
- layout.marginHeight = 0;
- layout.marginWidth = 0;
- layout.verticalSpacing = 1;
- composite.setLayout(layout);
-
- mTimescale = new Timescale(composite);
- gridData = new GridData(GridData.FILL_HORIZONTAL);
- gridData.heightHint = topMargin;
- mTimescale.setLayoutData(gridData);
-
- mSurface = new Surface(composite);
- gridData = new GridData(GridData.FILL_BOTH);
- mSurface.setLayoutData(gridData);
- mSashForm.setWeights(new int[] { 1, 5 });
-
- final ScrollBar vBar = mSurface.getVerticalBar();
- vBar.addListener(SWT.Selection, new Listener() {
- public void handleEvent(Event e) {
- mScrollOffsetY = vBar.getSelection();
- Point dim = mSurface.getSize();
- int newScrollOffsetY = computeVisibleRows(dim.y);
- if (newScrollOffsetY != mScrollOffsetY) {
- mScrollOffsetY = newScrollOffsetY;
- vBar.setSelection(newScrollOffsetY);
- }
- mLabels.redraw();
- mSurface.redraw();
- }
- });
-
- mSurface.addListener(SWT.Resize, new Listener() {
- public void handleEvent(Event e) {
- Point dim = mSurface.getSize();
-
- // If we don't need the scroll bar then don't display it.
- if (dim.y >= mNumRows * rowYSpace) {
- vBar.setVisible(false);
- } else {
- vBar.setVisible(true);
- }
- int newScrollOffsetY = computeVisibleRows(dim.y);
- if (newScrollOffsetY != mScrollOffsetY) {
- mScrollOffsetY = newScrollOffsetY;
- vBar.setSelection(newScrollOffsetY);
- }
-
- int spaceNeeded = mNumRows * rowYSpace;
- vBar.setMaximum(spaceNeeded);
- vBar.setThumb(dim.y);
-
- mLabels.redraw();
- mSurface.redraw();
- }
- });
-
- mSurface.addMouseListener(new MouseAdapter() {
- @Override
- public void mouseUp(MouseEvent me) {
- mSurface.mouseUp(me);
- }
-
- @Override
- public void mouseDown(MouseEvent me) {
- mSurface.mouseDown(me);
- }
-
- @Override
- public void mouseDoubleClick(MouseEvent me) {
- mSurface.mouseDoubleClick(me);
- }
- });
-
- mSurface.addMouseMoveListener(new MouseMoveListener() {
- public void mouseMove(MouseEvent me) {
- mSurface.mouseMove(me);
- }
- });
-
- mTimescale.addMouseListener(new MouseAdapter() {
- @Override
- public void mouseUp(MouseEvent me) {
- mTimescale.mouseUp(me);
- }
-
- @Override
- public void mouseDown(MouseEvent me) {
- mTimescale.mouseDown(me);
- }
-
- @Override
- public void mouseDoubleClick(MouseEvent me) {
- mTimescale.mouseDoubleClick(me);
- }
- });
-
- mTimescale.addMouseMoveListener(new MouseMoveListener() {
- public void mouseMove(MouseEvent me) {
- mTimescale.mouseMove(me);
- }
- });
-
- mLabels.addMouseMoveListener(new MouseMoveListener() {
- public void mouseMove(MouseEvent me) {
- mLabels.mouseMove(me);
- }
- });
-
- setData(reader.getThreadTimeRecords());
- }
-
- public void update(Observable objservable, Object arg) {
- // Ignore updates from myself
- if (arg == "TimeLineView") // $NON-NLS-1$
- return;
- // System.out.printf("timeline update from %s\n", arg);
- boolean foundHighlight = false;
- ArrayList<Selection> selections;
- selections = mSelectionController.getSelections();
- for (Selection selection : selections) {
- Selection.Action action = selection.getAction();
- if (action != Selection.Action.Highlight)
- continue;
- String name = selection.getName();
- // System.out.printf(" timeline highlight %s from %s\n", name, arg);
- if (name == "MethodData") { // $NON-NLS-1$
- foundHighlight = true;
- mHighlightMethodData = (MethodData) selection.getValue();
- // System.out.printf(" method %s\n",
- // highlightMethodData.getName());
- mHighlightCall = null;
- startHighlighting();
- } else if (name == "Call") { // $NON-NLS-1$
- foundHighlight = true;
- mHighlightCall = (Call) selection.getValue();
- // System.out.printf(" call %s\n", highlightCall.getName());
- mHighlightMethodData = null;
- startHighlighting();
- }
- }
- if (foundHighlight == false)
- mSurface.clearHighlights();
- }
-
- public void setData(ArrayList<Record> records) {
- if (records == null)
- records = new ArrayList<Record>();
-
- if (false) {
- System.out.println("TimelineView() list of records:"); // $NON-NLS-1$
- for (Record r : records) {
- System.out.printf("row '%s' block '%s' [%d, %d]\n", r.row // $NON-NLS-1$
- .getName(), r.block.getName(), r.block.getStartTime(),
- r.block.getEndTime());
- if (r.block.getStartTime() > r.block.getEndTime()) {
- System.err.printf("Error: block startTime > endTime\n"); // $NON-NLS-1$
- System.exit(1);
- }
- }
- }
-
- // Sort the records into increasing start time, and decreasing end time
- Collections.sort(records, new Comparator<Record>() {
- public int compare(Record r1, Record r2) {
- long start1 = r1.block.getStartTime();
- long start2 = r2.block.getStartTime();
- if (start1 > start2)
- return 1;
- if (start1 < start2)
- return -1;
-
- // The start times are the same, so compare the end times
- long end1 = r1.block.getEndTime();
- long end2 = r2.block.getEndTime();
- if (end1 > end2)
- return -1;
- if (end1 < end2)
- return 1;
-
- return 0;
- }
- });
-
- // The records are sorted into increasing start time,
- // so the minimum start time is the start time of the first record.
- double minVal = 0;
- if (records.size() > 0)
- minVal = records.get(0).block.getStartTime();
-
- // Sum the time spent in each row and block, and
- // keep track of the maximum end time.
- double maxVal = 0;
- for (Record rec : records) {
- Row row = rec.row;
- Block block = rec.block;
- String rowName = row.getName();
- RowData rd = mRowByName.get(rowName);
- if (rd == null) {
- rd = new RowData(row);
- mRowByName.put(rowName, rd);
- }
- long blockStartTime = block.getStartTime();
- long blockEndTime = block.getEndTime();
- if (blockEndTime > rd.mEndTime) {
- long start = Math.max(blockStartTime, rd.mEndTime);
- rd.mElapsed += blockEndTime - start;
- mTotalElapsed += blockEndTime - start;
- rd.mEndTime = blockEndTime;
- }
- if (blockEndTime > maxVal)
- maxVal = blockEndTime;
-
- // Keep track of nested blocks by using a stack (for each row).
- // Create a Segment object for each visible part of a block.
- Block top = rd.top();
- if (top == null) {
- rd.push(block);
- continue;
- }
-
- long topStartTime = top.getStartTime();
- long topEndTime = top.getEndTime();
- if (topEndTime >= blockStartTime) {
- // Add this segment if it has a non-zero elapsed time.
- if (topStartTime < blockStartTime) {
- Segment segment = new Segment(rd, top, topStartTime,
- blockStartTime);
- mSegmentList.add(segment);
- }
-
- // If this block starts where the previous (top) block ends,
- // then pop off the top block.
- if (topEndTime == blockStartTime)
- rd.pop();
- rd.push(block);
- } else {
- // We may have to pop several frames here.
- popFrames(rd, top, blockStartTime);
- rd.push(block);
- }
- }
-
- // Clean up the stack of each row
- for (RowData rd : mRowByName.values()) {
- Block top = rd.top();
- popFrames(rd, top, Integer.MAX_VALUE);
- }
-
- mSurface.setRange(minVal, maxVal);
- mSurface.setLimitRange(minVal, maxVal);
-
- // Sort the rows into decreasing elapsed time
- Collection<RowData> rv = mRowByName.values();
- mRows = rv.toArray(new RowData[rv.size()]);
- Arrays.sort(mRows, new Comparator<RowData>() {
- public int compare(RowData rd1, RowData rd2) {
- return (int) (rd2.mElapsed - rd1.mElapsed);
- }
- });
-
- // Assign ranks to the sorted rows
- for (int ii = 0; ii < mRows.length; ++ii) {
- mRows[ii].mRank = ii;
- }
-
- // Compute the number of rows with data
- mNumRows = 0;
- for (int ii = 0; ii < mRows.length; ++ii) {
- if (mRows[ii].mElapsed == 0)
- break;
- mNumRows += 1;
- }
-
- // Sort the blocks into increasing rows, and within rows into
- // increasing start values.
- mSegments = mSegmentList.toArray(new Segment[mSegmentList.size()]);
- Arrays.sort(mSegments, new Comparator<Segment>() {
- public int compare(Segment bd1, Segment bd2) {
- RowData rd1 = bd1.mRowData;
- RowData rd2 = bd2.mRowData;
- int diff = rd1.mRank - rd2.mRank;
- if (diff == 0) {
- long timeDiff = bd1.mStartTime - bd2.mStartTime;
- if (timeDiff == 0)
- timeDiff = bd1.mEndTime - bd2.mEndTime;
- return (int) timeDiff;
- }
- return diff;
- }
- });
-
- if (false) {
- for (Segment segment : mSegments) {
- System.out.printf("seg '%s' [%6d, %6d] %s\n",
- segment.mRowData.mName, segment.mStartTime,
- segment.mEndTime, segment.mBlock.getName());
- if (segment.mStartTime > segment.mEndTime) {
- System.err.printf("Error: segment startTime > endTime\n");
- System.exit(1);
- }
- }
- }
- }
-
- private void popFrames(RowData rd, Block top, long startTime) {
- long topEndTime = top.getEndTime();
- long lastEndTime = top.getStartTime();
- while (topEndTime <= startTime) {
- if (topEndTime > lastEndTime) {
- Segment segment = new Segment(rd, top, lastEndTime, topEndTime);
- mSegmentList.add(segment);
- lastEndTime = topEndTime;
- }
- rd.pop();
- top = rd.top();
- if (top == null)
- return;
- topEndTime = top.getEndTime();
- }
-
- // If we get here, then topEndTime > startTime
- if (lastEndTime < startTime) {
- Segment bd = new Segment(rd, top, lastEndTime, startTime);
- mSegmentList.add(bd);
- }
- }
-
- private class RowLabels extends Canvas {
-
- /** The space between the row label and the sash line */
- private static final int labelMarginX = 2;
-
- public RowLabels(Composite parent) {
- super(parent, SWT.NO_BACKGROUND);
- addPaintListener(new PaintListener() {
- public void paintControl(PaintEvent pe) {
- draw(pe.display, pe.gc);
- }
- });
- }
-
- private void mouseMove(MouseEvent me) {
- int rownum = (me.y + mScrollOffsetY) / rowYSpace;
- if (mMouseRow != rownum) {
- mMouseRow = rownum;
- redraw();
- mSurface.redraw();
- }
- }
-
- private void draw(Display display, GC gc) {
- if (mSegments.length == 0) {
- // gc.setBackground(colorBackground);
- // gc.fillRectangle(getBounds());
- return;
- }
- Point dim = getSize();
-
- // Create an image for double-buffering
- Image image = new Image(display, getBounds());
-
- // Set up the off-screen gc
- GC gcImage = new GC(image);
- if (mSetFonts)
- gcImage.setFont(mFontRegistry.get("medium")); // $NON-NLS-1$
-
- if (mNumRows > 2) {
- // Draw the row background stripes
- gcImage.setBackground(mColorRowBack);
- for (int ii = 1; ii < mNumRows; ii += 2) {
- RowData rd = mRows[ii];
- int y1 = rd.mRank * rowYSpace - mScrollOffsetY;
- gcImage.fillRectangle(0, y1, dim.x, rowYSpace);
- }
- }
-
- // Draw the row labels
- int offsetY = rowYMarginHalf - mScrollOffsetY;
- for (int ii = mStartRow; ii <= mEndRow; ++ii) {
- RowData rd = mRows[ii];
- int y1 = rd.mRank * rowYSpace + offsetY;
- Point extent = gcImage.stringExtent(rd.mName);
- int x1 = dim.x - extent.x - labelMarginX;
- gcImage.drawString(rd.mName, x1, y1, true);
- }
-
- // Draw a highlight box on the row where the mouse is.
- if (mMouseRow >= mStartRow && mMouseRow <= mEndRow) {
- gcImage.setForeground(mColorGray);
- int y1 = mMouseRow * rowYSpace - mScrollOffsetY;
- gcImage.drawRectangle(0, y1, dim.x, rowYSpace);
- }
-
- // Draw the off-screen buffer to the screen
- gc.drawImage(image, 0, 0);
-
- // Clean up
- image.dispose();
- gcImage.dispose();
- }
- }
-
- private class BlankCorner extends Canvas {
- public BlankCorner(Composite parent) {
- //super(parent, SWT.NO_BACKGROUND);
- super(parent, SWT.NONE);
- addPaintListener(new PaintListener() {
- public void paintControl(PaintEvent pe) {
- draw(pe.display, pe.gc);
- }
- });
- }
-
- private void draw(Display display, GC gc) {
- // Create a blank image and draw it to the canvas
- Image image = new Image(display, getBounds());
- gc.drawImage(image, 0, 0);
-
- // Clean up
- image.dispose();
- }
- }
-
- private class Timescale extends Canvas {
- private Point mMouse = new Point(LeftMargin, 0);
- private Cursor mZoomCursor;
- private String mMethodName = null;
- private Color mMethodColor = null;
- private int mMethodStartY;
- private int mMarkStartX;
- private int mMarkEndX;
-
- /** The space between the colored block and the method name */
- private static final int METHOD_BLOCK_MARGIN = 10;
-
- public Timescale(Composite parent) {
- //super(parent, SWT.NO_BACKGROUND);
- super(parent, SWT.NONE);
- Display display = getDisplay();
- mZoomCursor = new Cursor(display, SWT.CURSOR_SIZEWE);
- setCursor(mZoomCursor);
- mMethodStartY = mSmallFontHeight + 1;
- addPaintListener(new PaintListener() {
- public void paintControl(PaintEvent pe) {
- draw(pe.display, pe.gc);
- }
- });
- }
-
- public void setVbarPosition(int x) {
- mMouse.x = x;
- }
-
- public void setMarkStart(int x) {
- mMarkStartX = x;
- }
-
- public void setMarkEnd(int x) {
- mMarkEndX = x;
- }
-
- public void setMethodName(String name) {
- mMethodName = name;
- }
-
- public void setMethodColor(Color color) {
- mMethodColor = color;
- }
-
- private void mouseMove(MouseEvent me) {
- me.y = -1;
- mSurface.mouseMove(me);
- }
-
- private void mouseDown(MouseEvent me) {
- mSurface.startScaling(me.x);
- mSurface.redraw();
- }
-
- private void mouseUp(MouseEvent me) {
- mSurface.stopScaling(me.x);
- }
-
- private void mouseDoubleClick(MouseEvent me) {
- mSurface.resetScale();
- mSurface.redraw();
- }
-
- private void draw(Display display, GC gc) {
- Point dim = getSize();
-
- // Create an image for double-buffering
- Image image = new Image(display, getBounds());
-
- // Set up the off-screen gc
- GC gcImage = new GC(image);
- if (mSetFonts)
- gcImage.setFont(mFontRegistry.get("medium")); // $NON-NLS-1$
-
- if (mSurface.drawingSelection()) {
- drawSelection(display, gcImage);
- }
-
- drawTicks(display, gcImage);
-
- // Draw the vertical bar where the mouse is
- gcImage.setForeground(mColorDarkGray);
- gcImage.drawLine(mMouse.x, timeLineOffsetY, mMouse.x, dim.y);
-
- // Draw the current millseconds
- drawTickLegend(display, gcImage);
-
- // Draw the method name and color, if needed
- drawMethod(display, gcImage);
-
- // Draw the off-screen buffer to the screen
- gc.drawImage(image, 0, 0);
-
- // Clean up
- image.dispose();
- gcImage.dispose();
- }
-
- private void drawSelection(Display display, GC gc) {
- Point dim = getSize();
- gc.setForeground(mColorGray);
- gc.drawLine(mMarkStartX, timeLineOffsetY, mMarkStartX, dim.y);
- gc.setBackground(mColorZoomSelection);
- int x, width;
- if (mMarkStartX < mMarkEndX) {
- x = mMarkStartX;
- width = mMarkEndX - mMarkStartX;
- } else {
- x = mMarkEndX;
- width = mMarkStartX - mMarkEndX;
- }
- if (width > 1) {
- gc.fillRectangle(x, timeLineOffsetY, width, dim.y);
- }
- }
-
- private void drawTickLegend(Display display, GC gc) {
- int mouseX = mMouse.x - LeftMargin;
- double mouseXval = mScaleInfo.pixelToValue(mouseX);
- String info = mUnits.labelledString(mouseXval);
- gc.setForeground(mColorForeground);
- gc.drawString(info, LeftMargin + 2, 1, true);
-
- // Display the maximum data value
- double maxVal = mScaleInfo.getMaxVal();
- info = mUnits.labelledString(maxVal);
- info = String.format(" max %s ", info); // $NON-NLS-1$
- Point extent = gc.stringExtent(info);
- Point dim = getSize();
- int x1 = dim.x - RightMargin - extent.x;
- gc.drawString(info, x1, 1, true);
- }
-
- private void drawMethod(Display display, GC gc) {
- if (mMethodName == null) {
- return;
- }
-
- int x1 = LeftMargin;
- int y1 = mMethodStartY;
- gc.setBackground(mMethodColor);
- int width = 2 * mSmallFontWidth;
- gc.fillRectangle(x1, y1, width, mSmallFontHeight);
- x1 += width + METHOD_BLOCK_MARGIN;
- gc.drawString(mMethodName, x1, y1, true);
- }
-
- private void drawTicks(Display display, GC gc) {
- Point dim = getSize();
- int y2 = majorTickLength + timeLineOffsetY;
- int y3 = minorTickLength + timeLineOffsetY;
- int y4 = y2 + tickToFontSpacing;
- gc.setForeground(mColorForeground);
- gc.drawLine(LeftMargin, timeLineOffsetY, dim.x - RightMargin,
- timeLineOffsetY);
- double minVal = mScaleInfo.getMinVal();
- double maxVal = mScaleInfo.getMaxVal();
- double minMajorTick = mScaleInfo.getMinMajorTick();
- double tickIncrement = mScaleInfo.getTickIncrement();
- double minorTickIncrement = tickIncrement / 5;
- double pixelsPerRange = mScaleInfo.getPixelsPerRange();
-
- // Draw the initial minor ticks, if any
- if (minVal < minMajorTick) {
- gc.setForeground(mColorGray);
- double xMinor = minMajorTick;
- for (int ii = 1; ii <= 4; ++ii) {
- xMinor -= minorTickIncrement;
- if (xMinor < minVal)
- break;
- int x1 = LeftMargin
- + (int) (0.5 + (xMinor - minVal) * pixelsPerRange);
- gc.drawLine(x1, timeLineOffsetY, x1, y3);
- }
- }
-
- if (tickIncrement <= 10) {
- // TODO avoid rendering the loop when tickIncrement is invalid. It can be zero
- // or too small.
- // System.out.println(String.format("Timescale.drawTicks error: tickIncrement=%1f", tickIncrement));
- return;
- }
- for (double x = minMajorTick; x <= maxVal; x += tickIncrement) {
- int x1 = LeftMargin
- + (int) (0.5 + (x - minVal) * pixelsPerRange);
-
- // Draw a major tick
- gc.setForeground(mColorForeground);
- gc.drawLine(x1, timeLineOffsetY, x1, y2);
- if (x > maxVal)
- break;
-
- // Draw the tick text
- String tickString = mUnits.valueOf(x);
- gc.drawString(tickString, x1, y4, true);
-
- // Draw 4 minor ticks between major ticks
- gc.setForeground(mColorGray);
- double xMinor = x;
- for (int ii = 1; ii <= 4; ii++) {
- xMinor += minorTickIncrement;
- if (xMinor > maxVal)
- break;
- x1 = LeftMargin
- + (int) (0.5 + (xMinor - minVal) * pixelsPerRange);
- gc.drawLine(x1, timeLineOffsetY, x1, y3);
- }
- }
- }
- }
-
- private static enum GraphicsState {
- Normal, Marking, Scaling, Animating
- };
-
- private class Surface extends Canvas {
-
- public Surface(Composite parent) {
- super(parent, SWT.NO_BACKGROUND | SWT.V_SCROLL);
- Display display = getDisplay();
- mNormalCursor = new Cursor(display, SWT.CURSOR_CROSS);
- mIncreasingCursor = new Cursor(display, SWT.CURSOR_SIZEE);
- mDecreasingCursor = new Cursor(display, SWT.CURSOR_SIZEW);
-
- initZoomFractionsWithExp();
-
- addPaintListener(new PaintListener() {
- public void paintControl(PaintEvent pe) {
- draw(pe.display, pe.gc);
- }
- });
-
- mZoomAnimator = new Runnable() {
- public void run() {
- animateZoom();
- }
- };
-
- mHighlightAnimator = new Runnable() {
- public void run() {
- animateHighlight();
- }
- };
- }
-
- private void initZoomFractionsWithExp() {
- mZoomFractions = new double[ZOOM_STEPS];
- int next = 0;
- for (int ii = 0; ii < ZOOM_STEPS / 2; ++ii, ++next) {
- mZoomFractions[next] = (double) (1 << ii)
- / (double) (1 << (ZOOM_STEPS / 2));
- // System.out.printf("%d %f\n", next, zoomFractions[next]);
- }
- for (int ii = 2; ii < 2 + ZOOM_STEPS / 2; ++ii, ++next) {
- mZoomFractions[next] = (double) ((1 << ii) - 1)
- / (double) (1 << ii);
- // System.out.printf("%d %f\n", next, zoomFractions[next]);
- }
- }
-
- @SuppressWarnings("unused")
- private void initZoomFractionsWithSinWave() {
- mZoomFractions = new double[ZOOM_STEPS];
- for (int ii = 0; ii < ZOOM_STEPS; ++ii) {
- double offset = Math.PI * (double) ii / (double) ZOOM_STEPS;
- mZoomFractions[ii] = (Math.sin((1.5 * Math.PI + offset)) + 1.0) / 2.0;
- // System.out.printf("%d %f\n", ii, zoomFractions[ii]);
- }
- }
-
- public void setRange(double minVal, double maxVal) {
- mMinDataVal = minVal;
- mMaxDataVal = maxVal;
- mScaleInfo.setMinVal(minVal);
- mScaleInfo.setMaxVal(maxVal);
- }
-
- public void setLimitRange(double minVal, double maxVal) {
- mLimitMinVal = minVal;
- mLimitMaxVal = maxVal;
- }
-
- public void resetScale() {
- mScaleInfo.setMinVal(mLimitMinVal);
- mScaleInfo.setMaxVal(mLimitMaxVal);
- }
-
- private void draw(Display display, GC gc) {
- if (mSegments.length == 0) {
- // gc.setBackground(colorBackground);
- // gc.fillRectangle(getBounds());
- return;
- }
-
- // Create an image for double-buffering
- Image image = new Image(display, getBounds());
-
- // Set up the off-screen gc
- GC gcImage = new GC(image);
- if (mSetFonts)
- gcImage.setFont(mFontRegistry.get("small")); // $NON-NLS-1$
-
- // Draw the background
- // gcImage.setBackground(colorBackground);
- // gcImage.fillRectangle(image.getBounds());
-
- if (mGraphicsState == GraphicsState.Scaling) {
- double diff = mMouse.x - mMouseMarkStartX;
- if (diff > 0) {
- double newMinVal = mScaleMinVal - diff / mScalePixelsPerRange;
- if (newMinVal < mLimitMinVal)
- newMinVal = mLimitMinVal;
- mScaleInfo.setMinVal(newMinVal);
- // System.out.printf("diff %f scaleMin %f newMin %f\n",
- // diff, scaleMinVal, newMinVal);
- } else if (diff < 0) {
- double newMaxVal = mScaleMaxVal - diff / mScalePixelsPerRange;
- if (newMaxVal > mLimitMaxVal)
- newMaxVal = mLimitMaxVal;
- mScaleInfo.setMaxVal(newMaxVal);
- // System.out.printf("diff %f scaleMax %f newMax %f\n",
- // diff, scaleMaxVal, newMaxVal);
- }
- }
-
- // Recompute the ticks and strips only if the size has changed,
- // or we scrolled so that a new row is visible.
- Point dim = getSize();
- if (mStartRow != mCachedStartRow || mEndRow != mCachedEndRow
- || mScaleInfo.getMinVal() != mCachedMinVal
- || mScaleInfo.getMaxVal() != mCachedMaxVal) {
- mCachedStartRow = mStartRow;
- mCachedEndRow = mEndRow;
- int xdim = dim.x - TotalXMargin;
- mScaleInfo.setNumPixels(xdim);
- boolean forceEndPoints = (mGraphicsState == GraphicsState.Scaling
- || mGraphicsState == GraphicsState.Animating);
- mScaleInfo.computeTicks(forceEndPoints);
- mCachedMinVal = mScaleInfo.getMinVal();
- mCachedMaxVal = mScaleInfo.getMaxVal();
- if (mLimitMinVal > mScaleInfo.getMinVal())
- mLimitMinVal = mScaleInfo.getMinVal();
- if (mLimitMaxVal < mScaleInfo.getMaxVal())
- mLimitMaxVal = mScaleInfo.getMaxVal();
-
- // Compute the strips
- computeStrips();
- }
-
- if (mNumRows > 2) {
- // Draw the row background stripes
- gcImage.setBackground(mColorRowBack);
- for (int ii = 1; ii < mNumRows; ii += 2) {
- RowData rd = mRows[ii];
- int y1 = rd.mRank * rowYSpace - mScrollOffsetY;
- gcImage.fillRectangle(0, y1, dim.x, rowYSpace);
- }
- }
-
- if (drawingSelection()) {
- drawSelection(display, gcImage);
- }
-
- String blockName = null;
- Color blockColor = null;
-
- if (mDebug) {
- double pixelsPerRange = mScaleInfo.getPixelsPerRange();
- System.out
- .printf(
- "dim.x %d pixels %d minVal %f, maxVal %f ppr %f rpp %f\n",
- dim.x, dim.x - TotalXMargin, mScaleInfo
- .getMinVal(), mScaleInfo.getMaxVal(),
- pixelsPerRange, 1.0 / pixelsPerRange);
- }
-
- // Draw the strips
- Block selectBlock = null;
- for (Strip strip : mStripList) {
- if (strip.mColor == null) {
- // System.out.printf("strip.color is null\n");
- continue;
- }
- gcImage.setBackground(strip.mColor);
- gcImage.fillRectangle(strip.mX, strip.mY - mScrollOffsetY, strip.mWidth,
- strip.mHeight);
- if (mMouseRow == strip.mRowData.mRank) {
- if (mMouse.x >= strip.mX
- && mMouse.x < strip.mX + strip.mWidth) {
- blockName = strip.mSegment.mBlock.getName();
- blockColor = strip.mColor;
- }
- if (mMouseSelect.x >= strip.mX
- && mMouseSelect.x < strip.mX + strip.mWidth) {
- selectBlock = strip.mSegment.mBlock;
- }
- }
- }
- mMouseSelect.x = 0;
- mMouseSelect.y = 0;
-
- if (selectBlock != null) {
- ArrayList<Selection> selections = new ArrayList<Selection>();
- // Get the row label
- RowData rd = mRows[mMouseRow];
- selections.add(Selection.highlight("Thread", rd.mName)); // $NON-NLS-1$
- selections.add(Selection.highlight("Call", selectBlock)); // $NON-NLS-1$
-
- int mouseX = mMouse.x - LeftMargin;
- double mouseXval = mScaleInfo.pixelToValue(mouseX);
- selections.add(Selection.highlight("Time", mouseXval)); // $NON-NLS-1$
-
- mSelectionController.change(selections, "TimeLineView"); // $NON-NLS-1$
- mHighlightMethodData = null;
- mHighlightCall = (Call) selectBlock;
- startHighlighting();
- }
-
- // Draw a highlight box on the row where the mouse is.
- // Except don't draw the box if we are animating the
- // highlighing of a call or method because the inclusive
- // highlight bar passes through the highlight box and
- // causes an annoying flashing artifact.
- if (mMouseRow >= 0 && mMouseRow < mNumRows && mHighlightStep == 0) {
- gcImage.setForeground(mColorGray);
- int y1 = mMouseRow * rowYSpace - mScrollOffsetY;
- gcImage.drawLine(0, y1, dim.x, y1);
- gcImage.drawLine(0, y1 + rowYSpace, dim.x, y1 + rowYSpace);
- }
-
- // Highlight a selected method, if any
- drawHighlights(gcImage, dim);
-
- // Draw a vertical line where the mouse is.
- gcImage.setForeground(mColorDarkGray);
- int lineEnd = Math.min(dim.y, mNumRows * rowYSpace);
- gcImage.drawLine(mMouse.x, 0, mMouse.x, lineEnd);
-
- if (blockName != null) {
- mTimescale.setMethodName(blockName);
- mTimescale.setMethodColor(blockColor);
- mShowHighlightName = false;
- } else if (mShowHighlightName) {
- // Draw the highlighted method name
- MethodData md = mHighlightMethodData;
- if (md == null && mHighlightCall != null)
- md = mHighlightCall.getMethodData();
- if (md == null)
- System.out.printf("null highlight?\n"); // $NON-NLS-1$
- if (md != null) {
- mTimescale.setMethodName(md.getProfileName());
- mTimescale.setMethodColor(md.getColor());
- }
- } else {
- mTimescale.setMethodName(null);
- mTimescale.setMethodColor(null);
- }
- mTimescale.redraw();
-
- // Draw the off-screen buffer to the screen
- gc.drawImage(image, 0, 0);
-
- // Clean up
- image.dispose();
- gcImage.dispose();
- }
-
- private void drawHighlights(GC gc, Point dim) {
- int height = highlightHeight;
- if (height <= 0)
- return;
- for (Range range : mHighlightExclusive) {
- gc.setBackground(range.mColor);
- int xStart = range.mXdim.x;
- int width = range.mXdim.y;
- gc.fillRectangle(xStart, range.mY - height - mScrollOffsetY, width, height);
- }
-
- // Draw the inclusive lines a bit shorter
- height -= 1;
- if (height <= 0)
- height = 1;
-
- // Highlight the inclusive ranges
- gc.setForeground(mColorDarkGray);
- gc.setBackground(mColorDarkGray);
- for (Range range : mHighlightInclusive) {
- int x1 = range.mXdim.x;
- int x2 = range.mXdim.y;
- boolean drawLeftEnd = false;
- boolean drawRightEnd = false;
- if (x1 >= LeftMargin)
- drawLeftEnd = true;
- else
- x1 = LeftMargin;
- if (x2 >= LeftMargin)
- drawRightEnd = true;
- else
- x2 = dim.x - RightMargin;
- int y1 = range.mY + rowHeight + 2 - mScrollOffsetY;
-
- // If the range is very narrow, then just draw a small
- // rectangle.
- if (x2 - x1 < MinInclusiveRange) {
- int width = x2 - x1;
- if (width < 2)
- width = 2;
- gc.fillRectangle(x1, y1, width, height);
- continue;
- }
- if (drawLeftEnd) {
- if (drawRightEnd) {
- // Draw both ends
- int[] points = { x1, y1, x1, y1 + height, x2,
- y1 + height, x2, y1 };
- gc.drawPolyline(points);
- } else {
- // Draw the left end
- int[] points = { x1, y1, x1, y1 + height, x2,
- y1 + height };
- gc.drawPolyline(points);
- }
- } else {
- if (drawRightEnd) {
- // Draw the right end
- int[] points = { x1, y1 + height, x2, y1 + height, x2,
- y1 };
- gc.drawPolyline(points);
- } else {
- // Draw neither end, just the line
- int[] points = { x1, y1 + height, x2, y1 + height };
- gc.drawPolyline(points);
- }
- }
-
- // Draw the arrowheads, if necessary
- if (drawLeftEnd == false) {
- int[] points = { x1 + 7, y1 + height - 4, x1, y1 + height,
- x1 + 7, y1 + height + 4 };
- gc.fillPolygon(points);
- }
- if (drawRightEnd == false) {
- int[] points = { x2 - 7, y1 + height - 4, x2, y1 + height,
- x2 - 7, y1 + height + 4 };
- gc.fillPolygon(points);
- }
- }
- }
-
- private boolean drawingSelection() {
- return mGraphicsState == GraphicsState.Marking
- || mGraphicsState == GraphicsState.Animating;
- }
-
- private void drawSelection(Display display, GC gc) {
- Point dim = getSize();
- gc.setForeground(mColorGray);
- gc.drawLine(mMouseMarkStartX, 0, mMouseMarkStartX, dim.y);
- gc.setBackground(mColorZoomSelection);
- int width;
- int mouseX = (mGraphicsState == GraphicsState.Animating) ? mMouseMarkEndX : mMouse.x;
- int x;
- if (mMouseMarkStartX < mouseX) {
- x = mMouseMarkStartX;
- width = mouseX - mMouseMarkStartX;
- } else {
- x = mouseX;
- width = mMouseMarkStartX - mouseX;
- }
- gc.fillRectangle(x, 0, width, dim.y);
- }
-
- private void computeStrips() {
- double minVal = mScaleInfo.getMinVal();
- double maxVal = mScaleInfo.getMaxVal();
-
- // Allocate space for the pixel data
- Pixel[] pixels = new Pixel[mNumRows];
- for (int ii = 0; ii < mNumRows; ++ii)
- pixels[ii] = new Pixel();
-
- // Clear the per-block pixel data
- for (int ii = 0; ii < mSegments.length; ++ii) {
- mSegments[ii].mBlock.clearWeight();
- }
-
- mStripList.clear();
- mHighlightExclusive.clear();
- mHighlightInclusive.clear();
- MethodData callMethod = null;
- long callStart = 0;
- long callEnd = -1;
- RowData callRowData = null;
- int prevMethodStart = -1;
- int prevCallStart = -1;
- if (mHighlightCall != null) {
- int callPixelStart = -1;
- int callPixelEnd = -1;
- callStart = mHighlightCall.mGlobalStartTime;
- callEnd = mHighlightCall.mGlobalEndTime;
- callMethod = mHighlightCall.mMethodData;
- if (callStart >= minVal)
- callPixelStart = mScaleInfo.valueToPixel(callStart);
- if (callEnd <= maxVal)
- callPixelEnd = mScaleInfo.valueToPixel(callEnd);
- // System.out.printf("callStart,End %d,%d minVal,maxVal %f,%f
- // callPixelStart,End %d,%d\n",
- // callStart, callEnd, minVal, maxVal, callPixelStart,
- // callPixelEnd);
- int threadId = mHighlightCall.getThreadId();
- String threadName = mThreadLabels.get(threadId);
- callRowData = mRowByName.get(threadName);
- int y1 = callRowData.mRank * rowYSpace + rowYMarginHalf;
- Color color = callMethod.getColor();
- mHighlightInclusive.add(new Range(callPixelStart + LeftMargin,
- callPixelEnd + LeftMargin, y1, color));
- }
- for (Segment segment : mSegments) {
- if (segment.mEndTime <= minVal)
- continue;
- if (segment.mStartTime >= maxVal)
- continue;
- Block block = segment.mBlock;
- Color color = block.getColor();
- if (color == null)
- continue;
-
- double recordStart = Math.max(segment.mStartTime, minVal);
- double recordEnd = Math.min(segment.mEndTime, maxVal);
- if (recordStart == recordEnd)
- continue;
- int pixelStart = mScaleInfo.valueToPixel(recordStart);
- int pixelEnd = mScaleInfo.valueToPixel(recordEnd);
- int width = pixelEnd - pixelStart;
-
- RowData rd = segment.mRowData;
- MethodData md = block.getMethodData();
-
- // We will add the scroll offset later when we draw the strips
- int y1 = rd.mRank * rowYSpace + rowYMarginHalf;
-
- // If we can't display any more rows, then quit
- if (rd.mRank > mEndRow)
- break;
-
- // System.out.printf("segment %s val: [%.1f, %.1f] frac [%f, %f]
- // pixel: [%d, %d] pix.start %d weight %.2f %s\n",
- // block.getName(), recordStart, recordEnd,
- // scaleInfo.valueToPixelFraction(recordStart),
- // scaleInfo.valueToPixelFraction(recordEnd),
- // pixelStart, pixelEnd, pixels[rd.rank].start,
- // pixels[rd.rank].maxWeight,
- // pixels[rd.rank].segment != null
- // ? pixels[rd.rank].segment.block.getName()
- // : "null");
-
- if (mHighlightMethodData != null) {
- if (mHighlightMethodData == md) {
- if (prevMethodStart != pixelStart) {
- prevMethodStart = pixelStart;
- int rangeWidth = width;
- if (rangeWidth == 0)
- rangeWidth = 1;
- mHighlightExclusive.add(new Range(pixelStart
- + LeftMargin, rangeWidth, y1, color));
- Call call = (Call) block;
- callStart = call.mGlobalStartTime;
- int callPixelStart = -1;
- if (callStart >= minVal)
- callPixelStart = mScaleInfo.valueToPixel(callStart);
- if (prevCallStart != callPixelStart) {
- prevCallStart = callPixelStart;
- int callPixelEnd = -1;
- callEnd = call.mGlobalEndTime;
- if (callEnd <= maxVal)
- callPixelEnd = mScaleInfo.valueToPixel(callEnd);
- mHighlightInclusive.add(new Range(
- callPixelStart + LeftMargin,
- callPixelEnd + LeftMargin, y1, color));
- }
- }
- } else if (mFadeColors) {
- color = md.getFadedColor();
- }
- } else if (mHighlightCall != null) {
- if (segment.mStartTime >= callStart
- && segment.mEndTime <= callEnd && callMethod == md
- && callRowData == rd) {
- if (prevMethodStart != pixelStart) {
- prevMethodStart = pixelStart;
- int rangeWidth = width;
- if (rangeWidth == 0)
- rangeWidth = 1;
- mHighlightExclusive.add(new Range(pixelStart
- + LeftMargin, rangeWidth, y1, color));
- }
- } else if (mFadeColors) {
- color = md.getFadedColor();
- }
- }
-
- // Cases:
- // 1. This segment starts on a different pixel than the
- // previous segment started on. In this case, emit
- // the pixel strip, if any, and:
- // A. If the width is 0, then add this segment's
- // weight to the Pixel.
- // B. If the width > 0, then emit a strip for this
- // segment (no partial Pixel data).
- //
- // 2. Otherwise (the new segment starts on the same
- // pixel as the previous segment): add its "weight"
- // to the current pixel, and:
- // A. If the new segment has width 1,
- // then emit the pixel strip and then
- // add the segment's weight to the pixel.
- // B. If the new segment has width > 1,
- // then emit the pixel strip, and emit the rest
- // of the strip for this segment (no partial Pixel
- // data).
-
- Pixel pix = pixels[rd.mRank];
- if (pix.mStart != pixelStart) {
- if (pix.mSegment != null) {
- // Emit the pixel strip. This also clears the pixel.
- emitPixelStrip(rd, y1, pix);
- }
-
- if (width == 0) {
- // Compute the "weight" of this segment for the first
- // pixel. For a pixel N, the "weight" of a segment is
- // how much of the region [N - 0.5, N + 0.5] is covered
- // by the segment.
- double weight = computeWeight(recordStart, recordEnd,
- pixelStart);
- weight = block.addWeight(pixelStart, rd.mRank, weight);
- if (weight > pix.mMaxWeight) {
- pix.setFields(pixelStart, weight, segment, color,
- rd);
- }
- } else {
- int x1 = pixelStart + LeftMargin;
- Strip strip = new Strip(x1, y1, width, rowHeight, rd,
- segment, color);
- mStripList.add(strip);
- }
- } else {
- double weight = computeWeight(recordStart, recordEnd,
- pixelStart);
- weight = block.addWeight(pixelStart, rd.mRank, weight);
- if (weight > pix.mMaxWeight) {
- pix.setFields(pixelStart, weight, segment, color, rd);
- }
- if (width == 1) {
- // Emit the pixel strip. This also clears the pixel.
- emitPixelStrip(rd, y1, pix);
-
- // Compute the weight for the next pixel
- pixelStart += 1;
- weight = computeWeight(recordStart, recordEnd,
- pixelStart);
- weight = block.addWeight(pixelStart, rd.mRank, weight);
- pix.setFields(pixelStart, weight, segment, color, rd);
- } else if (width > 1) {
- // Emit the pixel strip. This also clears the pixel.
- emitPixelStrip(rd, y1, pix);
-
- // Emit a strip for the rest of the segment.
- pixelStart += 1;
- width -= 1;
- int x1 = pixelStart + LeftMargin;
- Strip strip = new Strip(x1, y1, width, rowHeight, rd,
- segment, color);
- mStripList.add(strip);
- }
- }
- }
-
- // Emit the last pixels of each row, if any
- for (int ii = 0; ii < mNumRows; ++ii) {
- Pixel pix = pixels[ii];
- if (pix.mSegment != null) {
- RowData rd = pix.mRowData;
- int y1 = rd.mRank * rowYSpace + rowYMarginHalf;
- // Emit the pixel strip. This also clears the pixel.
- emitPixelStrip(rd, y1, pix);
- }
- }
-
- if (false) {
- System.out.printf("computeStrips()\n");
- for (Strip strip : mStripList) {
- System.out.printf("%3d, %3d width %3d height %d %s\n",
- strip.mX, strip.mY, strip.mWidth, strip.mHeight,
- strip.mSegment.mBlock.getName());
- }
- }
- }
-
- private double computeWeight(double start, double end, int pixel) {
- double pixelStartFraction = mScaleInfo.valueToPixelFraction(start);
- double pixelEndFraction = mScaleInfo.valueToPixelFraction(end);
- double leftEndPoint = Math.max(pixelStartFraction, pixel - 0.5);
- double rightEndPoint = Math.min(pixelEndFraction, pixel + 0.5);
- double weight = rightEndPoint - leftEndPoint;
- return weight;
- }
-
- private void emitPixelStrip(RowData rd, int y, Pixel pixel) {
- Strip strip;
-
- if (pixel.mSegment == null)
- return;
-
- int x = pixel.mStart + LeftMargin;
- // Compute the percentage of the row height proportional to
- // the weight of this pixel. But don't let the proportion
- // exceed 3/4 of the row height so that we can easily see
- // if a given time range includes more than one method.
- int height = (int) (pixel.mMaxWeight * rowHeight * 0.75);
- if (height < mMinStripHeight)
- height = mMinStripHeight;
- int remainder = rowHeight - height;
- if (remainder > 0) {
- strip = new Strip(x, y, 1, remainder, rd, pixel.mSegment,
- mFadeColors ? mColorGray : mColorBlack);
- mStripList.add(strip);
- // System.out.printf("emitPixel (%d, %d) height %d black\n",
- // x, y, remainder);
- }
- strip = new Strip(x, y + remainder, 1, height, rd, pixel.mSegment,
- pixel.mColor);
- mStripList.add(strip);
- // System.out.printf("emitPixel (%d, %d) height %d %s\n",
- // x, y + remainder, height, pixel.segment.block.getName());
- pixel.mSegment = null;
- pixel.mMaxWeight = 0.0;
- }
-
- private void mouseMove(MouseEvent me) {
- if (false) {
- if (mHighlightMethodData != null) {
- mHighlightMethodData = null;
- // Force a recomputation of the strip colors
- mCachedEndRow = -1;
- }
- }
- Point dim = mSurface.getSize();
- int x = me.x;
- if (x < LeftMargin)
- x = LeftMargin;
- if (x > dim.x - RightMargin)
- x = dim.x - RightMargin;
- mMouse.x = x;
- mMouse.y = me.y;
- mTimescale.setVbarPosition(x);
- if (mGraphicsState == GraphicsState.Marking) {
- mTimescale.setMarkEnd(x);
- }
-
- if (mGraphicsState == GraphicsState.Normal) {
- // Set the cursor to the normal state.
- mSurface.setCursor(mNormalCursor);
- } else if (mGraphicsState == GraphicsState.Marking) {
- // Make the cursor point in the direction of the sweep
- if (mMouse.x >= mMouseMarkStartX)
- mSurface.setCursor(mIncreasingCursor);
- else
- mSurface.setCursor(mDecreasingCursor);
- }
- int rownum = (mMouse.y + mScrollOffsetY) / rowYSpace;
- if (me.y < 0 || me.y >= dim.y) {
- rownum = -1;
- }
- if (mMouseRow != rownum) {
- mMouseRow = rownum;
- mLabels.redraw();
- }
- redraw();
- }
-
- private void mouseDown(MouseEvent me) {
- Point dim = mSurface.getSize();
- int x = me.x;
- if (x < LeftMargin)
- x = LeftMargin;
- if (x > dim.x - RightMargin)
- x = dim.x - RightMargin;
- mMouseMarkStartX = x;
- mGraphicsState = GraphicsState.Marking;
- mSurface.setCursor(mIncreasingCursor);
- mTimescale.setMarkStart(mMouseMarkStartX);
- mTimescale.setMarkEnd(mMouseMarkStartX);
- redraw();
- }
-
- private void mouseUp(MouseEvent me) {
- mSurface.setCursor(mNormalCursor);
- if (mGraphicsState != GraphicsState.Marking) {
- mGraphicsState = GraphicsState.Normal;
- return;
- }
- mGraphicsState = GraphicsState.Animating;
- Point dim = mSurface.getSize();
-
- // If the user released the mouse outside the drawing area then
- // cancel the zoom.
- if (me.y <= 0 || me.y >= dim.y) {
- mGraphicsState = GraphicsState.Normal;
- redraw();
- return;
- }
-
- int x = me.x;
- if (x < LeftMargin)
- x = LeftMargin;
- if (x > dim.x - RightMargin)
- x = dim.x - RightMargin;
- mMouseMarkEndX = x;
-
- // If the user clicked and released the mouse at the same point
- // (+/- a pixel or two) then cancel the zoom (but select the
- // method).
- int dist = mMouseMarkEndX - mMouseMarkStartX;
- if (dist < 0)
- dist = -dist;
- if (dist <= 2) {
- mGraphicsState = GraphicsState.Normal;
-
- // Select the method underneath the mouse
- mMouseSelect.x = mMouseMarkStartX;
- mMouseSelect.y = me.y;
- redraw();
- return;
- }
-
- // Make mouseEndX be the higher end point
- if (mMouseMarkEndX < mMouseMarkStartX) {
- int temp = mMouseMarkEndX;
- mMouseMarkEndX = mMouseMarkStartX;
- mMouseMarkStartX = temp;
- }
-
- // If the zoom area is the whole window (or nearly the whole
- // window) then cancel the zoom.
- if (mMouseMarkStartX <= LeftMargin + MinZoomPixelMargin
- && mMouseMarkEndX >= dim.x - RightMargin - MinZoomPixelMargin) {
- mGraphicsState = GraphicsState.Normal;
- redraw();
- return;
- }
-
- // Compute some variables needed for zooming.
- // It's probably easiest to explain by an example. There
- // are two scales (or dimensions) involved: one for the pixels
- // and one for the values (microseconds). To keep the example
- // simple, suppose we have pixels in the range [0,16] and
- // values in the range [100, 260], and suppose the user
- // selects a zoom window from pixel 4 to pixel 8.
- //
- // usec: 100 140 180 260
- // |-------|ZZZZZZZ|---------------|
- // pixel: 0 4 8 16
- //
- // I've drawn the pixels starting at zero for simplicity, but
- // in fact the drawable area is offset from the left margin
- // by the value of "LeftMargin".
- //
- // The "pixels-per-range" (ppr) in this case is 0.1 (a tenth of
- // a pixel per usec). What we want is to redraw the screen in
- // several steps, each time increasing the zoom window until the
- // zoom window fills the screen. For simplicity, assume that
- // we want to zoom in four equal steps. Then the snapshots
- // of the screen at each step would look something like this:
- //
- // usec: 100 140 180 260
- // |-------|ZZZZZZZ|---------------|
- // pixel: 0 4 8 16
- //
- // usec: ? 140 180 ?
- // |-----|ZZZZZZZZZZZZZ|-----------|
- // pixel: 0 3 10 16
- //
- // usec: ? 140 180 ?
- // |---|ZZZZZZZZZZZZZZZZZZZ|-------|
- // pixel: 0 2 12 16
- //
- // usec: ?140 180 ?
- // |-|ZZZZZZZZZZZZZZZZZZZZZZZZZ|---|
- // pixel: 0 1 14 16
- //
- // usec: 140 180
- // |ZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZZ|
- // pixel: 0 16
- //
- // The problem is how to compute the endpoints (denoted by ?)
- // for each step. This is a little tricky. We first need to
- // compute the "fixed point": this is the point in the selection
- // that doesn't move left or right. Then we can recompute the
- // "ppr" (pixels per range) at each step and then find the
- // endpoints. The computation of the end points is done
- // in animateZoom(). This method computes the fixed point
- // and some other variables needed in animateZoom().
-
- double minVal = mScaleInfo.getMinVal();
- double maxVal = mScaleInfo.getMaxVal();
- double ppr = mScaleInfo.getPixelsPerRange();
- mZoomMin = minVal + ((mMouseMarkStartX - LeftMargin) / ppr);
- mZoomMax = minVal + ((mMouseMarkEndX - LeftMargin) / ppr);
-
- // Clamp the min and max values to the actual data min and max
- if (mZoomMin < mMinDataVal)
- mZoomMin = mMinDataVal;
- if (mZoomMax > mMaxDataVal)
- mZoomMax = mMaxDataVal;
-
- // Snap the min and max points to the grid determined by the
- // TickScaler
- // before we zoom.
- int xdim = dim.x - TotalXMargin;
- TickScaler scaler = new TickScaler(mZoomMin, mZoomMax, xdim,
- PixelsPerTick);
- scaler.computeTicks(false);
- mZoomMin = scaler.getMinVal();
- mZoomMax = scaler.getMaxVal();
-
- // Also snap the mouse points (in pixel space) to be consistent with
- // zoomMin and zoomMax (in value space).
- mMouseMarkStartX = (int) ((mZoomMin - minVal) * ppr + LeftMargin);
- mMouseMarkEndX = (int) ((mZoomMax - minVal) * ppr + LeftMargin);
- mTimescale.setMarkStart(mMouseMarkStartX);
- mTimescale.setMarkEnd(mMouseMarkEndX);
-
- // Compute the mouse selection end point distances
- mMouseEndDistance = dim.x - RightMargin - mMouseMarkEndX;
- mMouseStartDistance = mMouseMarkStartX - LeftMargin;
- mZoomMouseStart = mMouseMarkStartX;
- mZoomMouseEnd = mMouseMarkEndX;
- mZoomStep = 0;
-
- // Compute the fixed point in both value space and pixel space.
- mMin2ZoomMin = mZoomMin - minVal;
- mZoomMax2Max = maxVal - mZoomMax;
- mZoomFixed = mZoomMin + (mZoomMax - mZoomMin) * mMin2ZoomMin
- / (mMin2ZoomMin + mZoomMax2Max);
- mZoomFixedPixel = (mZoomFixed - minVal) * ppr + LeftMargin;
- mFixedPixelStartDistance = mZoomFixedPixel - LeftMargin;
- mFixedPixelEndDistance = dim.x - RightMargin - mZoomFixedPixel;
-
- mZoomMin2Fixed = mZoomFixed - mZoomMin;
- mFixed2ZoomMax = mZoomMax - mZoomFixed;
-
- getDisplay().timerExec(ZOOM_TIMER_INTERVAL, mZoomAnimator);
- redraw();
- update();
- }
-
- // No defined behavior yet for double-click.
- private void mouseDoubleClick(MouseEvent me) {
- }
-
- public void startScaling(int mouseX) {
- Point dim = mSurface.getSize();
- int x = mouseX;
- if (x < LeftMargin)
- x = LeftMargin;
- if (x > dim.x - RightMargin)
- x = dim.x - RightMargin;
- mMouseMarkStartX = x;
- mGraphicsState = GraphicsState.Scaling;
- mScalePixelsPerRange = mScaleInfo.getPixelsPerRange();
- mScaleMinVal = mScaleInfo.getMinVal();
- mScaleMaxVal = mScaleInfo.getMaxVal();
- }
-
- public void stopScaling(int mouseX) {
- mGraphicsState = GraphicsState.Normal;
- }
-
- private void animateHighlight() {
- mHighlightStep += 1;
- if (mHighlightStep >= HIGHLIGHT_STEPS) {
- mFadeColors = false;
- mHighlightStep = 0;
- // Force a recomputation of the strip colors
- mCachedEndRow = -1;
- } else {
- mFadeColors = true;
- mShowHighlightName = true;
- highlightHeight = highlightHeights[mHighlightStep];
- getDisplay().timerExec(HIGHLIGHT_TIMER_INTERVAL, mHighlightAnimator);
- }
- redraw();
- }
-
- private void clearHighlights() {
- // System.out.printf("clearHighlights()\n");
- mShowHighlightName = false;
- highlightHeight = 0;
- mHighlightMethodData = null;
- mHighlightCall = null;
- mFadeColors = false;
- mHighlightStep = 0;
- // Force a recomputation of the strip colors
- mCachedEndRow = -1;
- redraw();
- }
-
- private void animateZoom() {
- mZoomStep += 1;
- if (mZoomStep > ZOOM_STEPS) {
- mGraphicsState = GraphicsState.Normal;
- // Force a normal recomputation
- mCachedMinVal = mScaleInfo.getMinVal() + 1;
- } else if (mZoomStep == ZOOM_STEPS) {
- mScaleInfo.setMinVal(mZoomMin);
- mScaleInfo.setMaxVal(mZoomMax);
- mMouseMarkStartX = LeftMargin;
- Point dim = getSize();
- mMouseMarkEndX = dim.x - RightMargin;
- mTimescale.setMarkStart(mMouseMarkStartX);
- mTimescale.setMarkEnd(mMouseMarkEndX);
- getDisplay().timerExec(ZOOM_TIMER_INTERVAL, mZoomAnimator);
- } else {
- // Zoom in slowly at first, then speed up, then slow down.
- // The zoom fractions are precomputed to save time.
- double fraction = mZoomFractions[mZoomStep];
- mMouseMarkStartX = (int) (mZoomMouseStart - fraction * mMouseStartDistance);
- mMouseMarkEndX = (int) (mZoomMouseEnd + fraction * mMouseEndDistance);
- mTimescale.setMarkStart(mMouseMarkStartX);
- mTimescale.setMarkEnd(mMouseMarkEndX);
-
- // Compute the new pixels-per-range. Avoid division by zero.
- double ppr;
- if (mZoomMin2Fixed >= mFixed2ZoomMax)
- ppr = (mZoomFixedPixel - mMouseMarkStartX) / mZoomMin2Fixed;
- else
- ppr = (mMouseMarkEndX - mZoomFixedPixel) / mFixed2ZoomMax;
- double newMin = mZoomFixed - mFixedPixelStartDistance / ppr;
- double newMax = mZoomFixed + mFixedPixelEndDistance / ppr;
- mScaleInfo.setMinVal(newMin);
- mScaleInfo.setMaxVal(newMax);
-
- getDisplay().timerExec(ZOOM_TIMER_INTERVAL, mZoomAnimator);
- }
- redraw();
- }
-
- private static final int TotalXMargin = LeftMargin + RightMargin;
- private static final int yMargin = 1; // blank space on top
- // The minimum margin on each side of the zoom window, in pixels.
- private static final int MinZoomPixelMargin = 10;
- private GraphicsState mGraphicsState = GraphicsState.Normal;
- private Point mMouse = new Point(LeftMargin, 0);
- private int mMouseMarkStartX;
- private int mMouseMarkEndX;
- private boolean mDebug = false;
- private ArrayList<Strip> mStripList = new ArrayList<Strip>();
- private ArrayList<Range> mHighlightExclusive = new ArrayList<Range>();
- private ArrayList<Range> mHighlightInclusive = new ArrayList<Range>();
- private int mMinStripHeight = 2;
- private double mCachedMinVal;
- private double mCachedMaxVal;
- private int mCachedStartRow;
- private int mCachedEndRow;
- private double mScalePixelsPerRange;
- private double mScaleMinVal;
- private double mScaleMaxVal;
- private double mLimitMinVal;
- private double mLimitMaxVal;
- private double mMinDataVal;
- private double mMaxDataVal;
- private Cursor mNormalCursor;
- private Cursor mIncreasingCursor;
- private Cursor mDecreasingCursor;
- private static final int ZOOM_TIMER_INTERVAL = 10;
- private static final int HIGHLIGHT_TIMER_INTERVAL = 50;
- private static final int ZOOM_STEPS = 8; // must be even
- private int highlightHeight = 4;
- private final int[] highlightHeights = { 0, 2, 4, 5, 6, 5, 4, 2, 4, 5,
- 6 };
- private final int HIGHLIGHT_STEPS = highlightHeights.length;
- private boolean mFadeColors;
- private boolean mShowHighlightName;
- private double[] mZoomFractions;
- private int mZoomStep;
- private int mZoomMouseStart;
- private int mZoomMouseEnd;
- private int mMouseStartDistance;
- private int mMouseEndDistance;
- private Point mMouseSelect = new Point(0, 0);
- private double mZoomFixed;
- private double mZoomFixedPixel;
- private double mFixedPixelStartDistance;
- private double mFixedPixelEndDistance;
- private double mZoomMin2Fixed;
- private double mMin2ZoomMin;
- private double mFixed2ZoomMax;
- private double mZoomMax2Max;
- private double mZoomMin;
- private double mZoomMax;
- private Runnable mZoomAnimator;
- private Runnable mHighlightAnimator;
- private int mHighlightStep;
- }
-
- private int computeVisibleRows(int ydim) {
- // If we resize, then move the bottom row down. Don't allow the scroll
- // to waste space at the bottom.
- int offsetY = mScrollOffsetY;
- int spaceNeeded = mNumRows * rowYSpace;
- if (offsetY + ydim > spaceNeeded) {
- offsetY = spaceNeeded - ydim;
- if (offsetY < 0) {
- offsetY = 0;
- }
- }
- mStartRow = offsetY / rowYSpace;
- mEndRow = (offsetY + ydim) / rowYSpace;
- if (mEndRow >= mNumRows) {
- mEndRow = mNumRows - 1;
- }
-
- return offsetY;
- }
-
- private void startHighlighting() {
- // System.out.printf("startHighlighting()\n");
- mSurface.mHighlightStep = 0;
- mSurface.mFadeColors = true;
- // Force a recomputation of the color strips
- mSurface.mCachedEndRow = -1;
- getDisplay().timerExec(0, mSurface.mHighlightAnimator);
- }
-
- private static class RowData {
- RowData(Row row) {
- mName = row.getName();
- mStack = new ArrayList<Block>();
- }
-
- public void push(Block block) {
- mStack.add(block);
- }
-
- public Block top() {
- if (mStack.size() == 0)
- return null;
- return mStack.get(mStack.size() - 1);
- }
-
- public void pop() {
- if (mStack.size() == 0)
- return;
- mStack.remove(mStack.size() - 1);
- }
-
- private String mName;
- private int mRank;
- private long mElapsed;
- private long mEndTime;
- private ArrayList<Block> mStack;
- }
-
- private static class Segment {
- Segment(RowData rowData, Block block, long startTime, long endTime) {
- mRowData = rowData;
- mBlock = block;
- mStartTime = startTime;
- mEndTime = endTime;
- }
-
- private RowData mRowData;
- private Block mBlock;
- private long mStartTime;
- private long mEndTime;
- }
-
- private static class Strip {
- Strip(int x, int y, int width, int height, RowData rowData,
- Segment segment, Color color) {
- mX = x;
- mY = y;
- mWidth = width;
- mHeight = height;
- mRowData = rowData;
- mSegment = segment;
- mColor = color;
- }
-
- int mX;
- int mY;
- int mWidth;
- int mHeight;
- RowData mRowData;
- Segment mSegment;
- Color mColor;
- }
-
- private static class Pixel {
- public void setFields(int start, double weight, Segment segment,
- Color color, RowData rowData) {
- mStart = start;
- mMaxWeight = weight;
- mSegment = segment;
- mColor = color;
- mRowData = rowData;
- }
-
- int mStart = -2; // some value that won't match another pixel
- double mMaxWeight;
- Segment mSegment;
- Color mColor; // we need the color here because it may be faded
- RowData mRowData;
- }
-
- private static class Range {
- Range(int xStart, int width, int y, Color color) {
- mXdim.x = xStart;
- mXdim.y = width;
- mY = y;
- mColor = color;
- }
-
- Point mXdim = new Point(0, 0);
- int mY;
- Color mColor;
- }
-}
diff --git a/traceview/src/com/android/traceview/TraceReader.java b/traceview/src/com/android/traceview/TraceReader.java
deleted file mode 100644
index ae75876..0000000
--- a/traceview/src/com/android/traceview/TraceReader.java
+++ /dev/null
@@ -1,55 +0,0 @@
-/*
- * Copyright (C) 2006 The Android Open Source Project
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
-
-package com.android.traceview;
-
-import java.util.ArrayList;
-import java.util.HashMap;
-
-public abstract class TraceReader {
-
- private TraceUnits mTraceUnits;
-
- public TraceUnits getTraceUnits() {
- if (mTraceUnits == null)
- mTraceUnits = new TraceUnits();
- return mTraceUnits;
- }
-
- public ArrayList<TimeLineView.Record> getThreadTimeRecords() {
- return null;
- }
-
- public HashMap<Integer, String> getThreadLabels() {
- return null;
- }
-
- public MethodData[] getMethods() {
- return null;
- }
-
- public ThreadData[] getThreads() {
- return null;
- }
-
- public long getEndTime() {
- return 0;
- }
-
- public ProfileProvider getProfileProvider() {
- return null;
- }
-}
diff --git a/traceview/src/com/android/traceview/TraceUnits.java b/traceview/src/com/android/traceview/TraceUnits.java
deleted file mode 100644
index 20938f5..0000000
--- a/traceview/src/com/android/traceview/TraceUnits.java
+++ /dev/null
@@ -1,93 +0,0 @@
-/*
- * Copyright (C) 2006 The Android Open Source Project
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
-
-package com.android.traceview;
-
-import java.text.DecimalFormat;
-
-// This should be a singleton.
-public class TraceUnits {
-
- private TimeScale mTimeScale = TimeScale.MicroSeconds;
- private double mScale = 1.0;
- DecimalFormat mFormatter = new DecimalFormat();
-
- public double getScaledValue(long value) {
- return value * mScale;
- }
-
- public double getScaledValue(double value) {
- return value * mScale;
- }
-
- public String valueOf(long value) {
- return valueOf((double) value);
- }
-
- public String valueOf(double value) {
- String pattern;
- double scaled = value * mScale;
- if ((int) scaled == scaled)
- pattern = "###,###";
- else
- pattern = "###,###.###";
- mFormatter.applyPattern(pattern);
- return mFormatter.format(scaled);
- }
-
- public String labelledString(double value) {
- String units = label();
- String num = valueOf(value);
- return String.format("%s: %s", units, num);
- }
-
- public String labelledString(long value) {
- return labelledString((double) value);
- }
-
- public String label() {
- if (mScale == 1.0)
- return "usec";
- if (mScale == 0.001)
- return "msec";
- if (mScale == 0.000001)
- return "sec";
- return null;
- }
-
- public void setTimeScale(TimeScale val) {
- mTimeScale = val;
- switch (val) {
- case Seconds:
- mScale = 0.000001;
- break;
- case MilliSeconds:
- mScale = 0.001;
- break;
- case MicroSeconds:
- mScale = 1.0;
- break;
- }
- }
-
- public TimeScale getTimeScale() {
- return mTimeScale;
- }
-
- public enum TimeScale {
- Seconds, MilliSeconds, MicroSeconds
- };
-}