blob: 650953cf71e1b1644bca48ddefd48b2015367fef [file] [log] [blame]
// Copyright (c) 2021, 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.retrace.api;
import static org.junit.Assert.assertEquals;
import com.android.tools.r8.TestParameters;
import com.android.tools.r8.references.ClassReference;
import com.android.tools.r8.references.MethodReference;
import com.android.tools.r8.references.Reference;
import com.android.tools.r8.retrace.MappingProvider;
import com.android.tools.r8.retrace.ProguardMapProducer;
import com.android.tools.r8.retrace.ProguardMappingProvider;
import com.android.tools.r8.retrace.RetraceFrameElement;
import com.android.tools.r8.retrace.RetraceStackTraceContext;
import com.android.tools.r8.retrace.RetracedMethodReference;
import com.android.tools.r8.retrace.RetracedSingleFrame;
import com.android.tools.r8.retrace.Retracer;
import java.util.List;
import java.util.OptionalInt;
import java.util.stream.Collectors;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.junit.runners.Parameterized;
@RunWith(Parameterized.class)
public class RetraceApiOutlineInOutlineStackTrace extends RetraceApiTestBase {
public RetraceApiOutlineInOutlineStackTrace(TestParameters parameters) {
super(parameters);
}
@Override
protected Class<? extends RetraceApiBinaryTest> binaryTestClass() {
return ApiTest.class;
}
public static class ApiTest implements RetraceApiBinaryTest {
private final ClassReference outline1Renamed = Reference.classFromTypeName("a");
private final ClassReference outline2Renamed = Reference.classFromTypeName("b");
private final ClassReference callsiteOriginal = Reference.classFromTypeName("some.Class");
private final ClassReference callsiteRenamed = Reference.classFromTypeName("c");
private final String mapping =
"# { id: 'com.android.tools.r8.mapping', version: '2.0' }\n"
+ "outline1.Class -> "
+ outline1Renamed.getTypeName()
+ ":\n"
+ " 3:4:int outline():0:0 -> a\n"
+ "# { 'id':'com.android.tools.r8.outline' }\n"
+ "outline2.Class -> "
+ outline2Renamed.getTypeName()
+ ":\n"
+ " 6:6:int outline():0:0 -> a\n"
+ "# { 'id':'com.android.tools.r8.outlineCallsite', "
+ " 'positions': { '3': 42, '4': 43 } }\n"
+ " 42:43:int outline():0:0 -> a\n" // This is another call to the outline
+ "# { 'id':'com.android.tools.r8.outline' }\n"
+ callsiteOriginal.getTypeName()
+ " -> "
+ callsiteRenamed.getTypeName()
+ ":\n"
+ " 1:1:void foo.bar.Baz.qux():42:42 -> s\n"
+ " 10:11:int foo.bar.baz.outlineCaller(int):98:99 -> s\n"
+ " 28:28:int outlineCaller(int):0:0 -> s\n" // This is the actual call to the outline
+ "# { 'id':'com.android.tools.r8.outlineCallsite', "
+ " 'positions': { '42': 10, '43': 11 } }\n";
@Test
public void test() {
MappingProvider mappingProvider =
ProguardMappingProvider.builder()
.setProguardMapProducer(ProguardMapProducer.fromString(mapping))
.allowLookupAllClasses()
.build();
Retracer retracer = Retracer.builder().setMappingProvider(mappingProvider).build();
// Retrace the first outline.
RetraceStackTraceContext outlineContext =
retraceOutline(
retracer,
Reference.methodFromDescriptor(outline1Renamed, "a", "()I"),
4,
RetraceStackTraceContext.empty());
// Retrace the second outline using the context of the first retracing.
outlineContext =
retraceOutline(
retracer,
Reference.methodFromDescriptor(outline2Renamed, "a", "()I"),
6,
outlineContext);
List<RetraceFrameElement> retraceOutlineCallee =
retracer
.retraceFrame(
outlineContext,
OptionalInt.of(28),
Reference.methodFromDescriptor(callsiteRenamed, "s", "(I)V"))
.stream()
.collect(Collectors.toList());
assertEquals(1, retraceOutlineCallee.size());
List<RetracedMethodReference> outlineCallSiteFrames =
retraceOutlineCallee.get(0).stream()
.map(RetracedSingleFrame::getMethodReference)
.collect(Collectors.toList());
assertEquals(1, outlineCallSiteFrames.size());
assertEquals(99, outlineCallSiteFrames.get(0).getOriginalPositionOrDefault(28));
}
private RetraceStackTraceContext retraceOutline(
Retracer retracer,
MethodReference reference,
int position,
RetraceStackTraceContext context) {
List<RetraceFrameElement> outlineRetraced =
retracer.retraceFrame(context, OptionalInt.of(position), reference).stream()
.collect(Collectors.toList());
// The retrace result should not be ambiguous or empty.
assertEquals(1, outlineRetraced.size());
RetraceFrameElement retraceFrameElement = outlineRetraced.get(0);
// Check that visiting all frames report the outline.
List<RetracedMethodReference> allMethodReferences =
retraceFrameElement.stream()
.map(RetracedSingleFrame::getMethodReference)
.collect(Collectors.toList());
assertEquals(1, allMethodReferences.size());
assertEquals(0, allMethodReferences.get(0).getOriginalPositionOrDefault(position));
// Check that visiting rewritten frames will not report anything.
List<RetracedMethodReference> rewrittenReferences =
retraceFrameElement
.streamRewritten(RetraceStackTraceContext.empty())
.map(RetracedSingleFrame::getMethodReference)
.collect(Collectors.toList());
assertEquals(0, rewrittenReferences.size());
return retraceFrameElement.getRetraceStackTraceContext();
}
}
}