test/jdk/java/lang/StackWalker/EmbeddedStackWalkTest.java
author phh
Sat, 30 Nov 2019 14:33:05 -0800
changeset 59330 5b96c12f909d
parent 47216 71c04702a3d5
permissions -rw-r--r--
8234541: C1 emits an empty message when it inlines successfully Summary: Use "inline" as the message when successfull Reviewed-by: thartmann, mdoerr Contributed-by: navy.xliu@gmail.com

/*
 * Copyright (c) 2015, Oracle and/or its affiliates. All rights reserved.
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
 *
 * This code is free software; you can redistribute it and/or modify it
 * under the terms of the GNU General Public License version 2 only, as
 * published by the Free Software Foundation.
 *
 * This code is distributed in the hope that it will be useful, but WITHOUT
 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
 * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
 * version 2 for more details (a copy is included in the LICENSE file that
 * accompanied this code).
 *
 * You should have received a copy of the GNU General Public License version
 * 2 along with this work; if not, write to the Free Software Foundation,
 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
 *
 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
 * or visit www.oracle.com if you need additional information or have any
 * questions.
 */


/**
 * @test
 * @bug 8143214
 * @summary Verify StackWalker works well when embedded in another
 *          StackWalker's functions.
 * @run testng/othervm EmbeddedStackWalkTest
 */

import java.lang.StackWalker.StackFrame;
import static java.lang.StackWalker.Option.*;
import java.lang.invoke.MethodHandle;
import java.lang.invoke.MethodHandles;
import java.lang.invoke.MethodType;
import java.util.Arrays;
import java.util.EnumSet;

import org.testng.annotations.*;
import static org.testng.Assert.*;

public class EmbeddedStackWalkTest {
    static final StackWalker WALKERS[] = new StackWalker[] {
            StackWalker.getInstance(RETAIN_CLASS_REFERENCE),
            StackWalker.getInstance(EnumSet.of(SHOW_REFLECT_FRAMES, RETAIN_CLASS_REFERENCE)),
            StackWalker.getInstance(EnumSet.of(SHOW_HIDDEN_FRAMES, RETAIN_CLASS_REFERENCE))
    };

    static final int BIG_LOOP   = 30;
    static final int SMALL_LOOP = 5;

    @DataProvider
    public StackWalker[][] walkerProvider() {
        return new StackWalker[][] {
                new StackWalker[] { WALKERS[0] },
                new StackWalker[] { WALKERS[1] },
                new StackWalker[] { WALKERS[2] }
        };
    }

    @Test(dataProvider = "walkerProvider")
    public void test(StackWalker walker) {
        C1.call(walker, BIG_LOOP);
    }

    // line numbers are hardcoded for now.
    // Should annotate the line numbers and auto-generated these constants
    // for test verification instead
    static final int BEGIN_LINE = 71;   // the begin line number of approximate range.
    static final int END_LINE   = 136;  // the end line number of approximate range.
    static class C1 { // here is the begin line number of approximate range, L71.
        public static void call(StackWalker walker, int loops) {
            if (loops == 0) {
                String caller = walker.walk(s ->
                    s.map(StackFrame::getClassName)
                     .filter(cn -> !cn.startsWith("jdk.internal.reflect.") && !cn.startsWith("java.lang.invoke"))
                     .skip(2).findFirst()
                ).get();
                assertEquals(caller, C1.class.getName());

                walker.forEach(f -> C2.testEmbeddedWalker());
            } else {
                call(walker, --loops);
            }
        }
    }

    static class C2 {
        static final StackWalker embeddedWalkers[] = new StackWalker[] {
            StackWalker.getInstance(),
            StackWalker.getInstance(SHOW_REFLECT_FRAMES),
            StackWalker.getInstance(SHOW_HIDDEN_FRAMES)
        };

        public static void testEmbeddedWalker() {
            walk(SMALL_LOOP);
        }

        static void walk(int loops) {
            if (loops == 0) {
                Arrays.stream(embeddedWalkers)
                      .forEach(walker -> run(walker));
            } else {
                walk(--loops);
            }
        }

        static void run(StackWalker walker) {
            MethodHandles.Lookup lookup = MethodHandles.lookup();
            MethodHandle handle = null;
            try {
                handle = lookup.findStatic(C2.class, "call",
                        MethodType.methodType(void.class, StackWalker.class));
                handle.invoke(walker);
            } catch(Throwable t) {
                throw new RuntimeException(t);
            }
        }

        static void call(StackWalker walker) {
            String caller = walker.walk(s ->
                s.map(StackFrame::getClassName)
                 .filter(cn -> !cn.startsWith("jdk.internal.reflect.") && !cn.startsWith("java.lang.invoke"))
                 .skip(2).findFirst()
            ).get();
            assertEquals(caller, C2.class.getName());

            verify(walker, C1.class, "call");
            verify(walker, C2.class, "call");
            verify(walker, C2.class, "run");
            verify(walker, C2.class, "walk");
            verify(walker, C2.class, "testEmbeddedWalker");
        } // here is the end line number of approximate range, L136.

        static void verify(StackWalker walker, Class<?> c, String mn) {
            final String fileName = "EmbeddedStackWalkTest.java";
            walker.walk(s -> {
                s.limit(BIG_LOOP)
                 .filter(f -> c.getName().equals(f.getClassName()) && mn.equals(f.getMethodName()))
                 .forEach(f -> {
                    assertEquals(f.getFileName(), fileName);
                    int line = f.getLineNumber();
                    assertTrue(line >= BEGIN_LINE && line <= END_LINE);

                    StackTraceElement st = f.toStackTraceElement();
                    assertEquals(c.getName(), st.getClassName());
                    assertEquals(mn, st.getMethodName());
                    assertEquals(st.getFileName(), fileName);
                    line = st.getLineNumber();
                    assertTrue(line >= BEGIN_LINE && line <= END_LINE);
                });
                return null;
            });
        }
    }
}