1*795d594fSAndroid Build Coastguard Worker /* 2*795d594fSAndroid Build Coastguard Worker * Copyright (C) 2022 The Android Open Source Project 3*795d594fSAndroid Build Coastguard Worker * 4*795d594fSAndroid Build Coastguard Worker * Licensed under the Apache License, Version 2.0 (the "License"); 5*795d594fSAndroid Build Coastguard Worker * you may not use this file except in compliance with the License. 6*795d594fSAndroid Build Coastguard Worker * You may obtain a copy of the License at 7*795d594fSAndroid Build Coastguard Worker * 8*795d594fSAndroid Build Coastguard Worker * http://www.apache.org/licenses/LICENSE-2.0 9*795d594fSAndroid Build Coastguard Worker * 10*795d594fSAndroid Build Coastguard Worker * Unless required by applicable law or agreed to in writing, software 11*795d594fSAndroid Build Coastguard Worker * distributed under the License is distributed on an "AS IS" BASIS, 12*795d594fSAndroid Build Coastguard Worker * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13*795d594fSAndroid Build Coastguard Worker * See the License for the specific language governing permissions and 14*795d594fSAndroid Build Coastguard Worker * limitations under the License. 15*795d594fSAndroid Build Coastguard Worker */ 16*795d594fSAndroid Build Coastguard Worker 17*795d594fSAndroid Build Coastguard Worker import java.lang.ref.Reference; 18*795d594fSAndroid Build Coastguard Worker import java.lang.ref.WeakReference; 19*795d594fSAndroid Build Coastguard Worker import java.lang.ref.SoftReference; 20*795d594fSAndroid Build Coastguard Worker import java.math.BigInteger; 21*795d594fSAndroid Build Coastguard Worker import java.util.ArrayList; 22*795d594fSAndroid Build Coastguard Worker 23*795d594fSAndroid Build Coastguard Worker /** 24*795d594fSAndroid Build Coastguard Worker * We construct a main thread and worker threads, each retrieving stack traces 25*795d594fSAndroid Build Coastguard Worker * from the other. Since there are multiple workers, we may get a large number 26*795d594fSAndroid Build Coastguard Worker * of simultaneous stack trace attempts. 27*795d594fSAndroid Build Coastguard Worker */ 28*795d594fSAndroid Build Coastguard Worker public class Main { 29*795d594fSAndroid Build Coastguard Worker static final int NUM_THREADS = 5; 30*795d594fSAndroid Build Coastguard Worker static Thread mainThread; 31*795d594fSAndroid Build Coastguard Worker static volatile boolean pleaseStop = false; 32*795d594fSAndroid Build Coastguard Worker getTrace(Thread t)33*795d594fSAndroid Build Coastguard Worker private static void getTrace(Thread t) { 34*795d594fSAndroid Build Coastguard Worker StackTraceElement trace[] = t.getStackTrace(); 35*795d594fSAndroid Build Coastguard Worker if (!pleaseStop && (trace.length < 1 || trace.length > 20)) { 36*795d594fSAndroid Build Coastguard Worker // If called from traceGetter, we were started by the main thread, and it was still 37*795d594fSAndroid Build Coastguard Worker // running after the trace, so the main thread should have at least one frame on 38*795d594fSAndroid Build Coastguard Worker // the stack. If called by main(), we waited for all the traceGetters to start, 39*795d594fSAndroid Build Coastguard Worker // and didn't yet allow them to stop, so the same should be true. 40*795d594fSAndroid Build Coastguard Worker System.out.println("Stack trace for " + t.getName() + " has size " + trace.length); 41*795d594fSAndroid Build Coastguard Worker for (StackTraceElement e : trace) { 42*795d594fSAndroid Build Coastguard Worker System.out.println(e.toString()); 43*795d594fSAndroid Build Coastguard Worker } 44*795d594fSAndroid Build Coastguard Worker } 45*795d594fSAndroid Build Coastguard Worker } 46*795d594fSAndroid Build Coastguard Worker 47*795d594fSAndroid Build Coastguard Worker /** 48*795d594fSAndroid Build Coastguard Worker * Repeatedly get and minimally check stack trace of main thread. 49*795d594fSAndroid Build Coastguard Worker */ 50*795d594fSAndroid Build Coastguard Worker static Runnable traceGetter = new Runnable() { 51*795d594fSAndroid Build Coastguard Worker public void run() { 52*795d594fSAndroid Build Coastguard Worker System.out.println("Starting helper"); 53*795d594fSAndroid Build Coastguard Worker while (!pleaseStop) { 54*795d594fSAndroid Build Coastguard Worker getTrace(mainThread); 55*795d594fSAndroid Build Coastguard Worker } 56*795d594fSAndroid Build Coastguard Worker } 57*795d594fSAndroid Build Coastguard Worker }; 58*795d594fSAndroid Build Coastguard Worker main(String[] args)59*795d594fSAndroid Build Coastguard Worker public static void main(String[] args) throws Exception { 60*795d594fSAndroid Build Coastguard Worker System.out.println("Starting"); 61*795d594fSAndroid Build Coastguard Worker Thread[] t = new Thread[NUM_THREADS]; 62*795d594fSAndroid Build Coastguard Worker mainThread = Thread.currentThread(); 63*795d594fSAndroid Build Coastguard Worker for (int i = 0; i < NUM_THREADS; ++i) { 64*795d594fSAndroid Build Coastguard Worker t[i] = new Thread(traceGetter); 65*795d594fSAndroid Build Coastguard Worker t[i].start(); 66*795d594fSAndroid Build Coastguard Worker } 67*795d594fSAndroid Build Coastguard Worker try { 68*795d594fSAndroid Build Coastguard Worker Thread.sleep(1000); 69*795d594fSAndroid Build Coastguard Worker } catch (InterruptedException e) { 70*795d594fSAndroid Build Coastguard Worker System.out.println("Unexpectedly interrupted"); 71*795d594fSAndroid Build Coastguard Worker } 72*795d594fSAndroid Build Coastguard Worker for (int i = 0; i < NUM_THREADS; ++i) { 73*795d594fSAndroid Build Coastguard Worker getTrace(t[i]); 74*795d594fSAndroid Build Coastguard Worker } 75*795d594fSAndroid Build Coastguard Worker System.out.println("Finished worker stack traces"); 76*795d594fSAndroid Build Coastguard Worker long now = System.currentTimeMillis(); 77*795d594fSAndroid Build Coastguard Worker while (System.currentTimeMillis() - now < 2000) { 78*795d594fSAndroid Build Coastguard Worker try { 79*795d594fSAndroid Build Coastguard Worker Thread.sleep(1); 80*795d594fSAndroid Build Coastguard Worker } catch (InterruptedException e) { 81*795d594fSAndroid Build Coastguard Worker System.out.println("Unexpectedly interrupted"); 82*795d594fSAndroid Build Coastguard Worker } 83*795d594fSAndroid Build Coastguard Worker } 84*795d594fSAndroid Build Coastguard Worker pleaseStop = true; 85*795d594fSAndroid Build Coastguard Worker System.out.println("Finished"); 86*795d594fSAndroid Build Coastguard Worker } 87*795d594fSAndroid Build Coastguard Worker } 88