| /* |
| * Copyright (C) 2015 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. |
| */ |
| |
| import java.io.BufferedReader; |
| import java.io.FileReader; |
| import java.io.InputStreamReader; |
| import java.util.Arrays; |
| import java.util.Comparator; |
| |
| public class Main extends Base implements Comparator<Main> { |
| // Whether to test local unwinding. |
| private boolean testLocal; |
| |
| // Unwinding another process, modelling debuggerd. |
| private boolean testRemote; |
| |
| // We fork ourself to create the secondary process for remote unwinding. |
| private boolean secondary; |
| |
| // Expect the symbols to contain full method signatures including parameters. |
| private boolean fullSignatures; |
| |
| private boolean passed; |
| |
| public Main(String[] args) throws Exception { |
| System.loadLibrary(args[0]); |
| for (String arg : args) { |
| if (arg.equals("--test-local")) { |
| testLocal = true; |
| } |
| if (arg.equals("--test-remote")) { |
| testRemote = true; |
| } |
| if (arg.equals("--secondary")) { |
| secondary = true; |
| } |
| if (arg.equals("--full-signatures")) { |
| fullSignatures = true; |
| } |
| } |
| if (!testLocal && !testRemote) { |
| System.out.println("No test selected."); |
| } |
| } |
| |
| public static void main(String[] args) throws Exception { |
| new Main(args).runBase(); |
| } |
| |
| public void runImpl() { |
| if (secondary) { |
| if (!testRemote) { |
| throw new RuntimeException("Should not be running secondary!"); |
| } |
| runSecondary(); |
| } else { |
| runPrimary(); |
| } |
| } |
| |
| private void runSecondary() { |
| foo(); |
| throw new RuntimeException("Didn't expect to get back..."); |
| } |
| |
| private void runPrimary() { |
| // First do the in-process unwinding. |
| if (testLocal && !foo()) { |
| System.out.println("Unwinding self failed."); |
| } |
| |
| if (!testRemote) { |
| // Skip the remote step. |
| return; |
| } |
| |
| // Fork the secondary. |
| String[] cmdline = getCmdLine(); |
| String[] secCmdLine = new String[cmdline.length + 1]; |
| System.arraycopy(cmdline, 0, secCmdLine, 0, cmdline.length); |
| secCmdLine[secCmdLine.length - 1] = "--secondary"; |
| Process p = exec(secCmdLine); |
| |
| try { |
| int pid = getPid(p); |
| if (pid <= 0) { |
| throw new RuntimeException("Couldn't parse process"); |
| } |
| |
| // Wait until the forked process had time to run until its sleep phase. |
| BufferedReader lineReader; |
| try { |
| InputStreamReader stdout = new InputStreamReader(p.getInputStream(), "UTF-8"); |
| lineReader = new BufferedReader(stdout); |
| while (!lineReader.readLine().contains("Going to sleep")) { |
| } |
| } catch (Exception e) { |
| throw new RuntimeException(e); |
| } |
| |
| if (!unwindOtherProcess(fullSignatures, pid)) { |
| System.out.println("Unwinding other process failed."); |
| |
| // In this case, log all the output. |
| // Note: this is potentially non-terminating code, if the secondary is totally stuck. |
| // We rely on the run-test timeout infrastructure to terminate the primary in |
| // such a case. |
| try { |
| String tmp; |
| System.out.println("Output from the secondary:"); |
| while ((tmp = lineReader.readLine()) != null) { |
| System.out.println("Secondary: " + tmp); |
| } |
| } catch (Exception e) { |
| e.printStackTrace(System.out); |
| } |
| } |
| |
| try { |
| lineReader.close(); |
| } catch (Exception e) { |
| e.printStackTrace(System.out); |
| } |
| } finally { |
| // Kill the forked process if it is not already dead. |
| p.destroy(); |
| } |
| } |
| |
| private static Process exec(String[] args) { |
| try { |
| return Runtime.getRuntime().exec(args); |
| } catch (Exception exc) { |
| throw new RuntimeException(exc); |
| } |
| } |
| |
| private static int getPid(Process p) { |
| // Could do reflection for the private pid field, but String parsing is easier. |
| String s = p.toString(); |
| if (s.startsWith("Process[pid=")) { |
| return Integer.parseInt(s.substring("Process[pid=".length(), s.indexOf(","))); |
| } else { |
| return -1; |
| } |
| } |
| |
| // Read /proc/self/cmdline to find the invocation command line (so we can fork another runtime). |
| private static String[] getCmdLine() { |
| try { |
| BufferedReader in = new BufferedReader(new FileReader("/proc/self/cmdline")); |
| String s = in.readLine(); |
| in.close(); |
| return s.split("\0"); |
| } catch (Exception exc) { |
| throw new RuntimeException(exc); |
| } |
| } |
| |
| public boolean foo() { |
| // Call bar via Arrays.binarySearch. |
| // This tests that we can unwind from framework code. |
| Main[] array = { this, this, this }; |
| Arrays.binarySearch(array, 0, 3, this /* value */, this /* comparator */); |
| return passed; |
| } |
| |
| public int compare(Main lhs, Main rhs) { |
| passed = bar(secondary); |
| // Returning "equal" ensures that we terminate search |
| // after first item and thus call bar() only once. |
| return 0; |
| } |
| |
| public boolean bar(boolean b) { |
| if (b) { |
| return sleep(2, b, 1.0); |
| } else { |
| return unwindInProcess(fullSignatures, 1, b); |
| } |
| } |
| |
| // Native functions. Note: to avoid deduping, they must all have different signatures. |
| |
| public native boolean sleep(int i, boolean b, double dummy); |
| |
| public native boolean unwindInProcess(boolean fullSignatures, int i, boolean b); |
| public native boolean unwindOtherProcess(boolean fullSignatures, int pid); |
| } |