/*- * Written by H. Mitasova, I. Kosinovsky, D. Gerdes Fall 1993 * University of Illinois * US Army Construction Engineering Research Lab * Copyright 1993, H. Mitasova (University of Illinois), * I. Kosinovsky, (USA-CERL), and D.Gerdes (USA-CERL) * * updated by Mitasova Nov. 96, no changes necessary */ #include #include #include #include #include struct multfunc * MT_functions_new (int (*compare)(struct triple *, struct quaddata *), struct quaddata **(*divide_data)(struct quaddata *, int, double), int (*add_data)(struct triple *, struct quaddata *, double), int (*intersect)(struct quaddata *, struct quaddata *), int (*division_check)(struct quaddata *, int), int (*get_points)(struct quaddata *, struct quaddata *, int)) /* Initializes FUNCTIONS structure with given arguments*/ { struct multfunc *functions; if (!(functions = (struct multfunc *) malloc (sizeof (struct multfunc)))) { return NULL; } functions->compare = compare; functions->divide_data = divide_data; functions->add_data = add_data; functions->intersect = intersect; functions->division_check = division_check; functions->get_points = get_points; return functions; } struct tree_info * MT_tree_info_new (struct multtree *root, struct multfunc *functions, double dmin, int kmax) /*Initializes TREE_INFO using given arguments*/ { struct tree_info *info; if (!(info = (struct tree_info *) malloc (sizeof (struct tree_info)))) { return NULL; } info->root = root; info->functions = functions; info->dmin = dmin; info->kmax = kmax; return info; } struct multtree *MT_tree_new (struct quaddata *data, struct multtree **leafs, struct multtree *parent, int multant) /*Initializes TREE using given arguments*/ { struct multtree *tree; if (!(tree = (struct multtree *) malloc (sizeof (struct multtree)))) { return NULL; } tree->data = data; tree->leafs = leafs; tree->parent = parent; tree->multant = multant; return tree; } int MT_insert (struct triple *point, struct tree_info *info, struct multtree *tree, int n_leafs) /*First checks for dividing cond. (if n_points>=KMAX) and TREE is a leaf by calling one of tree's functions (division_check()). If TREE is not a leaf (is a node) uses function compare to determine into which "son" we need to insert the point and calls MT_insert() with this son as a n argument. If TREE is a leaf but we don't need to divide it (n_pointsdata == NULL) { fprintf(stderr,"insert: tree->data is NULL\n"); return -5; } i = info->functions->division_check (tree->data,info->kmax); if (i<=0) { if (i==-1) { comp=info->functions->compare(point,tree->data); if ((comp<1)||(comp>n_leafs)) return -3; j=MT_insert(point,info,tree->leafs[comp-1],n_leafs); } else { if (i==0) { j=info->functions->add_data (point, tree->data, info->dmin); } } } else { k=MT_divide(info,tree,n_leafs); if(k==1) j=MT_insert(point,info,tree,n_leafs); if (k == -3) { static int once = 0; if (!once) { fprintf (stderr, "Point out of range!\n"); once = 1; } } if(k<0) return k; } return j; } int MT_divide (struct tree_info *info, struct multtree *tree, int n_leafs) /* Divides the tree by calling one of tree's functions (divide_data()) and returns the result of divide_data() */ { int i; struct quaddata **datas; struct multtree *par; struct multtree **leafs; datas=info->functions->divide_data(tree->data,info->kmax,info->dmin); if (datas==NULL) { fprintf(stderr,"datas is NULL\n"); return -7; } par = tree; leafs = (struct multtree **)malloc(sizeof(struct multtree * ) *n_leafs); for (i=1;i<=n_leafs;i++) { leafs[i-1]=MT_tree_new(datas[i],NULL,par,i); } tree->leafs=leafs; return 1; } int MT_region_data ( struct tree_info *info, struct multtree *tree, struct quaddata *data, int MAX, /* max number of points we can add (KMAX2) */ int n_leafs ) /* Gets points inside the region defined by DATA from TREE and adds them to DATA. If the number of eligible point is more than MAX returns MAX+1 othervise returns number of points added to DATA. Uses tree's functions intersect() to find leafs that intersect given region and get_points() to get points from such leafs. */ { int n = 0,j; if (tree == NULL) { fprintf(stderr,"MT_region_data: tree is NULL\n"); return n; } if (tree->data == NULL) { fprintf(stderr,"MT_region_data: data is NULL\n"); return n; } if (info->functions->intersect (data, tree->data)) { if (tree->leafs != NULL) { for(j=0;jleafs[j],data,MAX-n,n_leafs))>MAX) return n; } } else { n=info->functions->get_points(data,tree->data,MAX); } return n; } return 0; }