/*---------------------------------------------------------------------------* * HashMapImpl.c * * * * Copyright 2007, 2008 Nuance Communciations, Inc. * * * * 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. * * * *---------------------------------------------------------------------------*/ #include "HashMap.h" #include "HashMapImpl.h" #include "plog.h" #include "pmemory.h" #include "string.h" #define MTAG NULL static ESR_ReturnCode HashMapCreate_Internal(PHashTableArgs *hashArgs, HashMap **self) { HashMapImpl* impl; ESR_ReturnCode rc = ESR_SUCCESS; if (self == NULL) return ESR_INVALID_ARGUMENT; impl = NEW(HashMapImpl, MTAG); if (impl == NULL) return ESR_OUT_OF_MEMORY; if ((rc = PHashTableCreate(hashArgs, MTAG, &impl->table)) != ESR_SUCCESS) { FREE(impl); return rc; } impl->Interface.put = &HashMap_Put; impl->Interface.remove = &HashMap_Remove; impl->Interface.removeAndFree = &HashMap_RemoveAndFree; impl->Interface.removeAll = &HashMap_RemoveAll; impl->Interface.removeAndFreeAll = &HashMap_RemoveAndFreeAll; impl->Interface.removeAtIndex = &HashMap_RemoveAtIndex; impl->Interface.containsKey = &HashMap_ContainsKey; impl->Interface.getKeyAtIndex = &HashMap_GetKeyAtIndex; impl->Interface.get = &HashMap_Get; impl->Interface.getValueAtIndex = &HashMap_GetValueAtIndex; impl->Interface.getSize = &HashMap_GetSize; impl->Interface.destroy = &HashMap_Destroy; *self = (HashMap*) impl; return ESR_SUCCESS; } ESR_ReturnCode HashMapCreate(HashMap** self) { return HashMapCreate_Internal(NULL, self); } ESR_ReturnCode HashMapCreateBins(size_t nbBins, HashMap** self) { PHashTableArgs hashArgs; hashArgs.capacity = nbBins; hashArgs.maxLoadFactor = PHASH_TABLE_DEFAULT_MAX_LOAD_FACTOR; hashArgs.hashFunction = PHASH_TABLE_DEFAULT_HASH_FUNCTION; hashArgs.compFunction = PHASH_TABLE_DEFAULT_COMP_FUNCTION; return HashMapCreate_Internal(&hashArgs, self); } ESR_ReturnCode HashMap_Put(HashMap* self, const LCHAR* key, void* value) { HashMapImpl* impl = (HashMapImpl*) self; PHashTableEntry *entry = NULL; ESR_ReturnCode rc; ESR_BOOL exists; CHKLOG(rc, PHashTableContainsKey(impl->table, key, &exists)); if (!exists) { /* Not found, clone the key and insert it. */ LCHAR *clone = (LCHAR *) MALLOC(sizeof(LCHAR) * (LSTRLEN(key) + 1), MTAG); if (clone == NULL) return ESR_OUT_OF_MEMORY; LSTRCPY(clone, key); if ((rc = PHashTablePutValue(impl->table, clone, value, NULL)) != ESR_SUCCESS) { FREE(clone); } } else { /* Key already present in table, just change the value. */ CHKLOG(rc, PHashTableGetEntry(impl->table, key, &entry)); rc = PHashTableEntrySetValue(entry, value, NULL); } return rc; CLEANUP: return rc; } static ESR_ReturnCode HashMap_Remove_Internal(HashMapImpl* impl, const LCHAR* key, ESR_BOOL freeValue) { PHashTableEntry *entry = NULL; ESR_ReturnCode rc = ESR_SUCCESS; LCHAR *clonedKey = NULL; void *value = NULL; CHK(rc, PHashTableGetEntry(impl->table, key, &entry)); CHK(rc, PHashTableEntryGetKeyValue(entry, (void **)&clonedKey, (void **)&value)); if (clonedKey) FREE(clonedKey); if (freeValue && value) FREE(value); return PHashTableEntryRemove(entry); CLEANUP: return rc; } ESR_ReturnCode HashMap_Remove(HashMap* self, const LCHAR* key) { return HashMap_Remove_Internal((HashMapImpl*) self, key, ESR_FALSE); } ESR_ReturnCode HashMap_RemoveAndFree(HashMap* self, const LCHAR* key) { return HashMap_Remove_Internal((HashMapImpl*) self, key, ESR_TRUE); } static ESR_ReturnCode HashMap_RemoveAll_Internal(HashMapImpl *impl, ESR_BOOL freeValues) { PHashTableEntry *entry1 = NULL; PHashTableEntry *entry2 = NULL; ESR_ReturnCode rc = ESR_SUCCESS; LCHAR *key = NULL; void *value = NULL; if ((rc = PHashTableEntryGetFirst(impl->table, &entry1)) != ESR_SUCCESS) goto end; while (entry1 != NULL) { if ((rc = PHashTableEntryGetKeyValue(entry1, (void **)&key, (void **)&value)) != ESR_SUCCESS) goto end; if (key) FREE(key); if (freeValues && value) FREE(value); entry2 = entry1; if ((rc = PHashTableEntryAdvance(&entry1)) != ESR_SUCCESS) goto end; if ((rc = PHashTableEntryRemove(entry2)) != ESR_SUCCESS) goto end; } end: return rc; } ESR_ReturnCode HashMap_RemoveAll(HashMap* self) { return HashMap_RemoveAll_Internal((HashMapImpl *) self, ESR_FALSE); } ESR_ReturnCode HashMap_RemoveAndFreeAll(HashMap* self) { return HashMap_RemoveAll_Internal((HashMapImpl *) self, ESR_TRUE); } ESR_ReturnCode HashMap_ContainsKey(HashMap* self, const LCHAR* key, ESR_BOOL* exists) { HashMapImpl* impl = (HashMapImpl*) self; ESR_ReturnCode rc = ESR_SUCCESS; CHKLOG(rc, PHashTableContainsKey(impl->table, key, exists)); return rc; CLEANUP: return rc; } ESR_ReturnCode HashMap_Get(HashMap* self, const LCHAR* key, void** value) { HashMapImpl* impl = (HashMapImpl*) self; PHashTableEntry *entry = NULL; ESR_ReturnCode rc = ESR_SUCCESS; CHK(rc, PHashTableGetEntry(impl->table, key, &entry)); CHK(rc, PHashTableEntryGetKeyValue(entry, (void **)NULL, (void **)value)); return ESR_SUCCESS; CLEANUP: return rc; } static ESR_ReturnCode HashMap_GetEntryAtIndex(HashMapImpl *impl, const size_t index, PHashTableEntry **entry) { ESR_ReturnCode rc = ESR_SUCCESS; size_t i = 0; if ((rc = PHashTableEntryGetFirst(impl->table, entry)) != ESR_SUCCESS) goto end; while (*entry != NULL && i < index) { ++i; if ((rc = PHashTableEntryAdvance(entry)) != ESR_SUCCESS) goto end; } if (*entry == NULL) rc = ESR_ARGUMENT_OUT_OF_BOUNDS; end: return rc; } ESR_ReturnCode HashMap_GetKeyAtIndex(HashMap* self, const size_t index, LCHAR** key) { HashMapImpl* impl = (HashMapImpl*) self; PHashTableEntry *entry = NULL; ESR_ReturnCode rc; if ((rc = HashMap_GetEntryAtIndex(impl, index, &entry)) != ESR_SUCCESS) goto end; rc = PHashTableEntryGetKeyValue(entry, (void **) key, (void **) NULL); end: return rc; } ESR_ReturnCode HashMap_GetValueAtIndex(HashMap* self, const size_t index, void** value) { HashMapImpl* impl = (HashMapImpl*) self; PHashTableEntry *entry = NULL; ESR_ReturnCode rc; if ((rc = HashMap_GetEntryAtIndex(impl, index, &entry)) != ESR_SUCCESS) goto end; rc = PHashTableEntryGetKeyValue(entry, (void **)NULL, (void **)value); end: return rc; } ESR_ReturnCode HashMap_RemoveAtIndex(HashMap* self, const size_t index) { HashMapImpl* impl = (HashMapImpl*) self; PHashTableEntry *entry = NULL; ESR_ReturnCode rc; void *key; if ((rc = HashMap_GetEntryAtIndex(impl, index, &entry)) != ESR_SUCCESS) goto end; if ((rc = PHashTableEntryGetKeyValue(entry, (void **)&key, (void **)NULL)) != ESR_SUCCESS) goto end; if (key != NULL) FREE(key); rc = PHashTableEntryRemove(entry); end: return rc; } ESR_ReturnCode HashMap_GetSize(HashMap* self, size_t* size) { HashMapImpl* impl = (HashMapImpl*) self; return PHashTableGetSize(impl->table, size); } ESR_ReturnCode HashMap_Destroy(HashMap* self) { HashMapImpl* impl = (HashMapImpl*) self; ESR_ReturnCode rc = ESR_SUCCESS; if ((rc = self->removeAll(self)) != ESR_SUCCESS) goto end; if (impl->table != NULL) rc = PHashTableDestroy(impl->table); FREE(impl); end: return rc; }