Java tutorial
/* * SonarQube Java * Copyright (C) 2012-2019 SonarSource SA * mailto:info AT sonarsource DOT com * * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 3 of the License, or (at your option) any later version. * * This program 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 * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public License * along with this program; if not, write to the Free Software Foundation, * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. */ package org.sonar.java.bytecode.cfg; import com.google.common.collect.HashMultiset; import com.google.common.collect.Iterables; import com.google.common.collect.Lists; import com.google.common.collect.Multiset; import java.io.File; import java.io.IOException; import java.util.Arrays; import java.util.List; import java.util.Objects; import java.util.function.Predicate; import java.util.stream.Collectors; import java.util.stream.Stream; import org.junit.Test; import org.objectweb.asm.ClassReader; import org.objectweb.asm.Handle; import org.objectweb.asm.Label; import org.objectweb.asm.Opcodes; import org.objectweb.asm.tree.AbstractInsnNode; import org.objectweb.asm.tree.ClassNode; import org.objectweb.asm.tree.MethodNode; import org.objectweb.asm.util.Printer; import org.sonar.java.ast.parser.JavaParser; import org.sonar.java.bytecode.cfg.testdata.CFGTestData; import org.sonar.java.bytecode.loader.SquidClassLoader; import org.sonar.java.resolve.BytecodeCompleter; import org.sonar.java.resolve.Convert; import org.sonar.java.resolve.SemanticModel; import org.sonar.java.se.SETestUtils; import org.sonar.plugins.java.api.semantic.Symbol; import org.sonar.plugins.java.api.tree.ClassTree; import org.sonar.plugins.java.api.tree.CompilationUnitTree; import org.sonar.plugins.java.api.tree.MethodTree; import org.sonar.plugins.java.api.tree.Tree; import static org.assertj.core.api.Assertions.assertThat; import static org.objectweb.asm.Opcodes.H_INVOKESTATIC; import static org.objectweb.asm.Opcodes.INVOKEINTERFACE; import static org.objectweb.asm.Opcodes.JSR; import static org.objectweb.asm.Opcodes.NOP; import static org.sonar.java.bytecode.cfg.Instructions.FIELD_INSN; import static org.sonar.java.bytecode.cfg.Instructions.INT_INSN; import static org.sonar.java.bytecode.cfg.Instructions.JUMP_INSN; import static org.sonar.java.bytecode.cfg.Instructions.METHOD_INSN; import static org.sonar.java.bytecode.cfg.Instructions.NO_OPERAND_INSN; import static org.sonar.java.bytecode.cfg.Instructions.TYPE_INSN; import static org.sonar.java.bytecode.cfg.Instructions.VAR_INSN; public class BytecodeCFGBuilderTest { @Test public void test() throws Exception { BytecodeCFG cfg = getCFGForMethod("fun"); StringBuilder sb = new StringBuilder(); cfg.blocks.forEach(b -> sb.append(b.printBlock())); assertThat(sb.toString()) .isEqualTo("B0(Exit)\n" + "B1\n" + "0: ILOAD\n" + "IFEQ Jumps to: B2(true) B3(false) \n" + "B2\n" + "0: LDC\n" + "1: ARETURN\n" + "Jumps to: B0 \n" + "B3\n" + "0: ALOAD\n" + "IFNONNULL Jumps to: B4(true) B5(false) \n" + "B4\n" + "0: LDC\n" + "1: ARETURN\n" + "Jumps to: B0 \n" + "B5\n" + "0: ACONST_NULL\n" + "1: ARETURN\n" + "Jumps to: B0 \n"); } static class InnerClass { private Object fun(boolean a, Object b) { if (a) { if (b == null) { return null; } return ""; } else { return "not a"; } } private boolean label_goto(Object b) { return b == null; } } @Test public void label_goto_successors() throws Exception { BytecodeCFG cfg = getCFGForMethod("label_goto"); StringBuilder sb = new StringBuilder(); cfg.blocks.forEach(b -> sb.append(b.printBlock())); assertThat(sb.toString()).isEqualTo("B0(Exit)\n" + "B1\n" + "0: ALOAD\n" + "IFNONNULL Jumps to: B2(true) B3(false) \n" + "B2\n" + "0: ICONST_0\n" + "Jumps to: B4 \n" + "B3\n" + "0: ICONST_1\n" + "GOTO Jumps to: B4 \n" + "B4\n" + "0: IRETURN\n" + "Jumps to: B0 \n"); } private BytecodeCFG getCFGForMethod(String methodName) { SquidClassLoader squidClassLoader = new SquidClassLoader( Lists.newArrayList(new File("target/test-classes"), new File("target/classes"))); File file = new File("src/test/java/org/sonar/java/bytecode/cfg/BytecodeCFGBuilderTest.java"); CompilationUnitTree tree = (CompilationUnitTree) JavaParser.createParser().parse(file); SemanticModel.createFor(tree, squidClassLoader); Symbol.TypeSymbol innerClass = ((Symbol.TypeSymbol) ((ClassTree) tree.types().get(0)).symbol() .lookupSymbols("InnerClass").iterator().next()); Symbol.MethodSymbol symbol = (Symbol.MethodSymbol) innerClass.lookupSymbols(methodName).iterator().next(); return SETestUtils.bytecodeCFG(symbol.signature(), squidClassLoader); } @Test public void test_all_instructions_are_part_of_CFG() throws Exception { SquidClassLoader squidClassLoader = new SquidClassLoader( Lists.newArrayList(new File("target/test-classes"), new File("target/classes"))); File file = new File("src/test/java/org/sonar/java/bytecode/cfg/testdata/CFGTestData.java"); CompilationUnitTree tree = (CompilationUnitTree) JavaParser.createParser().parse(file); SemanticModel.createFor(tree, squidClassLoader); Symbol.TypeSymbol testClazz = ((ClassTree) tree.types().get(0)).symbol(); ClassReader cr = new ClassReader(squidClassLoader .getResourceAsStream(Convert.bytecodeName(CFGTestData.class.getCanonicalName()) + ".class")); ClassNode classNode = new ClassNode(BytecodeCompleter.ASM_API_VERSION); cr.accept(classNode, 0); for (MethodNode method : classNode.methods) { Multiset<String> opcodes = Arrays.stream(method.instructions.toArray()).map(AbstractInsnNode::getOpcode) .filter(opcode -> opcode != -1).map(opcode -> Printer.OPCODES[opcode]) .collect(Collectors.toCollection(HashMultiset::create)); Symbol methodSymbol = Iterables.getOnlyElement(testClazz.lookupSymbols(method.name)); BytecodeCFG bytecodeCFG = SETestUtils.bytecodeCFG(((Symbol.MethodSymbol) methodSymbol).signature(), squidClassLoader); Multiset<String> cfgOpcodes = cfgOpcodes(bytecodeCFG); assertThat(cfgOpcodes).isEqualTo(opcodes); } } private Multiset<String> cfgOpcodes(BytecodeCFG bytecodeCFG) { return bytecodeCFG.blocks.stream() .flatMap(block -> Stream.concat(block.instructions.stream(), Stream.of(block.terminator))) .filter(Objects::nonNull).map(Instruction::opcode).map(opcode -> Printer.OPCODES[opcode]) .collect(Collectors.toCollection(HashMultiset::create)); } @Test public void all_opcodes_should_be_visited() throws Exception { Instructions ins = new Instructions(); Predicate<Integer> filterReturnAndThrow = opcode -> !((Opcodes.IRETURN <= opcode && opcode <= Opcodes.RETURN) || opcode == Opcodes.ATHROW); NO_OPERAND_INSN.stream().filter(filterReturnAndThrow).forEach(ins::visitInsn); INT_INSN.forEach(i -> ins.visitIntInsn(i, 0)); VAR_INSN.forEach(i -> ins.visitVarInsn(i, 0)); TYPE_INSN.forEach(i -> ins.visitTypeInsn(i, "java/lang/Object")); FIELD_INSN.forEach(i -> ins.visitFieldInsn(i, "java/lang/Object", "foo", "D(D)")); METHOD_INSN.forEach(i -> ins.visitMethodInsn(i, "java/lang/Object", "foo", "()V", i == INVOKEINTERFACE)); JUMP_INSN.forEach(i -> { Label jumpLabel = new Label(); ins.visitJumpInsn(i, jumpLabel); ins.visitLabel(jumpLabel); }); ins.visitLdcInsn("a"); ins.visitIincInsn(0, 1); Handle handle = new Handle(H_INVOKESTATIC, "", "", "()V", false); ins.visitInvokeDynamicInsn("sleep", "()V", handle); ins.visitLookupSwitchInsn(new Label(), new int[] {}, new Label[] {}); ins.visitMultiANewArrayInsn("B", 1); Label l0 = new Label(); Label dflt = new Label(); Label case0 = new Label(); ins.visitTableSwitchInsn(0, 1, dflt, case0); ins.visitLabel(dflt); ins.visitInsn(NOP); ins.visitLabel(l0); ins.visitInsn(NOP); BytecodeCFG cfg = ins.cfg(); Multiset<String> cfgOpcodes = cfgOpcodes(cfg); List<String> collect = Instructions.OPCODES.stream().filter(filterReturnAndThrow) .map(op -> Printer.OPCODES[op]).collect(Collectors.toList()); assertThat(cfgOpcodes).containsAll(collect); } @Test public void visited_label_should_be_assigned_to_true_successor() throws Exception { Label label0 = new Label(); Label label1 = new Label(); BytecodeCFG cfg = new Instructions().visitVarInsn(Opcodes.ALOAD, 0).visitJumpInsn(Opcodes.IFNULL, label0) .visitJumpInsn(Opcodes.IFEQ, label0).visitInsn(Opcodes.ICONST_0).visitJumpInsn(Opcodes.GOTO, label1) .visitLabel(label0).visitInsn(Opcodes.ICONST_1).visitLabel(label1).visitInsn(Opcodes.IRETURN).cfg(); BytecodeCFG.Block block3 = cfg.blocks.get(3); assertThat(block3.terminator.opcode).isEqualTo(Opcodes.IFEQ); assertThat(block3.falseSuccessor()).isNotNull().isSameAs(cfg.blocks.get(4)); assertThat(block3.trueSuccessor()).isNotNull().isSameAs(cfg.blocks.get(2)); assertThat(block3.successors).hasSize(2); assertThat(block3.successors()).hasSize(2); } @Test public void goto_successors() throws Exception { Label label0 = new Label(); Label label1 = new Label(); BytecodeCFG cfg = new Instructions().visitVarInsn(Opcodes.ALOAD, 0).visitJumpInsn(Opcodes.IFNULL, label0) .visitVarInsn(Opcodes.ALOAD, 0).visitJumpInsn(Opcodes.IFNULL, label1).visitVarInsn(Opcodes.ALOAD, 0) .visitVarInsn(Opcodes.ALOAD, 0).visitJumpInsn(Opcodes.IFEQ, label0).visitInsn(Opcodes.ICONST_0) .visitJumpInsn(Opcodes.GOTO, label1).visitLabel(label0).visitInsn(Opcodes.ICONST_1) .visitLabel(label1).visitInsn(Opcodes.IRETURN).cfg(); assertThat(cfg.blocks.get(6).successors).containsExactly(cfg.blocks.get(4)); } @Test public void isNotBlank_goto_followed_by_label() throws Exception { SquidClassLoader classLoader = new SquidClassLoader( Lists.newArrayList(new File("src/test/commons-lang-2.1"))); // apache commons 2.1 isNotBlank has a goto followed by an unreferenced label : see SONARJAVA-2461 BytecodeCFG bytecodeCFG = SETestUtils .bytecodeCFG("org.apache.commons.lang.StringUtils#isNotBlank(Ljava/lang/String;)Z", classLoader); assertThat(bytecodeCFG).isNotNull(); assertThat(bytecodeCFG.blocks).hasSize(11); assertThat(bytecodeCFG.blocks.get(4).successors).containsExactly(bytecodeCFG.blocks.get(6)); } @Test public void supportJSRandRET() throws Exception { SquidClassLoader classLoader = new SquidClassLoader(Lists.newArrayList(new File("src/test/JsrRet"))); BytecodeCFG bytecodeCFG = SETestUtils.bytecodeCFG("jdk3.AllInstructions#jsrAndRetInstructions(I)I", classLoader); assertThat(bytecodeCFG).isNotNull(); bytecodeCFG.blocks.stream().map(b -> b.terminator).filter(Objects::nonNull) .forEach(t -> assertThat(t.opcode).isNotEqualTo(JSR)); } @Test public void try_catch_finally() throws Exception { String methodName = "tryCatch"; String expectedCFG = "B0(Exit)\n" + "B1\n" + "Jumps to: B2 \n" + "B2\n" + "0: ALOAD\n" + "1: INVOKESPECIAL\n" + "2: ALOAD\n" + "3: INVOKESPECIAL\n" + "Jumps to: B3 B5(Exception:java.io.IOException) B7(Exception:!UncaughtException!) \n" + "B3\n" + "0: GETSTATIC\n" + "1: LDC\n" + "2: INVOKEVIRTUAL\n" + "GOTO Jumps to: B4 \n" + "B4\n" + "0: RETURN\n" + "Jumps to: B0 \n" + "B5\n" + "0: ASTORE\n" + "1: ALOAD\n" + "2: INVOKEVIRTUAL\n" + "Jumps to: B6 B7(Exception:!UncaughtException!) \n" + "B6\n" + "0: GETSTATIC\n" + "1: LDC\n" + "2: INVOKEVIRTUAL\n" + "GOTO Jumps to: B4 \n" + "B7\n" + "0: ASTORE\n" + "1: GETSTATIC\n" + "2: LDC\n" + "3: INVOKEVIRTUAL\n" + "4: ALOAD\n" + "5: ATHROW\n" + "Jumps to: B0 \n"; assertCFGforMethod(methodName, expectedCFG); } @Test public void try_catch_finally_with_multiplePaths_in_try() throws Exception { String methodName = "tryCatchWithMultiplePaths"; assertCFGforMethod(methodName, "B0(Exit)\n" + "B1\n" + "Jumps to: B2 \n" + "B2\n" + "0: ALOAD\n" + "1: INVOKESPECIAL\n" + "2: ILOAD\n" + "IFNE Jumps to: B3(true) B4(false) B8(Exception:java.io.IOException) B10(Exception:!UncaughtException!) \n" + "B3\n" + "0: ALOAD\n" + "1: INVOKESPECIAL\n" + "Jumps to: B5 B8(Exception:java.io.IOException) B10(Exception:!UncaughtException!) \n" + "B4\n" + "0: ALOAD\n" + "1: INVOKESPECIAL\n" + "GOTO Jumps to: B5 B8(Exception:java.io.IOException) B10(Exception:!UncaughtException!) \n" + "B5\n" + "0: GETSTATIC\n" + "1: LDC\n" + "2: INVOKEVIRTUAL\n" + "Jumps to: B6 B8(Exception:java.io.IOException) B10(Exception:!UncaughtException!) \n" + "B6\n" + "0: GETSTATIC\n" + "1: LDC\n" + "2: INVOKEVIRTUAL\n" + "GOTO Jumps to: B7 \n" + "B7\n" + "0: RETURN\n" + "Jumps to: B0 \n" + "B8\n" + "0: ASTORE\n" + "1: ALOAD\n" + "2: INVOKEVIRTUAL\n" + "Jumps to: B9 B10(Exception:!UncaughtException!) \n" + "B9\n" + "0: GETSTATIC\n" + "1: LDC\n" + "2: INVOKEVIRTUAL\n" + "GOTO Jumps to: B7 \n" + "B10\n" + "0: ASTORE\n" + "1: GETSTATIC\n" + "2: LDC\n" + "3: INVOKEVIRTUAL\n" + "4: ALOAD\n" + "5: ATHROW\n" + "Jumps to: B0 \n"); } @Test public void last_block_is_ends_with_GOTO() throws Exception { assertCFGforMethod("loopWithoutStopCondition", "B0(Exit)\n" + "B1\n" + "0: ICONST_0\n" + "1: ISTORE\n" + "Jumps to: B2 \n" + "B2\n" + "0: ILOAD\n" + "1: GETSTATIC\n" + "2: ILOAD\n" + "3: IALOAD\n" + "IF_ICMPGT Jumps to: B3(true) B4(false) \n" + "B3\n" + "0: IINC\n" + "GOTO Jumps to: B2 \n" + "B4\n" + "0: ILOAD\n" + "1: ICONST_1\n" + "2: IADD\n" + "3: IRETURN\n" + "Jumps to: B0 \n"); } @Test public void test_class_not_found_logs() throws Exception { SquidClassLoader squidClassLoader = new SquidClassLoader( Lists.newArrayList(new File("target/test-classes"), new File("target/classes"))); BytecodeCFG cfg = SETestUtils.bytecodeCFG("nonsense#foo", squidClassLoader); assertThat(cfg).isNull(); } private void assertCFGforMethod(String methodName, String expectedCFG) { BytecodeCFG cfg = getBytecodeCFG(methodName, "src/test/java/org/sonar/java/bytecode/cfg/BytecodeCFGBuilderTest.java"); StringBuilder sb = new StringBuilder(); cfg.blocks.forEach(b -> sb.append(b.printBlock())); assertThat(sb.toString()).isEqualTo(expectedCFG); } public static BytecodeCFG getBytecodeCFG(String methodName, String filename) { SquidClassLoader squidClassLoader = new SquidClassLoader( Lists.newArrayList(new File("target/test-classes"), new File("target/classes"))); File file = new File(filename); CompilationUnitTree tree = (CompilationUnitTree) JavaParser.createParser().parse(file); SemanticModel.createFor(tree, squidClassLoader); List<Tree> classMembers = ((ClassTree) tree.types().get(0)).members(); Symbol.MethodSymbol symbol = classMembers.stream().filter(m -> m instanceof MethodTree) .map(m -> ((MethodTree) m).symbol()).filter(s -> methodName.equals(s.name())).findFirst() .orElseThrow(IllegalStateException::new); return SETestUtils.bytecodeCFG(symbol.signature(), squidClassLoader); } private void tryCatch() { try { bar(); fun(); } catch (IOException e) { e.printStackTrace(); } finally { System.out.println("finally"); } } private void tryCatchWithMultiplePaths(int i) { try { bar(); if (i == 0) { fun(); } else { fun2(); } System.out.println("endOfTry"); } catch (IOException e) { e.printStackTrace(); } finally { System.out.println("finally"); } } private void bar() { } private void fun() throws IOException { } private void fun2() throws IOException { } final static int[] sizeTable = { 9, 99, 999, 9999, 99999, 999999, 9999999, 99999999, 999999999, Integer.MAX_VALUE }; static int loopWithoutStopCondition(int x) { for (int i = 0;; i++) if (x <= sizeTable[i]) return i + 1; } }