8298809: Clean up vm/compiler/InterfaceCalls JMH

Reviewed-by: kvn
This commit is contained in:
Eric Caspole 2022-12-16 16:30:31 +00:00
parent 3696711efa
commit 81e23ab340

View File

@ -1,5 +1,5 @@
/* /*
* Copyright (c) 2014, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 2014, 2022, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
* *
* This code is free software; you can redistribute it and/or modify it * This code is free software; you can redistribute it and/or modify it
@ -44,250 +44,132 @@ import java.util.concurrent.TimeUnit;
@Fork(value = 3) @Fork(value = 3)
public class InterfaceCalls { public class InterfaceCalls {
interface AnInterface { interface FirstInterface {
public int getInt(); public int getIntFirst();
} }
interface SecondInterface { interface SecondInterface {
public int get1(); public int getIntSecond();
} }
interface OnlyHasOneImplInterface { class FirstClass implements FirstInterface, SecondInterface {
public int getLong(); public int getIntFirst() {
} return 1;
}
interface AloneInterface { public int getIntSecond() {
public int getNumber();
}
class SingleImplementor implements OnlyHasOneImplInterface {
public int getLong() {
return 1; return 1;
} }
} }
class Extender1 extends SingleImplementor { class SecondClass implements FirstInterface, SecondInterface {
} public int getIntFirst() {
class FirstClass implements AnInterface {
public int getInt() {
return 1;
}
}
class SecondClass implements AnInterface {
public int getInt() {
return 2; return 2;
} }
public int getIntSecond() {
return 1;
}
} }
class ThirdClass implements AnInterface { class ThirdClass implements FirstInterface, SecondInterface {
public int getInt() { public int getIntFirst() {
return -3; return -3;
} }
}
class FourthClass implements AnInterface { public int getIntSecond() {
public int getInt() {
return -4;
}
}
class FifthClass implements AnInterface {
public int getInt() {
return -5;
}
}
class MultiClass1 implements AnInterface, SecondInterface {
public int get1() {
return 1; return 1;
} }
}
public int getInt() { class FourthClass implements FirstInterface, SecondInterface {
return 2; public int getIntFirst() {
return -4;
}
public int getIntSecond() {
return 1;
} }
} }
class MultiClass2 implements AnInterface, SecondInterface { class FifthClass implements FirstInterface, SecondInterface {
public int get1() { public int getIntFirst() {
return -1; return -5;
} }
public int getInt() { public int getIntSecond() {
return -2; return 1;
} }
} }
class Aloner implements AloneInterface { final int asLength = 5;
public int getNumber() { public FirstInterface[] as = new FirstInterface[asLength];
return 7;
}
}
public Object dummy1;
public Object dummy2;
public Object dummy3;
public AnInterface multi1a, multi2a;
public SecondInterface multi1b, multi2b;
public Object multic, multic2;
public AnInterface[] as = new AnInterface[5];
public AnInterface multi;
public OnlyHasOneImplInterface single1;
public OnlyHasOneImplInterface single2;
public AloneInterface alone;
int count;
@Setup @Setup
public void setupSubclass() { public void setupSubclass() {
dummy1 = new FirstClass();
dummy2 = new SecondClass();
dummy3 = new ThirdClass();
as[0] = new FirstClass(); as[0] = new FirstClass();
as[1] = new SecondClass(); as[1] = new SecondClass();
as[2] = new ThirdClass(); as[2] = new ThirdClass();
as[3] = new FourthClass(); as[3] = new FourthClass();
as[4] = new FifthClass(); as[4] = new FifthClass();
MultiClass1 mc1 = new MultiClass1();
multi1a = mc1;
multi1b = mc1;
multic = mc1;
MultiClass2 mc2 = new MultiClass2();
multi2a = mc2;
multi2b = mc2;
multic2 = mc2;
single1 = new SingleImplementor();
single2 = new Extender1();
alone = new Aloner();
}
private void swapMultiParts() {
AnInterface tmpa = multi1a;
SecondInterface tmpb = multi1b;
multi1a = multi2a;
multi2a = tmpa;
multi1b = multi2b;
multi2b = tmpb;
}
@SuppressWarnings("unused")
private void swapMulti() {
Object tmp = multic;
multic = multic2;
multic2 = tmp;
} }
/** /**
* Tests a call where there are multiple implementors but only one of the implementors is every used here so the * Tests a call where there are multiple implementors but only one of the
* call-site is monomorphic * implementors is every used here so the call-site is monomorphic
*/ */
@Benchmark @Benchmark
public int testMonomorphic() { public int testMonomorphic() {
return as[0].getInt(); return as[0].getIntFirst();
} }
/** Tests a interface call that only has one single implementation */ int l = 0;
@Benchmark
public int testSingle() {
return alone.getNumber();
}
/** /**
* Tests a call where there is a single implementation but multiple classes that inherit that implementation and both * Interface call address computation within loop but the receiver preexists
* these implementors are used. * the loop and the ac can be moved outside of the loop
*/ */
@Benchmark @Benchmark
public int testSingle2() { public int test1stInt2Types() {
OnlyHasOneImplInterface oi; FirstInterface ai = as[l];
if ((count & 1) == 0) {
oi = single1;
} else {
oi = single2;
}
count++;
return oi.getLong();
}
/**
* Tests calling two different interface methods in two different interfaces on the same receiver. Make sure to switch
* between two different types of receivers to achieve polymorhpism
*/
@Benchmark
public void testCall2Poly2(Blackhole bh) {
bh.consume(multi1a.getInt());
bh.consume(multi1b.get1());
swapMultiParts();
}
@Benchmark
public int testCallMulti1Poly2NoSwap() {
return multi1a.getInt();
}
/**
* This test goes together with Multi2 below It tests if a class implements multiple interfaces if the different
* interfaces take different amounts of time (They do for hotspot)
*/
@Benchmark
public int testCallMulti1Poly2() {
swapMultiParts();
return multi1a.getInt();
}
/**
* This test goes together with Multi2 below It tests if a class implements multiple interfaces if the different
* interfaces take different amounts of time (They do for hotspot)
*/
@Benchmark
public int testCallMulti2Poly2() {
swapMultiParts();
return multi1b.get1();
}
/** Interface call with three different receivers */
@Benchmark
public void testCallPoly3(Blackhole bh) {
for (int kk = 0; kk < 3; kk++) {
bh.consume(as[kk].getInt());
}
}
/** Interface call with five different receivers. */
@Benchmark
public void testCallPoly5(Blackhole bh) {
for (int kk = 0; kk < 5; kk++) {
bh.consume(as[kk].getInt());
}
}
int l;
/**
* Interface call address computation within loop but the receiver preexists the loop and the ac can be moved outside
* of the loop
*/
@Benchmark
public int testAC1() {
AnInterface ai = as[l];
l = 1 - l; l = 1 - l;
return ai.getInt(); return ai.getIntFirst();
} }
/** Tests an interface cast followed by an interface call. */
@Benchmark @Benchmark
public int testInterfaceCastAndCall() throws Exception { public int test1stInt3Types() {
return ((AnInterface) dummy1).getInt() + ((AnInterface) dummy2).getInt() FirstInterface ai = as[l];
+ ((AnInterface) dummy3).getInt(); l = ++ l % 3;
return ai.getIntFirst();
} }
@Benchmark
public int test1stInt5Types() {
FirstInterface ai = as[l];
l = ++ l % asLength;
return ai.getIntFirst();
}
@Benchmark
public int test2ndInt2Types() {
SecondInterface ai = (SecondInterface) as[l];
l = 1 - l;
return ai.getIntSecond();
}
@Benchmark
public int test2ndInt3Types() {
SecondInterface ai = (SecondInterface) as[l];
l = ++ l % 3;
return ai.getIntSecond();
}
@Benchmark
public int test2ndInt5Types() {
SecondInterface ai = (SecondInterface) as[l];
l = ++ l % asLength;
return ai.getIntSecond();
}
} }