kmath/benchmarks/build.gradle.kts

280 lines
9.5 KiB
Plaintext
Raw Normal View History

2024-08-21 11:56:05 +03:00
import com.fasterxml.jackson.module.kotlin.jacksonObjectMapper
import com.fasterxml.jackson.module.kotlin.readValue
import kotlinx.benchmark.gradle.BenchmarksExtension
import java.time.LocalDateTime
import java.time.ZoneId
import java.util.*
2021-06-15 13:18:40 +07:00
2021-04-16 22:43:10 +03:00
plugins {
kotlin("multiplatform")
alias(spclibs.plugins.kotlin.plugin.allopen)
2024-08-21 11:56:05 +03:00
alias(spclibs.plugins.kotlinx.benchmark)
2021-04-16 22:43:10 +03:00
}
allOpen.annotation("org.openjdk.jmh.annotations.State")
sourceSets.register("benchmarks")
repositories {
mavenCentral()
}
kotlin {
2024-10-04 14:55:15 +03:00
jvmToolchain(17)
compilerOptions {
optIn.addAll(
"space.kscience.kmath.UnstableKMathAPI"
)
}
2021-04-16 22:43:10 +03:00
jvm()
2022-01-25 23:02:35 +07:00
js(IR) {
nodejs()
}
2021-04-16 22:43:10 +03:00
sourceSets {
2022-01-25 23:02:35 +07:00
all {
languageSettings {
progressiveMode = true
2023-05-12 20:57:55 +03:00
optIn("kotlin.contracts.ExperimentalContracts")
optIn("kotlin.ExperimentalUnsignedTypes")
optIn("space.kscience.kmath.UnstableKMathAPI")
2022-01-25 23:02:35 +07:00
}
}
2021-04-16 22:43:10 +03:00
val commonMain by getting {
dependencies {
implementation(project(":kmath-ast"))
implementation(project(":kmath-core"))
implementation(project(":kmath-coroutines"))
implementation(project(":kmath-complex"))
implementation(project(":kmath-stat"))
implementation(project(":kmath-dimensions"))
implementation(project(":kmath-for-real"))
2021-10-06 12:25:32 +03:00
implementation(project(":kmath-tensors"))
2024-08-09 10:22:37 +03:00
implementation(libs.multik.default)
implementation(spclibs.kotlinx.benchmark.runtime)
2021-04-16 22:43:10 +03:00
}
}
val jvmMain by getting {
dependencies {
2025-01-12 11:40:51 +03:00
implementation(projects.kmathCommons)
implementation(projects.kmathEjml)
implementation(projects.kmathNd4j)
implementation(projects.kmathKotlingrad)
implementation(projects.kmathViktor)
implementation(projects.kmathOjalgo)
2022-02-17 22:46:17 +03:00
implementation(projects.kmath.kmathTensorflow)
2025-01-12 11:40:51 +03:00
implementation(projects.kmathMultik)
2022-02-17 22:46:17 +03:00
implementation("org.tensorflow:tensorflow-core-platform:0.4.0")
implementation("org.nd4j:nd4j-native:1.0.0-M1")
2021-04-16 22:43:10 +03:00
// uncomment if your system supports AVX2
// val os = System.getProperty("os.name")
//
// if (System.getProperty("os.arch") in arrayOf("x86_64", "amd64")) when {
// os.startsWith("Windows") -> implementation("org.nd4j:nd4j-native:1.0.0-beta7:windows-x86_64-avx2")
// os == "Linux" -> implementation("org.nd4j:nd4j-native:1.0.0-beta7:linux-x86_64-avx2")
// os == "Mac OS X" -> implementation("org.nd4j:nd4j-native:1.0.0-beta7:macosx-x86_64-avx2")
// } else
// implementation("org.nd4j:nd4j-native-platform:1.0.0-beta7")
}
}
}
}
// Configure benchmark
benchmark {
// Setup configurations
targets {
register("jvm")
2024-07-07 11:02:49 +03:00
register("js")
2021-04-16 22:43:10 +03:00
}
fun kotlinx.benchmark.gradle.BenchmarkConfiguration.commonConfiguration() {
warmups = 2
iterations = 5
iterationTime = 2000
iterationTimeUnit = "ms"
}
2021-04-16 22:43:10 +03:00
configurations.register("buffer") {
commonConfiguration()
2021-04-16 22:43:10 +03:00
include("BufferBenchmark")
}
2021-10-06 12:25:32 +03:00
configurations.register("nd") {
commonConfiguration()
include("NDFieldBenchmark")
}
2021-04-16 22:43:10 +03:00
configurations.register("dot") {
commonConfiguration()
2021-04-16 22:43:10 +03:00
include("DotBenchmark")
}
configurations.register("expressions") {
2022-01-25 23:02:35 +07:00
// Some extra precision
warmups = 2
iterations = 10
2022-01-26 20:33:44 +07:00
iterationTime = 10
iterationTimeUnit = "s"
outputTimeUnit = "s"
2021-04-16 22:43:10 +03:00
include("ExpressionsInterpretersBenchmark")
}
configurations.register("matrixInverse") {
commonConfiguration()
2021-04-16 22:43:10 +03:00
include("MatrixInverseBenchmark")
}
configurations.register("bigInt") {
commonConfiguration()
2021-04-16 22:43:10 +03:00
include("BigIntBenchmark")
}
2021-06-08 19:19:55 +05:30
configurations.register("jafamaDouble") {
commonConfiguration()
include("JafamaBenchmark")
}
2021-09-21 21:24:27 +03:00
2022-02-20 02:21:52 +03:00
configurations.register("tensorAlgebra") {
commonConfiguration()
include("TensorAlgebraBenchmark")
}
2021-09-21 21:24:27 +03:00
configurations.register("viktor") {
commonConfiguration()
include("ViktorBenchmark")
}
configurations.register("viktorLog") {
commonConfiguration()
include("ViktorLogBenchmark")
}
2023-02-03 19:32:53 +03:00
configurations.register("integration") {
commonConfiguration()
include("IntegrationBenchmark")
}
2021-04-16 22:43:10 +03:00
}
2024-08-21 11:56:05 +03:00
private data class JmhReport(
val jmhVersion: String,
val benchmark: String,
val mode: String,
val threads: Int,
val forks: Int,
val jvm: String,
val jvmArgs: List<String>,
val jdkVersion: String,
val vmName: String,
val vmVersion: String,
val warmupIterations: Int,
val warmupTime: String,
val warmupBatchSize: Int,
val measurementIterations: Int,
val measurementTime: String,
val measurementBatchSize: Int,
val params: Map<String, String> = emptyMap(),
val primaryMetric: PrimaryMetric,
val secondaryMetrics: Map<String, SecondaryMetric>,
) {
interface Metric {
val score: Double
val scoreError: Double
val scoreConfidence: List<Double>
val scorePercentiles: Map<Double, Double>
val scoreUnit: String
}
data class PrimaryMetric(
override val score: Double,
override val scoreError: Double,
override val scoreConfidence: List<Double>,
override val scorePercentiles: Map<Double, Double>,
override val scoreUnit: String,
val rawDataHistogram: List<List<List<List<Double>>>>? = null,
val rawData: List<List<Double>>? = null,
) : Metric
data class SecondaryMetric(
override val score: Double,
override val scoreError: Double,
override val scoreConfidence: List<Double>,
override val scorePercentiles: Map<Double, Double>,
override val scoreUnit: String,
val rawData: List<List<Double>>,
) : Metric
}
2021-04-16 22:43:10 +03:00
readme {
2022-07-29 15:58:02 +03:00
maturity = space.kscience.gradle.Maturity.EXPERIMENTAL
2021-06-15 13:18:40 +07:00
2024-08-21 11:56:05 +03:00
val jsonMapper = jacksonObjectMapper()
fun noun(number: Number, singular: String, plural: String) = if (number.toLong() == 1L) singular else plural
extensions.findByType(BenchmarksExtension::class.java)?.configurations?.forEach { cfg ->
2024-08-26 13:37:49 +03:00
val propertyName =
"benchmark${cfg.name.replaceFirstChar { if (it.isLowerCase()) it.titlecase(Locale.getDefault()) else it.toString() }}"
logger.info("Processing benchmark data from benchmark ${cfg.name} into readme property $propertyName")
2024-08-21 11:56:05 +03:00
2024-08-26 13:37:49 +03:00
val launches = layout.buildDirectory.dir("reports/benchmarks/${cfg.name}").get().asFile
if (!launches.exists()) return@forEach
property(propertyName) {
val resDirectory = launches.listFiles()?.maxByOrNull {
LocalDateTime.parse(it.name).atZone(ZoneId.systemDefault()).toInstant()
2024-08-21 11:56:05 +03:00
}
if (resDirectory == null || !(resDirectory.resolve("jvm.json")).exists()) {
"> **Can't find appropriate benchmark data. Try generating readme files after running benchmarks**."
} else {
val reports: List<JmhReport> =
jsonMapper.readValue<List<JmhReport>>(resDirectory.resolve("jvm.json"))
buildString {
2024-08-26 13:37:49 +03:00
appendLine("## Report for benchmark configuration <code>${cfg.name}</code>")
2024-08-21 11:56:05 +03:00
appendLine()
val first = reports.first()
appendLine("* Run on ${first.vmName} (build ${first.vmVersion}) with Java process:")
appendLine()
appendLine("```")
appendLine(
"${first.jvm} ${
first.jvmArgs.joinToString(" ")
}"
)
appendLine("```")
appendLine(
"* JMH ${first.jmhVersion} was used in `${first.mode}` mode with ${first.warmupIterations} warmup ${
noun(first.warmupIterations, "iteration", "iterations")
} by ${first.warmupTime} and ${first.measurementIterations} measurement ${
noun(first.measurementIterations, "iteration", "iterations")
} by ${first.measurementTime}."
)
2024-08-26 13:37:49 +03:00
reports.groupBy { it.benchmark.substringBeforeLast(".") }.forEach { (cl, compare) ->
appendLine("### [${cl.substringAfterLast(".")}](src/jvmMain/kotlin/${cl.replace(".","/")}.kt)")
appendLine()
appendLine("| Benchmark | Score |")
appendLine("|:---------:|:-----:|")
compare.forEach { report ->
val benchmarkName = report.benchmark.substringAfterLast(".")
val score = String.format("%.2G", report.primaryMetric.score)
val error = String.format("%.2G", report.primaryMetric.scoreError)
appendLine("|`$benchmarkName`|$score &plusmn; $error ${report.primaryMetric.scoreUnit}|")
}
2024-08-21 11:56:05 +03:00
}
}
}
}
}
}