/* Copyright (C) 2003-2015 LiveCode Ltd.
This file is part of LiveCode.
LiveCode is free software; you can redistribute it and/or modify it under
the terms of the GNU General Public License v3 as published by the Free
Software Foundation.
LiveCode is distributed in the hope that it will be useful, but WITHOUT ANY
WARRANTY; without even the implied warranty of MERCHANTABILITY or
FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
for more details.
You should have received a copy of the GNU General Public License
along with LiveCode. If not see . */
#include
#include
#include
#include "foundation-private.h"
////////////////////////////////////////////////////////////////////////////////
// Creates an indirect mutable array with contents.
static bool __MCArrayCreateIndirect(__MCArray *contents, __MCArray*& r_array);
// Returns the index of the table size of the array.
static uindex_t __MCArrayGetTableSizeIndex(__MCArray *self);
// Returns true if the array is indirect.
static bool __MCArrayIsIndirect(__MCArray *self);
// Replaces all the values in the array with immutable copies.
static bool __MCArrayMakeContentsImmutable(__MCArray *self);
// Creates an immutable array from this one, changing 'self' to indirect.
static bool __MCArrayMakeIndirect(__MCArray *self);
// Ensures the given mutable but indirect array is direct.
static bool __MCArrayResolveIndirect(__MCArray *self);
// Rehash the table adjusting capacity by delta.
static bool __MCArrayRehash(__MCArray *self, index_t by);
// Returns the number of entries in the key-value table for the array.
static uindex_t __MCArrayGetTableSize(__MCArray *self);
// Returns the maximum number of entries for a given array size that minimises rehashing.
static uindex_t __MCArrayGetTableCapacity(__MCArray *self);
// Looks for a key-value slot in the array with the given key. If the key was
// found 'true' is returned; otherwise 'false'. On return 'slot' will be the
// slot in which the key is found, could be placed, or UINDEX_MAX if there is
// no more room (and the key isn't there).
static bool __MCArrayFindKeyValueSlot(__MCArray *self, bool case_sensitive, MCNameRef key, uindex_t& r_slot);
////////////////////////////////////////////////////////////////////////////////
MC_DLLEXPORT_DEF
bool MCArrayCreate(bool p_case_sensitive, const MCNameRef *p_keys, const MCValueRef *p_values, uindex_t p_length, MCArrayRef& r_array)
{
if (p_length == 0)
{
if (nil != kMCEmptyArray)
{
r_array = MCValueRetain(kMCEmptyArray);
return true;
}
}
else
{
MCAssert(nil != p_keys);
MCAssert(nil != p_values);
}
bool t_success;
t_success = true;
MCArrayRef t_array;
t_array = nil;
if (t_success)
t_success = MCArrayCreateMutable(t_array);
if (t_success)
for(uindex_t i = 0; i < p_length && t_success; i++)
t_success = MCArrayStoreValue(t_array, p_case_sensitive, p_keys[i], p_values[i]);
if (t_success)
return MCArrayCopyAndRelease(t_array, r_array);
MCValueRelease(t_array);
return false;
}
MC_DLLEXPORT_DEF
bool MCArrayCreateMutable(MCArrayRef& r_array)
{
if (!__MCValueCreate(kMCValueTypeCodeArray, r_array))
return false;
r_array -> flags |= kMCArrayFlagIsMutable;
return true;
}
MC_DLLEXPORT_DEF
bool MCArrayCopy(MCArrayRef self, MCArrayRef& r_new_array)
{
__MCAssertIsArray(self);
// If we aren't mutable, then we can just copy directly.
if (!MCArrayIsMutable(self))
{
r_new_array = MCValueRetain(self);
return true;
}
// If we mutable, but still indirect just take the contents array.
if (__MCArrayIsIndirect(self))
{
r_new_array = MCValueRetain(self -> contents);
return true;
}
// Make the contents immutable.
if (!__MCArrayMakeContentsImmutable(self))
return false;
// Make the array indirect.
if (!__MCArrayMakeIndirect(self))
return false;
// Return a copy of the contents.
r_new_array = MCValueRetain(self -> contents);
return true;
}
MC_DLLEXPORT_DEF
bool MCArrayCopyAndRelease(MCArrayRef self, MCArrayRef& r_new_array)
{
__MCAssertIsArray(self);
// If we aren't mutable, then new array is just us.
if (!MCArrayIsMutable(self))
{
r_new_array = self;
return true;
}
// If we are indirect, then new array is the contents, we are released.
if (__MCArrayIsIndirect(self))
{
r_new_array = MCValueRetain(self -> contents);
MCValueRelease(self);
return true;
}
// We need to make an immutable copy, so first change all our contents
// to immutable.
if (!__MCArrayMakeContentsImmutable(self))
return false;
// If we have a reference count of one 'self' becomes the immutable copy.
if (self -> references == 1)
{
// We are no longer immutable.
self -> flags &= ~kMCArrayFlagIsMutable;
// Return this as the new array.
r_new_array = self;
return true;
}
// Otherwise we must build a new indirect value.
if (!__MCArrayMakeIndirect(self))
return false;
// Reduce our reference count.
self -> references -= 1;
// Return a copy of the contents.
r_new_array = MCValueRetain(self -> contents);
return true;
}
MC_DLLEXPORT_DEF
bool MCArrayMutableCopy(MCArrayRef self, MCArrayRef& r_new_array)
{
__MCAssertIsArray(self);
// If the array is immutable, then the new mutable array will be indirect
// referencing it. [ non-mutable arrays cannot be indirect so self does not
// need resolving ].
if (!MCArrayIsMutable(self))
return __MCArrayCreateIndirect(self, r_new_array);
// If the array is already indirect, we just create a new reference to the
// same contents.
if (__MCArrayIsIndirect(self))
return __MCArrayCreateIndirect(self -> contents, r_new_array);
// If the array is mutable, we make it immutable and indirect and share
// the indirect copy.
if (!__MCArrayMakeContentsImmutable(self))
return false;
// Make self indirect, and fetch its contents.
if (!__MCArrayMakeIndirect(self))
return false;
// Finally, create a new indirect array with contents of self.
return __MCArrayCreateIndirect(self -> contents, r_new_array);
}
MC_DLLEXPORT_DEF
bool MCArrayMutableCopyAndRelease(MCArrayRef self, MCArrayRef& r_new_array)
{
__MCAssertIsArray(self);
if (self -> references == 1)
{
if (!MCArrayIsMutable(self))
self -> flags |= kMCArrayFlagIsMutable;
r_new_array = self;
return true;
}
if (!MCArrayMutableCopy(self, r_new_array))
return false;
self -> references -= 1;
return true;
}
MC_DLLEXPORT_DEF
bool MCArrayApply(MCArrayRef self, MCArrayApplyCallback p_callback, void *p_context)
{
__MCAssertIsArray(self);
MCAssert(nil != p_callback);
// Make sure we are iterating over the correct contents.
MCArrayRef t_contents;
if (!__MCArrayIsIndirect(self))
t_contents = self;
else
t_contents = self -> contents;
uindex_t t_used;
t_used = t_contents -> key_value_count;
uindex_t t_count;
t_count = __MCArrayGetTableSize(t_contents);
for(uindex_t i = 0; t_used > 0 && i < t_count; i++)
{
if (t_contents -> key_values[i] . value == UINTPTR_MIN || t_contents -> key_values[i] . value == UINTPTR_MAX)
continue;
if (!p_callback(p_context, self, t_contents -> key_values[i] . key, (MCValueRef)t_contents -> key_values[i] . value))
return false;
t_used -= 1;
}
return true;
}
MC_DLLEXPORT_DEF
bool MCArrayIterate(MCArrayRef self, uintptr_t& x_iterator, MCNameRef& r_key, MCValueRef& r_value)
{
__MCAssertIsArray(self);
// Make sure we are iterating over the correct contents.
MCArrayRef t_contents;
if (!__MCArrayIsIndirect(self))
t_contents = self;
else
t_contents = self -> contents;
uindex_t t_count;
t_count = __MCArrayGetTableSize(t_contents);
if (x_iterator == t_count)
return false;
for(uindex_t i = x_iterator; i < t_count; i += 1)
{
x_iterator += 1;
if (t_contents -> key_values[i] . value != UINTPTR_MIN && t_contents -> key_values[i] . value != UINTPTR_MAX)
{
r_key = t_contents -> key_values[i] . key;
r_value = (MCValueRef)t_contents -> key_values[i] . value;
return true;
}
}
return false;
}
////////////////////////////////////////////////////////////////////////////////
MC_DLLEXPORT_DEF
bool MCArrayIsMutable(MCArrayRef self)
{
__MCAssertIsArray(self);
return (self -> flags & kMCArrayFlagIsMutable) != 0;
}
MC_DLLEXPORT_DEF
uindex_t MCArrayGetCount(MCArrayRef self)
{
__MCAssertIsArray(self);
if (!__MCArrayIsIndirect(self))
return self -> key_value_count;
return self -> contents -> key_value_count;
}
////////////////////////////////////////////////////////////////////////////////
MC_DLLEXPORT_DEF
bool MCArrayFetchValue(MCArrayRef self, bool p_case_sensitive, MCNameRef p_key, MCValueRef& r_value)
{
return MCArrayFetchValueOnPath(self, p_case_sensitive, &p_key, 1, r_value);
}
MC_DLLEXPORT_DEF
bool MCArrayFetchValueOnPath(MCArrayRef self, bool p_case_sensitive, const MCNameRef *p_path, uindex_t p_path_length, MCValueRef& r_value)
{
__MCAssertIsArray(self);
MCAssert(nil != p_path);
MCAssert(0 < p_path_length);
__MCAssertIsName(p_path[0]);
// If the array is indirect, get the contents.
MCArrayRef t_contents;
if (!__MCArrayIsIndirect(self))
t_contents = self;
else
t_contents = self -> contents;
// Lookup the slot for the first part of the path.
uindex_t t_slot;
if (!__MCArrayFindKeyValueSlot(t_contents, p_case_sensitive, p_path[0], t_slot))
return false;
// We found a slot successfully matching the key so get the value.
MCValueRef t_value;
t_value = (MCValueRef)t_contents -> key_values[t_slot] . value;
// If the path length is one, then we are done.
if (p_path_length == 1)
{
r_value = t_value;
return true;
}
// If the value isn't an array then we can't continue with the lookup.
if (MCValueGetTypeCode(t_value) != kMCValueTypeCodeArray)
return false;
// Otherwise, look up the next step in the path.
return MCArrayFetchValueOnPath((MCArrayRef)t_value, p_case_sensitive, p_path + 1, p_path_length - 1, r_value);
}
bool MCArrayFetchValueOnPath(MCArrayRef array,
bool case_sensitive,
const MCSpan path,
MCValueRef& value)
{
return MCArrayFetchValueOnPath(array, case_sensitive,
path.data(), path.size(),
value);
}
//////////////////////
MC_DLLEXPORT_DEF
bool MCArrayStoreValue(MCArrayRef self, bool p_case_sensitive, MCNameRef p_key, MCValueRef p_value)
{
return MCArrayStoreValueOnPath(self, p_case_sensitive, &p_key, 1, p_value);
}
MC_DLLEXPORT_DEF
bool MCArrayStoreValueOnPath(MCArrayRef self, bool p_case_sensitive, const MCNameRef *p_path, uindex_t p_path_length, MCValueRef p_new_value)
{
// The array must be mutable.
MCAssert(MCArrayIsMutable(self));
MCAssert(nil != p_path);
MCAssert(0 < p_path_length);
__MCAssertIsName(p_path[0]);
// Ensure it is not indirect.
if (__MCArrayIsIndirect(self))
if (!__MCArrayResolveIndirect(self))
return false;
// Lookup the slot for the first element in the path.
bool t_found;
uindex_t t_slot;
t_found = __MCArrayFindKeyValueSlot(self, p_case_sensitive, p_path[0], t_slot);
if (t_found)
{
// Get the value.
MCValueRef t_value;
t_value = (MCValueRef)self -> key_values[t_slot] . value;
// If the path length is 1, then just set the value and return.
if (p_path_length == 1)
{
MCValueRelease(t_value);
self -> key_values[t_slot] . value = (uintptr_t)MCValueRetain(p_new_value);
return true;
}
// If the value is an array then recurse.
if (MCValueGetTypeCode(t_value) == kMCValueTypeCodeArray)
{
// If the array isn't mutable, then make a mutable copy replacing the existing
// value.
MCArrayRef t_mutable_array;
if (!MCArrayIsMutable((MCArrayRef)t_value))
{
if (!MCArrayMutableCopyAndRelease((MCArrayRef)t_value, t_mutable_array))
return false;
self -> key_values[t_slot] . value = (uintptr_t)t_mutable_array;
}
else
t_mutable_array = (MCArrayRef)t_value;
return MCArrayStoreValueOnPath(t_mutable_array, p_case_sensitive, p_path + 1, p_path_length - 1, p_new_value);
}
}
else
{
// AL-2014-07-15: [[ Bug 12532 ]] Rehash according to hash table capacities rather than sizes
if (t_slot == UINDEX_MAX || self -> key_value_count >= __MCArrayGetTableCapacity(self))
{
if (!__MCArrayRehash(self, 1))
return false;
__MCArrayFindKeyValueSlot(self, p_case_sensitive, p_path[0], t_slot);
}
if (p_path_length == 1)
{
self -> key_values[t_slot] . key = MCValueRetain(p_path[0]);
self -> key_values[t_slot] . value = (uintptr_t)MCValueRetain(p_new_value);
self -> key_value_count += 1;
return true;
}
}
// If the value isn't an array, then create one.
MCArrayRef t_array;
if (!MCArrayCreateMutable(t_array))
return false;
// Build up the new array's value.
if (!MCArrayStoreValueOnPath(t_array, p_case_sensitive, p_path + 1, p_path_length - 1, p_new_value))
{
MCValueRelease(t_array);
return false;
}
// We've successfully built the rest of the path, so replace or add the new key-value.
if (t_found)
MCValueRelease((MCValueRef)self -> key_values[t_slot] . value);
else
{
self -> key_values[t_slot] . key = MCValueRetain(p_path[0]);
self -> key_value_count += 1;
}
self -> key_values[t_slot] . value = (uintptr_t)t_array;
return true;
}
bool MCArrayStoreValueOnPath(MCArrayRef array,
bool case_sensitive,
const MCSpan path,
MCValueRef value)
{
return MCArrayStoreValueOnPath(array, case_sensitive,
path.data(), path.size(),
value);
}
//////////////////////
MC_DLLEXPORT_DEF
bool MCArrayRemoveValue(MCArrayRef self, bool p_case_sensitive, MCNameRef p_key)
{
return MCArrayRemoveValueOnPath(self, p_case_sensitive, &p_key, 1);
}
MC_DLLEXPORT_DEF
bool MCArrayRemoveValueOnPath(MCArrayRef self, bool p_case_sensitive, const MCNameRef *p_path, uindex_t p_path_length)
{
// The array must be mutable.
MCAssert(MCArrayIsMutable(self));
MCAssert(nil != p_path);
MCAssert(0 < p_path_length);
// Ensure it is not indirect.
if (__MCArrayIsIndirect(self))
if (!__MCArrayResolveIndirect(self))
return false;
// Look up the first slot in the path.
uindex_t t_slot;
if (__MCArrayFindKeyValueSlot(self, p_case_sensitive, p_path[0], t_slot))
{
MCValueRef t_value;
t_value = (MCValueRef)self -> key_values[t_slot] . value;
// If the path length is one, then just remove the key.
if (p_path_length == 1)
{
MCValueRelease(self -> key_values[t_slot] . key);
MCValueRelease(t_value);
self -> key_values[t_slot] . key = nil;
self -> key_values[t_slot] . value = UINTPTR_MAX;
self -> key_value_count -= 1;
if (__MCArrayGetTableSizeIndex(self) > 2 &&
self -> key_value_count < __kMCValueHashTableCapacities[__MCArrayGetTableSizeIndex(self) - 2])
__MCArrayRehash(self, -1);
return true;
}
// If the value is an array then recurse.
if (MCValueGetTypeCode(t_value) == kMCValueTypeCodeArray)
{
// If the array isn't mutable, then make a mutable copy replacing the existing
// value.
MCArrayRef t_mutable_array;
if (!MCArrayIsMutable((MCArrayRef)t_value))
{
if (!MCArrayMutableCopyAndRelease((MCArrayRef)t_value, t_mutable_array))
return false;
self -> key_values[t_slot] . value = (uintptr_t)t_mutable_array;
}
else
t_mutable_array = (MCArrayRef)t_value;
return MCArrayRemoveValueOnPath(t_mutable_array, p_case_sensitive, p_path + 1, p_path_length - 1);
}
}
// Otherwise there is nothing more to do.
return true;
}
bool MCArrayRemoveValueOnPath(MCArrayRef array,
bool case_sensitive,
MCSpan path)
{
return MCArrayRemoveValueOnPath(array, case_sensitive,
path.data(), path.size());
}
////////////////////////////////////////////////////////////////////////////////
MC_DLLEXPORT_DEF
bool MCArrayFetchValueAtIndex(MCArrayRef self, index_t p_index, MCValueRef& r_value)
{
__MCAssertIsArray(self);
char t_index_str[16];
sprintf(t_index_str, "%d", p_index);
MCNewAutoNameRef t_key;
if (!MCNameCreateWithNativeChars((const char_t *)t_index_str, strlen(t_index_str), &t_key))
return false;
return MCArrayFetchValue(self, true, *t_key, r_value);
}
MC_DLLEXPORT_DEF
bool MCArrayStoreValueAtIndex(MCArrayRef self, index_t p_index, MCValueRef p_value)
{
__MCAssertIsArray(self);
char t_index_str[16];
sprintf(t_index_str, "%d", p_index);
MCNewAutoNameRef t_key;
if (!MCNameCreateWithNativeChars((const char_t *)t_index_str, strlen(t_index_str), &t_key))
return false;
return MCArrayStoreValue(self, true, *t_key, p_value);
}
bool
MCArrayRemoveValueAtIndex(MCArrayRef self, index_t p_index)
{
char t_index_str[16];
sprintf(t_index_str, "%d", p_index);
MCNewAutoNameRef t_key;
if (!MCNameCreateWithNativeChars((const char_t *)t_index_str,
strlen(t_index_str),
&t_key))
return false;
return MCArrayRemoveValue(self, true, *t_key);
}
////////////////////////////////////////////////////////////////////////////////
MC_DLLEXPORT_DEF
bool MCArrayIsEmpty(MCArrayRef self)
{
return MCArrayGetCount(self) == 0;
}
////////////////////////////////////////////////////////////////////////////////
MC_DLLEXPORT_DEF
bool MCArrayConvertToProperList(MCArrayRef p_array, MCProperListRef& r_list)
{
MCAutoProperListRef t_list;
if (!MCProperListCreateMutable(&t_list))
{
return false;
}
for(uindex_t t_index = 1; t_index <= MCArrayGetCount(p_array); t_index++)
{
MCValueRef t_value;
if (!MCArrayFetchValueAtIndex(p_array, t_index, t_value))
{
r_list = nullptr;
return true;
}
if (!MCProperListPushElementOntoBack(*t_list, t_value))
{
return false;
}
}
if (!t_list.MakeImmutable())
{
return false;
}
r_list = t_list.Take();
return true;
}
////////////////////////////////////////////////////////////////////////////////
void __MCArrayDestroy(__MCArray *self)
{
if (__MCArrayIsIndirect(self))
MCValueRelease(self -> contents);
else
{
uindex_t t_used;
t_used = self -> key_value_count;
uindex_t t_count;
t_count = __MCArrayGetTableSize(self);
for(uindex_t i = 0; t_used > 0 && i < t_count; i++)
{
if (self -> key_values[i] . value == UINTPTR_MIN || self -> key_values[i] . value == UINTPTR_MAX)
continue;
MCValueRelease(self -> key_values[i] . key);
MCValueRelease((MCValueRef)self -> key_values[i] . value);
t_used -= 1;
}
MCMemoryDeleteArray(self -> key_values);
}
}
hash_t __MCArrayHash(__MCArray *self)
{
return (hash_t)self -> key_value_count;
}
bool __MCArrayIsEqualTo(__MCArray *self, __MCArray *other_self)
{
// If the array is indirect, get the contents.
MCArrayRef t_contents;
if (!__MCArrayIsIndirect(self))
t_contents = self;
else
t_contents = self -> contents;
// If the other array is indirect, get its contents.
MCArrayRef t_other_contents;
if (!__MCArrayIsIndirect(other_self))
t_other_contents = other_self;
else
t_other_contents = other_self -> contents;
if (t_contents -> key_value_count != t_other_contents -> key_value_count)
return false;
uindex_t t_used;
t_used = t_contents -> key_value_count;
uindex_t t_count;
t_count = __MCArrayGetTableSize(t_contents);
for(uindex_t i = 0; t_used > 0 && i < t_count; i++)
{
// If the given slot is not used, then skip it.
if (t_contents -> key_values[i] . value == UINTPTR_MIN || t_contents -> key_values[i] . value == UINTPTR_MAX)
continue;
// If we don't find a key in the other array matching one in this then
// the arrays aren't equal.
uindex_t t_slot;
if (!__MCArrayFindKeyValueSlot(t_other_contents, true, t_contents -> key_values[i] . key, t_slot))
return false;
// Otherwise, they are only equal if the values are the same.
if (t_contents -> key_values[i] . value != t_other_contents -> key_values[t_slot] . value)
return false;
// We've compared one more key, so used count goes down.
t_used -= 1;
}
// If we get here it means all key/values match.
return true;
}
bool __MCArrayCopyDescription(__MCArray *self, MCStringRef& r_string)
{
/* Shortcut for empty arrays */
if (MCArrayIsEmpty (self))
return MCStringCopy (MCSTR("{}"), r_string);
MCAutoListRef t_contents_list;
if (!MCListCreateMutable (MCSTR(", "), &t_contents_list))
return false;
uintptr_t t_iter = 0;
MCNameRef t_key;
MCValueRef t_value;
while (MCArrayIterate (self, t_iter, t_key, t_value))
{
// AL-2015-06-19:[[ Bug 15529 ]] Call MCValueCopyDescription to convert arbitrary array values to string
MCAutoStringRef t_value_string;
if (!MCValueCopyDescription(t_value, &t_value_string))
return false;
if (!MCListAppendFormat (*t_contents_list, "%@: %@", t_key, *t_value_string))
return false;
}
MCAutoStringRef t_contents_string;
if (!MCListCopyAsString (*t_contents_list, &t_contents_string))
return false;
return MCStringFormat(r_string, "{%@}", *t_contents_string);
}
bool __MCArrayImmutableCopy(__MCArray *self, bool p_release, __MCArray*& r_immutable_self)
{
if (!p_release)
return MCArrayCopy(self, r_immutable_self);
return MCArrayCopyAndRelease(self, r_immutable_self);
}
////////////////////////////////////////////////////////////////////////////////
static bool __MCArrayCreateIndirect(__MCArray *p_contents, __MCArray*& r_array)
{
MCArrayRef self;
if (!__MCValueCreate(kMCValueTypeCodeArray, self))
return false;
self -> flags |= kMCArrayFlagIsMutable | kMCArrayFlagIsIndirect;
self -> contents = MCValueRetain(p_contents);
r_array = self;
return true;
}
static uindex_t __MCArrayGetTableSizeIndex(__MCArray *self)
{
return self -> flags & kMCArrayFlagCapacityIndexMask;
}
static void __MCArraySetTableSizeIndex(__MCArray *self, uindex_t p_new_index)
{
self -> flags = (self -> flags & ~kMCArrayFlagCapacityIndexMask) | p_new_index;
}
static uindex_t __MCArrayGetTableSize(__MCArray *self)
{
return __kMCValueHashTableSizes[self -> flags & kMCArrayFlagCapacityIndexMask];
}
static uindex_t __MCArrayGetTableCapacity(__MCArray *self)
{
return __kMCValueHashTableCapacities[self -> flags & kMCArrayFlagCapacityIndexMask];
}
static bool __MCArrayIsIndirect(__MCArray *self)
{
return (self -> flags & kMCArrayFlagIsIndirect) != 0;
}
static bool __MCArrayMakeContentsImmutable(__MCArray *self)
{
uindex_t t_used, t_count;
t_used = self -> key_value_count;
t_count = __MCArrayGetTableSize(self);
for(uindex_t i = 0; t_used > 0 && i < t_count; i++)
{
if (self -> key_values[i] . value != UINTPTR_MIN && self -> key_values[i] . value != UINTPTR_MAX)
{
__MCValue *t_new_value;
if (!__MCValueImmutableCopy((__MCValue *)self -> key_values[i] . value, true, t_new_value))
return false;
self -> key_values[i] . value = (uintptr_t)t_new_value;
}
}
return true;
}
static bool __MCArrayMakeIndirect(__MCArray *self)
{
// If we are already indirect, there's nothing to do.
if (__MCArrayIsIndirect(self))
return true;
// Our key-values are now all immutable, so create a new immutable array
// with them.
MCArrayRef t_array;
if (!__MCValueCreate(kMCValueTypeCodeArray, t_array))
return false;
// Fill in our new array.
t_array -> flags |= self -> flags & kMCArrayFlagCapacityIndexMask;
t_array -> key_value_count = self -> key_value_count;
t_array -> key_values = self -> key_values;
// 'self' now becomes indirect with a reference to the new array.
self -> flags |= kMCArrayFlagIsIndirect;
self -> contents = t_array;
return true;
}
static bool __MCArrayResolveIndirect(__MCArray *self)
{
// Make sure we are indirect.
MCAssert(__MCArrayIsIndirect(self));
// Fetch the contents.
MCArrayRef t_contents;
t_contents = self -> contents;
// If the contents only has a single reference, then re-absorb; otherwise
// copy.
if (self -> contents -> references == 1)
{
self -> key_values = t_contents -> key_values;
self -> key_value_count = t_contents -> key_value_count;
t_contents -> key_values = nil;
t_contents -> key_value_count = 0;
}
else
{
uindex_t t_size;
t_size = __MCArrayGetTableSize(t_contents);
if (!MCMemoryNewArray(t_size, self -> key_values))
return false;
self -> key_value_count = t_contents -> key_value_count;
for(uindex_t i = 0; i < t_size; i++)
{
if (t_contents -> key_values[i] . value != UINTPTR_MIN && t_contents -> key_values[i] . value != UINTPTR_MAX)
{
self -> key_values[i] . value = (uintptr_t)MCValueRetain((MCValueRef)t_contents -> key_values[i] . value);
self -> key_values[i] . key = MCValueRetain(t_contents -> key_values[i] . key);
}
else
{
// Ensure we don't break any hash chains
self -> key_values[i] = t_contents -> key_values[i];
}
}
}
// Make sure we take the index from the flags.
__MCArraySetTableSizeIndex(self, __MCArrayGetTableSizeIndex(t_contents));
// Make sure the array is no longer marked as indirect.
self -> flags &= ~kMCArrayFlagIsIndirect;
// Destroy the contents.
MCValueRelease(t_contents);
return true;
}
static bool __MCArrayFindKeyValueSlot(__MCArray *self, bool p_case_sensitive, MCNameRef p_key, uindex_t& r_slot)
{
// Get the table size.
uindex_t t_size;
t_size = __MCArrayGetTableSize(self);
if (t_size == 0 || self -> key_values == nil)
{
r_slot = UINDEX_MAX;
return false;
}
// Get the hash.
uindex_t t_hash;
t_hash = MCValueHash(p_key);
// Fold the hash code appropriately.
uindex_t t_h1;
#if defined(__ARM__) && 0 // TODO
t_h1 = __MCHashFold(t_hash, __MCArrayGetTableSizeIndex(self));
#else
t_h1 = t_hash % t_size;
#endif
// The initial index to probe.
uindex_t t_probe;
t_probe = t_h1;
// The target for a new entry - if it ends up being UINDEX_MAX it means the
// table is full.
uindex_t t_target_slot;
t_target_slot = UINDEX_MAX;
// Loop over all key value pairs - starting at probe.
for(uindex_t i = 0; i < t_size; i++)
{
__MCArrayKeyValue *t_entry;
t_entry = &self -> key_values[t_probe];
if (t_entry -> value == UINTPTR_MIN)
{
// The end of the chain - we are done.
if (t_target_slot == UINDEX_MAX)
t_target_slot = t_probe;
r_slot = t_probe;
return false;
}
if (t_entry -> value == UINTPTR_MAX)
{
if (t_target_slot == UINDEX_MAX)
t_target_slot = t_probe;
}
else
{
if (MCNameIsEqualTo(t_entry -> key, p_key, !p_case_sensitive ? kMCStringOptionCompareCaseless : kMCStringOptionCompareExact))
{
r_slot = t_probe;
return true;
}
}
t_probe += 1;
if (t_size <= t_probe)
t_probe -= t_size;
}
// If we get here the name wasn't found.
r_slot = t_target_slot;
return false;
}
static bool __MCArrayRehash(__MCArray *self, index_t p_by)
{
uindex_t t_new_capacity_idx;
t_new_capacity_idx = __MCArrayGetTableSizeIndex(self);
if (p_by != 0)
{
if (p_by < 0)
p_by = 0;
uindex_t t_new_capacity_req;
t_new_capacity_req = self -> key_value_count + p_by;
for(t_new_capacity_idx = 0;
t_new_capacity_req > __kMCValueHashTableCapacities[t_new_capacity_idx];
++t_new_capacity_idx);
}
uindex_t t_old_capacity;
__MCArrayKeyValue *t_old_key_values;
t_old_capacity = __MCArrayGetTableSize(self);
t_old_key_values = self -> key_values;
uindex_t t_new_capacity;
__MCArrayKeyValue *t_new_key_values;
t_new_capacity = __kMCValueHashTableSizes[t_new_capacity_idx];
if (!MCMemoryNewArray(t_new_capacity, t_new_key_values))
return false;
__MCArraySetTableSizeIndex(self, t_new_capacity_idx);
self -> key_values = t_new_key_values;
for(uindex_t i = 0; i < t_old_capacity; i++)
{
if (t_old_key_values[i] . value != UINTPTR_MIN && t_old_key_values[i] . value != UINTPTR_MAX)
{
uindex_t t_target_slot;
__MCArrayFindKeyValueSlot(self, true, t_old_key_values[i] . key, t_target_slot);
MCAssert(t_target_slot != UINDEX_MAX);
t_new_key_values[t_target_slot] = t_old_key_values[i];
}
}
MCMemoryDeleteArray(t_old_key_values);
return true;
}
////////////////////////////////////////////////////////////////////////////////
MC_DLLEXPORT_DEF MCArrayRef kMCEmptyArray;
bool __MCArrayInitialize(void)
{
if (!MCArrayCreate(false, nil, nil, 0, kMCEmptyArray))
return false;
return true;
}
void __MCArrayFinalize(void)
{
MCValueRelease(kMCEmptyArray);
}
////////////////////////////////////////////////////////////////////////////////
void __MCArrayDump(MCArrayRef array)
{
// If the array is indirect, get the contents.
MCArrayRef t_contents;
if (!__MCArrayIsIndirect(array))
t_contents = array;
else
t_contents = array -> contents;
uindex_t t_size;
t_size = __MCArrayGetTableSize(t_contents);
for(uindex_t i = 0; i < t_size; i++)
{
__MCArrayKeyValue *t_entry;
t_entry = &t_contents -> key_values[i];
if (t_entry -> value != UINTPTR_MIN && t_entry -> value != UINTPTR_MAX)
{
MCNameRef t_key;
t_key = t_entry -> key;
MCValueRef t_value;
t_value = (MCValueRef)t_entry -> value;
if (MCValueGetTypeCode(t_value) == kMCValueTypeCodeArray)
{
MCLog("[%@]:", t_key);
__MCArrayDump((MCArrayRef)t_value);
}
else
{
MCAutoStringRef t_desc;
MCValueCopyDescription(t_value, &t_desc);
MCLog("[%@] = %@", t_key, *t_desc);
}
}
}
}
/*
bool MCArrayFetchValue(MCArrayRef self, bool p_case_sensitive, const MCNameRef *p_path, uindex_t p_path_length, MCValueRef& r_value)
{
MCArrayRef t_container;
if (p_path_length == 1)
t_container = self;
else
t_container = __MCArrayResolvePath(self, p_case_sensitive, p_path, p_path_length - 1);
uindex_t t_slot;
if (__MCArrayFindKeyValueSlotForFetch(t_container, p_case_sensitive, p_path[p_path_length - 1], t_slot))
{
r_value = (MCValueRef)t_container -> key_values[t_slot] . value;
return true;
}
return false;
}
bool MCArrayStoreValue(MCArrayRef self, bool p_case_sensitive, const MCNameRef *p_path, uindex_t p_path_length, MCValueRef p_value)
{
MCArrayRef t_container;
if (p_path_length == 1)
t_container = self;
else if (!__MCArrayResolveMutablePath(self, p_case_sensitive, p_path, p_path_length - 1, t_container))
return false;
uindex_t t_slot;
if (__MCArrayFindKeyValueSlotForStore(t_container, p_case_sensitive, p_path[p_path_length - 1], t_slot))
{
uintptr_t t_value_ptr;
t_value_ptr = t_container -> key_values[t_slot] . value;
MCValueRetain(p_value);
if (t_value_ptr != UINTPTR_MIN && t_value_ptr != UINTPTR_MAX)
{
MCValueRelease((MCValueRef)t_value_ptr);
t_container -> key_values[t_slot] . value = p_value;
}
else
{
t_container -> key_values[t_slot] . name = MCValueRetain(p_path[p_path_length - 1]);
t_container -> key_values[t_slot] . value = (uintptr_t)p_value;
t_container -> key_value_count += 1;
}
}
else
return false;
return true;
}
bool MCArrayRemoveValue(MCArrayRef self, bool p_case_sensitive, const MCNameRef *p_path, uindex_t p_path_length)
{
MCArrayRef t_container;
if (p_path_length == 1)
t_container = self;
else if (!__MCArrayResolveMutablePath(self, p_case_sensitive, p_path, p_path_length - 1, t_container))
return false;
uindex_t t_slot;
if (__MCArrayFindKeyValueSlotForRemove(t_container, p_case_sensitive, p_path[p_path_length - 1], t_slot))
{
MCValueRelease(t_container -> key_values[t_slot] . key);
MCValueRelease((MCValueRef)t_container -> key_values[t_slot] . value);
}
return true;
}
////////////////////////////////////////////////////////////////////////////////
static MCArrayRef __MCArrayResolvePath(MCArrayRef self, bool p_case_sensitive, const MCNameRef *p_path, uindex_t p_path_length)
{
MCArrayRef t_container;
t_container = self;
for(uindex_t i = 0; i < p_path_length)
{
uindex_t t_slot;
if (__MCArrayFindKeyValueSlotForFetch(t_container, p_case_sensitive, p_path[i], t_slot))
{
MCValueRef t_value;
t_value = (MCValueRef)t_container -> key_values[t_slot] . value;
if (MCValueGetTypeCode(t_value) == kMCValueTypeCodeArray)
{
t_container = (MCArrayRef)t_value;
continue;
}
}
// If we get here then either the key wasn't found, or the value found
// was not an array - either way the path resolves to empty.
break;
}
return t_container;
}
static bool __MCArrayResolveMutablePath(MCArrayRef self, bool p_case_sensitive, const MCNameRef *p_path, uindex_t p_path_length, MCArrayRef& r_array)
{
MCArrayRef t_container;
t_container = self;
for(t_path_index = 0; t_path_index < p_path_length; t_path_index++)
{
MCArrayRef t_next_array;
// Lookup the slot for the given key.
uindex_t t_slot;
if (__MCArrayFindKeyValueSlotForStore(t_container, p_case_sensitive, p_path[t_path_index], t_slot))
{
uintptr_t t_value_ptr;
t_value_ptr = t_container -> key_values[t_slot] . value;
if (t_value_ptr != UINTPTR_MIN && t_value_ptr != UINTPTR_MAX)
{
// At this point we have a slot and it has a value within it. So fetch it.
MCValueRef t_value;
t_value = (MCValueRef)t_container -> key_values[t_slot] . value;
// If the value is already an array then must ensure it is mutable; otherwise
// we must change it to an array.
if (MCValueGetTypeCode(t_value) == kMCValueTypeCodeArray)
{
if (MCArrayIsMutable((MCArrayRef)t_value))
t_next_array = (MCArrayRef)t_value;
else
{
if (MCArrayMutableCopyAndRelease((MCArrayRef)t_value, t_next_array))
t_container -> key_values[t_slot] . value = (uintptr_t)t_next_array;
else
return false;
}
}
else if (MCArrayCreateMutable(t_next_array))
{
MCValueRelease(t_value);
t_container -> key_values[t_slot] . value = (uintptr_t)t_next_array;
}
else
return false;
}
else
{
// At this point a new slot has been created but has yet to be
// filled. So create a new array and fill it.
if (MCArrayCreateMutable(t_next_array))
{
t_container -> key_values[t_slot] . key = MCValueRetain(p_path[t_path_index]);
t_container -> key_values[t_slot] . value = (uintptr_t)t_next_array;
t_container -> key_value_count += 1;
}
else
return false;
}
}
else
return false;
t_container = t_next_array;
}
// At this point 't_container' contains a mutable array for the last key
// in the path.
r_array = t_container;
return true;
}*/