aboutsummaryrefslogtreecommitdiffstats
path: root/metis/ccnx/forwarder/metis/core/metis_NumberSet.c
diff options
context:
space:
mode:
Diffstat (limited to 'metis/ccnx/forwarder/metis/core/metis_NumberSet.c')
-rw-r--r--metis/ccnx/forwarder/metis/core/metis_NumberSet.c225
1 files changed, 225 insertions, 0 deletions
diff --git a/metis/ccnx/forwarder/metis/core/metis_NumberSet.c b/metis/ccnx/forwarder/metis/core/metis_NumberSet.c
new file mode 100644
index 00000000..6828014a
--- /dev/null
+++ b/metis/ccnx/forwarder/metis/core/metis_NumberSet.c
@@ -0,0 +1,225 @@
+/*
+ * Copyright (c) 2017 Cisco and/or its affiliates.
+ * 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.
+ */
+
+/**
+ * Currently uses an unsorted array of numbers.
+ *
+ */
+
+#include <config.h>
+#include <stdio.h>
+#include <parc/algol/parc_Memory.h>
+#include <ccnx/forwarder/metis/core/metis_NumberSet.h>
+#include <parc/algol/parc_ArrayList.h>
+
+#include <LongBow/runtime.h>
+
+struct metis_number_set {
+ MetisNumber *arrayOfNumbers;
+ size_t length;
+ size_t limit;
+ unsigned refcount;
+};
+
+static void metisNumberSet_Expand(MetisNumberSet *set);
+
+MetisNumberSet *
+metisNumberSet_Create()
+{
+ MetisNumberSet *set = parcMemory_AllocateAndClear(sizeof(MetisNumberSet));
+ assertNotNull(set, "parcMemory_AllocateAndClear(%zu) returned NULL", sizeof(MetisNumberSet));
+ set->arrayOfNumbers = parcMemory_AllocateAndClear(sizeof(MetisNumber) * 16);
+ assertNotNull((set->arrayOfNumbers), "parcMemory_AllocateAndClear(%zu) returned NULL", sizeof(MetisNumber) * 16);
+ set->length = 0;
+ set->limit = 16;
+ set->refcount = 1;
+ return set;
+}
+
+MetisNumberSet *
+metisNumberSet_Acquire(const MetisNumberSet *original)
+{
+ assertNotNull(original, "Parameter original must be non-null");
+ MetisNumberSet *copy = (MetisNumberSet *) original;
+ copy->refcount++;
+ return copy;
+}
+
+void
+metisNumberSet_Release(MetisNumberSet **setPtr)
+{
+ assertNotNull(setPtr, "Parameter must be non-null double pointer");
+ assertNotNull(*setPtr, "Parameter must dereference to non-null pointer");
+
+ MetisNumberSet *set = *setPtr;
+ assertTrue(set->refcount > 0, "Invalid state: calling destroy on an object with 0 reference count");
+ set->refcount--;
+
+ if (set->refcount == 0) {
+ parcMemory_Deallocate((void **) &(set->arrayOfNumbers));
+ parcMemory_Deallocate((void **) &set);
+ *setPtr = NULL;
+ }
+}
+
+/**
+ * @function metisNumberSet_AddNoChecks
+ * @abstract Add a number we know is not already in the set
+ * @discussion
+ * Used by other functions that already know the number is unique in the set,
+ * Does not do the expensive Contains check.
+ *
+ * @param <#param1#>
+ */
+static void
+metisNumberSet_AddNoChecks(MetisNumberSet *set, MetisNumber number)
+{
+ if (set->length == set->limit) {
+ metisNumberSet_Expand(set);
+ }
+
+ set->arrayOfNumbers[ set->length ] = number;
+ set->length++;
+}
+
+bool
+metisNumberSet_Add(MetisNumberSet *set, MetisNumber number)
+{
+ assertNotNull(set, "Parameter set must be non-null");
+ if (metisNumberSet_Contains(set, number)) {
+ return false;
+ }
+
+ metisNumberSet_AddNoChecks(set, number);
+ return true;
+}
+
+size_t
+metisNumberSet_Length(const MetisNumberSet *set)
+{
+ assertNotNull(set, "Parameter set must be non-null");
+ return set->length;
+}
+
+MetisNumber
+metisNumberSet_GetItem(const MetisNumberSet *set, size_t ordinalIndex)
+{
+ assertNotNull(set, "Parameter set must be non-null");
+ assertTrue(ordinalIndex < set->length, "Limit beyond end of set, length %zu got %zu", set->length, ordinalIndex);
+
+ return set->arrayOfNumbers[ordinalIndex];
+}
+
+bool
+metisNumberSet_Contains(const MetisNumberSet *set, MetisNumber number)
+{
+ assertNotNull(set, "Parameter set must be non-null");
+ for (size_t i = 0; i < set->length; i++) {
+ if (set->arrayOfNumbers[i] == number) {
+ return true;
+ }
+ }
+ return false;
+}
+
+void
+metisNumberSet_AddSet(MetisNumberSet *destinationSet, const MetisNumberSet *setToAdd)
+{
+ assertNotNull(destinationSet, "Parameter destinationSet must be non-null");
+ assertNotNull(setToAdd, "Parameter setToAdd must be non-null");
+
+ for (size_t i = 0; i < setToAdd->length; i++) {
+ metisNumberSet_Add(destinationSet, setToAdd->arrayOfNumbers[i]);
+ }
+}
+
+MetisNumberSet *
+metisNumberSet_Subtract(const MetisNumberSet *minuend, const MetisNumberSet *subtrahend)
+{
+ // because the underlying ADT is not sorted, this is pretty ineffient, could be O(n^2).
+
+ MetisNumberSet *difference = metisNumberSet_Create();
+
+ for (size_t i = 0; i < minuend->length; i++) {
+ bool unique = true;
+ for (size_t j = 0; j < subtrahend->length && unique; j++) {
+ if (minuend->arrayOfNumbers[i] == subtrahend->arrayOfNumbers[j]) {
+ unique = false;
+ }
+ }
+
+ if (unique) {
+ metisNumberSet_AddNoChecks(difference, minuend->arrayOfNumbers[i]);
+ }
+ }
+ return difference;
+}
+
+bool
+metisNumberSet_Equals(const MetisNumberSet *a, const MetisNumberSet *b)
+{
+ if (a == NULL && b == NULL) {
+ return true;
+ }
+
+ if (a == NULL || b == NULL) {
+ return false;
+ }
+
+ if (a->length == b->length) {
+ for (size_t i = 0; i < a->length; i++) {
+ bool found = false;
+ for (size_t j = 0; j < b->length && !found; j++) {
+ if (a->arrayOfNumbers[i] == b->arrayOfNumbers[j]) {
+ found = true;
+ }
+ }
+ if (!found) {
+ return false;
+ }
+ }
+ return true;
+ }
+
+ return false;
+}
+
+void
+metisNumberSet_Remove(MetisNumberSet *set, MetisNumber number)
+{
+ assertNotNull(set, "Parameter set must be non-null");
+ for (size_t i = 0; i < set->length; i++) {
+ if (set->arrayOfNumbers[i] == number) {
+ set->length--;
+ if (set->length > 0) {
+ // move the last element to the removed element to keep the array packed.
+ set->arrayOfNumbers[i] = set->arrayOfNumbers[set->length];
+ }
+ return;
+ }
+ }
+}
+
+// =====================================================
+
+static void
+metisNumberSet_Expand(MetisNumberSet *set)
+{
+ size_t newlimit = set->limit * 2;
+ size_t newbytes = newlimit * sizeof(MetisNumber);
+
+ set->arrayOfNumbers = parcMemory_Reallocate(set->arrayOfNumbers, newbytes);
+ set->limit = newlimit;
+}