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

dev.tauri.choam.internal.mcas.AbstractHamt.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 scala.util.hashing.MurmurHash3
import scala.collection.AbstractIterator

private[mcas] abstract class AbstractHamt[K <: Hamt.HasHash, V <: Hamt.HasKey[K], E, T1, T2, H <: AbstractHamt[K, V, E, T1, T2, H]] protected[mcas] () { self: H =>

  protected def newArray(size: Int): Array[E]

  protected def convertForArray(a: V, tok: T1, flag: Boolean): E

  protected def predicateForForAll(a: V, tok: T2): Boolean

  protected def isBlue(a: V): Boolean

  def size: Int

  protected def contentsArr: Array[AnyRef]

  protected def insertInternal(v: V): H

  /**
   * Evaluates `predicateForForAll` (implemented
   * in a subclass) for the values, short-circuits
   * on `false`.
   */
  final def forAll(tok: T2): Boolean = {
    this.forAllInternal(tok)
  }

  /** Only call on the root! */
  final def valuesIterator: Iterator[V] = {
    new AbstractIterator[V] {

      private[this] val arrays =
        new Array[Array[AnyRef]](11)

      private[this] val indices =
        new Array[Int](11)

      private[this] var depth: Int =
        0

      private[this] var loadedNext: V =
        nullOf[V]

      locally {
        this.arrays(0) = self.contentsArr
        this.indices(0) = -1
        this.loadNext()
      }

      final override def hasNext: Boolean = {
        !isNull(this.loadedNext)
      }

      final override def next(): V = {
        if (this.hasNext) {
          val res = this.loadedNext
          this.loadedNext = nullOf[V]
          this.loadNext()
          res
        } else {
          throw new NoSuchElementException
        }
      }

      @tailrec
      private[this] final def loadNext(): Unit = {
        val d = this.depth
        val idx = this.indices(d) + 1
        this.indices(d) = idx
        val arr = this.arrays(d)
        if (idx < arr.length) {
          arr(idx) match {
            case null =>
              // skip empty slot:
              this.loadNext()
            case node: AbstractHamt[_, _, _, _, _, _] =>
              // descend:
              val newDepth = d + 1
              this.depth = newDepth
              this.indices(newDepth) = -1
              this.arrays(newDepth) = node.contentsArr
              this.loadNext()
            case a =>
              // found it:
              this.loadedNext = a.asInstanceOf[V]
          }
        } else {
          // ascend:
          val newDepth = d - 1
          this.depth = newDepth
          if (newDepth >= 0) {
            this.loadNext()
          } // else: at end
        }
      }
    }
  }

  final def toString(pre: String, post: String): String = {
    val sb = new java.lang.StringBuilder(pre)
    val _ = this.toStringInternal(sb, first = true)
    sb.append(post)
    sb.toString()
  }

  protected final def copyToArrayInternal(tok: T1, flag: Boolean, nullIfBlue: Boolean): Array[E] = {
    val arr = this.newArray(this.size)
    val endAndBlue = this.copyIntoArray(arr, 0, tok, flag = flag)
    assert(unpackSize(endAndBlue) == arr.length)
    if (nullIfBlue && unpackBlue(endAndBlue)) {
      null
    } else {
      arr
    }
  }

  private final def copyIntoArray(arr: Array[E], start: Int, tok: T1, flag: Boolean): Int = {
    val contents = this.contentsArr
    var i = 0
    var arrIdx = start
    var isBlueSt = true
    val len = contents.length
    while (i < len) {
      contents(i) match {
        case null =>
          ()
        case node: AbstractHamt[_, _, _, _, _, _] =>
          val arrIdxAndBlue = node.asInstanceOf[H].copyIntoArray(arr, arrIdx, tok, flag = flag)
          arrIdx = unpackSize(arrIdxAndBlue)
          isBlueSt &= unpackBlue(arrIdxAndBlue)
        case a =>
          // temporary assertion to diagnose a bug here:
          if (arrIdx >= arr.length) {
            throw new AssertionError(
              s"indexing array of length ${arr.length} with index ${arrIdx} (" +
              s"a = ${a}; arr = ${arr.mkString("[", ", ", "]")}; " +
              s"contents = ${contents.mkString("[", ", ", "]")})"
            )
          }
          // end of temporary assertion
          arr(arrIdx) = convertForArray(a.asInstanceOf[V], tok, flag = flag)
          isBlueSt &= isBlue(a.asInstanceOf[V])
          arrIdx += 1
      }
      i += 1
    }
    packSizeAndBlue(arrIdx, isBlueSt)
  }

  protected final def insertIntoHamt(into: AbstractHamt[_, _, _, _, _, _]): H = {
    val contents = this.contentsArr
    var i = 0
    var curr = into
    val len = contents.length
    while (i < len) {
      contents(i) match {
        case null =>
          ()
        case node: AbstractHamt[_, _, _, _, _, _] =>
          curr = node.insertIntoHamt(curr)
        case a =>
          curr = curr.asInstanceOf[H].insertInternal(a.asInstanceOf[V])
      }
      i += 1
    }
    curr.asInstanceOf[H]
  }

  private final def forAllInternal(tok: T2): Boolean = {
    val contents = this.contentsArr
    var i = 0
    val len = contents.length
    while (i < len) {
      contents(i) match {
        case null =>
          ()
        case node: AbstractHamt[_, _, _, _, _, _] =>
          if (!node.asInstanceOf[H].forAllInternal(tok)) {
            return false // scalafix:ok
          }
        case a =>
          if (!this.predicateForForAll(a.asInstanceOf[V], tok)) {
            return false // scalafix:ok
          }
      }
      i += 1
    }

    true
  }

  protected def equalsInternal(that: AbstractHamt[_, _, _, _, _, _]): Boolean = {
    // Insertions are not order-dependent, and
    // there is no deletion, so HAMTs with the
    // same values always have the same tree
    // structure. So we can just traverse the
    // 2 trees at the same time.
    val thisContents = this.contentsArr
    val thatContents = that.contentsArr
    val thisLen = thisContents.length
    val thatLen = thatContents.length
    if (thisLen == thatLen) {
      var i = 0
      while (i < thisLen) {
        val iOk = thisContents(i) match {
          case null =>
            thatContents(i) eq null
          case thisNode: AbstractHamt[_, _, _, _, _, _] =>
            thatContents(i) match {
              case thatNode: AbstractHamt[_, _, _, _, _, _] =>
                thisNode.equalsInternal(thatNode)
              case _ => // including null
                false
            }
          case thisValue =>
            thatContents(i) match {
              case null | (_: Hamt[_, _, _, _, _, _]) =>
                false
              case thatValue =>
                thisValue == thatValue
            }
        }
        if (iOk) {
          i += 1
        } else {
          return false // scalafix:ok
        }
      }
      true
    } else {
      false
    }
  }

  protected final def hashCodeInternal(s: Int): Int = {
    val contents = this.contentsArr
    var i = 0
    var curr = s
    val len = contents.length
    while (i < len) {
      contents(i) match {
        case null =>
          ()
        case node: AbstractHamt[_, _, _, _, _, _] =>
          curr = node.hashCodeInternal(curr)
        case a =>
          curr = MurmurHash3.mix(curr, (a.asInstanceOf[V].key.hash >>> 32).toInt)
          curr = MurmurHash3.mix(curr, a.##)
      }
      i += 1
    }
    curr
  }

  private final def toStringInternal(sb: java.lang.StringBuilder, first: Boolean): Boolean = {
    val contents = this.contentsArr
    var i = 0
    val len = contents.length
    var fst = first
    while (i < len) {
      contents(i) match {
        case null =>
          ()
        case node: AbstractHamt[_, _, _, _, _, _] =>
          fst = node.toStringInternal(sb, fst)
        case a =>
          if (!fst) {
            sb.append(", ")
          } else {
            fst = false
          }
          sb.append(a.toString)
      }
      i += 1
    }
    fst
  }

  // TODO: this is duplicated with `Hamt`
  protected[this] final def packSizeAndBlue(size: Int, isBlue: Boolean): Int = {
    val x = (-1) * java.lang.Math.abs(java.lang.Boolean.compare(isBlue, true))
    size * ((x << 1) + 1)
  }

  @inline
  protected[this] final def unpackSize(sb: Int): Int = {
    java.lang.Math.abs(sb)
  }

  @inline
  protected[this] final def unpackBlue(sb: Int): Boolean = {
    sb >= 0
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy