diff --git a/.github/workflows/build-and-run-tests.yml b/.github/workflows/build-and-run-tests.yml index 1c1eae220d..ada025afd2 100644 --- a/.github/workflows/build-and-run-tests.yml +++ b/.github/workflows/build-and-run-tests.yml @@ -40,7 +40,7 @@ jobs: DEST_DIR="arkanalyzer" MAX_RETRIES=10 RETRY_DELAY=3 # Delay between retries in seconds - BRANCH="neo/2024-08-07" + BRANCH="neo/2024-12-04" for ((i=1; i<=MAX_RETRIES; i++)); do git clone --depth=1 --branch $BRANCH $REPO_URL $DEST_DIR && break diff --git a/.gitignore b/.gitignore index 3322abd66b..adbbc452c4 100644 --- a/.gitignore +++ b/.gitignore @@ -15,3 +15,6 @@ buildSrc/.gradle # Ignore Python execution cache __pycache__/ run_python_with_gdb.sh + +# Ignore Kotlin build directory +.kotlin/ diff --git a/buildSrc/build.gradle.kts b/buildSrc/build.gradle.kts index cbd46db18e..5075d846cd 100644 --- a/buildSrc/build.gradle.kts +++ b/buildSrc/build.gradle.kts @@ -12,9 +12,8 @@ repositories { maven("https://jitpack.io") } - dependencies { implementation("org.jetbrains.kotlin:kotlin-gradle-plugin:$kotlinVersion") implementation("io.gitlab.arturbosch.detekt:detekt-gradle-plugin:$detektVersion") implementation("org.glavo:gjavah:$gjavahVersion") -} \ No newline at end of file +} diff --git a/buildSrc/settings.gradle.kts b/buildSrc/settings.gradle.kts index c6e4cfd742..27752ba2fd 100644 --- a/buildSrc/settings.gradle.kts +++ b/buildSrc/settings.gradle.kts @@ -1,2 +1 @@ rootProject.name="usvm-conventions" - diff --git a/buildSrc/src/main/kotlin/Dependencies.kt b/buildSrc/src/main/kotlin/Dependencies.kt index b9c3771821..0ad33fec6e 100644 --- a/buildSrc/src/main/kotlin/Dependencies.kt +++ b/buildSrc/src/main/kotlin/Dependencies.kt @@ -3,7 +3,8 @@ import org.gradle.plugin.use.PluginDependenciesSpec object Versions { - const val detekt = "1.18.1" + const val clikt = "5.0.0" + const val detekt = "1.23.7" const val ini4j = "0.5.4" const val jacodb = "ad5e1f170e" const val juliet = "1.3.2" @@ -18,7 +19,7 @@ object Versions { const val mockk = "1.13.4" const val rd = "2023.2.0" const val sarif4k = "0.5.0" - const val shadow = "8.1.1" + const val shadow = "8.3.3" const val slf4j = "1.6.1" // versions for jvm samples @@ -121,6 +122,11 @@ object Libs { name = "jacodb-core", version = Versions.jacodb ) + val jacodb_api_common = dep( + group = jacodbPackage, + name = "jacodb-api-common", + version = Versions.jacodb + ) val jacodb_api_jvm = dep( group = jacodbPackage, name = "jacodb-api-jvm", @@ -136,16 +142,6 @@ object Libs { name = "jacodb-storage", version = Versions.jacodb ) - val jacodb_ets = dep( - group = jacodbPackage, - name = "jacodb-ets", - version = Versions.jacodb - ) - val jacodb_api_common = dep( - group = jacodbPackage, - name = "jacodb-api-common", - version = Versions.jacodb - ) val jacodb_approximations = dep( group = jacodbPackage, name = "jacodb-approximations", @@ -156,6 +152,11 @@ object Libs { name = "jacodb-taint-configuration", version = Versions.jacodb ) + val jacodb_ets = dep( + group = jacodbPackage, + name = "jacodb-ets", + version = Versions.jacodb + ) // https://github.com/Kotlin/kotlinx.coroutines val kotlinx_coroutines_core = dep( @@ -240,6 +241,13 @@ object Libs { name = "PythonTypesAPI", version = Versions.pythonTypesAPI ) + + // https://github.com/ajalt/clikt + val clikt = dep( + group = "com.github.ajalt.clikt", + name = "clikt", + version = Versions.clikt + ) } object Plugins { @@ -258,9 +266,9 @@ object Plugins { version = Versions.rd ) - // https://github.com/johnrengelman/shadow + // https://github.com/GradleUp/shadow object Shadow : ProjectPlugin( - id = "com.github.johnrengelman.shadow", + id = "com.gradleup.shadow", version = Versions.shadow ) } diff --git a/buildSrc/src/main/kotlin/usvmpython/tasks/CPythonBuildTasks.kt b/buildSrc/src/main/kotlin/usvmpython/tasks/CPythonBuildTasks.kt index 3eaefc8b6c..5610cb5112 100644 --- a/buildSrc/src/main/kotlin/usvmpython/tasks/CPythonBuildTasks.kt +++ b/buildSrc/src/main/kotlin/usvmpython/tasks/CPythonBuildTasks.kt @@ -127,4 +127,4 @@ fun Project.registerCPythonDistClean(): TaskProvider { commandLine(windowsBuildScript.canonicalPath, "-t", "CleanAll") } } -} \ No newline at end of file +} diff --git a/buildSrc/src/main/kotlin/usvmpython/tasks/JNIHeaderTask.kt b/buildSrc/src/main/kotlin/usvmpython/tasks/JNIHeaderTask.kt index cf8559de73..847d463911 100644 --- a/buildSrc/src/main/kotlin/usvmpython/tasks/JNIHeaderTask.kt +++ b/buildSrc/src/main/kotlin/usvmpython/tasks/JNIHeaderTask.kt @@ -17,4 +17,4 @@ fun Project.generateJNIForCPythonAdapterTask() { } task.addClass(CPYTHON_ADAPTER_CLASS) task.run() -} \ No newline at end of file +} diff --git a/buildSrc/src/main/kotlin/usvmpython/tasks/PythonSamplesTasks.kt b/buildSrc/src/main/kotlin/usvmpython/tasks/PythonSamplesTasks.kt index f405d46cb0..a486687aec 100644 --- a/buildSrc/src/main/kotlin/usvmpython/tasks/PythonSamplesTasks.kt +++ b/buildSrc/src/main/kotlin/usvmpython/tasks/PythonSamplesTasks.kt @@ -51,4 +51,4 @@ fun Project.registerBuildSamplesTask(): TaskProvider { environment("PYTHONHOME" to cpythonBuildPath) mainClass.set(BUILD_SAMPLES_ENTRY_POINT) } -} \ No newline at end of file +} diff --git a/gradle/wrapper/gradle-wrapper.jar b/gradle/wrapper/gradle-wrapper.jar index d64cd49177..2c3521197d 100644 Binary files a/gradle/wrapper/gradle-wrapper.jar and b/gradle/wrapper/gradle-wrapper.jar differ diff --git a/gradle/wrapper/gradle-wrapper.properties b/gradle/wrapper/gradle-wrapper.properties index a80b22ce5c..7cf748e743 100644 --- a/gradle/wrapper/gradle-wrapper.properties +++ b/gradle/wrapper/gradle-wrapper.properties @@ -1,6 +1,6 @@ distributionBase=GRADLE_USER_HOME distributionPath=wrapper/dists -distributionUrl=https\://services.gradle.org/distributions/gradle-8.6-bin.zip +distributionUrl=https\://services.gradle.org/distributions/gradle-8.11-all.zip networkTimeout=10000 validateDistributionUrl=true zipStoreBase=GRADLE_USER_HOME diff --git a/gradlew b/gradlew index 1aa94a4269..f5feea6d6b 100755 --- a/gradlew +++ b/gradlew @@ -15,6 +15,8 @@ # See the License for the specific language governing permissions and # limitations under the License. # +# SPDX-License-Identifier: Apache-2.0 +# ############################################################################## # @@ -55,7 +57,7 @@ # Darwin, MinGW, and NonStop. # # (3) This script is generated from the Groovy template -# https://github.com/gradle/gradle/blob/HEAD/subprojects/plugins/src/main/resources/org/gradle/api/internal/plugins/unixStartScript.txt +# https://github.com/gradle/gradle/blob/HEAD/platforms/jvm/plugins-application/src/main/resources/org/gradle/api/internal/plugins/unixStartScript.txt # within the Gradle project. # # You can find Gradle at https://github.com/gradle/gradle/. @@ -84,7 +86,8 @@ done # shellcheck disable=SC2034 APP_BASE_NAME=${0##*/} # Discard cd standard output in case $CDPATH is set (https://github.com/gradle/gradle/issues/25036) -APP_HOME=$( cd "${APP_HOME:-./}" > /dev/null && pwd -P ) || exit +APP_HOME=$( cd -P "${APP_HOME:-./}" > /dev/null && printf '%s +' "$PWD" ) || exit # Use the maximum available, or set MAX_FD != -1 to use that value. MAX_FD=maximum diff --git a/gradlew.bat b/gradlew.bat index 25da30dbde..9d21a21834 100644 --- a/gradlew.bat +++ b/gradlew.bat @@ -13,6 +13,8 @@ @rem See the License for the specific language governing permissions and @rem limitations under the License. @rem +@rem SPDX-License-Identifier: Apache-2.0 +@rem @if "%DEBUG%"=="" @echo off @rem ########################################################################## diff --git a/settings.gradle.kts b/settings.gradle.kts index 121c7fcbd5..7e98903912 100644 --- a/settings.gradle.kts +++ b/settings.gradle.kts @@ -8,6 +8,7 @@ include("usvm-jvm-instrumentation") include("usvm-sample-language") include("usvm-dataflow") include("usvm-jvm-dataflow") +include("usvm-dataflow-ts") include("usvm-python") include("usvm-python:cpythonadapter") @@ -21,6 +22,11 @@ findProject(":usvm-python:usvm-python-runner")?.name = "usvm-python-runner" include("usvm-python:usvm-python-commons") findProject(":usvm-python:usvm-python-commons")?.name = "usvm-python-commons" +// Actually, `includeBuild("../jacodb")` is enough, but there is a bug in IDEA when path is a symlink. +// As a workaround, we convert it to a real absolute path. +// See IDEA bug: https://youtrack.jetbrains.com/issue/IDEA-329756 +// includeBuild(file("../jacodb").toPath().toRealPath().toAbsolutePath()) + pluginManagement { resolutionStrategy { eachPlugin { diff --git a/usvm-dataflow-ts/.gitignore b/usvm-dataflow-ts/.gitignore new file mode 100644 index 0000000000..333c1e910a --- /dev/null +++ b/usvm-dataflow-ts/.gitignore @@ -0,0 +1 @@ +logs/ diff --git a/usvm-dataflow-ts/README.md b/usvm-dataflow-ts/README.md new file mode 100644 index 0000000000..59dd9cae7f --- /dev/null +++ b/usvm-dataflow-ts/README.md @@ -0,0 +1,167 @@ +# USVM Dataflow TS + +## Type Inference + +In order to run type inference on an arbitrary TypeScript project, you need the following: +1. IR dumped into JSON files: either from TS sources or from +binary ABC/HAP files. +2. USVM with type inference CLI: `usvm-dataflow-ts-all.jar` "fat" JAR. + +**NOTE:** the instructions below are given for Linux. If you are using Windows, you need to adjust the paths and commands accordingly, or use WSL. Overall, the process should be similar, and USVM should work on any platform that supports Java. + +### ArkTS IR + +- Below, we use the term "ArkIR" to refer to the representation of ArkTS inside ArkAnalyzer in a form of TypeScript classes and interfaces, such as `ArkMethod`, `ArkAssignStmt`, `ArkInstanceInvokeExpr`. + +- In USVM, we also have a similar model of representing ArkTS, but in the form of Java/Kotlin classes. In order to differentiate between the two models, we use the prefix "Ets" for the classes in USVM, such as `EtsMethod`, `EtsAssignStmt`, `EtsInstanceCallExpr`. + +### Setup ArkAnalyzer + +First of all, you need to clone the ArkAnalyzer repo. Here, we use the fork of the repo and the specific branch (named `neo/`) that is consistent with USVM internals. Note that this branch might change in the future. +```bash +cd ~ +git clone -b neo/2024-10-31 https://gitee.com/Lipenx/arkanalyzer arkanalyzer-usvm +cd arkanalyzer-usvm +``` + +Then, you need to install the dependencies and build the project. +```bash +npm install +npm run build +``` + +**Note:** after building the ArkAnalyzer project, the script for serializing ArkIR will be located at `out/save/serializeArkIR.js` and can be run with Node.js. + +**Note:** you can also use TS script directly using `npx ts-node src/save/serializeArkIR.ts` instead of building the whole project. + +### Serialize ArkIR to JSON + +Now, you can run the `serializeArkIR` script on your TS project in order to construct its ArkIR representation and dump it into JSON files, which can later be used by USVM. +```bash +node ~/arkanalyzer-usvm/out/src/save/serializeArkIR.js --help +``` +```text +Usage: serializeArkIR [options] + +Serialize ArkIR for TypeScript files or projects to JSON + +Arguments: + input Input file or directory + output Output file or directory + +Options: + -p, --project Flag to indicate the input is a project directory (default: false) + -t, --infer-types [times] Infer types in the ArkIR + -v, --verbose Verbose output (default: false) + -h, --help display help for command +``` + +If you have a single TS file `sample.ts`, just run the following command: +```bash +node .../serializeArkIR.js sample.ts sample.json +``` +The resulting `sample.json` file will contain the ArkIR in JSON format. + +If you have a TS project in the `project` directory, use `-p` flag: +```bash +node .../serializeArkIR.js -p project etsir +``` +The resulting `etsir` directory will contain the ArkIR in JSON format. The structure of the resulting directory (hierarchy of subfolders) will be the same as the structure of the input project, but all the files will be `*.ts.json`. + +_Note:_ We call the result "EtsIR" since it is a modified version of the ArkIR model suitable for serialization. When we load IR from JSONs in USVM (Java/Kotlin), the resulting data model (structure of classes) is very similar to ArkIR in ArkAnalyzer (TypeScript), but has some minor differences. The term "EtsIR" is used to distinguish between the two. + +If you have a TS project with multiple modules, run the serialization for each module separately: +```bash +node .../serializeArkIR.js -p project/entry etsir/entry +node .../serializeArkIR.js -p project/common etsir/common +node .../serializeArkIR.js -p project/feature etsir/feature +``` + +### Type Inference with USVM + +In order to run USVM type inference, you need to obtain `usvm-dataflow-ts-all.jar` "fat" JAR (download or build it yourself) and either use it directly or use a wrapper script `src/usvm/inferTypes.ts` in ArkAnalyzer repo. + +#### Build `usvm-type-inference` binary + +In order to build the USVM binary, you need to clone the USVM repo (and also its dependency `jacodb` in the _sibling directory_) and build the project using Gradle. +```bash +cd ~ +git clone -b lipen/usvm-type-inference https://github.com/UnitTestBot/jacodb +git clone -b lipen/type-inference https://github.com/UnitTestBot/usvm +cd usvm +./gradlew :usvm-dataflow-ts:installDist +``` +The last command will build the project and create the binary at `usvm-dataflow-ts/build/install/usvm-dataflow-ts/bin/usvm-type-inference` (on Windows, the corresponding "binary" is with `.bat` extension). + +#### Build "Fat" JAR + +Alternatively, you can build the "fat" JAR (also known as "Uber JAR" or "shadow JAR") that contains all the dependencies. +```bash +./gradlew :usvm-dataflow-ts:shadowJar +``` + +#### Run Type Inference + +You can run the type inference manually using USVM CLI: +```bash +usvm-dataflow-ts/build/install/usvm-dataflow-ts/bin/usvm-type-inference --help +# OR +java -jar usvm-dataflow-ts/build/libs/usvm-dataflow-ts-all.jar --help +``` +```text +Usage: infer-types [] + +Options: +* -i, --input= Input file or directory with IR (required) +* -o, --output= Output file with inferred types in JSON format (required) + -h, --help Show this message and exit +``` + +_Note:_ `-i` option can be supplied multiple times for multi-module projects. All input IR will be merged. + +For example, if you have the `project/entry` and `project/common` directories with the dumped ArkIR, you can run the following command: +```bash +java -jar usvm-dataflow-ts/build/libs/usvm-dataflow-ts-all.jar -i project/entry -i project/common -o inferred.json +``` + +### Type Inference with Wrapper Script + +You can also use the wrapper script `src/usvm/inferTypes.ts` from the ArkAnalyzer repo. This script will run the serialization of ArkIR and type inference with USVM in a single command. + +```bash +node ~/arkanalyzer-usvm/out/src/usvm/inferTypes.js --help +``` +```text +Usage: inferTypes [options] + +Arguments: + input input directory with ETS project + +Options: + -v, --verbose Verbose output (default: false) + -t, --aa-types Run type inference in ArkAnalyzer (default: false) + -s, --substitute Substitute inferred types (default: false) + -h, --help display help for command +``` + +For example: +```bash +node .../inferTypes.js myproject/entry +``` +```text +Building scene... +Serializing Scene to '/tmp/2f8aa8b34548b808167a8f6b30121dcc/etsir'... +... +USVM command: ~/usvm/usvm-dataflow-ts/build/install/usvm-dataflow-ts/bin/usvm-type-inference --input=/tmp/2f8aa8b34548b808167a8f6b30121dcc/etsir --output=/tmp/2f8aa8b34548b808167a8f6b30121dcc/inference-result --no-skip-anonymous +... +=== Inferred Types Statistics === +Total Classes: 10 +Total Methods: 305 +... +Deserialization successful. +... +Substituting inferred types... +... +Substituting type of local '$temp16' in method '@entry/model/Calculator.ts: _DEFAULT_ARK_CLASS.getFloatNum(unknown, unknown, unknown)' from unknown to number +... +``` diff --git a/usvm-dataflow-ts/build.gradle.kts b/usvm-dataflow-ts/build.gradle.kts new file mode 100644 index 0000000000..2da676a1e1 --- /dev/null +++ b/usvm-dataflow-ts/build.gradle.kts @@ -0,0 +1,132 @@ +import java.io.FileNotFoundException +import java.nio.file.Files +import java.nio.file.attribute.FileTime + +plugins { + id("usvm.kotlin-conventions") + kotlin("plugin.serialization") version Versions.kotlin + application + id(Plugins.Shadow) + `java-test-fixtures` +} + +dependencies { + api(project(":usvm-dataflow")) + implementation(project(":usvm-util")) + + api(Libs.jacodb_api_common) + api(Libs.jacodb_ets) + implementation(Libs.jacodb_taint_configuration) + implementation(Libs.kotlinx_collections) + implementation(Libs.kotlinx_serialization_json) + implementation(Libs.clikt) + + testImplementation(Libs.mockk) + testImplementation(Libs.junit_jupiter_params) + testImplementation(Libs.logback) + testImplementation(Libs.kotlinx_serialization_core) + + testFixturesImplementation(Libs.kotlin_logging) + testFixturesImplementation(Libs.junit_jupiter_api) +} + +tasks.withType { + maxHeapSize = "4G" +} + +val generateTestResources by tasks.registering { + group = "build" + description = "Generates JSON resources from TypeScript files using ArkAnalyzer." + doLast { + val envVarName = "ARKANALYZER_DIR" + val defaultArkAnalyzerDir = "../arkanalyzer" + + val arkAnalyzerDir = rootDir.resolve(System.getenv(envVarName) ?: run { + println("Please, set $envVarName environment variable. Using default value: '$defaultArkAnalyzerDir'") + defaultArkAnalyzerDir + }) + if (!arkAnalyzerDir.exists()) { + throw FileNotFoundException("ArkAnalyzer directory does not exist: '$arkAnalyzerDir'. Did you forget to set the '$envVarName' environment variable?") + } + + val scriptSubPath = "src/save/serializeArkIR" + val script = arkAnalyzerDir.resolve("out").resolve("$scriptSubPath.js") + if (!script.exists()) { + throw FileNotFoundException("Script file not found: '$script'. Did you forget to execute 'npm run build' in the arkanalyzer project?") + } + + val resources = projectDir.resolve("src/test/resources") + val inputDir = resources.resolve("ts") + val outputDir = resources.resolve("ir") + println("Generating test resources in '${outputDir.relativeTo(projectDir)}'...") + + inputDir.walkTopDown().filter { it.isFile }.forEach { input -> + val output = outputDir + .resolve(input.relativeTo(inputDir)) + .resolveSibling(input.name + ".json") + val inputFileTime = Files.getLastModifiedTime(input.toPath()) + val outputFileTime = if (output.exists()) { + Files.getLastModifiedTime(output.toPath()) + } else { + FileTime.fromMillis(0) + } + + if (!output.exists() || inputFileTime > outputFileTime) { + println("Regenerating JSON for '${output.relativeTo(outputDir)}'...") + + val cmd: List = listOf( + "node", + script.absolutePath, + input.relativeTo(resources).path, + output.relativeTo(resources).path, + ) + println("Running: '${cmd.joinToString(" ")}'") + val process = ProcessBuilder(cmd).directory(resources).start() + val ok = process.waitFor(10, TimeUnit.MINUTES) + + val stdout = process.inputStream.bufferedReader().readText().trim() + if (stdout.isNotBlank()) { + println("[STDOUT]:\n--------\n$stdout\n--------") + } + val stderr = process.errorStream.bufferedReader().readText().trim() + if (stderr.isNotBlank()) { + println("[STDERR]:\n--------\n$stderr\n--------") + } + + if (!ok) { + println("Timeout!") + process.destroy() + } else { + println("Done running!") + } + } else { + println("Skipping '${output.relativeTo(outputDir)}'") + } + } + } +} + +// tasks.test { +// dependsOn(generateTestResources) +// } + +application { + mainClass = "org.usvm.dataflow.ts.infer.cli.InferTypesKt" + applicationDefaultJvmArgs = listOf("-Dfile.encoding=UTF-8", "-Dsun.stdout.encoding=UTF-8") +} + +tasks.startScripts { + applicationName = "usvm-type-inference" +} + +tasks.shadowJar { + minimize { + // Note: keep 'mordant' dependency inside shadowJar, or else the following error occurs: + // ``` + // Exception in thread "main" java.util.ServiceConfigurationError: + // com.github.ajalt.mordant.terminal.TerminalInterfaceProvider: + // Provider com.github.ajalt.mordant.terminal.terminalinterface.jna.TerminalInterfaceProviderJna not found + // ``` + exclude(dependency("com.github.ajalt.mordant:.*:.*")) + } +} diff --git a/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/graph/EtsApplicationGraph.kt b/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/graph/EtsApplicationGraph.kt new file mode 100644 index 0000000000..1b11b5e2e9 --- /dev/null +++ b/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/graph/EtsApplicationGraph.kt @@ -0,0 +1,282 @@ +/* + * Copyright 2022 UnitTestBot contributors (utbot.org) + *

+ * 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. + */ + +package org.usvm.dataflow.ts.graph + +import mu.KotlinLogging +import org.jacodb.ets.base.CONSTRUCTOR_NAME +import org.jacodb.ets.base.EtsAssignStmt +import org.jacodb.ets.base.EtsClassType +import org.jacodb.ets.base.EtsNewExpr +import org.jacodb.ets.base.EtsStmt +import org.jacodb.ets.base.UNKNOWN_FILE_NAME +import org.jacodb.ets.model.EtsClass +import org.jacodb.ets.model.EtsClassSignature +import org.jacodb.ets.model.EtsFileSignature +import org.jacodb.ets.model.EtsMethod +import org.jacodb.ets.model.EtsMethodSignature +import org.jacodb.ets.model.EtsScene +import org.jacodb.ets.utils.Maybe +import org.jacodb.ets.utils.callExpr +import org.jacodb.ets.utils.onSome +import org.usvm.dataflow.graph.ApplicationGraph + +private val logger = KotlinLogging.logger {} + +interface EtsApplicationGraph : ApplicationGraph { + val cp: EtsScene +} + +private fun EtsFileSignature?.isUnknown(): Boolean = + this == null || fileName.isBlank() || fileName == UNKNOWN_FILE_NAME + +private fun EtsClassSignature.isUnknown(): Boolean = + name.isBlank() + +private fun EtsClassSignature.isIdeal(): Boolean = + !isUnknown() && !file.isUnknown() + +enum class ComparisonResult { + Equal, + NotEqual, + Unknown, +} + +fun compareFileSignatures( + sig1: EtsFileSignature?, + sig2: EtsFileSignature?, +): ComparisonResult = when { + sig1.isUnknown() -> ComparisonResult.Unknown + sig2.isUnknown() -> ComparisonResult.Unknown + sig1?.fileName == sig2?.fileName -> ComparisonResult.Equal + else -> ComparisonResult.NotEqual +} + +fun compareClassSignatures( + sig1: EtsClassSignature, + sig2: EtsClassSignature, +): ComparisonResult = when { + sig1.isUnknown() -> ComparisonResult.Unknown + sig2.isUnknown() -> ComparisonResult.Unknown + sig1.name == sig2.name -> compareFileSignatures(sig1.file, sig2.file) + else -> ComparisonResult.NotEqual +} + +class EtsApplicationGraphImpl( + override val cp: EtsScene, +) : EtsApplicationGraph { + + override fun predecessors(node: EtsStmt): Sequence { + val graph = node.method.flowGraph() + val predecessors = graph.predecessors(node) + return predecessors.asSequence() + } + + override fun successors(node: EtsStmt): Sequence { + val graph = node.method.flowGraph() + val successors = graph.successors(node) + return successors.asSequence() + } + + private val cacheClassWithIdealSignature: MutableMap> = hashMapOf() + private val cacheMethodWithIdealSignature: MutableMap> = hashMapOf() + private val cachePartiallyMatchedCallees: MutableMap> = hashMapOf() + + private fun lookupClassWithIdealSignature(signature: EtsClassSignature): Maybe { + require(signature.isIdeal()) + + if (signature in cacheClassWithIdealSignature) { + return cacheClassWithIdealSignature.getValue(signature) + } + + val matched = cp.projectAndSdkClasses + .asSequence() + .filter { it.signature == signature } + .toList() + if (matched.isEmpty()) { + cacheClassWithIdealSignature[signature] = Maybe.none() + return Maybe.none() + } else { + val s = matched.singleOrNull() + ?: error("Multiple classes with the same signature: $matched") + cacheClassWithIdealSignature[signature] = Maybe.some(s) + return Maybe.some(s) + } + } + + override fun callees(node: EtsStmt): Sequence { + val expr = node.callExpr ?: return emptySequence() + + val callee = expr.method + + // Note: the resolving code below expects that at least the current method signature is known. + check(node.method.enclosingClass.isIdeal()) { + "Incomplete signature in method: ${node.method}" + } + + // Note: specific resolve for constructor: + if (callee.name == CONSTRUCTOR_NAME) { + if (!callee.enclosingClass.isIdeal()) { + val prevStmt = predecessors(node).singleOrNull() + if (prevStmt == null) { + // Constructor call is the first statement in the method. + // We can't resolve it without the class signature. + return emptySequence() + } + + if (prevStmt is EtsAssignStmt && prevStmt.rhv is EtsNewExpr) { + val cls = prevStmt.rhv.type + if (cls !is EtsClassType) { + return emptySequence() + } + + val sig = cls.signature + if (sig.isIdeal()) { + lookupClassWithIdealSignature(sig).onSome { c -> + return sequenceOf(c.ctor) + } + } else { + val resolved = cp.projectAndSdkClasses + .asSequence() + .filter { compareClassSignatures(it.signature, sig) != ComparisonResult.NotEqual } + .singleOrNull() + if (resolved != null) { + return sequenceOf(resolved.ctor) + } + } + } + + // Constructor signature is garbage. Sorry, can't do anything in such case. + return emptySequence() + } + + // Here, we assume that the constructor signature is ideal. + check(callee.enclosingClass.isIdeal()) + + val cls = lookupClassWithIdealSignature(callee.enclosingClass) + if (cls.isSome) { + return sequenceOf(cls.getOrThrow().ctor) + } else { + return emptySequence() + } + } + + // If the callee signature is ideal, resolve it directly: + if (callee.enclosingClass.isIdeal()) { + if (callee in cacheMethodWithIdealSignature) { + val resolved = cacheMethodWithIdealSignature.getValue(callee) + if (resolved.isSome) { + return sequenceOf(resolved.getOrThrow()) + } else { + return emptySequence() + } + } + + val cls = lookupClassWithIdealSignature(callee.enclosingClass) + + val resolved = run { + if (cls.isNone) { + emptySequence() + } else { + cls.getOrThrow().methods.asSequence().filter { it.name == callee.name } + } + } + if (resolved.none()) { + cacheMethodWithIdealSignature[callee] = Maybe.none() + return emptySequence() + } + val r = resolved.singleOrNull() + ?: error("Multiple methods with the same complete signature: ${resolved.toList()}") + cacheMethodWithIdealSignature[callee] = Maybe.some(r) + return sequenceOf(r) + } + + // If the callee signature is not ideal, resolve it via a partial match... + check(!callee.enclosingClass.isIdeal()) + + val cls = lookupClassWithIdealSignature(node.method.enclosingClass).let { + if (it.isNone) { + error("Could not find the enclosing class: ${node.method.enclosingClass}") + } + it.getOrThrow() + } + + // If the complete signature match failed, + // try to find the unique not-the-same neighbour method in the same class: + val neighbors = cls.methods + .asSequence() + .filter { it.name == callee.name } + .filterNot { it.name == node.method.name } + .toList() + if (neighbors.isNotEmpty()) { + val s = neighbors.singleOrNull() + ?: error("Multiple methods with the same name: $neighbors") + cachePartiallyMatchedCallees[callee] = listOf(s) + return sequenceOf(s) + } + + // NOTE: cache lookup MUST be performed AFTER trying to match the neighbour! + if (callee in cachePartiallyMatchedCallees) { + return cachePartiallyMatchedCallees.getValue(callee).asSequence() + } + + // If the neighbour match failed, + // try to *uniquely* resolve the callee via a partial signature match: + val resolved = cp.projectAndSdkClasses + .asSequence() + .filter { compareClassSignatures(it.signature, callee.enclosingClass) != ComparisonResult.NotEqual } + // Note: exclude current class: + .filterNot { compareClassSignatures(it.signature, node.method.enclosingClass) != ComparisonResult.NotEqual } + // Note: omit constructors! + .flatMap { it.methods.asSequence() } + .filter { it.name == callee.name } + .toList() + if (resolved.isEmpty()) { + cachePartiallyMatchedCallees[callee] = emptyList() + return emptySequence() + } + val r = resolved.singleOrNull() ?: run { + logger.warn { "Multiple methods with the same partial signature '$callee': $resolved" } + cachePartiallyMatchedCallees[callee] = emptyList() + return emptySequence() + } + cachePartiallyMatchedCallees[callee] = listOf(r) + return sequenceOf(r) + } + + override fun callers(method: EtsMethod): Sequence { + // Note: currently, nobody uses `callers`, so if is safe to disable it for now. + // Note: comparing methods by signature may be incorrect, and comparing only by name fails for constructors. + TODO("disabled for now, need re-design") + // return cp.classes.asSequence() + // .flatMap { it.methods } + // .flatMap { it.cfg.instructions } + // .filterIsInstance() + // .filter { it.expr.method == method.signature } + } + + override fun entryPoints(method: EtsMethod): Sequence { + return method.flowGraph().entries.asSequence() + } + + override fun exitPoints(method: EtsMethod): Sequence { + return method.flowGraph().exits.asSequence() + } + + override fun methodOf(node: EtsStmt): EtsMethod { + return node.location.method + } +} diff --git a/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/ifds/UnitResolver.kt b/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/ifds/UnitResolver.kt new file mode 100644 index 0000000000..ba9bba9971 --- /dev/null +++ b/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/ifds/UnitResolver.kt @@ -0,0 +1,62 @@ +/* + * Copyright 2022 UnitTestBot contributors (utbot.org) + *

+ * 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. + */ + +package org.usvm.dataflow.ts.ifds + +import org.jacodb.ets.model.EtsClassSignature +import org.jacodb.ets.model.EtsMethod +import org.jacodb.ets.model.EtsMethodSignature +import org.usvm.dataflow.ifds.SingletonUnit +import org.usvm.dataflow.ifds.UnitResolver +import org.usvm.dataflow.ifds.UnitType + +data class MethodUnit(val method: EtsMethodSignature) : UnitType { + override fun toString(): String { + return "MethodUnit(${method.name})" + } +} + +data class ClassUnit(val clazz: EtsClassSignature) : UnitType { + override fun toString(): String { + return "ClassUnit(${clazz.name})" + } +} + +// TODO: PackageUnit +// data class PackageUnit(val packageName: String) : UnitType { +// override fun toString(): String { +// return "PackageUnit($packageName)" +// } +// } + +fun interface EtsUnitResolver : UnitResolver + +val MethodUnitResolver = EtsUnitResolver { method -> + MethodUnit(method.signature) +} + +val ClassUnitResolver = EtsUnitResolver { method -> + ClassUnit(method.signature.enclosingClass) +} + +// TODO: PackageUnitResolver +// val PackageUnitResolver = EtsUnitResolver { method -> +// PackageUnit(method.enclosingClass.packageName) +// } + +val SingletonUnitResolver = EtsUnitResolver { + SingletonUnit +} diff --git a/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/AccessPath.kt b/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/AccessPath.kt new file mode 100644 index 0000000000..6544a6f090 --- /dev/null +++ b/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/AccessPath.kt @@ -0,0 +1,98 @@ +package org.usvm.dataflow.ts.infer + +import org.jacodb.ets.base.EtsArrayAccess +import org.jacodb.ets.base.EtsCastExpr +import org.jacodb.ets.base.EtsConstant +import org.jacodb.ets.base.EtsEntity +import org.jacodb.ets.base.EtsInstanceFieldRef +import org.jacodb.ets.base.EtsLocal +import org.jacodb.ets.base.EtsParameterRef +import org.jacodb.ets.base.EtsStaticFieldRef +import org.jacodb.ets.base.EtsThis +import org.jacodb.ets.base.EtsValue + +data class AccessPath(val base: AccessPathBase, val accesses: List) { + operator fun plus(accessor: Accessor) = AccessPath(base, accesses + accessor) + operator fun plus(accessors: List) = AccessPath(base, accesses + accessors) + + override fun toString(): String { + return base.toString() + accesses.joinToString("") { it.toSuffix() } + } +} + +fun List.startsWith(other: List): Boolean { + return this.take(other.size) == other +} + +fun AccessPath?.startsWith(other: AccessPath?): Boolean { + if (this == null || other == null) { + return false + } + if (this.base != other.base) { + return false + } + return this.accesses.startsWith(other.accesses) +} + +fun List.hasDuplicateFields(limit: Int = 2): Boolean { + val counts = this.groupingBy { it }.eachCount() + return counts.any { it.value >= limit } +} + +sealed interface AccessPathBase { + object This : AccessPathBase { + override fun toString(): String = "" + } + + object Static : AccessPathBase { + override fun toString(): String = "" + } + + data class Arg(val index: Int) : AccessPathBase { + override fun toString(): String = "arg($index)" + } + + data class Local(val name: String) : AccessPathBase { + override fun toString(): String = "local($name)" + } + + data class Const(val constant: EtsConstant) : AccessPathBase { + override fun toString(): String = "const($constant)" + } +} + +fun EtsValue.toBase(): AccessPathBase = when (this) { + is EtsConstant -> AccessPathBase.Const(this) + is EtsLocal -> if (name == "this") AccessPathBase.This else AccessPathBase.Local(name) + is EtsThis -> AccessPathBase.This + is EtsParameterRef -> AccessPathBase.Arg(index) + else -> error("$this is not access path base") +} + +fun EtsEntity.toPathOrNull(): AccessPath? = when (this) { + is EtsConstant -> AccessPath(toBase(), emptyList()) + + is EtsLocal -> AccessPath(toBase(), emptyList()) + + is EtsThis -> AccessPath(toBase(), emptyList()) + + is EtsParameterRef -> AccessPath(toBase(), emptyList()) + + is EtsArrayAccess -> array.toPathOrNull()?.let { + it + ElementAccessor + } + + is EtsInstanceFieldRef -> instance.toPathOrNull()?.let { + it + FieldAccessor(field.name) + } + + is EtsStaticFieldRef -> AccessPath(AccessPathBase.Static, listOf(FieldAccessor(field.name))) + + is EtsCastExpr -> arg.toPathOrNull() + + else -> null +} + +fun EtsEntity.toPath(): AccessPath { + return toPathOrNull() ?: error("Unable to build access path for value $this") +} diff --git a/usvm-jvm-dataflow/src/samples/java/NullAssumptionAnalysisExample.java b/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/Accessors.kt similarity index 63% rename from usvm-jvm-dataflow/src/samples/java/NullAssumptionAnalysisExample.java rename to usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/Accessors.kt index abdcec3540..7473b06672 100644 --- a/usvm-jvm-dataflow/src/samples/java/NullAssumptionAnalysisExample.java +++ b/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/Accessors.kt @@ -14,16 +14,20 @@ * limitations under the License. */ -public class NullAssumptionAnalysisExample { - public void test1(String a) { - System.out.println("Hello from test1"); - System.out.println(a.length()); - } +package org.usvm.dataflow.ts.infer - public void test2(Object a) { - System.out.println("Hello from test2"); - System.out.println(a.hashCode()); - String x = (String) a; - System.out.println(x.length()); - } +sealed interface Accessor { + fun toSuffix(): String +} + +data class FieldAccessor( + val name: String, +) : Accessor { + override fun toSuffix(): String = ".$name" + override fun toString(): String = name +} + +object ElementAccessor : Accessor { + override fun toSuffix(): String = "[*]" + override fun toString(): String = "*" } diff --git a/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/Alias.kt b/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/Alias.kt new file mode 100644 index 0000000000..1669320d9c --- /dev/null +++ b/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/Alias.kt @@ -0,0 +1,324 @@ +/* + * Copyright 2022 UnitTestBot contributors (utbot.org) + *

+ * 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. + */ + +package org.usvm.dataflow.ts.infer + +import kotlinx.collections.immutable.PersistentMap +import kotlinx.collections.immutable.mutate +import kotlinx.collections.immutable.persistentHashMapOf +import org.jacodb.ets.base.EtsAssignStmt +import org.jacodb.ets.base.EtsCallExpr +import org.jacodb.ets.base.EtsCastExpr +import org.jacodb.ets.base.EtsConstant +import org.jacodb.ets.base.EtsLocal +import org.jacodb.ets.base.EtsNewArrayExpr +import org.jacodb.ets.base.EtsNewExpr +import org.jacodb.ets.base.EtsParameterRef +import org.jacodb.ets.base.EtsRef +import org.jacodb.ets.base.EtsStmt +import org.jacodb.ets.model.EtsMethod + +sealed interface Allocation { + class New : Allocation + data class Arg(val index: Int) : Allocation + class CallResult : Allocation + class Imm : Allocation +} + +class AliasInfo( + // B: Base -> Object + val baseToAlloc: PersistentMap, + // F: Object x Field -> Object + val allocToFields: PersistentMap>, +) { + // A: Base x Field* -> Object + fun find(path: AccessPath): Allocation? { + val b = baseToAlloc[path.base] ?: return null + if (path.accesses.isEmpty()) return b + // TODO: handle multiple accesses + check(path.accesses.size == 1) + return when (val a = path.accesses.single()) { + is FieldAccessor -> { + val f = allocToFields[b] ?: return null + f[a.name] + } + + is ElementAccessor -> { + null + } + } + } + + fun merge(other: AliasInfo): AliasInfo { + // Intersect B: + val newB = persistentHashMapOf().mutate { newB -> + for ((base, obj) in this.baseToAlloc) { + if (other.baseToAlloc[base] == obj) { + newB[base] = obj + } + } + } + + // Intersect F: + val newF = persistentHashMapOf>().mutate { newF -> + for ((obj, fields) in this.allocToFields) { + newF[obj] = persistentHashMapOf().mutate { newFields -> + val otherFields = other.allocToFields[obj] + if (otherFields != null) { + for ((field, alloc) in fields) { + if (otherFields[field] == alloc) { + newFields[field] = alloc + } + } + } + } + } + } + + return AliasInfo(newB, newF) + } + + /** + * Returns the set of access paths that must alias with the given path (excluding the path itself). + */ + fun getAliases(path: AccessPath): Set { + val obj = find(path) ?: return emptySet() + return getAliases(obj) - path + } + + private fun getAliases(obj: Allocation): Set { + val paths = mutableSetOf() + + val allocToFields = hashMapOf>>() + for ((obj1, fields) in this.allocToFields) { + for ((field, obj2) in fields) { + allocToFields.computeIfAbsent(obj2) { hashMapOf() } + allocToFields.computeIfAbsent(obj2) { hashMapOf() } + .computeIfAbsent(field) { mutableListOf() } + .add(obj1) + } + } + + val allocToBases = hashMapOf>() + for ((base, alloc) in baseToAlloc) { + allocToBases.computeIfAbsent(alloc) { mutableListOf() } + .add(base) + } + + val queue = ArrayDeque>>(listOf(obj to emptyList())) + while (queue.isNotEmpty()) { + val (cur, path) = queue.removeFirst() + // TODO: eliminate loops as in computeAliases via DFS with PATH/STACK + // TODO: think about loop-edges + if (path.size > MAX_PATH_SIZE) continue + if (cur in allocToBases) { + for (base in allocToBases.getValue(cur)) { + paths.add(AccessPath(base, path.reversed())) + } + } + if (cur in allocToFields) { + for ((field, objs) in allocToFields.getValue(cur)) { + for (alloc in objs) { + queue.add(alloc to path + FieldAccessor(field)) + } + } + } + } + + return paths + } + + companion object { + // Maximum number of fields in an access path. Used to avoid infinite loops. + private const val MAX_PATH_SIZE = 10 + } +} + +fun computeAliases(method: EtsMethod): Map> { + val preAliases = mutableMapOf() + val postAliases = mutableMapOf() + + val visited: MutableSet = hashSetOf() + val order: MutableList = mutableListOf() + val preds: MutableMap> = hashMapOf() + + fun postOrderDfs(node: EtsStmt) { + if (visited.add(node)) { + for (next in method.cfg.successors(node)) { + if (next !in visited) { + preds.computeIfAbsent(next) { mutableListOf() } += node + } + postOrderDfs(next) + } + order += node + } + } + + val root = method.cfg.stmts[0] + postOrderDfs(root) + order.reverse() + + fun computePostAliases(stmt: EtsStmt): AliasInfo { + if (stmt in postAliases) return postAliases.getValue(stmt) + + val pre = preAliases.getValue(stmt) + var newF = pre.allocToFields + val newB = pre.baseToAlloc.mutate { newB -> + newF = newF.mutate { newF -> + if (stmt is EtsAssignStmt) { + val lhv = stmt.lhv + val rhv = stmt.rhv + + if (rhv is EtsLocal || rhv is EtsRef || (rhv is EtsCastExpr && rhv.arg is EtsRef)) { + val lhs = lhv.toPath() + val rhs = rhv.toPath() + + if (rhv is EtsParameterRef) { + check(lhs.accesses.isEmpty()) + newB[lhs.base] = Allocation.Arg(rhv.index) + } else { + if (lhs.accesses.isEmpty() && rhs.accesses.isEmpty()) { + // x := y + newB[lhs.base] = newB.computeIfAbsent(rhs.base) { Allocation.Imm() } + } else if (lhs.accesses.isEmpty()) { + // x := y.f OR x := y[i] + when (val a = rhs.accesses.single()) { + is FieldAccessor -> { + // x := y.f + val b: Allocation = newB.computeIfAbsent(rhs.base) { Allocation.Imm() } + newF[b] = newF.getOrElse(b) { persistentHashMapOf() }.mutate { f -> + newB[lhs.base] = f.computeIfAbsent(a.name) { Allocation.Imm() } + } + } + + is ElementAccessor -> { + // x := y[i] + newB.remove(lhs.base) + } + } + } else if (rhs.accesses.isEmpty()) { + // x.f := y OR x[i] := y + when (val a = lhs.accesses.single()) { + is FieldAccessor -> { + // x.f := y + val b: Allocation = newB.computeIfAbsent(rhs.base) { Allocation.Imm() } + newF[b] = newF.getOrElse(b) { persistentHashMapOf() }.mutate { f -> + f[a.name] = newB.computeIfAbsent(rhs.base) { Allocation.Imm() } + } + } + + is ElementAccessor -> { + // x[i] := y + // do nothing + } + } + } else { + error("Incorrect 3AC: $stmt") + } + } + } + + if (rhv is EtsConstant || (rhv is EtsCastExpr && rhv.arg is EtsConstant)) { + val lhs = lhv.toPath() + if (lhs.accesses.isEmpty()) { + // x := const + newB.remove(lhs.base) + } else { + when (val a = lhs.accesses.single()) { + is FieldAccessor -> { + // x.f := const + val b = newB.computeIfAbsent(lhs.base) { Allocation.Imm() } + newF[b] = newF.getOrElse(b) { persistentHashMapOf() }.mutate { f -> + f.remove(a.name) + } + } + + is ElementAccessor -> { + // x[i] := const + // do nothing + } + } + } + } + + if (rhv is EtsNewExpr || rhv is EtsNewArrayExpr) { + val lhs = lhv.toPath() + if (lhs.accesses.isEmpty()) { + // x := new() + newB.computeIfAbsent(lhs.base) { Allocation.New() } + } else { + when (val a = lhs.accesses.single()) { + is FieldAccessor -> { + // x.f := new() + val b = newB.computeIfAbsent(lhs.base) { Allocation.Imm() } + newF[b] = newF.getOrElse(b) { persistentHashMapOf() }.mutate { f -> + f[a.name] = Allocation.New() + } + } + + is ElementAccessor -> { + // x[i] := new() + // do nothing + } + } + } + } + + if (rhv is EtsCastExpr) { + check(rhv.arg !is EtsCallExpr) + check(rhv.arg is EtsLocal || rhv.arg is EtsRef || rhv.arg is EtsConstant) + } + + if (rhv is EtsCallExpr) { + val lhs = lhv.toPath() + check(lhs.accesses.isEmpty()) + newB[lhs.base] = Allocation.CallResult() + } + } + } + } + + val newInfo = AliasInfo(newB, newF) + postAliases[stmt] = newInfo + return newInfo + } + + fun computePreAliases(stmt: EtsStmt): AliasInfo { + if (stmt in preAliases) return preAliases.getValue(stmt) + + val merged = preds[stmt] + ?.map { postAliases.getValue(it) } + ?.reduceOrNull { a, b -> a.merge(b) } + ?: AliasInfo(persistentHashMapOf(), persistentHashMapOf()) + + preAliases[stmt] = merged + return merged + } + + for (stmt in order) { + computePreAliases(stmt) + computePostAliases(stmt) + } + + for (stmt in method.cfg.stmts) { + check(stmt in preAliases) + check(stmt in postAliases) + } + + return method.cfg.stmts.associateWithTo(HashMap()) { stmt -> + Pair(preAliases.getValue(stmt), postAliases.getValue(stmt)) + } +} diff --git a/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/AnalyzerEvent.kt b/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/AnalyzerEvent.kt new file mode 100644 index 0000000000..c44ebbb026 --- /dev/null +++ b/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/AnalyzerEvent.kt @@ -0,0 +1,25 @@ +package org.usvm.dataflow.ts.infer + +import org.jacodb.ets.base.EtsStmt +import org.jacodb.ets.model.EtsMethod +import org.usvm.dataflow.ifds.Vertex + +sealed interface AnalyzerEvent + +data class ForwardSummaryAnalyzerEvent( + val method: EtsMethod, + val initialVertex: Vertex, + val exitVertex: Vertex, +) : AnalyzerEvent { + val initialFact get() = initialVertex.fact + val exitFact get() = exitVertex.fact +} + +data class BackwardSummaryAnalyzerEvent( + val method: EtsMethod, + val initialVertex: Vertex, + val exitVertex: Vertex, +) : AnalyzerEvent { + val initialFact get() = initialVertex.fact + val exitFact get() = exitVertex.fact +} diff --git a/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/ApplicationGraph.kt b/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/ApplicationGraph.kt new file mode 100644 index 0000000000..4493764869 --- /dev/null +++ b/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/ApplicationGraph.kt @@ -0,0 +1,78 @@ +/* + * Copyright 2022 UnitTestBot contributors (utbot.org) + *

+ * 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. + */ + +package org.usvm.dataflow.ts.infer + +import org.jacodb.ets.base.EtsInstLocation +import org.jacodb.ets.base.EtsNopStmt +import org.jacodb.ets.base.EtsStmt +import org.jacodb.ets.model.EtsMethod +import org.jacodb.ets.model.EtsScene +import org.usvm.dataflow.ts.graph.EtsApplicationGraph +import org.usvm.dataflow.ts.graph.EtsApplicationGraphImpl + +fun createApplicationGraph(cp: EtsScene): EtsApplicationGraph { + val base = EtsApplicationGraphImpl(cp) + val explicit = EtsApplicationGraphWithExplicitEntryPoint(base) + return explicit +} + +class EtsApplicationGraphWithExplicitEntryPoint( + private val graph: EtsApplicationGraphImpl, +) : EtsApplicationGraph { + + override val cp: EtsScene + get() = graph.cp + + override fun methodOf(node: EtsStmt): EtsMethod = node.location.method + + override fun exitPoints(method: EtsMethod): Sequence = graph.exitPoints(method) + + private fun methodEntryPoint(method: EtsMethod) = + EtsNopStmt(EtsInstLocation(method, index = -1)) + + override fun entryPoints(method: EtsMethod): Sequence = sequenceOf(methodEntryPoint(method)) + + override fun callers(method: EtsMethod): Sequence = graph.callers(method) + + override fun callees(node: EtsStmt): Sequence = graph.callees(node) + + override fun successors(node: EtsStmt): Sequence { + val method = methodOf(node) + val methodEntry = methodEntryPoint(method) + + if (node == methodEntry) { + return graph.entryPoints(method) + } + + return graph.successors(node) + } + + override fun predecessors(node: EtsStmt): Sequence { + val method = methodOf(node) + val methodEntry = methodEntryPoint(method) + + if (node == methodEntry) { + return emptySequence() + } + + if (node in graph.entryPoints(method)) { + return sequenceOf(methodEntry) + } + + return graph.predecessors(node) + } +} diff --git a/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/BackwardAnalyzer.kt b/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/BackwardAnalyzer.kt new file mode 100644 index 0000000000..114b1ca269 --- /dev/null +++ b/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/BackwardAnalyzer.kt @@ -0,0 +1,44 @@ +package org.usvm.dataflow.ts.infer + +import org.jacodb.ets.base.EtsStmt +import org.jacodb.ets.base.EtsType +import org.jacodb.ets.model.EtsMethod +import org.jacodb.impl.cfg.graphs.GraphDominators +import org.usvm.dataflow.graph.ApplicationGraph +import org.usvm.dataflow.ifds.Analyzer +import org.usvm.dataflow.ifds.Edge +import org.usvm.dataflow.ifds.Vertex + +class BackwardAnalyzer( + val graph: ApplicationGraph, + savedTypes: MutableMap>, + dominators: (EtsMethod) -> GraphDominators, +) : Analyzer { + + override val flowFunctions = BackwardFlowFunctions(graph, dominators, savedTypes) + + override fun handleCrossUnitCall( + caller: Vertex, + callee: Vertex, + ): List { + error("No cross unit calls") + } + + override fun handleNewEdge(edge: Edge): List { + val (startVertex, currentVertex) = edge + val (current, currentFact) = currentVertex + + val method = graph.methodOf(current) + val currentIsExit = current in graph.exitPoints(method) + + if (!currentIsExit) return emptyList() + + return listOf( + BackwardSummaryAnalyzerEvent( + method = method, + initialVertex = startVertex, + exitVertex = currentVertex, + ) + ) + } +} diff --git a/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/BackwardFlowFunctions.kt b/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/BackwardFlowFunctions.kt new file mode 100644 index 0000000000..c8867cb1e3 --- /dev/null +++ b/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/BackwardFlowFunctions.kt @@ -0,0 +1,578 @@ +package org.usvm.dataflow.ts.infer + +import mu.KotlinLogging +import org.jacodb.ets.base.EtsAssignStmt +import org.jacodb.ets.base.EtsCastExpr +import org.jacodb.ets.base.EtsEntity +import org.jacodb.ets.base.EtsEqExpr +import org.jacodb.ets.base.EtsIfStmt +import org.jacodb.ets.base.EtsInExpr +import org.jacodb.ets.base.EtsInstanceCallExpr +import org.jacodb.ets.base.EtsLValue +import org.jacodb.ets.base.EtsNewExpr +import org.jacodb.ets.base.EtsNumberConstant +import org.jacodb.ets.base.EtsRef +import org.jacodb.ets.base.EtsReturnStmt +import org.jacodb.ets.base.EtsStmt +import org.jacodb.ets.base.EtsStringConstant +import org.jacodb.ets.base.EtsThrowStmt +import org.jacodb.ets.base.EtsType +import org.jacodb.ets.base.EtsValue +import org.jacodb.ets.model.EtsMethod +import org.jacodb.ets.utils.callExpr +import org.jacodb.impl.cfg.graphs.GraphDominators +import org.usvm.dataflow.graph.ApplicationGraph +import org.usvm.dataflow.ifds.FlowFunction +import org.usvm.dataflow.ifds.FlowFunctions +import org.usvm.dataflow.ts.infer.BackwardTypeDomainFact.TypedVariable +import org.usvm.dataflow.ts.infer.BackwardTypeDomainFact.Zero +import org.usvm.util.Maybe + +private val logger = KotlinLogging.logger {} + +class BackwardFlowFunctions( + val graph: ApplicationGraph, + val dominators: (EtsMethod) -> GraphDominators, + val savedTypes: MutableMap>, +) : FlowFunctions { + + // private val aliasesCache: MutableMap>> = hashMapOf() + // + // private fun getAliases(method: EtsMethod): Map> { + // return aliasesCache.computeIfAbsent(method) { computeAliases(method) } + // } + + override fun obtainPossibleStartFacts(method: EtsMethod) = listOf(Zero) + + override fun obtainSequentFlowFunction( + current: EtsStmt, + next: EtsStmt, + ): FlowFunction = FlowFunction { fact -> + if (current is EtsAssignStmt) { + val lhvPath = current.lhv.toPathOrNull() + val rhvPath = current.rhv.toPathOrNull() + if (lhvPath != null && rhvPath != null && lhvPath == rhvPath) { + return@FlowFunction listOf(fact) + } + } + when (fact) { + Zero -> sequentZero(current) + is TypedVariable -> sequent(current, fact).myFilter() + } + } + + private fun TypedVariable.withTypeGuards(current: EtsStmt): TypedVariable { + val dominators = dominators(current.method).dominators(current).asReversed() + + var result = this + + for (stmt in dominators.filterIsInstance()) { + val (guardedVariable, typeGuard) = resolveTypeGuard(stmt) ?: continue + + if (guardedVariable != result.variable) continue + + val branches = graph.predecessors(stmt).toList() // graph is reversed + check(branches.size == 2) { "Unexpected IF branches: $branches" } + + val (falseBranch, trueBranch) = branches + + val isTrueStatement = current.isReachableFrom(trueBranch) + val isFalseStatement = current.isReachableFrom(falseBranch) + + if (isTrueStatement && !isFalseStatement) { + val type = result.type.withGuard(typeGuard, guardNegated = false) + result = TypedVariable(result.variable, type) + } + + if (!isTrueStatement && isFalseStatement) { + val type = result.type.withGuard(typeGuard, guardNegated = true) + result = TypedVariable(result.variable, type) + } + } + + return result + } + + private fun resolveTypeGuard(branch: EtsIfStmt): Pair? { + val condition = branch.condition as? EtsEqExpr ?: return null + + if (condition.right == EtsNumberConstant(0.0)) { + return resolveTypeGuard(condition.left, branch) + } + + return null + } + + private fun resolveTypeGuard( + value: EtsEntity, + stmt: EtsStmt, + ): Pair? { + val valueAssignment = findAssignment(value, stmt) ?: return null + + return when (val rhv = valueAssignment.rhv) { + is EtsRef, is EtsLValue -> { + resolveTypeGuard(rhv, valueAssignment) + } + + is EtsInExpr -> { + resolveTypeGuardFromIn(rhv.left, rhv.right) + } + + else -> null + } + } + + private fun findAssignment(value: EtsEntity, stmt: EtsStmt): EtsAssignStmt? { + val cache = hashMapOf>() + findAssignment(value, stmt, cache) + val maybeValue = cache.getValue(stmt) + + return if (maybeValue.isNone) null else maybeValue.getOrThrow() + } + + private fun findAssignment( + value: EtsEntity, + stmt: EtsStmt, + cache: MutableMap>, + ) { + if (stmt in cache) return + + if (stmt is EtsAssignStmt && stmt.lhv == value) { + cache[stmt] = Maybe.some(stmt) + return + } + + // val predecessors = graph.successors(stmt) // graph is reversed + // val predecessors = dominators(stmt.method).dominators(stmt) - stmt + val predecessors = dominators(stmt.method).dominators(stmt).toSet().intersect(graph.successors(stmt).toSet()) + predecessors.forEach { findAssignment(value, it, cache) } + + val predecessorValues = predecessors.map { cache.getValue(it) } + if (predecessorValues.any { it.isNone }) { + cache[stmt] = Maybe.none() + return + } + + val values = predecessorValues.map { it.getOrThrow() }.toHashSet() + if (values.size == 1) { + cache[stmt] = Maybe.some(values.single()) + return + } + + cache[stmt] = Maybe.none() + } + + private fun resolveTypeGuardFromIn( + left: EtsEntity, + right: EtsEntity, + ): Pair? { + if (left !is EtsStringConstant) return null + + check(right is EtsValue) + val base = right.toBase() + val type = EtsTypeFact.ObjectEtsTypeFact( + cls = null, + properties = mapOf(left.value to EtsTypeFact.UnknownEtsTypeFact) + ) + return base to type + } + + private fun EtsStmt.isReachableFrom(stmt: EtsStmt): Boolean { + val visited = hashSetOf() + val queue = mutableListOf(stmt) + + while (queue.isNotEmpty()) { + val s = queue.removeLast() + if (this == s) return true + + if (!visited.add(s)) continue + + val successors = graph.predecessors(s) // graph is reversed + queue.addAll(successors) + } + + return false + } + + private fun sequentZero(current: EtsStmt): List { + val result = mutableListOf(Zero) + + // Case `return x` + // ∅ |= x:unknown + if (current is EtsReturnStmt) { + val variable = current.returnValue?.toBase() + if (variable != null) { + result += TypedVariable(variable, EtsTypeFact.UnknownEtsTypeFact) + } + } + + if (current is EtsAssignStmt) { + val rhv = when (val r = current.rhv) { + is EtsRef -> r.toPath() // This, FieldRef, ArrayAccess + is EtsLValue -> r.toPath() // Local + else -> { + // logger.info { "TODO backward assign zero: $current" } + null + } + } + + // When RHS is not const-like, handle possible new facts for RHV: + if (rhv != null) { + val y = rhv.base + + if (rhv.accesses.isEmpty()) { + // Case `x... := y` + // ∅ |= y:unknown + result += TypedVariable(y, EtsTypeFact.UnknownEtsTypeFact) + } else { + // Case `x := y.f` OR `x := y[i]` + + check(rhv.accesses.size == 1) + when (val accessor = rhv.accesses.single()) { + // Case `x := y.f` + // ∅ |= y:{f:unknown} + is FieldAccessor -> { + val type = EtsTypeFact.ObjectEtsTypeFact( + cls = null, + properties = mapOf(accessor.name to EtsTypeFact.UnknownEtsTypeFact) + ) + result += TypedVariable(y, type).withTypeGuards(current) + } + + // Case `x := y[i]` + // ∅ |= y:Array + is ElementAccessor -> { + // Note: ElementAccessor guarantees that `y` is an array, + // since `y[i]` for property access (i.e. access property + // with name "i") is represented via FieldAccessor. + val type = EtsTypeFact.ArrayEtsTypeFact( + elementType = EtsTypeFact.UnknownEtsTypeFact + ) + result += TypedVariable(y, type).withTypeGuards(current) + } + } + } + } + + val lhv = when (val r = current.lhv) { + is EtsRef -> r.toPath() // This, FieldRef, ArrayAccess + is EtsLValue -> r.toPath() // Local + else -> { + logger.info { "TODO backward assign zero: $current" } + error("Unexpected LHV in assignment: $current") + } + } + + // Handle new possible facts for LHS: + if (lhv.accesses.isNotEmpty()) { + // Case `x.f := y` OR `x[i] := y` + val x = lhv.base + + when (val a = lhv.accesses.single()) { + // Case `x.f := y` + // ∅ |= x:{f:unknown} + is FieldAccessor -> { + val type = EtsTypeFact.ObjectEtsTypeFact( + cls = null, + properties = mapOf(a.name to EtsTypeFact.UnknownEtsTypeFact) + ) + result += TypedVariable(x, type).withTypeGuards(current) + } + + // Case `x[i] := y` + // ∅ |= x:Array + is ElementAccessor -> { + // Note: ElementAccessor guarantees that `y` is an array, + // since `y[i]` for property access (i.e. access property + // with name "i") is represented via FieldAccessor. + val type = EtsTypeFact.ArrayEtsTypeFact( + elementType = EtsTypeFact.UnknownEtsTypeFact + ) + result += TypedVariable(x, type) + } + } + } + } + + return result + } + + private fun sequent( + current: EtsStmt, + fact: TypedVariable, + ): List { + if (current !is EtsAssignStmt) { + return listOf(fact) + } + + val lhv = current.lhv.toPath() + + val rhv = when (val r = current.rhv) { + is EtsRef -> r.toPath() // This, FieldRef, ArrayAccess + is EtsLValue -> r.toPath() // Local + is EtsCastExpr -> r.toPath() // Cast + is EtsNewExpr -> { + // TODO: what about `x.f := new T()` ? + // `x := new T()` with fact `x:U` => `saved[T] += U` + if (fact.variable == lhv.base) { + savedTypes.getOrPut(r.type) { mutableListOf() }.add(fact.type) + } + return listOf(fact) + } + + else -> { + // logger.info { "TODO backward assign: $current" } + return listOf(fact) + } + } + + // Pass-through completely unrelated facts: + if (fact.variable != lhv.base) return listOf(fact) + + // val (preAliases, _) = getAliases(current.method)[current]!! + + if (lhv.accesses.isEmpty() && rhv.accesses.isEmpty()) { + // Case `x := y` + + // x:T |= x:T (keep) + y:T + val y = rhv.base + val newFact = TypedVariable(y, fact.type).withTypeGuards(current) + return listOf(fact, newFact) + + } else if (lhv.accesses.isEmpty()) { + // Case `x := y.f` OR `x := y[i]` + + check(rhv.accesses.size == 1) + when (val a = rhv.accesses.single()) { + // Case `x := y.f` + is FieldAccessor -> { + // // Drop facts containing duplicate fields + // if (fact.type is EtsTypeFact.ObjectEtsTypeFact && a.name in fact.type.properties) { + // // can just drop? + // return listOf(fact) + // } + + // x:T |= x:T (keep) + y:{f:T} + aliases + val result = mutableListOf(fact) + val y = rhv.base + val type = EtsTypeFact.ObjectEtsTypeFact( + cls = null, + properties = mapOf(a.name to fact.type) + ) + result += TypedVariable(y, type).withTypeGuards(current) + // aliases: +|= z:{f:T} + // for (z in preAliases.getAliases(AccessPath(y, emptyList()))) { + // val type2 = unrollAccessorsToTypeFact(z.accesses + a, fact.type) + // result += TypedVariable(z.base, type2).withTypeGuards(current) + // } + return result + } + + // Case `x := y[i]` + is ElementAccessor -> { + // x:T |= x:T (keep) + y:Array + val y = rhv.base + val type = EtsTypeFact.ArrayEtsTypeFact(elementType = fact.type) + val newFact = TypedVariable(y, type).withTypeGuards(current) + return listOf(fact, newFact) + } + } + + } else if (rhv.accesses.isEmpty()) { + // Case `x.f := y` OR `x[i] := y` + + check(lhv.accesses.size == 1) + when (val a = lhv.accesses.single()) { + // Case `x.f := y` + is FieldAccessor -> { + if (fact.type is EtsTypeFact.UnionEtsTypeFact) { + TODO("Support union type for x.f := y in BW-sequent") + } + + if (fact.type is EtsTypeFact.IntersectionEtsTypeFact) { + TODO("Support intersection type for x.f := y in BW-sequent") + } + + // x:primitive |= x:primitive (pass) + if (fact.type !is EtsTypeFact.ObjectEtsTypeFact) { + return listOf(fact) + } + + // x:{no f} |= only keep x:{..} + + // x:{f:T} |= x:{f:T} (keep) + y:T + val (typeWithoutProperty, removedPropertyType) = fact.type.removePropertyType(a.name) + // val updatedFact = TypedVariable(fact.variable, typeWithoutProperty) + val updatedFact = fact + val y = rhv.base + val newType = removedPropertyType?.let { type -> TypedVariable(y, type).withTypeGuards(current) } + return listOfNotNull(updatedFact, newType) + } + + // Case `x[i] := y` + is ElementAccessor -> { + if (fact.type is EtsTypeFact.UnionEtsTypeFact) { + TODO("Support union type for x[i] := y in BW-sequent") + } + + if (fact.type is EtsTypeFact.IntersectionEtsTypeFact) { + TODO("Support intersection type for x[i] := y in BW-sequent") + } + + // x:Array |= x:Array (pass) + if (fact.type !is EtsTypeFact.ArrayEtsTypeFact) { + return listOf(fact) + } + + // x:Array |= x:Array (keep) + y:T + val y = rhv.base + val type = fact.type.elementType + val newFact = TypedVariable(y, type).withTypeGuards(current) + return listOf(fact, newFact) + } + } + } else { + error("Incorrect 3AC: $current") + } + } + + private fun EtsTypeFact.ObjectEtsTypeFact.removePropertyType(propertyName: String): Pair { + val propertyType = properties[propertyName] + val updatedThis = EtsTypeFact.ObjectEtsTypeFact(cls, properties - propertyName) + return updatedThis to propertyType + } + + override fun obtainCallToReturnSiteFlowFunction( + callStatement: EtsStmt, + returnSite: EtsStmt, + ): FlowFunction = FlowFunction { fact -> + when (fact) { + Zero -> callZero(callStatement) + is TypedVariable -> call(callStatement, fact) + } + } + + private fun callZero( + callStatement: EtsStmt, + ): List { + val result = mutableListOf(Zero) + + val callExpr = callStatement.callExpr ?: error("No call") + + if (callExpr is EtsInstanceCallExpr) { + val instance = callExpr.instance + val path = instance.toBase() + val objectWithMethod = EtsTypeFact.ObjectEtsTypeFact( + cls = null, + properties = mapOf( + callExpr.method.name to EtsTypeFact.FunctionEtsTypeFact + ) + ) + result += TypedVariable(path, objectWithMethod) + } + + return result + } + + private fun call( + callStatement: EtsStmt, + fact: TypedVariable, + ): List { + val result = mutableListOf() + + val callResult = (callStatement as? EtsAssignStmt)?.lhv?.toBase() + if (callResult != null) { + // If fact was for LHS, drop it as overridden + if (fact.variable == callResult) return result + } + + result += fact + return result + } + + override fun obtainCallToStartFlowFunction( + callStatement: EtsStmt, + calleeStart: EtsStmt, + ): FlowFunction = FlowFunction { fact -> + when (fact) { + Zero -> listOf(fact) + is TypedVariable -> start(callStatement, calleeStart, fact) + } + } + + private fun start( + callStatement: EtsStmt, + calleeStart: EtsStmt, + fact: TypedVariable, + ): List { + val callResult = (callStatement as? EtsAssignStmt)?.lhv?.toBase() ?: return emptyList() + + if (fact.variable != callResult) return emptyList() + + if (calleeStart is EtsThrowStmt) return emptyList() // TODO support throwStmt + + check(calleeStart is EtsReturnStmt) + + val exitValue = calleeStart.returnValue?.toBase() ?: return emptyList() + + return listOf(TypedVariable(exitValue, fact.type)) + } + + override fun obtainExitToReturnSiteFlowFunction( + callStatement: EtsStmt, + returnSite: EtsStmt, + exitStatement: EtsStmt, + ): FlowFunction = FlowFunction { fact -> + when (fact) { + Zero -> listOf(fact) + is TypedVariable -> exit(callStatement, fact) + } + } + + private fun exit( + callStatement: EtsStmt, + fact: TypedVariable, + ): List { + val callExpr = callStatement.callExpr ?: error("No call") + + when (fact.variable) { + is AccessPathBase.This -> { + if (callExpr !is EtsInstanceCallExpr) { + return emptyList() + } + + val instance = callExpr.instance + val instancePath = instance.toBase() + return listOf(TypedVariable(instancePath, fact.type)) + } + + is AccessPathBase.Arg -> { + val arg = callExpr.args.getOrNull(fact.variable.index)?.toBase() ?: return emptyList() + return listOf(TypedVariable(arg, fact.type)) + } + + else -> return emptyList() + } + } +} + +private const val COMPLEXITY_LIMIT = 5 + +private fun Iterable.myFilter(): List = filter { + if (it.type.complexity() >= COMPLEXITY_LIMIT) { + logger.warn { "Dropping too complex fact: $it" } + return@filter false + } + true +} + +/** + * Complexity of a type fact is the maximum depth of nested types. + */ +private fun EtsTypeFact.complexity(): Int = when (this) { + is EtsTypeFact.ObjectEtsTypeFact -> (properties.values.maxOfOrNull { it.complexity() } ?: 0) + 1 + is EtsTypeFact.ArrayEtsTypeFact -> elementType.complexity() + 1 + is EtsTypeFact.UnionEtsTypeFact -> (types.maxOfOrNull { it.complexity() } ?: 0) + 1 + is EtsTypeFact.IntersectionEtsTypeFact -> (types.maxOfOrNull { it.complexity() } ?: 0) + 1 + else -> 0 +} diff --git a/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/EntryPointsProcessor.kt b/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/EntryPointsProcessor.kt new file mode 100644 index 0000000000..a686fcf463 --- /dev/null +++ b/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/EntryPointsProcessor.kt @@ -0,0 +1,25 @@ +package org.usvm.dataflow.ts.infer + +import org.jacodb.ets.model.EtsMethod +import org.jacodb.ets.model.EtsScene + +object EntryPointsProcessor { + fun extractEntryPoints( + scene: EtsScene, + ): ArtificialMainWithAllMethods { + val artificialMainMethods = scene.projectClasses + .asSequence() + .flatMap { it.methods } + .filter { it.name == "@dummyMain" } + .toList() + return ArtificialMainWithAllMethods( + mainMethods = artificialMainMethods, + allMethods = scene.projectClasses.flatMap { it.methods }, + ) + } +} + +data class ArtificialMainWithAllMethods( + val mainMethods: List, + val allMethods: List, +) diff --git a/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/EtsMethodTypeFacts.kt b/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/EtsMethodTypeFacts.kt new file mode 100644 index 0000000000..058cba690d --- /dev/null +++ b/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/EtsMethodTypeFacts.kt @@ -0,0 +1,8 @@ +package org.usvm.dataflow.ts.infer + +import org.jacodb.ets.model.EtsMethod + +data class EtsMethodTypeFacts( + val method: EtsMethod, + val types: Map, +) diff --git a/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/EtsTypeFact.kt b/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/EtsTypeFact.kt new file mode 100644 index 0000000000..24b2118431 --- /dev/null +++ b/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/EtsTypeFact.kt @@ -0,0 +1,531 @@ +package org.usvm.dataflow.ts.infer + +import mu.KotlinLogging +import org.jacodb.ets.base.ANONYMOUS_CLASS_PREFIX +import org.jacodb.ets.base.CONSTRUCTOR_NAME +import org.jacodb.ets.base.EtsAnyType +import org.jacodb.ets.base.EtsArrayType +import org.jacodb.ets.base.EtsBooleanType +import org.jacodb.ets.base.EtsClassType +import org.jacodb.ets.base.EtsFunctionType +import org.jacodb.ets.base.EtsNullType +import org.jacodb.ets.base.EtsNumberType +import org.jacodb.ets.base.EtsStringType +import org.jacodb.ets.base.EtsType +import org.jacodb.ets.base.EtsUnclearRefType +import org.jacodb.ets.base.EtsUndefinedType +import org.jacodb.ets.base.EtsUnionType +import org.jacodb.ets.base.EtsUnknownType +import org.jacodb.ets.base.INSTANCE_INIT_METHOD_NAME + +private val logger = KotlinLogging.logger {} + +sealed interface EtsTypeFact { + + fun toPrettyString(): String { + return toString() + } + + sealed interface BasicType : EtsTypeFact + + fun union(other: EtsTypeFact): EtsTypeFact { + if (this == other) return this + + return when { + this is ObjectEtsTypeFact && other is ObjectEtsTypeFact -> union(this, other) + this is ObjectEtsTypeFact && other is StringEtsTypeFact -> union(this, other) + this is UnionEtsTypeFact -> union(this, other) + this is IntersectionEtsTypeFact -> union(this, other) + this is GuardedTypeFact -> union(this, other) + other is UnionEtsTypeFact -> union(other, this) + other is IntersectionEtsTypeFact -> union(other, this) + other is GuardedTypeFact -> union(other, this) + else -> mkUnionType(this, other) + } + } + + fun intersect(other: EtsTypeFact): EtsTypeFact? { + if (this == other) return this + + if (other is UnknownEtsTypeFact) return this + if (other is AnyEtsTypeFact) return other + + return when (this) { + is UnknownEtsTypeFact -> other + + is AnyEtsTypeFact -> this + + is StringEtsTypeFact, + is NumberEtsTypeFact, + is BooleanEtsTypeFact, + is NullEtsTypeFact, + is UndefinedEtsTypeFact, + -> when (other) { + is UnionEtsTypeFact -> intersect(other, this) + is IntersectionEtsTypeFact -> intersect(other, this) + is GuardedTypeFact -> intersect(other, this) + else -> null + } + + is FunctionEtsTypeFact -> when (other) { + is ObjectEtsTypeFact -> mkIntersectionType(this, other) + is UnionEtsTypeFact -> intersect(other, this) + is IntersectionEtsTypeFact -> intersect(other, this) + is GuardedTypeFact -> intersect(other, this) + else -> null + } + + is ArrayEtsTypeFact -> when (other) { + is ArrayEtsTypeFact -> { + val t = elementType.intersect(other.elementType) + if (t == null) { + logger.warn { "Empty intersection of array element types: $elementType & ${other.elementType}" } + null + } else { + ArrayEtsTypeFact(t) + } + } + + else -> null + } + + is ObjectEtsTypeFact -> when (other) { + is ObjectEtsTypeFact -> intersect(this, other) + is StringEtsTypeFact -> intersect(this, other) + is FunctionEtsTypeFact -> mkIntersectionType(this, other) + is UnionEtsTypeFact -> intersect(other, this) + is IntersectionEtsTypeFact -> intersect(other, this) + is GuardedTypeFact -> intersect(other, this) + else -> null + } + + is UnionEtsTypeFact -> intersect(this, other) + is IntersectionEtsTypeFact -> intersect(this, other) + is GuardedTypeFact -> intersect(this, other) + } + } + + object UnknownEtsTypeFact : EtsTypeFact, BasicType { + override fun toString(): String = "unknown" + } + + object AnyEtsTypeFact : BasicType { + override fun toString(): String = "any" + } + + object StringEtsTypeFact : BasicType { + override fun toString(): String = "string" + } + + object NumberEtsTypeFact : BasicType { + override fun toString(): String = "number" + } + + object BooleanEtsTypeFact : BasicType { + override fun toString(): String = "boolean" + } + + object NullEtsTypeFact : BasicType { + override fun toString(): String = "null" + } + + object UndefinedEtsTypeFact : BasicType { + override fun toString(): String = "undefined" + } + + // object VoidEtsTypeFact : BasicType { + // override fun toString(): String = "void" + // } + // + // object NeverEtsTypeFact : BasicType { + // override fun toString(): String = "never" + // } + + object FunctionEtsTypeFact : BasicType { + override fun toString(): String = "function" + } + + data class ArrayEtsTypeFact( + val elementType: EtsTypeFact, + ) : BasicType { + override fun toString(): String = "Array<$elementType>" + } + + data class ObjectEtsTypeFact( + val cls: EtsType?, + val properties: Map, + ) : BasicType { + override fun toString(): String { + val clsName = cls?.typeName?.takeUnless { it.startsWith(ANONYMOUS_CLASS_PREFIX) } ?: "Object" + val funProps = properties.entries + .filter { it.value is FunctionEtsTypeFact } + .filterNot { it.key == CONSTRUCTOR_NAME } + .filterNot { it.key == INSTANCE_INIT_METHOD_NAME } + .sortedBy { it.key } + val nonFunProps = properties.entries + .filter { it.value !is FunctionEtsTypeFact } + .sortedBy { it.key } + val props = (funProps + nonFunProps).joinToString(", ") { (name, type) -> "$name: $type" } + return "$clsName { $props }" + } + + // Object { + // ..foo: Object { + // ....bar: string + // ..} + // } + override fun toPrettyString(): String { + val clsName = cls?.typeName ?: "Object" + val funProps = properties.entries + .filter { it.value is FunctionEtsTypeFact } + .sortedBy { it.key } + val nonFunProps = properties.entries + .filter { it.value !is FunctionEtsTypeFact } + .sortedBy { it.key } + return buildString { + appendLine("$clsName {") + for ((name, type) in funProps) { + appendLine(" $name: $type") + } + for ((name, type) in nonFunProps) { + appendLine("$name: ${type.toPrettyString()}".lines().joinToString("\n") { " $it" }) + } + append("}") + } + } + + override fun equals(other: Any?): Boolean { + if (other !is ObjectEtsTypeFact) return false + + if (other.cls != null && other.cls == cls) return true + + return properties == other.properties + } + + override fun hashCode(): Int { + if (cls == null) return properties.hashCode() + + return cls.hashCode() + } + } + + data class UnionEtsTypeFact( + val types: Set, + ) : EtsTypeFact { + init { + require(types.isNotEmpty()) { + "An empty set of types is passed as an union type" + } + } + + override fun toString(): String { + return types.map { + when (it) { + is UnionEtsTypeFact, is IntersectionEtsTypeFact -> "($it)" + else -> it.toString() + } + }.sorted().joinToString(" | ") + } + + override fun toPrettyString(): String { + return types.map { + when (it) { + is UnionEtsTypeFact, is IntersectionEtsTypeFact -> "($it)" + else -> it.toString() + } + }.sorted().joinToString(" | ") + } + } + + data class IntersectionEtsTypeFact( + val types: Set, + ) : EtsTypeFact { + init { + require(types.isNotEmpty()) { + "An empty set of types is passed as an intersection type" + } + } + + override fun toString(): String { + return types.map { + when (it) { + is UnionEtsTypeFact, is IntersectionEtsTypeFact -> "($it)" + else -> it.toString() + } + }.sorted().joinToString(" & ") + } + + override fun toPrettyString(): String { + return types.map { + when (it) { + is UnionEtsTypeFact, is IntersectionEtsTypeFact -> "($it)" + else -> it.toString() + } + }.sorted().joinToString(" & ") + } + } + + data class GuardedTypeFact( + val guard: BasicType, + val guardNegated: Boolean, + val type: EtsTypeFact, + ) : EtsTypeFact + + companion object { + internal val allStringProperties = listOf( + "length", + "constructor", + "anchor", + "at", + "big", + "blink", + "bold", + "charAt", + "charCodeAt", + "codePointAt", + "concat", + "endsWith", + "fontcolor", + "fontsize", + "fixed", + "includes", + "indexOf", + "isWellFormed", + "italics", + "lastIndexOf", + "link", + "localeCompare", + "match", + "matchAll", + "normalize", + "padEnd", + "padStart", + "repeat", + "replace", + "replaceAll", + "search", + "slice", + "small", + "split", + "strike", + "sub", + "substr", + "substring", + "sup", + "startsWith", + "toString", + "toWellFormed", + "trim", + "trimStart", + "trimLeft", + "trimEnd", + "trimRight", + "toLocaleLowerCase", + "toLocaleUpperCase", + "toLowerCase", + "toUpperCase", + "valueOf", + ) + + private fun intersect(unionType: UnionEtsTypeFact, other: EtsTypeFact): EtsTypeFact { + // todo: push intersection + return mkIntersectionType(unionType, other) + } + + private fun intersect(intersectionType: IntersectionEtsTypeFact, other: EtsTypeFact): EtsTypeFact? { + val result = hashSetOf() + for (type in intersectionType.types) { + val intersection = type.intersect(other) ?: return null + if (intersection is IntersectionEtsTypeFact) { + result.addAll(intersection.types) + } else { + result.add(intersection) + } + } + return mkIntersectionType(result) + } + + private fun intersect(guardedType: GuardedTypeFact, other: EtsTypeFact): EtsTypeFact? { + if (other is GuardedTypeFact) { + if (other.guard == guardedType.guard) { + return if (other.guardNegated == guardedType.guardNegated) { + guardedType.type.intersect(other.type)?.withGuard(guardedType.guard, guardedType.guardNegated) + } else { + guardedType.type.union(other.type) + } + } + } + + // todo: evaluate types + return mkIntersectionType(guardedType, other) + } + + private fun intersect(obj1: ObjectEtsTypeFact, obj2: ObjectEtsTypeFact): EtsTypeFact? { + val intersectionProperties = obj1.properties.toMutableMap() + for ((property, type) in obj2.properties) { + val currentType = intersectionProperties[property] + if (currentType == null) { + intersectionProperties[property] = type + continue + } + + intersectionProperties[property] = currentType.intersect(type) ?: return null + } + + val intersectionCls = if (obj1.cls != null && obj2.cls != null) { + obj1.cls.takeIf { it == obj2.cls } + } else { + obj1.cls ?: obj2.cls + } + return ObjectEtsTypeFact(intersectionCls, intersectionProperties) + } + + private fun intersect(obj: ObjectEtsTypeFact, string: StringEtsTypeFact): EtsTypeFact? { + if (obj.cls == EtsStringType) return string + if (obj.cls != null) return null + + val intersectionProperties = obj.properties + .filter { it.key in allStringProperties } + .mapValues { (_, type) -> + // TODO: intersect with the corresponding type of String's property + type + } + + return ObjectEtsTypeFact(obj.cls, intersectionProperties) + } + + private fun union(unionType: UnionEtsTypeFact, other: EtsTypeFact): EtsTypeFact { + val result = hashSetOf() + for (type in unionType.types) { + val union = type.union(other) + if (union is UnionEtsTypeFact) { + result.addAll(union.types) + } else { + result.add(union) + } + } + return mkUnionType(result) + } + + private fun union(guardedType: GuardedTypeFact, other: EtsTypeFact): EtsTypeFact { + // todo: evaluate types + return mkUnionType(guardedType, other) + } + + private fun union(intersectionType: IntersectionEtsTypeFact, other: EtsTypeFact): EtsTypeFact { + // todo: push union + return mkUnionType(intersectionType, other) + } + + private fun union(obj1: ObjectEtsTypeFact, obj2: ObjectEtsTypeFact): EtsTypeFact { + if (obj1.cls != null && obj2.cls != null && obj1.cls != obj2.cls) { + return mkUnionType(obj1, obj2) + } + + val commonProperties = obj1.properties.keys.intersect(obj2.properties.keys).associateWith { property -> + val thisType = obj1.properties.getValue(property) + val otherType = obj2.properties.getValue(property) + thisType.union(otherType) + } + + val o1OnlyProperties = obj1.properties.filter { it.key !in obj2.properties } + val o2OnlyProperties = obj2.properties.filter { it.key !in obj1.properties } + + val o1 = ObjectEtsTypeFact(obj1.cls, o1OnlyProperties) + val o2 = ObjectEtsTypeFact(obj2.cls, o2OnlyProperties) + + if (commonProperties.isEmpty()) { + return mkUnionType(o1, o2) + } + + val commonCls = obj1.cls.takeIf { it == obj2.cls } + val commonObject = ObjectEtsTypeFact(commonCls, commonProperties) + + if (o1OnlyProperties.isEmpty() && o2OnlyProperties.isEmpty()) { + return commonObject + } + + return mkIntersectionType(commonObject, mkUnionType(o1, o2)) + } + + private fun union(obj: ObjectEtsTypeFact, string: StringEtsTypeFact): EtsTypeFact { + if (obj.cls == EtsStringType) return string + if (obj.cls != null) return mkUnionType(obj, string) + + for (p in obj.properties.keys) { + if (p !in allStringProperties) { + return mkUnionType(obj, string) + } + } + + return string + } + + fun mkUnionType(vararg types: EtsTypeFact): EtsTypeFact = mkUnionType(types.toHashSet()) + + fun mkUnionType(types: Set): EtsTypeFact { + if (types.size == 1) return types.single() + return UnionEtsTypeFact(types) + } + + fun mkIntersectionType(vararg types: EtsTypeFact): EtsTypeFact = mkIntersectionType(types.toHashSet()) + + fun mkIntersectionType(types: Set): EtsTypeFact { + if (types.size == 1) return types.single() + return IntersectionEtsTypeFact(types) + } + + fun from(type: EtsType): EtsTypeFact { + return when (type) { + is EtsAnyType -> AnyEtsTypeFact + is EtsUnknownType -> UnknownEtsTypeFact + is EtsUnionType -> UnionEtsTypeFact(type.types.map { from(it) }.toSet()) + is EtsBooleanType -> BooleanEtsTypeFact + is EtsNumberType -> NumberEtsTypeFact + is EtsStringType -> StringEtsTypeFact + is EtsNullType -> NullEtsTypeFact + is EtsUndefinedType -> UndefinedEtsTypeFact + is EtsClassType -> ObjectEtsTypeFact(type, emptyMap()) + is EtsFunctionType -> FunctionEtsTypeFact + // is EtsArrayType -> ObjectEtsTypeFact( + // cls = type, + // properties = mapOf( + // "index" to ObjectEtsTypeFact( + // cls = null, + // properties = mapOf( + // "name" to StringEtsTypeFact, + // "value" to from(type.elementType) + // ) + // ), + // "length" to NumberEtsTypeFact + // ) + // ) + is EtsArrayType -> ArrayEtsTypeFact(elementType = from(type.elementType)) + is EtsUnclearRefType -> ObjectEtsTypeFact(type, emptyMap()) + // is EtsGenericType -> TODO() + else -> { + logger.error { "Unsupported type: $type" } + UnknownEtsTypeFact + } + } + } + } +} + +fun EtsTypeFact.withGuard(guard: EtsTypeFact.BasicType, guardNegated: Boolean): EtsTypeFact { + val duplicateGuard = findDuplicateGuard(this, guard) + + if (duplicateGuard != null) { + if (guardNegated == duplicateGuard.guardNegated) return this + + TODO("Same guard with different sign") + } + + return EtsTypeFact.GuardedTypeFact(guard, guardNegated, this) +} + +private fun findDuplicateGuard(fact: EtsTypeFact, guard: EtsTypeFact.BasicType): EtsTypeFact.GuardedTypeFact? { + if (fact !is EtsTypeFact.GuardedTypeFact) return null + if (fact.guard == guard) return fact + return findDuplicateGuard(fact.type, guard) +} diff --git a/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/ForwardAnalyzer.kt b/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/ForwardAnalyzer.kt new file mode 100644 index 0000000000..a8aebbe64f --- /dev/null +++ b/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/ForwardAnalyzer.kt @@ -0,0 +1,44 @@ +package org.usvm.dataflow.ts.infer + +import org.jacodb.ets.base.EtsStmt +import org.jacodb.ets.base.EtsType +import org.jacodb.ets.model.EtsMethod +import org.usvm.dataflow.ifds.Analyzer +import org.usvm.dataflow.ifds.Edge +import org.usvm.dataflow.ifds.Vertex +import org.usvm.dataflow.ts.graph.EtsApplicationGraph + +class ForwardAnalyzer( + val graph: EtsApplicationGraph, + methodInitialTypes: Map, + typeInfo: Map, + doAddKnownTypes: Boolean = true, +) : Analyzer { + + override val flowFunctions = ForwardFlowFunctions(graph, methodInitialTypes, typeInfo, doAddKnownTypes) + + override fun handleCrossUnitCall( + caller: Vertex, + callee: Vertex, + ): List { + error("No cross unit calls") + } + + override fun handleNewEdge(edge: Edge): List { + val (startVertex, currentVertex) = edge + val (current, currentFact) = currentVertex + + val method = graph.methodOf(current) + val currentIsExit = current in graph.exitPoints(method) + + if (!currentIsExit) return emptyList() + + return listOf( + ForwardSummaryAnalyzerEvent( + method = method, + initialVertex = startVertex, + exitVertex = currentVertex, + ) + ) + } +} diff --git a/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/ForwardFlowFunctions.kt b/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/ForwardFlowFunctions.kt new file mode 100644 index 0000000000..b9971a846f --- /dev/null +++ b/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/ForwardFlowFunctions.kt @@ -0,0 +1,586 @@ +package org.usvm.dataflow.ts.infer + +import mu.KotlinLogging +import org.jacodb.ets.base.EtsAnyType +import org.jacodb.ets.base.EtsArithmeticExpr +import org.jacodb.ets.base.EtsAssignStmt +import org.jacodb.ets.base.EtsBooleanConstant +import org.jacodb.ets.base.EtsCastExpr +import org.jacodb.ets.base.EtsFieldRef +import org.jacodb.ets.base.EtsInstanceCallExpr +import org.jacodb.ets.base.EtsLValue +import org.jacodb.ets.base.EtsNewArrayExpr +import org.jacodb.ets.base.EtsNewExpr +import org.jacodb.ets.base.EtsNullConstant +import org.jacodb.ets.base.EtsNumberConstant +import org.jacodb.ets.base.EtsRef +import org.jacodb.ets.base.EtsRelationExpr +import org.jacodb.ets.base.EtsReturnStmt +import org.jacodb.ets.base.EtsStmt +import org.jacodb.ets.base.EtsStringConstant +import org.jacodb.ets.base.EtsType +import org.jacodb.ets.base.EtsUndefinedConstant +import org.jacodb.ets.base.EtsUnknownType +import org.jacodb.ets.model.EtsMethod +import org.jacodb.ets.utils.callExpr +import org.usvm.dataflow.ifds.FlowFunction +import org.usvm.dataflow.ifds.FlowFunctions +import org.usvm.dataflow.ts.graph.EtsApplicationGraph +import org.usvm.dataflow.ts.infer.ForwardTypeDomainFact.TypedVariable +import org.usvm.dataflow.ts.infer.ForwardTypeDomainFact.Zero + +private val logger = KotlinLogging.logger {} + +class ForwardFlowFunctions( + val graph: EtsApplicationGraph, + val methodInitialTypes: Map, + val typeInfo: Map, + val doAddKnownTypes: Boolean = true, +) : FlowFunctions { + + private val aliasesCache: MutableMap>> = hashMapOf() + + private fun getAliases(method: EtsMethod): Map> { + return aliasesCache.computeIfAbsent(method) { computeAliases(method) } + } + + override fun obtainPossibleStartFacts(method: EtsMethod): Collection { + val result = mutableListOf(Zero) + + val initialTypes = methodInitialTypes[method] + if (initialTypes != null) { + for ((base, type) in initialTypes.types) { + val path = AccessPath(base, accesses = emptyList()) + addTypes(path, type, result) + } + } + + if (doAddKnownTypes) { + for (local in method.locals) { + if (local.type != EtsUnknownType && local.type != EtsAnyType) { + val path = AccessPath(AccessPathBase.Local(local.name), accesses = emptyList()) + val type = EtsTypeFact.from(local.type) + if (type != EtsTypeFact.UnknownEtsTypeFact && type != EtsTypeFact.AnyEtsTypeFact) { + logger.debug { "Adding known type for $path: $type" } + addTypes(path, type, result) + } + } + } + } + + return result + } + + private fun addTypes( + path: AccessPath, + type: EtsTypeFact, + facts: MutableList, + ) { + when (type) { + EtsTypeFact.UnknownEtsTypeFact -> { + facts += TypedVariable(path, EtsTypeFact.AnyEtsTypeFact) + } + + is EtsTypeFact.ObjectEtsTypeFact -> { + for ((propertyName, propertyType) in type.properties) { + val propertyPath = path + FieldAccessor(propertyName) + addTypes(propertyPath, propertyType, facts) + } + + facts += TypedVariable(path, type) + } + + is EtsTypeFact.ArrayEtsTypeFact -> { + // check(type.elementType !is EtsTypeFact.ArrayEtsTypeFact) + facts += TypedVariable(path, type) + addTypes(path + ElementAccessor, type.elementType, facts) + } + + is EtsTypeFact.GuardedTypeFact -> { + addTypes(path, type.type, facts) + } + + is EtsTypeFact.IntersectionEtsTypeFact -> { + type.types.forEach { addTypes(path, it, facts) } + } + + is EtsTypeFact.UnionEtsTypeFact -> { + type.types.forEach { addTypes(path, it, facts) } + } + + else -> { + facts += TypedVariable(path, type) + } + } + } + + override fun obtainSequentFlowFunction( + current: EtsStmt, + next: EtsStmt, + ): FlowFunction = FlowFunction { fact -> + if (current is EtsAssignStmt) { + val lhvPath = current.lhv.toPathOrNull() + val rhvPath = current.rhv.toPathOrNull() + if (lhvPath != null && rhvPath != null && lhvPath == rhvPath) { + return@FlowFunction listOf(fact) + } + } + when (fact) { + Zero -> sequentZero(current) + is TypedVariable -> sequentFact(current, fact).myFilter() + } + } + + private fun sequentZero(current: EtsStmt): List { + if (current !is EtsAssignStmt) return listOf(Zero) + + val lhv = current.lhv.toPath() + val result = mutableListOf(Zero) + val (preAliases, _) = getAliases(current.method).getValue(current) + + fun addTypeFactWithAliases(path: AccessPath, type: EtsTypeFact) { + result += TypedVariable(path, type) + if (path.accesses.isNotEmpty()) { + check(path.accesses.size == 1) + val base = AccessPath(path.base, emptyList()) + for (alias in preAliases.getAliases(base)) { + val newPath = alias + path.accesses.single() + result += TypedVariable(newPath, type) + } + } + } + + when (val rhv = current.rhv) { + is EtsNewExpr -> { + // val newType = rhv.type + // if (newType is EtsClassType) { + // val cls = graph.cp.classes + // .firstOrNull { it.name == newType.typeName } + // if (cls != null) { + // for (f in cls.fields) { + // val path = lhv + FieldAccessor(f.name) + // result += TypedVariable(path, EtsTypeFact.from(f.type)) + // } + // } + // } + + val type = typeInfo[rhv.type] + ?: EtsTypeFact.ObjectEtsTypeFact(cls = rhv.type, properties = emptyMap()) + addTypeFactWithAliases(lhv, type) + } + + is EtsNewArrayExpr -> { + // TODO: check + val elementType = EtsTypeFact.from(rhv.elementType) + val type = EtsTypeFact.ArrayEtsTypeFact(elementType = elementType) + result += TypedVariable(lhv, type) + result += TypedVariable(lhv + ElementAccessor, elementType) + } + + is EtsStringConstant -> { + addTypeFactWithAliases(lhv, EtsTypeFact.StringEtsTypeFact) + } + + is EtsNumberConstant -> { + addTypeFactWithAliases(lhv, EtsTypeFact.NumberEtsTypeFact) + } + + is EtsBooleanConstant -> { + addTypeFactWithAliases(lhv, EtsTypeFact.BooleanEtsTypeFact) + } + + is EtsNullConstant -> { + addTypeFactWithAliases(lhv, EtsTypeFact.NullEtsTypeFact) + } + + is EtsUndefinedConstant -> { + addTypeFactWithAliases(lhv, EtsTypeFact.UndefinedEtsTypeFact) + } + + // Note: do not handle cast in forward ff! + // is EtsCastExpr -> { + // result += TypedVariable(lhv, EtsTypeFact.from(rhv.type)) + // } + + is EtsFieldRef -> { + if (doAddKnownTypes && rhv.type != EtsUnknownType && rhv.type != EtsAnyType) { + val type = EtsTypeFact.from(rhv.type) + logger.debug { "Adding known type for $lhv from $rhv: $type" } + addTypeFactWithAliases(lhv, type) + } + } + + is EtsArithmeticExpr -> { + result += TypedVariable(lhv, EtsTypeFact.StringEtsTypeFact) + result += TypedVariable(lhv, EtsTypeFact.NumberEtsTypeFact) + } + + is EtsRelationExpr -> { + result += TypedVariable(lhv, EtsTypeFact.BooleanEtsTypeFact) + } + + else -> { + // logger.info { "TODO: forward assign $current" } + } + } + + return result + } + + private fun sequentFact(current: EtsStmt, fact: TypedVariable): List { + if (current !is EtsAssignStmt) return listOf(fact) + + val lhv = current.lhv.toPath() + + val rhv = when (val r = current.rhv) { + is EtsRef -> r.toPath() // This, FieldRef, ArrayAccess + is EtsLValue -> r.toPath() // Local + is EtsCastExpr -> r.toPath() // Cast + else -> { + // logger.info { "TODO forward assign: $current" } + null + } + } + + val (preAliases, _) = getAliases(current.method).getValue(current) + + // Override LHS when RHS is const-like: + if (rhv == null) { + if (lhv.accesses.isEmpty()) { + // x := const + + // TODO: `x := const as T` + + // x.*:T |= drop + if (fact.variable.startsWith(lhv)) { + return emptyList() + } + + } else { + // x.f := const OR x[i] := const + + check(lhv.accesses.size == 1) + when (val a = lhv.accesses.single()) { + // x.f := const + is FieldAccessor -> { + val base = AccessPath(lhv.base, emptyList()) + + // x.f.*:T |= drop + if (fact.variable.startsWith(lhv)) { + return emptyList() + } + // z in G(x), z.f.*:T |= drop + if (preAliases.getAliases(base).any { fact.variable.startsWith(it + a) }) { + return emptyList() + } + } + + // x[i] := const + is ElementAccessor -> { + // do nothing, pass-through + } + } + } + + // Pass-through unrelated facts: + return listOf(fact) + } + + if (lhv.accesses.isEmpty() && rhv.accesses.isEmpty()) { + // x := y + + // TODO: x := x + // Note: handled outside + + // x.*:T |= drop + if (fact.variable.startsWith(lhv)) { + return emptyList() + } + + // y.*:T |= y.*:T (keep) + x.*:T (same tail) + if (fact.variable.startsWith(rhv)) { + // Extra case with cast: `x := y as U`: + // `y.*:T` |= keep + new fact `x.*:W`, where `W = T intersect U` + // TODO: Currently, we just take the type from the CastExpr, without intersecting. + // The problem is that when we have fact `y:any`, the intersection (though probably correctly) + // produces `x:any`, so we just lose type information from the cast. + // Using the cast type directly is just a temporary solution to satisfy simple tests. + if (current.rhv is EtsCastExpr) { + val path = AccessPath(lhv.base, fact.variable.accesses) + // val type = EtsTypeFact.from((current.rhv as EtsCastExpr).type).intersect(fact.type) ?: fact.type + val type = EtsTypeFact.from((current.rhv as EtsCastExpr).type) + return listOf(fact, TypedVariable(path, type)) + } + + val path = AccessPath(lhv.base, fact.variable.accesses) + return listOf(fact, TypedVariable(path, fact.type)) + } + + } else if (lhv.accesses.isEmpty()) { + // x := y.f OR x := y[i] + + // TODO: x := x.f + // ??????? x.f:T |= drop + + // x.*:T |= drop + if (fact.variable.startsWith(lhv)) { + return emptyList() + } + + check(rhv.accesses.size == 1) + when (val a = rhv.accesses.single()) { + // x := y.f + is FieldAccessor -> { + // y.f.*:T |= y.f.*:T (keep) + x.*:T (same tail after .f) + if (fact.variable.startsWith(rhv)) { + val path = lhv + fact.variable.accesses.drop(1) + return listOf(fact, TypedVariable(path, fact.type)) + } + // Note: the following is unnecessary due to `z := y` alias + // // z in G(y), z.f.*:T |= z.f.*:T (keep) + x.*:T (same tail after .f) + // val y = AccessPath(rhv.base, emptyList()) + // for (z in preAliases.getAliases(y)) { + // if (fact.variable.startsWith(z + a)) { + // val path = lhv + fact.variable.accesses.drop(z.accesses.size + 1) + // return listOf(fact, TypedVariable(path, fact.type)) + // } + // } + } + + // x := y[i] + is ElementAccessor -> { + // do nothing, pass-through + // TODO: ??? + + // TODO: do we need to add type fact `x.*:T` here? + // y[i].*:T |= y[i].*:T (keep) + x.*:T (same tail after [i]) + if (fact.variable.startsWith(rhv)) { + val path = lhv + fact.variable.accesses.drop(1) + return listOf(fact, TypedVariable(path, fact.type)) + } + } + } + + } else if (rhv.accesses.isEmpty()) { + // x.f := y OR x[i] := y + + check(lhv.accesses.size == 1) + when (val a = lhv.accesses.single()) { + // x.f := y + is FieldAccessor -> { + // TODO: x.f := x + + // x.f.*:T |= drop + if (fact.variable.startsWith(lhv)) { + return emptyList() + } + // z in G(x), z.f.*:T |= drop + val x = AccessPath(lhv.base, emptyList()) + if (preAliases.getAliases(x).any { z -> fact.variable.startsWith(z + a) }) { + return emptyList() + } + + // x.*:T |= x.*:T (keep) + // Note: .* does NOT start with .f, which is handled above + if (fact.variable.base == lhv.base) { + return listOf(fact) + } + + // y.*:T |= y.*:T (keep) + x.f.*:T (same tail) + aliases + // aliases: z in G(x), z.f.*:T |= x.f.*:T (same tail) + if (fact.variable.startsWith(rhv)) { + val result = mutableListOf(fact) + // skip duplicate fields + // if (fact.variable.accesses.firstOrNull() != a) { + val path1 = lhv + fact.variable.accesses + result += TypedVariable(path1, fact.type) + // } + for (z in preAliases.getAliases(x)) { + // skip duplicate fields + // if (z.accesses.firstOrNull() != a) { + // TODO: what about z.accesses.last == a ? + val path2 = z + a + fact.variable.accesses + result += TypedVariable(path2, fact.type) + // } + } + return result + } + // Note: the following is unnecessary due to `z := y` alias + // // z in G(y), z.*:T |= x.f.*:T (same tail) + // for (z in preAliases.getAliases(rhv)) { + // if (fact.variable.startsWith(z)) { + // val path = lhv + fact.variable.accesses + // return listOf(fact, TypedVariable(path, fact.type)) + // } + // } + } + + // x[i] := y + is ElementAccessor -> { + // do nothing, pass-through + // TODO: ??? + + // TODO: do we really want to add type fact `x[i]:T` here? + // y.*:T |= y.*:T (keep) + x[i].*:T (same tail) + // TODO: what about aliases of x? + if (fact.variable.startsWith(rhv)) { + val path = lhv + fact.variable.accesses + return listOf(fact, TypedVariable(path, fact.type)) + } + } + } + + } else { + error("Incorrect 3AC: $current") + } + + return listOf(fact) + } + + override fun obtainCallToReturnSiteFlowFunction( + callStatement: EtsStmt, + returnSite: EtsStmt, + ): FlowFunction = FlowFunction { fact -> + when (fact) { + // TODO: add known return type of the function call + Zero -> listOf(Zero) + is TypedVariable -> call(callStatement, fact) + } + } + + private fun call( + callStatement: EtsStmt, + fact: TypedVariable, + ): List { + val callResultValue = (callStatement as? EtsAssignStmt)?.lhv?.toPath() + if (callResultValue != null) { + // Drop fact on LHS as it will be overwritten by the call result + if (fact.variable.base == callResultValue.base) return emptyList() + } + + val callExpr = callStatement.callExpr ?: error("No call") + + // todo: hack, keep fact if call was not resolved + if (graph.callees(callStatement).none()) { + return listOf(fact) + } + + if (callExpr is EtsInstanceCallExpr) { + val instance = callExpr.instance.toPath() + if (fact.variable.base == instance.base) return emptyList() + } + + for (arg in callExpr.args) { + val argPath = arg.toPath() + if (fact.variable.base == argPath.base) return emptyList() + } + + return listOf(fact) + } + + override fun obtainCallToStartFlowFunction( + callStatement: EtsStmt, + calleeStart: EtsStmt, + ): FlowFunction = FlowFunction { fact -> + when (fact) { + Zero -> listOf(Zero) + is TypedVariable -> start(callStatement, fact) + } + } + + private fun start( + callStatement: EtsStmt, + fact: TypedVariable, + ): List { + val result = mutableListOf() + + val callExpr = callStatement.callExpr ?: error("No call") + + if (callExpr is EtsInstanceCallExpr) { + val instance = callExpr.instance.toPath() + if (fact.variable.base == instance.base) { + val path = AccessPath(AccessPathBase.This, fact.variable.accesses) + result += TypedVariable(path, fact.type) + } + } + + for ((index, arg) in callExpr.args.withIndex()) { + val argPath = arg.toPath() + if (fact.variable.base == argPath.base) { + val path = AccessPath(AccessPathBase.Arg(index), fact.variable.accesses) + result += TypedVariable(path, fact.type) + } + } + + return result + } + + override fun obtainExitToReturnSiteFlowFunction( + callStatement: EtsStmt, + returnSite: EtsStmt, + exitStatement: EtsStmt, + ): FlowFunction = FlowFunction { fact -> + when (fact) { + Zero -> listOf(Zero) + is TypedVariable -> exit(callStatement, exitStatement, fact) + } + } + + private fun exit( + callStatement: EtsStmt, + exitStatement: EtsStmt, + fact: TypedVariable, + ): List { + val factVariableBase = fact.variable.base + val callExpr = callStatement.callExpr ?: error("No call") + + when (factVariableBase) { + is AccessPathBase.This -> { + // Drop facts on This if the call was static + if (callExpr !is EtsInstanceCallExpr) { + return emptyList() + } + + val instance = callExpr.instance.toPath() + check(instance.accesses.isEmpty()) + + val path = AccessPath(instance.base, fact.variable.accesses) + return listOf(TypedVariable(path, fact.type)) + } + + is AccessPathBase.Arg -> { + val arg = callExpr.args.getOrNull(factVariableBase.index)?.toPath() ?: return emptyList() + val path = AccessPath(arg.base, fact.variable.accesses) + return listOf(TypedVariable(path, fact.type)) + } + + else -> { + if (exitStatement !is EtsReturnStmt) return emptyList() + val exitValue = exitStatement.returnValue?.toPath() ?: return emptyList() + + if (fact.variable.base != exitValue.base) return emptyList() + + val callResult = (callStatement as? EtsAssignStmt)?.lhv?.toPath() ?: return emptyList() + check(callResult.accesses.isEmpty()) + + val path = AccessPath(callResult.base, fact.variable.accesses) + return listOf(TypedVariable(path, fact.type)) + } + } + } +} + +private const val ACCESSES_LIMIT = 5 +private const val DUPLICATE_FIELDS_LIMIT = 3 + +private fun Iterable.myFilter(): List = filter { + if (it.variable.accesses.size > ACCESSES_LIMIT) { + logger.warn { "Dropping too long fact: $it" } + return@filter false + } + if (it.variable.accesses.hasDuplicateFields(DUPLICATE_FIELDS_LIMIT)) { + logger.warn { "Dropping fact with duplicate fields: $it" } + return@filter false + } + true +} diff --git a/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/TypeDomainFact.kt b/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/TypeDomainFact.kt new file mode 100644 index 0000000000..196dea887c --- /dev/null +++ b/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/TypeDomainFact.kt @@ -0,0 +1,21 @@ +package org.usvm.dataflow.ts.infer + +sealed interface BackwardTypeDomainFact { + data object Zero : BackwardTypeDomainFact + + // Requirement + data class TypedVariable( + val variable: AccessPathBase, + val type: EtsTypeFact, + ) : BackwardTypeDomainFact +} + +sealed interface ForwardTypeDomainFact { + data object Zero : ForwardTypeDomainFact + + // Exact type + data class TypedVariable( + val variable: AccessPath, + val type: EtsTypeFact, // primitive or Object + ) : ForwardTypeDomainFact +} diff --git a/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/TypeGuesser.kt b/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/TypeGuesser.kt new file mode 100644 index 0000000000..87358c78e0 --- /dev/null +++ b/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/TypeGuesser.kt @@ -0,0 +1,269 @@ +/* + * Copyright 2022 UnitTestBot contributors (utbot.org) + *

+ * 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. + */ + +package org.usvm.dataflow.ts.infer + +import mu.KotlinLogging +import org.jacodb.ets.base.EtsClassType +import org.jacodb.ets.model.EtsClass +import org.jacodb.ets.model.EtsMethod +import org.jacodb.ets.model.EtsScene + +private val logger = KotlinLogging.logger {} + +fun guessTypes( + scene: EtsScene, + facts: Map>, + propertyNameToClasses: Map>, +): Map> { + return facts.mapValues { (method, types) -> + if (types.isEmpty()) { + logger.warn { "Facts are empty for method ${method.signature}" } + return@mapValues types + } + + val updatedTypes = types.mapValues { (accessPath, fact) -> + // logger.info { + // "Resolving a type for a fact \"$fact\" for access path \"$accessPath\" in the method \"$method\"" + // } + + val resultingType = fact.resolveType(scene, propertyNameToClasses) + // logger.info { "The result is $resultingType" } + + resultingType + } + + updatedTypes + } +} + +fun EtsTypeFact.resolveType( + scene: EtsScene, + propertyNameToClasses: Map>, +): EtsTypeFact { + return when (val simplifiedFact = simplify()) { + is EtsTypeFact.ArrayEtsTypeFact -> simplifiedFact.resolveArrayTypeFact(scene, propertyNameToClasses) + is EtsTypeFact.ObjectEtsTypeFact -> simplifiedFact.resolveObjectTypeFact(scene, propertyNameToClasses) + is EtsTypeFact.FunctionEtsTypeFact -> simplifiedFact + is EtsTypeFact.GuardedTypeFact -> TODO("guarded") + is EtsTypeFact.IntersectionEtsTypeFact -> { + val updatedTypes = simplifiedFact.types.mapTo(hashSetOf()) { + it.resolveType(scene, propertyNameToClasses) + } + EtsTypeFact.mkIntersectionType(updatedTypes).simplify() + } + + is EtsTypeFact.UnionEtsTypeFact -> { + val updatedTypes = simplifiedFact.types.mapNotNullTo(mutableSetOf()) { type -> + val resolvedType = type.resolveType(scene, propertyNameToClasses) + resolvedType.takeIf { it !is EtsTypeFact.AnyEtsTypeFact } + } + + if (updatedTypes.isEmpty()) { + EtsTypeFact.AnyEtsTypeFact + } else { + EtsTypeFact.mkUnionType(updatedTypes).simplify() + } + } + + else -> simplify() + } +} + +private fun EtsTypeFact.ObjectEtsTypeFact.resolveObjectTypeFact( + scene: EtsScene, + propertyNameToClasses: Map>, +): EtsTypeFact { + if (cls != null) { + return this + } + + val touchedPropertiesNames = properties.keys + val classesInSystem = collectSuitableClasses(touchedPropertiesNames, propertyNameToClasses) + + if (classesInSystem.isEmpty()) { + return tryToDetermineSpecialObjects(scene, touchedPropertiesNames, propertyNameToClasses) + } + + val suitableTypes = resolveTypesFromClasses(classesInSystem, scene, propertyNameToClasses) + + // TODO process arrays here (and strings) + + return when { + suitableTypes.isEmpty() -> error("Should be processed earlier") + suitableTypes.size == 1 -> suitableTypes.single() + suitableTypes.size in 2..5 -> EtsTypeFact.mkUnionType(suitableTypes).simplify() + else -> this + } +} + +private fun EtsTypeFact.ObjectEtsTypeFact.resolveTypesFromClasses( + classesInSystem: Iterable, + scene: EtsScene, + propertyNameToClasses: Map>, +) = classesInSystem + .mapTo(hashSetOf()) { cls -> + EtsTypeFact.ObjectEtsTypeFact( + cls = EtsClassType(signature = cls.signature), + properties = properties.mapValues { + it.value.resolveType(scene, propertyNameToClasses) + } + ) + } + +private fun collectSuitableClasses( + touchedPropertiesNames: Set, + propertyNameToClasses: Map>, +): Set { + val classesWithProperties = touchedPropertiesNames.map { propertyNameToClasses[it].orEmpty() } + + return classesWithProperties.reduceOrNull { a, b -> a intersect b }.orEmpty() +} + +private fun EtsTypeFact.ObjectEtsTypeFact.tryToDetermineSpecialObjects( + scene: EtsScene, + touchedPropertiesNames: Set, + propertyNameToClasses: Map>, +): EtsTypeFact.BasicType { + val indicesProperties = properties.filter { (k, _) -> k.toIntOrNull() != null } + if (indicesProperties.isNotEmpty()) { + val elementTypeFacts = indicesProperties.mapTo(hashSetOf()) { + it.value.resolveType(scene, propertyNameToClasses) + } + + val typeFact = EtsTypeFact.mkUnionType(elementTypeFacts).simplify() + + return EtsTypeFact.ArrayEtsTypeFact(typeFact) + } + + if ("length" in touchedPropertiesNames && "splice" in touchedPropertiesNames) { + return EtsTypeFact.ArrayEtsTypeFact(EtsTypeFact.AnyEtsTypeFact) + } + + return this +} + +private fun EtsTypeFact.ArrayEtsTypeFact.resolveArrayTypeFact( + scene: EtsScene, + propertyNameToClasses: Map>, +): EtsTypeFact.ArrayEtsTypeFact { + return if (elementType is EtsTypeFact.UnknownEtsTypeFact) { + this + } else { + val updatedElementType = elementType.resolveType(scene, propertyNameToClasses) + + if (updatedElementType === elementType) this else copy(elementType = elementType) + } +} + +fun EtsTypeFact.simplify(): EtsTypeFact { + return when (this) { + is EtsTypeFact.UnionEtsTypeFact -> simplifyUnionTypeFact() + is EtsTypeFact.IntersectionEtsTypeFact -> simplifyIntersectionTypeFact() + is EtsTypeFact.GuardedTypeFact -> TODO("Guarded type facts are unsupported in simplification") + is EtsTypeFact.ArrayEtsTypeFact -> { + val elementType = elementType.simplify() + + if (elementType === this.elementType) this else copy(elementType = elementType) + } + + is EtsTypeFact.ObjectEtsTypeFact -> { + if (cls != null) { + return this + } + + val props = properties.mapValues { it.value.simplify() } + copy(properties = props) + } + + else -> this + } +} + +private fun EtsTypeFact.IntersectionEtsTypeFact.simplifyIntersectionTypeFact(): EtsTypeFact { + val simplifiedArgs = types.map { it.simplify() } + + simplifiedArgs.singleOrNull()?.let { return it } + + val updatedTypeFacts = hashSetOf() + + val (objectClassFacts, otherFacts) = simplifiedArgs.partition { + it is EtsTypeFact.ObjectEtsTypeFact && it.cls == null + } + + updatedTypeFacts.addAll(otherFacts) + + if (objectClassFacts.isNotEmpty()) { + val allProperties = hashMapOf>().withDefault { hashSetOf() } + + objectClassFacts.forEach { fact -> + fact as EtsTypeFact.ObjectEtsTypeFact + + fact.properties.forEach { (name, propertyFact) -> + allProperties.getValue(name).add(propertyFact) + } + } + + val mergedAllProperties = hashMapOf() + allProperties.forEach { (name, propertyFact) -> + mergedAllProperties[name] = EtsTypeFact.mkUnionType(propertyFact) + } + + updatedTypeFacts += EtsTypeFact.ObjectEtsTypeFact(cls = null, properties = mergedAllProperties) + } + + return EtsTypeFact.mkIntersectionType(updatedTypeFacts) +} + +private fun EtsTypeFact.UnionEtsTypeFact.simplifyUnionTypeFact(): EtsTypeFact { + val simplifiedArgs = types.map { it.simplify() } + + simplifiedArgs.singleOrNull()?.let { return it } + + val updatedTypeFacts = hashSetOf() + + var atLeastOneNonEmptyObjectFound = false + var emptyTypeObjectFact: EtsTypeFact? = null + + simplifiedArgs.forEach { + if (it !is EtsTypeFact.ObjectEtsTypeFact) { + updatedTypeFacts += it + return@forEach + } + + if (it.cls != null) { + atLeastOneNonEmptyObjectFound = true + updatedTypeFacts += it + return@forEach + } + + if (it.properties.isEmpty() && emptyTypeObjectFact == null) { + emptyTypeObjectFact = it + } else { + updatedTypeFacts += it + atLeastOneNonEmptyObjectFound = true + } + } + + // take a fact `Object {}` only if there were no other objects in the facts + emptyTypeObjectFact?.let { + if (!atLeastOneNonEmptyObjectFound) { + updatedTypeFacts += it + } + } + + return EtsTypeFact.mkUnionType(updatedTypeFacts) +} diff --git a/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/TypeInferenceManager.kt b/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/TypeInferenceManager.kt new file mode 100644 index 0000000000..09e37311b5 --- /dev/null +++ b/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/TypeInferenceManager.kt @@ -0,0 +1,712 @@ +package org.usvm.dataflow.ts.infer + +import kotlinx.coroutines.CompletableDeferred +import kotlinx.coroutines.CoroutineScope +import kotlinx.coroutines.CoroutineStart +import kotlinx.coroutines.Dispatchers +import kotlinx.coroutines.cancelAndJoin +import kotlinx.coroutines.coroutineScope +import kotlinx.coroutines.launch +import kotlinx.coroutines.runBlocking +import kotlinx.coroutines.withTimeout +import org.jacodb.ets.base.ANONYMOUS_CLASS_PREFIX +import org.jacodb.ets.base.CONSTRUCTOR_NAME +import org.jacodb.ets.base.EtsReturnStmt +import org.jacodb.ets.base.EtsStmt +import org.jacodb.ets.base.EtsStringType +import org.jacodb.ets.base.EtsType +import org.jacodb.ets.base.INSTANCE_INIT_METHOD_NAME +import org.jacodb.ets.graph.findDominators +import org.jacodb.ets.model.EtsMethod +import org.jacodb.impl.cfg.graphs.GraphDominators +import org.usvm.dataflow.graph.reversed +import org.usvm.dataflow.ifds.ControlEvent +import org.usvm.dataflow.ifds.Edge +import org.usvm.dataflow.ifds.Manager +import org.usvm.dataflow.ifds.QueueEmptinessChanged +import org.usvm.dataflow.ifds.SingletonUnit +import org.usvm.dataflow.ifds.UniRunner +import org.usvm.dataflow.ts.graph.EtsApplicationGraph +import org.usvm.dataflow.ts.infer.EtsTypeFact.Companion.allStringProperties +import org.usvm.dataflow.ts.util.EtsTraits +import java.util.concurrent.ConcurrentHashMap +import kotlin.time.Duration.Companion.seconds + +class TypeInferenceManager( + val traits: EtsTraits, + val graph: EtsApplicationGraph, +) : Manager { + private lateinit var runnerFinished: CompletableDeferred + + private val backwardSummaries = ConcurrentHashMap>() + private val forwardSummaries = ConcurrentHashMap>() + + private val methodDominatorsCache = ConcurrentHashMap>() + + private fun methodDominators(method: EtsMethod): GraphDominators = + methodDominatorsCache.computeIfAbsent(method) { + method.flowGraph().findDominators() + } + + private val savedTypes: ConcurrentHashMap> = ConcurrentHashMap() + + fun analyze( + entrypoints: List, + allMethods: List = entrypoints, + doAddKnownTypes: Boolean = true, + doInferAllLocals: Boolean = true, + ): TypeInferenceResult = runBlocking(Dispatchers.Default) { + val methodTypeScheme = collectSummaries( + startMethods = entrypoints, + doAddKnownTypes = doAddKnownTypes, + ) + val remainingMethodsForAnalysis = allMethods.filter { it !in methodTypeScheme.keys } + + val updatedTypeScheme = if (remainingMethodsForAnalysis.isEmpty()) { + methodTypeScheme + } else { + collectSummaries( + startMethods = remainingMethodsForAnalysis, + doAddKnownTypes = doAddKnownTypes, + ) + } + + createResultsFromSummaries(updatedTypeScheme, doInferAllLocals) + } + + private suspend fun collectSummaries( + startMethods: List, + doAddKnownTypes: Boolean = true, + ): Map = coroutineScope { + logger.info { "Preparing backward analysis" } + val backwardGraph = graph.reversed + val backwardAnalyzer = BackwardAnalyzer(backwardGraph, savedTypes, ::methodDominators) + val backwardRunner = UniRunner( + traits = traits, + manager = this@TypeInferenceManager, + graph = backwardGraph, + analyzer = backwardAnalyzer, + unitResolver = { SingletonUnit }, + unit = SingletonUnit, + zeroFact = BackwardTypeDomainFact.Zero, + ) + + val backwardJob = launch(start = CoroutineStart.LAZY) { + backwardRunner.run(startMethods) + } + + logger.info { "Running backward analysis" } + runnerFinished = CompletableDeferred() + backwardJob.start() + runnerFinished.await() + backwardJob.cancelAndJoin() + logger.info { "Backward analysis finished" } + + // logger.info { + // buildString { + // appendLine("Backward summaries: (${backwardSummaries.size})") + // for ((method, summaries) in backwardSummaries) { + // appendLine("=== Backward summaries for ${method.signature.enclosingClass.name}::${method.name}: (${summaries.size})") + // for (summary in summaries) { + // appendLine(" ${summary.initialFact} -> ${summary.exitFact}") + // } + // } + // } + // } + + val methodTypeScheme = methodTypeScheme() + + logger.info { + buildString { + appendLine("Backward types:") + for ((method, typeFacts) in methodTypeScheme) { + appendLine("Backward types for ${method.enclosingClass.name}::${method.name} in ${method.enclosingClass.file}:") + for ((base, fact) in typeFacts.types.entries.sortedBy { + when (val key = it.key) { + is AccessPathBase.This -> 0 + is AccessPathBase.Arg -> key.index + 1 + else -> 1_000_000 + } + }) { + appendLine("$base: ${fact.toPrettyString()}") + } + } + } + } + + val typeInfo: Map = savedTypes.mapValues { (type, facts) -> + val typeFact = EtsTypeFact.ObjectEtsTypeFact(type, properties = emptyMap()) + facts.fold(typeFact as EtsTypeFact) { acc, it -> + acc.intersect(it) ?: run { + logger.error { "Empty intersection type: $acc & $it" } + acc + } + } + } + + logger.info { "Preparing forward analysis" } + val forwardGraph = graph + val forwardAnalyzer = ForwardAnalyzer(forwardGraph, methodTypeScheme, typeInfo, doAddKnownTypes) + val forwardRunner = UniRunner( + traits = traits, + manager = this@TypeInferenceManager, + graph = forwardGraph, + analyzer = forwardAnalyzer, + unitResolver = { SingletonUnit }, + unit = SingletonUnit, + zeroFact = ForwardTypeDomainFact.Zero, + ) + + val forwardJob = launch(start = CoroutineStart.LAZY) { + forwardRunner.run(startMethods) + } + + logger.info { "Running forward analysis" } + runnerFinished = CompletableDeferred() + forwardJob.start() + withTimeout(3600.seconds) { + runnerFinished.await() + } + forwardJob.cancelAndJoin() + logger.info { "Forward analysis finished" } + + // logger.info { + // buildString { + // appendLine("Forward summaries: (${forwardSummaries.size})") + // for ((method, summaries) in forwardSummaries) { + // appendLine("=== Forward summaries for ${method.signature.enclosingClass.name}::${method.name}: (${summaries.size})") + // for (summary in summaries) { + // appendLine(" ${summary.initialFact} -> ${summary.exitFact}") + // } + // } + // } + // } + + methodTypeScheme + } + + private fun createResultsFromSummaries( + methodTypeScheme: Map, + doInferAllLocals: Boolean, + ): TypeInferenceResult { + val refinedTypes = refineMethodTypes(methodTypeScheme).toMutableMap() + logger.info { + buildString { + appendLine("Forward types:") + for ((method, typeFacts) in refinedTypes) { + appendLine("Forward types for ${method.signature.enclosingClass.name}::${method.name} in ${method.signature.enclosingClass.file}:") + for ((base, fact) in typeFacts.types.entries.sortedBy { + when (val key = it.key) { + is AccessPathBase.This -> 0 + is AccessPathBase.Arg -> key.index + 1 + else -> 1_000_000 + } + }) { + appendLine("$base: ${fact.toPrettyString()}") + } + } + } + } + + // Infer types for 'this' in each class + val inferredCombinedThisTypes = run { + val allClasses = methodTypeScheme.keys + .map { it.enclosingClass } + .distinct() + .map { sig -> graph.cp.projectAndSdkClasses.first { cls -> cls.signature == sig } } + .filterNot { it.name.startsWith(ANONYMOUS_CLASS_PREFIX) } + allClasses.mapNotNull { cls -> + val combinedBackwardType = + methodTypeScheme.asSequence().filter { (method, _) -> method in (cls.methods + cls.ctor) } + .mapNotNull { (_, facts) -> facts.types[AccessPathBase.This] }.reduceOrNull { acc, type -> + val intersection = acc.intersect(type) + + if (intersection == null) { + System.err.println("Empty intersection type: $acc & $type") + } + + intersection ?: acc + } + logger.info { + buildString { + appendLine("Combined backward type for This in class '${cls.signature}': ${combinedBackwardType?.toPrettyString()}") + } + } + + if (combinedBackwardType == null) { + return@mapNotNull null + } + + val typeFactsOnThisMethods = forwardSummaries.asSequence() + .filter { (method, _) -> method.enclosingClass == cls.signature } + .filter { (method, _) -> method.name != INSTANCE_INIT_METHOD_NAME } + .flatMap { (_, summaries) -> summaries.asSequence() } + .mapNotNull { it.initialFact as? ForwardTypeDomainFact.TypedVariable } + .filter { it.variable.base is AccessPathBase.This } + .toList() + .distinct() + + val typeFactsOnThisCtor = forwardSummaries.asSequence() + .filter { (method, _) -> method.enclosingClass == cls.signature } + .filter { (method, _) -> method.name == CONSTRUCTOR_NAME || method.name == INSTANCE_INIT_METHOD_NAME } + .flatMap { (_, summaries) -> summaries.asSequence() } + .mapNotNull { it.exitFact as? ForwardTypeDomainFact.TypedVariable } + .filter { it.variable.base is AccessPathBase.This } + .toList() + .distinct() + + val typeFactsOnThis = (typeFactsOnThisMethods + typeFactsOnThisCtor).distinct() + + val propertyRefinements = typeFactsOnThis + .groupBy({ it.variable.accesses }, { it.type }) + .mapValues { (_, types) -> types.reduce { acc, t -> acc.union(t) } } + + logger.info { + buildString { + appendLine("Property refinements for This in class '${cls.signature}':") + for ((property, propertyType) in propertyRefinements.toList() + .sortedBy { it.first.joinToString(".") }) { + appendLine(" this.${property.joinToString(".")}: $propertyType") + } + } + } + + var refined: EtsTypeFact = combinedBackwardType + for ((property, propertyType) in propertyRefinements) { + refined = refined.refineProperty(property, propertyType) ?: this@TypeInferenceManager.run { + System.err.println("Empty intersection type: $combinedBackwardType[$property] & $propertyType") + refined + } + } + + combinedBackwardType.let {} + typeFactsOnThisMethods.let {} + typeFactsOnThisCtor.let {} + typeFactsOnThis.let {} + propertyRefinements.let {} + cls.let {} + + cls.signature to refined + }.toMap() + } + logger.info { + buildString { + appendLine("Combined and refined types for This:") + for ((cls, type) in inferredCombinedThisTypes) { + appendLine("Combined This in class '${cls}': ${type.toPrettyString()}") + } + } + } + + // Infer return types for each method + val inferredReturnTypes = run { + forwardSummaries.asSequence().mapNotNull { (method, summaries) -> + val typeFacts = summaries.asSequence().map { it.exitVertex }.mapNotNull { + val stmt = it.statement as? EtsReturnStmt ?: return@mapNotNull null + val fact = it.fact as? ForwardTypeDomainFact.TypedVariable ?: return@mapNotNull null + val r = stmt.returnValue?.toPath() ?: return@mapNotNull null + check(r.accesses.isEmpty()) + if (fact.variable.base != r.base) return@mapNotNull null + r.base to fact + }.groupBy({ it.first }, { it.second }) + + val returnFact = typeFacts.mapValues { (_, typeRefinements) -> + val propertyRefinements = typeRefinements.groupBy({ it.variable.accesses }, { it.type }) + .mapValues { (_, types) -> types.reduce { acc, t -> acc.union(t) } } + + val rootType = propertyRefinements[emptyList()] ?: run { + if (propertyRefinements.keys.any { it.isNotEmpty() }) { + EtsTypeFact.ObjectEtsTypeFact(null, emptyMap()) + } else { + EtsTypeFact.AnyEtsTypeFact + } + } + + val refined = rootType.refineProperties(emptyList(), propertyRefinements) + + refined + }.values.reduceOrNull { acc, type -> acc.union(type) } + if (returnFact != null) { + method to returnFact + } else { + null + } + }.toMap() + } + logger.info { + buildString { + appendLine("Return types:") + for ((method, type) in inferredReturnTypes) { + appendLine("Return type for ${method.signature.enclosingClass.file}::${method.signature.enclosingClass.name}::${method.name}: ${type.toPrettyString()}") + } + } + } + + val inferredLocalTypes: Map>? = if (doInferAllLocals) { + forwardSummaries.asSequence().map { (method, summaries) -> + val typeFacts = summaries.asSequence() + .mapNotNull { it.exitVertex.fact as? ForwardTypeDomainFact.TypedVariable } + .filter { it.variable.base is AccessPathBase.Local } + .groupBy { it.variable.base } + + val localTypes = typeFacts.mapValues { (_, typeFacts) -> + val propertyRefinements = typeFacts + .groupBy({ it.variable.accesses }, { it.type }) + .mapValues { (_, types) -> types.reduce { acc, t -> acc.union(t) } } + + val rootType = propertyRefinements[emptyList()] ?: run { + if (propertyRefinements.keys.any { it.isNotEmpty() }) { + EtsTypeFact.ObjectEtsTypeFact(null, emptyMap()) + } else { + EtsTypeFact.AnyEtsTypeFact + } + } + + val refined = rootType.refineProperties(emptyList(), propertyRefinements) + + refined + } + + method to localTypes + }.toMap() + } else { + null + } + + if (inferredLocalTypes != null) { + logger.info { + buildString { + appendLine("Local types:") + for ((method, localTypes) in inferredLocalTypes) { + appendLine("Local types for ${method.signature.enclosingClass.name}::${method.name} in ${method.signature.enclosingClass.file}:") + for ((base, fact) in localTypes.entries.sortedBy { (it.key as AccessPathBase.Local).name }) { + appendLine("$base: ${fact.toPrettyString()}") + } + } + } + } + + for ((method, localFacts) in inferredLocalTypes) { + val facts = refinedTypes.getValue(method) + refinedTypes[method] = facts.copy(types = facts.types + localFacts) + } + } + + val inferredTypes = refinedTypes + // Extract 'types': + .mapValues { (_, facts) -> facts.types } + // Sort by 'base': + .mapValues { (_, types) -> + types.entries.sortedBy { + when (val key = it.key) { + is AccessPathBase.This -> 0 + is AccessPathBase.Arg -> key.index + 1 + else -> 1_000_000 + } + }.associate { it.key to it.value } + }.mapValues { (_, methodFacts) -> + methodFacts.mapValues { (_, fact) -> + fact.simplify() + } + } + + return TypeInferenceResult( + inferredTypes = inferredTypes, + inferredReturnType = inferredReturnTypes, + inferredCombinedThisType = inferredCombinedThisTypes, + ) + } + + private fun methodTypeScheme(): Map = + backwardSummaries.mapValues { (method, summaries) -> + buildMethodTypeScheme(method, summaries) + } + + private fun refineMethodTypes(schema: Map): Map = + schema.mapValues { (method, facts) -> + val summaries = forwardSummaries[method].orEmpty() + refineMethodTypes(facts, summaries) + } + + private fun buildMethodTypeScheme( + method: EtsMethod, + summaries: Iterable, + ): EtsMethodTypeFacts { + val types = summaries + .mapNotNull { it.exitFact as? BackwardTypeDomainFact.TypedVariable } + .groupBy({ it.variable }, { it.type }) + .filter { (base, _) -> base is AccessPathBase.This || base is AccessPathBase.Arg } + .mapValues { (_, typeFacts) -> + typeFacts.reduce { acc, typeFact -> + val intersection = acc.intersect(typeFact) + + if (intersection == null) { + System.err.println("Empty intersection type: $acc & $typeFact") + } + + intersection ?: acc + } + } + + return EtsMethodTypeFacts(method, types) + } + + private fun refineMethodTypes( + facts: EtsMethodTypeFacts, // backward types + summaries: Iterable, + ): EtsMethodTypeFacts { + // Contexts + val typeFacts = summaries.asSequence() + .mapNotNull { it.initialFact as? ForwardTypeDomainFact.TypedVariable } + .filter { it.variable.base is AccessPathBase.This || it.variable.base is AccessPathBase.Arg } + .groupBy { it.variable.base } + + val refinedTypes = facts.types.mapValues { (base, type) -> + val typeRefinements = typeFacts[base] ?: return@mapValues type + + val propertyRefinements = typeRefinements + .groupBy({ it.variable.accesses }, { it.type }) + .mapValues { (_, types) -> types.reduce { acc, t -> acc.union(t) } } + + val rootType = propertyRefinements[emptyList()] ?: run { + if (propertyRefinements.keys.any { it.isNotEmpty() }) { + EtsTypeFact.ObjectEtsTypeFact(null, emptyMap()) + } else { + EtsTypeFact.AnyEtsTypeFact + } + } + + val refined = rootType.refineProperties(emptyList(), propertyRefinements) + + refined + } + + typeFacts.let {} + + return EtsMethodTypeFacts(facts.method, refinedTypes) + } + + private fun EtsTypeFact.refineProperties( + pathFromRootObject: List, + typeRefinements: Map, EtsTypeFact>, + ): EtsTypeFact = when (this) { + is EtsTypeFact.NumberEtsTypeFact -> this + is EtsTypeFact.StringEtsTypeFact -> this + is EtsTypeFact.FunctionEtsTypeFact -> this + is EtsTypeFact.AnyEtsTypeFact -> this + is EtsTypeFact.BooleanEtsTypeFact -> this + is EtsTypeFact.NullEtsTypeFact -> this + is EtsTypeFact.UndefinedEtsTypeFact -> this + + is EtsTypeFact.UnknownEtsTypeFact -> { + // logger.warn { "Unknown type after forward analysis" } + EtsTypeFact.AnyEtsTypeFact + } + + is EtsTypeFact.ArrayEtsTypeFact -> { + // TODO: array types + logger.error { "TODO: Array type $this" } + + val elementPath = pathFromRootObject + ElementAccessor + val refinedElemType = + typeRefinements[elementPath]?.intersect(elementType) ?: elementType // todo: mb exception??? + val elemType = refinedElemType.refineProperties(elementPath, typeRefinements) + + EtsTypeFact.ArrayEtsTypeFact(elemType) + } + + is EtsTypeFact.ObjectEtsTypeFact -> refineProperties(pathFromRootObject, typeRefinements) + + is EtsTypeFact.UnionEtsTypeFact -> EtsTypeFact.mkUnionType( + types.mapTo(hashSetOf()) { + it.refineProperties( + pathFromRootObject, + typeRefinements, + ) + } + ) + + is EtsTypeFact.IntersectionEtsTypeFact -> EtsTypeFact.mkIntersectionType( + types.mapTo(hashSetOf()) { + it.refineProperties( + pathFromRootObject, + typeRefinements, + ) + } + ) + + is EtsTypeFact.GuardedTypeFact -> type + .refineProperties(pathFromRootObject, typeRefinements) + .withGuard(guard, guardNegated) + } + + private fun EtsTypeFact.ObjectEtsTypeFact.refineProperties( + pathFromRootObject: List, + typeRefinements: Map, EtsTypeFact>, + ): EtsTypeFact { + val refinedProperties = properties.mapValues { (propertyName, type) -> + val propertyAccessor = FieldAccessor(propertyName) + val propertyPath = pathFromRootObject + propertyAccessor + val refinedType = typeRefinements[propertyPath]?.intersect(type) ?: type // todo: mb exception??? + refinedType.refineProperties(propertyPath, typeRefinements) + } + return EtsTypeFact.ObjectEtsTypeFact(cls, refinedProperties) + } + + private fun EtsTypeFact.refineProperty(property: List, type: EtsTypeFact): EtsTypeFact? = when (this) { + is EtsTypeFact.BasicType -> refineProperty(property, type) + + is EtsTypeFact.GuardedTypeFact -> this.type.refineProperty(property, type)?.withGuard(guard, guardNegated) + + is EtsTypeFact.IntersectionEtsTypeFact -> EtsTypeFact.mkIntersectionType(types.mapTo(hashSetOf()) { + it.refineProperty( + property, + type + ) ?: return null + }) + + is EtsTypeFact.UnionEtsTypeFact -> EtsTypeFact.mkUnionType(types.mapNotNullTo(hashSetOf()) { + it.refineProperty( + property, + type + ) + }) + } + + private fun EtsTypeFact.BasicType.refineProperty( + property: List, + type: EtsTypeFact, + ): EtsTypeFact? { + when (this) { + EtsTypeFact.AnyEtsTypeFact, + EtsTypeFact.FunctionEtsTypeFact, + EtsTypeFact.NumberEtsTypeFact, + EtsTypeFact.BooleanEtsTypeFact, + EtsTypeFact.StringEtsTypeFact, + EtsTypeFact.NullEtsTypeFact, + EtsTypeFact.UndefinedEtsTypeFact, + -> return if (property.isNotEmpty()) this else intersect(type) + + is EtsTypeFact.ArrayEtsTypeFact -> { + // TODO: the following check(property.size == 1) fails on multiple projects + // check(property.size == 1) + if (property.size == 1) { + // val p = property.single() + // check(p is ElementAccessor) + val t = elementType.intersect(type) ?: error("Empty intersection") + return EtsTypeFact.ArrayEtsTypeFact(elementType = t) + } else { + return EtsTypeFact.AnyEtsTypeFact + } + } + + is EtsTypeFact.UnknownEtsTypeFact -> { + // .f.g:T -> {f: {g: T}} + // .f[i].g:T -> {f: Array<{g: T}>} + var t = type + for (p in property.reversed()) { + if (p is FieldAccessor) { + t = EtsTypeFact.ObjectEtsTypeFact( + cls = null, + properties = mapOf(p.name to t), + ) + } else { + t = EtsTypeFact.ArrayEtsTypeFact(elementType = t) + } + } + return t + } + + is EtsTypeFact.ObjectEtsTypeFact -> { + val propertyAccessor = property.firstOrNull() as? FieldAccessor + if (propertyAccessor == null) { + // TODO: handle 'type=union' by exploding it into multiple ObjectFacts (later combined with union) with class names from union. + if (type is EtsTypeFact.UnionEtsTypeFact) { + return type.types.map { + refineProperty(property, it) ?: return null + }.reduce { acc: EtsTypeFact, t: EtsTypeFact -> acc.union(t) } + } + + if (type is EtsTypeFact.StringEtsTypeFact) { + // intersect(this:object, type:string) + + if (cls == EtsStringType) return type + if (cls != null) return null + + val intersectionProperties = properties + .filter { it.key in allStringProperties } + .mapValues { (_, type) -> + // TODO: intersect with the corresponding type of String's property + type + } + + return EtsTypeFact.ObjectEtsTypeFact(cls, intersectionProperties) + } + + if (type is EtsTypeFact.NullEtsTypeFact) { + // intersect(this:object, type:null) + // return EtsTypeFact.NullEtsTypeFact + return EtsTypeFact.mkUnionType(this, EtsTypeFact.NullEtsTypeFact) + } + + if (type is EtsTypeFact.UndefinedEtsTypeFact) { + // intersect(this:object, type:undefined) + // return EtsTypeFact.UndefinedEtsTypeFact + return EtsTypeFact.mkUnionType(this, EtsTypeFact.UndefinedEtsTypeFact) + } + + if (type !is EtsTypeFact.ObjectEtsTypeFact || cls != null) { + // todo: hack + if (type is EtsTypeFact.AnyEtsTypeFact) return this + + // TODO("Unexpected: $this & $type") + return this + } + + return EtsTypeFact.ObjectEtsTypeFact(type.cls, properties) + } + + val propertyType = properties[propertyAccessor.name] ?: return this + val refinedProperty = propertyType.refineProperty(property.drop(1), type) ?: return null + val properties = this.properties + (propertyAccessor.name to refinedProperty) + return EtsTypeFact.ObjectEtsTypeFact(cls, properties) + } + } + } + + override fun handleEvent(event: AnalyzerEvent) { + when (event) { + is BackwardSummaryAnalyzerEvent -> { + backwardSummaries.computeIfAbsent(event.method) { + ConcurrentHashMap.newKeySet() + }.add(event) + } + + is ForwardSummaryAnalyzerEvent -> { + forwardSummaries.computeIfAbsent(event.method) { + ConcurrentHashMap.newKeySet() + }.add(event) + } + } + } + + override fun subscribeOnSummaryEdges( + method: EtsMethod, + scope: CoroutineScope, + handler: (Edge) -> Unit, + ) { + error("No cross unit subscriptions") + } + + override fun handleControlEvent(event: ControlEvent) { + if (event is QueueEmptinessChanged) { + if (event.isEmpty) { + runnerFinished.complete(Unit) + } + } + } + + companion object { + val logger = mu.KotlinLogging.logger {} + } +} diff --git a/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/TypeInferenceResult.kt b/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/TypeInferenceResult.kt new file mode 100644 index 0000000000..d4d991bab7 --- /dev/null +++ b/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/TypeInferenceResult.kt @@ -0,0 +1,57 @@ +/* + * Copyright 2022 UnitTestBot contributors (utbot.org) + *

+ * 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. + */ + +package org.usvm.dataflow.ts.infer + +import org.jacodb.ets.model.EtsClass +import org.jacodb.ets.model.EtsClassSignature +import org.jacodb.ets.model.EtsMethod +import org.jacodb.ets.model.EtsScene + +data class TypeInferenceResult( + val inferredTypes: Map>, + val inferredReturnType: Map, + val inferredCombinedThisType: Map, +) { + fun withGuessedTypes(scene: EtsScene): TypeInferenceResult { + val propertyNameToClasses = precalculateCaches(scene) + + return TypeInferenceResult( + inferredTypes = guessTypes(scene, inferredTypes, propertyNameToClasses), + inferredReturnType = inferredReturnType.mapValues { (_, fact) -> + fact.resolveType(scene, propertyNameToClasses) + }, + inferredCombinedThisType = inferredCombinedThisType.mapValues { (_, fact) -> + fact.resolveType(scene, propertyNameToClasses = propertyNameToClasses) + }, + ) + } + + private fun precalculateCaches(scene: EtsScene): Map> { + val result = hashMapOf>() + + scene.projectAndSdkClasses.forEach { clazz -> + clazz.methods.forEach { + result.computeIfAbsent(it.name) { hashSetOf() }.add(clazz) + } + clazz.fields.forEach { + result.computeIfAbsent(it.name) { hashSetOf() }.add(clazz) + } + } + + return result + } +} diff --git a/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/annotation/ExprTypeAnnotator.kt b/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/annotation/ExprTypeAnnotator.kt new file mode 100644 index 0000000000..16cdbf1419 --- /dev/null +++ b/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/annotation/ExprTypeAnnotator.kt @@ -0,0 +1,279 @@ +/* + * Copyright 2022 UnitTestBot contributors (utbot.org) + *

+ * 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. + */ + +package org.usvm.dataflow.ts.infer.annotation + +import org.jacodb.ets.base.EtsAddExpr +import org.jacodb.ets.base.EtsAndExpr +import org.jacodb.ets.base.EtsAwaitExpr +import org.jacodb.ets.base.EtsBitAndExpr +import org.jacodb.ets.base.EtsBitNotExpr +import org.jacodb.ets.base.EtsBitOrExpr +import org.jacodb.ets.base.EtsBitXorExpr +import org.jacodb.ets.base.EtsCastExpr +import org.jacodb.ets.base.EtsClassType +import org.jacodb.ets.base.EtsCommaExpr +import org.jacodb.ets.base.EtsDeleteExpr +import org.jacodb.ets.base.EtsDivExpr +import org.jacodb.ets.base.EtsEntity +import org.jacodb.ets.base.EtsEqExpr +import org.jacodb.ets.base.EtsExpExpr +import org.jacodb.ets.base.EtsExpr +import org.jacodb.ets.base.EtsGtEqExpr +import org.jacodb.ets.base.EtsGtExpr +import org.jacodb.ets.base.EtsInExpr +import org.jacodb.ets.base.EtsInstanceCallExpr +import org.jacodb.ets.base.EtsInstanceOfExpr +import org.jacodb.ets.base.EtsLeftShiftExpr +import org.jacodb.ets.base.EtsLengthExpr +import org.jacodb.ets.base.EtsLocal +import org.jacodb.ets.base.EtsLtEqExpr +import org.jacodb.ets.base.EtsLtExpr +import org.jacodb.ets.base.EtsMulExpr +import org.jacodb.ets.base.EtsNegExpr +import org.jacodb.ets.base.EtsNewArrayExpr +import org.jacodb.ets.base.EtsNewExpr +import org.jacodb.ets.base.EtsNotEqExpr +import org.jacodb.ets.base.EtsNotExpr +import org.jacodb.ets.base.EtsNullishCoalescingExpr +import org.jacodb.ets.base.EtsOrExpr +import org.jacodb.ets.base.EtsPostDecExpr +import org.jacodb.ets.base.EtsPostIncExpr +import org.jacodb.ets.base.EtsPreDecExpr +import org.jacodb.ets.base.EtsPreIncExpr +import org.jacodb.ets.base.EtsPtrCallExpr +import org.jacodb.ets.base.EtsRemExpr +import org.jacodb.ets.base.EtsRightShiftExpr +import org.jacodb.ets.base.EtsStaticCallExpr +import org.jacodb.ets.base.EtsStrictEqExpr +import org.jacodb.ets.base.EtsStrictNotEqExpr +import org.jacodb.ets.base.EtsSubExpr +import org.jacodb.ets.base.EtsTernaryExpr +import org.jacodb.ets.base.EtsTypeOfExpr +import org.jacodb.ets.base.EtsUnaryPlusExpr +import org.jacodb.ets.base.EtsUnsignedRightShiftExpr +import org.jacodb.ets.base.EtsValue +import org.jacodb.ets.base.EtsVoidExpr +import org.jacodb.ets.base.EtsYieldExpr +import org.jacodb.ets.model.EtsScene + +class ExprTypeAnnotator( + private val valueAnnotator: ValueTypeAnnotator, + private val scene: EtsScene, +) : EtsExpr.Visitor { + private fun inferEntity(entity: EtsEntity) = when (entity) { + is EtsExpr -> entity.accept(this) + is EtsValue -> entity.accept(valueAnnotator) + else -> error("Unsupported entity") + } + + private fun inferValue(value: EtsValue) = value.accept(valueAnnotator) + + override fun visit(expr: EtsNewExpr) = expr + + override fun visit(expr: EtsNewArrayExpr) = expr + + override fun visit(expr: EtsLengthExpr) = expr + + override fun visit(expr: EtsCastExpr) = expr.copy(arg = inferEntity(expr.arg)) + + override fun visit(expr: EtsInstanceOfExpr) = expr.copy(arg = inferEntity(expr.arg)) + + override fun visit(expr: EtsDeleteExpr) = expr.copy(arg = inferEntity(expr.arg)) + + override fun visit(expr: EtsAwaitExpr) = expr.copy(arg = inferEntity(expr.arg)) + + override fun visit(expr: EtsYieldExpr) = expr.copy(arg = inferEntity(expr.arg)) + + override fun visit(expr: EtsTypeOfExpr) = expr.copy(arg = inferEntity(expr.arg)) + + override fun visit(expr: EtsVoidExpr) = expr.copy(arg = inferEntity(expr.arg)) + + override fun visit(expr: EtsNotExpr) = expr.copy(arg = inferEntity(expr.arg)) + + override fun visit(expr: EtsBitNotExpr) = expr.copy(arg = inferEntity(expr.arg)) + + override fun visit(expr: EtsNegExpr) = expr.copy(arg = inferEntity(expr.arg)) + + override fun visit(expr: EtsUnaryPlusExpr) = expr.copy(arg = inferEntity(expr.arg)) + + override fun visit(expr: EtsPreIncExpr) = expr.copy(arg = inferEntity(expr.arg)) + + override fun visit(expr: EtsPreDecExpr) = expr.copy(arg = inferEntity(expr.arg)) + + override fun visit(expr: EtsPostIncExpr) = expr.copy(arg = inferEntity(expr.arg)) + + override fun visit(expr: EtsPostDecExpr) = expr.copy(arg = inferEntity(expr.arg)) + + override fun visit(expr: EtsEqExpr) = expr.copy( + left = inferEntity(expr.left), + right = inferEntity(expr.right) + ) + + override fun visit(expr: EtsNotEqExpr) = expr.copy( + left = inferEntity(expr.left), + right = inferEntity(expr.right) + ) + + override fun visit(expr: EtsStrictEqExpr) = expr.copy( + left = inferEntity(expr.left), + right = inferEntity(expr.right) + ) + + override fun visit(expr: EtsStrictNotEqExpr) = expr.copy( + left = inferEntity(expr.left), + right = inferEntity(expr.right) + ) + + override fun visit(expr: EtsLtExpr) = expr.copy( + left = inferEntity(expr.left), + right = inferEntity(expr.right) + ) + + override fun visit(expr: EtsLtEqExpr) = expr.copy( + left = inferEntity(expr.left), + right = inferEntity(expr.right) + ) + + override fun visit(expr: EtsGtExpr) = expr.copy( + left = inferEntity(expr.left), + right = inferEntity(expr.right) + ) + + override fun visit(expr: EtsGtEqExpr) = expr.copy( + left = inferEntity(expr.left), + right = inferEntity(expr.right) + ) + + override fun visit(expr: EtsInExpr) = expr.copy( + left = inferEntity(expr.left), + right = inferEntity(expr.right) + ) + + override fun visit(expr: EtsAddExpr) = expr.copy( + left = inferEntity(expr.left), + right = inferEntity(expr.right) + ) + + override fun visit(expr: EtsSubExpr) = expr.copy( + left = inferEntity(expr.left), + right = inferEntity(expr.right) + ) + + override fun visit(expr: EtsMulExpr) = expr.copy( + left = inferEntity(expr.left), + right = inferEntity(expr.right) + ) + + override fun visit(expr: EtsDivExpr) = expr.copy( + left = inferEntity(expr.left), + right = inferEntity(expr.right) + ) + + override fun visit(expr: EtsRemExpr) = expr.copy( + left = inferEntity(expr.left), + right = inferEntity(expr.right) + ) + + override fun visit(expr: EtsExpExpr) = expr.copy( + left = inferEntity(expr.left), + right = inferEntity(expr.right) + ) + + override fun visit(expr: EtsBitAndExpr) = expr.copy( + left = inferEntity(expr.left), + right = inferEntity(expr.right) + ) + + override fun visit(expr: EtsBitOrExpr) = expr.copy( + left = inferEntity(expr.left), + right = inferEntity(expr.right) + ) + + override fun visit(expr: EtsBitXorExpr) = expr.copy( + left = inferEntity(expr.left), + right = inferEntity(expr.right) + ) + + override fun visit(expr: EtsLeftShiftExpr) = expr.copy( + left = inferEntity(expr.left), + right = inferEntity(expr.right) + ) + + override fun visit(expr: EtsRightShiftExpr) = expr.copy( + left = inferEntity(expr.left), + right = inferEntity(expr.right) + ) + + override fun visit(expr: EtsUnsignedRightShiftExpr) = expr.copy( + left = inferEntity(expr.left), + right = inferEntity(expr.right) + ) + + override fun visit(expr: EtsAndExpr) = expr.copy( + left = inferEntity(expr.left), + right = inferEntity(expr.right) + ) + + override fun visit(expr: EtsOrExpr) = expr.copy( + left = inferEntity(expr.left), + right = inferEntity(expr.right) + ) + + override fun visit(expr: EtsNullishCoalescingExpr) = expr.copy( + left = inferEntity(expr.left), + right = inferEntity(expr.right) + ) + + override fun visit(expr: EtsInstanceCallExpr): EtsExpr { + val baseInferred = inferValue(expr.instance) as EtsLocal + val argsInferred = expr.args.map(this::inferValue) + val methodInferred = when (val baseType = baseInferred.type) { + is EtsClassType -> { + val etsClass = scene.projectAndSdkClasses.find { it.signature == baseType.signature } + ?: return expr.copy(instance = baseInferred, args = argsInferred) + val callee = etsClass.methods.find { it.signature == expr.method } + ?: return expr.copy(instance = baseInferred, args = argsInferred) + callee.signature + } + + else -> expr.method + } + return EtsInstanceCallExpr(baseInferred, methodInferred, argsInferred) + } + + override fun visit(expr: EtsStaticCallExpr): EtsExpr { + val argsInferred = expr.args.map(this::inferValue) + return EtsStaticCallExpr(expr.method, argsInferred) + } + + override fun visit(expr: EtsPtrCallExpr): EtsExpr { + val ptrInferred = inferValue(expr.ptr) as EtsLocal + val argsInferred = expr.args.map(this::inferValue) + return EtsPtrCallExpr(ptrInferred, expr.method, argsInferred) + } + + override fun visit(expr: EtsCommaExpr) = expr.copy( + left = inferEntity(expr.left), + right = inferEntity(expr.right) + ) + + override fun visit(expr: EtsTernaryExpr) = expr.copy( + condition = inferEntity(expr.condition), + thenExpr = inferEntity(expr.thenExpr), + elseExpr = inferEntity(expr.elseExpr) + ) +} diff --git a/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/annotation/StmtTypeAnnotator.kt b/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/annotation/StmtTypeAnnotator.kt new file mode 100644 index 0000000000..110ca68abb --- /dev/null +++ b/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/annotation/StmtTypeAnnotator.kt @@ -0,0 +1,83 @@ +/* + * Copyright 2022 UnitTestBot contributors (utbot.org) + *

+ * 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. + */ + +package org.usvm.dataflow.ts.infer.annotation + +import org.jacodb.ets.base.EtsAssignStmt +import org.jacodb.ets.base.EtsCallExpr +import org.jacodb.ets.base.EtsCallStmt +import org.jacodb.ets.base.EtsEntity +import org.jacodb.ets.base.EtsExpr +import org.jacodb.ets.base.EtsGotoStmt +import org.jacodb.ets.base.EtsIfStmt +import org.jacodb.ets.base.EtsNopStmt +import org.jacodb.ets.base.EtsReturnStmt +import org.jacodb.ets.base.EtsStmt +import org.jacodb.ets.base.EtsSwitchStmt +import org.jacodb.ets.base.EtsThrowStmt +import org.jacodb.ets.base.EtsValue +import org.jacodb.ets.model.EtsScene +import org.usvm.dataflow.ts.infer.AccessPathBase +import org.usvm.dataflow.ts.infer.EtsTypeFact + +class StmtTypeAnnotator( + types: Map, + thisType: EtsTypeFact?, + scene: EtsScene, +) : EtsStmt.Visitor { + private val valueAnnotator = ValueTypeAnnotator(types, thisType, scene) + private val exprAnnotator = ExprTypeAnnotator(valueAnnotator, scene) + + private fun inferValue(value: EtsValue) = value.accept(valueAnnotator) + + private fun inferExpr(expr: EtsExpr) = expr.accept(exprAnnotator) + + private fun inferEntity(entity: EtsEntity) = when (entity) { + is EtsExpr -> entity.accept(exprAnnotator) + is EtsValue -> entity.accept(valueAnnotator) + else -> error("Unsupported entity") + } + + override fun visit(stmt: EtsNopStmt) = stmt + + override fun visit(stmt: EtsAssignStmt) = stmt.copy( + lhv = inferValue(stmt.lhv), + rhv = inferEntity(stmt.rhv) + ) + + override fun visit(stmt: EtsCallStmt) = stmt.copy( + expr = inferExpr(stmt.expr) as EtsCallExpr + ) + + override fun visit(stmt: EtsReturnStmt) = stmt.copy( + returnValue = stmt.returnValue?.let(this::inferValue) + ) + + override fun visit(stmt: EtsThrowStmt) = stmt.copy( + arg = inferEntity(stmt.arg) + ) + + override fun visit(stmt: EtsGotoStmt) = stmt + + override fun visit(stmt: EtsIfStmt) = stmt.copy( + condition = inferEntity(stmt.condition) + ) + + override fun visit(stmt: EtsSwitchStmt) = stmt.copy( + arg = inferEntity(stmt.arg), + cases = stmt.cases.map(this::inferEntity) + ) +} diff --git a/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/annotation/ValueTypeAnnotator.kt b/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/annotation/ValueTypeAnnotator.kt new file mode 100644 index 0000000000..01df46cae6 --- /dev/null +++ b/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/annotation/ValueTypeAnnotator.kt @@ -0,0 +1,113 @@ +/* + * Copyright 2022 UnitTestBot contributors (utbot.org) + *

+ * 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. + */ + +package org.usvm.dataflow.ts.infer.annotation + +import org.jacodb.ets.base.EtsArrayAccess +import org.jacodb.ets.base.EtsArrayType +import org.jacodb.ets.base.EtsClassType +import org.jacodb.ets.base.EtsInstanceFieldRef +import org.jacodb.ets.base.EtsLocal +import org.jacodb.ets.base.EtsParameterRef +import org.jacodb.ets.base.EtsStaticFieldRef +import org.jacodb.ets.base.EtsThis +import org.jacodb.ets.base.EtsType +import org.jacodb.ets.base.EtsValue +import org.jacodb.ets.model.EtsClassSignature +import org.jacodb.ets.model.EtsField +import org.jacodb.ets.model.EtsScene +import org.usvm.dataflow.ts.infer.AccessPathBase +import org.usvm.dataflow.ts.infer.EtsTypeFact +import org.usvm.dataflow.ts.infer.dto.toType + +class ValueTypeAnnotator( + private val types: Map, + private val thisType: EtsTypeFact?, + private val scene: EtsScene, +) : EtsValue.Visitor.Default { + private inline fun V.infer(base: AccessPathBase, apply: V.(T) -> V): V { + val type = types[base]?.toType() as? T ?: return this + return apply.invoke(this, type) + } + + override fun visit(value: EtsLocal): EtsLocal = + value.infer(AccessPathBase.Local(value.name)) { copy(type = it) } + + override fun visit(value: EtsThis): EtsValue = + (thisType?.toType() as? EtsClassType)?.let { value.copy(type = it) } + ?: value.infer(AccessPathBase.This) { copy(type = it) } + + override fun visit(value: EtsParameterRef) = + value.infer(AccessPathBase.Arg(value.index)) { copy(type = it) } + + override fun visit(value: EtsArrayAccess): EtsArrayAccess { + val arrayInferred = value.array.accept(this) + val arrayType = arrayInferred.type as? EtsArrayType ?: return value + val indexInferred = value.index.accept(this) + return EtsArrayAccess(arrayInferred, indexInferred, arrayType.elementType) + } + + // TODO: discuss (labeled with (Q)) + override fun visit(value: EtsInstanceFieldRef): EtsValue { + val instance = visit(value.instance) + val name = value.field.name + + fun findInClass(signature: EtsClassSignature): EtsField? = + scene.projectClasses + .singleOrNull { it.signature == signature } + ?.fields + ?.singleOrNull { it.name == name } + + // Try to determine field type using the scene + // (Q) Do we really need this step? + with(value.field) { + val etsField = findInClass(enclosingClass) ?: return@with + // Field was found in the scene + + // Check that inferred type is same with the declared one + // (Q) How should we (do we should?) handle the check violation here? + check(instance.type == EtsClassType(enclosingClass)) + + return EtsInstanceFieldRef(instance = instance, field = etsField.signature) + } + + // Field was not found by signature, then try infer instance type + val instanceTypeInfo = types[AccessPathBase.Local(instance.name)] as? EtsTypeFact.ObjectEtsTypeFact + // Instance type was neither specified in signature nor inferred, so no type info can be provided + // (Q) Should we check special properties of primitives (like `string.length`)? + ?: return value.copy(instance = instance) + + // Find field signature in inferred class fields + (instanceTypeInfo.cls as? EtsClassType)?.run { + val etsField = findInClass(signature) ?: return@run + // Field was found in the inferred class + return EtsInstanceFieldRef(instance = instance, field = etsField.signature) + } + + // Find field type in inferred fields - we don't know precisely the base class, but can infer the field type + instanceTypeInfo.properties[name]?.toType()?.let { fieldType -> + val fieldSubSignature = value.field.sub.copy(type = fieldType) + // (Q) General: Should we use our inferred type if there is a non-trivial type in the original scene? + return EtsInstanceFieldRef(instance = instance, field = value.field.copy(sub = fieldSubSignature)) + } + + return value.copy(instance = instance) + } + + override fun visit(value: EtsStaticFieldRef): EtsValue = value + + override fun defaultVisit(value: EtsValue): EtsValue = value +} diff --git a/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/annotation/withInferredTypes.kt b/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/annotation/withInferredTypes.kt new file mode 100644 index 0000000000..ee71d2cbe9 --- /dev/null +++ b/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/annotation/withInferredTypes.kt @@ -0,0 +1,91 @@ +/* + * Copyright 2022 UnitTestBot contributors (utbot.org) + *

+ * 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. + */ + +@file:Suppress("MemberVisibilityCanBePrivate") + +package org.usvm.dataflow.ts.infer.annotation + +import org.jacodb.ets.graph.EtsCfg +import org.jacodb.ets.model.EtsClass +import org.jacodb.ets.model.EtsClassImpl +import org.jacodb.ets.model.EtsFile +import org.jacodb.ets.model.EtsMethod +import org.jacodb.ets.model.EtsMethodImpl +import org.jacodb.ets.model.EtsScene +import org.usvm.dataflow.ts.infer.AccessPathBase +import org.usvm.dataflow.ts.infer.EtsTypeFact +import org.usvm.dataflow.ts.infer.TypeInferenceResult + +data class EtsTypeAnnotator( + val scene: EtsScene, + val typeInferenceResult: TypeInferenceResult, +) { + private fun selectTypesFor(method: EtsMethod) = typeInferenceResult.inferredTypes[method] ?: emptyMap() + + private fun combinedThisFor(method: EtsMethod) = typeInferenceResult.inferredCombinedThisType[method.enclosingClass] + + fun annotateWithTypes(scene: EtsScene) = with(scene) { + EtsScene( + projectFiles = projectFiles.map { annotateWithTypes(it) } + ) + } + + fun annotateWithTypes(file: EtsFile) = with(file) { + EtsFile( + signature = signature, + classes = classes.map { annotateWithTypes(it) }, + namespaces = namespaces, + ) + } + + fun annotateWithTypes(clazz: EtsClass) = with(clazz) { + EtsClassImpl( + signature = signature, + fields = fields, + methods = methods.map { annotateWithTypes(it) }, + ctor = annotateWithTypes(ctor), + superClass = superClass, // TODO: replace with inferred superclass + modifiers = modifiers, + decorators = decorators, + typeParameters = typeParameters, + ) + } + + fun annotateWithTypes(method: EtsMethod) = with(method) { + EtsMethodImpl( + signature = signature, + typeParameters = typeParameters, + locals = locals, + modifiers = modifiers, + decorators = decorators, + ).also { + it._cfg = annotateWithTypes(cfg, selectTypesFor(this), combinedThisFor(this)) + } + } + + fun annotateWithTypes( + cfg: EtsCfg, + types: Map, + thisType: EtsTypeFact?, + ) = with(cfg) { + with(StmtTypeAnnotator(types, thisType, scene)) { + EtsCfg( + stmts = stmts.map { it.accept(this) }, + successorMap = stmts.associateWith { successors(it).toList() }, + ) + } + } +} diff --git a/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/cli/InferTypes.kt b/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/cli/InferTypes.kt new file mode 100644 index 0000000000..42aaad2877 --- /dev/null +++ b/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/cli/InferTypes.kt @@ -0,0 +1,143 @@ +/* + * Copyright 2022 UnitTestBot contributors (utbot.org) + *

+ * 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. + */ + +@file:Suppress("MemberVisibilityCanBePrivate") + +package org.usvm.dataflow.ts.infer.cli + +import com.github.ajalt.clikt.core.CliktCommand +import com.github.ajalt.clikt.core.context +import com.github.ajalt.clikt.core.main +import com.github.ajalt.clikt.output.MordantHelpFormatter +import com.github.ajalt.clikt.parameters.options.convert +import com.github.ajalt.clikt.parameters.options.default +import com.github.ajalt.clikt.parameters.options.flag +import com.github.ajalt.clikt.parameters.options.multiple +import com.github.ajalt.clikt.parameters.options.option +import com.github.ajalt.clikt.parameters.options.required +import com.github.ajalt.clikt.parameters.types.path +import kotlinx.serialization.ExperimentalSerializationApi +import kotlinx.serialization.json.Json +import kotlinx.serialization.json.encodeToStream +import mu.KotlinLogging +import org.jacodb.ets.base.ANONYMOUS_CLASS_PREFIX +import org.jacodb.ets.base.ANONYMOUS_METHOD_PREFIX +import org.jacodb.ets.utils.loadEtsProjectFromMultipleIR +import org.usvm.dataflow.ts.infer.EntryPointsProcessor +import org.usvm.dataflow.ts.infer.TypeInferenceManager +import org.usvm.dataflow.ts.infer.TypeInferenceResult +import org.usvm.dataflow.ts.infer.createApplicationGraph +import org.usvm.dataflow.ts.infer.dto.toDto +import org.usvm.dataflow.ts.util.EtsTraits +import java.nio.file.Path +import kotlin.io.path.outputStream +import kotlin.time.measureTimedValue + +private val logger = KotlinLogging.logger {} + +class InferTypes : CliktCommand() { + init { + context { + helpFormatter = { + MordantHelpFormatter( + it, + requiredOptionMarker = "*", + showDefaultValues = true, + showRequiredTag = true + ) + } + } + } + + val input by option("-i", "--input", help = "Input file or directory with IR").path().multiple(required = true) + val output by option("-o", "--output", help = "Output file with inferred types in JSON format").path().required() + + val sdkPaths by option( + "--sdk", + help = "Path to SDK directory" + ).path().multiple() + + val sdkNames by option( + "--sdk-names", + help = "Comma-separated list of SDK names" + ).convert { it.split(",") }.default(emptyList(), defaultForHelp = "empty") + + val skipAnonymous by option( + "--skip-anonymous", + help = "Skip anonymous classes and method" + ).flag("--no-skip-anonymous", default = false) + + override fun run() { + logger.info { "Running InferTypes" } + val startTime = System.currentTimeMillis() + + logger.info { "Input: $input" } + logger.info { "Output: $output" } + + val project = loadEtsProjectFromMultipleIR(input, sdkPaths) + val graph = createApplicationGraph(project) + + val (dummyMains, allMethods) = EntryPointsProcessor.extractEntryPoints(project) + val publicMethods = allMethods.filter { m -> m.isPublic } + + val manager = TypeInferenceManager(EtsTraits(), graph) + + val (result, timeAnalyze) = measureTimedValue { + manager.analyze(dummyMains, publicMethods).withGuessedTypes(project) + } + logger.info { "Inferred types for ${result.inferredTypes.size} methods in $timeAnalyze" } + + dumpTypeInferenceResult(result, output, skipAnonymous) + + logger.info { "All done in %.1f s".format((System.currentTimeMillis() - startTime) / 1000.0) } + } +} + +fun main(args: Array) { + InferTypes().main(args) +} + +@OptIn(ExperimentalSerializationApi::class) +fun dumpTypeInferenceResult( + result: TypeInferenceResult, + path: Path, + skipAnonymous: Boolean = true, +) { + logger.info { "Dumping inferred types to '$path'" } + val dto = result.toDto() + // Filter out anonymous classes and methods + .let { dto -> + if (skipAnonymous) { + dto.copy( + classes = dto.classes.filterNot { cls -> + cls.signature.name.startsWith(ANONYMOUS_CLASS_PREFIX) + }, + methods = dto.methods.filterNot { method -> + method.signature.declaringClass.name.startsWith(ANONYMOUS_CLASS_PREFIX) + || method.signature.name.startsWith(ANONYMOUS_METHOD_PREFIX) + } + ) + } else { + dto + } + } + path.outputStream().use { stream -> + val json = Json { + prettyPrint = true + } + json.encodeToStream(dto, stream) + } +} diff --git a/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/dto/Convert.kt b/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/dto/Convert.kt new file mode 100644 index 0000000000..eda5340277 --- /dev/null +++ b/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/dto/Convert.kt @@ -0,0 +1,195 @@ +/* + * Copyright 2022 UnitTestBot contributors (utbot.org) + *

+ * 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. + */ + +package org.usvm.dataflow.ts.infer.dto + +import mu.KotlinLogging +import org.jacodb.ets.base.EtsAnyType +import org.jacodb.ets.base.EtsArrayObjectType +import org.jacodb.ets.base.EtsArrayType +import org.jacodb.ets.base.EtsBooleanType +import org.jacodb.ets.base.EtsClassType +import org.jacodb.ets.base.EtsFunctionType +import org.jacodb.ets.base.EtsGenericType +import org.jacodb.ets.base.EtsLiteralType +import org.jacodb.ets.base.EtsNeverType +import org.jacodb.ets.base.EtsNullType +import org.jacodb.ets.base.EtsNumberType +import org.jacodb.ets.base.EtsStringType +import org.jacodb.ets.base.EtsTupleType +import org.jacodb.ets.base.EtsType +import org.jacodb.ets.base.EtsUnclearRefType +import org.jacodb.ets.base.EtsUndefinedType +import org.jacodb.ets.base.EtsUnionType +import org.jacodb.ets.base.EtsUnknownType +import org.jacodb.ets.base.EtsVoidType +import org.jacodb.ets.dto.AnyTypeDto +import org.jacodb.ets.dto.ArrayTypeDto +import org.jacodb.ets.dto.BooleanTypeDto +import org.jacodb.ets.dto.ClassSignatureDto +import org.jacodb.ets.dto.ClassTypeDto +import org.jacodb.ets.dto.FileSignatureDto +import org.jacodb.ets.dto.FunctionTypeDto +import org.jacodb.ets.dto.GenericTypeDto +import org.jacodb.ets.dto.LiteralTypeDto +import org.jacodb.ets.dto.MethodParameterDto +import org.jacodb.ets.dto.MethodSignatureDto +import org.jacodb.ets.dto.NamespaceSignatureDto +import org.jacodb.ets.dto.NeverTypeDto +import org.jacodb.ets.dto.NullTypeDto +import org.jacodb.ets.dto.NumberTypeDto +import org.jacodb.ets.dto.PrimitiveLiteralDto +import org.jacodb.ets.dto.StringTypeDto +import org.jacodb.ets.dto.TupleTypeDto +import org.jacodb.ets.dto.TypeDto +import org.jacodb.ets.dto.UnclearReferenceTypeDto +import org.jacodb.ets.dto.UndefinedTypeDto +import org.jacodb.ets.dto.UnionTypeDto +import org.jacodb.ets.dto.UnknownTypeDto +import org.jacodb.ets.dto.VoidTypeDto +import org.jacodb.ets.model.EtsClassSignature +import org.jacodb.ets.model.EtsFileSignature +import org.jacodb.ets.model.EtsMethodParameter +import org.jacodb.ets.model.EtsMethodSignature +import org.jacodb.ets.model.EtsNamespaceSignature +import org.usvm.dataflow.ts.infer.EtsTypeFact + +private val logger = KotlinLogging.logger {} + +fun EtsTypeFact.toType(): EtsType? = when (this) { + is EtsTypeFact.ObjectEtsTypeFact -> if (cls is EtsClassType) cls else null + is EtsTypeFact.ArrayEtsTypeFact -> EtsArrayType( + elementType = elementType.toType() ?: EtsUnknownType, + dimensions = 1, + ) + + EtsTypeFact.AnyEtsTypeFact -> EtsAnyType + EtsTypeFact.BooleanEtsTypeFact -> EtsBooleanType + EtsTypeFact.FunctionEtsTypeFact -> null // TODO: function type + EtsTypeFact.NullEtsTypeFact -> EtsNullType + EtsTypeFact.NumberEtsTypeFact -> EtsNumberType + EtsTypeFact.StringEtsTypeFact -> EtsStringType + EtsTypeFact.UndefinedEtsTypeFact -> EtsUndefinedType + EtsTypeFact.UnknownEtsTypeFact -> EtsUnknownType + + is EtsTypeFact.GuardedTypeFact -> null + is EtsTypeFact.IntersectionEtsTypeFact -> null + is EtsTypeFact.UnionEtsTypeFact -> { + val types = this.types.map { it.toType() } + + if (types.any { it == null }) { + logger.warn { "Cannot convert union type fact to type: $this" } + null + } else { + EtsUnionType(types.map { it!! }) + } + } +} + +fun EtsType.toDto(): TypeDto = when (this) { + is EtsAnyType -> AnyTypeDto + is EtsUnknownType -> UnknownTypeDto + is EtsUnionType -> UnionTypeDto(types = this.types.map { it.toDto() }) + is EtsTupleType -> TupleTypeDto(types = this.types.map { it.toDto() }) + is EtsBooleanType -> BooleanTypeDto + is EtsNumberType -> NumberTypeDto + is EtsStringType -> StringTypeDto + is EtsNullType -> NullTypeDto + is EtsUndefinedType -> UndefinedTypeDto + is EtsVoidType -> VoidTypeDto + is EtsNeverType -> NeverTypeDto + + is EtsLiteralType -> { + val literal = when { + this.literalTypeName.equals("true", ignoreCase = true) -> PrimitiveLiteralDto.BooleanLiteral(true) + this.literalTypeName.equals("false", ignoreCase = true) -> PrimitiveLiteralDto.BooleanLiteral(false) + else -> { + val x = this.literalTypeName.toDoubleOrNull() + if (x != null) { + PrimitiveLiteralDto.NumberLiteral(x) + } else { + PrimitiveLiteralDto.StringLiteral(this.literalTypeName) + } + } + } + LiteralTypeDto(literal = literal) + } + + is EtsClassType -> ClassTypeDto( + signature = this.signature.toDto(), + typeParameters = this.typeParameters.map { it.toDto() } + ) + + is EtsFunctionType -> FunctionTypeDto( + signature = this.method.toDto(), + typeParameters = this.typeParameters.map { it.toDto() } + ) + + is EtsArrayType -> ArrayTypeDto( + elementType = this.elementType.toDto(), + dimensions = this.dimensions + ) + + is EtsArrayObjectType -> TODO("EtsArrayObjectType was removed") + + is EtsUnclearRefType -> UnclearReferenceTypeDto( + name = this.typeName, + typeParameters = this.typeParameters.map { it.toDto() } + ) + + is EtsGenericType -> GenericTypeDto( + name = this.typeName, + defaultType = this.defaultType?.toDto(), + constraint = this.constraint?.toDto(), + ) + + else -> error("Cannot convert ${this::class.java} to DTO: $this") +} + +fun EtsClassSignature.toDto(): ClassSignatureDto = + ClassSignatureDto( + name = this.name, + declaringFile = this.file.toDto(), + declaringNamespace = this.namespace?.toDto(), + ) + +fun EtsFileSignature.toDto(): FileSignatureDto = + FileSignatureDto( + projectName = this.projectName, + fileName = this.fileName, + ) + +fun EtsNamespaceSignature.toDto(): NamespaceSignatureDto = + NamespaceSignatureDto( + name = this.name, + declaringFile = this.file.toDto(), + declaringNamespace = this.namespace?.toDto(), + ) + +fun EtsMethodSignature.toDto(): MethodSignatureDto = + MethodSignatureDto( + declaringClass = this.enclosingClass.toDto(), + name = this.name, + parameters = this.parameters.map { it.toDto() }, + returnType = this.returnType.toDto(), + ) + +fun EtsMethodParameter.toDto(): MethodParameterDto = + MethodParameterDto( + name = this.name, + type = this.type.toDto(), + isOptional = this.isOptional, + ) diff --git a/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/dto/DTO.kt b/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/dto/DTO.kt new file mode 100644 index 0000000000..467443d0ec --- /dev/null +++ b/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/dto/DTO.kt @@ -0,0 +1,96 @@ +package org.usvm.dataflow.ts.infer.dto + +import kotlinx.serialization.Serializable +import org.jacodb.ets.dto.ClassSignatureDto +import org.jacodb.ets.dto.MethodSignatureDto +import org.jacodb.ets.dto.TypeDto +import org.usvm.dataflow.ts.infer.AccessPathBase +import org.usvm.dataflow.ts.infer.EtsTypeFact +import org.usvm.dataflow.ts.infer.TypeInferenceResult + +@Serializable +data class InferredTypesDto( + val classes: List, + val methods: List, +) + +@Serializable +data class ClassTypeResultDto( + val signature: ClassSignatureDto, + val fields: List, + val methods: List, +) + +@Serializable +data class FieldTypeResultDto( + val name: String, + val type: TypeDto, +) + +@Serializable +data class MethodTypeResultDto( + val signature: MethodSignatureDto, + val args: List, + val returnType: TypeDto? = null, + val locals: List, +) + +@Serializable +data class ArgumentTypeResultDto( + val index: Int, + val type: TypeDto, +) + +@Serializable +data class LocalTypeResultDto( + val name: String, + val type: TypeDto, +) + +fun TypeInferenceResult.toDto(): InferredTypesDto { + val classTypeInferenceResult = inferredCombinedThisType.map { (clazz, fact) -> + val properties = (fact as? EtsTypeFact.ObjectEtsTypeFact)?.properties ?: emptyMap() + val methods = properties + .filter { it.value is EtsTypeFact.FunctionEtsTypeFact } + .keys + .sortedBy { it } + val fields = properties + .filterNot { it.value is EtsTypeFact.FunctionEtsTypeFact } + .mapNotNull { (name, fact) -> + fact.toType()?.let { + FieldTypeResultDto(name, it.toDto()) + } + } + .sortedBy { it.name } + ClassTypeResultDto(clazz.toDto(), fields, methods) + }.sortedBy { + it.signature.toString() + } + + val methodTypeInferenceResult = inferredTypes.map { (method, facts) -> + val args = facts.mapNotNull { (base, fact) -> + if (base is AccessPathBase.Arg) { + val type = fact.toType() + if (type != null) { + return@mapNotNull ArgumentTypeResultDto(base.index, type.toDto()) + } + } + null + }.sortedBy { it.index } + val returnType = inferredReturnType[method]?.toType()?.toDto() + val locals = facts.mapNotNull { (base, fact) -> + if (base is AccessPathBase.Local) { + val type = fact.toType() + if (type != null) { + return@mapNotNull LocalTypeResultDto(base.name, type.toDto()) + } + } + null + }.sortedBy { it.name } + MethodTypeResultDto(method.signature.toDto(), args, returnType, locals) + }.sortedBy { + it.signature.toString() + } + + return InferredTypesDto(classTypeInferenceResult, methodTypeInferenceResult) +} diff --git a/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/verify/EntityId.kt b/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/verify/EntityId.kt new file mode 100644 index 0000000000..73c0d88270 --- /dev/null +++ b/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/verify/EntityId.kt @@ -0,0 +1,79 @@ +/* + * Copyright 2022 UnitTestBot contributors (utbot.org) + *

+ * 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. + */ + +package org.usvm.dataflow.ts.infer.verify + +import org.jacodb.ets.base.EtsLocal +import org.jacodb.ets.base.EtsParameterRef +import org.jacodb.ets.model.EtsClassSignature +import org.jacodb.ets.model.EtsFieldSignature +import org.jacodb.ets.model.EtsMethodParameter +import org.jacodb.ets.model.EtsMethodSignature + +sealed interface EntityId + +data class ClassId( + val signature: EtsClassSignature, +) : EntityId + +data class MethodId( + val name: String, + val enclosingClass: ClassId, +) { + constructor(signature: EtsMethodSignature) + : this(signature.name, ClassId(signature.enclosingClass)) +} + +data class FieldId( + val name: String, + val enclosingClass: ClassId, +) : EntityId { + constructor(signature: EtsFieldSignature) + : this(signature.name, ClassId(signature.enclosingClass)) +} + +data class ParameterId( + val index: Int, + val method: MethodId, +) : EntityId { + constructor(parameter: EtsMethodParameter, methodSignature: EtsMethodSignature) + : this(parameter.index, MethodId(methodSignature)) + + constructor(etsParameterRef: EtsParameterRef, methodSignature: EtsMethodSignature) + : this(etsParameterRef.index, MethodId(methodSignature)) +} + +data class ReturnId( + val method: MethodId, +) : EntityId { + constructor(methodSignature: EtsMethodSignature) + : this(MethodId(methodSignature)) +} + +data class LocalId( + val name: String, + val method: MethodId, +) : EntityId { + constructor(local: EtsLocal, methodSignature: EtsMethodSignature) + : this(local.name, MethodId(methodSignature)) +} + +data class ThisId( + val method: MethodId, +) : EntityId { + constructor(methodSignature: EtsMethodSignature) + : this(MethodId(methodSignature)) +} diff --git a/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/verify/VerificationResult.kt b/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/verify/VerificationResult.kt new file mode 100644 index 0000000000..1859346d90 --- /dev/null +++ b/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/verify/VerificationResult.kt @@ -0,0 +1,34 @@ +/* + * Copyright 2022 UnitTestBot contributors (utbot.org) + *

+ * 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. + */ + +package org.usvm.dataflow.ts.infer.verify + +import org.jacodb.ets.base.EtsType + +sealed interface VerificationResult { + data class Success(val mapping: Map) : VerificationResult + + data class Fail(val mapping: Map>) : VerificationResult + + companion object { + fun from(mapping: Map>): VerificationResult = + if (mapping.values.all { it.size == 1 }) { + Success(mapping.mapValues { (_, types) -> types.single() }) + } else { + Fail(mapping) + } + } +} diff --git a/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/verify/Verify.kt b/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/verify/Verify.kt new file mode 100644 index 0000000000..ede486afe6 --- /dev/null +++ b/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/verify/Verify.kt @@ -0,0 +1,28 @@ +/* + * Copyright 2022 UnitTestBot contributors (utbot.org) + *

+ * 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. + */ + +package org.usvm.dataflow.ts.infer.verify + +import org.jacodb.ets.base.EtsType +import org.jacodb.ets.model.EtsScene +import org.usvm.dataflow.ts.infer.verify.collectors.ClassSummaryCollector + +fun collectSummary(scene: EtsScene): Map> = + ClassSummaryCollector(mutableMapOf()).apply { + scene.projectAndSdkClasses.forEach { collect(it) } + }.typeSummary + +fun verify(scene: EtsScene) = VerificationResult.from(collectSummary(scene)) diff --git a/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/verify/collectors/ClassSummaryCollector.kt b/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/verify/collectors/ClassSummaryCollector.kt new file mode 100644 index 0000000000..03bcef0d02 --- /dev/null +++ b/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/verify/collectors/ClassSummaryCollector.kt @@ -0,0 +1,39 @@ +/* + * Copyright 2022 UnitTestBot contributors (utbot.org) + *

+ * 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. + */ + +package org.usvm.dataflow.ts.infer.verify.collectors + +import org.jacodb.ets.base.EtsType +import org.jacodb.ets.model.EtsClass +import org.usvm.dataflow.ts.infer.verify.EntityId + +class ClassSummaryCollector( + override val typeSummary: MutableMap>, +) : SummaryCollector { + fun collect(etsClass: EtsClass) { + etsClass.fields.forEach { field -> + yield(field.signature) + } + + etsClass.methods.forEach { method -> + yield(method.signature) + val stmtCollector = StmtSummaryCollector(method.signature, typeSummary) + method.cfg.stmts.forEach { + it.accept(stmtCollector) + } + } + } +} diff --git a/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/verify/collectors/ExprSummaryCollector.kt b/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/verify/collectors/ExprSummaryCollector.kt new file mode 100644 index 0000000000..960f770074 --- /dev/null +++ b/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/verify/collectors/ExprSummaryCollector.kt @@ -0,0 +1,93 @@ +/* + * Copyright 2022 UnitTestBot contributors (utbot.org) + *

+ * 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. + */ + +package org.usvm.dataflow.ts.infer.verify.collectors + +import org.jacodb.ets.base.EtsBinaryExpr +import org.jacodb.ets.base.EtsEntity +import org.jacodb.ets.base.EtsExpr +import org.jacodb.ets.base.EtsInstanceCallExpr +import org.jacodb.ets.base.EtsInstanceOfExpr +import org.jacodb.ets.base.EtsLengthExpr +import org.jacodb.ets.base.EtsPtrCallExpr +import org.jacodb.ets.base.EtsStaticCallExpr +import org.jacodb.ets.base.EtsTernaryExpr +import org.jacodb.ets.base.EtsType +import org.jacodb.ets.base.EtsUnaryExpr +import org.jacodb.ets.base.EtsValue +import org.jacodb.ets.model.EtsMethodSignature +import org.usvm.dataflow.ts.infer.verify.EntityId + +class ExprSummaryCollector( + override val enclosingMethod: EtsMethodSignature, + override val typeSummary: MutableMap>, +) : MethodSummaryCollector, EtsExpr.Visitor.Default { + private val valueSummaryCollector by lazy { + ValueSummaryCollector(enclosingMethod, typeSummary) + } + + private fun collect(entity: EtsEntity) { + when (entity) { + is EtsValue -> entity.accept(valueSummaryCollector) + is EtsExpr -> entity.accept(this) + else -> error("Unsupported entity kind") + } + } + + override fun defaultVisit(expr: EtsExpr) = when (expr) { + is EtsUnaryExpr -> { + collect(expr.arg) + } + + is EtsBinaryExpr -> { + collect(expr.left) + collect(expr.right) + } + + is EtsTernaryExpr -> { + collect(expr.condition) + collect(expr.thenExpr) + collect(expr.elseExpr) + } + + is EtsInstanceCallExpr -> { + yield(expr.method) + collect(expr.instance) + expr.args.forEach { collect(it) } + } + + is EtsStaticCallExpr -> { + yield(expr.method) + expr.args.forEach { collect(it) } + } + + is EtsPtrCallExpr -> { + yield(expr.method) + collect(expr.ptr) + expr.args.forEach { collect(it) } + } + + is EtsLengthExpr -> { + collect(expr.arg) + } + + is EtsInstanceOfExpr -> { + collect(expr.arg) + } + + else -> {} + } +} diff --git a/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/verify/collectors/StmtSummaryCollector.kt b/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/verify/collectors/StmtSummaryCollector.kt new file mode 100644 index 0000000000..20d0ce61fd --- /dev/null +++ b/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/verify/collectors/StmtSummaryCollector.kt @@ -0,0 +1,80 @@ +/* + * Copyright 2022 UnitTestBot contributors (utbot.org) + *

+ * 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. + */ + +package org.usvm.dataflow.ts.infer.verify.collectors + +import org.jacodb.ets.base.EtsAssignStmt +import org.jacodb.ets.base.EtsCallStmt +import org.jacodb.ets.base.EtsEntity +import org.jacodb.ets.base.EtsExpr +import org.jacodb.ets.base.EtsGotoStmt +import org.jacodb.ets.base.EtsIfStmt +import org.jacodb.ets.base.EtsNopStmt +import org.jacodb.ets.base.EtsReturnStmt +import org.jacodb.ets.base.EtsStmt +import org.jacodb.ets.base.EtsSwitchStmt +import org.jacodb.ets.base.EtsThrowStmt +import org.jacodb.ets.base.EtsType +import org.jacodb.ets.base.EtsValue +import org.jacodb.ets.model.EtsMethodSignature +import org.usvm.dataflow.ts.infer.verify.EntityId + +class StmtSummaryCollector( + override val enclosingMethod: EtsMethodSignature, + override val typeSummary: MutableMap>, +) : EtsStmt.Visitor, MethodSummaryCollector { + private val exprCollector = ExprSummaryCollector(enclosingMethod, typeSummary) + private val valueCollector = ValueSummaryCollector(enclosingMethod, typeSummary) + + private fun collect(entity: EtsEntity) { + when (entity) { + is EtsValue -> entity.accept(valueCollector) + is EtsExpr -> entity.accept(exprCollector) + else -> error("Unsupported entity kind") + } + } + + override fun visit(stmt: EtsNopStmt) {} + + override fun visit(stmt: EtsAssignStmt) { + collect(stmt.lhv) + collect(stmt.rhv) + } + + override fun visit(stmt: EtsCallStmt) { + collect(stmt.expr) + } + + override fun visit(stmt: EtsReturnStmt) { + stmt.returnValue?.let { collect(it) } + } + + override fun visit(stmt: EtsThrowStmt) { + collect(stmt.arg) + } + + override fun visit(stmt: EtsGotoStmt) {} + + override fun visit(stmt: EtsIfStmt) { + collect(stmt.condition) + } + + override fun visit(stmt: EtsSwitchStmt) { + collect(stmt.arg) + stmt.cases.forEach { collect(it) } + } + +} diff --git a/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/verify/collectors/SummaryCollector.kt b/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/verify/collectors/SummaryCollector.kt new file mode 100644 index 0000000000..618f5e3104 --- /dev/null +++ b/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/verify/collectors/SummaryCollector.kt @@ -0,0 +1,76 @@ +/* + * Copyright 2022 UnitTestBot contributors (utbot.org) + *

+ * 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. + */ + +package org.usvm.dataflow.ts.infer.verify.collectors + +import org.jacodb.ets.base.EtsLocal +import org.jacodb.ets.base.EtsParameterRef +import org.jacodb.ets.base.EtsThis +import org.jacodb.ets.base.EtsType +import org.jacodb.ets.model.EtsFieldSignature +import org.jacodb.ets.model.EtsMethodSignature +import org.usvm.dataflow.ts.infer.verify.EntityId +import org.usvm.dataflow.ts.infer.verify.FieldId +import org.usvm.dataflow.ts.infer.verify.LocalId +import org.usvm.dataflow.ts.infer.verify.ParameterId +import org.usvm.dataflow.ts.infer.verify.ReturnId +import org.usvm.dataflow.ts.infer.verify.ThisId + +interface SummaryCollector { + val typeSummary: MutableMap> + + fun yield(field: EtsFieldSignature) { + if (!field.type.isUnresolved) { + typeSummary.getOrPut(FieldId(field), ::mutableSetOf) + .add(field.type) + } + } + + fun yield(method: EtsMethodSignature) { + if (!method.returnType.isUnresolved) { + typeSummary.getOrPut(ReturnId(method), ::mutableSetOf) + .add(method.returnType) + } + method.parameters.forEach { + if (!it.type.isUnresolved) { + typeSummary.getOrPut(ParameterId(it, method), ::mutableSetOf) + .add(it.type) + } + } + } +} + +interface MethodSummaryCollector : SummaryCollector { + val enclosingMethod: EtsMethodSignature + fun yield(parameter: EtsParameterRef) { + if (!parameter.type.isUnresolved) { + typeSummary.getOrPut(ParameterId(parameter, enclosingMethod), ::mutableSetOf) + .add(parameter.type) + } + } + + fun yield(local: EtsLocal) { + if (!local.type.isUnresolved) { + typeSummary.getOrPut(LocalId(local, enclosingMethod), ::mutableSetOf) + .add(local.type) + } + } + + fun yield(etsThis: EtsThis) { + typeSummary.getOrPut(ThisId(enclosingMethod), ::mutableSetOf) + .add(etsThis.type) + } +} diff --git a/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/verify/collectors/Utils.kt b/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/verify/collectors/Utils.kt new file mode 100644 index 0000000000..b7a77c8e9d --- /dev/null +++ b/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/verify/collectors/Utils.kt @@ -0,0 +1,34 @@ +/* + * Copyright 2022 UnitTestBot contributors (utbot.org) + *

+ * 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. + */ + +package org.usvm.dataflow.ts.infer.verify.collectors + +import org.jacodb.ets.base.EtsAnyType +import org.jacodb.ets.base.EtsArrayType +import org.jacodb.ets.base.EtsTupleType +import org.jacodb.ets.base.EtsType +import org.jacodb.ets.base.EtsUnionType +import org.jacodb.ets.base.EtsUnknownType + +val EtsType.isUnresolved: Boolean + get() = when (this) { + is EtsAnyType -> true + is EtsUnknownType -> true + is EtsUnionType -> types.any { it.isUnresolved } + is EtsTupleType -> types.any { it.isUnresolved } + is EtsArrayType -> elementType.isUnresolved + else -> false + } \ No newline at end of file diff --git a/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/verify/collectors/ValueSummaryCollector.kt b/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/verify/collectors/ValueSummaryCollector.kt new file mode 100644 index 0000000000..707e76dad0 --- /dev/null +++ b/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/infer/verify/collectors/ValueSummaryCollector.kt @@ -0,0 +1,85 @@ +/* + * Copyright 2022 UnitTestBot contributors (utbot.org) + *

+ * 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. + */ + +package org.usvm.dataflow.ts.infer.verify.collectors + +import org.jacodb.ets.base.EtsArrayAccess +import org.jacodb.ets.base.EtsArrayLiteral +import org.jacodb.ets.base.EtsEntity +import org.jacodb.ets.base.EtsExpr +import org.jacodb.ets.base.EtsInstanceFieldRef +import org.jacodb.ets.base.EtsLocal +import org.jacodb.ets.base.EtsObjectLiteral +import org.jacodb.ets.base.EtsParameterRef +import org.jacodb.ets.base.EtsStaticFieldRef +import org.jacodb.ets.base.EtsThis +import org.jacodb.ets.base.EtsType +import org.jacodb.ets.base.EtsValue +import org.jacodb.ets.model.EtsMethodSignature +import org.usvm.dataflow.ts.infer.verify.EntityId + +class ValueSummaryCollector( + override val enclosingMethod: EtsMethodSignature, + override val typeSummary: MutableMap>, +) : MethodSummaryCollector, EtsValue.Visitor.Default { + private val exprSummaryCollector by lazy { + ExprSummaryCollector(enclosingMethod, typeSummary) + } + + private fun collect(entity: EtsEntity) { + when (entity) { + is EtsValue -> entity.accept(this) + is EtsExpr -> entity.accept(exprSummaryCollector) + else -> error("Unsupported entity kind") + } + } + + override fun defaultVisit(value: EtsValue) {} + + override fun visit(value: EtsLocal) { + yield(value) + } + + override fun visit(value: EtsArrayLiteral) { + value.elements.forEach { collect(it) } + } + + override fun visit(value: EtsObjectLiteral) { + value.properties.forEach { (_, it) -> collect(it) } + } + + override fun visit(value: EtsThis) { + yield(value) + } + + override fun visit(value: EtsParameterRef) { + yield(value) + } + + override fun visit(value: EtsArrayAccess) { + value.array.accept(this) + value.index.accept(this) + } + + override fun visit(value: EtsInstanceFieldRef) { + yield(value.field) + value.instance.accept(this) + } + + override fun visit(value: EtsStaticFieldRef) { + yield(value.field) + } +} diff --git a/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/util/EtsTraits.kt b/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/util/EtsTraits.kt new file mode 100644 index 0000000000..81820a9f0a --- /dev/null +++ b/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/util/EtsTraits.kt @@ -0,0 +1,252 @@ +/* + * Copyright 2022 UnitTestBot contributors (utbot.org) + *

+ * 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. + */ + +package org.usvm.dataflow.ts.util + +import org.jacodb.api.common.CommonMethodParameter +import org.jacodb.api.common.cfg.CommonAssignInst +import org.jacodb.api.common.cfg.CommonCallExpr +import org.jacodb.api.common.cfg.CommonExpr +import org.jacodb.api.common.cfg.CommonValue +import org.jacodb.ets.base.CONSTRUCTOR_NAME +import org.jacodb.ets.base.EtsArrayAccess +import org.jacodb.ets.base.EtsAssignStmt +import org.jacodb.ets.base.EtsBinaryExpr +import org.jacodb.ets.base.EtsBooleanConstant +import org.jacodb.ets.base.EtsCallExpr +import org.jacodb.ets.base.EtsCastExpr +import org.jacodb.ets.base.EtsClassType +import org.jacodb.ets.base.EtsConstant +import org.jacodb.ets.base.EtsEntity +import org.jacodb.ets.base.EtsIfStmt +import org.jacodb.ets.base.EtsImmediate +import org.jacodb.ets.base.EtsInstanceFieldRef +import org.jacodb.ets.base.EtsNewArrayExpr +import org.jacodb.ets.base.EtsNumberConstant +import org.jacodb.ets.base.EtsParameterRef +import org.jacodb.ets.base.EtsStaticFieldRef +import org.jacodb.ets.base.EtsStmt +import org.jacodb.ets.base.EtsStringConstant +import org.jacodb.ets.base.EtsThis +import org.jacodb.ets.base.EtsUnaryExpr +import org.jacodb.ets.base.EtsValue +import org.jacodb.ets.model.EtsMethod +import org.jacodb.ets.model.EtsMethodImpl +import org.jacodb.ets.model.EtsMethodParameter +import org.jacodb.ets.utils.callExpr +import org.jacodb.ets.utils.getOperands +import org.jacodb.ets.utils.getValues +import org.jacodb.taint.configuration.ConstantBooleanValue +import org.jacodb.taint.configuration.ConstantIntValue +import org.jacodb.taint.configuration.ConstantStringValue +import org.jacodb.taint.configuration.ConstantValue +import org.jacodb.taint.configuration.TypeMatches +import org.usvm.dataflow.ifds.AccessPath +import org.usvm.dataflow.ifds.ElementAccessor +import org.usvm.dataflow.ifds.FieldAccessor +import org.usvm.dataflow.util.Traits + +/** + * ETS-specific extensions for analysis. + */ +class EtsTraits : Traits { + + override fun convertToPathOrNull(expr: CommonExpr): AccessPath? { + check(expr is EtsEntity) + return expr.toPathOrNull() + } + + override fun convertToPathOrNull(value: CommonValue): AccessPath? { + check(value is EtsEntity) + return value.toPathOrNull() + } + + override fun convertToPath(value: CommonValue): AccessPath { + check(value is EtsEntity) + return value.toPath() + } + + override fun getThisInstance(method: EtsMethod): EtsThis { + return EtsThis(EtsClassType(method.signature.enclosingClass)) + } + + override fun getArgument(param: CommonMethodParameter): EtsParameterRef { + check(param is EtsMethodParameter) + return EtsParameterRef(index = param.index, type = param.type) + } + + override fun getArgumentsOf(method: EtsMethod): List { + return method.parameters.map { getArgument(it) } + } + + override fun getCallee(callExpr: CommonCallExpr): EtsMethod { + check(callExpr is EtsCallExpr) + return EtsMethodImpl(callExpr.method) + } + + override fun getCallExpr(statement: EtsStmt): EtsCallExpr? { + return statement.callExpr + } + + override fun getValues(expr: CommonExpr): Set { + check(expr is EtsEntity) + return expr.getValues().toSet() + } + + override fun getOperands(statement: EtsStmt): List { + return statement.getOperands().toList() + } + + override fun getArrayAllocation(statement: EtsStmt): EtsEntity? { + if (statement !is EtsAssignStmt) return null + return statement.rhv as? EtsNewArrayExpr + } + + override fun getArrayAccessIndex(statement: EtsStmt): EtsValue? { + if (statement !is EtsAssignStmt) return null + + val lhv = statement.lhv + if (lhv is EtsArrayAccess) return lhv.index + + val rhv = statement.rhv + if (rhv is EtsArrayAccess) return rhv.index + + return null + } + + override fun getBranchExprCondition(statement: EtsStmt): EtsEntity? { + if (statement !is EtsIfStmt) return null + return statement.condition + } + + override fun isConstant(value: CommonValue): Boolean { + check(value is EtsValue) + return value is EtsConstant + } + + override fun eqConstant(value: CommonValue, constant: ConstantValue): Boolean { + check(value is EtsValue) + return when (constant) { + is ConstantBooleanValue -> { + value is EtsBooleanConstant && value.value == constant.value + } + + is ConstantIntValue -> { + value is EtsNumberConstant && value.value == constant.value.toDouble() + } + + is ConstantStringValue -> { + value is EtsStringConstant && value.value == constant.value + } + } + } + + override fun ltConstant(value: CommonValue, constant: ConstantValue): Boolean { + check(value is EtsValue) + return when (constant) { + is ConstantIntValue -> { + value is EtsNumberConstant && value.value < constant.value.toDouble() + } + + else -> error("Unexpected constant: $constant") + } + } + + override fun gtConstant(value: CommonValue, constant: ConstantValue): Boolean { + check(value is EtsValue) + return when (constant) { + is ConstantIntValue -> { + value is EtsNumberConstant && value.value > constant.value.toDouble() + } + + else -> error("Unexpected constant: $constant") + } + } + + override fun matches(value: CommonValue, pattern: String): Boolean { + check(value is EtsValue) + val s = value.toString() + val re = pattern.toRegex() + return re.matches(s) + } + + override fun typeMatches(value: CommonValue, condition: TypeMatches): Boolean { + check(value is EtsValue) + TODO("Not yet implemented") + } + + override fun isConstructor(method: EtsMethod): Boolean { + return method.name == CONSTRUCTOR_NAME + } + + override fun isLoopHead(statement: EtsStmt): Boolean { + TODO("Not yet implemented") + } + + override fun lineNumber(statement: EtsStmt): Int { + TODO("Not yet implemented") + } + + override fun locationFQN(statement: EtsStmt): String { + TODO("Not yet implemented") + } + + override fun taintFlowRhsValues(statement: CommonAssignInst): List { + check(statement is EtsAssignStmt) + return when (val rhv = statement.rhv) { + is EtsBinaryExpr -> listOf(rhv.left, rhv.right) + is EtsUnaryExpr -> listOf(rhv.arg) + is EtsCastExpr -> listOf(rhv.arg) + else -> listOf(rhv) + } + } + + override fun taintPassThrough(statement: EtsStmt): List>? { + return null + } +} + +fun EtsEntity.toPathOrNull(): AccessPath? = when (this) { + is EtsImmediate -> AccessPath(this, emptyList()) + + is EtsThis -> AccessPath(this, emptyList()) + + is EtsParameterRef -> AccessPath(this, emptyList()) + + is EtsArrayAccess -> { + array.toPathOrNull()?.let { + it + ElementAccessor + } + } + + is EtsInstanceFieldRef -> { + instance.toPathOrNull()?.let { + it + FieldAccessor(field.name) + } + } + + is EtsStaticFieldRef -> { + AccessPath(null, listOf(FieldAccessor(field.name, isStatic = true))) + } + + is EtsCastExpr -> arg.toPathOrNull() + + else -> null +} + +fun EtsEntity.toPath(): AccessPath { + return toPathOrNull() ?: error("Unable to build access path for value $this") +} diff --git a/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/util/TypeInferenceStatistics.kt b/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/util/TypeInferenceStatistics.kt new file mode 100644 index 0000000000..54fb50ff70 --- /dev/null +++ b/usvm-dataflow-ts/src/main/kotlin/org/usvm/dataflow/ts/util/TypeInferenceStatistics.kt @@ -0,0 +1,696 @@ +package org.usvm.dataflow.ts.util + +import org.jacodb.ets.base.DEFAULT_ARK_CLASS_NAME +import org.jacodb.ets.base.DEFAULT_ARK_METHOD_NAME +import org.jacodb.ets.base.EtsAnyType +import org.jacodb.ets.base.EtsArrayType +import org.jacodb.ets.base.EtsAssignStmt +import org.jacodb.ets.base.EtsBooleanType +import org.jacodb.ets.base.EtsClassType +import org.jacodb.ets.base.EtsFunctionType +import org.jacodb.ets.base.EtsNullType +import org.jacodb.ets.base.EtsNumberType +import org.jacodb.ets.base.EtsParameterRef +import org.jacodb.ets.base.EtsStringType +import org.jacodb.ets.base.EtsType +import org.jacodb.ets.base.EtsUnclearRefType +import org.jacodb.ets.base.EtsUndefinedType +import org.jacodb.ets.base.EtsUnionType +import org.jacodb.ets.base.EtsUnknownType +import org.jacodb.ets.base.INSTANCE_INIT_METHOD_NAME +import org.jacodb.ets.base.STATIC_INIT_METHOD_NAME +import org.jacodb.ets.model.EtsClassSignature +import org.jacodb.ets.model.EtsMethod +import org.jacodb.ets.model.EtsScene +import org.usvm.dataflow.ts.graph.EtsApplicationGraph +import org.usvm.dataflow.ts.infer.AccessPathBase +import org.usvm.dataflow.ts.infer.EtsTypeFact +import org.usvm.dataflow.ts.infer.TypeInferenceResult +import java.io.File + +class TypeInferenceStatistics { + private val noTypesInferred: MutableSet = hashSetOf() + private val allTypesAndFacts: MutableMap> = + hashMapOf() + + private var overallTypes: Long = 0 + + private var exactTypeInferredPreviouslyUnknown = 0L + private var exactTypeInferredCorrectlyPreviouslyKnown = 0L + private var exactTypeInferredPreviouslyWasAny = 0L + private var exactTypeInferredIncorrectlyPreviouslyKnown = 0L + + private var typeInfoInferredPreviouslyUnknown = 0L + private var typeInfoInferredPreviouslyKnownExactly = 0L + private var arrayInfoPreviouslyUnknown = 0L + private var arrayInfoPreviouslyKnown = 0L + + private var noInfoInferredPreviouslyKnown = 0L + private var noInfoInferredPreviouslyUnknown = 0L + + private var unhandled = 0L + + private var undefinedUnknownCombination = 0L + private var unknownAnyCombination = 0L + + fun compareSingleMethodFactsWithTypesInScene( + methodTypeFacts: MethodTypesFacts, + method: EtsMethod, + graph: EtsApplicationGraph, + ) { + overallTypes += 1 // thisType + overallTypes += method.parameters.size + method.locals.size + + methodTypeFacts.apply { + if (combinedThisFact == null + && argumentsFacts.all { it == null } + && returnFact == null + && localFacts.isEmpty() + ) { + noTypesInferred += method + } + } + + val thisType = getEtsClassType(method.enclosingClass, graph.cp) + val argTypes = method.parameters.map { it.type } + val locals = method.locals + + val methodFacts = mutableListOf() + + thisType?.let { + val thisPosition = AccessPathBase.This + val fact = methodTypeFacts.combinedThisFact + + val status = if (fact == null) { + // TODO check how unknown is represented + if (it.signature.name == "Unknown") { + noInfoInferredPreviouslyUnknown++ + InferenceStatus.NO_INFO_PREVIOUSLY_UNKNOWN + } else { + noInfoInferredPreviouslyKnown++ + InferenceStatus.NO_INFO_PREVIOUSLY_KNOWN + } + } else { + when { + fact.matchesWith(it) -> { + exactTypeInferredCorrectlyPreviouslyKnown++ + InferenceStatus.EXACT_MATCH_PREVIOUSLY_KNOWN + } + + (fact as? EtsTypeFact.ObjectEtsTypeFact)?.cls != null -> { + exactTypeInferredIncorrectlyPreviouslyKnown++ + InferenceStatus.DIFFERENT_TYPE_FOUND + } + + else -> { + typeInfoInferredPreviouslyKnownExactly++ + InferenceStatus.TYPE_INFO_FOUND_PREV_KNOWN + } + } + } + + methodFacts += InferenceResult(thisPosition, it, fact, status) + } + + argTypes.forEachIndexed { index, type -> + val fact = methodTypeFacts.argumentsFacts.getOrNull(index) + + val status = if (fact == null) { + if (type is EtsUnknownType) { + noInfoInferredPreviouslyUnknown++ + InferenceStatus.NO_INFO_PREVIOUSLY_UNKNOWN + } else { + noInfoInferredPreviouslyKnown++ + InferenceStatus.NO_INFO_PREVIOUSLY_KNOWN + } + } else { + checkForFact(fact, type) + } + + methodFacts += InferenceResult(AccessPathBase.Arg(index), type, fact, status) + } + + + + locals.forEach { + val type = it.type + val local = AccessPathBase.Local(it.name) + val fact = methodTypeFacts.localFacts[local] + + val status = if (fact == null) { + if (type is EtsUnknownType) { + noInfoInferredPreviouslyUnknown++ + InferenceStatus.NO_INFO_PREVIOUSLY_UNKNOWN + } else { + noInfoInferredPreviouslyKnown++ + InferenceStatus.NO_INFO_PREVIOUSLY_KNOWN + } + } else { + checkForFact(fact, type) + } + + methodFacts += InferenceResult(local, type, fact, status) + } + + allTypesAndFacts[method] = methodFacts + } + + private fun checkForFact(fact: EtsTypeFact, type: EtsType): InferenceStatus { + return when (fact) { + EtsTypeFact.AnyEtsTypeFact -> { + when (type) { + is EtsUnknownType -> { + unknownAnyCombination++ + InferenceStatus.UNKNOWN_ANY_COMBINATION + } + + is EtsAnyType -> { + exactTypeInferredPreviouslyWasAny++ + InferenceStatus.EXACT_MATCH_PREVIOUSLY_WAS_ANY + } + + else -> { + exactTypeInferredIncorrectlyPreviouslyKnown++ + InferenceStatus.DIFFERENT_TYPE_FOUND + } + } + } + + is EtsTypeFact.ArrayEtsTypeFact -> { + when (type) { + is EtsArrayType -> { + arrayInfoPreviouslyKnown++ + InferenceStatus.ARRAY_INFO + } + + is EtsUnclearRefType -> { + arrayInfoPreviouslyKnown++ + InferenceStatus.ARRAY_INFO + } + + is EtsUnknownType -> { + arrayInfoPreviouslyUnknown++ + InferenceStatus.ARRAY_INFO_PREV_UNKNOWN + } + + else -> { + arrayInfoPreviouslyKnown++ + InferenceStatus.ARRAY_INFO + } + } + } + + EtsTypeFact.BooleanEtsTypeFact -> { + when (type) { + is EtsBooleanType -> { + exactTypeInferredCorrectlyPreviouslyKnown++ + InferenceStatus.EXACT_MATCH_PREVIOUSLY_KNOWN + } + + is EtsUnknownType -> { + exactTypeInferredPreviouslyUnknown++ + InferenceStatus.EXACT_MATCH_PREVIOUSLY_UNKNOWN + } + + else -> { + when { + (type as? EtsClassType)?.typeName == "Boolean" -> { + exactTypeInferredCorrectlyPreviouslyKnown++ + InferenceStatus.EXACT_MATCH_PREVIOUSLY_KNOWN + } + + (type as? EtsUnclearRefType)?.typeName == "Boolean" -> { + exactTypeInferredCorrectlyPreviouslyKnown++ + InferenceStatus.EXACT_MATCH_PREVIOUSLY_KNOWN + } + + else -> { + exactTypeInferredIncorrectlyPreviouslyKnown++ + InferenceStatus.DIFFERENT_TYPE_FOUND + } + } + } + } + } + + EtsTypeFact.FunctionEtsTypeFact -> { + when (type) { + is EtsUnknownType -> { + undefinedUnknownCombination++ + InferenceStatus.UNKNOWN_UNDEFINED_COMBINATION + } + + is EtsFunctionType -> { + exactTypeInferredCorrectlyPreviouslyKnown++ + InferenceStatus.EXACT_MATCH_PREVIOUSLY_KNOWN + } + + else -> { + exactTypeInferredIncorrectlyPreviouslyKnown++ + InferenceStatus.DIFFERENT_TYPE_FOUND + } + } + } + + EtsTypeFact.NullEtsTypeFact -> { + when (type) { + is EtsUnknownType -> { + undefinedUnknownCombination++ + InferenceStatus.UNKNOWN_UNDEFINED_COMBINATION + } + + is EtsNullType -> { + exactTypeInferredCorrectlyPreviouslyKnown++ + InferenceStatus.EXACT_MATCH_PREVIOUSLY_KNOWN + } + + else -> { + exactTypeInferredIncorrectlyPreviouslyKnown++ + InferenceStatus.DIFFERENT_TYPE_FOUND + } + } + } + + EtsTypeFact.NumberEtsTypeFact -> { + when (type) { + is EtsNumberType -> { + exactTypeInferredCorrectlyPreviouslyKnown++ + InferenceStatus.EXACT_MATCH_PREVIOUSLY_KNOWN + } + + is EtsUnknownType -> { + exactTypeInferredPreviouslyUnknown++ + InferenceStatus.EXACT_MATCH_PREVIOUSLY_UNKNOWN + } + + else -> { + when { + (type as? EtsClassType)?.typeName == "Number" -> { + exactTypeInferredCorrectlyPreviouslyKnown++ + InferenceStatus.EXACT_MATCH_PREVIOUSLY_KNOWN + } + + (type as? EtsUnclearRefType)?.typeName == "Number" -> { + exactTypeInferredCorrectlyPreviouslyKnown++ + InferenceStatus.EXACT_MATCH_PREVIOUSLY_KNOWN + } + + else -> { + exactTypeInferredIncorrectlyPreviouslyKnown++ + InferenceStatus.DIFFERENT_TYPE_FOUND + } + } + } + } + } + + is EtsTypeFact.ObjectEtsTypeFact -> { + if (type is EtsUnknownType) { + return if (fact.cls != null) { + exactTypeInferredPreviouslyUnknown++ + InferenceStatus.EXACT_MATCH_PREVIOUSLY_UNKNOWN + } else { + typeInfoInferredPreviouslyUnknown++ + InferenceStatus.TYPE_INFO_FOUND_PREV_UNKNOWN + } + } + + if (type is EtsAnyType) { + return if (fact.cls != null) { + exactTypeInferredPreviouslyWasAny++ + InferenceStatus.EXACT_MATCH_PREVIOUSLY_WAS_ANY + } else { + typeInfoInferredPreviouslyKnownExactly++ + InferenceStatus.TYPE_INFO_FOUND_PREV_KNOWN + } + } + + if (fact.cls == null) { + return InferenceStatus.TYPE_INFO_FOUND_PREV_KNOWN + } + + val typeName = fact.cls.typeName + + if ((type is EtsClassType || type is EtsUnclearRefType) && type.typeName == typeName) { + exactTypeInferredCorrectlyPreviouslyKnown++ + InferenceStatus.EXACT_MATCH_PREVIOUSLY_KNOWN + } else { + exactTypeInferredIncorrectlyPreviouslyKnown++ + InferenceStatus.DIFFERENT_TYPE_FOUND + } + } + + EtsTypeFact.StringEtsTypeFact -> { + when (type) { + is EtsStringType -> { + exactTypeInferredCorrectlyPreviouslyKnown++ + InferenceStatus.EXACT_MATCH_PREVIOUSLY_KNOWN + } + + is EtsUnknownType -> { + exactTypeInferredPreviouslyUnknown++ + InferenceStatus.EXACT_MATCH_PREVIOUSLY_UNKNOWN + } + + else -> { + when { + (type as? EtsClassType)?.typeName == "String" -> { + exactTypeInferredCorrectlyPreviouslyKnown++ + InferenceStatus.EXACT_MATCH_PREVIOUSLY_KNOWN + } + + (type as? EtsUnclearRefType)?.typeName == "String" -> { + exactTypeInferredCorrectlyPreviouslyKnown++ + InferenceStatus.EXACT_MATCH_PREVIOUSLY_KNOWN + } + + else -> { + exactTypeInferredIncorrectlyPreviouslyKnown++ + InferenceStatus.DIFFERENT_TYPE_FOUND + } + } + } + } + } + + EtsTypeFact.UndefinedEtsTypeFact -> { + when (type) { + is EtsUnknownType -> { + undefinedUnknownCombination++ + InferenceStatus.UNKNOWN_UNDEFINED_COMBINATION + } + + is EtsUndefinedType -> { + exactTypeInferredCorrectlyPreviouslyKnown++ + InferenceStatus.EXACT_MATCH_PREVIOUSLY_KNOWN + } + + else -> { + exactTypeInferredIncorrectlyPreviouslyKnown++ + InferenceStatus.DIFFERENT_TYPE_FOUND + } + } + } + + EtsTypeFact.UnknownEtsTypeFact -> { + if (type is EtsUnknownType) { + exactTypeInferredCorrectlyPreviouslyKnown++ + InferenceStatus.EXACT_MATCH_PREVIOUSLY_KNOWN + } else { + exactTypeInferredIncorrectlyPreviouslyKnown++ + InferenceStatus.DIFFERENT_TYPE_FOUND + } + } + + is EtsTypeFact.GuardedTypeFact -> { + unhandled++ + InferenceStatus.UNHANDLED + } + + is EtsTypeFact.IntersectionEtsTypeFact -> { + unhandled++ + InferenceStatus.UNHANDLED + } + + is EtsTypeFact.UnionEtsTypeFact -> { + when (type) { + is EtsUnionType -> { + unhandled++ + InferenceStatus.UNHANDLED + } + + is EtsUnknownType -> { + InferenceStatus.UNION_INSTEAD_OF_UNKNOWN + } + + else -> { + InferenceStatus.DIFFERENT_TYPE_FOUND + } + } + } + } + } + + fun dumpStatistics(outputFilePath: String? = null) { + val data = buildString { + appendLine(this@TypeInferenceStatistics.toString()) + appendLine() + + appendLine("Specifically: ${"=".repeat(42)}") + + val comparator = + Comparator { fst, snd -> + when (fst.position) { + is AccessPathBase.This -> when { + snd.position is AccessPathBase.This -> 0 + else -> -1 + } + + is AccessPathBase.Arg -> when (snd.position) { + is AccessPathBase.This -> 1 + is AccessPathBase.Arg -> { + fst.position.index.compareTo(snd.position.index) + } + + else -> -1 + } + + else -> when (snd.position) { + is AccessPathBase.This, is AccessPathBase.Arg -> 1 + else -> if (fst.position is AccessPathBase.Local && snd.position is AccessPathBase.Local) { + fst.position.name.compareTo(snd.position.name) + } else { + 0 + } + } + } + } + + allTypesAndFacts + .toList() + .sortedBy { it.first.signature.toString() } + .forEach { (method, types) -> + appendLine("${method.signature}:") + + types + .sortedWith(comparator) + .forEach { (path, typeInfo, typeFact, status) -> + appendLine("[${status.message}]: ${path}: $typeInfo -> $typeFact ") + } + appendLine() + } + + appendLine() + appendLine("=".repeat(42)) + appendLine("No types inferred for methods:") + + noTypesInferred + .filterNot { it.name == INSTANCE_INIT_METHOD_NAME || it.name == STATIC_INIT_METHOD_NAME } + .filterNot { it.name == DEFAULT_ARK_METHOD_NAME && it.enclosingClass.name == DEFAULT_ARK_CLASS_NAME } + .sortedBy { it.signature.toString() } + .forEach { + appendLine(it) + } + } + + if (outputFilePath == null) { + println(data) + return + } + + val file = File(outputFilePath) + println("File with statistics is located: ${file.absolutePath}") + file.writeText(data) + } + + private fun calculateImprovement(): Double { + val newExactTypes = exactTypeInferredPreviouslyUnknown + + exactTypeInferredPreviouslyWasAny + + exactTypeInferredIncorrectlyPreviouslyKnown + + arrayInfoPreviouslyUnknown + + val prevTypes = exactTypeInferredCorrectlyPreviouslyKnown + + typeInfoInferredPreviouslyKnownExactly + + arrayInfoPreviouslyKnown + + noInfoInferredPreviouslyKnown + + return newExactTypes.toDouble() / prevTypes * 100 + } + + override fun toString(): String = """ + Total types number: $overallTypes + + Compared to the first state of the Scene: + + Improvement: ${calculateImprovement()}% + + Inferred types that were unknown: $exactTypeInferredPreviouslyUnknown + Inferred types that were already inferred: $exactTypeInferredCorrectlyPreviouslyKnown + Inferred types that were previously inferred as any: $exactTypeInferredPreviouslyWasAny + Inferred types are different from the ones in the Scene: $exactTypeInferredIncorrectlyPreviouslyKnown + + Array types instead of unknown: $arrayInfoPreviouslyUnknown + Array types instead of previously known type: $arrayInfoPreviouslyKnown + + Some facts found about unknown type: $typeInfoInferredPreviouslyUnknown + Some facts found about already inferred type: $typeInfoInferredPreviouslyKnownExactly + + Unhandled type info: $unhandled + + Lost info about type: $noInfoInferredPreviouslyKnown + Nothing inferred, but it was unknown previously as well: $noInfoInferredPreviouslyUnknown + + Was unknown, became undefined: $undefinedUnknownCombination + Was unknown, became any: $unknownAnyCombination + """.trimIndent() + +} + +private fun getEtsClassType(signature: EtsClassSignature, scene: EtsScene): EtsClassType? { + if (signature.name == DEFAULT_ARK_CLASS_NAME || signature.name.isBlank()) { + return null + } + + val clazz = scene.projectAndSdkClasses.firstOrNull { it.signature == signature } + ?: error("No class found in the classpath with signature $signature") + return EtsClassType(clazz.signature) +} + +data class MethodTypesFacts( + val combinedThisFact: EtsTypeFact?, + val argumentsFacts: List, + val localFacts: Map, + val returnFact: EtsTypeFact?, +) { + companion object { + fun from( + result: TypeInferenceResult, + m: EtsMethod, + ): MethodTypesFacts { + val combinedThisFact = result.inferredCombinedThisType.entries.firstOrNull { + it.key.name == m.enclosingClass.name + }?.value + + val factsForMethod = result.inferredTypes.entries.singleOrNull { + compareByMethodNameAndEnclosingClass(it.key, m) + }?.value + + val inferredReturnType = result.inferredReturnType.entries.firstOrNull { + compareByMethodNameAndEnclosingClass(it.key, m) + }?.value + + val arguments = m.parameters.indices.map { + val stmts = m.cfg.stmts + if (stmts.isEmpty()) return@map null + + val realIndex = ((stmts[it] as EtsAssignStmt).rhv as EtsParameterRef).index + factsForMethod?.get(AccessPathBase.Arg(realIndex)) + } + + val locals = factsForMethod?.filterKeys { it is AccessPathBase.Local }.orEmpty() + + return MethodTypesFacts(combinedThisFact, arguments, locals, inferredReturnType) + } + } +} + +// TODO hack because of an issue with signatures +private val compareByMethodNameAndEnclosingClass = { fst: EtsMethod, snd: EtsMethod -> + fst.name === snd.name && fst.enclosingClass.name === snd.enclosingClass.name +} + +private fun EtsTypeFact.matchesWith(type: EtsType): Boolean { + val result = when (this) { + EtsTypeFact.AnyEtsTypeFact -> { + type is EtsAnyType + } + + is EtsTypeFact.ObjectEtsTypeFact -> { + val typeName = this.cls?.typeName + + if (type is EtsUnknownType || type is EtsAnyType) { + this.cls != null + } else { + (type is EtsClassType || type is EtsUnclearRefType) && type.typeName == typeName + } + } + + is EtsTypeFact.ArrayEtsTypeFact -> when (type) { + is EtsArrayType -> this.elementType.matchesWith(type.elementType) + + is EtsUnclearRefType -> { + val elementType = this.elementType as? EtsTypeFact.ObjectEtsTypeFact + elementType?.cls?.typeName == type.typeName + } + + else -> false + } + + EtsTypeFact.BooleanEtsTypeFact -> { + type is EtsBooleanType + || type is EtsUnknownType + || (type as? EtsClassType)?.typeName == "Boolean" + || (type as? EtsUnclearRefType)?.typeName == "Boolean" + } + + EtsTypeFact.FunctionEtsTypeFact -> type is EtsFunctionType || type is EtsUnknownType + EtsTypeFact.NullEtsTypeFact -> type is EtsNullType || type is EtsUnknownType + EtsTypeFact.NumberEtsTypeFact -> { + type is EtsNumberType + || type is EtsUnknownType + || (type as? EtsClassType)?.typeName == "Number" + || (type as? EtsUnclearRefType)?.typeName == "Number" + } + + EtsTypeFact.StringEtsTypeFact -> { + type is EtsStringType + || type is EtsUnknownType + || (type as? EtsClassType)?.typeName == "String" + || (type as? EtsUnclearRefType)?.typeName == "String" + } + + EtsTypeFact.UndefinedEtsTypeFact -> type is EtsUndefinedType + EtsTypeFact.UnknownEtsTypeFact -> type is EtsUnknownType + is EtsTypeFact.GuardedTypeFact -> TODO() + is EtsTypeFact.IntersectionEtsTypeFact -> { + // TODO intersections checks are not supported yet + false + } + + is EtsTypeFact.UnionEtsTypeFact -> types.all { it.matchesWith(type) } + } + + return result +} + +private data class InferenceResult( + val position: AccessPathBase, + val type: EtsType, + val typeFact: EtsTypeFact?, + val status: InferenceStatus, +) + +private enum class InferenceStatus(val message: String) { + EXACT_MATCH_PREVIOUSLY_KNOWN("Exactly matched, previously known"), + EXACT_MATCH_PREVIOUSLY_UNKNOWN("Exactly matched, previously unknown"), + EXACT_MATCH_PREVIOUSLY_WAS_ANY("Exactly matched with any type"), + + TYPE_INFO_FOUND_PREV_KNOWN("Some type facts found, previously known exactly"), + TYPE_INFO_FOUND_PREV_UNKNOWN("Some type facts found, previously unknown"), + DIFFERENT_TYPE_FOUND("Another type is inferred"), + + UNHANDLED("Unhandled type info"), + + UNKNOWN_ANY_COMBINATION("Unknown any combination"), + UNKNOWN_UNDEFINED_COMBINATION("Unknown undefined combination"), + + ARRAY_INFO_PREV_UNKNOWN("Found an array type, previously unknown"), + + ARRAY_INFO("Array information"), + + UNION_INSTEAD_OF_UNKNOWN("Discovered union type, previously unknown"), + + NO_INFO_PREVIOUSLY_UNKNOWN("Not inferred, previously unknown"), + NO_INFO_PREVIOUSLY_KNOWN("Not inferred, previously known") + +} diff --git a/usvm-dataflow-ts/src/main/resources/logback.xml b/usvm-dataflow-ts/src/main/resources/logback.xml new file mode 100644 index 0000000000..656c393170 --- /dev/null +++ b/usvm-dataflow-ts/src/main/resources/logback.xml @@ -0,0 +1,19 @@ + + + + %highlight([%level]) %replace(%c{0}){'(\$Companion)?\$logger\$1',''} - %msg%n + + + + + logs/app.log + + %highlight([%level]) %replace(%c{0}){'(\$Companion)?\$logger\$1',''} - %msg%n + + + + + + + + diff --git a/usvm-dataflow-ts/src/test/kotlin/org/usvm/dataflow/ts/test/EtsIfdsTest.kt b/usvm-dataflow-ts/src/test/kotlin/org/usvm/dataflow/ts/test/EtsIfdsTest.kt new file mode 100644 index 0000000000..d40ef9a675 --- /dev/null +++ b/usvm-dataflow-ts/src/test/kotlin/org/usvm/dataflow/ts/test/EtsIfdsTest.kt @@ -0,0 +1,437 @@ +/* + * Copyright 2022 UnitTestBot contributors (utbot.org) + *

+ * 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. + */ + +package org.usvm.dataflow.ts.test + +import org.jacodb.ets.model.EtsFile +import org.jacodb.ets.model.EtsMethod +import org.jacodb.ets.model.EtsScene +import org.jacodb.taint.configuration.Argument +import org.jacodb.taint.configuration.AssignMark +import org.jacodb.taint.configuration.ConstantTrue +import org.jacodb.taint.configuration.ContainsMark +import org.jacodb.taint.configuration.CopyAllMarks +import org.jacodb.taint.configuration.RemoveMark +import org.jacodb.taint.configuration.Result +import org.jacodb.taint.configuration.TaintConfigurationItem +import org.jacodb.taint.configuration.TaintMark +import org.jacodb.taint.configuration.TaintMethodSink +import org.jacodb.taint.configuration.TaintMethodSource +import org.jacodb.taint.configuration.TaintPassThrough +import org.junit.jupiter.api.Assertions +import org.junit.jupiter.api.Disabled +import org.junit.jupiter.api.Test +import org.junit.jupiter.api.condition.EnabledIf +import org.usvm.dataflow.ifds.SingletonUnit +import org.usvm.dataflow.ifds.UnitResolver +import org.usvm.dataflow.taint.TaintAnalysisOptions +import org.usvm.dataflow.taint.TaintManager +import org.usvm.dataflow.ts.graph.EtsApplicationGraphImpl +import org.usvm.dataflow.ts.loadEtsFileFromResource +import org.usvm.dataflow.ts.util.EtsTraits +import kotlin.io.path.exists +import kotlin.io.path.toPath +import kotlin.time.Duration.Companion.seconds + +private val logger = mu.KotlinLogging.logger {} + +@Disabled("Have several issues with EtsIR") +class EtsIfdsTest { + + companion object { + private const val BASE_PATH = "/etsir/samples" + + private fun loadSample(programName: String): EtsFile { + return loadEtsFileFromResource("$BASE_PATH/${programName}.ts.json") + } + } + + private fun projectAvailable(): Boolean { + val resource = object {}::class.java.getResource("/samples/source/project1")?.toURI() + return resource != null && resource.toPath().exists() + } + + @Test + fun `test taint analysis on MethodCollision`() { + val file = loadSample("MethodCollision") + val project = EtsScene(listOf(file)) + val graph = EtsApplicationGraphImpl(project) + val unitResolver = UnitResolver { SingletonUnit } + val getConfigForMethod: (EtsMethod) -> List? = + { method -> + val rules = buildList { + if (method.name == "isSame" && method.signature.enclosingClass.name == "Foo") add( + TaintMethodSource( + method = method, + condition = ConstantTrue, + actionsAfter = listOf( + AssignMark(mark = TaintMark("TAINT"), position = Result), + ), + ) + ) + if (method.name == "log") add( + TaintMethodSink( + method = method, + ruleNote = "CUSTOM SINK", // FIXME + cwe = listOf(), // FIXME + condition = ContainsMark(position = Argument(0), mark = TaintMark("TAINT")) + ) + ) + } + rules.ifEmpty { null } + } + val manager = TaintManager( + traits = EtsTraits(), + graph = graph, + unitResolver = unitResolver, + getConfigForMethod = getConfigForMethod, + ) + + val methods = project.projectClasses.flatMap { it.methods }.filter { it.name == "main" } + logger.info { "Methods: ${methods.size}" } + for (method in methods) { + logger.info { " ${method.name}" } + } + val sinks = manager.analyze(methods, timeout = 60.seconds) + logger.info { "Sinks: $sinks" } + Assertions.assertTrue(sinks.isNotEmpty()) + } + + @Test + fun `test taint analysis on TypeMismatch`() { + val file = loadSample("TypeMismatch") + val project = EtsScene(listOf(file)) + val graph = EtsApplicationGraphImpl(project) + val unitResolver = UnitResolver { SingletonUnit } + val getConfigForMethod: (EtsMethod) -> List? = + { method -> + val rules = buildList { + if (method.name == "add") add( + TaintMethodSource( + method = method, + condition = ConstantTrue, + actionsAfter = listOf( + AssignMark(mark = TaintMark("TAINT"), position = Result), + ) + ) + ) + if (method.name == "log") add( + TaintMethodSink( + method = method, + ruleNote = "CUSTOM SINK", // FIXME + cwe = listOf(), // FIXME + condition = ContainsMark(position = Argument(1), mark = TaintMark("TAINT")) + ) + ) + } + rules.ifEmpty { null } + } + val manager = TaintManager( + traits = EtsTraits(), + graph = graph, + unitResolver = unitResolver, + getConfigForMethod = getConfigForMethod, + ) + + val methods = project.projectClasses.flatMap { it.methods } + logger.info { "Methods: ${methods.size}" } + for (method in methods) { + logger.info { " ${method.name}" } + } + val sinks = manager.analyze(methods, timeout = 60.seconds) + logger.info { "Sinks: $sinks" } + Assertions.assertTrue(sinks.isNotEmpty()) + } + + @Disabled("TODO: Sink should be detected in the 'good' method") + @Test + fun `test taint analysis on DataFlowSecurity`() { + val file = loadSample("DataFlowSecurity") + val project = EtsScene(listOf(file)) + val graph = EtsApplicationGraphImpl(project) + val unitResolver = UnitResolver { SingletonUnit } + val getConfigForMethod: (EtsMethod) -> List? = + { method -> + val rules = buildList { + if (method.name == "samples/source") add( + TaintMethodSource( + method = method, + condition = ConstantTrue, + actionsAfter = listOf( + AssignMark(mark = TaintMark("TAINT"), position = Result), + ), + ) + ) + if (method.name == "sink") add( + TaintMethodSink( + method = method, + ruleNote = "SINK", // FIXME + cwe = listOf(), // FIXME + condition = ContainsMark(position = Argument(0), mark = TaintMark("TAINT")) + ) + ) + if (method.name == "pass") add( + TaintPassThrough( + method = method, + condition = ConstantTrue, + actionsAfter = listOf( + CopyAllMarks(from = Argument(0), to = Result) + ), + ) + ) + if (method.name == "validate") add( + TaintPassThrough( + method = method, + condition = ConstantTrue, + actionsAfter = listOf( + RemoveMark(mark = TaintMark("TAINT"), position = Argument(0)) + ), + ) + ) + } + rules.ifEmpty { null } + } + val manager = TaintManager( + traits = EtsTraits(), + graph = graph, + unitResolver = unitResolver, + getConfigForMethod = getConfigForMethod, + ) + + val goodMethod = project.projectClasses.flatMap { it.methods }.single { it.name == "good" } + logger.info { "good() method: $goodMethod" } + val goodSinks = manager.analyze(listOf(goodMethod), timeout = 60.seconds) + logger.info { "Sinks in good(): $goodSinks" } + Assertions.assertTrue(goodSinks.isEmpty()) + + val badMethod = project.projectClasses.flatMap { it.methods }.single { it.name == "bad" } + logger.info { "bad() method: $badMethod" } + val badSinks = manager.analyze(listOf(badMethod), timeout = 60.seconds) + logger.info { "Sinks in bad(): $badSinks" } + Assertions.assertTrue(badSinks.isNotEmpty()) + } + + @Test + fun `test taint analysis on case1 - untrusted loop bound scenario`() { + val file = loadSample("cases/case1") + val project = EtsScene(listOf(file)) + val graph = EtsApplicationGraphImpl(project) + val unitResolver = UnitResolver { SingletonUnit } + val getConfigForMethod: (EtsMethod) -> List? = + { method -> + val rules = buildList { + if (method.name == "readInt") add( + TaintMethodSource( + method = method, + condition = ConstantTrue, + actionsAfter = listOf( + AssignMark(mark = TaintMark("UNTRUSTED"), position = Result), + ), + ) + ) + } + rules.ifEmpty { null } + } + val manager = TaintManager( + traits = EtsTraits(), + graph = graph, + unitResolver = unitResolver, + getConfigForMethod = getConfigForMethod, + ) + TaintAnalysisOptions.UNTRUSTED_LOOP_BOUND_SINK = true + + val methods = project.projectClasses.flatMap { it.methods } + logger.info { "Methods: ${methods.size}" } + for (method in methods) { + logger.info { " ${method.name}" } + } + val sinks = manager.analyze(methods, timeout = 60.seconds) + logger.info { "Sinks: $sinks" } + Assertions.assertTrue(sinks.isNotEmpty()) + } + + @Test + fun `test taint analysis on case2 - untrusted array buffer size scenario`() { + val file = loadSample("cases/case2") + val project = EtsScene(listOf(file)) + val graph = EtsApplicationGraphImpl(project) + val unitResolver = UnitResolver { SingletonUnit } + val getConfigForMethod: (EtsMethod) -> List? = + { method -> + val rules = buildList { + if (method.name == "readInt") add( + TaintMethodSource( + method = method, + condition = ConstantTrue, + actionsAfter = listOf( + AssignMark(mark = TaintMark("UNTRUSTED"), position = Result), + ), + ) + ) + } + rules.ifEmpty { null } + } + val manager = TaintManager( + traits = EtsTraits(), + graph = graph, + unitResolver = unitResolver, + getConfigForMethod = getConfigForMethod, + ) + TaintAnalysisOptions.UNTRUSTED_ARRAY_SIZE_SINK = true + + val methods = project.projectClasses.flatMap { it.methods } + logger.info { "Methods: ${methods.size}" } + for (method in methods) { + logger.info { " ${method.name}" } + } + val sinks = manager.analyze(methods, timeout = 60.seconds) + logger.info { "Sinks: $sinks" } + Assertions.assertTrue(sinks.isNotEmpty()) + } + + // TODO(): support AnyArgument Position type for more flexible configs + @Test + fun `test taint analysis on case3 - send plain information with sensitive data`() { + val file = loadSample("cases/case3") + val project = EtsScene(listOf(file)) + val graph = EtsApplicationGraphImpl(project) + val unitResolver = UnitResolver { SingletonUnit } + val getConfigForMethod: (EtsMethod) -> List? = + { method -> + val rules = buildList { + if (method.name == "getPassword") add( + TaintMethodSource( + method = method, + condition = ConstantTrue, + actionsAfter = listOf( + AssignMark(mark = TaintMark("TAINT"), position = Result), + ), + ) + ) + if (method.name == "publishEvent") add( + TaintMethodSink( + method = method, ruleNote = "SINK", // FIXME + cwe = listOf(), // FIXME + condition = ContainsMark(position = Argument(1), mark = TaintMark("TAINT")) + ) + ) + } + rules.ifEmpty { null } + } + val manager = TaintManager( + traits = EtsTraits(), + graph = graph, + unitResolver = unitResolver, + getConfigForMethod = getConfigForMethod, + ) + + val methods = project.projectClasses.flatMap { it.methods } + logger.info { "Methods: ${methods.size}" } + for (method in methods) { + logger.info { " ${method.name}" } + } + val sinks = manager.analyze(methods, timeout = 60.seconds) + logger.info { "Sinks: $sinks" } + Assertions.assertTrue(sinks.isNotEmpty()) + } + + @EnabledIf("projectAvailable") + @Test + fun `test taint analysis on AccountManager`() { + val file = loadEtsFileFromResource("/etsir/project1/entry/src/main/ets/base/account/AccountManager.ts.json") + val project = EtsScene(listOf(file)) + val graph = EtsApplicationGraphImpl(project) + val unitResolver = UnitResolver { SingletonUnit } + val getConfigForMethod: (EtsMethod) -> List? = + { method -> + val rules = buildList { + // adhoc taint second argument (cursor: string) + if (method.name == "taintSink") add( + TaintMethodSink( + method = method, + cwe = listOf(), + ruleNote = "SINK", + condition = ContainsMark(position = Argument(0), mark = TaintMark("TAINT")), + ) + ) +// // encodeURI* +// if (method.name.startsWith("encodeURI")) add( +// TaintMethodSource( +// method = method, +// condition = ContainsMark(position = Argument(0), mark = TaintMark("UNSAFE")), +// actionsAfter = listOf( +// RemoveMark(position = Result, mark = TaintMark("UNSAFE")), +// ), +// ) +// ) +// // RequestOption.setUrl +// if (method.name == "setUrl") add( +// TaintMethodSource( +// method = method, +// condition = ConstantTrue, +// actionsAfter = listOf( +// CopyMark( +// mark = TaintMark("UNSAFE"), +// from = Argument(0), +// to = Result +// ), +// ), +// ) +// ) +// // HttpManager.requestSync +// if (method.name == "requestSync") add( +// TaintMethodSink( +// method = method, +// ruleNote = "Unsafe request", // FIXME +// cwe = listOf(), // FIXME +// condition = ContainsMark(position = Argument(0), mark = TaintMark("UNSAFE")) +// ) +// ) + // SyncUtil.requestGet + if (method.name == "requestGet") add( + TaintMethodSource( + method = method, + condition = ConstantTrue, + actionsAfter = listOf(AssignMark(position = Result, mark = TaintMark("TAINT"))) + ) + ) + } + rules.ifEmpty { null } + } + val manager = TaintManager( + traits = EtsTraits(), + graph = graph, + unitResolver = unitResolver, + getConfigForMethod = getConfigForMethod, + ) + + val methodNames = setOf( + "getDeviceIdListWithCursor", + "requestGet", + "taintRun", + "taintSink" + ) + + val methods = project.projectClasses.flatMap { it.methods }.filter { it.name in methodNames } + logger.info { "Methods: ${methods.size}" } + for (method in methods) { + logger.info { " ${method.name}" } + } + val sinks = manager.analyze(methods, timeout = 60.seconds) + logger.info { "Sinks: $sinks" } + Assertions.assertTrue(sinks.isNotEmpty()) + } +} diff --git a/usvm-dataflow-ts/src/test/kotlin/org/usvm/dataflow/ts/test/EtsProjectAnalysisTest.kt b/usvm-dataflow-ts/src/test/kotlin/org/usvm/dataflow/ts/test/EtsProjectAnalysisTest.kt new file mode 100644 index 0000000000..c8af281ee6 --- /dev/null +++ b/usvm-dataflow-ts/src/test/kotlin/org/usvm/dataflow/ts/test/EtsProjectAnalysisTest.kt @@ -0,0 +1,155 @@ +/* + * Copyright 2022 UnitTestBot contributors (utbot.org) + *

+ * 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. + */ + +package org.usvm.dataflow.ts.test + +import mu.KotlinLogging +import org.jacodb.ets.base.EtsStmt +import org.jacodb.ets.model.EtsFile +import org.jacodb.ets.model.EtsMethod +import org.jacodb.ets.model.EtsScene +import org.junit.jupiter.api.Test +import org.junit.jupiter.api.condition.EnabledIf +import org.usvm.dataflow.ifds.SingletonUnit +import org.usvm.dataflow.ifds.UnitResolver +import org.usvm.dataflow.taint.TaintManager +import org.usvm.dataflow.taint.TaintVulnerability +import org.usvm.dataflow.ts.getResourcePath +import org.usvm.dataflow.ts.getResourceStream +import org.usvm.dataflow.ts.graph.EtsApplicationGraphImpl +import org.usvm.dataflow.ts.loadEtsFileFromResource +import org.usvm.dataflow.ts.test.utils.getConfigForMethod +import org.usvm.dataflow.ts.test.utils.loadRules +import org.usvm.dataflow.ts.util.EtsTraits +import org.usvm.dataflow.util.getPathEdges +import java.nio.file.Files +import kotlin.io.path.exists +import kotlin.io.path.toPath +import kotlin.time.Duration +import kotlin.time.Duration.Companion.milliseconds +import kotlin.time.Duration.Companion.seconds + +private val logger = KotlinLogging.logger {} + +class EtsProjectAnalysisTest { + private var tsLinesSuccess = 0L + private var tsLinesFailed = 0L + private var analysisTime: Duration = Duration.ZERO + private var totalPathEdges = 0 + private var totalSinks: MutableList> = mutableListOf() + + companion object { + private const val SOURCE_PROJECT_PATH = + "/projects/applications_app_samples/source/applications_app_samples/code/SuperFeature/DistributedAppDev/ArkTSDistributedCalc" + private const val PROJECT_PATH = "/projects/applications_app_samples/etsir/ast/ArkTSDistributedCalc" + private const val START_PATH = "/entry/src/main/ets" + private const val BASE_PATH = PROJECT_PATH + private const val SOURCE_BASE_PATH = SOURCE_PROJECT_PATH + START_PATH + + private fun loadFromProject(filename: String): EtsFile { + return loadEtsFileFromResource("$BASE_PATH/$filename.json") + } + + private fun countFileLines(path: String): Long { + return getResourceStream(path).bufferedReader().use { reader -> + reader.lines().count() + } + } + + val rules by lazy { loadRules("config1.json") } + } + + private fun projectAvailable(): Boolean { + val path = object {}::class.java.getResource(PROJECT_PATH)?.toURI()?.toPath() + return path != null && path.exists() + } + + @EnabledIf("projectAvailable") + @Test + fun processAllFiles() { + val baseDir = getResourcePath(BASE_PATH) + Files.walk(baseDir) + .filter { it.toString().endsWith(".json") } + .map { baseDir.relativize(it).toString().replace("\\", "/").substringBeforeLast('.') } + .forEach { filename -> + handleFile(filename) + } + makeReport() + } + + private fun makeReport() { + logger.info { "Analysis Report On $PROJECT_PATH" } + logger.info { "====================" } + logger.info { "Total files processed: ${tsLinesSuccess + tsLinesFailed}" } + logger.info { "Successfully processed lines: $tsLinesSuccess" } + logger.info { "Failed lines: $tsLinesFailed" } + logger.info { "Total analysis time: $analysisTime" } + logger.info { "Total path edges: $totalPathEdges" } + logger.info { "Found sinks: ${totalSinks.size}" } + + if (totalSinks.isNotEmpty()) { + totalSinks.forEachIndexed { idx, sink -> + logger.info { + """Detailed Sink Information: + | + |Sink ID: $idx + |Statement: ${sink.sink.statement} + |Fact: ${sink.sink.fact} + |Condition: ${sink.rule?.condition} + | + """.trimMargin() + } + } + } else { + logger.info { "No sinks found." } + } + logger.info { "====================" } + logger.info { "End of report" } + } + + private fun handleFile(filename: String) { + val fileLines = countFileLines("$SOURCE_BASE_PATH/$filename") + try { + logger.info { "Processing '$filename'" } + val file = loadFromProject(filename) + val project = EtsScene(listOf(file)) + val startTime = System.currentTimeMillis() + runAnalysis(project) + val endTime = System.currentTimeMillis() + analysisTime += (endTime - startTime).milliseconds + tsLinesSuccess += fileLines + } catch (e: Exception) { + logger.warn { "Failed to process '$filename': $e" } + logger.warn { e.stackTraceToString() } + tsLinesFailed += fileLines + } + } + + private fun runAnalysis(project: EtsScene) { + val graph = EtsApplicationGraphImpl(project) + val unitResolver = UnitResolver { SingletonUnit } + val manager = TaintManager( + traits = EtsTraits(), + graph = graph, + unitResolver = unitResolver, + getConfigForMethod = { method -> getConfigForMethod(method, rules) }, + ) + val methods = project.projectClasses.flatMap { it.methods } + val sinks = manager.analyze(methods, timeout = 10.seconds) + totalPathEdges += manager.runnerForUnit.values.sumOf { it.getPathEdges().size } + totalSinks += sinks + } +} diff --git a/usvm-dataflow-ts/src/test/kotlin/org/usvm/dataflow/ts/test/EtsSceneTest.kt b/usvm-dataflow-ts/src/test/kotlin/org/usvm/dataflow/ts/test/EtsSceneTest.kt new file mode 100644 index 0000000000..5ed683007b --- /dev/null +++ b/usvm-dataflow-ts/src/test/kotlin/org/usvm/dataflow/ts/test/EtsSceneTest.kt @@ -0,0 +1,184 @@ +/* + * Copyright 2022 UnitTestBot contributors (utbot.org) + *

+ * 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. + */ + +package org.usvm.dataflow.ts.test + +import org.jacodb.ets.base.CONSTRUCTOR_NAME +import org.jacodb.ets.base.EtsArrayType +import org.jacodb.ets.base.EtsAssignStmt +import org.jacodb.ets.base.EtsCallStmt +import org.jacodb.ets.base.EtsClassType +import org.jacodb.ets.base.EtsInstLocation +import org.jacodb.ets.base.EtsInstanceCallExpr +import org.jacodb.ets.base.EtsLocal +import org.jacodb.ets.base.EtsReturnStmt +import org.jacodb.ets.base.EtsStringType +import org.jacodb.ets.base.EtsThis +import org.jacodb.ets.base.EtsVoidType +import org.jacodb.ets.graph.EtsCfg +import org.jacodb.ets.model.EtsClassImpl +import org.jacodb.ets.model.EtsClassSignature +import org.jacodb.ets.model.EtsFieldImpl +import org.jacodb.ets.model.EtsFieldSignature +import org.jacodb.ets.model.EtsFieldSubSignature +import org.jacodb.ets.model.EtsFile +import org.jacodb.ets.model.EtsFileSignature +import org.jacodb.ets.model.EtsMethodImpl +import org.jacodb.ets.model.EtsMethodSignature +import org.jacodb.ets.model.EtsScene +import org.usvm.dataflow.ts.graph.EtsApplicationGraphImpl +import kotlin.test.Test +import kotlin.test.assertEquals + +class EtsSceneTest { + + @Test + fun `test create EtsScene with multiple files`() { + val fileCatSignature = EtsFileSignature( + projectName = "TestProject", + fileName = "cat.ts", + ) + val classCatSignature = EtsClassSignature( + name = "Cat", + file = fileCatSignature, + namespace = null, + ) + val fieldName = EtsFieldImpl( + signature = EtsFieldSignature( + enclosingClass = classCatSignature, + sub = EtsFieldSubSignature( + name = "name", + type = EtsStringType, + ) + ) + ) + val methodMeow = EtsMethodImpl( + signature = EtsMethodSignature( + enclosingClass = classCatSignature, + name = "meow", + parameters = emptyList(), + returnType = EtsVoidType, + ) + ) + val ctorCat = EtsMethodImpl( + signature = EtsMethodSignature( + enclosingClass = classCatSignature, + name = CONSTRUCTOR_NAME, + parameters = emptyList(), + returnType = EtsVoidType, + ), + ) + val classCat = EtsClassImpl( + signature = classCatSignature, + fields = listOf(fieldName), + methods = listOf(methodMeow), + ctor = ctorCat, + ) + val fileCat = EtsFile( + signature = fileCatSignature, + classes = listOf(classCat), + namespaces = emptyList(), + ) + + val fileBoxSignature = EtsFileSignature( + projectName = "TestProject", + fileName = "box.ts", + ) + val classBoxSignature = EtsClassSignature( + name = "Box", + file = fileBoxSignature, + namespace = null, + ) + val fieldCats = EtsFieldImpl( + signature = EtsFieldSignature( + enclosingClass = classBoxSignature, + sub = EtsFieldSubSignature( + name = "cats", + type = EtsArrayType(EtsClassType(classCatSignature), 1), + ) + ) + ) + val methodTouch = EtsMethodImpl( + signature = EtsMethodSignature( + enclosingClass = classBoxSignature, + name = "touch", + parameters = emptyList(), + returnType = EtsVoidType, + ) + ).also { + var index = 0 + val stmts = listOf( + EtsAssignStmt( + location = EtsInstLocation(it, index++), + lhv = EtsLocal("this", EtsClassType(classBoxSignature)), + rhv = EtsThis(EtsClassType(classBoxSignature)), + ), + EtsCallStmt( + location = EtsInstLocation(it, index++), + expr = EtsInstanceCallExpr( + instance = EtsLocal("this", EtsClassType(classBoxSignature)), + method = methodMeow.signature, + args = emptyList(), + ) + ), + EtsReturnStmt( + location = EtsInstLocation(it, index++), + returnValue = null, + ) + ) + check(index == stmts.size) + val successors = mapOf( + stmts[0] to listOf(stmts[1]), + stmts[1] to listOf(stmts[2]), + ) + it._cfg = EtsCfg( + stmts = stmts, + successorMap = successors, + ) + } + val ctorBox = EtsMethodImpl( + signature = EtsMethodSignature( + enclosingClass = classBoxSignature, + name = CONSTRUCTOR_NAME, + parameters = emptyList(), + returnType = EtsVoidType, + ), + ) + val classBox = EtsClassImpl( + signature = classBoxSignature, + fields = listOf(fieldCats), + methods = listOf(methodTouch), + ctor = ctorBox, + ) + val fileBox = EtsFile( + signature = fileBoxSignature, + classes = listOf(classBox), + namespaces = emptyList(), + ) + + val project = EtsScene(listOf(fileCat, fileBox)) + val graph = EtsApplicationGraphImpl(project) + + val callStmt = project.projectClasses + .asSequence() + .flatMap { it.methods } + .filter { it.name == "touch" } + .flatMap { it.cfg.stmts } + .filterIsInstance() + .first() + assertEquals(methodMeow, graph.callees(callStmt).first()) + } +} diff --git a/usvm-dataflow-ts/src/test/kotlin/org/usvm/dataflow/ts/test/EtsTaintAnalysisTest.kt b/usvm-dataflow-ts/src/test/kotlin/org/usvm/dataflow/ts/test/EtsTaintAnalysisTest.kt new file mode 100644 index 0000000000..6592c48f83 --- /dev/null +++ b/usvm-dataflow-ts/src/test/kotlin/org/usvm/dataflow/ts/test/EtsTaintAnalysisTest.kt @@ -0,0 +1,141 @@ +/* + * Copyright 2022 UnitTestBot contributors (utbot.org) + *

+ * 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. + */ + +package org.usvm.dataflow.ts.test + +import mu.KotlinLogging +import org.jacodb.ets.model.EtsFile +import org.jacodb.ets.model.EtsMethod +import org.jacodb.ets.model.EtsScene +import org.jacodb.taint.configuration.Argument +import org.jacodb.taint.configuration.AssignMark +import org.jacodb.taint.configuration.ConstantTrue +import org.jacodb.taint.configuration.ContainsMark +import org.jacodb.taint.configuration.CopyAllMarks +import org.jacodb.taint.configuration.RemoveMark +import org.jacodb.taint.configuration.Result +import org.jacodb.taint.configuration.TaintConfigurationItem +import org.jacodb.taint.configuration.TaintMark +import org.jacodb.taint.configuration.TaintMethodSink +import org.jacodb.taint.configuration.TaintMethodSource +import org.jacodb.taint.configuration.TaintPassThrough +import org.junit.jupiter.api.Disabled +import org.junit.jupiter.api.Test +import org.usvm.dataflow.ifds.SingletonUnit +import org.usvm.dataflow.ifds.UnitResolver +import org.usvm.dataflow.taint.TaintManager +import org.usvm.dataflow.ts.graph.EtsApplicationGraphImpl +import org.usvm.dataflow.ts.loadEtsFileFromResource +import org.usvm.dataflow.ts.util.EtsTraits +import kotlin.test.assertTrue +import kotlin.time.Duration.Companion.seconds + +private val logger = KotlinLogging.logger {} + +@Disabled("Need to update IR") +class EtsTaintAnalysisTest { + + companion object { + private const val BASE_PATH = "/samples/etsir/ast" + private const val DECOMPILED_PATH = "/decompiled" + + private fun loadFromProject(name: String): EtsFile { + return loadEtsFileFromResource("$BASE_PATH/$name.ts.json") + } + + private fun loadDecompiled(name: String): EtsFile { + return loadEtsFileFromResource("$DECOMPILED_PATH/$name.abc.json") + } + + val getConfigForMethod: (EtsMethod) -> List? = + { method -> + val rules = buildList { + if (method.name == "source") add( + TaintMethodSource( + method = method, + condition = ConstantTrue, + actionsAfter = listOf( + AssignMark(mark = TaintMark("TAINT"), position = Result), + ), + ) + ) + if (method.name == "sink") add( + TaintMethodSink( + method = method, + ruleNote = "SINK", // FIXME + cwe = listOf(), // FIXME + condition = ContainsMark(position = Argument(0), mark = TaintMark("TAINT")) + ) + ) + if (method.name == "pass") add( + TaintPassThrough( + method = method, + condition = ConstantTrue, + actionsAfter = listOf( + CopyAllMarks(from = Argument(0), to = Result) + ), + ) + ) + if (method.name == "validate") add( + TaintPassThrough( + method = method, + condition = ConstantTrue, + actionsAfter = listOf( + RemoveMark(mark = TaintMark("TAINT"), position = Argument(0)) + ), + ) + ) + } + rules.ifEmpty { null } + } + } + + fun runTaintAnalysis(project: EtsScene) { + val graph = EtsApplicationGraphImpl(project) + val unitResolver = UnitResolver { SingletonUnit } + + val manager = TaintManager( + traits = EtsTraits(), + graph = graph, + unitResolver = unitResolver, + getConfigForMethod = getConfigForMethod, + ) + + val methods = project.projectClasses.flatMap { it.methods }.filter { it.name == "bad" } + logger.info { "Methods: ${methods.size}" } + for (method in methods) { + logger.info { " ${method.name}" } + } + val sinks = manager.analyze(methods, timeout = 60.seconds) + logger.info { "Sinks: $sinks" } + assertTrue(sinks.isNotEmpty()) + } + + @Test + fun `test taint analysis`() { + val file = loadFromProject("TaintAnalysis") + val project = EtsScene(listOf(file)) + runTaintAnalysis(project) + } + + @Disabled("Need to update the EtsIR-ABC json file") + @Test + fun `test taint analysis on decompiled file`() { + val file = loadDecompiled("TaintAnalysis") + val project = EtsScene(listOf(file)) + runTaintAnalysis(project) + } +} diff --git a/usvm-dataflow-ts/src/test/kotlin/org/usvm/dataflow/ts/test/EtsTypeAnnotationTest.kt b/usvm-dataflow-ts/src/test/kotlin/org/usvm/dataflow/ts/test/EtsTypeAnnotationTest.kt new file mode 100644 index 0000000000..6683bf7fcf --- /dev/null +++ b/usvm-dataflow-ts/src/test/kotlin/org/usvm/dataflow/ts/test/EtsTypeAnnotationTest.kt @@ -0,0 +1,193 @@ +/* + * Copyright 2022 UnitTestBot contributors (utbot.org) + *

+ * 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. + */ + +package org.usvm.dataflow.ts.test + +import org.jacodb.ets.base.CONSTRUCTOR_NAME +import org.jacodb.ets.base.EtsAddExpr +import org.jacodb.ets.base.EtsAssignStmt +import org.jacodb.ets.base.EtsClassType +import org.jacodb.ets.base.EtsInstLocation +import org.jacodb.ets.base.EtsLocal +import org.jacodb.ets.base.EtsNumberType +import org.jacodb.ets.base.EtsParameterRef +import org.jacodb.ets.base.EtsReturnStmt +import org.jacodb.ets.base.EtsStmt +import org.jacodb.ets.base.EtsStringType +import org.jacodb.ets.base.EtsThis +import org.jacodb.ets.base.EtsType +import org.jacodb.ets.base.EtsUnknownType +import org.jacodb.ets.graph.EtsCfg +import org.jacodb.ets.model.EtsClassImpl +import org.jacodb.ets.model.EtsClassSignature +import org.jacodb.ets.model.EtsDecorator +import org.jacodb.ets.model.EtsFile +import org.jacodb.ets.model.EtsFileSignature +import org.jacodb.ets.model.EtsMethod +import org.jacodb.ets.model.EtsMethodParameter +import org.jacodb.ets.model.EtsMethodSignature +import org.jacodb.ets.model.EtsModifiers +import org.jacodb.ets.model.EtsScene +import org.usvm.dataflow.ts.infer.AccessPathBase +import org.usvm.dataflow.ts.infer.EtsTypeFact +import org.usvm.dataflow.ts.infer.TypeInferenceResult +import org.usvm.dataflow.ts.infer.annotation.EtsTypeAnnotator +import org.usvm.dataflow.ts.infer.verify.LocalId +import org.usvm.dataflow.ts.infer.verify.MethodId +import org.usvm.dataflow.ts.infer.verify.ParameterId +import org.usvm.dataflow.ts.infer.verify.VerificationResult +import org.usvm.dataflow.ts.infer.verify.verify +import kotlin.test.Test + +internal class EtsTypeAnnotationTest { + @Test + fun `test EtsTypeAnnotator`() { + val typeInferenceResult = TypeInferenceResult( + inferredTypes = mapOf( + mainMethod to mapOf( + AccessPathBase.Arg(1) to EtsTypeFact.StringEtsTypeFact, + AccessPathBase.Arg(2) to EtsTypeFact.NumberEtsTypeFact, + AccessPathBase.Local("v1") to EtsTypeFact.StringEtsTypeFact, + AccessPathBase.Local("v2") to EtsTypeFact.NumberEtsTypeFact, + AccessPathBase.Local("v3") to EtsTypeFact.StringEtsTypeFact, + ) + ), + inferredReturnType = mapOf( + mainMethod to EtsTypeFact.StringEtsTypeFact + ), + inferredCombinedThisType = mapOf( + mainClassSignature to EtsTypeFact.ObjectEtsTypeFact( + cls = EtsClassType(mainClassSignature), + properties = mapOf(), + ) + ) + ) + + val annotatedScene = EtsTypeAnnotator(sampleScene, typeInferenceResult) + .annotateWithTypes(sampleScene) + + val verificationResult = verify(annotatedScene) + + assert(verificationResult is VerificationResult.Success) + + with(verificationResult as VerificationResult.Success) { + val main = MethodId(mainMethodSignature) + + assert(mapping[ParameterId(1, main)] == EtsStringType) + assert(mapping[ParameterId(2, main)] == EtsNumberType) + + assert(mapping[LocalId("v1", main)] == EtsStringType) + assert(mapping[LocalId("v2", main)] == EtsNumberType) + assert(mapping[LocalId("v3", main)] == EtsStringType) + } + } + + private val mainTs = EtsFileSignature( + projectName = "sampleProject", + fileName = "main.ts", + ) + + private val mainClassSignature = EtsClassSignature( + name = "MainClass", + file = mainTs, + namespace = null, + ) + + private val mainMethodSignature = EtsMethodSignature( + enclosingClass = mainClassSignature, + name = "mainMethod", + parameters = parameters(2), + returnType = EtsUnknownType, + ) + + private val mainMethod = buildMethod(mainMethodSignature) { + val arg1 = EtsParameterRef(1, EtsUnknownType) + val arg2 = EtsParameterRef(2, EtsUnknownType) + val v1 = EtsLocal("v1", EtsUnknownType) + val v2 = EtsLocal("v2", EtsUnknownType) + val v3 = EtsLocal("v3", EtsUnknownType) + + push(EtsAssignStmt(location, v1, arg1)) + push(EtsAssignStmt(location, v2, arg2)) + push(EtsAssignStmt(location, v3, EtsAddExpr(EtsUnknownType, v1, v2))) + push(EtsReturnStmt(location, v3)) + } + + private val mainClassCtorSignature = EtsMethodSignature( + enclosingClass = mainClassSignature, + name = CONSTRUCTOR_NAME, + parameters = parameters(1), + returnType = EtsUnknownType, + ) + + private val mainClassCtor = buildMethod(mainClassCtorSignature) { + val etsThis = EtsThis(EtsClassType(mainClassSignature)) + push(EtsReturnStmt(location, etsThis)) + } + + private val mainClass = EtsClassImpl( + signature = mainClassSignature, + fields = listOf(), + methods = listOf(mainMethod), + ctor = mainClassCtor, + superClass = null, + ) + + private val mainFile = EtsFile( + signature = mainTs, + classes = listOf(mainClass), + namespaces = listOf(), + ) + + private val sampleScene = EtsScene(listOf(mainFile)) + + private class CfgBuilderContext( + val method: EtsMethod, + ) { + private val stmts = mutableListOf() + private val successorsMap = mutableMapOf>() + + fun build(): EtsCfg = EtsCfg(stmts, successorsMap) + + fun push(stmt: EtsStmt) { + stmts.lastOrNull()?.let { link(it, stmt) } + successorsMap[stmt] = mutableListOf() + stmts.add(stmt) + } + + fun link(from: EtsStmt, to: EtsStmt) { + successorsMap.getOrPut(from, ::mutableListOf).add(to) + } + + val location: EtsInstLocation + get() = EtsInstLocation(method, stmts.size) + } + + private fun buildMethod( + signature: EtsMethodSignature, + cfgBuilder: CfgBuilderContext.() -> Unit, + ) = object : EtsMethod { + override val signature = signature + override val typeParameters: List = emptyList() + override val modifiers: EtsModifiers = EtsModifiers.EMPTY + override val decorators: List = emptyList() + override val locals: List = emptyList() + override val cfg = CfgBuilderContext(this).apply(cfgBuilder).build() + } + + private fun parameters(n: Int) = + List(n) { EtsMethodParameter(it, "a$it", EtsUnknownType) } +} diff --git a/usvm-dataflow-ts/src/test/kotlin/org/usvm/dataflow/ts/test/EtsTypeInferenceTest.kt b/usvm-dataflow-ts/src/test/kotlin/org/usvm/dataflow/ts/test/EtsTypeInferenceTest.kt new file mode 100644 index 0000000000..bf11d2f5d6 --- /dev/null +++ b/usvm-dataflow-ts/src/test/kotlin/org/usvm/dataflow/ts/test/EtsTypeInferenceTest.kt @@ -0,0 +1,593 @@ +/* + * Copyright 2022 UnitTestBot contributors (utbot.org) + *

+ * 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. + */ + +package org.usvm.dataflow.ts.test + +import kotlinx.coroutines.TimeoutCancellationException +import kotlinx.coroutines.runBlocking +import kotlinx.coroutines.runInterruptible +import kotlinx.coroutines.withTimeout +import kotlinx.serialization.SerializationException +import mu.KotlinLogging +import org.jacodb.ets.base.CONSTRUCTOR_NAME +import org.jacodb.ets.base.EtsAnyType +import org.jacodb.ets.base.EtsAssignStmt +import org.jacodb.ets.base.EtsLocal +import org.jacodb.ets.base.EtsStringConstant +import org.jacodb.ets.base.EtsType +import org.jacodb.ets.base.EtsUnknownType +import org.jacodb.ets.dto.EtsFileDto +import org.jacodb.ets.dto.convertToEtsFile +import org.jacodb.ets.model.EtsFile +import org.jacodb.ets.model.EtsScene +import org.jacodb.ets.utils.loadEtsFileAutoConvert +import org.junit.jupiter.api.Disabled +import org.junit.jupiter.api.Test +import org.junit.jupiter.api.TestFactory +import org.junit.jupiter.api.condition.EnabledIf +import org.usvm.dataflow.ts.getResourcePath +import org.usvm.dataflow.ts.getResourcePathOrNull +import org.usvm.dataflow.ts.infer.AccessPathBase +import org.usvm.dataflow.ts.infer.EtsTypeFact +import org.usvm.dataflow.ts.infer.TypeInferenceManager +import org.usvm.dataflow.ts.infer.TypeInferenceResult +import org.usvm.dataflow.ts.infer.annotation.EtsTypeAnnotator +import org.usvm.dataflow.ts.infer.createApplicationGraph +import org.usvm.dataflow.ts.infer.dto.toType +import org.usvm.dataflow.ts.loadEtsProjectFromResources +import org.usvm.dataflow.ts.testFactory +import org.usvm.dataflow.ts.util.EtsTraits +import java.io.File +import kotlin.io.path.div +import kotlin.io.path.exists +import kotlin.io.path.toPath +import kotlin.test.assertContains +import kotlin.test.assertEquals +import kotlin.test.assertIs +import kotlin.test.assertNotEquals +import kotlin.test.assertTrue + +private val logger = KotlinLogging.logger {} + +class EtsTypeInferenceTest { + + companion object { + private fun load(path: String): EtsFile { + return loadEtsFileAutoConvert(getResourcePath(path), useArkAnalyzerTypeInference = null) + } + } + + @Test + fun `type inference for microphone`() { + val name = "microphone" + val file = load("/ts/$name.ts") + val project = EtsScene(listOf(file)) + val graph = createApplicationGraph(project) + + val entrypoints = project.projectClasses + .flatMap { it.methods } + .filter { it.name.startsWith("entrypoint") } + println("entrypoints: (${entrypoints.size})") + entrypoints.forEach { + println(" ${it.signature.enclosingClass.name}::${it.name}") + } + + val manager = TypeInferenceManager(EtsTraits(), graph) + val result = manager.analyze(entrypoints, doAddKnownTypes = false) + val types = result.inferredTypes + + run { + val m = types.keys.first { it.name == "getMicrophoneUuid" } + + // arg0 = 'devices' + val devices = types[m]!![AccessPathBase.Arg(0)]!! + assertIs(devices) + + val devicesCls = devices.cls + assertEquals("VirtualDevices", devicesCls?.typeName) + + assertContains(devices.properties, "microphone") + val microphone = devices.properties["microphone"]!! + assertIs(microphone) + + assertContains(microphone.properties, "uuid") + val uuid = microphone.properties["uuid"]!! + assertIs(uuid) + } + } + + @Test + fun `type inference for types`() { + val name = "types" + val file = load("/ts/$name.ts") + val project = EtsScene(listOf(file)) + val graph = createApplicationGraph(project) + + val entrypoints = project.projectClasses + .flatMap { it.methods } + .filter { it.name.startsWith("entrypoint") } + println("entrypoints: (${entrypoints.size})") + entrypoints.forEach { + println(" ${it.signature.enclosingClass.name}::${it.name}") + } + + val manager = TypeInferenceManager(EtsTraits(), graph) + manager.analyze(entrypoints) + } + + @Test + fun `type inference for data`() { + val name = "data" + val file = load("/ts/$name.ts") + val project = EtsScene(listOf(file)) + val graph = createApplicationGraph(project) + + val entrypoints = project.projectClasses + .flatMap { it.methods } + .filter { it.name.startsWith("entrypoint") } + println("entrypoints: (${entrypoints.size})") + entrypoints.forEach { + println(" ${it.signature.enclosingClass.name}::${it.name}") + } + + val manager = TypeInferenceManager(EtsTraits(), graph) + manager.analyze(entrypoints) + } + + @Test + fun `type inference for call`() { + val name = "call" + val file = load("/ts/$name.ts") + val project = EtsScene(listOf(file)) + val graph = createApplicationGraph(project) + + val entrypoints = project.projectClasses + .flatMap { it.methods } + .filter { it.name.startsWith("entrypoint") } + println("entrypoints: (${entrypoints.size})") + entrypoints.forEach { + println(" ${it.signature.enclosingClass.name}::${it.name}") + } + + val manager = TypeInferenceManager(EtsTraits(), graph) + manager.analyze(entrypoints) + } + + @Test + fun `type inference for nested_init`() { + val name = "nested_init" + val file = load("/ts/$name.ts") + val project = EtsScene(listOf(file)) + val graph = createApplicationGraph(project) + + val entrypoints = project.projectClasses + .flatMap { it.methods } + .filter { it.name.startsWith("entrypoint") } + println("entrypoints: (${entrypoints.size})") + entrypoints.forEach { + println(" ${it.signature.enclosingClass.name}::${it.name}") + } + + val manager = TypeInferenceManager(EtsTraits(), graph) + manager.analyze(entrypoints) + } + + private fun resourceAvailable(dirName: String): Boolean = + object {}::class.java.getResource(dirName) != null + + private fun testHapSet(setPath: String) { + val abcDir = object {}::class.java.getResource(setPath)?.toURI()?.toPath() + ?: error("Resource not found: $setPath") + val haps = abcDir.toFile().listFiles()?.toList() ?: emptyList() + processAllHAPs(haps) + } + + private val TEST_PROJECTS_PATH = "/projects/abcir/" + private fun testProjectsAvailable() = resourceAvailable(TEST_PROJECTS_PATH) + + @Test + @EnabledIf("testProjectsAvailable") + fun `type inference for test projects`() = testHapSet(TEST_PROJECTS_PATH) + + @Test + @Disabled("No project found") + fun `test single HAP`() { + val abcDirName = "/TestProjects/CertificateManager_240801_843398b" + val projectDir = object {}::class.java.getResource(abcDirName)?.toURI()?.toPath() + ?: error("Resource not found: $abcDirName") + val (scene, result) = testHap(projectDir.toString()) + val scene2 = EtsTypeAnnotator(scene, result).annotateWithTypes(scene) + } + + private fun processAllHAPs(haps: Collection) { + val succeed = mutableListOf() + val timeout = mutableListOf() + val failed = mutableListOf() + + haps.forEach { project -> + try { + runBlocking { + withTimeout(60_000) { + runInterruptible { + testHap(project.path) + } + } + } + succeed += project.name + println("$project - SUCCESS") + } catch (_: TimeoutCancellationException) { + timeout += project.name + println("$project - TIMEOUT") + } catch (e: SerializationException) { + e.printStackTrace() + error("Serialization exception") + } catch (e: Throwable) { + failed += project.name + println("$project - FAILED") + e.printStackTrace() + } + } + + println("Total: ${haps.size} HAPs") + println("Succeed: ${succeed.size}") + println("Timeout: ${timeout.size}") + println("Failed: ${failed.size}") + + println("PASSED") + succeed.forEach { + println(it) + } + println("TIMEOUT") + timeout.forEach { + println(it) + } + println("FAILED") + failed.forEach { + println(it) + } + } + + private fun testHap(projectDir: String): Pair { + val dir = File(projectDir).takeIf { it.isDirectory } ?: error("Not found project dir $projectDir") + println("Found project dir: '$dir'") + + val files = dir + .walk() + .filter { it.extension == "json" } + .toList() + println("Found files: (${files.size})") + for (path in files) { + println(" ${path.relativeTo(dir)}") + } + + println("Processing ${files.size} files...") + val etsFiles = files.map { convertToEtsFile(EtsFileDto.loadFromJson(it.inputStream())) } + val project = EtsScene(etsFiles) + val graph = createApplicationGraph(project) + + val entrypoints = project.projectClasses + .flatMap { it.methods + it.ctor } + .filter { it.isPublic || it.name == CONSTRUCTOR_NAME } + .filter { !it.enclosingClass.name.startsWith("AnonymousClass") } + println("entrypoints: (${entrypoints.size})") + entrypoints.forEach { + println(" ${it.signature.enclosingClass.name}::${it.name}") + } + + val manager = TypeInferenceManager(EtsTraits(), graph) + val result = manager.analyze(entrypoints) + return Pair(project, result) + } + + @Test + fun `test if guesser does anything`() { + val name = "testcases" + val file = load("/ts/$name.ts") + val project = EtsScene(listOf(file)) + val graph = createApplicationGraph(project) + + val entrypoints = project.projectClasses + .flatMap { it.methods } + .filter { it.name == "entrypoint" } + println("entrypoints: (${entrypoints.size})") + entrypoints.forEach { + println(" ${it.signature.enclosingClass.name}::${it.name}") + } + + val manager = TypeInferenceManager(EtsTraits(), graph) + val resultWithoutGuessed = manager.analyze(entrypoints) + val resultWithGuessed = resultWithoutGuessed.withGuessedTypes(project) + + assertNotEquals(resultWithoutGuessed.inferredTypes, resultWithGuessed.inferredTypes) + + println("=".repeat(42)) + println("Inferred types WITHOUT guesser: ") + for ((method, types) in resultWithoutGuessed.inferredTypes) { + println(method.enclosingClass.name to types) + } + + println("=".repeat(42)) + println("Inferred types with guesser: ") + for ((method, types) in resultWithGuessed.inferredTypes) { + println(method.enclosingClass.name to types) + } + } + + // TODO: support these complex tests + private val disabledTests = setOf( + "CaseAssignFieldToSelf", + "CaseLoop", + "CaseNew", + "CaseRecursion", + ) + + @TestFactory + fun `type inference on testcases`() = testFactory { + val file = load("/ts/testcases.ts") + val project = EtsScene(listOf(file)) + val graph = createApplicationGraph(project) + + val allCases = project.projectClasses.filter { it.name.startsWith("Case") } + + for (cls in allCases) { + if (cls.name in disabledTests) continue + test(name = cls.name) { + logger.info { "Analyzing testcase: ${cls.name}" } + + val inferMethod = cls.methods.single { it.name == "infer" } + logger.info { "Found infer: ${inferMethod.signature}" } + + val expectedTypeString = mutableMapOf() + var expectedReturnTypeString = "" + for (inst in inferMethod.cfg.stmts) { + if (inst is EtsAssignStmt) { + val lhv = inst.lhv + if (lhv is EtsLocal) { + val rhv = inst.rhv + if (lhv.name.startsWith("EXPECTED_ARG_")) { + check(rhv is EtsStringConstant) + val arg = lhv.name.removePrefix("EXPECTED_ARG_").toInt() + val pos = AccessPathBase.Arg(arg) + expectedTypeString[pos] = rhv.value + logger.info { "Expected type for $pos: ${rhv.value}" } + } else if (lhv.name == "EXPECTED_RETURN") { + check(rhv is EtsStringConstant) + expectedReturnTypeString = rhv.value + logger.info { "Expected return type: ${rhv.value}" } + } else if (lhv.name.startsWith("EXPECTED")) { + logger.error { "Skipping unexpected local: $lhv" } + } + } + } + } + + val entrypoint = cls.methods.single { it.name == "entrypoint" } + logger.info { "Found entrypoint: ${entrypoint.signature}" } + + val manager = TypeInferenceManager(EtsTraits(), graph) + val result = manager.analyze(listOf(entrypoint), doAddKnownTypes = false) + + val inferredTypes = result.inferredTypes[inferMethod] + ?: error("No inferred types for method ${inferMethod.enclosingClass.name}::${inferMethod.name}") + + for (position in expectedTypeString.keys.sortedBy { + when (it) { + is AccessPathBase.This -> -1 + is AccessPathBase.Arg -> it.index + else -> 1_000_000 + } + }) { + val expected = expectedTypeString[position]!! + val inferred = inferredTypes[position] + logger.info { "Inferred type for $position: $inferred" } + val passed = inferred.toString() == expected + assertTrue( + passed, + "Inferred type for $position does not match: inferred = $inferred, expected = $expected" + ) + } + if (expectedReturnTypeString.isNotBlank()) { + val expected = expectedReturnTypeString + val inferred = result.inferredReturnType[inferMethod] + logger.info { "Inferred return type: $inferred" } + val passed = inferred.toString() == expected + assertTrue( + passed, + "Inferred return type does not match: inferred = $inferred, expected = $expected" + ) + } + } + } + } + + @TestFactory + fun `test type inference on projects`() = testFactory { + val p = getResourcePathOrNull("/projects") ?: run { + logger.warn { "No projects directory found in resources" } + return@testFactory + } + val availableProjectNames = p.toFile().listFiles { f -> f.isDirectory }!!.map { it.name }.sorted() + logger.info { + buildString { + appendLine("Found projects: ${availableProjectNames.size}") + for (name in availableProjectNames) { + appendLine(" - $name") + } + } + } + if (availableProjectNames.isEmpty()) { + logger.warn { "No projects found" } + return@testFactory + } + for (projectName in availableProjectNames) { + // if (projectName != "Launcher") continue + // if (projectName != "Demo_Calc") continue + test("infer types in $projectName") { + logger.info { "Loading project: $projectName" } + val projectPath = getResourcePath("/projects/$projectName") + val etsirPath = projectPath / "etsir" + if (!etsirPath.exists()) { + logger.warn { "No etsir directory found for project $projectName" } + return@test + } + val modules = etsirPath.toFile().listFiles { f -> f.isDirectory }!!.map { it.name } + logger.info { "Found ${modules.size} modules: $modules" } + if (modules.isEmpty()) { + logger.warn { "No modules found for project $projectName" } + return@test + } + val project = loadEtsProjectFromResources(modules, "/projects/$projectName/etsir") + logger.info { + buildString { + appendLine("Loaded project with ${project.projectAndSdkClasses.size} classes and ${project.projectClasses.sumOf { it.methods.size }} methods") + for (cls in project.projectAndSdkClasses.sortedBy { it.name }) { + appendLine("= ${cls.signature} with ${cls.methods.size} methods:") + for (method in cls.methods.sortedBy { it.name }) { + appendLine(" - ${method.signature}") + } + } + } + } + val graph = createApplicationGraph(project) + + val entrypoints = project.projectAndSdkClasses + .flatMap { it.methods } + .filter { it.isPublic } + logger.info { "Found ${entrypoints.size} entrypoints" } + + val manager = TypeInferenceManager(EtsTraits(), graph) + val result = manager.analyze(entrypoints) + + logger.info { + buildString { + appendLine("Inferred types: ${result.inferredTypes.size}") + for ((method, types) in result.inferredTypes.entries.sortedBy { "${it.key.enclosingClass.name}::${it.key.name}" }) { + appendLine() + appendLine("- $method") + for ((pos, type) in types.entries.sortedBy { + when (val base = it.key) { + is AccessPathBase.This -> -1 + is AccessPathBase.Arg -> base.index + else -> 1_000_000 + } + }) { + appendLine("$pos: $type") + } + } + } + } + logger.info { + buildString { + appendLine("Inferred return types: ${result.inferredReturnType.size}") + for ((method, returnType) in result.inferredReturnType.entries.sortedBy { it.key.toString() }) { + appendLine("${method.enclosingClass.name}::${method.name}: $returnType") + } + } + } + logger.info { + buildString { + appendLine("Inferred combined this types: ${result.inferredCombinedThisType.size}") + for ((clazz, thisType) in result.inferredCombinedThisType.entries.sortedBy { it.key.toString() }) { + appendLine("${clazz.name} in ${clazz.file}: $thisType") + } + } + } + + var totalNumMatchedNormal = 0 + var totalNumMatchedUnknown = 0 + var totalNumMismatchedNormal = 0 + var totalNumLostNormal = 0 + var totalNumBetterThanUnknown = 0 + + for ((method, inferredTypes) in result.inferredTypes) { + var numMatchedNormal = 0 + var numMatchedUnknown = 0 + var numMismatchedNormal = 0 + var numLostNormal = 0 + var numBetterThanUnknown = 0 + + for (local in method.locals) { + val inferredType = inferredTypes[AccessPathBase.Local(local.name)]?.toType() + + logger.info { + "Local ${local.name} in ${method.enclosingClass.name}::${method.name}, known type: ${local.type}, inferred type: $inferredType" + } + + if (inferredType != null) { + if (local.type.isUnknown()) { + if (inferredType.isUnknown()) { + logger.info { "Matched unknown" } + numMatchedUnknown++ + } else { + logger.info { "Better than unknown" } + numBetterThanUnknown++ + } + } else { + if (inferredType == local.type) { + logger.info { "Matched normal" } + numMatchedNormal++ + } else { + logger.info { "Mismatched normal" } + numMismatchedNormal++ + } + } + } else { + if (local.type.isUnknown()) { + logger.info { "Matched (lost) unknown" } + numMatchedUnknown++ + } else { + logger.info { "Lost normal" } + numLostNormal++ + } + } + } + + logger.info { + buildString { + appendLine("Local type matching for ${method.enclosingClass.name}::${method.name}:") + appendLine(" Matched normal: $numMatchedNormal") + appendLine(" Matched unknown: $numMatchedUnknown") + appendLine(" Mismatched normal: $numMismatchedNormal") + appendLine(" Lost normal: $numLostNormal") + appendLine(" Better than unknown: $numBetterThanUnknown") + } + } + totalNumMatchedNormal += numMatchedNormal + totalNumMatchedUnknown += numMatchedUnknown + totalNumMismatchedNormal += numMismatchedNormal + totalNumLostNormal += numLostNormal + totalNumBetterThanUnknown += numBetterThanUnknown + } + + logger.info { + buildString { + appendLine("Total local type matching statistics:") + appendLine(" Matched normal: $totalNumMatchedNormal") + appendLine(" Matched unknown: $totalNumMatchedUnknown") + appendLine(" Mismatched normal: $totalNumMismatchedNormal") + appendLine(" Lost normal: $totalNumLostNormal") + appendLine(" Better than unknown: $totalNumBetterThanUnknown") + } + } + + logger.info { "Done analyzing project: $projectName" } + } + } + } +} + +private fun EtsType.isUnknown(): Boolean = + this == EtsUnknownType || this == EtsAnyType diff --git a/usvm-dataflow-ts/src/test/kotlin/org/usvm/dataflow/ts/test/EtsTypeResolverAbcTest.kt b/usvm-dataflow-ts/src/test/kotlin/org/usvm/dataflow/ts/test/EtsTypeResolverAbcTest.kt new file mode 100644 index 0000000000..41dd572fe3 --- /dev/null +++ b/usvm-dataflow-ts/src/test/kotlin/org/usvm/dataflow/ts/test/EtsTypeResolverAbcTest.kt @@ -0,0 +1,93 @@ +package org.usvm.dataflow.ts.test + +import org.jacodb.ets.utils.loadEtsProjectFromIR +import org.junit.jupiter.api.Test +import org.junit.jupiter.api.condition.EnabledIf +import org.usvm.dataflow.ts.infer.EntryPointsProcessor +import org.usvm.dataflow.ts.infer.TypeInferenceManager +import org.usvm.dataflow.ts.infer.createApplicationGraph +import org.usvm.dataflow.ts.util.EtsTraits +import org.usvm.dataflow.ts.util.MethodTypesFacts +import org.usvm.dataflow.ts.util.TypeInferenceStatistics +import kotlin.io.path.Path +import kotlin.io.path.exists + +@EnabledIf("projectAvailable") +class EtsTypeResolverAbcTest { + + companion object { + private val yourPrefixForTestFolders = "C:/work/TestProjects" + private val testProjectsVersion = "TestProjects_2024_12_5" + private val pathToSDK: String? = null // TODO: Put your path here + + @JvmStatic + private fun projectAvailable(): Boolean { + return Path(yourPrefixForTestFolders).exists() + } + } + + private fun runOnAbcProject(projectID: String, abcPath: String) { + val projectAbc = "$yourPrefixForTestFolders/$testProjectsVersion/$abcPath" + val abcScene = loadEtsProjectFromIR(Path(projectAbc), pathToSDK?.let { Path(it) }) + val graphAbc = createApplicationGraph(abcScene) + + val entrypoint = EntryPointsProcessor.extractEntryPoints(abcScene) + val allMethods = entrypoint.allMethods.filter { it.isPublic }.filter { it.cfg.stmts.isNotEmpty() } + + val manager = TypeInferenceManager(EtsTraits(), graphAbc) + val result = manager + .analyze(entrypoint.mainMethods, allMethods) + .withGuessedTypes(abcScene) + + val sceneStatistics = TypeInferenceStatistics() + entrypoint.allMethods + .filter { it.cfg.stmts.isNotEmpty() } + .forEach { + val methodTypeFacts = MethodTypesFacts.from(result, it) + sceneStatistics.compareSingleMethodFactsWithTypesInScene(methodTypeFacts, it, graphAbc) + } + sceneStatistics.dumpStatistics("${projectID}_scene_comparison.txt") + } + + @Test + fun testLoadAbcProject1() = runOnAbcProject( + projectID = "project1", + abcPath = "callui-default-signed", + ) + + @Test + fun testLoadAbcProject2() = runOnAbcProject( + projectID = "project2", + abcPath = "CertificateManager_240801_843398b", + ) + + @Test + fun testLoadAbcProject3() = runOnAbcProject( + projectID = "project3", + abcPath = "mobiledatasettings-callui-default-signed", + ) + + @Test + fun testLoadAbcProject4() = runOnAbcProject( + projectID = "project4", + abcPath = "Music_Demo_240727_98a3500", + ) + + @Test + fun testLoadAbcProject5() = runOnAbcProject( + projectID = "project5", + abcPath = "phone_photos-default-signed_20240905_151755", + ) + + @Test + fun testLoadAbcProject6() = runOnAbcProject( + projectID = "project6", + abcPath = "phone-default-signed_20240409_144519", + ) + + @Test + fun testLoadAbcProject7() = runOnAbcProject( + projectID = "project7", + abcPath = "SecurityPrivacyCenter_240801_843998b", + ) +} diff --git a/usvm-dataflow-ts/src/test/kotlin/org/usvm/dataflow/ts/test/EtsTypeResolverWithAstTest.kt b/usvm-dataflow-ts/src/test/kotlin/org/usvm/dataflow/ts/test/EtsTypeResolverWithAstTest.kt new file mode 100644 index 0000000000..74b4c87a76 --- /dev/null +++ b/usvm-dataflow-ts/src/test/kotlin/org/usvm/dataflow/ts/test/EtsTypeResolverWithAstTest.kt @@ -0,0 +1,366 @@ +package org.usvm.dataflow.ts.test + +import org.jacodb.ets.model.EtsFile +import org.jacodb.ets.model.EtsMethod +import org.jacodb.ets.model.EtsScene +import org.jacodb.ets.utils.loadEtsFileAutoConvert +import org.jacodb.ets.utils.loadEtsProjectAutoConvert +import org.jacodb.ets.utils.loadEtsProjectFromIR +import org.junit.jupiter.api.Assertions.assertFalse +import org.junit.jupiter.api.Test +import org.junit.jupiter.api.condition.EnabledIf +import org.usvm.dataflow.ts.graph.EtsApplicationGraph +import org.usvm.dataflow.ts.infer.AccessPathBase +import org.usvm.dataflow.ts.infer.EntryPointsProcessor +import org.usvm.dataflow.ts.infer.EtsApplicationGraphWithExplicitEntryPoint +import org.usvm.dataflow.ts.infer.EtsTypeFact +import org.usvm.dataflow.ts.infer.TypeInferenceManager +import org.usvm.dataflow.ts.infer.TypeInferenceResult +import org.usvm.dataflow.ts.infer.createApplicationGraph +import org.usvm.dataflow.ts.test.utils.ClassMatcherStatistics +import org.usvm.dataflow.ts.test.utils.ExpectedTypesExtractor +import org.usvm.dataflow.ts.util.EtsTraits +import org.usvm.dataflow.ts.util.MethodTypesFacts +import org.usvm.dataflow.ts.util.TypeInferenceStatistics +import java.nio.file.Paths +import kotlin.io.path.Path +import kotlin.io.path.exists +import kotlin.test.assertTrue + +@EnabledIf("projectAvailable") +class EtsTypeResolverWithAstTest { + companion object { + private val yourPrefixForTestFolders = "C:/work/TestProjects" + private val testProjectsVersion = "TestProjects_2024_12_5" + private val pathToSDK: String? = null // TODO: Put your path here + + @JvmStatic + private fun projectAvailable(): Boolean { + return Path(yourPrefixForTestFolders).exists() + } + + private fun load(name: String): EtsFile { + return loadEtsFileAutoConvert(Paths.get("/ts/$name.ts")) + } + } + + @Test + fun testTestHap() { + val projectAbc = "$yourPrefixForTestFolders/$testProjectsVersion/CallUI" + val abcScene = loadEtsProjectFromIR(Path(projectAbc), pathToSDK?.let { Path(it) }) + val graphAbc = createApplicationGraph(abcScene) + + val entrypoint = EntryPointsProcessor.extractEntryPoints(abcScene) // TODO fix error with abc and ast methods + + val manager = TypeInferenceManager(EtsTraits(), graphAbc) + val resultBasic = manager.analyze( + entrypoints = entrypoint.mainMethods, + allMethods = entrypoint.allMethods, + ) + val result = resultBasic.withGuessedTypes(abcScene) + + val classMatcherStatistics = ClassMatcherStatistics() + + // TODO fix error with abc and ast methods + saveTypeInferenceComparison( + entrypoint.allMethods, + entrypoint.allMethods, + graphAbc, + graphAbc, + result, + classMatcherStatistics, + abcScene, + ) + classMatcherStatistics.dumpStatistics("callkit.txt") + } + + fun runOnProjectWithAstComparison(projectID: String, abcPath: String, astPath: String) { + val projectAbc = "$yourPrefixForTestFolders/$testProjectsVersion/$abcPath" + val abcScene = loadEtsProjectFromIR(Path(projectAbc), pathToSDK?.let { Path(it) }) + + val projectAst = "$yourPrefixForTestFolders/AST/$astPath" + val astScene = loadEtsProjectAutoConvert(Paths.get(projectAst)) + + val graphAbc = createApplicationGraph(abcScene) as EtsApplicationGraphWithExplicitEntryPoint + val graphAst = createApplicationGraph(astScene) as EtsApplicationGraphWithExplicitEntryPoint + + val entrypoint = EntryPointsProcessor.extractEntryPoints(abcScene) + val astMethods = extractAllAstMethods(astScene, abcScene) + + println(entrypoint.mainMethods.map { it.signature }) + println(entrypoint.allMethods.map { it.signature }) + + val manager = TypeInferenceManager(EtsTraits(), graphAbc) + val resultBasic = manager.analyze( + entrypoints = entrypoint.mainMethods, + allMethods = entrypoint.allMethods.filter { it.isPublic }, + ) + val result = resultBasic.withGuessedTypes(abcScene) + + val classMatcherStatistics = ClassMatcherStatistics() + saveTypeInferenceComparison( + astMethods, + entrypoint.allMethods, + graphAst, + graphAbc, + result, + classMatcherStatistics, + abcScene + ) + classMatcherStatistics.dumpStatistics("$projectID.txt") + + val sceneStatistics = TypeInferenceStatistics() + abcScene.projectAndSdkClasses + .flatMap { it.methods } + .forEach { + val facts = MethodTypesFacts.from(result, it) + sceneStatistics.compareSingleMethodFactsWithTypesInScene(facts, it, graphAbc) + } + sceneStatistics.dumpStatistics("${projectID}_scene_comparison.txt") + } + + @Test + fun testLoadProject1() = runOnProjectWithAstComparison( + projectID = "project1", + abcPath = "callui-default-signed", + astPath = "16_CallUI/applications_call_230923_4de8" + ) + + @Test + fun testLoadProject2() = runOnProjectWithAstComparison( + projectID = "project2", + abcPath = "CertificateManager_240801_843398b", + astPath = "13_SecurityPrivacyCenter/security_privacy_center" + ) + + @Test + fun testLoadProject3() = runOnProjectWithAstComparison( + projectID = "project3", + abcPath = "mobiledatasettings-callui-default-signed", + astPath = "16_CallUI/applications_call_230923_4de8" + ) + + @Test + fun testLoadProject4() = runOnProjectWithAstComparison( + projectID = "project4", + abcPath = "Music_Demo_240727_98a3500", + astPath = "16_CallUI/applications_call_230923_4de8" + ) + + @Test + fun testLoadProject5() = runOnProjectWithAstComparison( + projectID = "project5", + abcPath = "phone_photos-default-signed_20240905_151755", + astPath = "15_Photos/applications_photos_240905_ea8d1" + ) + + @Test + fun testLoadProject6() = runOnProjectWithAstComparison( + projectID = "project6", + abcPath = "phone-default-signed_20240409_144519", + astPath = "17_Camera/applications_camera_240409_1da805f8" + ) + + @Test + fun testLoadProject7() = runOnProjectWithAstComparison( + projectID = "project7", + abcPath = "SecurityPrivacyCenter_240801_843998b", + astPath = "13_SecurityPrivacyCenter/security_privacy_center" + ) + + @Test + fun `use non unique fields`() { + val file = load("resolver_test") + + val project = EtsScene(listOf(file)) + val graph = createApplicationGraph(project) + + val entrypoint = project.projectClasses + .flatMap { it.methods } + .single { it.name == "useNonUniqueField" } + + val manager = TypeInferenceManager(EtsTraits(), graph) + val resultBasic = manager.analyze(listOf(entrypoint)) + val result = resultBasic.withGuessedTypes(project) + + checkAnObjectTypeOfSingleArgument(result.inferredTypes[entrypoint]!!) { typeFact: EtsTypeFact.ObjectEtsTypeFact -> + typeFact.cls == null && typeFact.properties.keys.single() == "defaultA" + } + + val expectedTypes = ExpectedTypesExtractor(graph).extractTypes(entrypoint) + val actualTypes = MethodTypesFacts.from(result, entrypoint) + + assertFalse(expectedTypes.matchesWithTypeFacts(actualTypes, ignoreReturnType = true, project)) + } + + @Test + fun `use unique fields`() { + val file = load("resolver_test") + + val project = EtsScene(listOf(file)) + val graph = createApplicationGraph(project) + + val entrypoint = project.projectClasses + .flatMap { it.methods } + .single { it.name == "useUniqueFields" } + + val manager = TypeInferenceManager(EtsTraits(), graph) + val resultBasic = manager.analyze(listOf(entrypoint)) + val result = resultBasic.withGuessedTypes(project) + + checkAnObjectTypeOfSingleArgument(result.inferredTypes[entrypoint]!!) { fact: EtsTypeFact.ObjectEtsTypeFact -> + fact.cls?.typeName == "FieldContainerToInfer" && fact.properties.isEmpty() + } + + val expectedTypes = ExpectedTypesExtractor(graph).extractTypes(entrypoint) + val actualTypes = MethodTypesFacts.from(result, entrypoint) + + assertTrue(expectedTypes.matchesWithTypeFacts(actualTypes, ignoreReturnType = true, project)) + } + + @Test + fun `use unique and non unique fields`() { + val file = load("resolver_test") + + val project = EtsScene(listOf(file)) + val graph = createApplicationGraph(project) + + val entrypoint = project.projectClasses + .flatMap { it.methods } + .single { it.name == "useBothA" } + + val expectedTypes = ExpectedTypesExtractor(graph).extractTypes(entrypoint) + + val manager = TypeInferenceManager(EtsTraits(), graph) + val resultBasic = manager.analyze(listOf(entrypoint)) + val result = resultBasic.withGuessedTypes(project) + + checkAnObjectTypeOfSingleArgument(result.inferredTypes[entrypoint]!!) { fact: EtsTypeFact.ObjectEtsTypeFact -> + fact.cls?.typeName == "FieldContainerToInfer" && fact.properties.isEmpty() + } + + val actualTypes = MethodTypesFacts.from(result, entrypoint) + + assertTrue(expectedTypes.matchesWithTypeFacts(actualTypes, ignoreReturnType = true, project)) + } + + @Test + fun `use unique methods`() { + val file = load("resolver_test") + + val project = EtsScene(listOf(file)) + val graph = createApplicationGraph(project) + + val entrypoint = project.projectClasses + .flatMap { it.methods } + .single { it.name == "useUniqueMethods" } + + val manager = TypeInferenceManager(EtsTraits(), graph) + val resultBasic = manager.analyze(listOf(entrypoint)) + val result = resultBasic.withGuessedTypes(project) + + checkAnObjectTypeOfSingleArgument(result.inferredTypes[entrypoint]!!) { typeFact: EtsTypeFact.ObjectEtsTypeFact -> + typeFact.cls?.typeName == "MethodsContainerToInfer" && typeFact.properties.isEmpty() + } + + val expectedTypes = ExpectedTypesExtractor(graph).extractTypes(entrypoint) + val actualTypes = MethodTypesFacts.from(result, entrypoint) + + assertTrue(expectedTypes.matchesWithTypeFacts(actualTypes, ignoreReturnType = true, project)) + } + + @Test + fun `use non unique methods`() { + val file = load("resolver_test") + + val project = EtsScene(listOf(file)) + val graph = createApplicationGraph(project) + + val entrypoint = project.projectClasses + .flatMap { it.methods } + .single { it.name == "useNotUniqueMethod" } + + val manager = TypeInferenceManager(EtsTraits(), graph) + val resultBasic = manager.analyze(listOf(entrypoint)) + val result = resultBasic.withGuessedTypes(project) + + checkAnObjectTypeOfSingleArgument(result.inferredTypes[entrypoint]!!) { typeFact: EtsTypeFact.ObjectEtsTypeFact -> + typeFact.cls == null && typeFact.properties.keys.single() == "notUniqueFunction" + } + + val expectedTypes = ExpectedTypesExtractor(graph).extractTypes(entrypoint) + val actualTypes = MethodTypesFacts.from(result, entrypoint) + + assertFalse(expectedTypes.matchesWithTypeFacts(actualTypes, ignoreReturnType = true, project)) + } + + @Test + fun `use function and field`() { + val file = load("resolver_test") + + val project = EtsScene(listOf(file)) + val graph = createApplicationGraph(project) + + val entrypoint = project.projectClasses + .flatMap { it.methods } + .single { it.name == "useFunctionAndField" } + + val manager = TypeInferenceManager(EtsTraits(), graph) + val resultBasic = manager.analyze(listOf(entrypoint)) + val result = resultBasic.withGuessedTypes(project) + + checkAnObjectTypeOfSingleArgument(result.inferredTypes[entrypoint]!!) { typeFact: EtsTypeFact.ObjectEtsTypeFact -> + typeFact.cls?.typeName == "FieldContainerToInfer" && typeFact.properties.isEmpty() + } + + val expectedTypes = ExpectedTypesExtractor(graph).extractTypes(entrypoint) + val actualTypes = MethodTypesFacts.from(result, entrypoint) + + assertTrue(expectedTypes.matchesWithTypeFacts(actualTypes, ignoreReturnType = true, project)) + } + + private fun extractAllAstMethods(astScene: EtsScene, abcScene: EtsScene): List { + val abcMethods = abcScene.projectAndSdkClasses + .flatMapTo(hashSetOf()) { + it.methods.map { method -> method.name } + } + return astScene.projectAndSdkClasses + .flatMap { it.methods } + .filter { it.name in abcMethods } + } + + private fun saveTypeInferenceComparison( + astMethods: List, + abcMethods: List, + graphAst: EtsApplicationGraph, + graphAbc: EtsApplicationGraph, + result: TypeInferenceResult, + classMatcherStatistics: ClassMatcherStatistics, + abcScene: EtsScene, + ) { + astMethods.forEach { m -> + val expectedTypes = ExpectedTypesExtractor(graphAst).extractTypes(m) + val abcMethod = abcMethods.singleOrNull { + it.name == m.name && it.enclosingClass.name == m.enclosingClass.name + } ?: return@forEach + val actualTypes = MethodTypesFacts.from(result, m) + classMatcherStatistics.calculateStats( + actualTypes, + expectedTypes, + abcScene, + m, + abcMethod, + graphAst, + graphAbc + ) + } + } + + private inline fun checkAnObjectTypeOfSingleArgument( + types: Map, + predicate: (T) -> Boolean, + ) { + val type = types.filterKeys { it is AccessPathBase.Arg }.values.single() as T + assertTrue(predicate(type)) + } +} diff --git a/usvm-dataflow-ts/src/test/kotlin/org/usvm/dataflow/ts/test/utils/ExpectedTypesExtractor.kt b/usvm-dataflow-ts/src/test/kotlin/org/usvm/dataflow/ts/test/utils/ExpectedTypesExtractor.kt new file mode 100644 index 0000000000..82aa4fb2dc --- /dev/null +++ b/usvm-dataflow-ts/src/test/kotlin/org/usvm/dataflow/ts/test/utils/ExpectedTypesExtractor.kt @@ -0,0 +1,377 @@ +package org.usvm.dataflow.ts.test.utils + +import org.jacodb.ets.base.DEFAULT_ARK_CLASS_NAME +import org.jacodb.ets.base.EtsAnyType +import org.jacodb.ets.base.EtsArrayType +import org.jacodb.ets.base.EtsBooleanType +import org.jacodb.ets.base.EtsClassType +import org.jacodb.ets.base.EtsFunctionType +import org.jacodb.ets.base.EtsNullType +import org.jacodb.ets.base.EtsNumberType +import org.jacodb.ets.base.EtsStringType +import org.jacodb.ets.base.EtsType +import org.jacodb.ets.base.EtsUnclearRefType +import org.jacodb.ets.base.EtsUndefinedType +import org.jacodb.ets.base.EtsUnknownType +import org.jacodb.ets.model.EtsMethod +import org.jacodb.ets.model.EtsScene +import org.usvm.dataflow.ts.graph.EtsApplicationGraph +import org.usvm.dataflow.ts.infer.AccessPathBase +import org.usvm.dataflow.ts.infer.EtsTypeFact +import org.usvm.dataflow.ts.util.MethodTypesFacts +import java.io.File + +private val logger = mu.KotlinLogging.logger {} + +class ExpectedTypesExtractor(private val graph: EtsApplicationGraph) { + fun extractTypes(method: EtsMethod): MethodTypes { + val returnType = method.returnType + val argumentsTypes = method.parameters.map { it.type } + val thisType = getEtsClassType(method, graph) + + return MethodTypes(thisType, argumentsTypes, returnType) + } + +} + +private fun getEtsClassType(method: EtsMethod, graph: EtsApplicationGraph) = + if (method.enclosingClass.name == DEFAULT_ARK_CLASS_NAME || method.enclosingClass.name.isBlank()) { + null + } else { + val clazz = graph.cp + .projectAndSdkClasses + // .filterNot { it.name.startsWith(ANONYMOUS_CLASS_PREFIX) } + // TODO different representation in abc and ast, replace with signatures + // .singleOrNull { it.name == method.enclosingClass.name } + // ?: error("TODO") + .firstOrNull { it.name == method.enclosingClass.name } + ?: error("") /*return MethodTypes(null, argumentsTypes, returnType)*/ + + EtsClassType(clazz.signature) + } + +class ClassMatcherStatistics { + private val overallTypes: Long + get() = overallThisTypes + overallArgsTypes + overallReturnTypes + private val matched: Long + get() = exactlyMatchedThisTypes + exactlyMatchedArgsTypes + exactlyMatchedReturnTypes + private val someFactsFound: Long + get() = someFactsAboutThisTypes + someFactsAboutArgsTypes + someFactsAboutReturnTypes + returnIsAnyType + argIsAnyType + + private var failedMethods: MutableList = mutableListOf() + + private var overallThisTypes: Long = 0L + private var exactlyMatchedThisTypes: Long = 0L + private var someFactsAboutThisTypes: Long = 0L + + private var overallArgsTypes: Long = 0L + private var exactlyMatchedArgsTypes: Long = 0L + private var someFactsAboutArgsTypes: Long = 0L + private var argIsAnyType: Long = 0L + + private var overallReturnTypes: Long = 0L + private var exactlyMatchedReturnTypes: Long = 0L + private var someFactsAboutReturnTypes: Long = 0L + private var returnIsAnyType: Long = 0L + + private val methodToTypes: MutableMap>> = + hashMapOf() + private val methodToReturnTypes: MutableMap> = hashMapOf() + + private fun EtsMethod.saveComparisonInfo( + position: AccessPathBase, + type: EtsType, + fact: EtsTypeFact?, + ) { + val methodTypes = methodToTypes.getOrPut(this) { hashMapOf() } + check(position !in methodTypes) + methodTypes[position] = type to fact + } + + fun calculateStats( + methodResults: MethodTypesFacts, + types: MethodTypes?, + scene: EtsScene, + astMethod: EtsMethod, + abcMethod: EtsMethod, + astGraph: EtsApplicationGraph, + abcGraph: EtsApplicationGraph, + ) { + methodResults.apply { + if (combinedThisFact == null && argumentsFacts.all { it == null } && returnFact == null && localFacts.isEmpty()) { + saveAbsentResult(astMethod) + return + } + } + + compareTypesWithExpected( + methodResults, + requireNotNull(types), + scene, + astMethod + ) + + } + + private fun compareTypesWithExpected( + facts: MethodTypesFacts, + types: MethodTypes, + scene: EtsScene, + method: EtsMethod, + ) { + // 'this' type + types.thisType?.let { + overallThisTypes++ + + method.saveComparisonInfo(AccessPathBase.This, it, facts.combinedThisFact) + + val thisFact = facts.combinedThisFact ?: return@let + if (thisFact.matchesWith(it, strictMode = false)) { + exactlyMatchedThisTypes++ + } else { + someFactsAboutThisTypes++ + } + } + + // args + types.argumentsTypes.forEachIndexed { index, type -> + overallArgsTypes++ + + val fact = facts.argumentsFacts.getOrNull(index) + + + method.saveComparisonInfo(AccessPathBase.Arg(index), type, fact) + + if (fact is EtsTypeFact.AnyEtsTypeFact) { + argIsAnyType++ + return@forEachIndexed + } + + if (fact == null) return@forEachIndexed + + if (fact.matchesWith(type, strictMode = false)) { + exactlyMatchedArgsTypes++ + } else { + someFactsAboutArgsTypes++ + } + } + + // return type + val inferredReturnType = facts.returnFact ?: EtsTypeFact.AnyEtsTypeFact + + overallReturnTypes++ + methodToReturnTypes[method] = method.returnType to inferredReturnType + + if (inferredReturnType is EtsTypeFact.AnyEtsTypeFact) { + returnIsAnyType++ + return + } + + if (inferredReturnType.matchesWith(types.returnType, strictMode = false)) { + exactlyMatchedReturnTypes++ + } else if (inferredReturnType.partialMatchedBy(types.returnType)) { + someFactsAboutReturnTypes++ + } + } + + fun saveAbsentResult(method: EtsMethod) { + failedMethods += method + } + + override fun toString(): String = """ + Total types number: $overallTypes + Exactly matched: $matched + Partially matched: $someFactsFound + Not found: ${overallTypes - matched - someFactsFound} + + Specifically: + + This types total: $overallThisTypes + Exactly matched this types: $exactlyMatchedThisTypes + Partially matched this types: $someFactsAboutThisTypes + Not found: ${overallThisTypes - exactlyMatchedThisTypes - someFactsAboutThisTypes} + + Args types total: $overallArgsTypes + Exactly matched args types: $exactlyMatchedArgsTypes + Partially matched args types: $someFactsAboutArgsTypes + Any type as arg: $argIsAnyType + Not found: ${overallArgsTypes - exactlyMatchedArgsTypes - someFactsAboutArgsTypes - argIsAnyType} + + Return types total: $overallReturnTypes + Exactly matched return types: $exactlyMatchedReturnTypes + Partially matched return types: $someFactsAboutReturnTypes + Any type is returned: $returnIsAnyType + Not found: ${overallReturnTypes - exactlyMatchedReturnTypes - someFactsAboutReturnTypes - returnIsAnyType} + + Didn't find any types for ${failedMethods.size} methods + """.trimIndent() + + fun dumpStatistics(outputFilePath: String? = null) { + val data = buildString { + appendLine(this@ClassMatcherStatistics.toString()) + appendLine() + + appendLine("Specifically: ${"=".repeat(42)}") + + val comparator = + Comparator>> { fst, snd -> + when (fst.key) { + is AccessPathBase.This -> when { + snd.key is AccessPathBase.This -> 0 + else -> -1 + } + + is AccessPathBase.Arg -> when (snd.key) { + is AccessPathBase.This -> 1 + is AccessPathBase.Arg -> { + (fst.key as AccessPathBase.Arg).index.compareTo((snd.key as AccessPathBase.Arg).index) + } + + else -> -1 + } + + else -> when (snd.key) { + is AccessPathBase.This, is AccessPathBase.Arg -> 1 + else -> 0 + } + } + } + + methodToTypes.forEach { (method, types) -> + appendLine("${method.signature}:") + + types + .entries + .sortedWith(comparator) + .forEach { (path, typeInfo) -> + appendLine("${path}: ${typeInfo.first} -> ${typeInfo.second}") + } + appendLine() + } + + appendLine() + appendLine("=".repeat(42)) + appendLine("Failed methods:") + failedMethods.forEach { + appendLine(it) + } + } + + if (outputFilePath == null) { + println(data) + return + } + + val file = File(outputFilePath) + println("File with statistics is located: ${file.absolutePath}") + file.writeText(data) + } +} + +data class MethodTypes( + val thisType: EtsType?, + val argumentsTypes: List, + val returnType: EtsType, +) { + fun matchesWithTypeFacts(other: MethodTypesFacts, ignoreReturnType: Boolean, scene: EtsScene): Boolean { + if (thisType == null && other.combinedThisFact != null) return false + + if (thisType != null && other.combinedThisFact != null) { + if (!other.combinedThisFact.matchesWith(thisType, strictMode = false)) return false + } + + for ((i, fact) in other.argumentsFacts.withIndex()) { + if (!fact.matchesWith(argumentsTypes[i], strictMode = false)) return false + } + + if (ignoreReturnType) return true + + return other.returnFact.matchesWith(returnType, strictMode = false) + } +} + +private fun EtsTypeFact?.matchesWith(type: EtsType, strictMode: Boolean): Boolean { + val result = when (this) { + null, EtsTypeFact.AnyEtsTypeFact -> { + // TODO any other combination? + type is EtsAnyType || (!strictMode && type is EtsUnknownType) + } + + is EtsTypeFact.ObjectEtsTypeFact -> { + // TODO it should be replaced with signatures + val typeName = this.cls?.typeName + + if ((type is EtsUnknownType || type is EtsAnyType) && !strictMode) { + this.cls != null + } else { + (type is EtsClassType || type is EtsUnclearRefType) && type.typeName == typeName + } + } + + is EtsTypeFact.ArrayEtsTypeFact -> when (type) { + is EtsArrayType -> this.elementType.matchesWith(type.elementType, strictMode) + + is EtsUnclearRefType -> { + val elementType = this.elementType as? EtsTypeFact.ObjectEtsTypeFact + elementType?.cls?.typeName == type.typeName + } + + else -> false + } + + EtsTypeFact.BooleanEtsTypeFact -> { + type is EtsBooleanType + || (type is EtsUnknownType && !strictMode) + || (type as? EtsClassType)?.typeName == "Boolean" + || (type as? EtsUnclearRefType)?.typeName == "Boolean" + } + + EtsTypeFact.FunctionEtsTypeFact -> type is EtsFunctionType || (type is EtsUnknownType && !strictMode) + EtsTypeFact.NullEtsTypeFact -> type is EtsNullType || (type is EtsUnknownType && !strictMode) + EtsTypeFact.NumberEtsTypeFact -> { + type is EtsNumberType + || (type is EtsUnknownType && !strictMode) + || (type as? EtsClassType)?.typeName == "Number" + || (type as? EtsUnclearRefType)?.typeName == "Number" + } + + EtsTypeFact.StringEtsTypeFact -> { + type is EtsStringType + || (type is EtsUnknownType && !strictMode) + || (type as? EtsClassType)?.typeName == "String" + || (type as? EtsUnclearRefType)?.typeName == "String" + } + + EtsTypeFact.UndefinedEtsTypeFact -> type is EtsUndefinedType + EtsTypeFact.UnknownEtsTypeFact -> type is EtsUnknownType + is EtsTypeFact.GuardedTypeFact -> TODO() + is EtsTypeFact.IntersectionEtsTypeFact -> { + // TODO intersections checks are not supported yet + false + } + + is EtsTypeFact.UnionEtsTypeFact -> if (strictMode) { + types.all { it.matchesWith(type, strictMode) } + } else { + types.any { it.matchesWith(type, strictMode) } + } + } + + if (!result) { + logger.warn { + """ + Fact: $this + Type: $type + + """.trimIndent() + } + } + + return result +} + +private fun EtsTypeFact.partialMatchedBy(type: EtsType): Boolean { + if (type is EtsUnknownType) return true + logger.warn { "Not implemented partial match for fact $this and type $type" } + return false +} diff --git a/usvm-dataflow-ts/src/test/kotlin/org/usvm/dataflow/ts/test/utils/TaintConfig.kt b/usvm-dataflow-ts/src/test/kotlin/org/usvm/dataflow/ts/test/utils/TaintConfig.kt new file mode 100644 index 0000000000..8402634d0f --- /dev/null +++ b/usvm-dataflow-ts/src/test/kotlin/org/usvm/dataflow/ts/test/utils/TaintConfig.kt @@ -0,0 +1,108 @@ +/* + * Copyright 2022 UnitTestBot contributors (utbot.org) + *

+ * 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. + */ + +package org.usvm.dataflow.ts.test.utils + +import kotlinx.serialization.json.Json +import kotlinx.serialization.modules.SerializersModule +import org.jacodb.api.common.CommonMethod +import org.jacodb.taint.configuration.NameExactMatcher +import org.jacodb.taint.configuration.NamePatternMatcher +import org.jacodb.taint.configuration.SerializedTaintCleaner +import org.jacodb.taint.configuration.SerializedTaintConfigurationItem +import org.jacodb.taint.configuration.SerializedTaintEntryPointSource +import org.jacodb.taint.configuration.SerializedTaintMethodSink +import org.jacodb.taint.configuration.SerializedTaintMethodSource +import org.jacodb.taint.configuration.SerializedTaintPassThrough +import org.jacodb.taint.configuration.TaintCleaner +import org.jacodb.taint.configuration.TaintConfigurationItem +import org.jacodb.taint.configuration.TaintEntryPointSource +import org.jacodb.taint.configuration.TaintMethodSink +import org.jacodb.taint.configuration.TaintMethodSource +import org.jacodb.taint.configuration.TaintPassThrough +import org.jacodb.taint.configuration.actionModule +import org.jacodb.taint.configuration.conditionModule +import org.usvm.dataflow.ts.getResourceStream + +private val json = Json { + classDiscriminator = "_" + serializersModule = SerializersModule { + include(conditionModule) + include(actionModule) + } +} + +fun loadRules(configFileName: String): List { + val configJson = getResourceStream("/$configFileName").bufferedReader().readText() + val rules: List = json.decodeFromString(configJson) + // println("Loaded ${rules.size} rules from '$configFileName'") + // for (rule in rules) { + // println(rule) + // } + return rules +} + +fun getConfigForMethod( + method: CommonMethod, + rules: List, +): List? { + val res = buildList { + for (item in rules) { + val matcher = item.methodInfo.functionName + if (matcher is NameExactMatcher) { + if (method.name == matcher.name) add(item.toItem(method)) + } else if (matcher is NamePatternMatcher) { + if (method.name.matches(matcher.pattern.toRegex())) add(item.toItem(method)) + } + } + } + return res.ifEmpty { null } +} + +fun SerializedTaintConfigurationItem.toItem(method: CommonMethod): TaintConfigurationItem { + return when (this) { + is SerializedTaintEntryPointSource -> TaintEntryPointSource( + method = method, + condition = condition, + actionsAfter = actionsAfter + ) + + is SerializedTaintMethodSource -> TaintMethodSource( + method = method, + condition = condition, + actionsAfter = actionsAfter + ) + + is SerializedTaintMethodSink -> TaintMethodSink( + method = method, + ruleNote = ruleNote, + cwe = cwe, + condition = condition + ) + + is SerializedTaintPassThrough -> TaintPassThrough( + method = method, + condition = condition, + actionsAfter = actionsAfter + ) + + is SerializedTaintCleaner -> TaintCleaner( + method = method, + condition = condition, + actionsAfter = actionsAfter + ) + } +} diff --git a/usvm-dataflow-ts/src/test/resources/logback.xml b/usvm-dataflow-ts/src/test/resources/logback.xml new file mode 100644 index 0000000000..30f8844725 --- /dev/null +++ b/usvm-dataflow-ts/src/test/resources/logback.xml @@ -0,0 +1,19 @@ + + + + %highlight([%level]) %replace(%c{0}){'(\$Companion)?\$logger\$1',''} - %msg%n + + + + + logs/app.log + + [%level] %replace(%c{0}){'(\$Companion)?\$logger\$1',''} - %msg%n + + + + + + + + diff --git a/usvm-dataflow-ts/src/test/resources/ts/call.ts b/usvm-dataflow-ts/src/test/resources/ts/call.ts new file mode 100644 index 0000000000..507c49e9df --- /dev/null +++ b/usvm-dataflow-ts/src/test/resources/ts/call.ts @@ -0,0 +1,13 @@ +function entrypoint() { + f(42); + let x= 7 + g(x); +} + +function f(x: any) { + console.log(x); +} + +function g(x: any) { + console.log(x); +} diff --git a/usvm-dataflow-ts/src/test/resources/ts/cast.ts b/usvm-dataflow-ts/src/test/resources/ts/cast.ts new file mode 100644 index 0000000000..53a869a2fd --- /dev/null +++ b/usvm-dataflow-ts/src/test/resources/ts/cast.ts @@ -0,0 +1,12 @@ +declare function getData(): any; + +interface Data {} + +function entrypoint() { + let x = getData() as Data; + infer(x); +} + +function infer(arg: any) { + console.log(arg); +} diff --git a/usvm-dataflow-ts/src/test/resources/ts/data.ts b/usvm-dataflow-ts/src/test/resources/ts/data.ts new file mode 100644 index 0000000000..3bbce38da3 --- /dev/null +++ b/usvm-dataflow-ts/src/test/resources/ts/data.ts @@ -0,0 +1,17 @@ +class Data { + a: number; + b: number; + constructor(a: number, b: number) { + this.a = a; + this.b = b; + } +} + +function entrypoint() { + let data = new Data(7, 42); + process(data); +} + +function process(data: any) { + console.log(data); +} diff --git a/usvm-dataflow-ts/src/test/resources/ts/microphone.ts b/usvm-dataflow-ts/src/test/resources/ts/microphone.ts new file mode 100644 index 0000000000..e436b5f498 --- /dev/null +++ b/usvm-dataflow-ts/src/test/resources/ts/microphone.ts @@ -0,0 +1,25 @@ +interface Microphone { + uuid: string +} + +class VirtualMicro implements Microphone { + uuid: string = "virtual_micro_v3" +} + +interface Devices { + microphone: Microphone +} + +class VirtualDevices implements Devices { + microphone: Microphone = new VirtualMicro() +} + +function getMicrophoneUuid(device: Devices): string { + return device.microphone.uuid +} + +function entrypoint() { + let devices = new VirtualDevices() + let uuid = getMicrophoneUuid(devices) + console.log(uuid) +} diff --git a/usvm-dataflow-ts/src/test/resources/ts/microphone_ctor.ts b/usvm-dataflow-ts/src/test/resources/ts/microphone_ctor.ts new file mode 100644 index 0000000000..0a5692a4c8 --- /dev/null +++ b/usvm-dataflow-ts/src/test/resources/ts/microphone_ctor.ts @@ -0,0 +1,33 @@ +interface Microphone { + uuid: string +} + +class VirtualMicro implements Microphone { + uuid: string; + + constructor() { + this.uuid = "virtual_micro_v3" + } +} + +interface Devices { + microphone: Microphone +} + +class VirtualDevices implements Devices { + microphone: Microphone; + + constructor() { + this.microphone = new VirtualMicro(); + } +} + +function getMicrophoneUuid(devices: Devices): string { + return devices.microphone.uuid; +} + +function entrypoint() { + let devices = new VirtualDevices() + let uuid = getMicrophoneUuid(devices) + console.log(uuid) +} diff --git a/usvm-dataflow-ts/src/test/resources/ts/nested_init.ts b/usvm-dataflow-ts/src/test/resources/ts/nested_init.ts new file mode 100644 index 0000000000..beda346630 --- /dev/null +++ b/usvm-dataflow-ts/src/test/resources/ts/nested_init.ts @@ -0,0 +1,23 @@ +class Foo { +} + +class Cat { + foo: Foo = new Foo(); +} + +// class Cat { +// foo: Foo; +// +// constructor() { +// this.foo = new Foo(); +// } +// } + +function entrypoint() { + let cat = new Cat(); + infer(cat); +} + +function infer(x: any) { + console.log(x); +} diff --git a/usvm-dataflow-ts/src/test/resources/ts/resolver_test.ts b/usvm-dataflow-ts/src/test/resources/ts/resolver_test.ts new file mode 100644 index 0000000000..338e41b26a --- /dev/null +++ b/usvm-dataflow-ts/src/test/resources/ts/resolver_test.ts @@ -0,0 +1,50 @@ +class FieldContainerToInfer { + defaultA: number = 13; + uniqueA: number = 12; + + notUniqueFunction(): number { + return 21 + } +} + +class MethodsContainerToInfer { + uniqueFunction() { + console.log("Hi") + } + + notUniqueFunction(): number { + return 42 + } +} + +class A { + defaultA: number = 23 +} + +function useNonUniqueField(x : A) { + return x.defaultA +} + +function useUniqueFields(x : FieldContainerToInfer) { + return x.uniqueA +} + +function useBothA(x : FieldContainerToInfer): number { + return x.uniqueA + x.defaultA +} + +function useUniqueMethods(x : MethodsContainerToInfer) { + x.uniqueFunction() +} + +function useNotUniqueMethod(x : MethodsContainerToInfer) { + x.notUniqueFunction() +} + +function useFunctionAndField(x : FieldContainerToInfer): number { + x.notUniqueFunction() + return x.defaultA +} + + +// TODO examples with another scope of view and anonymous properties \ No newline at end of file diff --git a/usvm-dataflow-ts/src/test/resources/ts/taint.ts b/usvm-dataflow-ts/src/test/resources/ts/taint.ts new file mode 100644 index 0000000000..ea6ee035b7 --- /dev/null +++ b/usvm-dataflow-ts/src/test/resources/ts/taint.ts @@ -0,0 +1,32 @@ +function source(): number | null { + return null; +} + +function pass(data: number | null): number | null { + return data; +} + +function validate(data: number | null): number { + if (data == null) { + return 0; + } + return data; +} + +function sink(data: number | null) { + if (data == null) { + throw new Error("Error!"); + } +} + +function bad() { + let data = source(); + data = pass(data); + sink(data); +} + +function good() { + let data = source(); + data = validate(data); + sink(data); +} diff --git a/usvm-dataflow-ts/src/test/resources/ts/testcases.ts b/usvm-dataflow-ts/src/test/resources/ts/testcases.ts new file mode 100644 index 0000000000..1731296432 --- /dev/null +++ b/usvm-dataflow-ts/src/test/resources/ts/testcases.ts @@ -0,0 +1,994 @@ +// Case `x := y` +class CaseAssignLocalToLocal { + entrypoint() { + let x = 52; // x: number + let y = x; // y: number + this.infer(y); + } + + infer(a: any) { + const EXPECTED_ARG_0 = "number"; + } +} + +// ---------------------------------------- + +// Case `x := y.f` +class CaseAssignFieldToLocal1 { + entrypoint(y: any) { + let x = y.f; // y: { f: any } + this.infer(y); + } + + infer(a: any) { + const EXPECTED_ARG_0 = "Object { f: any }"; + } +} + +// ---------------------------------------- + +// Case `x := y.f` +class CaseAssignFieldToLocal2 { + entrypoint() { + let y = {f: 42}; // y: { f: number } + let x = y.f; // x: number + this.infer(x); + } + + infer(a: any) { + const EXPECTED_ARG_0 = "number"; + } +} + +// ---------------------------------------- + +// Case `x := a.f` +class CaseAssignFieldToLocal3 { + entrypoint(y: any) { + this.infer(y); + } + + infer(a: any) { + let x = a.f; // a: { f: any } + const EXPECTED_ARG_0 = "Object { f: any }"; + } +} + +// ---------------------------------------- + +// Case `x := x.f` +class CaseAssignFieldToSelf { + entrypoint(a: any) { + let x = { f: a }; + x = x.f; // x: any + this.infer(x); + } + + infer(a: any) { + const EXPECTED_ARG_0 = "Object { f: any }" + } +} + +// Case `x.f := x` +class CaseAssignSelfToField { + entrypoint(a: any) { + let x = { f: a }; // x: { f: any } + x.f = x; // x: { f: any } + this.infer(x); + } + + infer(a: any) { + const EXPECTED_ARG_0 = "Object { f: any }" + } +} + +// ---------------------------------------- + +// Case `x.f := y` +class CaseAssignLocalNumberToField { + entrypoint(x: any) { + let y = 100; // y: number + x.f = y; // x: { f: number } + this.infer(x); + } + + infer(a: any) { + const EXPECTED_ARG_0 = "Object { f: number }"; + } +} + +// ---------------------------------------- + +// Case `x.f := y` +class CaseAssignLocalObjectToField { + entrypoint(x: any) { + let y = { t: 32 }; // y: { t: number } + x.f = y; // x: { f: { t: number } } + this.infer(x); + } + + infer(a: any) { + const EXPECTED_ARG_0 = "Object { f: Object { t: number } }"; + } +} + +// ---------------------------------------- + +// Case `x.f.f := const` +class CaseNestedDuplicateFields { + entrypoint(x: any) { + x.f.f = 2; // x: { f: { f: number } } + this.infer(x); + } + + infer(a: any) { + const EXPECTED_ARG_0 = "Object { f: Object { f: number } }"; + } +} + +// ---------------------------------------- + +// Case `y := [...]` +class CaseAssignArrayToLocal { + entrypoint() { + let y = [1, 2, 3]; // y: Array + this.infer(y); + } + + infer(a: any) { + const EXPECTED_ARG_0 = "Array"; + } +} + +// ---------------------------------------- + +// Case `x := y[i]` +class CaseAssignArrayElementToLocal1 { + entrypoint() { + let y = [33]; // y: Array + let x = y[0]; // y: Array + this.infer(y); + } + + infer(a: any) { + const EXPECTED_ARG_0 = "Array"; + } +} + +// ---------------------------------------- + +// Case `x := y[i]` +class CaseAssignArrayElementToLocal2 { + entrypoint() { + let y = [22]; // y: Array + let x = y[0]; // x: number + this.infer(x); + } + + infer(a: any) { + const EXPECTED_ARG_0 = "number"; + } +} + +// ---------------------------------------- + +// Case `x := y[i]` +class CaseAssignArgumentArrayElementToLocal1 { + entrypoint(y: number[]) { + let x = y[0]; // y: Array + this.infer(y); + } + + infer(a: any) { + const EXPECTED_ARG_0 = "Array"; + } +} + +// ---------------------------------------- + +// Case `x := y[i]` +class CaseAssignArgumentArrayElementToLocal2 { + entrypoint(y: number[]) { + let x = y[0]; // x: any + this.infer(x); + } + + infer(a: any) { + const EXPECTED_ARG_0 = "any"; + } +} + +// ---------------------------------------- + +// Case `x[i] := y` +class CaseAssignLocalToArrayElementNumber { + entrypoint(x: any[]) { + let y = 100; // y: number + x[0] = y; // x: Array + this.infer(x); + } + + infer(a: any) { + const EXPECTED_ARG_0 = "Array"; + } +} + +// ---------------------------------------- + +interface ICustom { + a: number; + b: string; +} + +// Case `x := y as T` +class CaseCastToInterface { + entrypoint(y: any) { + let x = y as ICustom; // x: ICustom + this.infer(x); + } + + infer(a: any) { + const EXPECTED_ARG_0 = "ICustom { }"; + } +} + +// ---------------------------------------- + +// Case `x := y as T[]` +class CaseCastToArrayInterface { + entrypoint(y: any) { + let x = y as ICustom[]; // x: Array + this.infer(x); + } + + infer(a: any) { + const EXPECTED_ARG_0 = "Array"; + } +} + +// // ---------------------------------------- +// +// // Case `x := y + z` +// class CaseAddNumbers { +// entrypoint() { +// let y = 5; // y: number +// let z = 10; // z: number +// let x = y + z; // x: number +// this.infer(x); +// } +// +// infer(a: any) { +// const EXPECTED_ARG_0 = "number"; +// } +// } +// +// // ---------------------------------------- +// +// // Case `x := y && z` +// class CaseLogicalAndBooleans { +// entrypoint() { +// let y = true; // y: boolean +// let z = false; // z: boolean +// let x = y && z; // x: boolean +// this.infer(x); +// } +// +// infer(a: any) { +// const EXPECTED_ARG_0 = "boolean"; +// } +// } +// +// // ---------------------------------------- +// +// // Case `x := y || z` +// class CaseLogicalOrBooleanAndString { +// entrypoint() { +// let y = false; // y: boolean +// let z = "default"; // z: string +// let x = y || z; // x: string +// this.infer(x); +// } +// +// infer(a: any) { +// const EXPECTED_ARG_0 = "boolean | string"; +// } +// } +// +// // ---------------------------------------- +// +// // Case `x := y + z` +// class CaseAddStrings { +// entrypoint() { +// let y = "Hello, "; // y: string +// let z = "World!"; // z: string +// let x = y + z; // x: string +// this.infer(x); +// } +// +// infer(a: any) { +// const EXPECTED_ARG_0 = "string"; +// } +// } +// +// // ---------------------------------------- +// +// // Case `x := number + string` +// class CaseAddNumberToString { +// entrypoint() { +// let y = 12; // y: number +// let z = " is the answer"; // z: string +// let x = y + z; // x: string +// this.infer(x); +// } +// +// infer(a: any) { +// const EXPECTED_ARG_0 = "string"; +// } +// } +// +// // ---------------------------------------- +// +// // Case `x := string + number` +// class CaseAddStringToNumber { +// entrypoint() { +// let y = "The answer is "; // y: string +// let z = 13; // z: number +// let x = y + z; // x: string +// this.infer(x); +// } +// +// infer(a: any) { +// const EXPECTED_ARG_0 = "string"; +// } +// } +// +// // ---------------------------------------- +// +// // Case `x := string - number` +// class CaseSubtractNumberFromString { +// entrypoint() { +// let y = "73"; // y: string +// let z = 10; // z: number +// let x = y - z; // x: number (63) +// this.infer(x); +// } +// +// infer(a: any) { +// const EXPECTED_ARG_0 = "number"; +// } +// } +// +// // ---------------------------------------- +// +// // Case `x := number - string` +// class CaseSubtractStringFromNumber { +// entrypoint() { +// let y = 96; // y: number +// let z = "51"; // z: string +// let x = y - z; // x: number (45) +// this.infer(x); +// } +// +// infer(a: any) { +// const EXPECTED_ARG_0 = "number"; +// } +// } +// +// // ---------------------------------------- +// +// // Case `x := string * number` +// class CaseMultiplyStringByNumber { +// entrypoint() { +// let y = "100"; // y: string +// let z = 30; // z: number +// let x = y * z; // x: number (3000) +// this.infer(x); +// } +// +// infer(a: any) { +// const EXPECTED_ARG_0 = "number"; +// } +// } +// +// // ---------------------------------------- +// +// // Case `x := number * string` +// class CaseMultiplyNumberByString { +// entrypoint() { +// let y = 40; // y: number +// let z = "500"; // z: string +// let x = y * z; // x: number (20000) +// this.infer(x); +// } +// +// infer(a: any) { +// const EXPECTED_ARG_0 = "number"; +// } +// } +// +// // ---------------------------------------- +// +// // Case `x := boolean + number` +// class CaseAddBooleanToNumber { +// entrypoint() { +// let y = true; // y: boolean +// let z = 1; // z: number +// let x = y + z; // x: number (2, as true is coerced to 1) +// this.infer(x); +// } +// +// infer(a: any) { +// const EXPECTED_ARG_0 = "number"; +// } +// } +// +// // ---------------------------------------- +// +// // Case `x := number + boolean` +// class CaseAddNumberToBoolean { +// entrypoint() { +// let y = 1; // y: number +// let z = false; // z: boolean +// let x = y + z; // x: number (1, as false is coerced to 0) +// this.infer(x); +// } +// +// infer(a: any) { +// const EXPECTED_ARG_0 = "number"; +// } +// } +// +// // ---------------------------------------- +// +// // Case `x := null + number` +// class CaseAddNullToNumber { +// entrypoint() { +// let y = null; // y: null +// let z = 105; // z: number +// let x = y + z; // x: number (10, as null is coerced to 0) +// this.infer(x); +// } +// +// infer(a: any) { +// const EXPECTED_ARG_0 = "number"; +// } +// } +// +// // ---------------------------------------- +// +// // Case `x := number + null` +// class CaseAddNumberToNull { +// entrypoint() { +// let y = 115; // y: number +// let z = null; // z: null +// let x = y + z; // x: number (10, as null is coerced to 0) +// this.infer(x); +// } +// +// infer(a: any) { +// const EXPECTED_ARG_0 = "number"; +// } +// } +// +// // ---------------------------------------- +// +// // Case `x := undefined + number` +// class CaseAddUndefinedToNumber { +// entrypoint() { +// let y = undefined; // y: undefined +// let z = 125; // z: number +// let x = y + z; // x: number (NaN) +// this.infer(x); +// } +// +// infer(a: any) { +// const EXPECTED_ARG_0 = "number"; +// } +// } +// +// // ---------------------------------------- +// +// // Case `x := number + undefined` +// class CaseAddNumberToUndefined { +// entrypoint() { +// let y = 135; // y: number +// let z = undefined; // z: undefined +// let x = y + z; // x: number (NaN) +// this.infer(x); +// } +// +// infer(a: any) { +// const EXPECTED_ARG_0 = "number"; +// } +// } +// +// // ---------------------------------------- +// +// // Case `x := string / number` +// class CaseDivideStringByNumber { +// entrypoint() { +// let y = "185"; // y: string +// let z = 5; // z: number +// let x = y / z; // x: number (37) +// this.infer(x); +// } +// +// infer(a: any) { +// const EXPECTED_ARG_0 = "number"; +// } +// } +// +// // ---------------------------------------- +// +// // Case `x := number / string` +// class CaseDivideNumberByString { +// entrypoint() { +// let y = 195; // y: number +// let z = "5"; // z: string +// let x = y / z; // x: number (39) +// this.infer(x); +// } +// +// infer(a: any) { +// const EXPECTED_ARG_0 = "number"; +// } +// } +// +// ---------------------------------------- + +// Case `return x` +class CaseReturnNumber { + entrypoint() { + this.infer(); + } + + infer(): any { + const EXPECTED_RETURN = "number"; + let x = 93; // x: number + return x; + } +} + +// ---------------------------------------- + + // Case `return arg` + class CaseReturnArgumentNumber { + entrypoint() { + let x = 94; // x: number + this.infer(x); + } + + infer(a: any): any { + const EXPECTED_RETURN = "number"; + return a; + } + } + +// ---------------------------------------- + +// Case `return obj` +class CaseReturnObject { + entrypoint() { + this.infer(); + } + + infer(): any { + const EXPECTED_RETURN = "Object { f: number }"; + let x = { f: 95 }; // x: Object { f: number } + return x; + } +} + +// ---------------------------------------- + +// Case `return obj` +class CaseReturnArgumentObject { + entrypoint() { + let x = { f: 96 }; // x: Object { f: number } + this.infer(x); + } + + infer(a: any): any { + const EXPECTED_RETURN = "Object { f: number }"; + return a; + } +} + +// ---------------------------------------- + +// // Case `x.f[0].g := y` +// class CaseAssignToNestedObjectField { +// entrypoint(x: any) { +// let y = 134; // y: number +// x.f[0].g = y; // x: { f: Array<{ g: number }> } +// this.infer(x); +// } +// +// infer(a: any) { +// const EXPECTED_ARG_0 = "Object { f: Array }" +// } +// } +// +// // ---------------------------------------- +// +// // Case `x.f.g.h := y` +// class CaseAssignDeeplyNestedField { +// entrypoint(x: any) { +// let y = "abc"; // y: string +// x.f.g.h = y; // x: { f: { g: { h: string } } } +// this.infer(x); +// } +// +// infer(a: any) { +// const EXPECTED_ARG_0 = "Object { f: Object { g: Object { h: string } } }" +// } +// } +// +// // ---------------------------------------- +// +// // Case `x.f[i].g.h := y` +// class CaseAssignToArrayObjectField { +// entrypoint(x: any) { +// let y = false; // y: boolean +// x.f[2].g.h = y; // x: { f: Array<{ g: { h: boolean } }> } +// this.infer(x); +// } +// +// infer(a: any) { +// const EXPECTED_ARG_0 = "Object { f: Array }" +// } +// } +// +// // ---------------------------------------- +// +// // Case `x[i].f.g := y` +// class CaseAssignArrayFieldToNestedObject { +// entrypoint(x: any) { +// let y = 219; // y: number +// x[0].f.g = y; // x: Array<{ f: { g: number } }> +// this.infer(x); +// } +// +// infer(a: any) { +// const EXPECTED_ARG_0 = "Array<{ f: Object { g: number } }>" +// } +// } +// +// // ---------------------------------------- +// +// // Case `x.f[i][j] := y` +// class CaseAssignToMultiDimensionalArray { +// entrypoint(x: any) { +// let y = "data"; // y: string +// x.f[1][2] = y; // x: { f: Array> } +// this.infer(x); +// } +// +// infer(a: any) { +// const EXPECTED_ARG_0 = "Object { f: Array> }" +// } +// } +// +// // ---------------------------------------- +// +// // Case `x.f[0].g[1].h := y` +// class CaseAssignToComplexNestedArrayField { +// entrypoint(x: any) { +// let y = true; // y: boolean +// x.f[0].g[1].h = y; // x: { f: Array<{ g: Array<{ h: boolean }> }> } +// this.infer(x); +// } +// +// infer(a: any) { +// const EXPECTED_ARG_0 = "Object { f: Array }> }" +// } +// } +// +// // ---------------------------------------- +// +// // Case `x.f.g.h[i] := y` +// class CaseAssignToArrayInNestedObject { +// entrypoint(x: any) { +// let y = 3.14; // y: number +// x.f.g.h[2] = y; // x: { f: { g: { h: Array } } } +// this.infer(x); +// } +// +// infer(a: any) { +// const EXPECTED_ARG_0 = "Object { f: Object { g: Object { h: Array } } }" +// } +// } +// +// // ---------------------------------------- +// +// // Case `x.f[0].g.h[i] := y` +// class CaseAssignToArrayInDeeplyNestedObject { +// entrypoint(x: any) { +// let y = null; // y: null +// x.f[0].g.h[3] = y; // x: { f: Array<{ g: { h: Array } }> } +// this.infer(x); +// } +// +// infer(a: any) { +// const EXPECTED_ARG_0 = "Object { f: Array } }> }" +// } +// } +// +// // ---------------------------------------- +// +// // Case `x.f.g[i].h.j := y` +// class CaseAssignToDeeplyNestedObjectArray { +// entrypoint(x: any) { +// let y = "nested"; // y: string +// x.f.g[1].h.j = y; // x: { f: { g: Array<{ h: { j: string } }> } } +// this.infer(x); +// } +// +// infer(a: any) { +// const EXPECTED_ARG_0 = "Object { f: Object { g: Array } }" +// } +// } +// +// // ---------------------------------------- +// +// // Case `x.f.g.h[0][i] := y` +// class CaseAssignToMultiDimensionalArrayField { +// entrypoint(x: any) { +// let y = 99; // y: number +// x.f.g.h[0][1] = y; // x: { f: { g: { h: Array> } } } +// this.infer(x); +// } +// +// infer(a: any) { +// const EXPECTED_ARG_0 = "Object { f: Object { g: { h: Array> } } }" +// } +// } + +// ---------------------------------------- + +class MyType { + f: number = 15; +} + +// Case `x := new T()` +class CaseNew { + entrypoint() { + let y = new MyType(); // y: MyType + // hidden: + // y := new MyType() + // -> y.constructor() + // -> this.() + // -> this.f := 15 + // -> this: { f: number } + // -> y: { f.number } + this.infer(y); + } + + infer(a: any): any { + const EXPECTED_ARG_0 = "MyType { f: number }" + } +} + +// ---------------------------------------- + +// Case `x := number | string` +class CaseUnion { + entrypoint() { + let x: string | number = "str"; // x: string + x = 42; // x: number + this.infer(x); + } + + infer(a: any): any { + const EXPECTED_ARG_0 = "number"; + } +} + +// Case `x := number | string` +class CaseArgumentUnion { + entrypoint(x: string | number) { + x = "kek"; + x = 42; + this.infer(x); + } + + infer(a: any): any { + const EXPECTED_ARG_0 = "number"; + } +} + +// Case `x := y` +class CaseUnion2 { + entrypoint() { + let y = "str"; + let x: string | number = y; + x = 42; + this.infer(y); + } + + infer(a: any): any { + const EXPECTED_ARG_0 = "string"; + } +} + +// Case `x := y` +class CaseArgumentUnion2 { + entrypoint(y: string) { + let x: string | number = y; + x = 42; + this.infer(y); + } + + infer(a: any): any { + const EXPECTED_ARG_0 = "any"; + } +} + +// Case `x := "string" | number` +class CaseUnion3 { + entrypoint() { + let x: string | number; + if (Math.random() > 0.5) { + x = "str"; + } else { + x = 42; + } + this.infer(x); + } + + infer(a: any): any { + const EXPECTED_ARG_0 = "number | string"; + } +} + +// Case `x := "string" | number` +class CaseArgumentUnion3 { + entrypoint(x: string | number) { + if (Math.random() > 0.5) { + x = "str"; + } else { + x = 42; + } + this.infer(x); + } + + infer(a: any): any { + const EXPECTED_ARG_0 = "number | string"; + } +} + +// Case `x := "string", x := number` +class CaseUnion4 { + entrypoint() { + let x: string | number = "str"; + if (Math.random() > 0.5) { + x = 42; + } + this.infer(x); + } + + infer(a: any): any { + const EXPECTED_ARG_0 = "number | string"; + } +} + +// Case `x := "string", x := number` +// class CaseUnion5 { +// entrypoint() { +// let x: string | number; +// if (Math.random() > 0.5) { +// x = 42; +// } +// this.infer(x); +// } +// +// infer(a: any): any { +// // Currently, `number | undefined` is inferred due to the lack of DeclareStmt +// const EXPECTED_ARG_0 = "number | string"; +// } +// } + +// ---------------------------------------- + +// Case `y := x.f.g` +class CaseAliasChain1 { + entrypoint(x: any) { + let y = x.f; // x: { f: any } + y.g = 42; // x: { f: { g: number } } + this.infer(x); + } + + infer(a: any): any { + const EXPECTED_ARG_0 = "Object { f: Object { g: number } }" + } +} + +// Case `x.f.g := number` +class CaseAssignNumberToNestedField { + entrypoint(x: any) { + x.f.g = 100; // x: { f: { g: number } } + this.infer(x); + } + + infer(a: any) { + const EXPECTED_ARG_0 = "Object { f: Object { g: number } }"; + } +} + +// Case `x.f := (y: number)` +class CaseAssignLocalNumberToNestedField { + entrypoint(x: any) { + let y = 98; // y: number + x.f.g = y; // x: { f: { g: number } } + this.infer(x); + } + + infer(a: any) { + const EXPECTED_ARG_0 = "Object { f: Object { g: number } }"; + } +} + +// ---------------------------------------- + +class CaseLoop { + entrypoint() { + let x: any = {}; + let a: any = 42; + for (let i = 0; i < 10; i++) { + x.f = a; + a = x; + } + this.infer(x); + } + + infer(a: any) { + const EXPECTED_ARG_0 = "any"; + } +} + +class CaseFindAssignmentAfterLoop { + entrypoint(y: number) { + let x = []; + for (let i = 0; i < y; i++) { + x.push(i); + } + if (x.length == 0) { + x.push(42); + } + + this.infer(x) + } + + infer(a: any) { + // + } +} + +// ---------------------------------------- + +class Tree { + children: Tree[] = []; + + getChildren(): Tree[] { + return this.children; + } +} + +class CaseRecursion { + entrypoint(root: Tree) { + this.traverse([root]); + this.infer(root); + } + + traverse(xs: Tree[]) { + for (let i = 0; i < xs.length; i++) { + let child: Tree = xs[i]; + let children: Tree[] = child.getChildren(); + this.traverse(children); + } + } + + infer(a: any) { + const EXPECTED_ARG_0 = "Tree { children: Array }"; + } +} diff --git a/usvm-dataflow-ts/src/test/resources/ts/types.ts b/usvm-dataflow-ts/src/test/resources/ts/types.ts new file mode 100644 index 0000000000..4e8d90ccb3 --- /dev/null +++ b/usvm-dataflow-ts/src/test/resources/ts/types.ts @@ -0,0 +1,55 @@ +interface A { + aStr: string + bObj: B +} + +interface B { + bNum: number +} + +function conditional(x: A, cond: boolean): number { + if (cond) { + return x.aStr.length + } else { + return x.bObj.bNum + } +} + +function entrypoint1(arg: A) { + console.log(conditional(arg, false)) +} + +interface X { + a: string +} + +interface Y { + b: number +} + +function foo(x: X | Y) { + if ("a" in x) { + strBar(x.a) + } else { + numberBar(x.b) + } +} + +function baz(x: X & Y) { + strBar(x.a) + numberBar(x.b) +} + +function strBar(x: string) { + +} + +function numberBar(x: number) { + +} + +function entrypoint2(arg0: X, arg1: Y) { + foo(arg0) + foo(arg1) + baz({...arg0, ...arg1}) +} diff --git a/usvm-dataflow-ts/src/testFixtures/kotlin/org/usvm/dataflow/ts/LoadEts.kt b/usvm-dataflow-ts/src/testFixtures/kotlin/org/usvm/dataflow/ts/LoadEts.kt new file mode 100644 index 0000000000..d6195a60b1 --- /dev/null +++ b/usvm-dataflow-ts/src/testFixtures/kotlin/org/usvm/dataflow/ts/LoadEts.kt @@ -0,0 +1,117 @@ +package org.usvm.dataflow.ts + +import mu.KotlinLogging +import org.jacodb.ets.dto.EtsFileDto +import org.jacodb.ets.dto.convertToEtsFile +import org.jacodb.ets.model.EtsFile +import org.jacodb.ets.model.EtsScene +import java.nio.file.Path +import kotlin.io.path.extension +import kotlin.io.path.inputStream +import kotlin.io.path.relativeTo +import kotlin.io.path.walk + +private val logger = KotlinLogging.logger {} + +/** + * Load an [EtsFileDto] from a resource file. + * + * For example, `resources/ets/sample.json` can be loaded with: + * ``` + * val dto: EtsFileDto = loadEtsFileDtoFromResource("/ets/sample.json") + * ``` + */ +fun loadEtsFileDtoFromResource(jsonPath: String): EtsFileDto { + logger.debug { "Loading EtsIR from resource: '$jsonPath'" } + require(jsonPath.endsWith(".json")) { "File must have a '.json' extension: '$jsonPath'" } + getResourceStream(jsonPath).use { stream -> + return EtsFileDto.loadFromJson(stream) + } +} + +/** + * Load an [EtsFile] from a resource file. + * + * For example, `resources/ets/sample.json` can be loaded with: + * ``` + * val file: EtsFile = loadEtsFileFromResource("/ets/sample.json") + * ``` + */ +fun loadEtsFileFromResource(jsonPath: String): EtsFile { + val etsFileDto = loadEtsFileDtoFromResource(jsonPath) + return convertToEtsFile(etsFileDto) +} + +/** + * Load multiple [EtsFile]s from a resource directory. + * + * For example, all files in `resources/project/` can be loaded with: + * ``` + * val files: Sequence = loadMultipleEtsFilesFromResourceDirectory("/project") + * ``` + */ +fun loadMultipleEtsFilesFromResourceDirectory(dirPath: String): Sequence { + val rootPath = getResourcePath(dirPath) + return rootPath.walk().filter { it.extension == "json" }.map { path -> + loadEtsFileFromResource("$dirPath/${path.relativeTo(rootPath)}") + } +} + +fun loadMultipleEtsFilesFromMultipleResourceDirectories( + dirPaths: List, +): Sequence { + return dirPaths.asSequence().flatMap { loadMultipleEtsFilesFromResourceDirectory(it) } +} + +fun loadEtsProjectFromResources( + modules: List, + prefix: String, +): EtsScene { + logger.info { "Loading Ets project with modules $modules from '$prefix/'" } + val dirPaths = modules.map { "$prefix/$it" } + val files = loadMultipleEtsFilesFromMultipleResourceDirectories(dirPaths).toList() + logger.info { "Loaded ${files.size} files" } + return EtsScene(files, sdkFiles = emptyList()) +} + +//----------------------------------------------------------------------------- + +/** + * Load an [EtsFileDto] from a file. + * + * For example, `data/sample.json` can be loaded with: + * ``` + * val dto: EtsFileDto = loadEtsFileDto(Path("data/sample.json")) + * ``` + */ +fun loadEtsFileDto(path: Path): EtsFileDto { + require(path.extension == "json") { "File must have a '.json' extension: $path" } + path.inputStream().use { stream -> + return EtsFileDto.loadFromJson(stream) + } +} + +/** + * Load an [EtsFile] from a file. + * + * For example, `data/sample.json` can be loaded with: + * ``` + * val file: EtsFile = loadEtsFile(Path("data/sample.json")) + * ``` + */ +fun loadEtsFile(path: Path): EtsFile { + val etsFileDto = loadEtsFileDto(path) + return convertToEtsFile(etsFileDto) +} + +/** + * Load multiple [EtsFile]s from a directory. + * + * For example, all files in `data` can be loaded with: + * ``` + * val files: Sequence = loadMultipleEtsFilesFromDirectory(Path("data")) + * ``` + */ +fun loadMultipleEtsFilesFromDirectory(dirPath: Path): Sequence { + return dirPath.walk().filter { it.extension == "json" }.map { loadEtsFile(it) } +} diff --git a/usvm-dataflow-ts/src/testFixtures/kotlin/org/usvm/dataflow/ts/Resources.kt b/usvm-dataflow-ts/src/testFixtures/kotlin/org/usvm/dataflow/ts/Resources.kt new file mode 100644 index 0000000000..6b00cd961a --- /dev/null +++ b/usvm-dataflow-ts/src/testFixtures/kotlin/org/usvm/dataflow/ts/Resources.kt @@ -0,0 +1,20 @@ +package org.usvm.dataflow.ts + +import java.io.InputStream +import java.nio.file.Path +import kotlin.io.path.toPath + +fun getResourcePathOrNull(res: String): Path? { + require(res.startsWith("/")) { "Resource path must start with '/': '$res'" } + return object {}::class.java.getResource(res)?.toURI()?.toPath() +} + +fun getResourcePath(res: String): Path { + return getResourcePathOrNull(res) ?: error("Resource not found: '$res'") +} + +fun getResourceStream(res: String): InputStream { + require(res.startsWith("/")) { "Resource path must start with '/': '$res'" } + return object {}::class.java.getResourceAsStream(res) + ?: error("Resource not found: '$res'") +} diff --git a/usvm-dataflow-ts/src/testFixtures/kotlin/org/usvm/dataflow/ts/TestFactoryDsl.kt b/usvm-dataflow-ts/src/testFixtures/kotlin/org/usvm/dataflow/ts/TestFactoryDsl.kt new file mode 100644 index 0000000000..a710f1c896 --- /dev/null +++ b/usvm-dataflow-ts/src/testFixtures/kotlin/org/usvm/dataflow/ts/TestFactoryDsl.kt @@ -0,0 +1,52 @@ +package org.usvm.dataflow.ts + +import org.junit.jupiter.api.DynamicContainer +import org.junit.jupiter.api.DynamicNode +import org.junit.jupiter.api.DynamicTest +import org.junit.jupiter.api.function.Executable +import java.util.stream.Stream + +private interface TestProvider { + fun test(name: String, test: () -> Unit) +} + +private interface ContainerProvider { + fun container(name: String, init: TestContainerBuilder.() -> Unit) +} + +class TestContainerBuilder(var name: String) : TestProvider, ContainerProvider { + private val nodes: MutableList = mutableListOf() + + override fun test(name: String, test: () -> Unit) { + nodes += dynamicTest(name, test) + } + + override fun container(name: String, init: TestContainerBuilder.() -> Unit) { + nodes += containerBuilder(name, init) + } + + fun build(): DynamicContainer = DynamicContainer.dynamicContainer(name, nodes) +} + +private fun containerBuilder(name: String, init: TestContainerBuilder.() -> Unit): DynamicContainer = + TestContainerBuilder(name).apply(init).build() + +class TestFactoryBuilder : TestProvider, ContainerProvider { + private val nodes: MutableList = mutableListOf() + + override fun test(name: String, test: () -> Unit) { + nodes += dynamicTest(name, test) + } + + override fun container(name: String, init: TestContainerBuilder.() -> Unit) { + nodes += containerBuilder(name, init) + } + + fun build(): Stream = nodes.stream() +} + +fun testFactory(init: TestFactoryBuilder.() -> Unit): Stream = + TestFactoryBuilder().apply(init).build() + +private fun dynamicTest(name: String, test: () -> Unit): DynamicTest = + DynamicTest.dynamicTest(name, Executable(test)) diff --git a/usvm-dataflow/build.gradle.kts b/usvm-dataflow/build.gradle.kts index a4724708fc..5de8fa92be 100644 --- a/usvm-dataflow/build.gradle.kts +++ b/usvm-dataflow/build.gradle.kts @@ -3,9 +3,10 @@ plugins { } dependencies { - implementation(Libs.jacodb_api_common) + implementation(project(":usvm-util")) + api(Libs.jacodb_api_common) implementation(Libs.jacodb_taint_configuration) - implementation(Libs.sarif4k) + api(Libs.sarif4k) } publishing { diff --git a/usvm-dataflow/src/main/kotlin/org/usvm/dataflow/config/Condition.kt b/usvm-dataflow/src/main/kotlin/org/usvm/dataflow/config/Condition.kt index ee9c915b16..95968d320b 100644 --- a/usvm-dataflow/src/main/kotlin/org/usvm/dataflow/config/Condition.kt +++ b/usvm-dataflow/src/main/kotlin/org/usvm/dataflow/config/Condition.kt @@ -35,11 +35,11 @@ import org.jacodb.taint.configuration.Or import org.jacodb.taint.configuration.PositionResolver import org.jacodb.taint.configuration.SourceFunctionMatches import org.jacodb.taint.configuration.TypeMatches -import org.usvm.dataflow.ifds.Maybe -import org.usvm.dataflow.ifds.onSome import org.usvm.dataflow.taint.Tainted import org.usvm.dataflow.util.Traits import org.usvm.dataflow.util.removeTrailingElementAccessors +import org.usvm.util.Maybe +import org.usvm.util.onSome open class BasicConditionEvaluator( val traits: Traits, diff --git a/usvm-dataflow/src/main/kotlin/org/usvm/dataflow/config/Position.kt b/usvm-dataflow/src/main/kotlin/org/usvm/dataflow/config/Position.kt index e51204caba..f8b95b9cde 100644 --- a/usvm-dataflow/src/main/kotlin/org/usvm/dataflow/config/Position.kt +++ b/usvm-dataflow/src/main/kotlin/org/usvm/dataflow/config/Position.kt @@ -30,10 +30,10 @@ import org.jacodb.taint.configuration.ResultAnyElement import org.jacodb.taint.configuration.This import org.usvm.dataflow.ifds.AccessPath import org.usvm.dataflow.ifds.ElementAccessor -import org.usvm.dataflow.ifds.Maybe -import org.usvm.dataflow.ifds.fmap -import org.usvm.dataflow.ifds.toMaybe import org.usvm.dataflow.util.Traits +import org.usvm.util.Maybe +import org.usvm.util.fmap +import org.usvm.util.toMaybe class CallPositionToAccessPathResolver( private val traits: Traits, diff --git a/usvm-dataflow/src/main/kotlin/org/usvm/dataflow/config/TaintAction.kt b/usvm-dataflow/src/main/kotlin/org/usvm/dataflow/config/TaintAction.kt index 622dc56767..5ef603b45a 100644 --- a/usvm-dataflow/src/main/kotlin/org/usvm/dataflow/config/TaintAction.kt +++ b/usvm-dataflow/src/main/kotlin/org/usvm/dataflow/config/TaintAction.kt @@ -16,17 +16,17 @@ package org.usvm.dataflow.config -import org.usvm.dataflow.ifds.AccessPath -import org.usvm.dataflow.ifds.Maybe -import org.usvm.dataflow.ifds.fmap -import org.usvm.dataflow.ifds.map -import org.usvm.dataflow.taint.Tainted import org.jacodb.taint.configuration.AssignMark import org.jacodb.taint.configuration.CopyAllMarks import org.jacodb.taint.configuration.CopyMark import org.jacodb.taint.configuration.PositionResolver import org.jacodb.taint.configuration.RemoveAllMarks import org.jacodb.taint.configuration.RemoveMark +import org.usvm.dataflow.ifds.AccessPath +import org.usvm.dataflow.taint.Tainted +import org.usvm.util.Maybe +import org.usvm.util.fmap +import org.usvm.util.map class TaintActionEvaluator( private val positionResolver: PositionResolver>, diff --git a/usvm-dataflow/src/main/kotlin/org/usvm/dataflow/ifds/Maybe.kt b/usvm-dataflow/src/main/kotlin/org/usvm/dataflow/ifds/Maybe.kt deleted file mode 100644 index e804812aa5..0000000000 --- a/usvm-dataflow/src/main/kotlin/org/usvm/dataflow/ifds/Maybe.kt +++ /dev/null @@ -1,60 +0,0 @@ -/* - * Copyright 2022 UnitTestBot contributors (utbot.org) - *

- * 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. - */ - -package org.usvm.dataflow.ifds - -@JvmInline -value class Maybe private constructor( - private val rawValue: Any?, -) { - val isSome: Boolean get() = rawValue !== NONE_VALUE - val isNone: Boolean get() = rawValue === NONE_VALUE - - fun getOrThrow(): T { - check(isSome) { "Maybe is None" } - @Suppress("UNCHECKED_CAST") - return rawValue as T - } - - companion object { - private val NONE_VALUE = Any() - private val NONE = Maybe(NONE_VALUE) - - fun none(): Maybe = NONE - - fun some(value: T): Maybe = Maybe(value) - - fun from(value: T?): Maybe = if (value == null) none() else some(value) - } -} - -inline fun Maybe.map(body: (T) -> Maybe): Maybe = - if (isNone) Maybe.none() else body(getOrThrow()) - -inline fun Maybe.fmap(body: (T) -> R): Maybe = - if (isNone) Maybe.none() else Maybe.some(body(getOrThrow())) - -inline fun Maybe.onSome(body: (T) -> Unit): Maybe { - if (isSome) body(getOrThrow()) - return this -} - -inline fun Maybe.onNone(body: () -> Unit): Maybe { - if (isNone) body() - return this -} - -fun T?.toMaybe(): Maybe = Maybe.from(this) diff --git a/usvm-dataflow/src/main/kotlin/org/usvm/dataflow/taint/TaintFlowFunctions.kt b/usvm-dataflow/src/main/kotlin/org/usvm/dataflow/taint/TaintFlowFunctions.kt index d64c6bed74..4858c09921 100644 --- a/usvm-dataflow/src/main/kotlin/org/usvm/dataflow/taint/TaintFlowFunctions.kt +++ b/usvm-dataflow/src/main/kotlin/org/usvm/dataflow/taint/TaintFlowFunctions.kt @@ -48,9 +48,9 @@ import org.usvm.dataflow.ifds.FlowFunctions import org.usvm.dataflow.ifds.isOnHeap import org.usvm.dataflow.ifds.isStatic import org.usvm.dataflow.ifds.minus -import org.usvm.dataflow.ifds.onSome import org.usvm.dataflow.util.Traits import org.usvm.dataflow.util.startsWith +import org.usvm.util.onSome private val logger = mu.KotlinLogging.logger {} @@ -462,7 +462,6 @@ class ForwardTaintFlowFunctions( } } } - class BackwardTaintFlowFunctions( private val traits: Traits, private val graph: ApplicationGraph, diff --git a/usvm-jvm-dataflow/build.gradle.kts b/usvm-jvm-dataflow/build.gradle.kts index e22790a6d9..cf5e9bb997 100644 --- a/usvm-jvm-dataflow/build.gradle.kts +++ b/usvm-jvm-dataflow/build.gradle.kts @@ -10,16 +10,14 @@ val samples by sourceSets.creating { dependencies { api(project(":usvm-dataflow")) + implementation(project(":usvm-util")) - implementation(Libs.jacodb_api_common) implementation(Libs.jacodb_api_jvm) implementation(Libs.jacodb_core) implementation(Libs.jacodb_api_storage) implementation(Libs.jacodb_storage) implementation(Libs.jacodb_taint_configuration) - implementation(Libs.sarif4k) - testImplementation(Libs.mockk) testImplementation(Libs.junit_jupiter_params) diff --git a/usvm-jvm-dataflow/src/main/kotlin/org/usvm/dataflow/jvm/npe/NpeFlowFunctions.kt b/usvm-jvm-dataflow/src/main/kotlin/org/usvm/dataflow/jvm/npe/NpeFlowFunctions.kt index 5f1fa363b4..a82e18d4a7 100644 --- a/usvm-jvm-dataflow/src/main/kotlin/org/usvm/dataflow/jvm/npe/NpeFlowFunctions.kt +++ b/usvm-jvm-dataflow/src/main/kotlin/org/usvm/dataflow/jvm/npe/NpeFlowFunctions.kt @@ -63,13 +63,13 @@ import org.usvm.dataflow.ifds.FlowFunctions import org.usvm.dataflow.ifds.isOnHeap import org.usvm.dataflow.ifds.isStatic import org.usvm.dataflow.ifds.minus -import org.usvm.dataflow.ifds.onSome import org.usvm.dataflow.jvm.graph.JcApplicationGraph import org.usvm.dataflow.jvm.util.JcTraits import org.usvm.dataflow.taint.TaintDomainFact import org.usvm.dataflow.taint.TaintZeroFact import org.usvm.dataflow.taint.Tainted import org.usvm.dataflow.util.startsWith +import org.usvm.util.onSome private val logger = mu.KotlinLogging.logger {} diff --git a/usvm-jvm-dataflow/src/test/kotlin/org/usvm/dataflow/jvm/impl/ConditionEvaluatorTest.kt b/usvm-jvm-dataflow/src/test/kotlin/org/usvm/dataflow/jvm/impl/ConditionEvaluatorTest.kt index 2141b9b026..e3545a3c28 100644 --- a/usvm-jvm-dataflow/src/test/kotlin/org/usvm/dataflow/jvm/impl/ConditionEvaluatorTest.kt +++ b/usvm-jvm-dataflow/src/test/kotlin/org/usvm/dataflow/jvm/impl/ConditionEvaluatorTest.kt @@ -53,10 +53,10 @@ import org.jacodb.taint.configuration.TypeMatches import org.junit.jupiter.api.Test import org.usvm.dataflow.config.BasicConditionEvaluator import org.usvm.dataflow.config.FactAwareConditionEvaluator -import org.usvm.dataflow.ifds.Maybe -import org.usvm.dataflow.ifds.toMaybe import org.usvm.dataflow.jvm.util.JcTraits import org.usvm.dataflow.taint.Tainted +import org.usvm.util.Maybe +import org.usvm.util.toMaybe import kotlin.test.assertFailsWith import kotlin.test.assertFalse import kotlin.test.assertTrue diff --git a/usvm-jvm-dataflow/src/test/kotlin/org/usvm/dataflow/jvm/impl/NullabilityAssumptionAnalysisTest.kt b/usvm-jvm-dataflow/src/test/kotlin/org/usvm/dataflow/jvm/impl/NullabilityAssumptionAnalysisTest.kt deleted file mode 100644 index 3d4bad0591..0000000000 --- a/usvm-jvm-dataflow/src/test/kotlin/org/usvm/dataflow/jvm/impl/NullabilityAssumptionAnalysisTest.kt +++ /dev/null @@ -1,69 +0,0 @@ -/* - * Copyright 2022 UnitTestBot contributors (utbot.org) - *

- * 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. - */ - -package org.usvm.dataflow.jvm.impl - -import NullAssumptionAnalysisExample -import org.jacodb.api.jvm.JcClassOrInterface -import org.jacodb.api.jvm.JcMethod -import org.jacodb.api.jvm.cfg.JcAssignInst -import org.jacodb.api.jvm.cfg.JcInstanceCallExpr -import org.jacodb.api.jvm.cfg.JcLocal -import org.jacodb.api.jvm.ext.findClass -import org.junit.jupiter.api.Assertions.assertTrue -import org.junit.jupiter.api.Test -import org.junit.jupiter.api.TestInstance -import org.junit.jupiter.api.TestInstance.Lifecycle.PER_CLASS -import org.usvm.dataflow.jvm.flow.NullAssumptionAnalysis - -@TestInstance(PER_CLASS) -class NullabilityAssumptionAnalysisTest : BaseAnalysisTest() { - - @Test - fun `null-assumption analysis should work`() { - val clazz = cp.findClass() - with(clazz.findMethod("test1").flowGraph()) { - val analysis = NullAssumptionAnalysis(this).also { - it.run() - } - val sout = (instructions[0] as JcAssignInst).lhv as JcLocal - val a = ((instructions[3] as JcAssignInst).rhv as JcInstanceCallExpr).instance - - assertTrue(analysis.isAssumedNonNullBefore(instructions[2], a)) - assertTrue(analysis.isAssumedNonNullBefore(instructions[0], sout)) - } - } - - @Test - fun `null-assumption analysis should work 2`() { - val clazz = cp.findClass() - with(clazz.findMethod("test2").flowGraph()) { - val analysis = NullAssumptionAnalysis(this).also { - it.run() - } - val sout = (instructions[0] as JcAssignInst).lhv as JcLocal - val a = ((instructions[3] as JcAssignInst).rhv as JcInstanceCallExpr).instance - val x = (instructions[5] as JcAssignInst).lhv as JcLocal - - assertTrue(analysis.isAssumedNonNullBefore(instructions[2], a)) - assertTrue(analysis.isAssumedNonNullBefore(instructions[0], sout)) - analysis.isAssumedNonNullBefore(instructions[5], x) - } - } - - private fun JcClassOrInterface.findMethod(name: String): JcMethod = declaredMethods.first { it.name == name } - -} diff --git a/usvm-ts/build.gradle.kts b/usvm-ts/build.gradle.kts index f1d7ed9655..f9121b88ce 100644 --- a/usvm-ts/build.gradle.kts +++ b/usvm-ts/build.gradle.kts @@ -4,6 +4,7 @@ plugins { dependencies { implementation(project(":usvm-core")) + implementation(project(":usvm-dataflow-ts")) implementation(Libs.jacodb_core) implementation(Libs.jacodb_ets) diff --git a/usvm-ts/src/main/kotlin/org/usvm/TSApplicationGraph.kt b/usvm-ts/src/main/kotlin/org/usvm/TSApplicationGraph.kt index 0d2d92a735..524a0974a5 100644 --- a/usvm-ts/src/main/kotlin/org/usvm/TSApplicationGraph.kt +++ b/usvm-ts/src/main/kotlin/org/usvm/TSApplicationGraph.kt @@ -1,12 +1,14 @@ package org.usvm import org.jacodb.ets.base.EtsStmt -import org.jacodb.ets.model.EtsFile import org.jacodb.ets.model.EtsMethod +import org.jacodb.ets.model.EtsScene +import org.usvm.dataflow.ts.graph.EtsApplicationGraph +import org.usvm.dataflow.ts.graph.EtsApplicationGraphImpl import org.usvm.statistics.ApplicationGraph -class TSApplicationGraph(project: EtsFile) : ApplicationGraph { - private val applicationGraph: ApplicationGraph = TODO() +class TSApplicationGraph(scene: EtsScene) : ApplicationGraph { + private val applicationGraph: EtsApplicationGraph = EtsApplicationGraphImpl(scene) override fun predecessors(node: EtsStmt): Sequence = applicationGraph.predecessors(node) diff --git a/usvm-ts/src/main/kotlin/org/usvm/TSExprResolver.kt b/usvm-ts/src/main/kotlin/org/usvm/TSExprResolver.kt index b383922130..cd26df8888 100644 --- a/usvm-ts/src/main/kotlin/org/usvm/TSExprResolver.kt +++ b/usvm-ts/src/main/kotlin/org/usvm/TSExprResolver.kt @@ -87,8 +87,8 @@ class TSExprResolver( fun resolveLValue(value: EtsValue): ULValue<*, *>? = when (value) { - is EtsParameterRef, - is EtsLocal -> simpleValueResolver.resolveLocal(value) + is EtsParameterRef, is EtsLocal -> simpleValueResolver.resolveLocal(value) + else -> error("Unexpected value: $value") } @@ -115,8 +115,6 @@ class TSExprResolver( return block(result0, result1) } - - override fun visit(value: EtsLocal): UExpr { return simpleValueResolver.visit(value) } @@ -157,6 +155,10 @@ class TSExprResolver( TODO("Not yet implemented") } + override fun visit(expr: EtsAwaitExpr): UExpr? { + TODO("Not yet implemented") + } + override fun visit(expr: EtsBitAndExpr): UExpr { TODO("Not yet implemented") } @@ -185,14 +187,6 @@ class TSExprResolver( TODO("Not yet implemented") } - override fun visit(expr: EtsAwaitExpr): UExpr? { - TODO("Not yet implemented") - } - - override fun visit(expr: EtsYieldExpr): UExpr? { - TODO("Not yet implemented") - } - override fun visit(expr: EtsDivExpr): UExpr { TODO("Not yet implemented") } @@ -337,6 +331,10 @@ class TSExprResolver( TODO("Not yet implemented") } + override fun visit(expr: EtsYieldExpr): UExpr? { + TODO("Not yet implemented") + } + override fun visit(value: EtsArrayAccess): UExpr { TODO("Not yet implemented") } diff --git a/usvm-ts/src/main/kotlin/org/usvm/TSMachine.kt b/usvm-ts/src/main/kotlin/org/usvm/TSMachine.kt index 4e56356488..4d9ac76b79 100644 --- a/usvm-ts/src/main/kotlin/org/usvm/TSMachine.kt +++ b/usvm-ts/src/main/kotlin/org/usvm/TSMachine.kt @@ -1,8 +1,8 @@ package org.usvm import org.jacodb.ets.base.EtsStmt -import org.jacodb.ets.model.EtsFile import org.jacodb.ets.model.EtsMethod +import org.jacodb.ets.model.EtsScene import org.usvm.ps.createPathSelector import org.usvm.state.TSMethodResult import org.usvm.state.TSState @@ -20,7 +20,7 @@ import org.usvm.stopstrategies.createStopStrategy import kotlin.time.Duration.Companion.seconds class TSMachine( - private val project: EtsFile, + private val project: EtsScene, private val options: UMachineOptions, ) : UMachine() { private val typeSystem = TSTypeSystem(typeOperationsTimeout = 1.seconds, project) diff --git a/usvm-ts/src/main/kotlin/org/usvm/TSTypeSystem.kt b/usvm-ts/src/main/kotlin/org/usvm/TSTypeSystem.kt index 9402a88501..d6f821ada2 100644 --- a/usvm-ts/src/main/kotlin/org/usvm/TSTypeSystem.kt +++ b/usvm-ts/src/main/kotlin/org/usvm/TSTypeSystem.kt @@ -1,14 +1,14 @@ package org.usvm import org.jacodb.ets.base.EtsType -import org.jacodb.ets.model.EtsFile +import org.jacodb.ets.model.EtsScene import org.usvm.types.UTypeStream import org.usvm.types.UTypeSystem import kotlin.time.Duration class TSTypeSystem( override val typeOperationsTimeout: Duration, - val project: EtsFile, + val project: EtsScene, ) : UTypeSystem { override fun isSupertype(supertype: EtsType, type: EtsType): Boolean { diff --git a/usvm-ts/src/test/kotlin/org/usvm/util/TSMethodTestRunner.kt b/usvm-ts/src/test/kotlin/org/usvm/util/TSMethodTestRunner.kt index fa82d2177d..970861388c 100644 --- a/usvm-ts/src/test/kotlin/org/usvm/util/TSMethodTestRunner.kt +++ b/usvm-ts/src/test/kotlin/org/usvm/util/TSMethodTestRunner.kt @@ -1,5 +1,6 @@ package org.usvm.util +import org.jacodb.ets.base.DEFAULT_ARK_CLASS_NAME import org.jacodb.ets.base.EtsAnyType import org.jacodb.ets.base.EtsBooleanType import org.jacodb.ets.base.EtsNumberType @@ -10,6 +11,7 @@ import org.jacodb.ets.dto.EtsFileDto import org.jacodb.ets.dto.convertToEtsFile import org.jacodb.ets.model.EtsFile import org.jacodb.ets.model.EtsMethod +import org.jacodb.ets.model.EtsScene import org.jacodb.ets.utils.loadEtsFileAutoConvert import org.usvm.NoCoverage import org.usvm.PathSelectionStrategy @@ -29,7 +31,7 @@ typealias CoverageChecker = (TSMethodCoverage) -> Boolean open class TSMethodTestRunner : TestRunner() { - protected val globalClassName = "_DEFAULT_ARK_CLASS" + protected val globalClassName = DEFAULT_ARK_CLASS_NAME protected val doNotCheckCoverage: CoverageChecker = { _ -> true } @@ -205,10 +207,11 @@ open class TSMethodTestRunner : TestRunner + TSMachine(project, options).use { machine -> val states = machine.analyze(listOf(method)) states.map { state -> val resolver = TSTestResolver() diff --git a/usvm-util/src/main/kotlin/org/usvm/util/Logging.kt b/usvm-util/src/main/kotlin/org/usvm/util/Logging.kt index cd74322328..5ece6e5a1e 100644 --- a/usvm-util/src/main/kotlin/org/usvm/util/Logging.kt +++ b/usvm-util/src/main/kotlin/org/usvm/util/Logging.kt @@ -40,10 +40,10 @@ inline fun LoggerWithLogMethod.bracket( val startNano = System.nanoTime() var alreadyLogged = false - var res : Maybe = Maybe.empty() + var res: Maybe = Maybe.none() try { // Note: don't replace this one with runCatching, otherwise return from lambda breaks "finished" logging. - res = Maybe(block()) + res = Maybe.some(block()) return res.getOrThrow() } catch (t: Throwable) { logMethod { "Finished (in ${elapsedSecFrom(startNano)}): $msg :: EXCEPTION :: ${closingComment(Result.failure(t))}" } @@ -51,7 +51,7 @@ inline fun LoggerWithLogMethod.bracket( throw t } finally { if (!alreadyLogged) { - if (res.hasValue) + if (res.isSome) logMethod { "Finished (in ${elapsedSecFrom(startNano)}): $msg ${closingComment(Result.success(res.getOrThrow()))}" } else logMethod { "Finished (in ${elapsedSecFrom(startNano)}): $msg " } @@ -75,4 +75,4 @@ inline fun KLogger.logException(block: () -> T): T { this.error("Exception occurred", e) throw e } -} \ No newline at end of file +} diff --git a/usvm-util/src/main/kotlin/org/usvm/util/Maybe.kt b/usvm-util/src/main/kotlin/org/usvm/util/Maybe.kt index c630f4d4d1..3ee025572e 100644 --- a/usvm-util/src/main/kotlin/org/usvm/util/Maybe.kt +++ b/usvm-util/src/main/kotlin/org/usvm/util/Maybe.kt @@ -1,23 +1,51 @@ package org.usvm.util -/** - * Analogue of java's [java.util.Optional] - */ -class Maybe private constructor(val hasValue: Boolean, val value:T? ) { - constructor(v: T) : this(true, v) - companion object { - fun empty() = Maybe(false, null) +@JvmInline +value class Maybe private constructor( + private val rawValue: Any?, +) { + val isSome: Boolean get() = rawValue !== NONE_VALUE + val isNone: Boolean get() = rawValue === NONE_VALUE + + fun getOrThrow(): T { + check(isSome) { "Maybe is None" } + @Suppress("UNCHECKED_CAST") + return rawValue as T + } + + override fun toString(): String { + return if (isSome) "Some($rawValue)" else "None" } - /** - * Returns [value] if [hasValue]. Otherwise, throws exception - */ - @Suppress("UNCHECKED_CAST") - fun getOrThrow() : T = if (!hasValue) { - error("Maybe hasn't value") - } else { - value as T + companion object { + private val NONE_VALUE = object { + // Note: toString() for debugger + override fun toString(): String = "None" + } + private val NONE = Maybe(NONE_VALUE) + + fun none(): Maybe = NONE + + fun some(value: T): Maybe = Maybe(value) + + fun from(value: T?): Maybe = if (value == null) none() else some(value) } +} + +inline fun Maybe.map(body: (T) -> Maybe): Maybe = + if (isNone) Maybe.none() else body(getOrThrow()) + +inline fun Maybe.fmap(body: (T) -> R): Maybe = + if (isNone) Maybe.none() else Maybe.some(body(getOrThrow())) + +inline fun Maybe.onSome(body: (T) -> Unit): Maybe { + if (isSome) body(getOrThrow()) + return this +} + +inline fun Maybe.onNone(body: () -> Unit): Maybe { + if (isNone) body() + return this +} - override fun toString(): String = if (hasValue) "Maybe($value)" else "" -} \ No newline at end of file +fun T?.toMaybe(): Maybe = Maybe.from(this)