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

org.jetbrains.kotlin.contracts.description.EventOccurrencesRange.kt Maven / Gradle / Ivy

There is a newer version: 2.1.0-RC
Show newest version
/*
 * Copyright 2010-2020 JetBrains s.r.o. and Kotlin Programming Language contributors.
 * Use of this source code is governed by the Apache 2.0 license that can be found in the license/LICENSE.txt file.
 */

package org.jetbrains.kotlin.contracts.description

import kotlin.math.max
import kotlin.math.min

enum class EventOccurrencesRange(private val left: Int, private val right: Int) {
    ZERO(0, 0),          // 0..0
    AT_MOST_ONCE(0, 1),  // 0..1
    EXACTLY_ONCE(1, 1),  // 1..1
    AT_LEAST_ONCE(1, 3), // 1..*
    MORE_THAN_ONCE(2, 3), // 2..*
    UNKNOWN(0, 3);       // 0..*

    companion object {
        private fun fromRange(left: Int, right: Int): EventOccurrencesRange = when (min(left, 2) to min(right, 3)) {
            0 to 0 -> ZERO
            0 to 1 -> AT_MOST_ONCE
            0 to 2 -> UNKNOWN
            0 to 3 -> UNKNOWN
            1 to 1 -> EXACTLY_ONCE
            1 to 2 -> AT_LEAST_ONCE
            1 to 3 -> AT_LEAST_ONCE
            2 to 2 -> MORE_THAN_ONCE
            2 to 3 -> MORE_THAN_ONCE
            3 to 3 -> MORE_THAN_ONCE
            else -> throw IllegalArgumentException()
        }

        fun or(x: EventOccurrencesRange, y: EventOccurrencesRange): EventOccurrencesRange =
            fromRange(min(x.left, y.left), max(x.right, y.right))

        fun plus(x: EventOccurrencesRange, y: EventOccurrencesRange): EventOccurrencesRange =
            fromRange(x.left + y.left, x.right + y.right)
    }

    infix fun or(other: EventOccurrencesRange): EventOccurrencesRange = Companion.or(this, other)
    operator fun plus(other: EventOccurrencesRange): EventOccurrencesRange = Companion.plus(this, other)
    operator fun contains(other: EventOccurrencesRange): Boolean = left <= other.left && other.right <= right

    fun  at(marker: D?): MarkedEventOccurrencesRange =
        when (this) {
            ZERO -> MarkedEventOccurrencesRange.Zero
            AT_MOST_ONCE -> MarkedEventOccurrencesRange.AtMostOnce(marker ?: throw AssertionError("AT_MOST_ONCE event requires location"))
            EXACTLY_ONCE -> MarkedEventOccurrencesRange.ExactlyOnce(marker ?: throw AssertionError("EXACTLY_ONCE event requires location"))
            AT_LEAST_ONCE -> MarkedEventOccurrencesRange.AtLeastOnce
            MORE_THAN_ONCE -> MarkedEventOccurrencesRange.MoreThanOnce
            UNKNOWN -> MarkedEventOccurrencesRange.Unknown
        }
}

// Extended version of `EventOccurrencesRange` that, for events that can only happen once,
// also carries the event location.
sealed class MarkedEventOccurrencesRange {
    open val location: D?
        get() = null

    data object Zero : MarkedEventOccurrencesRange()
    data class AtMostOnce(override val location: D) : MarkedEventOccurrencesRange()
    data class ExactlyOnce(override val location: D) : MarkedEventOccurrencesRange()
    data object AtLeastOnce : MarkedEventOccurrencesRange()
    data object MoreThanOnce : MarkedEventOccurrencesRange()
    data object Unknown : MarkedEventOccurrencesRange()

    val withoutMarker: EventOccurrencesRange
        get() = when (this) {
            Zero -> EventOccurrencesRange.ZERO
            is AtMostOnce -> EventOccurrencesRange.AT_MOST_ONCE
            is ExactlyOnce -> EventOccurrencesRange.EXACTLY_ONCE
            AtLeastOnce -> EventOccurrencesRange.AT_LEAST_ONCE
            MoreThanOnce -> EventOccurrencesRange.MORE_THAN_ONCE
            Unknown -> EventOccurrencesRange.UNKNOWN
        }
}

fun EventOccurrencesRange.isDefinitelyVisited(): Boolean =
    this == EventOccurrencesRange.EXACTLY_ONCE || this == EventOccurrencesRange.AT_LEAST_ONCE || this == EventOccurrencesRange.MORE_THAN_ONCE

fun EventOccurrencesRange.canBeVisited(): Boolean =
    this != EventOccurrencesRange.ZERO

fun EventOccurrencesRange.canBeRevisited(): Boolean =
    this == EventOccurrencesRange.UNKNOWN || this == EventOccurrencesRange.AT_LEAST_ONCE || this == EventOccurrencesRange.MORE_THAN_ONCE

fun MarkedEventOccurrencesRange<*>.isDefinitelyVisited(): Boolean =
    withoutMarker.isDefinitelyVisited()

fun MarkedEventOccurrencesRange<*>.canBeVisited(): Boolean =
    withoutMarker.canBeVisited()

fun MarkedEventOccurrencesRange<*>.canBeRevisited(): Boolean =
    withoutMarker.canBeRevisited()

val EventOccurrencesRange?.isInPlace: Boolean
    get() = this != null




© 2015 - 2024 Weber Informatics LLC | Privacy Policy