Path: blob/master/test/hotspot/jtreg/vmTestbase/vm/share/stack/StackUtils.java
41159 views
/*1* Copyright (c) 2007, 2018, Oracle and/or its affiliates. All rights reserved.2* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.3*4* This code is free software; you can redistribute it and/or modify it5* under the terms of the GNU General Public License version 2 only, as6* published by the Free Software Foundation.7*8* This code is distributed in the hope that it will be useful, but WITHOUT9* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or10* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License11* version 2 for more details (a copy is included in the LICENSE file that12* accompanied this code).13*14* You should have received a copy of the GNU General Public License version15* 2 along with this work; if not, write to the Free Software Foundation,16* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.17*18* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA19* or visit www.oracle.com if you need additional information or have any20* questions.21*/22package vm.share.stack;2324import java.util.List;25import java.util.Map;2627import nsk.share.TestFailure;28import nsk.share.log.Log;2930public final class StackUtils {31private StackUtils() {32}3334private static String replace(String s) {35return (s == null || s.length() == 0) ? "?" : s;36}3738/**39* String representation of stack trace element.40*41* Note that null and empty values are replaced with '?'.42*/43public static String strStackTraceElement(StackTraceElement element) {44return "at " + replace(element.getClassName()) + "." + replace(element.getMethodName()) + "(" + replace(element.getFileName()) + ":" + element.getLineNumber() + ")";45}4647public static void printStackTraceElement(Log log, StackTraceElement element) {48log.info(" " + strStackTraceElement(element));49}5051public static void printStackTrace(Log log, StackTraceElement[] elements) {52for (StackTraceElement element : elements)53printStackTraceElement(log, element);54}5556public static void printStackTrace(Log log, Iterable<StackTraceElement> elements) {57for (StackTraceElement element : elements)58printStackTraceElement(log, element);59}6061/**62* Check that element matches expected element.63*64* Expected element is used as pattern for matching. A null or empty65* field value means that no comparison is done.66*/67public static boolean matches(StackTraceElement element, StackTraceElement expected) {68return69(expected.getClassName() == null || expected.getClassName().length() == 0 || expected.getClassName().equals(element.getClassName())) &&70(expected.getMethodName() == null || expected.getMethodName().length() == 0 || expected.getMethodName().equals(element.getMethodName())) &&71(expected.isNativeMethod() == element.isNativeMethod());72}7374public static StackTraceElement expectedTraceElement(String className, String methodName, boolean nativeMethod) {75// Replace null className with empty because StackTraceElement constructor does not allow null className.76return new StackTraceElement(className == null ? "" : className, methodName, null, (nativeMethod ? -2 : 0));77}7879public static void addExpectedTraceElement(List<StackTraceElement> expectedTrace, String className, String methodName, boolean nativeMethod) {80expectedTrace.add(0, expectedTraceElement(className, methodName, nativeMethod));81}8283/**84* Check that trace elements starting from given index match expected elements.85*/86public static void checkMatches(StackTraceElement[] elements, List<StackTraceElement> expectedTrace, int i) {87if (elements.length - i < expectedTrace.size())88throw new TestFailure("Expected at least " + expectedTrace.size() + " trace elements, got only " + (i + 1));89for (int j = 0; j < expectedTrace.size(); ++j) {90StackTraceElement expected = expectedTrace.get((expectedTrace.size() - 1) - j);91int index = (elements.length - 1) - i - j;92StackTraceElement actual = elements[index];93if (!matches(actual, expected))94throw new TestFailure("Expected element at index " + index + " to match: " + strStackTraceElement(expected));95}96}9798/**99* Find matching stack trace element starting from top of the stack.100*/101public static int findMatch(StackTraceElement[] elements, StackTraceElement expected) {102for (int i = 0; i < elements.length; ++i)103if (StackUtils.matches(elements[elements.length - 1 - i], expected))104return i;105return -1;106}107108/**109* Find the stack trace element that contains the "main(String[])" method110*111* @return StackTraceElement containing "main" function, null if there are more than one112*/113public static StackTraceElement findMain() {114Map<Thread, StackTraceElement[]> stackTraces = Thread.getAllStackTraces();115StackTraceElement mainMethodFrame = null;116for(StackTraceElement[] current : stackTraces.values()) {117if (current.length > 0) {118StackTraceElement last = current[current.length - 1];119if ("main".equals(last.getMethodName())) {120if (mainMethodFrame == null) {121mainMethodFrame = last;122} else if (!mainMethodFrame.getClassName().equals(last.getClassName())) {123// more than one class has a "main"124return null;125}126}127}128}129return mainMethodFrame;130}131}132133134