replace at_key with at_keys, some bugs found by tests.

This commit is contained in:
Enno Rehling 2016-02-09 07:03:11 +01:00
parent dbcae5484e
commit 5e631fd145
3 changed files with 38 additions and 8 deletions

View file

@ -125,6 +125,7 @@ void register_attributes(void)
at_register(&at_raceprefix); at_register(&at_raceprefix);
at_register(&at_iceberg); at_register(&at_iceberg);
at_register(&at_key); at_register(&at_key);
at_register(&at_keys);
at_register(&at_follow); at_register(&at_follow);
at_register(&at_targetregion); at_register(&at_targetregion);
at_register(&at_orcification); at_register(&at_orcification);

View file

@ -29,7 +29,7 @@ OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
static void a_writekeys(const attrib *a, const void *o, storage *store) { static void a_writekeys(const attrib *a, const void *o, storage *store) {
int i, *keys = (int *)a->data.v; int i, *keys = (int *)a->data.v;
for (i = 0; i != keys[0]; ++i) { for (i = 0; i <= keys[0]; ++i) {
WRITE_INT(store, keys[i]); WRITE_INT(store, keys[i]);
} }
} }
@ -40,7 +40,7 @@ static int a_readkeys(attrib * a, void *owner, struct storage *store) {
assert(i < 4096 && i>0); assert(i < 4096 && i>0);
a->data.v = p = malloc(sizeof(int)*(i + 1)); a->data.v = p = malloc(sizeof(int)*(i + 1));
*p++ = i; *p++ = i;
while (--i) { while (i--) {
READ_INT(store, p++); READ_INT(store, p++);
} }
return AT_READ_OK; return AT_READ_OK;
@ -115,27 +115,55 @@ static attrib *find_key(attrib * alist, int key)
void key_set(attrib ** alist, int key) void key_set(attrib ** alist, int key)
{ {
int *keys, n = 1;
attrib *a; attrib *a;
assert(key != 0); assert(key != 0);
a = find_key(*alist, key); a = a_find(*alist, &at_keys);
if (!a) { if (!a) {
a = a_add(alist, make_key(key)); a = a_add(alist, a_new(&at_keys));
} }
keys = (int *)a->data.v;
if (keys) {
n = keys[0] + 1;
}
keys = realloc(keys, sizeof(int) *(n + 1));
// TODO: does insertion sort pay off here?
keys[0] = n;
keys[n] = key;
a->data.v = keys;
} }
void key_unset(attrib ** alist, int key) void key_unset(attrib ** alist, int key)
{ {
attrib *a; attrib *a;
assert(key != 0); assert(key != 0);
a = find_key(*alist, key); a = a_find(*alist, &at_keys);
if (a) { if (a) {
a_remove(alist, a); int i, *keys = (int *)a->data.v;
if (keys) {
for (i = 1; i <= keys[0]; ++i) {
if (keys[i] == key) {
keys[i] = keys[keys[0]];
keys[0]--;
}
}
}
} }
} }
bool key_get(attrib *alist, int key) { bool key_get(attrib *alist, int key) {
attrib *a; attrib *a;
assert(key != 0); assert(key != 0);
a = find_key(alist, key); a = a_find(alist, &at_keys);
return a != NULL; if (a) {
int i, *keys = (int *)a->data.v;
if (keys) {
for (i = 1; i <= keys[0]; ++i) {
if (keys[i] == key) {
return true;
}
}
}
}
return false;
} }

View file

@ -23,6 +23,7 @@ extern "C" {
#endif #endif
extern struct attrib_type at_key; extern struct attrib_type at_key;
extern struct attrib_type at_keys;
void key_set(struct attrib **alist, int key); void key_set(struct attrib **alist, int key);
void key_unset(struct attrib **alist, int key); void key_unset(struct attrib **alist, int key);