| // Copyright (c) 2018, 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. | 
 | package com.android.tools.r8; | 
 |  | 
 | import static org.junit.Assert.assertEquals; | 
 | import static org.junit.Assert.assertTrue; | 
 |  | 
 | import com.android.tools.r8.ClassFileConsumer.ArchiveConsumer; | 
 | import com.android.tools.r8.ToolHelper.ProcessResult; | 
 | import com.android.tools.r8.utils.DescriptorUtils; | 
 | import java.nio.file.Path; | 
 | import org.junit.Test; | 
 | import org.objectweb.asm.AnnotationVisitor; | 
 | import org.objectweb.asm.ClassWriter; | 
 | import org.objectweb.asm.FieldVisitor; | 
 | import org.objectweb.asm.Label; | 
 | import org.objectweb.asm.MethodVisitor; | 
 | import org.objectweb.asm.Opcodes; | 
 |  | 
 | /** Jacoco does invalid instrumentation when R8 clobber locals of arguments: TODO(b/117589870) */ | 
 | public class JacocoRegressionTest extends TestBase implements Opcodes { | 
 |  | 
 |   @Test | 
 |   public void test() throws Exception { | 
 |     Path output = temp.newFolder().toPath(); | 
 |     String name = "Test"; | 
 |     String desc = DescriptorUtils.javaTypeToDescriptor(name); | 
 |     byte[] bytes = dump(); | 
 |     Path path = output.resolve("out.jar"); | 
 |     ArchiveConsumer archiveConsumer = new ArchiveConsumer(path); | 
 |     archiveConsumer.accept(ByteDataView.of(bytes), desc, null); | 
 |     archiveConsumer.finished(null); | 
 |  | 
 |     String expected = "15" + System.lineSeparator(); | 
 |     ProcessResult result = ToolHelper.runJava(path, name); | 
 |     assertEquals(expected, result.stdout); | 
 |  | 
 |     Path agentOutput = output.resolve("agent.out"); | 
 |     ProcessResult result1 = | 
 |         ToolHelper.runJava( | 
 |             path, | 
 |             String.format( | 
 |                 "-javaagent:%s=destfile=%s,dumponexit=true,output=file", | 
 |                 ToolHelper.JACOCO_AGENT, agentOutput), | 
 |             name); | 
 |     assertEquals(1, result1.exitCode); | 
 |     assertTrue(result1.toString().contains("java.lang.VerifyError: Bad local variable type")); | 
 |   } | 
 |  | 
 |   public static byte[] dump() throws Exception { | 
 |  | 
 |     ClassWriter classWriter = new ClassWriter(0); | 
 |     FieldVisitor fieldVisitor; | 
 |     MethodVisitor methodVisitor; | 
 |     AnnotationVisitor annotationVisitor0; | 
 |  | 
 |     classWriter.visit( | 
 |         V1_8, ACC_FINAL | ACC_SUPER | ACC_PUBLIC, "Test", null, "java/lang/Object", null); | 
 |     classWriter.visitSource("Test.java", null); | 
 |  | 
 |     { | 
 |       methodVisitor = classWriter.visitMethod(ACC_PRIVATE, "<init>", "()V", null, null); | 
 |       methodVisitor.visitCode(); | 
 |       Label label0 = new Label(); | 
 |       methodVisitor.visitLabel(label0); | 
 |       methodVisitor.visitLineNumber(32, label0); | 
 |       methodVisitor.visitVarInsn(ALOAD, 0); | 
 |       methodVisitor.visitMethodInsn(INVOKESPECIAL, "java/lang/Object", "<init>", "()V", false); | 
 |       methodVisitor.visitInsn(RETURN); | 
 |       Label label1 = new Label(); | 
 |       methodVisitor.visitLabel(label1); | 
 |       methodVisitor.visitLocalVariable("this", "LTest;", null, label0, label1, 0); | 
 |       methodVisitor.visitMaxs(1, 1); | 
 |       methodVisitor.visitEnd(); | 
 |     } | 
 |  | 
 |     // This is the problematic part for Jacoco, where we clobber local 1 with long_2nd | 
 |     { | 
 |       methodVisitor = classWriter.visitMethod(ACC_STATIC, "foo", "(I)I", null, null); | 
 |       methodVisitor.visitCode(); | 
 |       methodVisitor.visitVarInsn(ILOAD, 0); | 
 |       methodVisitor.visitInsn(I2L); | 
 |       methodVisitor.visitVarInsn(LSTORE, 0); | 
 |       methodVisitor.visitIntInsn(BIPUSH, 15); | 
 |       methodVisitor.visitInsn(IRETURN); | 
 |       methodVisitor.visitMaxs(4, 4); | 
 |       methodVisitor.visitEnd(); | 
 |     } | 
 |  | 
 |     { | 
 |       methodVisitor = | 
 |           classWriter.visitMethod( | 
 |               ACC_PUBLIC | ACC_STATIC, "main", "([Ljava/lang/String;)V", null, null); | 
 |       methodVisitor.visitCode(); | 
 |       Label label0 = new Label(); | 
 |       methodVisitor.visitLabel(label0); | 
 |       methodVisitor.visitLineNumber(6, label0); | 
 |       methodVisitor.visitFieldInsn(GETSTATIC, "java/lang/System", "out", "Ljava/io/PrintStream;"); | 
 |       methodVisitor.visitIntInsn(BIPUSH, 42); | 
 |       methodVisitor.visitMethodInsn(INVOKESTATIC, "Test", "foo", "(I)I", false); | 
 |       methodVisitor.visitMethodInsn(INVOKEVIRTUAL, "java/io/PrintStream", "println", "(I)V", false); | 
 |       Label label1 = new Label(); | 
 |       methodVisitor.visitLabel(label1); | 
 |       methodVisitor.visitLineNumber(7, label1); | 
 |       methodVisitor.visitInsn(RETURN); | 
 |       methodVisitor.visitMaxs(2, 1); | 
 |       methodVisitor.visitEnd(); | 
 |     } | 
 |  | 
 |     classWriter.visitEnd(); | 
 |  | 
 |     return classWriter.toByteArray(); | 
 |   } | 
 | } |