summaryrefslogtreecommitdiffhomepage
path: root/src/dictmaptree.h
blob: 190683cb2059f691ebb943947e3ad7eb69341208 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
/*
 * Copyright © 2023 Michael Smith <mikesmiffy128@gmail.com>
 *
 * Permission to use, copy, modify, and/or distribute this software for any
 * purpose with or without fee is hereby granted, provided that the above
 * copyright notice and this permission notice appear in all copies.
 *
 * THE SOFTWARE IS PROVIDED “AS IS” AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH
 * REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY
 * AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT,
 * INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM
 * LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR
 * OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
 * PERFORMANCE OF THIS SOFTWARE.
 */

#ifndef INC_DICTMAPTREE_H
#define INC_DICTMAPTREE_H

#include "engineapi.h"
#include "intdefs.h"

/*
 * Valve's dict/map/tree structures come in various shapes and sizes, so here we
 * do the generic macro thing for future-proofing. For now we just define a
 * CUtlDict (map with string keys) of pointers, with ushort indices, which is
 * sufficient for server entity factory lookup, and probably some other stuff.
 * Functions for actually modifying the dicts/maps/trees aren't implemented.
 */

#define DECL_CUTLRBTREE_NODE(name, ktype, vtype, idxtype) \
typedef typeof(ktype) _cutlrbtree_##name##_key; \
typedef typeof(vtype) _cutlrbtree_##name##_val; \
typedef typeof(idxtype) _cutlrbtree_##name##_idx; \
struct name { \
	struct { \
		_cutlrbtree_##name##_idx l, r, p, tags; \
	} links; \
	_cutlrbtree_##name##_key k; \
	_cutlrbtree_##name##_val v; \
};

#define DEF_CUTLRBTREE(name, nodetype) \
struct name { \
	bool (*cmp)(const _cutlrbtree_##nodetype##_key *, \
			const _cutlrbtree_##nodetype##_key *); \
	struct CUtlMemory elems; \
	_cutlrbtree_##nodetype##_idx root, count, firstfree, lastalloc; \
	struct nodetype *nodes; \
}; \
\
static _cutlrbtree_##nodetype##_idx name##_find(const struct name *rb, \
		const _cutlrbtree_##nodetype##_key k) { \
	_cutlrbtree_##nodetype##_idx idx = rb->root; \
	while (idx != (_cutlrbtree_##nodetype##_idx)-1) { \
		struct nodetype *nodes = rb->elems.mem; \
		if (rb->cmp(&k, &nodes[idx].k)) idx = nodes[idx].links.l; \
		else if (rb->cmp(&nodes[idx].k, &k)) idx = nodes[idx].links.r; \
		else break; \
	} \
	return idx; \
} \
\
static inline _cutlrbtree_##nodetype##_val name##_findval(const struct name *rb, \
		const _cutlrbtree_##nodetype##_key k) { \
	const _cutlrbtree_##nodetype##_idx idx = name##_find(rb, k); \
	if (idx == (_cutlrbtree_##nodetype##_idx)-1) { \
		return (_cutlrbtree_CUtlDict_node_p_ushort_val){0}; \
	} \
	struct nodetype *nodes = rb->elems.mem; \
	return nodes[idx].v; \
}

DECL_CUTLRBTREE_NODE(CUtlDict_node_p_ushort, const char *, void *, ushort)
DEF_CUTLRBTREE(CUtlDict_p_ushort, CUtlDict_node_p_ushort)

#endif

// vi: sw=4 ts=4 noet tw=80 cc=80