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

dev.tauri.choam.internal.mcas.Mcas.scala Maven / Gradle / Ivy

/*
 * SPDX-License-Identifier: Apache-2.0
 * Copyright 2016-2024 Daniel Urban and contributors listed in NOTICE.txt
 *
 * 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 dev.tauri.choam
package internal
package mcas

import java.util.concurrent.ThreadLocalRandom

/** Common interface for MCAS (i.e., k-CAS) implementations */
sealed trait Mcas {

  /** Returns the context associated with the current thread */
  def currentContext(): Mcas.ThreadContext

  /**
   * The same as `ctx eq currentContext()`, but
   * might be more efficient.
   */
  def isCurrentContext(ctx: Mcas.ThreadContext): Boolean = {
    // subclasses should override if
    // they can be more efficient:
    ctx eq currentContext()
  }

  /** True iff `this` can be used to perform concurrent ops on separate threads */
  private[choam] def isThreadSafe: Boolean

  /** Only for testing/benchmarking */
  private[choam] def getRetryStats(): Mcas.RetryStats = {
    // implementations should override if
    // they collect statistics
    Mcas.RetryStats.zero
  }

  /** Only for testing/benchmarking */
  private[choam] def collectExchangerStats(): Map[Long, Map[AnyRef, AnyRef]] = {
    // implementations should override if
    // they collect statistics
    Map.empty
  }

  /** Only for testing/benchmarking */
  private[choam] def maxReusedWeakRefs(): Int = {
    // implementations should override if
    // they collect statistics
    0
  }
}

object Mcas extends McasCompanionPlatform { self =>

  private[mcas] trait UnsealedMcas extends Mcas

  private[mcas] trait UnsealedThreadContext extends ThreadContext

  sealed trait ThreadContext {

    type START <: AbstractDescriptor.Aux[START]

    // abstract:

    /** The `Mcas` instance from which this context was retrieved */
    def impl: Mcas

    /**
     * Reads the global version, and
     * Starts building a descriptor
     * (its `.validTs` will be the
     * current global version).
     */
    def start(): AbstractDescriptor.Aux[START]

    /**
     * Subclasses may override to provide a more
     * efficient implementation than the default
     * `snapshot(start())`.
     */
    def startSnap(): Descriptor = {
      this.snapshot(this.start())
    }

    private[mcas] def addVersionCas(desc: AbstractDescriptor): AbstractDescriptor.Aux[desc.D]

    /**
     * @return the current value of `ref`, as
     *         if read by `readIntoHwd(ref).nv`.
     */
    def readDirect[A](ref: MemoryLocation[A]): A

    /**
     * @return the current value and version of `ref`
     *         in a word descriptor object. The
     *         value and version are guaranteed to be
     *         consistent with each other. The descriptor's
     *         `.ov` and `.nv` are guaranteed to be the same.
     */
    def readIntoHwd[A](ref: MemoryLocation[A]): LogEntry[A]

    /**
     * @return the current version of `ref`, as if
     *         read by `readIntoHwd(ref).version`.
     */
    private[choam] def readVersion[A](ref: MemoryLocation[A]): Long

    def validateAndTryExtend(
      desc: AbstractDescriptor,
      hwd: LogEntry[_], // may be null
    ): AbstractDescriptor.Aux[desc.D]

    /**
     * Directly tries to perform the k-CAS described by `desc`
     *
     * @param optimism Specify `Consts.OPTIMISTIC`, to try first
     *                 in optimistic mode; `Consts.PESSIMISTIC`,
     *                 to not even try being optimistic.
     * @return either `EmcasStatus.Successful` (if successful);
     *         `EmcasStatus.FailedVal` (if failed due to an
     *         expected value not matching); or the current
     *         global version (if failed due to the version
     *         being newer than `desc.validTs`). Can also return
     *         `Version.Reserved`, which has the same semantics as
     *         `FailedVal`, but hints that further retries should
     *         be `PESSIMISTIC`.
     */
    private[mcas] def tryPerformInternal(desc: AbstractDescriptor, optimism: Long): Long

    /** @return a `ThreadLocalRandom` valid for the current thread */
    def random: ThreadLocalRandom

    /** @return the `RefIdGen` of the current thread (or a thread-safe one) */
    def refIdGen: RefIdGen

    // concrete:

    /** Utility to first try to read from the log, and only from the ref if not found */
    final def readMaybeFromLog[A](ref: MemoryLocation[A], log: AbstractDescriptor): Option[(A, AbstractDescriptor.Aux[log.D])] = {
      log.getOrElseNull(ref) match {
        case null =>
          // not in log
          this.readIntoLog(ref, log) match {
            case null =>
              None // None means rollback needed
            case newLog =>
              val a: A = newLog.getOrElseNull(ref).nv
              Some((a, newLog))
          }
        case hwd =>
          // found in log
          Some((hwd.cast[A].nv, log : AbstractDescriptor.Aux[log.D]))
      }
    }

    private[this] final def readIntoLog[A](ref: MemoryLocation[A], log: AbstractDescriptor): AbstractDescriptor.Aux[log.D] = {
      require(log.getOrElseNull(ref) eq null)
      val hwd = this.readIntoHwd(ref)
      val newLog: AbstractDescriptor.Aux[log.D] = log.add(hwd)
      if (!newLog.isValidHwd(hwd)) {
        // this returns null if we need to roll back
        // (and we pass on the null to our caller):
        val res = this.validateAndTryExtend(newLog, hwd = null)
        if (res ne null) {
          assert(res.isValidHwd(hwd))
        }
        res
      } else {
        newLog
      }
    }

    /**
     * Tries to perform the ops in `desc`, with adding a
     * version-CAS (if not read-only).
     *
     * @return either `EmcasStatus.Successful` (if successful);
     *         `EmcasStatus.FailedVal` (if failed due to an
     *         expected value not matching); or the current
     *         global version (if failed due to the version
     *         being newer than `desc.validTs`).
     */
    final def tryPerform(desc: AbstractDescriptor): Long = {
      this.tryPerform(desc, Consts.OPTIMISTIC)
    }

    final def tryPerform(desc: AbstractDescriptor, optimism: Long): Long = {
      if (desc.readOnly) {
        // we've validated each read,
        // so nothing to do here
        McasStatus.Successful
      } else {
        val finalDesc = this.addVersionCas(desc)
        val res = this.tryPerformInternal(finalDesc, optimism = optimism)
        assert((res == McasStatus.Successful) || (res == McasStatus.FailedVal) || (res == Version.Reserved) || Version.isValid(res))
        res
      }
    }

    final def tryPerformOk(desc: AbstractDescriptor): Boolean = {
      this.tryPerformOk(desc, Consts.OPTIMISTIC)
    }

    /** Like `tryPerform`, but returns whether it was successful */
    final def tryPerformOk(desc: AbstractDescriptor, optimism: Long): Boolean = {
      tryPerform(desc, optimism = optimism) == McasStatus.Successful
    }

    final def addCasFromInitial[A](desc: AbstractDescriptor, ref: MemoryLocation[A], ov: A, nv: A): AbstractDescriptor.Aux[desc.D] =
      this.addCasWithVersion(desc, ref, ov = ov, nv = nv, version = Version.Start)

    final def addCasWithVersion[A](
      desc: AbstractDescriptor,
      ref: MemoryLocation[A],
      ov: A,
      nv: A,
      version: Long
    ): AbstractDescriptor.Aux[desc.D] = {
      val wd = LogEntry(ref, ov, nv, version)
      desc.add(wd)
    }

    /**
     * Tries to (re)validate `desc` based on the current
     * versions of the refs it contains.
     *
     * @return true, iff `desc` is still valid.
     */
    private[mcas] final def validate(desc: AbstractDescriptor): Boolean = {
      desc.revalidate(this)
    }

    private[mcas] final def validateHwd[A](hwd: LogEntry[A]): Boolean = {
      hwd.revalidate(this)
    }

    /**
     * @return a snapshot of `desc`.
     */
    final def snapshot(desc: AbstractDescriptor): Descriptor =
      desc.toImmutable

    /**
     * Merges disjoint descriptors `to` and `from`.
     *
     * @return The merged descriptor, which contains
     *         all the ops either in `to` or `from`.
     */
    private[choam] final def addAll(to: Descriptor, from: Descriptor): Descriptor = {
      Descriptor.merge(to, from, this)
    }

    /**
     * Tries to perform a "bare" 1-CAS, without
     * changing the global version. This breaks
     * opacity guarantees! It may change the value
     * of a ref without changing its version!
     */
    final def singleCasDirect[A](ref: MemoryLocation[A], ov: A, nv: A): Boolean = {
      assert(!equ(ov, nv))
      val hwd = this.readIntoHwd(ref)
      val d0 = this.start() // do this after reading, so version is deemed valid
      assert(d0.isValidHwd(hwd))
      if (equ(hwd.ov, ov)) {
        // create a (dangerous) descriptor, which will set
        // the new version to `validTs` (which may equal the
        // old version):
        val d1 = d0.add(hwd.withNv(nv)).withNoNewVersion
        assert(d1.newVersion == d1.validTs)
        // we're intentionally NOT having a version-CAS:
        this.tryPerformInternal(d1, optimism = Consts.PESSIMISTIC) == McasStatus.Successful
      } else {
        false
      }
    }

    /**
     * Tries to perform a 1-CAS, but
     * also handles versions. Equivalent
     * to creating a descriptor containing a
     * single op, and calling `tryPerformOk`
     * on it (but may be more efficient).
     */
    final def tryPerformSingleCas[A](ref: MemoryLocation[A], ov: A, nv: A): Boolean = {
      // TODO: this could be optimized (probably)
      val d0 = this.start()
      val d1 = this.readIntoLog(ref, d0)
      assert(d1 ne null)
      val hwd = d1.getOrElseNull(ref)
      assert(hwd ne null)
      if (equ(hwd.ov, ov)) {
        val d2 = d1.overwrite(hwd.withNv(nv))
        this.tryPerform(d2, optimism = Consts.PESSIMISTIC) == McasStatus.Successful
      } else {
        false
      }
    }

    // statistics/testing/benchmarking:

    /** Only for testing */
    private[mcas] final def builder(): Builder = {
      new Builder(this, this.start())
    }

    /** Only for testing/benchmarking */
    private[choam] def recordCommit(@unused retries: Int, @unused committedRefs: Int, @unused descExtensions: Int): Unit = {
      // we ignore stats by default; implementations
      // can override if it matters
      ()
    }

    /** Only for testing/benchmarking */
    private[choam] def supportsStatistics: Boolean = {
      // we ignore stats by default; implementations
      // can override if it matters
      false
    }

    /** Only for testing/benchmarking */
    private[choam] def getStatisticsP(): Map[AnyRef, AnyRef] = {
      // we ignore stats by default; implementations
      // can override if it matters
      Map.empty
    }

    /** Only for testing/benchmarking */
    private[choam] def getStatisticsO(): Map[AnyRef, AnyRef] = {
      // we ignore stats by default; implementations
      // can override if it matters
      Map.empty
    }

    /** Only for testing/benchmarking */
    private[choam] def setStatisticsP(@unused stats: Map[AnyRef, AnyRef]): Unit = {
      // we ignore stats by default; implementations
      // can override if it matters
    }

    /** Only for testing/benchmarking */
    private[choam] def maxReusedWeakRefs(): Int = {
      0
    }
  }

  private[choam] final case class RetryStats(
    /** The number of successfully committed `Rxn`s */
    commits: Long,
    /** The number of retries overall */
    retries: Long,
    /** The number of times a `Descriptor` was revalidated and extended successfully */
    extensions: Long,
    /**
     * The number of internal `Mcas` attempts (can be more
     * than the `tryPerform` calls, e.g., `Emcas` makes up to
     * two internal attempts).
     */
    mcasAttempts: Long,
    /** The sum of the number of `Ref`s the committed `Rxn`s touched */
    committedRefs: Long,
    /** The number of times cycles were detected when helping */
    cyclesDetected: Long,
    /** The highest number of retries one `Rxn` had to perform */
    maxRetries: Long,
    /** The size (touched `Ref`s) of the biggest `Rxn` that committed */
    maxCommittedRefs: Int,
    /** The (estimated) maximum observed size of the cycle detection Bloom filter */
    maxBloomFilterSize: Int,
  ) {

    final def + (that: RetryStats): RetryStats = {
      RetryStats(
        commits = this.commits + that.commits,
        retries = this.retries + that.retries,
        extensions = this.extensions + that.extensions,
        mcasAttempts = this.mcasAttempts + that.mcasAttempts,
        committedRefs = this.committedRefs + that.committedRefs,
        cyclesDetected = this.cyclesDetected + that.cyclesDetected,
        maxRetries = java.lang.Math.max(this.maxRetries, that.maxRetries),
        maxCommittedRefs = java.lang.Math.max(this.maxCommittedRefs, that.maxCommittedRefs),
        maxBloomFilterSize = java.lang.Math.max(this.maxBloomFilterSize, that.maxBloomFilterSize),
      )
    }

    /** Delta between `this` and an _earlier_ stats instance `that` */
    final def - (that: RetryStats): RetryStats = {
      require(
        (that.maxRetries <= this.maxRetries) &&
        (that.maxCommittedRefs <= this.maxCommittedRefs) &&
        (that.maxBloomFilterSize <= this.maxBloomFilterSize)
      )
      RetryStats(
        commits = this.commits - that.commits,
        retries = this.retries - that.retries,
        extensions = this.extensions - that.extensions,
        mcasAttempts = this.mcasAttempts - that.mcasAttempts,
        committedRefs = this.committedRefs - that.committedRefs,
        cyclesDetected = this.cyclesDetected - that.cyclesDetected,
        maxRetries = this.maxRetries,
        maxCommittedRefs = this.maxCommittedRefs,
        maxBloomFilterSize = this.maxBloomFilterSize,
      )
    }

    final def avgRetriesPerCommit: Double = {
      val c = this.commits.toDouble
      if (c != 0.0) {
        val r = this.retries
        r.toDouble / c
      } else {
        Double.NaN
      }
    }

    final def avgTriesPerCommit: Double = {
      val c = this.commits.toDouble
      if (c != 0.0) {
        val r = this.retries
        (r.toDouble + c) / c
      } else {
        Double.NaN
      }
    }

    final def avgExtensionsPerCommit: Double = {
      val c = this.commits.toDouble
      if (c != 0.0) {
        val e = this.extensions.toDouble
        e / c
      } else {
        Double.NaN
      }
    }

    final def avgCyclesPerMcasAttempt: Double = {
      val a = this.mcasAttempts.toDouble
      if (a != 0.0) {
        val cd = this.cyclesDetected.toDouble
        cd / a
      } else {
        Double.NaN
      }
    }

    final def avgLogSize: Double = {
      val c = this.commits.toDouble
      if (c != 0.0) {
        val allCommittedRefs = this.committedRefs.toDouble
        allCommittedRefs / c
      } else {
        Double.NaN
      }
    }

    final def maxLogSize: Int = {
      this.maxCommittedRefs
    }
  }

  private[mcas] final object RetryStats {
    val zero: RetryStats =
      RetryStats(0L, 0L, 0L, 0L, 0L, 0L, 0L, 0, 0)
  }

  /** Only for testing */
  private[mcas] final class Builder(
    private[this] val ctx: ThreadContext,
    private[this] val desc: AbstractDescriptor,
  ) {

    final def updateRef[A](ref: MemoryLocation[A], f: A => A): Builder = {
      this.ctx.readMaybeFromLog(ref, this.desc) match {
        case Some((ov, newDesc)) =>
          val nv = f(ov)
          val newestDesc = newDesc.overwrite(
            newDesc.getOrElseNull(ref).withNv(nv)
          )
          new Builder(this.ctx, newestDesc)
        case None =>
          throw new IllegalStateException("couldn't extend, rollback is necessary")
      }
    }

    final def casRef[A](ref: MemoryLocation[A], from: A, to: A): Builder = {
      this.tryCasRef(ref, from, to).getOrElse(
        throw new IllegalStateException("couldn't extend, rollback is necessary")
      )
    }

    final def tryCasRef[A](ref: MemoryLocation[A], from: A, to: A): Option[Builder] = {
      this.ctx.readMaybeFromLog(ref, this.desc).map {
        case (ov, newDesc) =>
          val newestDesc = if (equ(ov, from)) {
            newDesc.overwrite(newDesc.getOrElseNull(ref).withNv(to))
          } else {
            val oldHwd = newDesc.getOrElseNull(ref)
            val newHwd = LogEntry(ref, from, to, oldHwd.version)
            newDesc.overwrite(newHwd)
          }
          new Builder(this.ctx, newestDesc)
      }
    }

    final def tryPerformOk(): Boolean = {
      this.tryPerformOk(Consts.OPTIMISTIC)
    }

    final def tryPerformOk(optimism: Long): Boolean = {
      this.ctx.tryPerformOk(this.desc, optimism)
    }
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy