2008-02-19 18:06:52 +00:00
|
|
|
/* Copyright (c) 2008, Avian Contributors
|
|
|
|
|
|
|
|
Permission to use, copy, modify, and/or distribute this software
|
|
|
|
for any purpose with or without fee is hereby granted, provided
|
|
|
|
that the above copyright notice and this permission notice appear
|
|
|
|
in all copies.
|
|
|
|
|
|
|
|
There is NO WARRANTY for this software. See license.txt for
|
|
|
|
details. */
|
|
|
|
|
2007-11-26 23:15:53 +00:00
|
|
|
#include "util.h"
|
|
|
|
|
2007-11-27 01:40:47 +00:00
|
|
|
using namespace vm;
|
|
|
|
|
2008-04-07 23:47:41 +00:00
|
|
|
namespace {
|
|
|
|
|
|
|
|
inline object
|
|
|
|
getTreeNodeValue(Thread*, object n)
|
|
|
|
{
|
|
|
|
return reinterpret_cast<object>
|
|
|
|
(cast<intptr_t>(n, TreeNodeValue) & PointerMask);
|
|
|
|
}
|
|
|
|
|
|
|
|
inline void
|
|
|
|
setTreeNodeValue(Thread* t, object n, object value)
|
|
|
|
{
|
|
|
|
intptr_t red = cast<intptr_t>(n, TreeNodeValue) & (~PointerMask);
|
|
|
|
|
|
|
|
set(t, n, TreeNodeValue, value);
|
|
|
|
|
|
|
|
cast<intptr_t>(n, TreeNodeValue) |= red;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline bool
|
|
|
|
treeNodeRed(Thread*, object n)
|
|
|
|
{
|
2008-04-08 00:00:59 +00:00
|
|
|
return (cast<intptr_t>(n, TreeNodeValue) & (~PointerMask)) == 1;
|
2008-04-07 23:47:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
inline void
|
2008-04-08 00:14:59 +00:00
|
|
|
setTreeNodeRed(Thread*, object n, bool red)
|
2008-04-07 23:47:41 +00:00
|
|
|
{
|
2008-04-08 00:14:59 +00:00
|
|
|
if (red) {
|
|
|
|
cast<intptr_t>(n, TreeNodeValue) |= 1;
|
|
|
|
} else {
|
|
|
|
cast<intptr_t>(n, TreeNodeValue) &= PointerMask;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
inline object
|
|
|
|
cloneTreeNode(Thread* t, object n)
|
|
|
|
{
|
2008-04-11 23:40:36 +00:00
|
|
|
PROTECT(t, n);
|
|
|
|
|
2008-04-08 00:14:59 +00:00
|
|
|
object newNode = makeTreeNode
|
|
|
|
(t, getTreeNodeValue(t, n), treeNodeLeft(t, n), treeNodeRight(t, n));
|
|
|
|
setTreeNodeRed(t, newNode, treeNodeRed(t, n));
|
|
|
|
return newNode;
|
2008-04-07 23:47:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
object
|
2008-07-12 04:01:42 +00:00
|
|
|
treeFind(Thread* t, object old, intptr_t key, object node, object sentinal,
|
|
|
|
intptr_t (*compare)(Thread* t, intptr_t key, object b))
|
2008-04-07 23:47:41 +00:00
|
|
|
{
|
|
|
|
PROTECT(t, old);
|
|
|
|
PROTECT(t, node);
|
|
|
|
PROTECT(t, sentinal);
|
|
|
|
|
|
|
|
object newRoot = cloneTreeNode(t, old);
|
|
|
|
PROTECT(t, newRoot);
|
|
|
|
|
|
|
|
object new_ = newRoot;
|
|
|
|
PROTECT(t, new_);
|
|
|
|
|
|
|
|
object ancestors = 0;
|
|
|
|
PROTECT(t, ancestors);
|
|
|
|
|
|
|
|
while (old != sentinal) {
|
|
|
|
ancestors = makePair(t, new_, ancestors);
|
|
|
|
|
2008-07-12 04:01:42 +00:00
|
|
|
intptr_t difference = compare(t, key, getTreeNodeValue(t, old));
|
2008-04-07 23:47:41 +00:00
|
|
|
|
|
|
|
if (difference < 0) {
|
|
|
|
old = treeNodeLeft(t, old);
|
|
|
|
object n = cloneTreeNode(t, old);
|
|
|
|
set(t, new_, TreeNodeLeft, n);
|
|
|
|
new_ = n;
|
|
|
|
} else if (difference > 0) {
|
|
|
|
old = treeNodeRight(t, old);
|
|
|
|
object n = cloneTreeNode(t, old);
|
|
|
|
set(t, new_, TreeNodeRight, n);
|
|
|
|
new_ = n;
|
|
|
|
} else {
|
|
|
|
return makeTreePath(t, false, new_, newRoot, pairSecond(t, ancestors));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
setTreeNodeValue(t, new_, getTreeNodeValue(t, node));
|
|
|
|
|
|
|
|
return makeTreePath(t, true, new_, newRoot, ancestors);
|
|
|
|
}
|
|
|
|
|
|
|
|
object
|
|
|
|
leftRotate(Thread* t, object n)
|
|
|
|
{
|
2008-04-12 00:05:03 +00:00
|
|
|
PROTECT(t, n);
|
|
|
|
|
2008-04-07 23:47:41 +00:00
|
|
|
object child = cloneTreeNode(t, treeNodeRight(t, n));
|
|
|
|
set(t, n, TreeNodeRight, treeNodeLeft(t, child));
|
|
|
|
set(t, child, TreeNodeLeft, n);
|
|
|
|
return child;
|
|
|
|
}
|
|
|
|
|
|
|
|
object
|
|
|
|
rightRotate(Thread* t, object n)
|
|
|
|
{
|
2008-04-12 00:05:03 +00:00
|
|
|
PROTECT(t, n);
|
|
|
|
|
2008-04-07 23:47:41 +00:00
|
|
|
object child = cloneTreeNode(t, treeNodeLeft(t, n));
|
|
|
|
set(t, n, TreeNodeLeft, treeNodeRight(t, child));
|
|
|
|
set(t, child, TreeNodeRight, n);
|
|
|
|
return child;
|
|
|
|
}
|
|
|
|
|
|
|
|
object
|
|
|
|
treeAdd(Thread* t, object path)
|
|
|
|
{
|
|
|
|
object new_ = treePathNode(t, path);
|
|
|
|
PROTECT(t, new_);
|
|
|
|
|
|
|
|
object newRoot = treePathRoot(t, path);
|
|
|
|
PROTECT(t, newRoot);
|
|
|
|
|
|
|
|
object ancestors = treePathAncestors(t, path);
|
|
|
|
PROTECT(t, ancestors);
|
|
|
|
|
|
|
|
// rebalance
|
2008-04-08 00:14:59 +00:00
|
|
|
setTreeNodeRed(t, new_, true);
|
2008-04-07 23:47:41 +00:00
|
|
|
while (ancestors != 0 and treeNodeRed(t, pairFirst(t, ancestors))) {
|
|
|
|
if (pairFirst(t, ancestors)
|
|
|
|
== treeNodeLeft(t, pairFirst(t, pairSecond(t, ancestors))))
|
|
|
|
{
|
|
|
|
if (treeNodeRed
|
|
|
|
(t, treeNodeRight(t, pairFirst(t, pairSecond(t, ancestors)))))
|
|
|
|
{
|
2008-11-22 21:14:52 +00:00
|
|
|
setTreeNodeRed(t, pairFirst(t, ancestors), false);
|
2008-04-07 23:47:41 +00:00
|
|
|
|
|
|
|
object n = cloneTreeNode
|
|
|
|
(t, treeNodeRight(t, pairFirst(t, pairSecond(t, ancestors))));
|
|
|
|
|
|
|
|
set(t, pairFirst(t, pairSecond(t, ancestors)), TreeNodeRight, n);
|
|
|
|
|
|
|
|
setTreeNodeRed
|
|
|
|
(t, treeNodeRight
|
2008-04-08 00:14:59 +00:00
|
|
|
(t, pairFirst(t, pairSecond(t, ancestors))), false);
|
2008-04-07 23:47:41 +00:00
|
|
|
|
2008-11-22 21:14:52 +00:00
|
|
|
setTreeNodeRed(t, pairFirst(t, pairSecond(t, ancestors)), true);
|
2008-04-07 23:47:41 +00:00
|
|
|
|
|
|
|
new_ = pairFirst(t, pairSecond(t, ancestors));
|
|
|
|
ancestors = pairSecond(t, pairSecond(t, ancestors));
|
|
|
|
} else {
|
|
|
|
if (new_ == treeNodeRight(t, pairFirst(t, ancestors))) {
|
|
|
|
new_ = pairFirst(t, ancestors);
|
|
|
|
ancestors = pairSecond(t, ancestors);
|
|
|
|
|
|
|
|
object n = leftRotate(t, new_);
|
|
|
|
|
|
|
|
if (new_ == treeNodeRight(t, pairFirst(t, ancestors))) {
|
|
|
|
set(t, pairFirst(t, ancestors), TreeNodeRight, n);
|
|
|
|
} else {
|
|
|
|
set(t, pairFirst(t, ancestors), TreeNodeLeft, n);
|
|
|
|
}
|
|
|
|
ancestors = makePair(t, n, ancestors);
|
|
|
|
}
|
2008-04-08 00:14:59 +00:00
|
|
|
setTreeNodeRed(t, pairFirst(t, ancestors), false);
|
|
|
|
setTreeNodeRed(t, pairFirst(t, pairSecond(t, ancestors)), true);
|
2008-04-07 23:47:41 +00:00
|
|
|
|
|
|
|
object n = rightRotate(t, pairFirst(t, pairSecond(t, ancestors)));
|
|
|
|
if (pairSecond(t, pairSecond(t, ancestors)) == 0) {
|
|
|
|
newRoot = n;
|
|
|
|
} else if (treeNodeRight
|
|
|
|
(t, pairFirst(t, pairSecond(t, pairSecond(t, ancestors))))
|
|
|
|
== pairFirst(t, pairSecond(t, ancestors)))
|
|
|
|
{
|
|
|
|
set(t, pairFirst(t, pairSecond(t, pairSecond(t, ancestors))),
|
|
|
|
TreeNodeRight, n);
|
|
|
|
} else {
|
|
|
|
set(t, pairFirst(t, pairSecond(t, pairSecond(t, ancestors))),
|
|
|
|
TreeNodeLeft, n);
|
|
|
|
}
|
|
|
|
// done
|
|
|
|
}
|
|
|
|
} else { // this is just the reverse of the code above (right and
|
|
|
|
// left swapped):
|
|
|
|
if (treeNodeRed
|
|
|
|
(t, treeNodeLeft(t, pairFirst(t, pairSecond(t, ancestors)))))
|
|
|
|
{
|
2008-11-22 21:14:52 +00:00
|
|
|
setTreeNodeRed(t, pairFirst(t, ancestors), false);
|
2008-04-07 23:47:41 +00:00
|
|
|
|
|
|
|
object n = cloneTreeNode
|
|
|
|
(t, treeNodeLeft(t, pairFirst(t, pairSecond(t, ancestors))));
|
|
|
|
|
|
|
|
set(t, pairFirst(t, pairSecond(t, ancestors)), TreeNodeLeft, n);
|
|
|
|
|
|
|
|
setTreeNodeRed
|
|
|
|
(t, treeNodeLeft
|
2008-04-08 00:14:59 +00:00
|
|
|
(t, pairFirst(t, pairSecond(t, ancestors))), false);
|
2008-04-07 23:47:41 +00:00
|
|
|
|
2008-11-22 21:14:52 +00:00
|
|
|
setTreeNodeRed(t, pairFirst(t, pairSecond(t, ancestors)), true);
|
2008-04-07 23:47:41 +00:00
|
|
|
|
|
|
|
new_ = pairFirst(t, pairSecond(t, ancestors));
|
|
|
|
ancestors = pairSecond(t, pairSecond(t, ancestors));
|
|
|
|
} else {
|
|
|
|
if (new_ == treeNodeLeft(t, pairFirst(t, ancestors))) {
|
|
|
|
new_ = pairFirst(t, ancestors);
|
|
|
|
ancestors = pairSecond(t, ancestors);
|
|
|
|
|
|
|
|
object n = rightRotate(t, new_);
|
|
|
|
|
|
|
|
if (new_ == treeNodeLeft(t, pairFirst(t, ancestors))) {
|
|
|
|
set(t, pairFirst(t, ancestors), TreeNodeLeft, n);
|
|
|
|
} else {
|
|
|
|
set(t, pairFirst(t, ancestors), TreeNodeRight, n);
|
|
|
|
}
|
|
|
|
ancestors = makePair(t, n, ancestors);
|
|
|
|
}
|
2008-04-08 00:14:59 +00:00
|
|
|
setTreeNodeRed(t, pairFirst(t, ancestors), false);
|
|
|
|
setTreeNodeRed(t, pairFirst(t, pairSecond(t, ancestors)), true);
|
2008-04-07 23:47:41 +00:00
|
|
|
|
|
|
|
object n = leftRotate(t, pairFirst(t, pairSecond(t, ancestors)));
|
|
|
|
if (pairSecond(t, pairSecond(t, ancestors)) == 0) {
|
|
|
|
newRoot = n;
|
|
|
|
} else if (treeNodeLeft
|
|
|
|
(t, pairFirst(t, pairSecond(t, pairSecond(t, ancestors))))
|
|
|
|
== pairFirst(t, pairSecond(t, ancestors)))
|
|
|
|
{
|
|
|
|
set(t, pairFirst(t, pairSecond(t, pairSecond(t, ancestors))),
|
|
|
|
TreeNodeLeft, n);
|
|
|
|
} else {
|
|
|
|
set(t, pairFirst(t, pairSecond(t, pairSecond(t, ancestors))),
|
|
|
|
TreeNodeRight, n);
|
|
|
|
}
|
|
|
|
// done
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-04-08 00:14:59 +00:00
|
|
|
setTreeNodeRed(t, newRoot, false);
|
2008-04-07 23:47:41 +00:00
|
|
|
|
|
|
|
return newRoot;
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace
|
|
|
|
|
2007-11-26 23:15:53 +00:00
|
|
|
namespace vm {
|
|
|
|
|
|
|
|
object
|
|
|
|
hashMapFindNode(Thread* t, object map, object key,
|
|
|
|
uint32_t (*hash)(Thread*, object),
|
|
|
|
bool (*equal)(Thread*, object, object))
|
|
|
|
{
|
|
|
|
bool weak = objectClass(t, map)
|
|
|
|
== arrayBody(t, t->m->types, Machine::WeakHashMapType);
|
|
|
|
|
|
|
|
object array = hashMapArray(t, map);
|
|
|
|
if (array) {
|
|
|
|
unsigned index = hash(t, key) & (arrayLength(t, array) - 1);
|
|
|
|
for (object n = arrayBody(t, array, index); n; n = tripleThird(t, n)) {
|
|
|
|
object k = tripleFirst(t, n);
|
|
|
|
if (weak) {
|
|
|
|
k = jreferenceTarget(t, k);
|
|
|
|
if (k == 0) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (equal(t, key, k)) {
|
|
|
|
return n;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
hashMapResize(Thread* t, object map, uint32_t (*hash)(Thread*, object),
|
|
|
|
unsigned size)
|
|
|
|
{
|
|
|
|
PROTECT(t, map);
|
|
|
|
|
|
|
|
object newArray = 0;
|
|
|
|
|
|
|
|
if (size) {
|
|
|
|
object oldArray = hashMapArray(t, map);
|
|
|
|
PROTECT(t, oldArray);
|
|
|
|
|
|
|
|
unsigned newLength = nextPowerOfTwo(size);
|
|
|
|
if (oldArray and arrayLength(t, oldArray) == newLength) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
newArray = makeArray(t, newLength, true);
|
|
|
|
|
2008-07-12 16:38:09 +00:00
|
|
|
if (oldArray != hashMapArray(t, map)) {
|
|
|
|
// a resize was performed during a GC via the makeArray call
|
|
|
|
// above; nothing left to do
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2007-11-26 23:15:53 +00:00
|
|
|
if (oldArray) {
|
|
|
|
bool weak = objectClass(t, map)
|
|
|
|
== arrayBody(t, t->m->types, Machine::WeakHashMapType);
|
|
|
|
|
|
|
|
for (unsigned i = 0; i < arrayLength(t, oldArray); ++i) {
|
|
|
|
object next;
|
|
|
|
for (object p = arrayBody(t, oldArray, i); p; p = next) {
|
|
|
|
next = tripleThird(t, p);
|
|
|
|
|
|
|
|
object k = tripleFirst(t, p);
|
|
|
|
if (weak) {
|
|
|
|
k = jreferenceTarget(t, k);
|
|
|
|
if (k == 0) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
unsigned index = hash(t, k) & (newLength - 1);
|
|
|
|
|
|
|
|
set(t, p, TripleThird, arrayBody(t, newArray, index));
|
|
|
|
set(t, newArray, ArrayBody + (index * BytesPerWord), p);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
set(t, map, HashMapArray, newArray);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
hashMapInsert(Thread* t, object map, object key, object value,
|
2008-07-12 01:26:46 +00:00
|
|
|
uint32_t (*hash)(Thread*, object))
|
2007-11-26 23:15:53 +00:00
|
|
|
{
|
2008-08-11 22:44:38 +00:00
|
|
|
// note that we reinitialize the array variable whenever an
|
|
|
|
// allocation (and thus possibly a collection) occurs, in case the
|
|
|
|
// array changes due to a table resize.
|
2008-01-19 20:12:16 +00:00
|
|
|
|
|
|
|
PROTECT(t, map);
|
|
|
|
|
|
|
|
uint32_t h = hash(t, key);
|
|
|
|
|
2007-11-26 23:15:53 +00:00
|
|
|
bool weak = objectClass(t, map)
|
|
|
|
== arrayBody(t, t->m->types, Machine::WeakHashMapType);
|
|
|
|
|
|
|
|
object array = hashMapArray(t, map);
|
|
|
|
|
|
|
|
++ hashMapSize(t, map);
|
|
|
|
|
|
|
|
if (array == 0 or hashMapSize(t, map) >= arrayLength(t, array) * 2) {
|
|
|
|
PROTECT(t, key);
|
|
|
|
PROTECT(t, value);
|
|
|
|
|
|
|
|
hashMapResize(t, map, hash, array ? arrayLength(t, array) * 2 : 16);
|
2008-01-19 20:12:16 +00:00
|
|
|
|
2007-11-26 23:15:53 +00:00
|
|
|
array = hashMapArray(t, map);
|
|
|
|
}
|
|
|
|
|
2008-01-19 20:12:16 +00:00
|
|
|
object k = key;
|
2007-11-26 23:15:53 +00:00
|
|
|
|
|
|
|
if (weak) {
|
|
|
|
PROTECT(t, key);
|
|
|
|
PROTECT(t, value);
|
|
|
|
|
|
|
|
object r = makeWeakReference(t, 0, 0, 0, 0);
|
|
|
|
jreferenceTarget(t, r) = key;
|
|
|
|
jreferenceVmNext(t, r) = t->m->weakReferences;
|
2008-01-19 20:12:16 +00:00
|
|
|
k = t->m->weakReferences = r;
|
|
|
|
|
|
|
|
array = hashMapArray(t, map);
|
2007-11-26 23:15:53 +00:00
|
|
|
}
|
|
|
|
|
2008-08-11 21:20:12 +00:00
|
|
|
object n = makeTriple(t, k, value, 0);
|
2008-01-19 20:12:16 +00:00
|
|
|
|
|
|
|
array = hashMapArray(t, map);
|
2008-08-11 22:44:38 +00:00
|
|
|
|
|
|
|
unsigned index = h & (arrayLength(t, array) - 1);
|
2007-11-26 23:15:53 +00:00
|
|
|
|
2008-08-11 21:20:12 +00:00
|
|
|
set(t, n, TripleThird, arrayBody(t, array, index));
|
2007-11-26 23:15:53 +00:00
|
|
|
set(t, array, ArrayBody + (index * BytesPerWord), n);
|
|
|
|
}
|
|
|
|
|
|
|
|
object
|
|
|
|
hashMapRemoveNode(Thread* t, object map, unsigned index, object p, object n)
|
|
|
|
{
|
|
|
|
if (p) {
|
|
|
|
set(t, p, TripleThird, tripleThird(t, n));
|
|
|
|
} else {
|
|
|
|
set(t, hashMapArray(t, map), ArrayBody + (index * BytesPerWord),
|
|
|
|
tripleThird(t, n));
|
|
|
|
}
|
|
|
|
-- hashMapSize(t, map);
|
|
|
|
return n;
|
|
|
|
}
|
|
|
|
|
|
|
|
object
|
|
|
|
hashMapRemove(Thread* t, object map, object key,
|
|
|
|
uint32_t (*hash)(Thread*, object),
|
|
|
|
bool (*equal)(Thread*, object, object))
|
|
|
|
{
|
|
|
|
bool weak = objectClass(t, map)
|
|
|
|
== arrayBody(t, t->m->types, Machine::WeakHashMapType);
|
|
|
|
|
|
|
|
object array = hashMapArray(t, map);
|
|
|
|
object o = 0;
|
|
|
|
if (array) {
|
|
|
|
unsigned index = hash(t, key) & (arrayLength(t, array) - 1);
|
|
|
|
object p = 0;
|
|
|
|
for (object n = arrayBody(t, array, index); n;) {
|
|
|
|
object k = tripleFirst(t, n);
|
|
|
|
if (weak) {
|
|
|
|
k = jreferenceTarget(t, k);
|
|
|
|
if (k == 0) {
|
|
|
|
n = tripleThird(t, hashMapRemoveNode(t, map, index, p, n));
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (equal(t, key, k)) {
|
|
|
|
o = tripleSecond(t, hashMapRemoveNode(t, map, index, p, n));
|
|
|
|
break;
|
|
|
|
} else {
|
|
|
|
p = n;
|
|
|
|
n = tripleThird(t, n);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (hashMapSize(t, map) <= arrayLength(t, array) / 3) {
|
|
|
|
PROTECT(t, o);
|
|
|
|
hashMapResize(t, map, hash, arrayLength(t, array) / 2);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return o;
|
|
|
|
}
|
|
|
|
|
|
|
|
object
|
|
|
|
hashMapIterator(Thread* t, object map)
|
|
|
|
{
|
|
|
|
object array = hashMapArray(t, map);
|
|
|
|
if (array) {
|
|
|
|
for (unsigned i = 0; i < arrayLength(t, array); ++i) {
|
|
|
|
if (arrayBody(t, array, i)) {
|
|
|
|
return makeHashMapIterator(t, map, arrayBody(t, array, i), i + 1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
object
|
|
|
|
hashMapIteratorNext(Thread* t, object it)
|
|
|
|
{
|
|
|
|
object map = hashMapIteratorMap(t, it);
|
|
|
|
object node = hashMapIteratorNode(t, it);
|
|
|
|
unsigned index = hashMapIteratorIndex(t, it);
|
|
|
|
|
|
|
|
if (tripleThird(t, node)) {
|
|
|
|
return makeHashMapIterator(t, map, tripleThird(t, node), index);
|
|
|
|
} else {
|
|
|
|
object array = hashMapArray(t, map);
|
|
|
|
for (unsigned i = index; i < arrayLength(t, array); ++i) {
|
|
|
|
if (arrayBody(t, array, i)) {
|
|
|
|
return makeHashMapIterator(t, map, arrayBody(t, array, i), i + 1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
listAppend(Thread* t, object list, object value)
|
|
|
|
{
|
|
|
|
PROTECT(t, list);
|
|
|
|
|
|
|
|
++ listSize(t, list);
|
|
|
|
|
|
|
|
object p = makePair(t, value, 0);
|
|
|
|
if (listFront(t, list)) {
|
|
|
|
set(t, listRear(t, list), PairSecond, p);
|
|
|
|
} else {
|
|
|
|
set(t, list, ListFront, p);
|
|
|
|
}
|
|
|
|
set(t, list, ListRear, p);
|
|
|
|
}
|
|
|
|
|
|
|
|
object
|
|
|
|
vectorAppend(Thread* t, object vector, object value)
|
|
|
|
{
|
|
|
|
if (vectorLength(t, vector) == vectorSize(t, vector)) {
|
|
|
|
PROTECT(t, vector);
|
|
|
|
PROTECT(t, value);
|
|
|
|
|
|
|
|
object newVector = makeVector
|
|
|
|
(t, vectorSize(t, vector), max(16, vectorSize(t, vector) * 2), false);
|
|
|
|
|
|
|
|
if (vectorSize(t, vector)) {
|
|
|
|
memcpy(&vectorBody(t, newVector, 0),
|
|
|
|
&vectorBody(t, vector, 0),
|
|
|
|
vectorSize(t, vector) * BytesPerWord);
|
|
|
|
}
|
|
|
|
|
|
|
|
memset(&vectorBody(t, newVector, vectorSize(t, vector) + 1),
|
|
|
|
0,
|
|
|
|
(vectorLength(t, newVector) - vectorSize(t, vector) - 1)
|
|
|
|
* BytesPerWord);
|
|
|
|
|
|
|
|
vector = newVector;
|
|
|
|
}
|
|
|
|
|
|
|
|
set(t, vector, VectorBody + (vectorSize(t, vector) * BytesPerWord), value);
|
|
|
|
++ vectorSize(t, vector);
|
|
|
|
return vector;
|
|
|
|
}
|
|
|
|
|
2008-04-07 23:47:41 +00:00
|
|
|
object
|
|
|
|
treeQuery(Thread* t, object tree, intptr_t key, object sentinal,
|
|
|
|
intptr_t (*compare)(Thread* t, intptr_t key, object b))
|
|
|
|
{
|
|
|
|
object node = tree;
|
|
|
|
while (node != sentinal) {
|
|
|
|
intptr_t difference = compare(t, key, getTreeNodeValue(t, node));
|
|
|
|
if (difference < 0) {
|
|
|
|
node = treeNodeLeft(t, node);
|
|
|
|
} else if (difference > 0) {
|
|
|
|
node = treeNodeRight(t, node);
|
|
|
|
} else {
|
2008-04-10 23:48:28 +00:00
|
|
|
return getTreeNodeValue(t, node);
|
2008-04-07 23:47:41 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
object
|
2008-07-12 04:01:42 +00:00
|
|
|
treeInsertNode(Thread* t, object tree, intptr_t key, object node,
|
|
|
|
object sentinal,
|
|
|
|
intptr_t (*compare)(Thread* t, intptr_t key, object b))
|
2008-04-07 23:47:41 +00:00
|
|
|
{
|
2008-04-10 23:48:28 +00:00
|
|
|
PROTECT(t, tree);
|
|
|
|
PROTECT(t, sentinal);
|
|
|
|
|
2008-07-12 04:01:42 +00:00
|
|
|
object path = treeFind(t, tree, key, node, sentinal, compare);
|
|
|
|
expect(t, treePathFresh(t, path));
|
|
|
|
return treeAdd(t, path);
|
2008-04-07 23:47:41 +00:00
|
|
|
}
|
|
|
|
|
2007-11-26 23:15:53 +00:00
|
|
|
} // namespace vm
|