blob: 8f1fd7e9cbc6f80a08ea4e7331494546cc9e665f [file] [log] [blame]
#
# Copyright 2016 The Android Open Source Project
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
#
import sys
max_conflict_depth = 20 # In practice does not go above 20 for reasonable IMT sizes
try:
imt_size = int(sys.argv[1])
except (IndexError, ValueError):
print("Usage: python ImtConflictBenchmarkGen.py <IMT_SIZE>")
sys.exit(1)
license = """\
/*
* Copyright 2016 The Android Open Source Project
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
"""
description = """
/**
* This file is script-generated by ImtConflictBenchmarkGen.py.
* It measures the performance impact of conflicts in interface method tables.
* Run `python ImtConflictBenchmarkGen.py > ImtConflictBenchmark.java` to regenerate.
*
* Each interface has 64 methods, which is the current size of an IMT. C0 implements
* one interface, C1 implements two, C2 implements three, and so on. The intent
* is that C0 has no conflicts in its IMT, C1 has depth-2 conflicts in
* its IMT, C2 has depth-3 conflicts, etc. This is currently guaranteed by
* the fact that we hash interface methods by taking their method index modulo 64.
* (Note that a "conflict depth" of 1 means no conflict at all.)
*/\
"""
print(license)
print("package benchmarks;")
print("import com.google.caliper.BeforeExperiment;")
print(description)
print("public class ImtConflictBenchmark {")
# Warm up interface method tables
print(" @BeforeExperiment")
print(" public void setup() {")
for i in xrange(max_conflict_depth):
print(" C{0} c{0} = new C{0}();".format(i))
for j in xrange(i+1):
print(" callF{}(c{});".format(imt_size * j, i))
print(" }")
# Print test cases--one for each conflict depth
for i in xrange(max_conflict_depth):
print(" public void timeConflictDepth{:02d}(int nreps) {{".format(i+1))
print(" C{0} c{0} = new C{0}();".format(i))
print(" for (int i = 0; i < nreps; i++) {")
# Cycle through each interface method in an IMT entry in order
# to test all conflict resolution possibilities
for j in xrange(max_conflict_depth):
print(" callF{}(c{});".format(imt_size * (j % (i + 1)), i))
print(" }")
print(" }")
# Make calls through the IMTs
for i in xrange(max_conflict_depth):
print(" public void callF{0}(I{1} i) {{ i.f{0}(); }}".format(imt_size*i, i))
# Class definitions, implementing varying amounts of interfaces
for i in xrange(max_conflict_depth):
interfaces = ", ".join(["I{}".format(j) for j in xrange(i+1)])
print(" static class C{} implements {} {{}}".format(i, interfaces))
# Interface definitions, each with enough methods to fill an entire IMT
for i in xrange(max_conflict_depth):
print(" static interface I{} {{".format(i))
for j in xrange(imt_size):
print(" default void f{}() {{}}".format(i*imt_size + j))
print(" }")
print "}"