/* ----------------------------------------------------------------------------- Copyright 2021,2023 Kevin P. Barry 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. ----------------------------------------------------------------------------- */ // Author: Kevin P. Barry [ta0kira@gmail.com] define HashedMap { @value Int itemCount @value [DefaultOrder> & ReadAt>] table new () { return #self{ 0, Vector:createSize>($ExprLookup[HASH_TABLE_MIN_SIZE]$) } } default () { return new() } duplicate () { return #self{ itemCount, table.defaultOrder() `OrderH:duplicateTo` Vector>.new() } } size () { return itemCount } member (k) { return `present` delegate -> `get` } get (k) { return table.readAt(getBin(k.hashed())).find(k) } set (k, v) { \ swap(k, v) return self } remove (k) { \ swap(k, empty) return self } weakSet (k, v) { optional #v value <- table.readAt(getBin(k.hashed())).weakReplace(k, v) if (`present` value) { return `require` value } else { itemCount <- itemCount+1 \ maybeResize() return v } } swap (k, v) (old) { old <- table.readAt(getBin(k.hashed())).exchange(k, v) if (!present(v) && present(old)) { itemCount <- itemCount-1 \ maybeResize() } elif (present(v) && !present(old)) { itemCount <- itemCount+1 \ maybeResize() } } defaultOrder () { return HashedMapOrder<#k, #v>.new(itemCount, table) } keyOrder () { return MapKeyOrder:new(defaultOrder()) } valueOrder () { return MapValueOrder:new(defaultOrder()) } @value getBin (Int) -> (Int) getBin (hash) { if (hash < 0) { return -hash % table.size() } else { return hash % table.size() } } @value maybeResize () -> () maybeResize () { optional [DefaultOrder> & ReadAt>] newTable <- empty if (itemCount >= $ExprLookup[HASH_TABLE_EXPAND_RATIO]$*table.size()) { newTable <- Vector:createSize>(2*table.size()) } elif ($ExprLookup[HASH_TABLE_CONTRACT_RATIO]$*itemCount < table.size() && table.size() > $ExprLookup[HASH_TABLE_MIN_SIZE]$) { newTable <- Vector:createSize>(table.size()/2) } if (`present` newTable) { [DefaultOrder> & ReadAt>] oldTable <- table table <- `require` newTable $Hidden[newTable]$ traverse (oldTable.defaultOrder() -> HashRoot<#k, #v> root) { traverse (root.defaultOrder() -> HashNode<#k, #v> node) { \ table.readAt(getBin(node.getKey().hashed())).exchange(node.getKey(), node.getValue()) } } } } } concrete HashNode<#k, #v> { refines Duplicate refines Order<#self> @type new (#k, #v, optional HashNode<#k, #v>) -> (#self) @value getKey () -> (#k) @value getValue () -> (#v) @value setValue (#v) -> () @value setNext (optional HashNode<#k, #v>) -> (optional #self) } define HashNode { $ReadOnly[key]$ @value #k key @value #v value @value optional HashNode<#k, #v> next new (key, value, next) { return delegate -> #self } duplicate () { optional HashNode<#k, #v> next2 <- empty if (`present` next) { next2 <- require(next).duplicate() } return #self{ key, value, next2 } } getKey () { return key } getValue () { return value } setValue (value2) { value <- value2 } get () { return self } next () { return next } setNext (next2) { cleanup { next <- next2 } in return next } } concrete HashRoot<#k, #v> { defines Default refines DefaultOrder> refines Duplicate #k defines Equals<#k> @value exchange (#k, optional #v) -> (optional #v) @value weakReplace (#k, #v) -> (optional #v) @value find (#k) -> (optional #v) // Override to allow returning HashNode directly. @value defaultOrder () -> (optional HashNode<#k, #v>) } define HashRoot { @value optional HashNode<#k, #v> root default () { return #self{ empty } } defaultOrder () { return root } duplicate () { if (`present` root) { return #self{ require(root).duplicate() } } else { return #self{ empty } } } exchange (k, v) (old) { old <- empty scoped { optional HashNode<#k, #v> prev <- empty } in traverse (root -> HashNode<#k, #v> node) { if (node.getKey() `#k.equals` k) { old <- node.getValue() if (`present` v) { \ node.setValue(`require` v) } else { optional HashNode<#k, #v> next <- node.setNext(empty) if (`present` prev) { \ require(prev).setNext(next) } else { root <- next } } return _ } } update { prev <- node } if (`present` v) { root <- HashNode<#k, #v>.new(k, `require` v, root) } } weakReplace (k, v) { traverse (root -> HashNode<#k, #v> node) { if (node.getKey() `#k.equals` k) { return node.getValue() } } root <- HashNode<#k, #v>.new(k, v, root) return empty } find (k) (v) { v <- empty traverse (root -> HashNode<#k, #v> node) { if (node.getKey() `#k.equals` k) { return node.getValue() } } } } concrete HashedMapOrder<#k, #v> { refines Order> @type new (Int, DefaultOrder>) -> (optional #self) } define HashedMapOrder { @value Int count @value optional Order> root @value HashNode<#k, #v> node new (count, table) (next) { next <- empty if (count > 0) { scoped { optional Order> root, optional HashNode<#k, #v> node <- seekNext(table.defaultOrder()) } in if (`present` node) { next <- #self{ count, root, `require` node } } } } get () { return SimpleKeyValue:new(node.getKey(), node.getValue()) } next () (next) { next <- empty if ((count <- count-1) > 0) { scoped { optional HashNode<#k, #v> newNode <- node.next() } in if (`present` newNode) { node <- `require` newNode return self } scoped { root, optional HashNode<#k, #v> newNode <- seekNext(root) } in if (`present` newNode) { node <- `require` newNode return self } } } @type seekNext (optional Order>) -> (optional Order>, optional HashNode<#k, #v>) seekNext (start) (root, node) { root <- start node <- empty $Hidden[start]$ while (`present` root) { node <- require(root).get().defaultOrder() root <- require(root).next() if (`present` node) { return _ } } } }