/* IGraph library. Copyright (C) 2007-2012 Gabor Csardi 334 Harvard street, Cambridge, MA 02139 USA This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 2 of the License, or (at your option) any later version. 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 */ #include "igraph_gml_tree.h" #include "igraph_memory.h" #include "igraph_error.h" #include "config.h" #include #include int igraph_gml_tree_init_integer(igraph_gml_tree_t *t, const char *name, int namelen, igraph_integer_t value) { igraph_integer_t *p; IGRAPH_UNUSED(namelen); IGRAPH_VECTOR_PTR_INIT_FINALLY(&t->names, 1); IGRAPH_CHECK(igraph_vector_char_init(&t->types, 1)); IGRAPH_FINALLY(igraph_vector_char_destroy, &t->types); IGRAPH_VECTOR_PTR_INIT_FINALLY(&t->children, 1); /* names */ VECTOR(t->names)[0] = (void*)name; /* types */ VECTOR(t->types)[0] = IGRAPH_I_GML_TREE_INTEGER; /* children */ p = igraph_Calloc(1, igraph_integer_t); if (!p) { IGRAPH_ERROR("Cannot create integer GML tree node", IGRAPH_ENOMEM); } *p = value; VECTOR(t->children)[0] = p; IGRAPH_FINALLY_CLEAN(3); return 0; } int igraph_gml_tree_init_real(igraph_gml_tree_t *t, const char *name, int namelen, igraph_real_t value) { igraph_real_t *p; IGRAPH_UNUSED(namelen); IGRAPH_VECTOR_PTR_INIT_FINALLY(&t->names, 1); IGRAPH_CHECK(igraph_vector_char_init(&t->types, 1)); IGRAPH_FINALLY(igraph_vector_char_destroy, &t->types); IGRAPH_VECTOR_PTR_INIT_FINALLY(&t->children, 1); /* names */ VECTOR(t->names)[0] = (void*) name; /* types */ VECTOR(t->types)[0] = IGRAPH_I_GML_TREE_REAL; /* children */ p = igraph_Calloc(1, igraph_real_t); if (!p) { IGRAPH_ERROR("Cannot create real GML tree node", IGRAPH_ENOMEM); } *p = value; VECTOR(t->children)[0] = p; IGRAPH_FINALLY_CLEAN(3); return 0; } int igraph_gml_tree_init_string(igraph_gml_tree_t *t, const char *name, int namelen, const char *value, int valuelen) { IGRAPH_UNUSED(namelen); IGRAPH_UNUSED(valuelen); IGRAPH_VECTOR_PTR_INIT_FINALLY(&t->names, 1); IGRAPH_CHECK(igraph_vector_char_init(&t->types, 1)); IGRAPH_FINALLY(igraph_vector_char_destroy, &t->types); IGRAPH_VECTOR_PTR_INIT_FINALLY(&t->children, 1); /* names */ VECTOR(t->names)[0] = (void*) name; /* types */ VECTOR(t->types)[0] = IGRAPH_I_GML_TREE_STRING; /* children */ VECTOR(t->children)[0] = (void*)value; IGRAPH_FINALLY_CLEAN(3); return 0; } int igraph_gml_tree_init_tree(igraph_gml_tree_t *t, const char *name, int namelen, igraph_gml_tree_t *value) { IGRAPH_UNUSED(namelen); IGRAPH_VECTOR_PTR_INIT_FINALLY(&t->names, 1); IGRAPH_CHECK(igraph_vector_char_init(&t->types, 1)); IGRAPH_FINALLY(igraph_vector_char_destroy, &t->types); IGRAPH_VECTOR_PTR_INIT_FINALLY(&t->children, 1); /* names */ VECTOR(t->names)[0] = (void*)name; /* types */ VECTOR(t->types)[0] = IGRAPH_I_GML_TREE_TREE; /* children */ VECTOR(t->children)[0] = value; IGRAPH_FINALLY_CLEAN(3); return 0; } /* merge is destructive, the _second_ tree is destroyed */ int igraph_gml_tree_mergedest(igraph_gml_tree_t *t1, igraph_gml_tree_t *t2) { long int i, n = igraph_vector_ptr_size(&t2->children); for (i = 0; i < n; i++) { IGRAPH_CHECK(igraph_vector_ptr_push_back(&t1->names, VECTOR(t2->names)[i])); IGRAPH_CHECK(igraph_vector_char_push_back(&t1->types, VECTOR(t2->types)[i])); IGRAPH_CHECK(igraph_vector_ptr_push_back(&t1->children, VECTOR(t2->children)[i])); } igraph_vector_ptr_destroy(&t2->names); igraph_vector_char_destroy(&t2->types); igraph_vector_ptr_destroy(&t2->children); return 0; } void igraph_gml_tree_destroy(igraph_gml_tree_t *t) { long int i, n = igraph_vector_ptr_size(&t->children); for (i = 0; i < n; i++) { int type = VECTOR(t->types)[i]; switch (type) { case IGRAPH_I_GML_TREE_TREE: igraph_gml_tree_destroy(VECTOR(t->children)[i]); igraph_Free(VECTOR(t->names)[i]); break; case IGRAPH_I_GML_TREE_INTEGER: igraph_Free(VECTOR(t->children)[i]); igraph_Free(VECTOR(t->names)[i]); break; case IGRAPH_I_GML_TREE_REAL: igraph_Free(VECTOR(t->children)[i]); igraph_Free(VECTOR(t->names)[i]); break; case IGRAPH_I_GML_TREE_STRING: igraph_Free(VECTOR(t->children)[i]); igraph_Free(VECTOR(t->names)[i]); break; case IGRAPH_I_GML_TREE_DELETED: break; } } igraph_vector_ptr_destroy(&t->names); igraph_vector_char_destroy(&t->types); igraph_vector_ptr_destroy(&t->children); igraph_Free(t); } long int igraph_gml_tree_length(const igraph_gml_tree_t *t) { return igraph_vector_ptr_size(&t->names); } long int igraph_gml_tree_find(const igraph_gml_tree_t *t, const char *name, long int from) { long int size = igraph_vector_ptr_size(&t->names); while ( from < size && (! VECTOR(t->names)[from] || strcmp(VECTOR(t->names)[from], name)) ) { from++; } if (from == size) { from = -1; } return from; } long int igraph_gml_tree_findback(const igraph_gml_tree_t *t, const char *name, long int from) { while ( from >= 0 && (! VECTOR(t->names)[from] || strcmp(VECTOR(t->names)[from], name)) ) { from--; } return from; } int igraph_gml_tree_type(const igraph_gml_tree_t *t, long int pos) { return VECTOR(t->types)[pos]; } const char *igraph_gml_tree_name(const igraph_gml_tree_t *t, long int pos) { return VECTOR(t->names)[pos]; } igraph_integer_t igraph_gml_tree_get_integer(const igraph_gml_tree_t *t, long int pos) { igraph_integer_t *i = VECTOR(t->children)[pos]; return *i; } igraph_real_t igraph_gml_tree_get_real(const igraph_gml_tree_t *t, long int pos) { igraph_real_t *d = VECTOR(t->children)[pos]; return *d; } const char *igraph_gml_tree_get_string(const igraph_gml_tree_t *t, long int pos) { const char *s = VECTOR(t->children)[pos]; return s; } igraph_gml_tree_t *igraph_gml_tree_get_tree(const igraph_gml_tree_t *t, long int pos) { igraph_gml_tree_t *tree = VECTOR(t->children)[pos]; return tree; } void igraph_gml_tree_delete(igraph_gml_tree_t *t, long int pos) { if (VECTOR(t->types)[pos] == IGRAPH_I_GML_TREE_TREE) { igraph_gml_tree_destroy(VECTOR(t->children)[pos]); } igraph_Free(VECTOR(t->names)[pos]); igraph_Free(VECTOR(t->children)[pos]); VECTOR(t->children)[pos] = 0; VECTOR(t->names)[pos] = 0; VECTOR(t->types)[pos] = IGRAPH_I_GML_TREE_DELETED; }