diff options
author | Ashlee Young <ashlee@onosfw.com> | 2015-09-09 22:21:41 -0700 |
---|---|---|
committer | Ashlee Young <ashlee@onosfw.com> | 2015-09-09 22:21:41 -0700 |
commit | 8879b125d26e8db1a5633de5a9c692eb2d1c4f83 (patch) | |
tree | c7259d85a991b83dfa85ab2e339360669fc1f58e /framework/src/suricata/src/util-bloomfilter-counting.c | |
parent | 13d05bc8458758ee39cb829098241e89616717ee (diff) |
suricata checkin based on commit id a4bce14770beee46a537eda3c3f6e8e8565d5d0a
Change-Id: I9a214fa0ee95e58fc640e50bd604dac7f42db48f
Diffstat (limited to 'framework/src/suricata/src/util-bloomfilter-counting.c')
-rw-r--r-- | framework/src/suricata/src/util-bloomfilter-counting.c | 410 |
1 files changed, 410 insertions, 0 deletions
diff --git a/framework/src/suricata/src/util-bloomfilter-counting.c b/framework/src/suricata/src/util-bloomfilter-counting.c new file mode 100644 index 00000000..443b8d79 --- /dev/null +++ b/framework/src/suricata/src/util-bloomfilter-counting.c @@ -0,0 +1,410 @@ +/* Copyright (C) 2007-2010 Open Information Security Foundation + * + * You can copy, redistribute or modify this Program under the terms of + * the GNU General Public License version 2 as published by the Free + * Software Foundation. + * + * This program 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 + * version 2 along with this program; if not, write to the Free Software + * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA + * 02110-1301, USA. + */ + +/** + * \file + * + * \author Victor Julien <victor@inliniac.net> + * + * Counting Bloom Filter implementation. Can be used with 8, 16, 32 bits + * counters. + */ + +#include "suricata-common.h" +#include "util-bloomfilter-counting.h" +#include "util-unittest.h" + +/* type: 1, 2 or 4 for 8, 16, or 32 bit counters + * + */ +BloomFilterCounting *BloomFilterCountingInit(uint32_t size, uint8_t type, uint8_t iter, uint32_t (*Hash)(void *, uint16_t, uint8_t, uint32_t)) { + BloomFilterCounting *bf = NULL; + + if (iter == 0) + goto error; + + if (Hash == NULL || size == 0) { + //printf("ERROR: BloomFilterCountingInit no Hash function\n"); + goto error; + } + + if (type != 1 && type != 2 && type != 4) { + //printf("ERROR: BloomFilterCountingInit only 1, 2 and 4 bytes are supported\n"); + goto error; + } + + /* setup the filter */ + bf = SCMalloc(sizeof(BloomFilterCounting)); + if (unlikely(bf == NULL)) + goto error; + memset(bf,0,sizeof(BloomFilterCounting)); + bf->type = type; /* size of the type: 1, 2, 4 */ + bf->array_size = size; + bf->hash_iterations = iter; + bf->Hash = Hash; + + /* setup the bitarray */ + bf->array = SCMalloc(bf->array_size * bf->type); + if (bf->array == NULL) + goto error; + memset(bf->array,0,bf->array_size * bf->type); + + return bf; + +error: + if (bf != NULL) { + if (bf->array != NULL) + SCFree(bf->array); + + SCFree(bf); + } + return NULL; +} + +void BloomFilterCountingFree(BloomFilterCounting *bf) +{ + if (bf != NULL) { + if (bf->array != NULL) + SCFree(bf->array); + + SCFree(bf); + } +} + +void BloomFilterCountingPrint(BloomFilterCounting *bf) +{ + printf("\n------ Counting Bloom Filter Stats ------\n"); + printf("Buckets: %" PRIu32 "\n", bf->array_size); + printf("Counter size: %" PRIu32 "\n", bf->type); + printf("Memory size: %" PRIu32 " bytes\n", bf->array_size * bf->type); + printf("Hash function pointer: %p\n", bf->Hash); + printf("Hash functions: %" PRIu32 "\n", bf->hash_iterations); + printf("-----------------------------------------\n"); +} + +int BloomFilterCountingAdd(BloomFilterCounting *bf, void *data, uint16_t datalen) +{ + uint8_t iter = 0; + uint32_t hash = 0; + + if (bf == NULL || data == NULL || datalen == 0) + return -1; + + for (iter = 0; iter < bf->hash_iterations; iter++) { + hash = bf->Hash(data, datalen, iter, bf->array_size) * bf->type; + if (bf->type == 1) { + uint8_t *u8 = (uint8_t *)&bf->array[hash]; + if ((*u8) != 255) + (*u8)++; + } else if (bf->type == 2) { + uint16_t *u16 = (uint16_t *)&bf->array[hash]; + if ((*u16) != 65535) + (*u16)++; + } else if (bf->type == 4) { + uint32_t *u32 = (uint32_t *)&bf->array[hash]; + if ((*u32) != 4294967295UL) + (*u32)++; + } + } + + return 0; +} + +int BloomFilterCountingRemove(BloomFilterCounting *bf, void *data, uint16_t datalen) +{ + uint8_t iter = 0; + uint32_t hash = 0; + + if (bf == NULL || data == NULL || datalen == 0) + return -1; + + /* only remove data that was actually added */ + if (BloomFilterCountingTest(bf, data, datalen) == 0) { + printf("ERROR: BloomFilterCountingRemove tried to remove data " + "that was never added to the set or was already removed.\n"); + return -1; + } + + /* decrease counters for every iteration */ + for (iter = 0; iter < bf->hash_iterations; iter++) { + hash = bf->Hash(data, datalen, iter, bf->array_size) * bf->type; + if (bf->type == 1) { + uint8_t *u8 = (uint8_t *)&bf->array[hash]; + if ((*u8) > 0) + (*u8)--; + else { + printf("ERROR: BloomFilterCountingRemove tried to decrease a " + "counter below zero.\n"); + return -1; + } + } else if (bf->type == 2) { + uint16_t *u16 = (uint16_t *)&bf->array[hash]; + if ((*u16) > 0) + (*u16)--; + else { + printf("ERROR: BloomFilterCountingRemove tried to decrease a " + "counter below zero.\n"); + return -1; + } + } else if (bf->type == 4) { + uint32_t *u32 = (uint32_t *)&bf->array[hash]; + if ((*u32) > 0) + (*u32)--; + else { + printf("ERROR: BloomFilterCountingRemove tried to decrease a " + "counter below zero.\n"); + return -1; + } + } + } + + return 0; +} + +/* Test if data matches our filter and is likely to be in the set + * + * returns 0: for no match + * 1: match + */ +int BloomFilterCountingTest(BloomFilterCounting *bf, void *data, uint16_t datalen) +{ + uint8_t iter = 0; + uint32_t hash = 0; + int hit = 1; + + /* check each hash iteration */ + for (iter = 0; iter < bf->hash_iterations; iter++) { + hash = bf->Hash(data, datalen, iter, bf->array_size) * bf->type; + if (bf->type == 1) { + uint8_t *u8 = (uint8_t *)&bf->array[hash]; + if ((*u8) == 0x00) { + hit = 0; + break; + } + } else if (bf->type == 2) { + uint16_t *u16 = (uint16_t *)&bf->array[hash]; + if ((*u16) == 0x0000) { + hit = 0; + break; + } + } else if (bf->type == 4) { + uint32_t *u32 = (uint32_t *)&bf->array[hash]; + if ((*u32) == 0x00000000) { + hit = 0; + break; + } + } + } + + return hit; +} + +/* + * ONLY TESTS BELOW THIS COMMENT + */ + +#ifdef UNITTESTS +static uint32_t BloomHash(void *data, uint16_t datalen, uint8_t iter, uint32_t hash_size) +{ + uint8_t *d = (uint8_t *)data; + uint32_t i; + uint32_t hash = 0; + + for (i = 0; i < datalen; i++) { + if (i == 0) hash += (((uint32_t)*d++)); + else if (i == 1) hash += (((uint32_t)*d++) * datalen); + else hash *= (((uint32_t)*d++) * i); + } + + hash *= (iter + datalen); + hash %= hash_size; + return hash; +} + +static int BloomFilterCountingTestInit01 (void) +{ + BloomFilterCounting *bf = BloomFilterCountingInit(1024, 4, 4, BloomHash); + if (bf == NULL) + return 0; + + BloomFilterCountingFree(bf); + return 1; +} + +/* no hash function, so it should fail */ +static int BloomFilterCountingTestInit02 (void) +{ + BloomFilterCounting *bf = BloomFilterCountingInit(1024, 4, 4, NULL); + if (bf == NULL) + return 1; + + BloomFilterCountingFree(bf); + return 0; +} + +static int BloomFilterCountingTestInit03 (void) +{ + int result = 0; + BloomFilterCounting *bf = BloomFilterCountingInit(1024, 4, 4, BloomHash); + if (bf == NULL) + return 0; + + if (bf->Hash == BloomHash) + result = 1; + + BloomFilterCountingFree(bf); + return result; +} + +static int BloomFilterCountingTestInit04 (void) +{ + BloomFilterCounting *bf = BloomFilterCountingInit(1024, 0, 4, BloomHash); + if (bf == NULL) + return 1; + + BloomFilterCountingFree(bf); + return 0; +} + +static int BloomFilterCountingTestInit05 (void) +{ + BloomFilterCounting *bf = BloomFilterCountingInit(0, 4, 4, BloomHash); + if (bf == NULL) + return 1; + + BloomFilterCountingFree(bf); + return 0; +} + +static int BloomFilterCountingTestInit06 (void) +{ + BloomFilterCounting *bf = BloomFilterCountingInit(32, 3, 4, BloomHash); + if (bf == NULL) + return 1; + + BloomFilterCountingFree(bf); + return 0; +} + +static int BloomFilterCountingTestAdd01 (void) +{ + int result = 0; + BloomFilterCounting *bf = BloomFilterCountingInit(1024, 4, 4, BloomHash); + if (bf == NULL) + return 0; + + int r = BloomFilterCountingAdd(bf, "test", 0); + if (r == 0) + goto end; + + /* all is good! */ + result = 1; +end: + if (bf != NULL) BloomFilterCountingFree(bf); + return result; +} + +static int BloomFilterCountingTestAdd02 (void) +{ + int result = 0; + BloomFilterCounting *bf = BloomFilterCountingInit(1024, 4, 4, BloomHash); + if (bf == NULL) + return 0; + + int r = BloomFilterCountingAdd(bf, NULL, 4); + if (r == 0) + goto end; + + /* all is good! */ + result = 1; +end: + if (bf != NULL) BloomFilterCountingFree(bf); + return result; +} + +static int BloomFilterCountingTestFull01 (void) +{ + int result = 0; + BloomFilterCounting *bf = BloomFilterCountingInit(32, 4, 4, BloomHash); + if (bf == NULL) { + printf("init failed: "); + goto end; + } + + int r = BloomFilterCountingAdd(bf, "test", 4); + if (r != 0) { + printf("first add: "); + goto end; + } + + r = BloomFilterCountingTest(bf, "test", 4); + if (r != 1) { + printf("2nd add: "); + goto end; + } + + r = BloomFilterCountingRemove(bf, "test", 4); + if (r != 0) { + printf("3rd add: "); + goto end; + } + + /* all is good! */ + result = 1; +end: + if (bf != NULL) + BloomFilterCountingFree(bf); + return result; +} + +static int BloomFilterCountingTestFull02 (void) +{ + int result = 0; + BloomFilterCounting *bf = BloomFilterCountingInit(32, 4, 4, BloomHash); + if (bf == NULL) + goto end; + + int r = BloomFilterCountingTest(bf, "test", 4); + if (r != 0) + goto end; + + /* all is good! */ + result = 1; +end: + if (bf != NULL) BloomFilterCountingFree(bf); + return result; +} +#endif + +void BloomFilterCountingRegisterTests(void) +{ +#ifdef UNITTESTS + UtRegisterTest("BloomFilterCountingTestInit01", BloomFilterCountingTestInit01, 1); + UtRegisterTest("BloomFilterCountingTestInit02", BloomFilterCountingTestInit02, 1); + UtRegisterTest("BloomFilterCountingTestInit03", BloomFilterCountingTestInit03, 1); + UtRegisterTest("BloomFilterCountingTestInit04", BloomFilterCountingTestInit04, 1); + UtRegisterTest("BloomFilterCountingTestInit05", BloomFilterCountingTestInit05, 1); + UtRegisterTest("BloomFilterCountingTestInit06", BloomFilterCountingTestInit06, 1); + + UtRegisterTest("BloomFilterCountingTestAdd01", BloomFilterCountingTestAdd01, 1); + UtRegisterTest("BloomFilterCountingTestAdd02", BloomFilterCountingTestAdd02, 1); + + UtRegisterTest("BloomFilterCountingTestFull01", BloomFilterCountingTestFull01, 1); + UtRegisterTest("BloomFilterCountingTestFull02", BloomFilterCountingTestFull02, 1); +#endif +} + |