All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.jetbrains.kotlin.js.inline.clean.EmptyStatementElimination.kt Maven / Gradle / Ivy

There is a newer version: 2.1.0-Beta1
Show newest version
/*
 * Copyright 2010-2017 JetBrains s.r.o.
 *
 * 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.jetbrains.kotlin.js.inline.clean

import org.jetbrains.kotlin.js.backend.ast.*
import org.jetbrains.kotlin.js.backend.ast.metadata.synthetic
import org.jetbrains.kotlin.js.translate.utils.JsAstUtils

internal class EmptyStatementElimination(private val root: JsStatement) {
    private var hasChanges = false

    fun apply(): Boolean {
        object : JsVisitorWithContextImpl() {
            override fun visit(x: JsFunction, ctx: JsContext<*>) = false

            override fun endVisit(x: JsLabel, ctx: JsContext) {
                if (x.synthetic) {
                    if (isEmpty(x.statement)) {
                        ctx.replaceMe(x.statement)
                        hasChanges = true
                    }
                }
            }

            override fun endVisit(x: JsBlock, ctx: JsContext<*>) {
                processStatements(x.statements)
            }

            override fun endVisit(x: JsIf, ctx: JsContext) {
                val thenEmpty = isEmpty(x.thenStatement)
                val elseEmpty = x.elseStatement?.let { isEmpty(it) } ?: true
                when {
                    thenEmpty && elseEmpty -> {
                        hasChanges = true
                        ctx.replaceMe(JsAstUtils.asSyntheticStatement(x.ifExpression))
                    }
                    elseEmpty -> {
                        if (x.elseStatement != null) {
                            hasChanges = true
                            x.elseStatement = null
                        }
                    }
                    thenEmpty -> {
                        hasChanges = true
                        x.thenStatement = x.elseStatement!!
                        x.elseStatement = null
                        x.ifExpression = JsAstUtils.notOptimized(x.ifExpression)
                    }
                }
            }

            override fun endVisit(x: JsTry, ctx: JsContext) {
                val finallyBlock = x.finallyBlock
                if (x.tryBlock.isEmpty) {
                    hasChanges = true
                    ctx.replaceMe(finallyBlock ?: JsEmpty)
                }
            }

            override fun endVisit(x: JsSwitch, ctx: JsContext) {
                for (case in x.cases) {
                    processStatements(case.statements)
                }
                if (x.cases.lastOrNull() is JsDefault && x.cases.dropLast(1).all { it.statements.isEmpty() }
                    || x.cases.all { it.statements.isEmpty() }
                ) {
                    hasChanges = true
                    val replacement = mutableListOf(JsAstUtils.asSyntheticStatement(x.expression))
                    x.cases.lastOrNull()?.apply { replacement.addAll(statements) }
                    ctx.replaceMe(JsBlock(replacement))
                }
            }

            private fun processStatements(statements: MutableList) {
                for ((index, statement) in statements.withIndex().reversed()) {
                    if (statement is JsEmpty) {
                        statements.removeAt(index)
                        hasChanges = true
                    }
                    else if (statement is JsBlock) {
                        statements.removeAt(index)
                        statements.addAll(index, statement.statements)
                    }
                }
            }

            private fun isEmpty(statement: JsStatement) = statement is JsBlock && statement.isEmpty || statement is JsEmpty
        }.accept(root)
        return hasChanges
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy