|  | // Copyright (c) 2016, the R8 project authors. Please see the AUTHORS file | 
|  | // for details. All rights reserved. Use of this source code is governed by a | 
|  | // BSD-style license that can be found in the LICENSE file. | 
|  |  | 
|  | // This code is not run directly. It needs to be compiled to dex code. | 
|  | // 'throwing.dex' is what is run. | 
|  |  | 
|  | package throwing; | 
|  |  | 
|  | import java.util.Collections; | 
|  | import java.util.List; | 
|  |  | 
|  | class Throwing { | 
|  |  | 
|  | static int[] used = new int[10]; | 
|  |  | 
|  | public static void main(String[] args) { | 
|  | try { | 
|  | used[0] = throwAtFistLine(42); | 
|  | } catch (Exception e) { | 
|  | printFrameHead(e); | 
|  | } | 
|  | try { | 
|  | used[1] = throwInMiddle(42); | 
|  | } catch (Exception e) { | 
|  | printFrameHead(e); | 
|  | } | 
|  | try { | 
|  | used[2] = throwAfterMultiInline(42); | 
|  | } catch (Exception e) { | 
|  | printFrameHead(e); | 
|  | } | 
|  | try { | 
|  | int value = magicNumber(42); | 
|  | // This throws after an inline, on the top-level. | 
|  | used[6] = value * 10; | 
|  | used[7] = anotherInlinedFunction(value); | 
|  | // | 
|  | // Some space to increase line numbers... | 
|  | // | 
|  | used[8] = value / (value & 0x0); | 
|  | } catch (Exception e) { | 
|  | printFrameHead(e); | 
|  | } | 
|  |  | 
|  | Nested nested = new Nested(); | 
|  |  | 
|  | try { | 
|  | used[3] = nested.justThrow(42); | 
|  | } catch (Exception e) { | 
|  | printFrameHead(e); | 
|  | } | 
|  |  | 
|  | nested.doSomethingUseless(); | 
|  |  | 
|  | used[0] += Nested.callAMethod(nested, 11); | 
|  | used[0] += Nested.callAMethod(nested, 42); | 
|  |  | 
|  | RenamedClass aInstance = RenamedClass.create(); | 
|  | aInstance.takeThingsForASpin(42); | 
|  |  | 
|  | System.out.print(used[0]); | 
|  |  | 
|  | try { | 
|  | throwInAFunctionThatIsNotInlinedAndCalledTwice(); | 
|  | } catch (Exception e) { | 
|  | printFrameHead(e); | 
|  | } | 
|  |  | 
|  | try { | 
|  | throwInAFunctionThatIsNotInlinedAndCalledTwice(); | 
|  | } catch (Exception e) { | 
|  | printFrameHead(e); | 
|  | } | 
|  |  | 
|  | try { | 
|  | aFunctionThatCallsAnInlinedMethodThatThrows(Collections.emptyList()); | 
|  | } catch (Exception e) { | 
|  | printFrameHead(e); | 
|  | } | 
|  |  | 
|  | try { | 
|  | anotherFunctionThatCallsAnInlinedMethodThatThrows("string"); | 
|  | } catch (Exception e) { | 
|  | printFrameHead(e); | 
|  | } | 
|  |  | 
|  | try { | 
|  | aFunctionsThatThrowsBeforeAnInlinedMethod(magicNumber(42)); | 
|  | } catch (Exception e) { | 
|  | printFrameHead(e); | 
|  | } | 
|  | } | 
|  |  | 
|  | public static int magicNumber(int value) { | 
|  | if (value < 0) { | 
|  | return magicNumber(value++); | 
|  | } | 
|  | return value; | 
|  | } | 
|  |  | 
|  | public static void printFrameHead(Exception e) { | 
|  | for (StackTraceElement element : e.getStackTrace()) { | 
|  | // Dalvik will print frames like: | 
|  | // FRAME: dalvik.system.NativeStart.main(Native Method) | 
|  | // ignore these to get same stacks as the JVM | 
|  | if (!element.toString().contains("dalvik")) { | 
|  | System.out.println("FRAME: " + element); | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | // This throws in the first line of the method. | 
|  | public static int throwAtFistLine(int value) { | 
|  | int aValue = value * 2 / (value & 0x0); | 
|  | return aValue; | 
|  | } | 
|  |  | 
|  | // This throws a little further down. | 
|  | public static int throwInMiddle(int value) { | 
|  | used[2] = value * 10; | 
|  | used[3] = value >> 3; | 
|  | used[4] = value / (value & 0x0); | 
|  | used[5] = value * 20; | 
|  | return value >> 5; | 
|  | } | 
|  |  | 
|  | // This throws after another inlined function. | 
|  | public static int throwAfterMultiInline(int value) { | 
|  | used[6] = value * 10; | 
|  | used[7] = anotherInlinedFunction(value); | 
|  | // | 
|  | // Some space to increase line numbers... | 
|  | // | 
|  | used[8] = value / (value & 0x0); | 
|  | return value >> 5; | 
|  | } | 
|  |  | 
|  | public static int throwInAFunctionThatIsNotInlinedAndCalledTwice() { | 
|  | for (int i = 0; i < 10; i++) { | 
|  | used[9] += i; | 
|  | System.out.println("Increment by one!"); | 
|  | } | 
|  | System.out.println("Incremented by 10."); | 
|  | used[9] = used[9] / (used[9] & 0x0); | 
|  | return used[9]; | 
|  | } | 
|  |  | 
|  | // Small method that throws and can be inlined. | 
|  | private static int anotherThrowingMethodToInline(int value) { | 
|  | used[4] = value / (value & 0x0); | 
|  | return value >> 5; | 
|  | } | 
|  |  | 
|  | // It is important that this function uses an argument type that is otherwise unused, so it gets | 
|  | // the same minified name. | 
|  | public static int aFunctionThatCallsAnInlinedMethodThatThrows(List aList) { | 
|  | used[9] = aList.size(); | 
|  | for (int i = 0; i < 10; i++) { | 
|  | used[9] += i; | 
|  | System.out.println("Increment by one!"); | 
|  | } | 
|  | System.out.println("Incremented by 10."); | 
|  | used[9] = anotherThrowingMethodToInline(used[9]); | 
|  | return used[9]; | 
|  | } | 
|  |  | 
|  | // Small method that throws and can be inlined. | 
|  | private static int yetAnotherThrowingMethodToInline(int value) { | 
|  | used[5] = value / (value & 0x0); | 
|  | return value >> 5; | 
|  | } | 
|  |  | 
|  | // It is important that this function uses an argument type that is otherwise unused, so it gets | 
|  | // the same minified name. | 
|  | public static int anotherFunctionThatCallsAnInlinedMethodThatThrows(String aString) { | 
|  | used[0] = aString.length(); | 
|  | for (int i = 0; i < 10; i++) { | 
|  | used[8] += i; | 
|  | System.out.println("Increment by one!"); | 
|  | } | 
|  | System.out.println("Incremented by 10."); | 
|  | used[8] = yetAnotherThrowingMethodToInline(used[8]); | 
|  | return used[8]; | 
|  | } | 
|  |  | 
|  | public static int aFunctionsThatThrowsBeforeAnInlinedMethod(int value) { | 
|  | used[1] = value / (value & 0x0); | 
|  | anotherInlinedFunction(used[1]); | 
|  | return used[1]; | 
|  | } | 
|  |  | 
|  | // This will be inlined above but does not throw | 
|  | public static int anotherInlinedFunction(int value) { | 
|  | return value / (value & 0xff); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * A nested class with different kind of methods to have inlining from a nested class and also | 
|  | * renamings of a nested class in the mapping file. | 
|  | * | 
|  | * <p>Some methods are recursive to avoid inlining. | 
|  | */ | 
|  | static class Nested { | 
|  |  | 
|  | int justThrow(int value) { | 
|  | return used[8] = value / (value & 0x0); | 
|  | } | 
|  |  | 
|  | // This will also be inlined. Not used in test but for generating interesting mapping files. | 
|  | void doSomethingUseless() { | 
|  | Throwing.used[9] = 11; | 
|  | } | 
|  |  | 
|  | static int callAMethod(Nested on, int value) { | 
|  | if (value > 20) { | 
|  | return callAMethod(on, value - 1); | 
|  | } else { | 
|  | return on.aMethod(value); | 
|  | } | 
|  | } | 
|  |  | 
|  | int aMethod(int value) { | 
|  | if (value > 10) { | 
|  | return aMethod(value - 1); | 
|  | } else { | 
|  | return value; | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | } |