215 lines
7.1 KiB
C
215 lines
7.1 KiB
C
|
// trie.c -- Basic implementation of a trie abstract data type
|
||
|
// Copyright (C) 2008 Markus Gutschke <markus@shellinabox.com>
|
||
|
//
|
||
|
// This program is free software; you can redistribute it and/or modify
|
||
|
// it 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 along
|
||
|
// with this program; if not, write to the Free Software Foundation, Inc.,
|
||
|
// 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
|
||
|
//
|
||
|
// In addition to these license terms, the author grants the following
|
||
|
// additional rights:
|
||
|
//
|
||
|
// If you modify this program, or any covered work, by linking or
|
||
|
// combining it with the OpenSSL project's OpenSSL library (or a
|
||
|
// modified version of that library), containing parts covered by the
|
||
|
// terms of the OpenSSL or SSLeay licenses, the author
|
||
|
// grants you additional permission to convey the resulting work.
|
||
|
// Corresponding Source for a non-source form of such a combination
|
||
|
// shall include the source code for the parts of OpenSSL used as well
|
||
|
// as that of the covered work.
|
||
|
//
|
||
|
// You may at your option choose to remove this additional permission from
|
||
|
// the work, or from any part of it.
|
||
|
//
|
||
|
// It is possible to build this program in a way that it loads OpenSSL
|
||
|
// libraries at run-time. If doing so, the following notices are required
|
||
|
// by the OpenSSL and SSLeay licenses:
|
||
|
//
|
||
|
// This product includes software developed by the OpenSSL Project
|
||
|
// for use in the OpenSSL Toolkit. (http://www.openssl.org/)
|
||
|
//
|
||
|
// This product includes cryptographic software written by Eric Young
|
||
|
// (eay@cryptsoft.com)
|
||
|
//
|
||
|
//
|
||
|
// The most up-to-date version of this program is always available from
|
||
|
// http://shellinabox.com
|
||
|
|
||
|
#include <stdlib.h>
|
||
|
#include <string.h>
|
||
|
|
||
|
#include "libhttp/trie.h"
|
||
|
#include "logging/logging.h"
|
||
|
|
||
|
struct Trie *newTrie(void (*destructor)(void *, char *), void *arg) {
|
||
|
struct Trie *trie;
|
||
|
check(trie = malloc(sizeof(struct Trie)));
|
||
|
initTrie(trie, destructor, arg);
|
||
|
return trie;
|
||
|
}
|
||
|
|
||
|
void initTrie(struct Trie *trie, void (*destructor)(void *, char *),
|
||
|
void *arg) {
|
||
|
trie->destructor = destructor;
|
||
|
trie->arg = arg;
|
||
|
trie->key = NULL;
|
||
|
trie->value = NULL;
|
||
|
trie->idx = -1;
|
||
|
trie->ch = '\000';
|
||
|
trie->children = NULL;
|
||
|
trie->numChildren = 0;
|
||
|
}
|
||
|
|
||
|
void destroyTrie(struct Trie *trie) {
|
||
|
if (trie) {
|
||
|
free(trie->key);
|
||
|
for (int i = 0; i < trie->numChildren; i++) {
|
||
|
if (trie->destructor && !trie->children[i].ch) {
|
||
|
trie->destructor(trie->arg, trie->children[i].value);
|
||
|
} else {
|
||
|
check(!trie->children[i].value);
|
||
|
}
|
||
|
destroyTrie(&trie->children[i]);
|
||
|
}
|
||
|
free(trie->children);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void deleteTrie(struct Trie *trie) {
|
||
|
destroyTrie(trie);
|
||
|
free(trie);
|
||
|
}
|
||
|
|
||
|
static void addLeafToTrie(struct Trie *trie, char ch, const char *key, int len,
|
||
|
void *value) {
|
||
|
check (len >= 0);
|
||
|
if (len) {
|
||
|
check(trie->key = malloc(len));
|
||
|
memcpy(trie->key, key, len);
|
||
|
} else {
|
||
|
trie->key = NULL;
|
||
|
}
|
||
|
trie->value = NULL;
|
||
|
trie->idx = len;
|
||
|
trie->ch = ch;
|
||
|
check(trie->children = malloc(sizeof(struct Trie)));
|
||
|
trie->numChildren = 1;
|
||
|
initTrie(trie->children, trie->destructor, trie->arg);
|
||
|
trie->children->value = value;
|
||
|
}
|
||
|
|
||
|
void addToTrie(struct Trie *trie, const char *key, char *value) {
|
||
|
if (trie->numChildren == 0) {
|
||
|
addLeafToTrie(trie, '\000', key, strlen(key), value);
|
||
|
} else {
|
||
|
nextNode:;
|
||
|
int len = strlen(key);
|
||
|
for (int i = 0; i < trie->idx; i++) {
|
||
|
if (key[i] != trie->key[i]) {
|
||
|
struct Trie *child;
|
||
|
check(child = malloc(2*sizeof(struct Trie)));
|
||
|
child->destructor = trie->destructor;
|
||
|
child->arg = trie->arg;
|
||
|
check(child->key = malloc(trie->idx - i - 1));
|
||
|
memcpy(child->key, trie->key + i + 1, trie->idx - i - 1);
|
||
|
child->value = trie->value;
|
||
|
child->idx = trie->idx - i - 1;
|
||
|
child->ch = trie->key[i];
|
||
|
child->children = trie->children;
|
||
|
child->numChildren = trie->numChildren;
|
||
|
trie->value = NULL;
|
||
|
trie->idx = i;
|
||
|
trie->children = child;
|
||
|
trie->numChildren = 2;
|
||
|
child++;
|
||
|
child->destructor = trie->destructor;
|
||
|
child->arg = trie->arg;
|
||
|
if (key[i]) {
|
||
|
addLeafToTrie(child, key[i], key + i + 1, len - i - 1, value);
|
||
|
} else {
|
||
|
initTrie(child, trie->destructor, trie->arg);
|
||
|
child->value = value;
|
||
|
}
|
||
|
return;
|
||
|
}
|
||
|
}
|
||
|
for (int i = 0; i < trie->numChildren; i++) {
|
||
|
if (key[trie->idx] == trie->children[i].ch) {
|
||
|
if (trie->children[i].ch) {
|
||
|
key += trie->idx + 1;
|
||
|
trie = &trie->children[i];
|
||
|
goto nextNode;
|
||
|
} else {
|
||
|
if (trie->destructor) {
|
||
|
trie->destructor(trie->arg, trie->children[i].value);
|
||
|
}
|
||
|
trie->children[i].value = value;
|
||
|
return;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
key += trie->idx;
|
||
|
len -= trie->idx;
|
||
|
check(trie->children = realloc(
|
||
|
trie->children, ++trie->numChildren*sizeof(struct Trie)));
|
||
|
struct Trie *newNode = &trie->children[trie->numChildren-1];
|
||
|
if (*key) {
|
||
|
newNode->destructor = trie->destructor;
|
||
|
newNode->arg = trie->arg;
|
||
|
addLeafToTrie(newNode, *key, key + 1, len - 1, value);
|
||
|
} else {
|
||
|
initTrie(newNode, trie->destructor, trie->arg);
|
||
|
trie->value = value;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
char *getFromTrie(const struct Trie *trie, const char *key, char **diff) {
|
||
|
if (diff) {
|
||
|
*diff = NULL;
|
||
|
}
|
||
|
for (;;) {
|
||
|
if (trie->idx > 0) {
|
||
|
if (memcmp(trie->key, key, trie->idx)) {
|
||
|
return NULL;
|
||
|
}
|
||
|
key += trie->idx;
|
||
|
}
|
||
|
int partial = -1;
|
||
|
for (int i = 0; ; i++) {
|
||
|
if (i >= trie->numChildren) {
|
||
|
if (diff && partial >= 0) {
|
||
|
// If the caller provided a "diff" pointer, then we allow partial
|
||
|
// matches for the longest possible prefix that is a key in the
|
||
|
// trie. Upon return, the "diff" pointer points to the first
|
||
|
// character in the key does not match.
|
||
|
*diff = (char *)key;
|
||
|
return trie->children[partial].value;
|
||
|
}
|
||
|
return NULL;
|
||
|
} else if (*key == trie->children[i].ch) {
|
||
|
if (!*key) {
|
||
|
if (diff) {
|
||
|
*diff = (char *)key;
|
||
|
}
|
||
|
return trie->children[i].value;
|
||
|
}
|
||
|
trie = &trie->children[i];
|
||
|
key++;
|
||
|
break;
|
||
|
} else if (!trie->children[i].ch) {
|
||
|
partial = i;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|